siphash.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930
  1. /* siphash.c
  2. *
  3. * Copyright (C) 2006-2022 wolfSSL Inc.
  4. *
  5. * This file is part of wolfSSL.
  6. *
  7. * wolfSSL is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * wolfSSL is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335, USA
  20. */
  21. #ifdef HAVE_CONFIG_H
  22. #include <config.h>
  23. #endif
  24. #include <wolfssl/wolfcrypt/settings.h>
  25. #include <wolfssl/wolfcrypt/types.h>
  26. #include <wolfssl/wolfcrypt/siphash.h>
  27. #include <wolfssl/wolfcrypt/error-crypt.h>
  28. #ifdef NO_INLINE
  29. #include <wolfssl/wolfcrypt/misc.h>
  30. #else
  31. #define WOLFSSL_MISC_INCLUDED
  32. #include <wolfcrypt/src/misc.c>
  33. #endif
  34. /* DESCRIPTION
  35. *
  36. * SipHash is a PseudoRandom Function (PRF) that can be used with small
  37. * messages (less than 256 bytes).
  38. * SipHash can be used for Message Authentication Codes (MACs) and as such must
  39. * be passed a secret key.
  40. * https://eprint.iacr.org/2012/351.pdf
  41. *
  42. * SipHash is commonly used in hash tables.
  43. * Do not use this as a hash not as a general purpose MAC.
  44. *
  45. * WOLFSSL_SIPHASH_CROUNDS and WOLFSSL_SIPHASH_DROUNDS can be defined at build
  46. * time to change the algorithm.
  47. * Default is SipHash-2-4:
  48. * WOLFSSL_SIPHASH_CROUNDS = 2
  49. * WOLFSSL_SIPHASH_DROUNDS = 4
  50. *
  51. * Inline assembly implementations of wc_SipHash() written for:
  52. * - GCC for Intel x86_64
  53. * - GCC for Aarch64.
  54. */
  55. #ifdef WOLFSSL_SIPHASH
  56. #ifdef LITTLE_ENDIAN_ORDER
  57. /**
  58. * Decode little-endian byte array to 64-bit number.
  59. *
  60. * @param [in] a Little-endian byte array.
  61. * @return 64-bit number.
  62. */
  63. #define GET_U64(a) (*(word64*)(a))
  64. /**
  65. * Decode little-endian byte array to 32-bit number.
  66. *
  67. * @param [in] a Little-endian byte array.
  68. * @return 32-bit number.
  69. */
  70. #define GET_U32(a) (*(word32*)(a))
  71. /**
  72. * Decode little-endian byte array to 16-bit number.
  73. *
  74. * @param [in] a Little-endian byte array.
  75. * @return 16-bit number.
  76. */
  77. #define GET_U16(a) (*(word16*)(a))
  78. /**
  79. * Encode 64-bit number to a little-endian byte array.
  80. *
  81. * @param [out] a Byte array to write into.
  82. * @param [in] n Number to encode.
  83. */
  84. #define SET_U64(a, n) ((*(word64*)(a)) = n)
  85. #else
  86. /**
  87. * Decode little-endian byte array to 64-bit number.
  88. *
  89. * @param [in] a Little-endian byte array.
  90. * @return 64-bit number.
  91. */
  92. #define GET_U64(a) (((word64)((a)[7]) << 56) | \
  93. ((word64)((a)[6]) << 48) | \
  94. ((word64)((a)[5]) << 40) | \
  95. ((word64)((a)[4]) << 32) | \
  96. ((word64)((a)[3]) << 24) | \
  97. ((word64)((a)[2]) << 16) | \
  98. ((word64)((a)[1]) << 8) | \
  99. ((word64)((a)[0]) ))
  100. /**
  101. * Decode little-endian byte array to 32-bit number.
  102. *
  103. * @param [in] a Little-endian byte array.
  104. * @return 32-bit number.
  105. */
  106. #define GET_U32(a) (((word64)((a)[3]) << 24) | \
  107. ((word32)((a)[2]) << 16) | \
  108. ((word32)((a)[1]) << 8) | \
  109. ((word32)((a)[0]) ))
  110. /**
  111. * Decode little-endian byte array to 16-bit number.
  112. *
  113. * @param [in] a Little-endian byte array.
  114. * @return 16-bit number.
  115. */
  116. #define GET_U16(a) (((word16)((a)[1]) << 8) | \
  117. ((word16)((a)[0]) ))
  118. /**
  119. * Encode 64-bit number to a little-endian byte array.
  120. *
  121. * @param [out] a Byte array to write into.
  122. * @param [in] n Number to encode.
  123. */
  124. #define SET_U64(a, n) (a)[0] = (byte)((n) ); \
  125. (a)[1] = (byte)((n) >> 8); \
  126. (a)[2] = (byte)((n) >> 16); \
  127. (a)[3] = (byte)((n) >> 24); \
  128. (a)[4] = (byte)((n) >> 32); \
  129. (a)[5] = (byte)((n) >> 40); \
  130. (a)[6] = (byte)((n) >> 48); \
  131. (a)[7] = (byte)((n) >> 56)
  132. #endif
  133. /**
  134. * Initialize SipHash operation with a key.
  135. *
  136. * @param [out] sipHash SipHash object.
  137. * @param [in] key 16 byte array - little endian.
  138. * @return BAD_FUNC_ARG when sipHash or key is NULL.
  139. * @return BAD_FUNC_ARG when outSz is neither 8 nor 16.
  140. * @return 0 on success.
  141. */
  142. int wc_InitSipHash(SipHash* sipHash, const unsigned char* key,
  143. unsigned char outSz)
  144. {
  145. int ret = 0;
  146. /* Validate parameters. */
  147. if ((sipHash == NULL) || (key == NULL) ||
  148. ((outSz != SIPHASH_MAC_SIZE_8) && (outSz != SIPHASH_MAC_SIZE_16))) {
  149. ret = BAD_FUNC_ARG;
  150. }
  151. if (ret == 0) {
  152. word64 k0 = GET_U64(key + 0);
  153. word64 k1 = GET_U64(key + 8);
  154. /* Initialize state with key. */
  155. sipHash->v[0] = 0x736f6d6570736575ULL;
  156. if (outSz == SIPHASH_MAC_SIZE_8) {
  157. sipHash->v[1] = 0x646f72616e646f6dULL;
  158. }
  159. else {
  160. sipHash->v[1] = 0x646f72616e646f83ULL;
  161. }
  162. sipHash->v[2] = 0x6c7967656e657261ULL;
  163. sipHash->v[3] = 0x7465646279746573ULL;
  164. sipHash->v[0] ^= k0;
  165. sipHash->v[1] ^= k1;
  166. sipHash->v[2] ^= k0;
  167. sipHash->v[3] ^= k1;
  168. /* No cached message bytes. */
  169. sipHash->cacheCnt = 0;
  170. /* No message bytes compressed yet. */
  171. sipHash->inCnt = 0;
  172. /* Keep the output size to check against final call. */
  173. sipHash->outSz = outSz;
  174. }
  175. return ret;
  176. }
  177. /**
  178. * One round of SipHash.
  179. *
  180. * @param [in, out] sipHash SipHash object.
  181. */
  182. static WC_INLINE void SipRound(SipHash *sipHash)
  183. {
  184. word64* v = sipHash->v;
  185. v[0] += v[1];
  186. v[2] += v[3];
  187. v[1] = rotlFixed64(v[1], 13);
  188. v[3] = rotlFixed64(v[3], 16);
  189. v[1] ^= v[0];
  190. v[3] ^= v[2];
  191. v[0] = rotlFixed64(v[0], 32);
  192. v[2] += v[1];
  193. v[0] += v[3];
  194. v[1] = rotlFixed64(v[1], 17);
  195. v[3] = rotlFixed64(v[3], 21);
  196. v[1] ^= v[2];
  197. v[3] ^= v[0];
  198. v[2] = rotlFixed64(v[2], 32);
  199. }
  200. /**
  201. * One step of the compression operation.
  202. *
  203. * @param [in, out] sipHash SipHash object.
  204. * @param [in] m Message to compress.
  205. */
  206. static WC_INLINE void SipHashCompress(SipHash* sipHash, const byte* m)
  207. {
  208. int i;
  209. sipHash->v[3] ^= GET_U64(m);
  210. for (i = 0; i < WOLFSSL_SIPHASH_CROUNDS; i++) {
  211. SipRound(sipHash);
  212. }
  213. sipHash->v[0] ^= GET_U64(m);
  214. }
  215. /**
  216. * Update the SipHash operation with more data.
  217. *
  218. * @param [in, out] sipHash SipHash object.
  219. * @param [in] in Input message.
  220. * @param [in] inSz Size of input message.
  221. * @return BAD_FUNC_ARG when sipHash is NULL.
  222. * @return BAD_FUNC_ARG when in is NULL and inSz is not zero.
  223. * @return 0 on success.
  224. */
  225. int wc_SipHashUpdate(SipHash* sipHash, const unsigned char* in, word32 inSz)
  226. {
  227. int ret = 0;
  228. /* Validate parameters. */
  229. if ((sipHash == NULL) || ((in == NULL) && (inSz != 0))) {
  230. ret = BAD_FUNC_ARG;
  231. }
  232. /* Process any message bytes. */
  233. if ((ret == 0) && (inSz > 0)) {
  234. /* Add to cache if already started. */
  235. if (sipHash->cacheCnt > 0) {
  236. byte len = SIPHASH_BLOCK_SIZE - sipHash->cacheCnt;
  237. if (len > inSz) {
  238. len = inSz;
  239. }
  240. XMEMCPY(sipHash->cache + sipHash->cacheCnt, in, len);
  241. in += len;
  242. inSz -= len;
  243. sipHash->cacheCnt += len;
  244. if (sipHash->cacheCnt == SIPHASH_BLOCK_SIZE) {
  245. /* Compress the block from the cache. */
  246. SipHashCompress(sipHash, sipHash->cache);
  247. sipHash->cacheCnt = 0;
  248. }
  249. }
  250. /* Process more blocks from message. */
  251. while (inSz >= SIPHASH_BLOCK_SIZE) {
  252. /* Compress the next block from the message data. */
  253. SipHashCompress(sipHash, in);
  254. in += SIPHASH_BLOCK_SIZE;
  255. inSz -= SIPHASH_BLOCK_SIZE;
  256. sipHash->inCnt += SIPHASH_BLOCK_SIZE;
  257. }
  258. if (inSz > 0) {
  259. /* Cache remaining message bytes less than a block. */
  260. XMEMCPY(sipHash->cache, in, inSz);
  261. sipHash->cacheCnt = inSz;
  262. }
  263. }
  264. return ret;
  265. }
  266. /**
  267. * Calculate 8-bytes of output.
  268. *
  269. * @param [in, out] sipHash SipHash object.
  270. * @param [out] out Buffer to place 8-bytes of MAC into.
  271. */
  272. static WC_INLINE void SipHashOut(SipHash* sipHash, byte* out)
  273. {
  274. word64 n;
  275. int i;
  276. for (i = 0; i < WOLFSSL_SIPHASH_DROUNDS; i++) {
  277. SipRound(sipHash);
  278. }
  279. n = sipHash->v[0] ^ sipHash->v[1] ^ sipHash->v[2] ^ sipHash->v[3];
  280. SET_U64(out, n);
  281. }
  282. /**
  283. * Finalize SipHash operation.
  284. *
  285. * @param [in, out] sipHash SipHash object.
  286. * @param [out] out Buffer to place MAC into.
  287. * @param [in] outSz Size of ouput MAC. 8 or 16 only.
  288. * @return BAD_FUNC_ARG when sipHash or out is NULL.
  289. * @return BAD_FUNC_ARG when outSz is not the same as initialized value.
  290. * @return 0 on success.
  291. */
  292. int wc_SipHashFinal(SipHash* sipHash, unsigned char* out, unsigned char outSz)
  293. {
  294. int ret = 0;
  295. /* Validate parameters. */
  296. if ((sipHash == NULL) || (out == NULL) || (outSz != sipHash->outSz)) {
  297. ret = BAD_FUNC_ARG;
  298. }
  299. if (ret == 0) {
  300. /* Put int remaining cached message bytes. */
  301. XMEMSET(sipHash->cache + sipHash->cacheCnt, 0, 7 - sipHash->cacheCnt);
  302. sipHash->cache[7] = (byte)(sipHash->inCnt + sipHash->cacheCnt);
  303. SipHashCompress(sipHash, sipHash->cache);
  304. sipHash->cacheCnt = 0;
  305. /* Output either 8 or 16 bytes. */
  306. if (outSz == SIPHASH_MAC_SIZE_8) {
  307. sipHash->v[2] ^= (word64)0xff;
  308. SipHashOut(sipHash, out);
  309. }
  310. else {
  311. sipHash->v[2] ^= (word64)0xee;
  312. SipHashOut(sipHash, out);
  313. sipHash->v[1] ^= (word64)0xdd;
  314. SipHashOut(sipHash, out + 8);
  315. }
  316. }
  317. return ret;
  318. }
  319. #if !defined(WOLFSSL_NO_ASM) && defined(__GNUC__) && defined(__x86_64__) && \
  320. (WOLFSSL_SIPHASH_CROUNDS == 1 || WOLFSSL_SIPHASH_CROUNDS == 2) && \
  321. (WOLFSSL_SIPHASH_DROUNDS == 2 || WOLFSSL_SIPHASH_DROUNDS == 4)
  322. #define SIPHASH_ROUND(v0, v1, v2, v3) \
  323. "addq " #v1 ", " #v0 "\n\t" \
  324. "addq " #v3 ", " #v2 "\n\t" \
  325. "rolq $13, " #v1 "\n\t" \
  326. "rolq $16, " #v3 "\n\t" \
  327. "xorq " #v0 ", " #v1 "\n\t" \
  328. "xorq " #v2 ", " #v3 "\n\t" \
  329. "rolq $32, " #v0 "\n\t" \
  330. "addq " #v1 ", " #v2 "\n\t" \
  331. "addq " #v3 ", " #v0 "\n\t" \
  332. "rolq $17, " #v1 "\n\t" \
  333. "rolq $21, " #v3 "\n\t" \
  334. "xorq " #v2 ", " #v1 "\n\t" \
  335. "xorq " #v0 ", " #v3 "\n\t" \
  336. "rolq $32, " #v2 "\n\t"
  337. #define SIPHASH_LAST_ROUND(v0, v1, v2, v3) \
  338. "addq " #v1 ", " #v0 "\n\t" \
  339. "addq " #v3 ", " #v2 "\n\t" \
  340. "rolq $13, " #v1 "\n\t" \
  341. "rolq $16, " #v3 "\n\t" \
  342. "xorq " #v0 ", " #v1 "\n\t" \
  343. "xorq " #v2 ", " #v3 "\n\t" \
  344. "addq " #v1 ", " #v2 "\n\t" \
  345. "rolq $17, " #v1 "\n\t" \
  346. "rolq $21, " #v3 "\n\t" \
  347. "xorq " #v2 ", " #v1 "\n\t" \
  348. "rolq $32, " #v2 "\n\t"
  349. /**
  350. * Perform SipHash operation on input with key.
  351. *
  352. * @param [in] key 16 byte array - little endian.
  353. * @param [in] in Input message.
  354. * @param [in] inSz Size of input message.
  355. * @param [out] out Buffer to place MAC into.
  356. * @param [in] outSz Size of ouput MAC. 8 or 16 only.
  357. * @return BAD_FUNC_ARG when key or out is NULL.
  358. * @return BAD_FUNC_ARG when in is NULL and inSz is not zero.
  359. * @return BAD_FUNC_ARG when outSz is neither 8 nor 16.
  360. * @return 0 on success.
  361. */
  362. int wc_SipHash(const unsigned char* key, const unsigned char* in, word32 inSz,
  363. unsigned char* out, unsigned char outSz)
  364. {
  365. if ((key == NULL) || ((in == NULL) && (inSz != 0)) || (out == NULL) ||
  366. ((outSz != SIPHASH_MAC_SIZE_8) && (outSz != SIPHASH_MAC_SIZE_16))) {
  367. return BAD_FUNC_ARG;
  368. }
  369. /* v0=%r8, v1=%r9, v2=%r10, v3=%r11 */
  370. __asm__ __volatile__ (
  371. "movq (%[key]), %%r12\n\t"
  372. "movq 8(%[key]), %%r13\n\t"
  373. "movabsq $0x736f6d6570736575, %%r8\n\t"
  374. "movabsq $0x646f72616e646f6d, %%r9\n\t"
  375. "movabsq $0x6c7967656e657261, %%r10\n\t"
  376. "movabsq $0x7465646279746573, %%r11\n\t"
  377. "xorq %%r12, %%r8\n\t"
  378. "xorq %%r13, %%r9\n\t"
  379. "xorq %%r12, %%r10\n\t"
  380. "xorq %%r13, %%r11\n\t"
  381. "cmp $8, %[outSz]\n\t"
  382. "mov %[inSz], %%r13d\n\t"
  383. "je L_siphash_8_top\n\t"
  384. "xorq $0xee, %%r9\n\t"
  385. "L_siphash_8_top:\n\t"
  386. "sub $8, %[inSz]\n\t"
  387. "jb L_siphash_done_input_8\n\t"
  388. "L_siphash_input:\n\t"
  389. "movq (%[in]), %%r12\n\t"
  390. "addq $8, %[in]\n\t"
  391. "xorq %%r12, %%r11\n\t"
  392. #if WOLFSSL_SIPHASH_CROUNDS == 1
  393. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  394. #elif WOLFSSL_SIPHASH_CROUNDS == 2
  395. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  396. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  397. #endif
  398. "xorq %%r12, %%r8\n\t"
  399. "sub $8, %[inSz]\n\t"
  400. "jge L_siphash_input\n\t"
  401. "L_siphash_done_input_8:\n\t"
  402. "add $8, %[inSz]\n\t"
  403. "shlq $56, %%r13\n\t"
  404. "cmp $0, %[inSz]\n\t"
  405. "je L_siphash_last_done\n\t"
  406. "cmp $4, %[inSz]\n\t"
  407. "jl L_siphash_last_lt4\n\t"
  408. "cmp $7, %[inSz]\n\t"
  409. "jl L_siphash_n7\n\t"
  410. "movzbq 6(%[in]), %%r12\n\t"
  411. "shlq $48, %%r12\n\t"
  412. "orq %%r12, %%r13\n\t"
  413. "L_siphash_n7:\n\t"
  414. "cmp $6, %[inSz]\n\t"
  415. "jl L_siphash_n6\n\t"
  416. "movzbq 5(%[in]), %%r12\n\t"
  417. "shlq $40, %%r12\n\t"
  418. "orq %%r12, %%r13\n\t"
  419. "L_siphash_n6:\n\t"
  420. "cmp $5, %[inSz]\n\t"
  421. "jl L_siphash_n5\n\t"
  422. "movzbq 4(%[in]), %%r12\n\t"
  423. "shlq $32, %%r12\n\t"
  424. "orq %%r12, %%r13\n\t"
  425. "L_siphash_n5:\n\t"
  426. "mov (%[in]), %%r12d\n\t"
  427. "orq %%r12, %%r13\n\t"
  428. "jmp L_siphash_last_done\n\t"
  429. "L_siphash_last_lt4:\n\t"
  430. "cmp $1, %[inSz]\n\t"
  431. "je L_siphash_last_1\n\t"
  432. "cmp $3, %[inSz]\n\t"
  433. "jl L_siphash_n3\n\t"
  434. "movzbq 2(%[in]), %%r12\n\t"
  435. "shlq $16, %%r12\n\t"
  436. "orq %%r12, %%r13\n\t"
  437. "L_siphash_n3:\n\t"
  438. "movw (%[in]), %%r12w\n\t"
  439. "or %%r12w, %%r13w\n\t"
  440. "jmp L_siphash_last_done\n\t"
  441. "L_siphash_last_1:\n\t"
  442. "movb (%[in]), %%r12b\n\t"
  443. "or %%r12b, %%r13b\n\t"
  444. "L_siphash_last_done:\n\t"
  445. "xorq %%r13, %%r11\n\t"
  446. #if WOLFSSL_SIPHASH_CROUNDS == 1
  447. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  448. #elif WOLFSSL_SIPHASH_CROUNDS == 2
  449. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  450. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  451. #endif
  452. "xorq %%r13, %%r8\n\t"
  453. "cmp $8, %[outSz]\n\t"
  454. "je L_siphash_8_end\n\t"
  455. "xor $0xee, %%r10b\n\t"
  456. #if WOLFSSL_SIPHASH_DROUNDS == 2
  457. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  458. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  459. #elif WOLFSSL_SIPHASH_DROUNDS == 4
  460. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  461. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  462. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  463. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  464. #endif
  465. "movq %%r8, %%r12\n\t"
  466. "xorq %%r9, %%r12\n\t"
  467. "xorq %%r10, %%r12\n\t"
  468. "xorq %%r11, %%r12\n\t"
  469. "movq %%r12, (%[out])\n\t"
  470. "xor $0xdd, %%r9b\n\t"
  471. #if WOLFSSL_SIPHASH_DROUNDS == 2
  472. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  473. SIPHASH_LAST_ROUND(%%r8, %%r9, %%r10, %%r11)
  474. #elif WOLFSSL_SIPHASH_DROUNDS == 4
  475. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  476. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  477. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  478. SIPHASH_LAST_ROUND(%%r8, %%r9, %%r10, %%r11)
  479. #endif
  480. "xorq %%r11, %%r9\n\t"
  481. "xorq %%r10, %%r9\n\t"
  482. "movq %%r9, 8(%[out])\n\t"
  483. "jmp L_siphash_done\n\t"
  484. "L_siphash_8_end:\n\t"
  485. "xor $0xff, %%r10b\n\t"
  486. #if WOLFSSL_SIPHASH_DROUNDS == 2
  487. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  488. SIPHASH_LAST_ROUND(%%r8, %%r9, %%r10, %%r11)
  489. #elif WOLFSSL_SIPHASH_DROUNDS == 4
  490. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  491. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  492. SIPHASH_ROUND(%%r8, %%r9, %%r10, %%r11)
  493. SIPHASH_LAST_ROUND(%%r8, %%r9, %%r10, %%r11)
  494. #endif
  495. "xorq %%r11, %%r9\n\t"
  496. "xorq %%r10, %%r9\n\t"
  497. "movq %%r9, (%[out])\n\t"
  498. "L_siphash_done:\n\t"
  499. : [in] "+r" (in), [inSz] "+r" (inSz)
  500. : [key] "r" (key), [out] "r" (out) , [outSz] "r" (outSz)
  501. : "memory", "%r8", "%r9", "%r10", "%r11", "%r12", "%r13"
  502. );
  503. return 0;
  504. }
  505. #elif !defined(WOLFSSL_NO_ASM) && defined(__GNUC__) && defined(__aarch64__) && \
  506. (WOLFSSL_SIPHASH_CROUNDS == 1 || WOLFSSL_SIPHASH_CROUNDS == 2) && \
  507. (WOLFSSL_SIPHASH_DROUNDS == 2 || WOLFSSL_SIPHASH_DROUNDS == 4)
  508. #define SIPHASH_ROUND(v0, v1, v2, v3) \
  509. "add " #v0 ", " #v0 ", " #v1 "\n\t" \
  510. "add " #v2 ", " #v2 ", " #v3 "\n\t" \
  511. "ror " #v1 ", " #v1 ", #51\n\t" \
  512. "ror " #v3 ", " #v3 ", #48\n\t" \
  513. "eor " #v1 ", " #v1 ", " #v0 "\n\t" \
  514. "eor " #v3 ", " #v3 ", " #v2 "\n\t" \
  515. "ror " #v0 ", " #v0 ", #32\n\t" \
  516. "add " #v2 ", " #v2 ", " #v1 "\n\t" \
  517. "add " #v0 ", " #v0 ", " #v3 "\n\t" \
  518. "ror " #v1 ", " #v1 ", #47\n\t" \
  519. "ror " #v3 ", " #v3 ", #43\n\t" \
  520. "eor " #v1 ", " #v1 ", " #v2 "\n\t" \
  521. "eor " #v3 ", " #v3 ", " #v0 "\n\t" \
  522. "ror " #v2 ", " #v2 ", #32\n\t"
  523. #define SIPHASH_LAST_ROUND(v0, v1, v2, v3) \
  524. "add " #v0 ", " #v0 ", " #v1 "\n\t" \
  525. "add " #v2 ", " #v2 ", " #v3 "\n\t" \
  526. "ror " #v1 ", " #v1 ", #51\n\t" \
  527. "ror " #v3 ", " #v3 ", #48\n\t" \
  528. "eor " #v1 ", " #v1 ", " #v0 "\n\t" \
  529. "eor " #v3 ", " #v3 ", " #v2 "\n\t" \
  530. "add " #v2 ", " #v2 ", " #v1 "\n\t" \
  531. "ror " #v1 ", " #v1 ", #47\n\t" \
  532. "ror " #v3 ", " #v3 ", #43\n\t" \
  533. "eor " #v1 ", " #v1 ", " #v2 "\n\t" \
  534. "ror " #v2 ", " #v2 ", #32\n\t"
  535. /**
  536. * Perform SipHash operation on input with key.
  537. *
  538. * @param [in] key 16 byte array - little endian.
  539. * @param [in] in Input message.
  540. * @param [in] inSz Size of input message.
  541. * @param [out] out Buffer to place MAC into.
  542. * @param [in] outSz Size of ouput MAC. 8 or 16 only.
  543. * @return BAD_FUNC_ARG when key or out is NULL.
  544. * @return BAD_FUNC_ARG when in is NULL and inSz is not zero.
  545. * @return BAD_FUNC_ARG when outSz is not 8 nor 16.
  546. * @return 0 on success.
  547. */
  548. int wc_SipHash(const unsigned char* key, const unsigned char* in, word32 inSz,
  549. unsigned char* out, unsigned char outSz)
  550. {
  551. if ((key == NULL) || ((in == NULL) && (inSz != 0)) || (out == NULL) ||
  552. ((outSz != SIPHASH_MAC_SIZE_8) && (outSz != SIPHASH_MAC_SIZE_16))) {
  553. return BAD_FUNC_ARG;
  554. }
  555. /* v0=x8, v1=x9, v2=x10, v3=x11 */
  556. __asm__ __volatile__ (
  557. "ldp x12, x13, [%[key]]\n\t"
  558. "mov x8, #0x6575\n\t"
  559. "movk x8, #0x7073, lsl #16\n\t"
  560. "movk x8, #0x6d65, lsl #32\n\t"
  561. "movk x8, #0x736f, lsl #48\n\t"
  562. "mov x9, #0x6f6d\n\t"
  563. "movk x9, #0x6e64, lsl #16\n\t"
  564. "movk x9, #0x7261, lsl #32\n\t"
  565. "movk x9, #0x646f, lsl #48\n\t"
  566. "mov x10, #0x7261\n\t"
  567. "movk x10, #0x6e65, lsl #16\n\t"
  568. "movk x10, #0x6765, lsl #32\n\t"
  569. "movk x10, #0x6c79, lsl #48\n\t"
  570. "mov x11, #0x6573\n\t"
  571. "movk x11, #0x7974, lsl #16\n\t"
  572. "movk x11, #0x6462, lsl #32\n\t"
  573. "movk x11, #0x7465, lsl #48\n\t"
  574. "eor x8, x8, x12\n\t"
  575. "eor x9, x9, x13\n\t"
  576. "eor x10, x10, x12\n\t"
  577. "eor x11, x11, x13\n\t"
  578. "mov w13, %w[inSz]\n\t"
  579. "cmp %w[outSz], #8\n\t"
  580. "b.eq L_siphash_8_top\n\t"
  581. "mov w12, #0xee\n\t"
  582. "eor x9, x9, x12\n\t"
  583. "L_siphash_8_top:\n\t"
  584. "subs %w[inSz], %w[inSz], #8\n\t"
  585. "b.mi L_siphash_done_input_8\n\t"
  586. "L_siphash_input:\n\t"
  587. "ldr x12, [%[in]], #8\n\t"
  588. "eor x11, x11, x12\n\t"
  589. #if WOLFSSL_SIPHASH_CROUNDS == 1
  590. SIPHASH_ROUND(x8, x9, x10, x11)
  591. #elif WOLFSSL_SIPHASH_CROUNDS == 2
  592. SIPHASH_ROUND(x8, x9, x10, x11)
  593. SIPHASH_ROUND(x8, x9, x10, x11)
  594. #endif
  595. "eor x8, x8, x12\n\t"
  596. "subs %w[inSz], %w[inSz], #8\n\t"
  597. "b.ge L_siphash_input\n\t"
  598. "L_siphash_done_input_8:\n\t"
  599. "add %w[inSz], %w[inSz], #8\n\t"
  600. "lsl x13, x13, #56\n\t"
  601. "cmp %w[inSz], #0\n\t"
  602. "b.eq L_siphash_last_done\n\t"
  603. "cmp %w[inSz], #4\n\t"
  604. "b.lt L_siphash_last_lt4\n\t"
  605. "cmp %w[inSz], #7\n\t"
  606. "b.lt L_siphash_n7\n\t"
  607. "ldrb w12, [%[in], 6]\n\t"
  608. "orr x13, x13, x12, lsl 48\n\t"
  609. "L_siphash_n7:\n\t"
  610. "cmp %w[inSz], #6\n\t"
  611. "b.lt L_siphash_n6\n\t"
  612. "ldrb w12, [%[in], 5]\n\t"
  613. "orr x13, x13, x12, lsl 40\n\t"
  614. "L_siphash_n6:\n\t"
  615. "cmp %w[inSz], #5\n\t"
  616. "b.lt L_siphash_n5\n\t"
  617. "ldrb w12, [%[in], 4]\n\t"
  618. "orr x13, x13, x12, lsl 32\n\t"
  619. "L_siphash_n5:\n\t"
  620. "ldr w12, [%[in]]\n\t"
  621. "orr x13, x13, x12\n\t"
  622. "b L_siphash_last_done\n\t"
  623. "L_siphash_last_lt4:\n\t"
  624. "cmp %w[inSz], #1\n\t"
  625. "b.eq L_siphash_last_1\n\t"
  626. "cmp %w[inSz], #3\n\t"
  627. "b.lt L_siphash_n3\n\t"
  628. "ldrb w12, [%[in], 2]\n\t"
  629. "orr x13, x13, x12, lsl 16\n\t"
  630. "L_siphash_n3:\n\t"
  631. "ldrh w12, [%[in]]\n\t"
  632. "orr x13, x13, x12\n\t"
  633. "b L_siphash_last_done\n\t"
  634. "L_siphash_last_1:\n\t"
  635. "ldrb w12, [%[in]]\n\t"
  636. "orr x13, x13, x12\n\t"
  637. "L_siphash_last_done:\n\t"
  638. "eor x11, x11, x13\n\t"
  639. #if WOLFSSL_SIPHASH_CROUNDS == 1
  640. SIPHASH_ROUND(x8, x9, x10, x11)
  641. #elif WOLFSSL_SIPHASH_CROUNDS == 2
  642. SIPHASH_ROUND(x8, x9, x10, x11)
  643. SIPHASH_ROUND(x8, x9, x10, x11)
  644. #endif
  645. "eor x8, x8, x13\n\t"
  646. "cmp %w[outSz], #8\n\t"
  647. "b.eq L_siphash_8_end\n\t"
  648. "mov w13, #0xee\n\t"
  649. "eor x10, x10, x13\n\t"
  650. #if WOLFSSL_SIPHASH_DROUNDS == 2
  651. SIPHASH_ROUND(x8, x9, x10, x11)
  652. SIPHASH_ROUND(x8, x9, x10, x11)
  653. #elif WOLFSSL_SIPHASH_DROUNDS == 4
  654. SIPHASH_ROUND(x8, x9, x10, x11)
  655. SIPHASH_ROUND(x8, x9, x10, x11)
  656. SIPHASH_ROUND(x8, x9, x10, x11)
  657. SIPHASH_ROUND(x8, x9, x10, x11)
  658. #endif
  659. "eor x12, x8, x9\n\t"
  660. "eor x13, x10, x11\n\t"
  661. "eor x12, x12, x13\n\t"
  662. "mov w13, #0xdd\n\t"
  663. "eor x9, x9, x13\n\t"
  664. #if WOLFSSL_SIPHASH_DROUNDS == 2
  665. SIPHASH_ROUND(x8, x9, x10, x11)
  666. SIPHASH_LAST_ROUND(x8, x9, x10, x11)
  667. #elif WOLFSSL_SIPHASH_DROUNDS == 4
  668. SIPHASH_ROUND(x8, x9, x10, x11)
  669. SIPHASH_ROUND(x8, x9, x10, x11)
  670. SIPHASH_ROUND(x8, x9, x10, x11)
  671. SIPHASH_LAST_ROUND(x8, x9, x10, x11)
  672. #endif
  673. "eor x13, x11, x9\n\t"
  674. "eor x13, x13, x10\n\t"
  675. "stp x12, x13, [%[out]]\n\t"
  676. "b L_siphash_done\n\t"
  677. "L_siphash_8_end:\n\t"
  678. "mov w13, #0xff\n\t"
  679. "eor x10, x10, x13\n\t"
  680. #if WOLFSSL_SIPHASH_DROUNDS == 2
  681. SIPHASH_ROUND(x8, x9, x10, x11)
  682. SIPHASH_LAST_ROUND(x8, x9, x10, x11)
  683. #elif WOLFSSL_SIPHASH_DROUNDS == 4
  684. SIPHASH_ROUND(x8, x9, x10, x11)
  685. SIPHASH_ROUND(x8, x9, x10, x11)
  686. SIPHASH_ROUND(x8, x9, x10, x11)
  687. SIPHASH_LAST_ROUND(x8, x9, x10, x11)
  688. #endif
  689. "eor x13, x11, x9\n\t"
  690. "eor x13, x13, x10\n\t"
  691. "str x13, [%[out]]\n\t"
  692. "L_siphash_done:\n\t"
  693. : [in] "+r" (in), [inSz] "+r" (inSz)
  694. : [key] "r" (key), [out] "r" (out) , [outSz] "r" (outSz)
  695. : "memory", "x8", "x9", "x10", "x11", "x12", "x13"
  696. );
  697. return 0;
  698. }
  699. #else
  700. #define SipRoundV(v0, v1, v2, v3) \
  701. v0 += v1; \
  702. v2 += v3; \
  703. v1 = rotlFixed64(v1, 13); \
  704. v3 = rotlFixed64(v3, 16); \
  705. v1 ^= v0; \
  706. v3 ^= v2; \
  707. v0 = rotlFixed64(v0, 32); \
  708. v2 += v1; \
  709. v0 += v3; \
  710. v1 = rotlFixed64(v1, 17); \
  711. v3 = rotlFixed64(v3, 21); \
  712. v1 ^= v2; \
  713. v3 ^= v0; \
  714. v2 = rotlFixed64(v2, 32);
  715. #define SipHashCompressV(v0, v1, v2, v3, m) \
  716. do { \
  717. int i; \
  718. v3 ^= m; \
  719. for (i = 0; i < WOLFSSL_SIPHASH_CROUNDS; i++) { \
  720. SipRoundV(v0, v1, v2, v3); \
  721. } \
  722. v0 ^= m; \
  723. } \
  724. while (0)
  725. #define SipHashOutV(v0, v1, v2, v3, out) \
  726. do { \
  727. word64 n; \
  728. int i; \
  729. \
  730. for (i = 0; i < WOLFSSL_SIPHASH_DROUNDS; i++) { \
  731. SipRoundV(v0, v1, v2, v3); \
  732. } \
  733. n = v0 ^ v1 ^ v2 ^ v3; \
  734. SET_U64(out, n); \
  735. } \
  736. while (0)
  737. /**
  738. * Perform SipHash operation on input with key.
  739. *
  740. * @param [in] key 16 byte array - little endian.
  741. * @param [in] in Input message.
  742. * @param [in] inSz Size of input message.
  743. * @param [out] out Buffer to place MAC into.
  744. * @param [in] outSz Size of ouput MAC. 8 or 16 only.
  745. * @return BAD_FUNC_ARG when key or out is NULL.
  746. * @return BAD_FUNC_ARG when in is NULL and inSz is not zero.
  747. * @return BAD_FUNC_ARG when outSz is not 8 nor 16.
  748. * @return 0 on success.
  749. */
  750. int wc_SipHash(const unsigned char* key, const unsigned char* in, word32 inSz,
  751. unsigned char* out, unsigned char outSz)
  752. {
  753. int ret = 0;
  754. if ((key == NULL) || ((in == NULL) && (inSz != 0)) || (out == NULL) ||
  755. ((outSz != SIPHASH_MAC_SIZE_8) && (outSz != SIPHASH_MAC_SIZE_16))) {
  756. ret = BAD_FUNC_ARG;
  757. }
  758. if (ret == 0) {
  759. word64 v0, v1, v2, v3;
  760. word64 k0 = GET_U64(key + 0);
  761. word64 k1 = GET_U64(key + 8);
  762. word64 b = (word64)((word64)inSz << 56);
  763. /* Initialize state with key. */
  764. v0 = 0x736f6d6570736575ULL;
  765. v1 = 0x646f72616e646f6dULL;
  766. v2 = 0x6c7967656e657261ULL;
  767. v3 = 0x7465646279746573ULL;
  768. if (outSz == SIPHASH_MAC_SIZE_16) {
  769. v1 ^= 0xee;
  770. }
  771. v0 ^= k0;
  772. v1 ^= k1;
  773. v2 ^= k0;
  774. v3 ^= k1;
  775. /* Process blocks from message. */
  776. while (inSz >= SIPHASH_BLOCK_SIZE) {
  777. word64 m = GET_U64(in);
  778. /* Compress the next block from the message data. */
  779. SipHashCompressV(v0, v1, v2, v3, m);
  780. in += SIPHASH_BLOCK_SIZE;
  781. inSz -= SIPHASH_BLOCK_SIZE;
  782. }
  783. switch (inSz) {
  784. case 7:
  785. b |= (word64)in[6] << 48;
  786. FALL_THROUGH;
  787. case 6:
  788. b |= (word64)in[5] << 40;
  789. FALL_THROUGH;
  790. case 5:
  791. b |= (word64)in[4] << 32;
  792. FALL_THROUGH;
  793. case 4:
  794. b |= (word64)GET_U32(in);
  795. break;
  796. case 3:
  797. b |= (word64)in[2] << 16;
  798. FALL_THROUGH;
  799. case 2:
  800. b |= (word64)GET_U16(in);
  801. break;
  802. case 1:
  803. b |= (word64)in[0];
  804. break;
  805. case 0:
  806. break;
  807. }
  808. SipHashCompressV(v0, v1, v2, v3, b);
  809. /* Output either 8 or 16 bytes. */
  810. if (outSz == SIPHASH_MAC_SIZE_8) {
  811. v2 ^= (word64)0xff;
  812. SipHashOutV(v0, v1, v2, v3, out);
  813. }
  814. else {
  815. v2 ^= (word64)0xee;
  816. SipHashOutV(v0, v1, v2, v3, out);
  817. v1 ^= (word64)0xdd;
  818. SipHashOutV(v0, v1, v2, v3, out + 8);
  819. }
  820. }
  821. return ret;
  822. }
  823. #endif /* !ASM */
  824. #endif /* WOLFSSL_SIPHASH */