You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

190 lines
5.3 KiB

  1. /** @brief Elligator high-level functions. */
  2. #include "word.h"
  3. #include "field.h"
  4. #include <decaf.h>
  5. /* Template stuff */
  6. #define API_NS(_id) $(c_ns)_##_id
  7. #define point_t API_NS(point_t)
  8. #define IMAGINE_TWIST $(imagine_twist)
  9. #define COFACTOR $(cofactor)
  10. static const int EDWARDS_D = $(d);
  11. #define RISTRETTO_FACTOR $(C_NS)_RISTRETTO_FACTOR
  12. extern const gf RISTRETTO_FACTOR;
  13. /* End of template stuff */
  14. extern mask_t API_NS(deisogenize) (
  15. gf_s *__restrict__ s,
  16. gf_s *__restrict__ inv_el_sum,
  17. gf_s *__restrict__ inv_el_m1,
  18. const point_t p,
  19. mask_t toggle_hibit_s,
  20. mask_t toggle_altx,
  21. mask_t toggle_rotation
  22. );
  23. void API_NS(point_from_hash_nonuniform) (
  24. point_t p,
  25. const unsigned char ser[SER_BYTES]
  26. ) {
  27. gf r0,r,a,b,c,N,e;
  28. const uint8_t mask = (uint8_t)(0xFE<<($((gf_bits-1)%8)));
  29. ignore_result(gf_deserialize(r0,ser,mask));
  30. gf_strong_reduce(r0);
  31. gf_sqr(a,r0);
  32. gf_mul_qnr(r,a);
  33. /* Compute D@c := (dr+a-d)(dr-ar-d) with a=1 */
  34. gf_sub(a,r,ONE);
  35. gf_mulw(b,a,EDWARDS_D); /* dr-d */
  36. gf_add(a,b,ONE);
  37. gf_sub(b,b,r);
  38. gf_mul(c,a,b);
  39. /* compute N := (r+1)(a-2d) */
  40. gf_add(a,r,ONE);
  41. gf_mulw(N,a,1-2*EDWARDS_D);
  42. /* e = +-sqrt(1/ND) or +-r0 * sqrt(qnr/ND) */
  43. gf_mul(a,c,N);
  44. mask_t square = gf_isr(b,a);
  45. gf_cond_sel(c,r0,ONE,square); /* r? = square ? 1 : r0 */
  46. gf_mul(e,b,c);
  47. /* s@a = +-|N.e| */
  48. gf_mul(a,N,e);
  49. gf_cond_neg(a,gf_lobit(a) ^ ~square);
  50. /* t@b = -+ cN(r-1)((a-2d)e)^2 - 1 */
  51. gf_mulw(c,e,1-2*EDWARDS_D); /* (a-2d)e */
  52. gf_sqr(b,c);
  53. gf_sub(e,r,ONE);
  54. gf_mul(c,b,e);
  55. gf_mul(b,c,N);
  56. gf_cond_neg(b,square);
  57. gf_sub(b,b,ONE);
  58. /* isogenize */
  59. #if IMAGINE_TWIST
  60. gf_mul(c,a,SQRT_MINUS_ONE);
  61. gf_copy(a,c);
  62. #endif
  63. gf_sqr(c,a); /* s^2 */
  64. gf_add(a,a,a); /* 2s */
  65. gf_add(e,c,ONE);
  66. gf_mul(p->t,a,e); /* 2s(1+s^2) */
  67. gf_mul(p->x,a,b); /* 2st */
  68. gf_sub(a,ONE,c);
  69. gf_mul(p->y,e,a); /* (1+s^2)(1-s^2) */
  70. gf_mul(p->z,a,b); /* (1-s^2)t */
  71. assert(API_NS(point_valid)(p));
  72. }
  73. void API_NS(point_from_hash_uniform) (
  74. point_t pt,
  75. const unsigned char hashed_data[2*SER_BYTES]
  76. ) {
  77. point_t pt2;
  78. API_NS(point_from_hash_nonuniform)(pt,hashed_data);
  79. API_NS(point_from_hash_nonuniform)(pt2,&hashed_data[SER_BYTES]);
  80. API_NS(point_add)(pt,pt,pt2);
  81. }
  82. /* Elligator_onto:
  83. * Make elligator-inverse onto at the cost of roughly halving the success probability.
  84. * Currently no effect for curves with field size 1 bit mod 8 (where the top bit
  85. * is chopped off). FUTURE MAGIC: automatic at least for brainpool-style curves; support
  86. * log p == 1 mod 8 brainpool curves maybe?
  87. */
  88. #define MAX(A,B) (((A)>(B)) ? (A) : (B))
  89. decaf_error_t
  90. API_NS(invert_elligator_nonuniform) (
  91. unsigned char recovered_hash[SER_BYTES],
  92. const point_t p,
  93. uint32_t hint_
  94. ) {
  95. mask_t hint = hint_;
  96. mask_t sgn_s = ~((1 - (hint & 1))*DECAF_MASK_ALL_SET), /* expand hint bit 0 to the whole mask without branching */
  97. sgn_altx = ~((1 - (hint>>1 & 1))*DECAF_MASK_ALL_SET),
  98. sgn_r0 = ~((1 - (hint>>2 & 1))*DECAF_MASK_ALL_SET),
  99. /* FUTURE MAGIC: eventually if there's a curve which needs sgn_ed_T but not sgn_r0,
  100. * change this mask extraction.
  101. */
  102. sgn_ed_T = ~((1 - (hint>>3 & 1))*DECAF_MASK_ALL_SET);
  103. gf a,b,c;
  104. API_NS(deisogenize)(a,b,c,p,sgn_s,sgn_altx,sgn_ed_T);
  105. mask_t is_identity = gf_eq(p->t,ZERO);
  106. #if COFACTOR==4
  107. gf_cond_sel(b,b,ONE,is_identity & sgn_altx);
  108. gf_cond_sel(c,c,ONE,is_identity & sgn_s &~ sgn_altx);
  109. #elif IMAGINE_TWIST
  110. /* Terrible, terrible special casing due to lots of 0/0 is deisogenize
  111. * Basically we need to generate -D and +- i*RISTRETTO_FACTOR
  112. */
  113. gf_mul_i(a,RISTRETTO_FACTOR);
  114. gf_cond_sel(b,b,ONE,is_identity);
  115. gf_cond_neg(a,sgn_altx);
  116. gf_cond_sel(c,c,a,is_identity & sgn_ed_T);
  117. gf_cond_sel(c,c,ZERO,is_identity & ~sgn_ed_T);
  118. gf_mulw(a,ONE,-EDWARDS_D);
  119. gf_cond_sel(c,c,a,is_identity & ~sgn_ed_T &~ sgn_altx);
  120. #else
  121. #error "Different special-casing goes here!"
  122. #endif
  123. #if IMAGINE_TWIST
  124. gf_mulw(a,b,-EDWARDS_D);
  125. #else
  126. gf_mulw(a,b,EDWARDS_D-1);
  127. #endif
  128. gf_add(b,a,b);
  129. gf_sub(a,a,c);
  130. gf_add(b,b,c);
  131. gf_cond_swap(a,b,sgn_s);
  132. gf_mul_qnr(c,b);
  133. gf_mul(b,c,a);
  134. mask_t succ = gf_isr(c,b);
  135. succ |= gf_eq(b,ZERO);
  136. gf_mul(b,c,a);
  137. #if $(gf_bits) == 8*SER_BYTES + 1 /* p521. */
  138. #error "this won't work because it needs to adjust high bit, not low bit"
  139. sgn_r0 = 0;
  140. #endif
  141. gf_cond_neg(b, sgn_r0^gf_lobit(b));
  142. /* Eliminate duplicate values for identity ... */
  143. succ &= ~(gf_eq(b,ZERO) & (sgn_r0 | sgn_s));
  144. // #if COFACTOR == 8
  145. // succ &= ~(is_identity & sgn_ed_T); /* NB: there are no preimages of rotated identity. */
  146. // #endif
  147. gf_serialize(recovered_hash,b);
  148. #if $(gf_bits%8)
  149. #if COFACTOR==8
  150. recovered_hash[SER_BYTES-1] ^= (hint>>4)<<$(gf_bits%8);
  151. #else
  152. recovered_hash[SER_BYTES-1] ^= (hint>>3)<<$(gf_bits%8);
  153. #endif
  154. #endif
  155. return decaf_succeed_if(mask_to_bool(succ));
  156. }
  157. decaf_error_t
  158. API_NS(invert_elligator_uniform) (
  159. unsigned char partial_hash[2*SER_BYTES],
  160. const point_t p,
  161. uint32_t hint
  162. ) {
  163. point_t pt2;
  164. API_NS(point_from_hash_nonuniform)(pt2,&partial_hash[SER_BYTES]);
  165. API_NS(point_sub)(pt2,p,pt2);
  166. return API_NS(invert_elligator_nonuniform)(partial_hash,pt2,hint);
  167. }