Skip to content

Commit cfa30d9

Browse files
authored
perf(bn254): include G2 membership check in ML (#1387)
1 parent dcd7cb3 commit cfa30d9

File tree

6 files changed

+37
-19
lines changed

6 files changed

+37
-19
lines changed

internal/stats/latest_stats.csv

+2-2
Original file line numberDiff line numberDiff line change
@@ -181,14 +181,14 @@ pairing_bls24315,bls24_315,plonk,0,0
181181
pairing_bls24315,bls24_317,plonk,0,0
182182
pairing_bls24315,bw6_761,plonk,0,0
183183
pairing_bls24315,bw6_633,plonk,141249,141249
184-
pairing_bn254,bn254,groth16,604783,990919
184+
pairing_bn254,bn254,groth16,607378,995098
185185
pairing_bn254,bls12_377,groth16,0,0
186186
pairing_bn254,bls12_381,groth16,0,0
187187
pairing_bn254,bls24_315,groth16,0,0
188188
pairing_bn254,bls24_317,groth16,0,0
189189
pairing_bn254,bw6_761,groth16,0,0
190190
pairing_bn254,bw6_633,groth16,0,0
191-
pairing_bn254,bn254,plonk,2319665,2030447
191+
pairing_bn254,bn254,plonk,2329131,2039205
192192
pairing_bn254,bls12_377,plonk,0,0
193193
pairing_bn254,bls12_381,plonk,0,0
194194
pairing_bn254,bls24_315,plonk,0,0

std/algebra/emulated/sw_bn254/g2.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -101,7 +101,7 @@ func (g2 *G2) phi(q *G2Affine) *G2Affine {
101101
return &G2Affine{
102102
P: g2AffP{
103103
X: *x,
104-
Y: q.P.Y,
104+
Y: *g2.Ext2.Neg(&q.P.Y),
105105
},
106106
}
107107
}

std/algebra/emulated/sw_bn254/g2_test.go

-1
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,6 @@ type endomorphismG2Circuit struct {
126126
func (c *endomorphismG2Circuit) Define(api frontend.API) error {
127127
g2 := NewG2(api)
128128
res1 := g2.phi(&c.In1)
129-
res1 = g2.neg(res1)
130129
res2 := g2.psi(&c.In1)
131130
res2 = g2.psi(res2)
132131
g2.AssertIsEqual(res1, res2)

std/algebra/emulated/sw_bn254/pairing.go

+14-8
Original file line numberDiff line numberDiff line change
@@ -81,10 +81,10 @@ func NewPairing(api frontend.API) (*Pairing, error) {
8181
}, nil
8282
}
8383

84-
// Pair calculates the reduced pairing for a set of points
85-
// ∏ᵢ e(Pᵢ, Qᵢ).
84+
// Pair calculates the reduced pairing for a set of points ∏ᵢ e(Pᵢ, Qᵢ).
8685
//
87-
// This function doesn't check that the inputs are in the correct subgroups. See AssertIsOnG1 and AssertIsOnG2.
86+
// This function checks that the Qᵢ are in the correct subgroup, but does not
87+
// check Pᵢ. See AssertIsOnG1.
8888
func (pr Pairing) Pair(P []*G1Affine, Q []*G2Affine) (*GTEl, error) {
8989
res, err := pr.MillerLoop(P, Q)
9090
if err != nil {
@@ -206,10 +206,13 @@ func (pr Pairing) AssertFinalExponentiationIsOne(a *GTEl) {
206206
pr.AssertIsEqual(t0, t2)
207207
}
208208

209-
// PairingCheck calculates the reduced pairing for a set of points and asserts if the result is One
210-
// ∏ᵢ e(Pᵢ, Qᵢ) =? 1
209+
// PairingCheck calculates the reduced pairing for a set of points and asserts
210+
// if the result is one:
211211
//
212-
// This function doesn't check that the inputs are in the correct subgroups. See AssertIsOnG1 and AssertIsOnG2.
212+
// ∏ᵢ e(Pᵢ, Qᵢ) =? 1
213+
//
214+
// This function checks that the Qᵢ are in the correct subgroup, but does not
215+
// check Pᵢ. See AssertIsOnG1.
213216
func (pr Pairing) PairingCheck(P []*G1Affine, Q []*G2Affine) error {
214217
f, err := pr.MillerLoop(P, Q)
215218
if err != nil {
@@ -266,6 +269,7 @@ func (pr Pairing) AssertIsOnG1(P *G1Affine) {
266269
pr.AssertIsOnCurve(P)
267270
}
268271

272+
// computeG2ShortVector computes ψ³([2x₀]Q) - ψ²([x₀]Q) - ψ([x₀]Q) - [x₀]Q
269273
func (pr Pairing) computeG2ShortVector(Q *G2Affine) (_Q *G2Affine) {
270274
// [x₀]Q
271275
xQ := pr.g2.scalarMulBySeed(Q)
@@ -278,7 +282,7 @@ func (pr Pairing) computeG2ShortVector(Q *G2Affine) (_Q *G2Affine) {
278282
psi3xxQ = pr.g2.psi(psi3xxQ)
279283

280284
// _Q = ψ³([2x₀]Q) - ψ²([x₀]Q) - ψ([x₀]Q) - [x₀]Q
281-
_Q = pr.g2.sub(psi2xQ, psi3xxQ)
285+
_Q = pr.g2.sub(psi3xxQ, psi2xQ)
282286
_Q = pr.g2.sub(_Q, psixQ)
283287
_Q = pr.g2.sub(_Q, xQ)
284288
return _Q
@@ -289,8 +293,10 @@ func (pr Pairing) AssertIsOnG2(Q *G2Affine) {
289293
pr.AssertIsOnTwist(Q)
290294

291295
// 2- Check Q has the right subgroup order
296+
// [r]Q == 0 <==> ψ³([2x₀]Q) - ψ²([x₀]Q) - ψ([x₀]Q) - [x₀]Q == Q
297+
// This is a valid short vector since x₀ ≠ 5422 mod 2196.
298+
// See Sec. 3.1.2 (Example 1) in https://eprint.iacr.org/2022/348.
292299
_Q := pr.computeG2ShortVector(Q)
293-
// [r]Q == 0 <==> _Q == Q
294300
pr.g2.AssertIsEqual(Q, _Q)
295301
}
296302

std/algebra/emulated/sw_bn254/precomputations.go

+18
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,10 @@ func precomputeLines(Q bn254.G2Affine) lineEvaluations {
3131

3232
func (p *Pairing) computeLines(Q *g2AffP) lineEvaluations {
3333

34+
// check Q is on curve
35+
Qaff := G2Affine{P: *Q, Lines: nil}
36+
p.IsOnTwist(&Qaff)
37+
3438
var cLines lineEvaluations
3539
Qacc := Q
3640
n := len(bn254.LoopCounter)
@@ -50,6 +54,20 @@ func (p *Pairing) computeLines(Q *g2AffP) lineEvaluations {
5054
}
5155
}
5256

57+
// Check that Q is on G2 subgroup:
58+
// [r]Q == 0 <==> [6x₀+2]Q + ψ(Q) + ψ³(Q) = ψ²(Q).
59+
// This is a valid short vector since x₀ ≠ 4 mod 13 and x₀ ≠ 92 mod 97.
60+
// See Sec. 3.1.2 (Remark 2) in https://eprint.iacr.org/2022/348.
61+
// This test is equivalent to [computeG2ShortVector] in [AssertIsOnG2].
62+
//
63+
// At this point Qacc = [6x₀+2]Q.
64+
psiQ := p.g2.psi(&Qaff) // ψ(Q)
65+
psi2Q := p.g2.phi(&Qaff) // ϕ(Q)=ψ²(Q)
66+
psi3Q := p.g2.psi(psi2Q) // ψ³(Q)
67+
lhs := p.g2.add(&G2Affine{P: *Qacc, Lines: nil}, psiQ)
68+
lhs = p.g2.add(lhs, psi3Q)
69+
p.g2.AssertIsEqual(lhs, psi2Q)
70+
5371
Q1X := p.Ext2.Conjugate(&Q.X)
5472
Q1X = p.Ext2.MulByNonResidue1Power2(Q1X)
5573
Q1Y := p.Ext2.Conjugate(&Q.Y)

std/evmprecompiles/08-bnpairing.go

+2-7
Original file line numberDiff line numberDiff line change
@@ -40,11 +40,8 @@ func ECPair(api frontend.API, P []*sw_bn254.G1Affine, Q []*sw_bn254.G2Affine) {
4040
if err != nil {
4141
panic(err)
4242
}
43-
// 1- Check that Pᵢ are on G1 (done in the zkEVM ⚠️ )
44-
// 2- Check that Qᵢ are on G2
45-
for i := 0; i < len(Q); i++ {
46-
pair.AssertIsOnG2(Q[i])
47-
}
43+
// 1- Check that Pᵢ are on G1 (done in the zkEVM ⚠️
44+
// 2- Check that Qᵢ are on G2 (done in `computeLines` in `MillerLoopAndMul` and `MillerLoopAndFinalExpCheck)
4845

4946
// 3- Check that ∏ᵢ e(Pᵢ, Qᵢ) == 1
5047
ml := pair.Ext12.One()
@@ -79,7 +76,6 @@ func ECPairMillerLoopAndMul(api frontend.API, accumulator *sw_bn254.GTEl, P *sw_
7976
if err != nil {
8077
return fmt.Errorf("new pairing: %w", err)
8178
}
82-
pairing.AssertIsOnG2(Q)
8379
ml, err := pairing.MillerLoopAndMul(P, Q, accumulator)
8480
if err != nil {
8581
return fmt.Errorf("miller loop and mul: %w", err)
@@ -97,7 +93,6 @@ func ECPairMillerLoopAndFinalExpCheck(api frontend.API, accumulator *sw_bn254.GT
9793
if err != nil {
9894
return fmt.Errorf("new pairing: %w", err)
9995
}
100-
pairing.AssertIsOnG2(Q)
10196

10297
isSuccess := pairing.IsMillerLoopAndFinalExpOne(P, Q, accumulator)
10398
api.AssertIsEqual(expectedIsSuccess, isSuccess)

0 commit comments

Comments
 (0)