/* Copyright 2018 0KIMS association. This file is part of circom (Zero Knowledge Circuit Compiler). circom is a free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. circom is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with circom. If not, see . */ include "montgomery.circom"; include "babyjub.circom"; include "comparators.circom"; template Multiplexor2() { signal input sel; signal input in[2][2]; signal output out[2]; out[0] <== (in[1][0] - in[0][0])*sel + in[0][0]; out[1] <== (in[1][1] - in[0][1])*sel + in[0][1]; } template BitElementMulAny() { signal input sel; signal input dblIn[2]; signal input addIn[2]; signal output dblOut[2]; signal output addOut[2]; component doubler = MontgomeryDouble(); component adder = MontgomeryAdd(); component selector = Multiplexor2(); sel ==> selector.sel; dblIn[0] ==> doubler.in[0]; dblIn[1] ==> doubler.in[1]; doubler.out[0] ==> adder.in1[0]; doubler.out[1] ==> adder.in1[1]; addIn[0] ==> adder.in2[0]; addIn[1] ==> adder.in2[1]; addIn[0] ==> selector.in[0][0]; addIn[1] ==> selector.in[0][1]; adder.out[0] ==> selector.in[1][0]; adder.out[1] ==> selector.in[1][1]; doubler.out[0] ==> dblOut[0]; doubler.out[1] ==> dblOut[1]; selector.out[0] ==> addOut[0]; selector.out[1] ==> addOut[1]; } // p is montgomery point // n must be <= 248 // returns out in twisted edwards // Double is in montgomery to be linked; template SegmentMulAny(n) { signal input e[n]; signal input p[2]; signal output out[2]; signal output dbl[2]; component bits[n-1]; component e2m = Edwards2Montgomery(); p[0] ==> e2m.in[0]; p[1] ==> e2m.in[1]; var i; bits[0] = BitElementMulAny(); e2m.out[0] ==> bits[0].dblIn[0] e2m.out[1] ==> bits[0].dblIn[1] e2m.out[0] ==> bits[0].addIn[0] e2m.out[1] ==> bits[0].addIn[1] e[1] ==> bits[0].sel; for (i=1; i bits[i].dblIn[0] bits[i-1].dblOut[1] ==> bits[i].dblIn[1] bits[i-1].addOut[0] ==> bits[i].addIn[0] bits[i-1].addOut[1] ==> bits[i].addIn[1] e[i+1] ==> bits[i].sel; } bits[n-2].dblOut[0] ==> dbl[0]; bits[n-2].dblOut[1] ==> dbl[1]; component m2e = Montgomery2Edwards(); bits[n-2].addOut[0] ==> m2e.in[0]; bits[n-2].addOut[1] ==> m2e.in[1]; component eadder = BabyAdd(); m2e.out[0] ==> eadder.x1; m2e.out[1] ==> eadder.y1; -p[0] ==> eadder.x2; p[1] ==> eadder.y2; component lastSel = Multiplexor2(); e[0] ==> lastSel.sel; eadder.xout ==> lastSel.in[0][0]; eadder.yout ==> lastSel.in[0][1]; m2e.out[0] ==> lastSel.in[1][0]; m2e.out[1] ==> lastSel.in[1][1]; lastSel.out[0] ==> out[0]; lastSel.out[1] ==> out[1]; } // This function assumes that p is in the subgroup and it is different to 0 template EscalarMulAny(n) { signal input e[n]; // Input in binary format signal input p[2]; // Point (Twisted format) signal output out[2]; // Point (Twisted format) var nsegments = (n-1)\148 +1; var nlastsegment = n - (nsegments-1)*148; component segments[nsegments]; component doublers[nsegments-1]; component m2e[nsegments-1]; component adders[nsegments-1]; component zeropoint = IsZero(); zeropoint.in <== p[0]; var s; var i; var nseg; for (s=0; s segments[s].e[i]; } if (s==0) { // force G8 point if input point is zero segments[s].p[0] <== p[0] + (5299619240641551281634865583518297030282874472190772894086521144482721001553 - p[0])*zeropoint.out; segments[s].p[1] <== p[1] + (16950150798460657717958625567821834550301663161624707787222815936182638968203 - p[1])*zeropoint.out; } else { doublers[s-1] = MontgomeryDouble(); m2e[s-1] = Montgomery2Edwards(); adders[s-1] = BabyAdd(); segments[s-1].dbl[0] ==> doublers[s-1].in[0]; segments[s-1].dbl[1] ==> doublers[s-1].in[1]; doublers[s-1].out[0] ==> m2e[s-1].in[0]; doublers[s-1].out[1] ==> m2e[s-1].in[1]; m2e[s-1].out[0] ==> segments[s].p[0]; m2e[s-1].out[1] ==> segments[s].p[1]; if (s==1) { segments[s-1].out[0] ==> adders[s-1].x1; segments[s-1].out[1] ==> adders[s-1].y1; } else { adders[s-2].xout ==> adders[s-1].x1; adders[s-2].yout ==> adders[s-1].y1; } segments[s].out[0] ==> adders[s-1].x2; segments[s].out[1] ==> adders[s-1].y2; } } if (nsegments == 1) { segments[0].out[0]*(1-zeropoint.out) ==> out[0]; segments[0].out[1]+(1-segments[0].out[1])*zeropoint.out ==> out[1]; } else { adders[nsegments-2].xout*(1-zeropoint.out) ==> out[0]; adders[nsegments-2].yout+(1-adders[nsegments-2].yout)*zeropoint.out ==> out[1]; } }