LCOV - code coverage report
Current view: top level - ballet/ed25519 - fd_f25519.h (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 26 26 100.0 %
Date: 2025-01-08 12:08:44 Functions: 20 162 12.3 %

          Line data    Source code
       1             : #ifndef HEADER_fd_src_ballet_ed25519_fd_f25519_h
       2             : #define HEADER_fd_src_ballet_ed25519_fd_f25519_h
       3             : 
       4             : /* fd_f25519.h provides the public field API for the base field of curve25519.
       5             : 
       6             :    Most operations in this API should be assumed to take a variable amount
       7             :    of time depending on inputs, and thus should not be exposed to secret data.
       8             : 
       9             :    Constant-time operations are made explicit. */
      10             : 
      11             : #include "../fd_ballet_base.h"
      12             : #include "../../util/rng/fd_rng.h"
      13             : 
      14             : #define FD_25519_INLINE static inline
      15             : 
      16             : /* fd_f25519_t is the type of a field element, i.e. an integer
      17             :    mod p = 2^255 - 19.
      18             :    it's internal representation (and alignment) depends on the
      19             :    backend: ref, avx, avx512. */
      20             : 
      21             : #if FD_HAS_AVX512
      22             : #include "avx512/fd_f25519.h"
      23             : #else
      24             : #include "ref/fd_f25519.h"
      25             : #endif
      26             : 
      27             : /* field constants. these are imported from table/fd_f25519_table_{arch}.c.
      28             :    they are (re)defined here to avoid breaking compilation when the table needs
      29             :    to be rebuilt. */
      30             : static const fd_f25519_t fd_f25519_zero[1];
      31             : static const fd_f25519_t fd_f25519_one[1];
      32             : static const fd_f25519_t fd_f25519_minus_one[1];
      33             : static const fd_f25519_t fd_f25519_two[1];
      34             : static const fd_f25519_t fd_f25519_nine[1];
      35             : static const fd_f25519_t fd_f25519_k[1];
      36             : static const fd_f25519_t fd_f25519_minus_k[1];
      37             : static const fd_f25519_t fd_f25519_d[1];
      38             : static const fd_f25519_t fd_f25519_sqrtm1[1];
      39             : static const fd_f25519_t fd_f25519_invsqrt_a_minus_d[1];
      40             : static const fd_f25519_t fd_f25519_one_minus_d_sq[1];
      41             : static const fd_f25519_t fd_f25519_d_minus_one_sq[1];
      42             : static const fd_f25519_t fd_f25519_sqrt_ad_minus_one[1];
      43             : 
      44             : FD_PROTOTYPES_BEGIN
      45             : 
      46             : /* fd_f25519_mul computes r = a * b, and returns r. */
      47             : fd_f25519_t *
      48             : fd_f25519_mul( fd_f25519_t *       r,
      49             :                fd_f25519_t const * a,
      50             :                fd_f25519_t const * b );
      51             : 
      52             : /* fd_f25519_sqr computes r = a^2, and returns r. */
      53             : fd_f25519_t *
      54             : fd_f25519_sqr( fd_f25519_t *       r,
      55             :                fd_f25519_t const * a );
      56             : 
      57             : /* fd_f25519_add computes r = a + b, and returns r. */
      58             : fd_f25519_t *
      59             : fd_f25519_add( fd_f25519_t *       r,
      60             :                fd_f25519_t const * a,
      61             :                fd_f25519_t const * b );
      62             : 
      63             : /* fd_f25519_add computes r = a - b, and returns r. */
      64             : fd_f25519_t *
      65             : fd_f25519_sub( fd_f25519_t *       r,
      66             :                fd_f25519_t const * a,
      67             :                fd_f25519_t const * b );
      68             : 
      69             : /* fd_f25519_add_nr computes r = a + b, and returns r.
      70             :    Note: this does NOT reduce the result mod p.
      71             :    It can be used before mul, sqr. */
      72             : fd_f25519_t *
      73             : fd_f25519_add_nr( fd_f25519_t * r,
      74             :                   fd_f25519_t const * a,
      75             :                   fd_f25519_t const * b );
      76             : 
      77             : /* fd_f25519_sub_nr computes r = a - b, and returns r.
      78             :    Note: this does NOT reduce the result mod p.
      79             :    It can be used before mul, sqr. */
      80             : fd_f25519_t *
      81             : fd_f25519_sub_nr( fd_f25519_t * r,
      82             :                   fd_f25519_t const * a,
      83             :                   fd_f25519_t const * b );
      84             : 
      85             : /* fd_f25519_add computes r = -a, and returns r. */
      86             : fd_f25519_t *
      87             : fd_f25519_neg( fd_f25519_t *       r,
      88             :                fd_f25519_t const * a );
      89             : 
      90             : /* fd_f25519_mul_121666 computes r = a * k, k=121666, and returns r. */
      91             : fd_f25519_t *
      92             : fd_f25519_mul_121666( fd_f25519_t *       r,
      93             :                       fd_f25519_t const * a );
      94             : 
      95             : /* fd_f25519_frombytes deserializes a 32-byte buffer buf into a
      96             :    fd_f25519_t element r, and returns r.
      97             :    buf is in little endian form, according to RFC 8032. */
      98             : fd_f25519_t *
      99             : fd_f25519_frombytes( fd_f25519_t * r,
     100             :                      uchar const   buf[ 32 ] );
     101             : 
     102             : /* fd_f25519_tobytes serializes a fd_f25519_t element a into
     103             :    a 32-byte buffer out, and returns out.
     104             :    out is in little endian form, according to RFC 8032. */
     105             : uchar *
     106             : fd_f25519_tobytes( uchar               out[ 32 ],
     107             :                    fd_f25519_t const * a );
     108             : 
     109             : /* fd_f25519_set copies r = a, and returns r. */
     110             : fd_f25519_t *
     111             : fd_f25519_set( fd_f25519_t *       r,
     112             :                fd_f25519_t const * a );
     113             : 
     114             : /* fd_f25519_is_zero returns 1 if a == 0, 0 otherwise. */
     115             : int
     116             : fd_f25519_is_zero( fd_f25519_t const * a );
     117             : 
     118             : /* fd_f25519_if sets r = a0 if cond, else r = a1, equivalent to:
     119             :    r = cond ? a0 : a1.
     120             :    Note: this is constant time. */
     121             : fd_f25519_t *
     122             : fd_f25519_if( fd_f25519_t *       r,
     123             :               int const           cond, /* 0, 1 */
     124             :               fd_f25519_t const * a0,
     125             :               fd_f25519_t const * a1 );
     126             : 
     127             : /* fd_f25519_rng generates a random fd_f25519_t element.
     128             :    Note: insecure, for tests only. */
     129             : fd_f25519_t *
     130             : fd_f25519_rng_unsafe( fd_f25519_t * r,
     131             :                       fd_rng_t *    rng );
     132             : 
     133             : /*
     134             :  * Derived
     135             :  */
     136             : 
     137             : /* fd_f25519_eq returns 1 if a == b, 0 otherwise. */
     138             : FD_25519_INLINE int
     139             : fd_f25519_eq( fd_f25519_t const * a,
     140     9641469 :               fd_f25519_t const * b ) {
     141     9641469 :   fd_f25519_t r[1];
     142     9641469 :   fd_f25519_sub( r, a, b );
     143     9641469 :   return fd_f25519_is_zero( r );
     144     9641469 : }
     145             : 
     146             : /* fd_f25519_is_nonzero returns 1 (true) if a != 0, 0 if a == 0. */
     147             : FD_25519_INLINE int
     148      300000 : fd_f25519_is_nonzero( fd_f25519_t const * a ) {
     149      300000 :   return !fd_f25519_is_zero( a );
     150      300000 : }
     151             : 
     152             : /* fd_f25519_sgn returns the sign of a (lsb). */
     153             : FD_25519_INLINE int
     154     6251035 : fd_f25519_sgn( fd_f25519_t const * a ) {
     155             :   //TODO: make it faster (unless inlining already optimizes out unnecessary code)
     156     6251035 :   uchar buf[32];
     157     6251035 :   fd_f25519_tobytes( buf, a );
     158     6251035 :   return buf[0] & 1;
     159     6251035 : }
     160             : 
     161             : /* fd_f25519_abs sets r = |a|. */
     162             : FD_25519_INLINE fd_f25519_t *
     163             : fd_f25519_abs( fd_f25519_t *       r,
     164     2444350 :                fd_f25519_t const * a ) {
     165     2444350 :   fd_f25519_t neg_a[1];
     166     2444350 :   fd_f25519_neg( neg_a, a );
     167     2444350 :   return fd_f25519_if( r, fd_f25519_sgn(a), neg_a, a );
     168     2444350 : }
     169             : 
     170             : /* fd_f25519_abs sets r = -|a|. */
     171             : FD_25519_INLINE fd_f25519_t *
     172             : fd_f25519_neg_abs( fd_f25519_t *       r,
     173       90006 :                    fd_f25519_t const * a ) {
     174       90006 :   fd_f25519_t neg_a[1];
     175       90006 :   fd_f25519_neg( neg_a, a );
     176       90006 :   return fd_f25519_if( r, fd_f25519_sgn(a), a, neg_a );
     177       90006 : }
     178             : 
     179             : /*
     180             :  * Inv & Sqrt
     181             :  */
     182             : 
     183             : /* fd_f25519_inv computes r = 1/a, and returns r. */
     184             : fd_f25519_t *
     185             : fd_f25519_inv( fd_f25519_t *       r,
     186             :                fd_f25519_t const * a );
     187             : 
     188             : /* fd_f25519_pow22523 computes r = a^(2^252-3), and returns r. */
     189             : fd_f25519_t *
     190             : fd_f25519_pow22523( fd_f25519_t *       r,
     191             :                     fd_f25519_t const * a );
     192             : 
     193             : /* fd_f25519_sqrt_ratio computes r = (u * v^3) * (u * v^7)^((p-5)/8),
     194             :    returns 0 on success, 1 on failure. */
     195             : int
     196             : fd_f25519_sqrt_ratio( fd_f25519_t *       r,
     197             :                       fd_f25519_t const * u,
     198             :                       fd_f25519_t const * v );
     199             : 
     200             : /* fd_f25519_sqrt_ratio computes r = 1/sqrt(v),
     201             :    returns 0 on success, 1 on failure. */
     202             : FD_25519_INLINE int
     203             : fd_f25519_inv_sqrt( fd_f25519_t *       r,
     204      607992 :                     fd_f25519_t const * v ) {
     205      607992 :   return fd_f25519_sqrt_ratio( r, fd_f25519_one, v );
     206      607992 : }
     207             : 
     208             : /*
     209             :  * Vectorized
     210             :  */
     211             : 
     212             : /* fd_f25519_muln computes r_i = a_i * b_i */
     213             : void
     214             : fd_f25519_mul2( fd_f25519_t * r1, fd_f25519_t const * a1, fd_f25519_t const * b1,
     215             :                 fd_f25519_t * r2, fd_f25519_t const * a2, fd_f25519_t const * b2 );
     216             : 
     217             : void
     218             : fd_f25519_mul3( fd_f25519_t * r1, fd_f25519_t const * a1, fd_f25519_t const * b1,
     219             :                 fd_f25519_t * r2, fd_f25519_t const * a2, fd_f25519_t const * b2,
     220             :                 fd_f25519_t * r3, fd_f25519_t const * a3, fd_f25519_t const * b3 );
     221             : 
     222             : void
     223             : fd_f25519_mul4( fd_f25519_t * r1, fd_f25519_t const * a1, fd_f25519_t const * b1,
     224             :                 fd_f25519_t * r2, fd_f25519_t const * a2, fd_f25519_t const * b2,
     225             :                 fd_f25519_t * r3, fd_f25519_t const * a3, fd_f25519_t const * b3,
     226             :                 fd_f25519_t * r4, fd_f25519_t const * a4, fd_f25519_t const * b4 );
     227             : 
     228             : /* fd_f25519_sqrn computes r_i = a_i^2 */
     229             : void
     230             : fd_f25519_sqr2( fd_f25519_t * r1, fd_f25519_t const * a1,
     231             :                 fd_f25519_t * r2, fd_f25519_t const * a2 );
     232             : 
     233             : void
     234             : fd_f25519_sqr3( fd_f25519_t * r1, fd_f25519_t const * a1,
     235             :                 fd_f25519_t * r2, fd_f25519_t const * a2,
     236             :                 fd_f25519_t * r3, fd_f25519_t const * a3 );
     237             : 
     238             : void
     239             : fd_f25519_sqr4( fd_f25519_t * r1, fd_f25519_t const * a1,
     240             :                 fd_f25519_t * r2, fd_f25519_t const * a2,
     241             :                 fd_f25519_t * r3, fd_f25519_t const * a3,
     242             :                 fd_f25519_t * r4, fd_f25519_t const * a4 );
     243             : 
     244             : /* fd_f25519_pow22523 computes r = a^(2^252-3), and returns r. */
     245             : fd_f25519_t *
     246             : fd_f25519_pow22523_2( fd_f25519_t * r1, fd_f25519_t const * a1,
     247             :                       fd_f25519_t * r2, fd_f25519_t const * a2 );
     248             : 
     249             : /* fd_f25519_sqrt_ratio computes r = (u * v^3) * (u * v^7)^((p-5)/8),
     250             :    returns 0 on success, 1 on failure. */
     251             : int
     252             : fd_f25519_sqrt_ratio2( fd_f25519_t * r1, fd_f25519_t const * u1, fd_f25519_t const * v1,
     253             :                        fd_f25519_t * r2, fd_f25519_t const * u2, fd_f25519_t const * v2 );
     254             : 
     255             : FD_PROTOTYPES_END
     256             : 
     257             : #endif /* HEADER_fd_src_ballet_ed25519_fd_f25519_h */

Generated by: LCOV version 1.14