-
Notifications
You must be signed in to change notification settings - Fork 154
Commit
- Loading branch information
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
// @note This code was taken from | ||
// https://github.com/yondonfu/sol-baby-jubjub/blob/master/contracts/CurveBabyJubJub.sol | ||
// Thanks to yondonfu for the code | ||
// Implementation cited on baby-jubjub's paper | ||
// https://eips.ethereum.org/EIPS/eip-2494#implementation | ||
|
||
// SPDX-License-Identifier: MIT | ||
pragma solidity ^0.8.20; | ||
|
||
library CurveBabyJubJub { | ||
// Curve parameters | ||
// E: 168700x^2 + y^2 = 1 + 168696x^2y^2 | ||
// A = 168700 | ||
uint256 public constant A = 0x292FC; | ||
// D = 168696 | ||
uint256 public constant D = 0x292F8; | ||
// Prime Q = 21888242871839275222246405745257275088548364400416034343698204186575808495617 | ||
uint256 public constant Q = 0x30644E72E131A029B85045B68181585D2833E84879B9709143E1F593F0000001; | ||
|
||
/** | ||
* @dev Add 2 points on baby jubjub curve | ||
* Formula for adding 2 points on a twisted Edwards curve: | ||
* x3 = (x1y2 + y1x2) / (1 + dx1x2y1y2) | ||
* y3 = (y1y2 - ax1x2) / (1 - dx1x2y1y2) | ||
*/ | ||
function pointAdd(uint256 _x1, uint256 _y1, uint256 _x2, uint256 _y2) internal view returns (uint256 x3, uint256 y3) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointAdd(uint256,uint256,uint256,uint256)._x2 is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointAdd(uint256,uint256,uint256,uint256)._y1 is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointAdd(uint256,uint256,uint256,uint256)._y2 is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointAdd(uint256,uint256,uint256,uint256)._x1 is not in mixedCase
|
||
if (_x1 == 0 && _y1 == 0) { | ||
return (_x2, _y2); | ||
} | ||
|
||
if (_x2 == 0 && _y1 == 0) { | ||
return (_x1, _y1); | ||
} | ||
|
||
uint256 x1x2 = mulmod(_x1, _x2, Q); | ||
uint256 y1y2 = mulmod(_y1, _y2, Q); | ||
uint256 dx1x2y1y2 = mulmod(D, mulmod(x1x2, y1y2, Q), Q); | ||
uint256 x3Num = addmod(mulmod(_x1, _y2, Q), mulmod(_y1, _x2, Q), Q); | ||
uint256 y3Num = submod(y1y2, mulmod(A, x1x2, Q), Q); | ||
|
||
x3 = mulmod(x3Num, inverse(addmod(1, dx1x2y1y2, Q)), Q); | ||
y3 = mulmod(y3Num, inverse(submod(1, dx1x2y1y2, Q)), Q); | ||
} | ||
Check warning Code scanning / Slither Dead-code Warning
CurveBabyJubJub.pointAdd(uint256,uint256,uint256,uint256) is never used and should be removed
|
||
|
||
/** | ||
* @dev Double a point on baby jubjub curve | ||
* Doubling can be performed with the same formula as addition | ||
*/ | ||
function pointDouble(uint256 _x1, uint256 _y1) internal view returns (uint256 x2, uint256 y2) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointDouble(uint256,uint256)._y1 is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointDouble(uint256,uint256)._x1 is not in mixedCase
|
||
return pointAdd(_x1, _y1, _x1, _y1); | ||
} | ||
Check warning Code scanning / Slither Dead-code Warning
CurveBabyJubJub.pointDouble(uint256,uint256) is never used and should be removed
|
||
|
||
/** | ||
* @dev Multiply a point on baby jubjub curve by a scalar | ||
* Use the double and add algorithm | ||
*/ | ||
function pointMul(uint256 _x1, uint256 _y1, uint256 _d) internal view returns (uint256 x2, uint256 y2) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointMul(uint256,uint256,uint256)._y1 is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointMul(uint256,uint256,uint256)._x1 is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.pointMul(uint256,uint256,uint256)._d is not in mixedCase
|
||
uint256 remaining = _d; | ||
|
||
uint256 px = _x1; | ||
uint256 py = _y1; | ||
uint256 ax = 0; | ||
uint256 ay = 0; | ||
|
||
while (remaining != 0) { | ||
if ((remaining & 1) != 0) { | ||
// Binary digit is 1 so add | ||
(ax, ay) = pointAdd(ax, ay, px, py); | ||
} | ||
|
||
(px, py) = pointDouble(px, py); | ||
|
||
remaining = remaining / 2; | ||
} | ||
|
||
x2 = ax; | ||
y2 = ay; | ||
} | ||
Check warning Code scanning / Slither Dead-code Warning
CurveBabyJubJub.pointMul(uint256,uint256,uint256) is never used and should be removed
|
||
|
||
/** | ||
* @dev Check if a given point is on the curve | ||
* (168700x^2 + y^2) - (1 + 168696x^2y^2) == 0 | ||
*/ | ||
function isOnCurve(uint256 _x, uint256 _y) internal pure returns (bool) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.isOnCurve(uint256,uint256)._x is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.isOnCurve(uint256,uint256)._y is not in mixedCase
|
||
uint256 xSq = mulmod(_x, _x, Q); | ||
uint256 ySq = mulmod(_y, _y, Q); | ||
uint256 lhs = addmod(mulmod(A, xSq, Q), ySq, Q); | ||
uint256 rhs = addmod(1, mulmod(mulmod(D, xSq, Q), ySq, Q), Q); | ||
return submod(lhs, rhs, Q) == 0; | ||
} | ||
|
||
/** | ||
* @dev Perform modular subtraction | ||
*/ | ||
function submod(uint256 _a, uint256 _b, uint256 _mod) internal pure returns (uint256) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.submod(uint256,uint256,uint256)._b is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.submod(uint256,uint256,uint256)._a is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.submod(uint256,uint256,uint256)._mod is not in mixedCase
|
||
uint256 aNN = _a; | ||
|
||
if (_a <= _b) { | ||
aNN += _mod; | ||
} | ||
|
||
return addmod(aNN - _b, 0, _mod); | ||
} | ||
|
||
/** | ||
* @dev Compute modular inverse of a number | ||
*/ | ||
function inverse(uint256 _a) internal view returns (uint256) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.inverse(uint256)._a is not in mixedCase
|
||
// We can use Euler's theorem instead of the extended Euclidean algorithm | ||
// Since m = Q and Q is prime we have: a^-1 = a^(m - 2) (mod m) | ||
return expmod(_a, Q - 2, Q); | ||
} | ||
Check warning Code scanning / Slither Dead-code Warning
CurveBabyJubJub.inverse(uint256) is never used and should be removed
|
||
|
||
/** | ||
* @dev Helper function to call the bigModExp precompile | ||
*/ | ||
function expmod(uint256 _b, uint256 _e, uint256 _m) internal view returns (uint256 o) { | ||
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.expmod(uint256,uint256,uint256)._e is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.expmod(uint256,uint256,uint256)._b is not in mixedCase
Check warning Code scanning / Slither Conformance to Solidity naming conventions Warning
Parameter CurveBabyJubJub.expmod(uint256,uint256,uint256)._m is not in mixedCase
|
||
assembly { | ||
let memPtr := mload(0x40) | ||
mstore(memPtr, 0x20) // Length of base _b | ||
mstore(add(memPtr, 0x20), 0x20) // Length of exponent _e | ||
mstore(add(memPtr, 0x40), 0x20) // Length of modulus _m | ||
mstore(add(memPtr, 0x60), _b) // Base _b | ||
mstore(add(memPtr, 0x80), _e) // Exponent _e | ||
mstore(add(memPtr, 0xa0), _m) // Modulus _m | ||
|
||
// The bigModExp precompile is at 0x05 | ||
let success := staticcall(gas(), 0x05, memPtr, 0xc0, memPtr, 0x20) | ||
switch success | ||
case 0 { | ||
revert(0x0, 0x0) | ||
} | ||
default { | ||
o := mload(memPtr) | ||
} | ||
} | ||
} | ||
Check warning Code scanning / Slither Assembly usage Warning Check warning Code scanning / Slither Dead-code Warning
CurveBabyJubJub.expmod(uint256,uint256,uint256) is never used and should be removed
|
||
} | ||
Check warning Code scanning / Slither Too many digits Warning
CurveBabyJubJub.slitherConstructorConstantVariables() uses literals with too many digits:
- Q = 0x30644E72E131A029B85045B68181585D2833E84879B9709143E1F593F0000001 |