1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2024-12-22 14:28:07 +00:00
trezor-firmware/crypto/elligator2.c
2024-05-10 20:05:37 +02:00

191 lines
5.1 KiB
C

/** Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions: The above copyright
* notice and this permission notice shall be included in all copies or
* substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES
* OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/
#include <assert.h>
#include <stdbool.h>
#include "ed25519-donna/ed25519-donna.h"
#include "memzero.h"
#include "elligator2.h"
// Returns 1 if a equals b, returns 0 otherwise.
static int curve25519_isequal(bignum25519 a, const bignum25519 b) {
bignum25519 difference;
curve25519_sub(difference, a, b);
int result = 1 - curve25519_isnonzero(difference);
memzero(difference, sizeof(difference));
return result;
}
// Sets out to a if c equals 0, sets out to b if c equals 1.
static void curve25519_cmov(bignum25519 out, const bignum25519 a,
const bignum25519 b, uint32_t c) {
assert((int)(c == 1) | (int)(c == 0));
bignum25519 a_copy = {0}, b_copy = {0};
curve25519_copy(a_copy, a);
curve25519_copy(b_copy, b);
curve25519_swap_conditional(a_copy, b_copy, c);
memzero(b_copy, sizeof(b_copy));
curve25519_copy(out, a_copy);
memzero(a_copy, sizeof(a_copy));
}
bool map_to_curve_elligator2_curve25519(const uint8_t input[32],
curve25519_key output) {
// https://www.rfc-editor.org/rfc/rfc9380.html#map-to-curve25519
// The procedure from the above link is used, with the exception that the
// y-coordinate of the output point is not computed, because it is not needed.
bignum25519 input_bignum = {0};
curve25519_expand(input_bignum, (unsigned char*)input);
// c3 = sqrt(-1)
bignum25519 c3 = {0};
curve25519_set_sqrtneg1(c3);
// J = 486662
bignum25519 j = {0};
curve25519_set(j, 486662);
// tv1 = u^2
bignum25519 tv1 = {0};
curve25519_square(tv1, input_bignum);
memzero(input_bignum, sizeof(input_bignum));
// tv1 = 2 * tv1
curve25519_add_reduce(tv1, tv1, tv1);
// xd = tv1 + 1
bignum25519 xd = {0};
bignum25519 one = {0};
curve25519_set(one, 1);
curve25519_add_reduce(xd, tv1, one);
memzero(one, sizeof(one));
// x1n = -J
bignum25519 x1n = {0};
curve25519_neg(x1n, j);
// tv2 = xd^2
bignum25519 tv2 = {0};
curve25519_square(tv2, xd);
// gxd = tv2 * xd
bignum25519 gxd = {0};
curve25519_mul(gxd, tv2, xd);
// gx1 = J * tv1
bignum25519 gx1 = {0};
curve25519_mul(gx1, j, tv1);
memzero(j, sizeof(j));
// gx1 = gx1 * x1n
curve25519_mul(gx1, gx1, x1n);
// gx1 = gx1 + tv2
curve25519_add_reduce(gx1, gx1, tv2);
// gx1 = gx1 * x1n
curve25519_mul(gx1, gx1, x1n);
// tv3 = gxd^2
bignum25519 tv3 = {0};
curve25519_square(tv3, gxd);
// tv2 = tv3^2
curve25519_square(tv2, tv3);
// tv3 = tv3 * gxd
curve25519_mul(tv3, tv3, gxd);
// tv3 = tv3 * gx1
curve25519_mul(tv3, tv3, gx1);
// tv2 = tv2 * tv3
curve25519_mul(tv2, tv2, tv3);
// y11 = tv2^c4
bignum25519 y11 = {0};
curve25519_pow_two252m3(y11, tv2);
// y11 = y11 * tv3
curve25519_mul(y11, y11, tv3);
memzero(tv3, sizeof(tv3));
// y12 = y11 * c3
bignum25519 y12 = {0};
curve25519_mul(y12, y11, c3);
memzero(c3, sizeof(c3));
// tv2 = y11^2
curve25519_square(tv2, y11);
// tv2 = tv2 * gxd
curve25519_mul(tv2, tv2, gxd);
// e1 = tv2 == gx1
int e1 = curve25519_isequal(tv2, gx1);
// y1 = CMOV(y12, y11, e1)
bignum25519 y1 = {0};
curve25519_cmov(y1, y12, y11, e1);
memzero(y12, sizeof(y12));
memzero(y11, sizeof(y11));
memzero(&e1, sizeof(e1));
// x2n = x1n * tv1
bignum25519 x2n = {0};
curve25519_mul(x2n, x1n, tv1);
memzero(tv1, sizeof(tv1));
// tv2 = y1^2
curve25519_square(tv2, y1);
memzero(y1, sizeof(y1));
// tv2 = tv2 * gxd
curve25519_mul(tv2, tv2, gxd);
memzero(gxd, sizeof(gxd));
// e3 = tv2 == gx1
int e3 = curve25519_isequal(tv2, gx1);
memzero(tv2, sizeof(tv2));
memzero(gx1, sizeof(gx1));
// xn = CMOV(x2n, x1n, e3)
bignum25519 xn = {0};
curve25519_cmov(xn, x2n, x1n, e3);
memzero(x1n, sizeof(x1n));
memzero(x2n, sizeof(x2n));
memzero(&e3, sizeof(e3));
// Compute the x-coordinate of the output point
// x = xn / xd
bignum25519 x = {0};
curve25519_recip(x, xd);
memzero(xd, sizeof(xd));
curve25519_mul(x, xn, x);
memzero(xn, sizeof(xn));
// output = x
curve25519_contract(output, x);
memzero(x, sizeof(x));
return true;
}