Introduction

In the previous post, we went over Fʀᴀᴄᴛᴀʟ at a very high level, describing its features and what kind of subprotocols we would need to make it work. This post is a little more dense and details how these subprotocols work.

Univariate Sumcheck for Rational Functions

As we mentioned in the previous post, there is a result that says sSf(s)=σ if and only if f(x)=Γg,σ(x) where Γg,σ=xg(x)+σ/|S| and g(x) is a polynomial of degree strictly less than |S|1. To extend this to rational functions, let's first do a little bit of algebra. What we wish to show is
p(α)q(α) S Γg,σ(α)
where fSg means that f and g agree on all values of S. Then we would have
p(α) S Γg,σ(α)q(α)
But we're not done. Due to the multiplication of Γg,σ and q, the polynomial function interpolating each side generally has much higher degree than we want. So we make one more adjustment to get
0 S Γg,σ(α)q(α)p(α).
Now we still have a "high" degree polynomial on the right-hand side, but it is zero on all of S. This doesn't necessarily mean that it is zero on all of F, but it does mean that the polynomial interpolating the right hand side is divisible by the nonzero polynomial of minimal degree that is zero on S, constructed as ZS(x)=sS(xs). So even though Γg,σ(x)q(x)p(x) isn't necessarily low-degree, the polynomial interpolating Γg,σ(x)q(x)p(x)ZS(x) on S is.

Holographic Lincheck

In the previous post, we mentioned the need for some u which is a vector of linearly independent polynomials. We could choose the entries to be something simple such as a power basis 1,x,x2 but to give it a little more structure, we will choose the entries of u to be the partial evaluations of the bivariate "diagonal polynomial on H"
DH(x,y)=ZH(x)ZH(y)xy
as y ranges over H. Two things to note about this polynomial are

  1. first, it does give linearly independent polynomial entries. To see this (and to see why we call it a diagonal polynomial) imagine a square matrix with indices in H×H which has the value DH(a,b) in the a,b entry. When a and b are both in H, but not equal to each other, the numerator of the rational function defining DH is zero, but the denominator is not. When a=b we get an indeterminate form, but it can be resolved because
  2. the minimal-degree nonzero polynomial which is zero on a multiplicative group is of the form ZH(x)=x|H|1 so the numerator is x|H|y|H|, which is divisible by xy no matter the size of H and so despite defining it as a rational function, DH does indeed describe a polynomial function.

The indexer I receives A, the sparse representation of the matrix A, and computes a sparse representation of A, a matrix defined so that Ma,b=Mb,aDH(b,b). The reason for defining it this way is so that the interpolating bivariate polynomial ^A(x,y) is equal to the bivariate polynomial ^uA(x,y) where uA(x,α) is the polynomial entry in the vector Au at the position indexed by α.

This is the oracle that V gets access to.

The holographic lincheck protocol proceeds as follows:

  1. V sends a random αF that is not in H (so that the polynomial ^uA(x,α) is defined, but there is no entry in the vector Au corresponding to α).
  2. P responds with the evaluation of t(x)=^uA(x,α) on L.
  3. P, V perform a sumcheck. i.e. P sends an evaluation of some function g on some coset LF where L is larger than H, but the degree of ˆg is strictly less than |H|1. V outputs two rational constraints, the first attesting that the degree of ˆg is indeed strictly less than |H|1 and the second one attests that the degree of ˆhd1 for a function h defined on L as
    h(x)=DH(x,a)fa(x)t(x)fz(x)Γg,0(x)ZH(x).
  4. V sends βFH uniformly at random.
  5. P sends γ=uA(β,α)=ˆt(β) and V outputs a boundary constraint asserting that γ=ˆt(β). So at this point P has comitted to the value of ˆt(β) because if it's not the case that ˆt(β)=γ then this boundary condition will not be satisfied so Fʀᴀᴄᴛᴀʟ's verifier will not accept the proof.\item P, V perform the sparse matrix arithmetization protocol to verify that Aβ,α=γ.

Sparse Matrix Arithmetization

If we're given a matrix M containing mostly zeroes, it's to our benefit to find a way to represent the matrix so that we only need to store information about the nonzero entries (as much as possible). The way we do this is with a sparse matrix representation M which is a function defined over some set KF whose values are all ordered triples of the form (a, b, γ) with a,bH and γF. So [assuming M has at most |K| nonzero entries], if for some kK we have M(k)=(a, b, γ) then the entry in the a,b position of M is γ. Any entries not covered by the sparse matrix arithmetization protocol are zero.

This is where holography comes in. The matrices A, B, and C are part of the index i. What makes this proof system holographic is that the verifier V does not have access to i. Instead, an indexer I will take i and output a set of oracles that V can query. The benefit of this, is that it allows the verifier's runtime to grow sublinearly with the size of the circuit. Without holography (or some form of preprocessing) this isn't possible, because at the very least the verifier would have to read the entire circuit. A "preprocessing" SNARG is a SNARG where a preprocessing step is computed once for each index (i.e. it doesn't need to be repeated for each proof) and any holographic IOP can be converted into a preprocessing SNARG by taking the indexer's oracle and putting it into a Merkle tree. Then the root of this tree is published as the index's 'verification key'. After this step, the prover can be trusted to answer the verifier's (real or simulated) queries to the holographic oracle if it provides an authentication path that can be checked against the verification key.

In this case, for each matrix M the indexer outputs oracles for the functions defined so that if M(k)=(a, b, γ)

rowM(k)=a

colM(k)=b

valM(k)=γDH(a,a)DH(b,b)

The Verifier

The verifier splits naturally into two parts; one verifying the RS-encoded IOP and a second part that checks the transformation to a SNARG was done correctly.

Both of these involve computing many hashes. Writing standard hash functions, such as SHA-256, in an arithmetic circuit requires a lot of gates; too many for us to feasibly express the verifier entirely within an arithmetic circuit. However, we need to be able to do this if we want to recursively verify Fʀᴀᴄᴛᴀʟ instances, as Fʀᴀᴄᴛᴀʟ only verifies computations in arithmetic circuits (converted to R1CS instances). The proof of correctness for Fʀᴀᴄᴛᴀʟ works in the Random Oracle Model, where the specific hash function is not specified, but is assumed to have some desirable properties. The hash function chosen for implementation is Rescue, which was designed to have low complexity when expressed as an arithmetic circuit.

Conclusion

Fʀᴀᴄᴛᴀʟ, and its predecessor Aurora, are difficult to compare qualitatively to other SNARKs and STARKs, and not just because of its unique achievements of post-quantum security and recursive composability. In almost every aspect of the construction above, the authors of Fʀᴀᴄᴛᴀʟ have come up with a solution or used fairly recent constructions tailored very specifically to their needs, from the RS encoding to the new hash function. In fact, every novel sub-protocol described in this post is required just to make the IOP work!

Written by Nat Bunner, zero-knowledge cryptography researcher & protocol developer at Metastate.

Image credits: Denali National Park and the photographer Ken Conger.

Metastate has ceased its activities on the 31st of January 2021. The team members have joined a new company and working on a new project. If you're interested in zero-knowledge cryptography and cutting-edge cryptographic protocols engineering positions in Rust, check out the open positions at Heliax.

If you have any feedback or questions on this article, please do not hesitate to contact us: hello@heliax.dev.