LCOV - code coverage report
Current view: top level - ballet/ed25519/ref - fd_curve25519.h (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 104 110 94.5 %
Date: 2025-01-08 12:08:44 Functions: 36 350 10.3 %

          Line data    Source code
       1             : #ifndef HEADER_fd_src_ballet_ed25519_fd_curve25519_h
       2             : #error "Do not include this directly; use fd_curve25519.h"
       3             : #endif
       4             : 
       5             : /* fd_curve25519.h provides the public Curve25519 API.
       6             : 
       7             :    Most operations in this API should be assumed to take a variable
       8             :    amount of time depending on inputs.  (And thus should not be exposed
       9             :    to secret data).
      10             : 
      11             :    Const time operations are made explicit, see fd_curve25519_secure.c */
      12             : 
      13             : #include "../../fd_ballet_base.h"
      14             : #include "../fd_f25519.h"
      15             : #include "../fd_curve25519_scalar.h"
      16             : 
      17             : /* CURVE25519_PRECOMP_XY turns on/off the precomputation of (Y-X), (Y+X)
      18             :    in precomputation tables. */
      19             : #define CURVE25519_PRECOMP_XY 1
      20             : 
      21             : /* struct fd_curve25519_edwards (aka fd_curve25519_edwards_t) represents
      22             :    a point in Extended Twisted Edwards Coordinates.
      23             :    https://eprint.iacr.org/2008/522 */
      24             : struct fd_curve25519_edwards {
      25             :   fd_f25519_t X[1];
      26             :   fd_f25519_t Y[1];
      27             :   fd_f25519_t T[1];
      28             :   fd_f25519_t Z[1];
      29             : };
      30             : typedef struct fd_curve25519_edwards fd_curve25519_edwards_t;
      31             : 
      32             : typedef fd_curve25519_edwards_t fd_ed25519_point_t;
      33             : typedef fd_curve25519_edwards_t fd_ristretto255_point_t;
      34             : 
      35             : #include "../table/fd_curve25519_table_ref.c"
      36             : 
      37             : FD_PROTOTYPES_BEGIN
      38             : 
      39             : /* fd_ed25519_point_set sets r = 0 (point at infinity). */
      40             : FD_25519_INLINE fd_ed25519_point_t *
      41     1714754 : fd_ed25519_point_set_zero( fd_ed25519_point_t * r ) {
      42     1714754 :   fd_f25519_set( r->X, fd_f25519_zero );
      43     1714754 :   fd_f25519_set( r->Y, fd_f25519_one );
      44     1714754 :   fd_f25519_set( r->Z, fd_f25519_one );
      45     1714754 :   fd_f25519_set( r->T, fd_f25519_zero );
      46     1714754 :   return r;
      47     1714754 : }
      48             : 
      49             : /* fd_ed25519_point_set_zero_precomputed sets r = 0 (point at infinity). */
      50             : FD_25519_INLINE fd_ed25519_point_t *
      51    75091456 : fd_ed25519_point_set_zero_precomputed( fd_ed25519_point_t * r ) {
      52    75091456 : #if CURVE25519_PRECOMP_XY
      53    75091456 :   fd_f25519_set( r->X, fd_f25519_one );  /* Y-X = 1-0 = 1 */
      54    75091456 :   fd_f25519_set( r->Y, fd_f25519_one );  /* Y+X = 1+0 = 1 */
      55    75091456 :   fd_f25519_set( r->Z, fd_f25519_one );  /* Z = 1 */
      56    75091456 :   fd_f25519_set( r->T, fd_f25519_zero ); /* kT = 0 */
      57    75091456 :   return r;
      58             : #else
      59             :   return fd_ed25519_point_set_zero( r );
      60             : #endif
      61    75091456 : }
      62             : 
      63             : /* fd_ed25519_point_set sets r = a. */
      64             : FD_25519_INLINE fd_ed25519_point_t * FD_FN_NO_ASAN
      65             : fd_ed25519_point_set( fd_ed25519_point_t *       r,
      66      681632 :                       fd_ed25519_point_t const * a ) {
      67      681632 :   fd_f25519_set( r->X, a->X );
      68      681632 :   fd_f25519_set( r->Y, a->Y );
      69      681632 :   fd_f25519_set( r->Z, a->Z );
      70      681632 :   fd_f25519_set( r->T, a->T );
      71      681632 :   return r;
      72      681632 : }
      73             : 
      74             : /* fd_ed25519_point_from sets r = (x : y : z : t). */
      75             : FD_25519_INLINE fd_ed25519_point_t *
      76             : fd_ed25519_point_from( fd_ed25519_point_t * r,
      77             :                        fd_f25519_t const *  x,
      78             :                        fd_f25519_t const *  y,
      79             :                        fd_f25519_t const *  z,
      80     1313512 :                        fd_f25519_t const *  t ) {
      81     1313512 :   fd_f25519_set( r->X, x );
      82     1313512 :   fd_f25519_set( r->Y, y );
      83     1313512 :   fd_f25519_set( r->Z, z );
      84     1313512 :   fd_f25519_set( r->T, t );
      85     1313512 :   return r;
      86     1313512 : }
      87             : 
      88             : /* fd_ed25519_point_from sets (x : y : z : t) = a. */
      89             : FD_25519_INLINE void
      90             : fd_ed25519_point_to( fd_f25519_t *  x,
      91             :                      fd_f25519_t *  y,
      92             :                      fd_f25519_t *  z,
      93             :                      fd_f25519_t *  t,
      94     2411242 :                      fd_ed25519_point_t const * a ) {
      95     2411242 :   fd_f25519_set( x, a->X );
      96     2411242 :   fd_f25519_set( y, a->Y );
      97     2411242 :   fd_f25519_set( z, a->Z );
      98     2411242 :   fd_f25519_set( t, a->T );
      99     2411242 : }
     100             : 
     101             : /* fd_ed25519_point_sub sets r = -a. */
     102             : FD_25519_INLINE fd_ed25519_point_t *
     103             : fd_ed25519_point_neg( fd_ed25519_point_t *       r,
     104      515050 :                       fd_ed25519_point_t const * a ) {
     105      515050 :   fd_f25519_neg( r->X, a->X );
     106      515050 :   fd_f25519_set( r->Y, a->Y );
     107      515050 :   fd_f25519_set( r->Z, a->Z );
     108      515050 :   fd_f25519_neg( r->T, a->T );
     109      515050 :   return r;
     110      515050 : }
     111             : 
     112             : /* fd_ed25519_point_is_zero returns 1 if a == 0 (point at infinity), 0 otherwise. */
     113             : FD_25519_INLINE int
     114           0 : fd_ed25519_point_is_zero( fd_ed25519_point_t const * a ) {
     115           0 :   return fd_f25519_is_zero( a->X ) & fd_f25519_eq( a->Y, a->Z );
     116           0 : }
     117             : 
     118             : /* fd_ed25519_point_eq returns 1 if a == b, 0 otherwise. */
     119             : FD_25519_INLINE int
     120             : fd_ed25519_point_eq( fd_ed25519_point_t const * a,
     121           2 :                      fd_ed25519_point_t const * b ) {
     122           2 :   fd_f25519_t x1[1], x2[1], y1[1], y2[1];
     123           2 :   fd_f25519_mul( x1, b->X, a->Z );
     124           2 :   fd_f25519_mul( x2, a->X, b->Z );
     125           2 :   fd_f25519_mul( y1, b->Y, a->Z );
     126           2 :   fd_f25519_mul( y2, a->Y, b->Z );
     127           2 :   return fd_f25519_eq( x1, x2 ) & fd_f25519_eq( y1, y2 );
     128           2 : }
     129             : 
     130             : /* fd_ed25519_point_eq returns 1 if a == b, 0 otherwise.
     131             :    b is a point with Z==1, e.g. a decompressed point. */
     132             : FD_25519_INLINE int
     133             : fd_ed25519_point_eq_z1( fd_ed25519_point_t const * a,
     134      515050 :                         fd_ed25519_point_t const * b ) { /* b.Z == 1, e.g. a decompressed point */
     135      515050 :   fd_f25519_t x1[1], y1[1];
     136      515050 :   fd_f25519_mul( x1, b->X, a->Z );
     137      515050 :   fd_f25519_mul( y1, b->Y, a->Z );
     138      515050 :   return fd_f25519_eq( x1, a->X ) & fd_f25519_eq( y1, a->Y );
     139      515050 : }
     140             : 
     141             : /* fd_curve25519_into_precomputed transforms a point into
     142             :    precomputed table format, e.g. replaces T -> kT to save
     143             :    1mul in the dbl-and-add loop. */
     144             : FD_25519_INLINE void
     145     5447872 : fd_curve25519_into_precomputed( fd_ed25519_point_t * r ) {
     146     5447872 : #if CURVE25519_PRECOMP_XY
     147     5447872 :   fd_f25519_t add[1], sub[1];
     148     5447872 :   fd_f25519_add_nr( add, r->Y, r->X );
     149     5447872 :   fd_f25519_sub_nr( sub, r->Y, r->X );
     150     5447872 :   fd_f25519_set( r->X, sub );
     151     5447872 :   fd_f25519_set( r->Y, add );
     152     5447872 : #endif
     153     5447872 :   fd_f25519_mul( r->T, r->T, fd_f25519_k );
     154     5447872 : }
     155             : 
     156             : /* fd_ed25519_point_add_final_mul computes just the final mul step in point add.
     157             :    See fd_ed25519_point_add_with_opts. */
     158             : FD_25519_INLINE fd_ed25519_point_t *
     159             : fd_ed25519_point_add_final_mul( fd_ed25519_point_t * restrict r,
     160    47445627 :                                 fd_ed25519_point_t const *    a ) {
     161    47445627 :   fd_f25519_t const *r1 = a->X;
     162    47445627 :   fd_f25519_t const *r2 = a->Y;
     163    47445627 :   fd_f25519_t const *r3 = a->Z;
     164    47445627 :   fd_f25519_t const *r4 = a->T;
     165             : 
     166    47445627 :   fd_f25519_mul4( r->X, r1, r2,
     167    47445627 :                   r->Y, r3, r4,
     168    47445627 :                   r->Z, r2, r3,
     169    47445627 :                   r->T, r1, r4 );
     170    47445627 :   return r;
     171    47445627 : }
     172             : 
     173             : /* fd_ed25519_point_add_final_mul_projective computes just the final mul step
     174             :    in point add, assuming the result is projective (X, Y, Z), i.e. ignoring T.
     175             :    This is useful because dbl only needs (X, Y, Z) in input, so we can save 1mul.
     176             :    See fd_ed25519_point_add_with_opts. */
     177             : FD_25519_INLINE fd_ed25519_point_t *
     178             : fd_ed25519_point_add_final_mul_projective( fd_ed25519_point_t * restrict r,
     179   134789719 :                                            fd_ed25519_point_t const *    a ) {
     180   134789719 :   fd_f25519_mul3( r->X, a->X, a->Y,
     181   134789719 :                   r->Y, a->Z, a->T,
     182   134789719 :                   r->Z, a->Y, a->Z );
     183   134789719 :   return r;
     184   134789719 : }
     185             : 
     186             : /* Dedicated dbl
     187             :    https://eprint.iacr.org/2008/522
     188             :    Sec 4.4.
     189             :    This uses sqr instead of mul. */
     190             : FD_25519_INLINE fd_ed25519_point_t *
     191             : fd_ed25519_partial_dbl( fd_ed25519_point_t *       r,
     192   136010565 :                         fd_ed25519_point_t const * a ) {
     193   136010565 :   fd_f25519_t r1[1], r2[1], r3[1], r4[1];
     194   136010565 :   fd_f25519_t r5[1];
     195             : 
     196   136010565 :   fd_f25519_add_nr( r1, a->X, a->Y );
     197             : 
     198   136010565 :   fd_f25519_sqr4( r2, a->X,
     199   136010565 :                   r3, a->Y,
     200   136010565 :                   r4, a->Z,
     201   136010565 :                   r5, r1 );
     202             : 
     203             :   /* important: reduce mod p (these values are used in add/sub) */
     204   136010565 :   fd_f25519_add( r4, r4, r4 );
     205   136010565 :   fd_f25519_add( r->T, r2, r3 );
     206   136010565 :   fd_f25519_sub( r->Z, r2, r3 );
     207             : 
     208   136010565 :   fd_f25519_add_nr( r->Y, r4, r->Z );
     209   136010565 :   fd_f25519_sub_nr( r->X, r->T, r5 );
     210   136010565 :   return r;
     211   136010565 : }
     212             : 
     213             : FD_25519_INLINE fd_ed25519_point_t * FD_FN_NO_ASAN
     214             : fd_ed25519_point_dbln( fd_ed25519_point_t *       r,
     215             :                        fd_ed25519_point_t const * a,
     216      680984 :                        int                        n ) {
     217      680984 :   fd_ed25519_point_t t[1];
     218      680984 :   fd_ed25519_partial_dbl( t, a );
     219      680984 :   for( uchar i=1; i<n; i++ ) {
     220           0 :     fd_ed25519_point_add_final_mul_projective( r, t );
     221           0 :     fd_ed25519_partial_dbl( t, r );
     222           0 :   }
     223      680984 :   return fd_ed25519_point_add_final_mul( r, t );
     224      680984 : }
     225             : 
     226             : FD_PROTOTYPES_END

Generated by: LCOV version 1.14