1
0

ed25519-hash.h 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. #if defined(ED25519_REFHASH)
  2. /* reference/slow SHA-512. really, do not use this */
  3. #define HASH_BLOCK_SIZE 128
  4. #define HASH_DIGEST_SIZE 64
  5. typedef struct sha512_state_t {
  6. uint64_t H[8];
  7. uint64_t T[2];
  8. uint32_t leftover;
  9. uint8_t buffer[HASH_BLOCK_SIZE];
  10. } sha512_state;
  11. typedef sha512_state ed25519_hash_context;
  12. static const uint64_t sha512_constants[80] = {
  13. 0x428a2f98d728ae22ull, 0x7137449123ef65cdull, 0xb5c0fbcfec4d3b2full, 0xe9b5dba58189dbbcull,
  14. 0x3956c25bf348b538ull, 0x59f111f1b605d019ull, 0x923f82a4af194f9bull, 0xab1c5ed5da6d8118ull,
  15. 0xd807aa98a3030242ull, 0x12835b0145706fbeull, 0x243185be4ee4b28cull, 0x550c7dc3d5ffb4e2ull,
  16. 0x72be5d74f27b896full, 0x80deb1fe3b1696b1ull, 0x9bdc06a725c71235ull, 0xc19bf174cf692694ull,
  17. 0xe49b69c19ef14ad2ull, 0xefbe4786384f25e3ull, 0x0fc19dc68b8cd5b5ull, 0x240ca1cc77ac9c65ull,
  18. 0x2de92c6f592b0275ull, 0x4a7484aa6ea6e483ull, 0x5cb0a9dcbd41fbd4ull, 0x76f988da831153b5ull,
  19. 0x983e5152ee66dfabull, 0xa831c66d2db43210ull, 0xb00327c898fb213full, 0xbf597fc7beef0ee4ull,
  20. 0xc6e00bf33da88fc2ull, 0xd5a79147930aa725ull, 0x06ca6351e003826full, 0x142929670a0e6e70ull,
  21. 0x27b70a8546d22ffcull, 0x2e1b21385c26c926ull, 0x4d2c6dfc5ac42aedull, 0x53380d139d95b3dfull,
  22. 0x650a73548baf63deull, 0x766a0abb3c77b2a8ull, 0x81c2c92e47edaee6ull, 0x92722c851482353bull,
  23. 0xa2bfe8a14cf10364ull, 0xa81a664bbc423001ull, 0xc24b8b70d0f89791ull, 0xc76c51a30654be30ull,
  24. 0xd192e819d6ef5218ull, 0xd69906245565a910ull, 0xf40e35855771202aull, 0x106aa07032bbd1b8ull,
  25. 0x19a4c116b8d2d0c8ull, 0x1e376c085141ab53ull, 0x2748774cdf8eeb99ull, 0x34b0bcb5e19b48a8ull,
  26. 0x391c0cb3c5c95a63ull, 0x4ed8aa4ae3418acbull, 0x5b9cca4f7763e373ull, 0x682e6ff3d6b2b8a3ull,
  27. 0x748f82ee5defb2fcull, 0x78a5636f43172f60ull, 0x84c87814a1f0ab72ull, 0x8cc702081a6439ecull,
  28. 0x90befffa23631e28ull, 0xa4506cebde82bde9ull, 0xbef9a3f7b2c67915ull, 0xc67178f2e372532bull,
  29. 0xca273eceea26619cull, 0xd186b8c721c0c207ull, 0xeada7dd6cde0eb1eull, 0xf57d4f7fee6ed178ull,
  30. 0x06f067aa72176fbaull, 0x0a637dc5a2c898a6ull, 0x113f9804bef90daeull, 0x1b710b35131c471bull,
  31. 0x28db77f523047d84ull, 0x32caab7b40c72493ull, 0x3c9ebe0a15c9bebcull, 0x431d67c49c100d4cull,
  32. 0x4cc5d4becb3e42b6ull, 0x597f299cfc657e2aull, 0x5fcb6fab3ad6faecull, 0x6c44198c4a475817ull
  33. };
  34. static uint64_t
  35. sha512_ROTR64(uint64_t x, int k) {
  36. return (x >> k) | (x << (64 - k));
  37. }
  38. static uint64_t
  39. sha512_LOAD64_BE(const uint8_t *p) {
  40. return
  41. ((uint64_t)p[0] << 56) |
  42. ((uint64_t)p[1] << 48) |
  43. ((uint64_t)p[2] << 40) |
  44. ((uint64_t)p[3] << 32) |
  45. ((uint64_t)p[4] << 24) |
  46. ((uint64_t)p[5] << 16) |
  47. ((uint64_t)p[6] << 8) |
  48. ((uint64_t)p[7] );
  49. }
  50. static void
  51. sha512_STORE64_BE(uint8_t *p, uint64_t v) {
  52. p[0] = (uint8_t)(v >> 56);
  53. p[1] = (uint8_t)(v >> 48);
  54. p[2] = (uint8_t)(v >> 40);
  55. p[3] = (uint8_t)(v >> 32);
  56. p[4] = (uint8_t)(v >> 24);
  57. p[5] = (uint8_t)(v >> 16);
  58. p[6] = (uint8_t)(v >> 8);
  59. p[7] = (uint8_t)(v );
  60. }
  61. #define Ch(x,y,z) (z ^ (x & (y ^ z)))
  62. #define Maj(x,y,z) (((x | y) & z) | (x & y))
  63. #define S0(x) (sha512_ROTR64(x, 28) ^ sha512_ROTR64(x, 34) ^ sha512_ROTR64(x, 39))
  64. #define S1(x) (sha512_ROTR64(x, 14) ^ sha512_ROTR64(x, 18) ^ sha512_ROTR64(x, 41))
  65. #define G0(x) (sha512_ROTR64(x, 1) ^ sha512_ROTR64(x, 8) ^ (x >> 7))
  66. #define G1(x) (sha512_ROTR64(x, 19) ^ sha512_ROTR64(x, 61) ^ (x >> 6))
  67. #define W0(in,i) (sha512_LOAD64_BE(&in[i * 8]))
  68. #define W1(i) (G1(w[i - 2]) + w[i - 7] + G0(w[i - 15]) + w[i - 16])
  69. #define STEP(i) \
  70. t1 = S0(r[0]) + Maj(r[0], r[1], r[2]); \
  71. t0 = r[7] + S1(r[4]) + Ch(r[4], r[5], r[6]) + sha512_constants[i] + w[i]; \
  72. r[7] = r[6]; \
  73. r[6] = r[5]; \
  74. r[5] = r[4]; \
  75. r[4] = r[3] + t0; \
  76. r[3] = r[2]; \
  77. r[2] = r[1]; \
  78. r[1] = r[0]; \
  79. r[0] = t0 + t1;
  80. static void
  81. sha512_blocks(sha512_state *S, const uint8_t *in, size_t blocks) {
  82. uint64_t r[8], w[80], t0, t1;
  83. size_t i;
  84. for (i = 0; i < 8; i++) r[i] = S->H[i];
  85. while (blocks--) {
  86. for (i = 0; i < 16; i++) { w[i] = W0(in, i); }
  87. for (i = 16; i < 80; i++) { w[i] = W1(i); }
  88. for (i = 0; i < 80; i++) { STEP(i); }
  89. for (i = 0; i < 8; i++) { r[i] += S->H[i]; S->H[i] = r[i]; }
  90. S->T[0] += HASH_BLOCK_SIZE * 8;
  91. S->T[1] += (!S->T[0]) ? 1 : 0;
  92. in += HASH_BLOCK_SIZE;
  93. }
  94. }
  95. static void
  96. ed25519_hash_init(sha512_state *S) {
  97. S->H[0] = 0x6a09e667f3bcc908ull;
  98. S->H[1] = 0xbb67ae8584caa73bull;
  99. S->H[2] = 0x3c6ef372fe94f82bull;
  100. S->H[3] = 0xa54ff53a5f1d36f1ull;
  101. S->H[4] = 0x510e527fade682d1ull;
  102. S->H[5] = 0x9b05688c2b3e6c1full;
  103. S->H[6] = 0x1f83d9abfb41bd6bull;
  104. S->H[7] = 0x5be0cd19137e2179ull;
  105. S->T[0] = 0;
  106. S->T[1] = 0;
  107. S->leftover = 0;
  108. }
  109. static void
  110. ed25519_hash_update(sha512_state *S, const uint8_t *in, size_t inlen) {
  111. size_t blocks, want;
  112. /* handle the previous data */
  113. if (S->leftover) {
  114. want = (HASH_BLOCK_SIZE - S->leftover);
  115. want = (want < inlen) ? want : inlen;
  116. memcpy(S->buffer + S->leftover, in, want);
  117. S->leftover += (uint32_t)want;
  118. if (S->leftover < HASH_BLOCK_SIZE)
  119. return;
  120. in += want;
  121. inlen -= want;
  122. sha512_blocks(S, S->buffer, 1);
  123. }
  124. /* handle the current data */
  125. blocks = (inlen & ~(HASH_BLOCK_SIZE - 1));
  126. S->leftover = (uint32_t)(inlen - blocks);
  127. if (blocks) {
  128. sha512_blocks(S, in, blocks / HASH_BLOCK_SIZE);
  129. in += blocks;
  130. }
  131. /* handle leftover data */
  132. if (S->leftover)
  133. memcpy(S->buffer, in, S->leftover);
  134. }
  135. static void
  136. ed25519_hash_final(sha512_state *S, uint8_t *hash) {
  137. uint64_t t0 = S->T[0] + (S->leftover * 8), t1 = S->T[1];
  138. S->buffer[S->leftover] = 0x80;
  139. if (S->leftover <= 111) {
  140. memset(S->buffer + S->leftover + 1, 0, 111 - S->leftover);
  141. } else {
  142. memset(S->buffer + S->leftover + 1, 0, 127 - S->leftover);
  143. sha512_blocks(S, S->buffer, 1);
  144. memset(S->buffer, 0, 112);
  145. }
  146. sha512_STORE64_BE(S->buffer + 112, t1);
  147. sha512_STORE64_BE(S->buffer + 120, t0);
  148. sha512_blocks(S, S->buffer, 1);
  149. sha512_STORE64_BE(&hash[ 0], S->H[0]);
  150. sha512_STORE64_BE(&hash[ 8], S->H[1]);
  151. sha512_STORE64_BE(&hash[16], S->H[2]);
  152. sha512_STORE64_BE(&hash[24], S->H[3]);
  153. sha512_STORE64_BE(&hash[32], S->H[4]);
  154. sha512_STORE64_BE(&hash[40], S->H[5]);
  155. sha512_STORE64_BE(&hash[48], S->H[6]);
  156. sha512_STORE64_BE(&hash[56], S->H[7]);
  157. }
  158. static void
  159. ed25519_hash(uint8_t *hash, const uint8_t *in, size_t inlen) {
  160. ed25519_hash_context ctx;
  161. ed25519_hash_init(&ctx);
  162. ed25519_hash_update(&ctx, in, inlen);
  163. ed25519_hash_final(&ctx, hash);
  164. }
  165. #elif defined(ED25519_CUSTOMHASH)
  166. #include "ed25519-hash-custom.h"
  167. #else
  168. #include <openssl/sha.h>
  169. typedef SHA512_CTX ed25519_hash_context;
  170. static void
  171. ed25519_hash_init(ed25519_hash_context *ctx) {
  172. SHA512_Init(ctx);
  173. }
  174. static void
  175. ed25519_hash_update(ed25519_hash_context *ctx, const uint8_t *in, size_t inlen) {
  176. SHA512_Update(ctx, in, inlen);
  177. }
  178. static void
  179. ed25519_hash_final(ed25519_hash_context *ctx, uint8_t *hash) {
  180. SHA512_Final(hash, ctx);
  181. }
  182. static void
  183. ed25519_hash(uint8_t *hash, const uint8_t *in, size_t inlen) {
  184. SHA512(in, inlen, hash);
  185. }
  186. #endif