curl_sha512_256.c 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850
  1. /***************************************************************************
  2. * _ _ ____ _
  3. * Project ___| | | | _ \| |
  4. * / __| | | | |_) | |
  5. * | (__| |_| | _ <| |___
  6. * \___|\___/|_| \_\_____|
  7. *
  8. * Copyright (C) Evgeny Grin (Karlson2k), <k2k@narod.ru>.
  9. *
  10. * This software is licensed as described in the file COPYING, which
  11. * you should have received as part of this distribution. The terms
  12. * are also available at https://curl.se/docs/copyright.html.
  13. *
  14. * You may opt to use, copy, modify, merge, publish, distribute and/or sell
  15. * copies of the Software, and permit persons to whom the Software is
  16. * furnished to do so, under the terms of the COPYING file.
  17. *
  18. * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
  19. * KIND, either express or implied.
  20. *
  21. * SPDX-License-Identifier: curl
  22. *
  23. ***************************************************************************/
  24. #include "curl_setup.h"
  25. #if !defined(CURL_DISABLE_DIGEST_AUTH) && !defined(CURL_DISABLE_SHA512_256)
  26. #include "curl_sha512_256.h"
  27. #include "warnless.h"
  28. /* The recommended order of the TLS backends:
  29. * * OpenSSL
  30. * * GnuTLS
  31. * * wolfSSL
  32. * * Schannel SSPI
  33. * * SecureTransport (Darwin)
  34. * * mbedTLS
  35. * * BearSSL
  36. * * rustls
  37. * Skip the backend if it does not support the required algorithm */
  38. #if defined(USE_OPENSSL)
  39. # include <openssl/opensslv.h>
  40. # if (!defined(LIBRESSL_VERSION_NUMBER) && \
  41. defined(OPENSSL_VERSION_NUMBER) && \
  42. (OPENSSL_VERSION_NUMBER >= 0x10101000L)) || \
  43. (defined(LIBRESSL_VERSION_NUMBER) && \
  44. (LIBRESSL_VERSION_NUMBER >= 0x3080000fL))
  45. # include <openssl/opensslconf.h>
  46. # if !defined(OPENSSL_NO_SHA) && !defined(OPENSSL_NO_SHA512)
  47. # include <openssl/evp.h>
  48. # define USE_OPENSSL_SHA512_256 1
  49. # define HAS_SHA512_256_IMPLEMENTATION 1
  50. # ifdef __NetBSD__
  51. /* Some NetBSD versions has a bug in SHA-512/256.
  52. * See https://gnats.netbsd.org/cgi-bin/query-pr-single.pl?number=58039
  53. * The problematic versions:
  54. * - NetBSD before 9.4
  55. * - NetBSD 9 all development versions (9.99.x)
  56. * - NetBSD 10 development versions (10.99.x) before 10.99.11
  57. * The bug was fixed in NetBSD 9.4 release, NetBSD 10.0 release,
  58. * NetBSD 10.99.11 development.
  59. * It is safe to apply the workaround even if the bug is not present, as
  60. * the workaround just reduces performance slightly. */
  61. # include <sys/param.h>
  62. # if __NetBSD_Version__ < 904000000 || \
  63. (__NetBSD_Version__ >= 999000000 && \
  64. __NetBSD_Version__ < 1000000000) || \
  65. (__NetBSD_Version__ >= 1099000000 && \
  66. __NetBSD_Version__ < 1099001100)
  67. # define NEED_NETBSD_SHA512_256_WORKAROUND 1
  68. # include <string.h>
  69. # endif
  70. # endif
  71. # endif
  72. # endif
  73. #endif /* USE_OPENSSL */
  74. #if !defined(HAS_SHA512_256_IMPLEMENTATION) && defined(USE_GNUTLS)
  75. # include <nettle/sha.h>
  76. # if defined(SHA512_256_DIGEST_SIZE)
  77. # define USE_GNUTLS_SHA512_256 1
  78. # define HAS_SHA512_256_IMPLEMENTATION 1
  79. # endif
  80. #endif /* ! HAS_SHA512_256_IMPLEMENTATION && USE_GNUTLS */
  81. #if defined(USE_OPENSSL_SHA512_256)
  82. /* OpenSSL does not provide macros for SHA-512/256 sizes */
  83. /**
  84. * Size of the SHA-512/256 single processing block in bytes.
  85. */
  86. #define SHA512_256_BLOCK_SIZE 128
  87. /**
  88. * Size of the SHA-512/256 resulting digest in bytes.
  89. * This is the final digest size, not intermediate hash.
  90. */
  91. #define SHA512_256_DIGEST_SIZE SHA512_256_DIGEST_LENGTH
  92. /**
  93. * Context type used for SHA-512/256 calculations
  94. */
  95. typedef EVP_MD_CTX *Curl_sha512_256_ctx;
  96. /**
  97. * Initialise structure for SHA-512/256 calculation.
  98. *
  99. * @param context the calculation context
  100. * @return CURLE_OK if succeed,
  101. * error code otherwise
  102. */
  103. static CURLcode
  104. Curl_sha512_256_init(void *context)
  105. {
  106. Curl_sha512_256_ctx *const ctx = (Curl_sha512_256_ctx *)context;
  107. *ctx = EVP_MD_CTX_create();
  108. if(!*ctx)
  109. return CURLE_OUT_OF_MEMORY;
  110. if(EVP_DigestInit_ex(*ctx, EVP_sha512_256(), NULL)) {
  111. /* Check whether the header and this file use the same numbers */
  112. DEBUGASSERT(EVP_MD_CTX_size(*ctx) == SHA512_256_DIGEST_SIZE);
  113. /* Check whether the block size is correct */
  114. DEBUGASSERT(EVP_MD_CTX_block_size(*ctx) == SHA512_256_BLOCK_SIZE);
  115. return CURLE_OK; /* Success */
  116. }
  117. /* Cleanup */
  118. EVP_MD_CTX_destroy(*ctx);
  119. return CURLE_FAILED_INIT;
  120. }
  121. /**
  122. * Process portion of bytes.
  123. *
  124. * @param context the calculation context
  125. * @param data bytes to add to hash
  126. * @return CURLE_OK if succeed,
  127. * error code otherwise
  128. */
  129. static CURLcode
  130. Curl_sha512_256_update(void *context,
  131. const unsigned char *data,
  132. size_t length)
  133. {
  134. Curl_sha512_256_ctx *const ctx = (Curl_sha512_256_ctx *)context;
  135. if(!EVP_DigestUpdate(*ctx, data, length))
  136. return CURLE_SSL_CIPHER;
  137. return CURLE_OK;
  138. }
  139. /**
  140. * Finalise SHA-512/256 calculation, return digest.
  141. *
  142. * @param context the calculation context
  143. * @param[out] digest set to the hash, must be #SHA512_256_DIGEST_SIZE bytes
  144. * @return CURLE_OK if succeed,
  145. * error code otherwise
  146. */
  147. static CURLcode
  148. Curl_sha512_256_finish(unsigned char *digest,
  149. void *context)
  150. {
  151. CURLcode ret;
  152. Curl_sha512_256_ctx *const ctx = (Curl_sha512_256_ctx *)context;
  153. #ifdef NEED_NETBSD_SHA512_256_WORKAROUND
  154. /* Use a larger buffer to work around a bug in NetBSD:
  155. https://gnats.netbsd.org/cgi-bin/query-pr-single.pl?number=58039 */
  156. unsigned char tmp_digest[SHA512_256_DIGEST_SIZE * 2];
  157. ret = EVP_DigestFinal_ex(*ctx,
  158. tmp_digest, NULL) ? CURLE_OK : CURLE_SSL_CIPHER;
  159. if(ret == CURLE_OK)
  160. memcpy(digest, tmp_digest, SHA512_256_DIGEST_SIZE);
  161. explicit_memset(tmp_digest, 0, sizeof(tmp_digest));
  162. #else /* ! NEED_NETBSD_SHA512_256_WORKAROUND */
  163. ret = EVP_DigestFinal_ex(*ctx, digest, NULL) ? CURLE_OK : CURLE_SSL_CIPHER;
  164. #endif /* ! NEED_NETBSD_SHA512_256_WORKAROUND */
  165. EVP_MD_CTX_destroy(*ctx);
  166. *ctx = NULL;
  167. return ret;
  168. }
  169. #elif defined(USE_GNUTLS_SHA512_256)
  170. /**
  171. * Context type used for SHA-512/256 calculations
  172. */
  173. typedef struct sha512_256_ctx Curl_sha512_256_ctx;
  174. /**
  175. * Initialise structure for SHA-512/256 calculation.
  176. *
  177. * @param context the calculation context
  178. * @return always CURLE_OK
  179. */
  180. static CURLcode
  181. Curl_sha512_256_init(void *context)
  182. {
  183. Curl_sha512_256_ctx *const ctx = (Curl_sha512_256_ctx *)context;
  184. /* Check whether the header and this file use the same numbers */
  185. DEBUGASSERT(SHA512_256_DIGEST_LENGTH == SHA512_256_DIGEST_SIZE);
  186. sha512_256_init(ctx);
  187. return CURLE_OK;
  188. }
  189. /**
  190. * Process portion of bytes.
  191. *
  192. * @param context the calculation context
  193. * @param data bytes to add to hash
  194. * @param length number of bytes in @a data
  195. * @return always CURLE_OK
  196. */
  197. static CURLcode
  198. Curl_sha512_256_update(void *context,
  199. const unsigned char *data,
  200. size_t length)
  201. {
  202. Curl_sha512_256_ctx *const ctx = (Curl_sha512_256_ctx *)context;
  203. DEBUGASSERT((data != NULL) || (length == 0));
  204. sha512_256_update(ctx, length, (const uint8_t *)data);
  205. return CURLE_OK;
  206. }
  207. /**
  208. * Finalise SHA-512/256 calculation, return digest.
  209. *
  210. * @param context the calculation context
  211. * @param[out] digest set to the hash, must be #SHA512_256_DIGEST_SIZE bytes
  212. * @return always CURLE_OK
  213. */
  214. static CURLcode
  215. Curl_sha512_256_finish(unsigned char *digest,
  216. void *context)
  217. {
  218. Curl_sha512_256_ctx *const ctx = (Curl_sha512_256_ctx *)context;
  219. sha512_256_digest(ctx, (size_t)SHA512_256_DIGEST_SIZE, (uint8_t *)digest);
  220. return CURLE_OK;
  221. }
  222. #else /* No system or TLS backend SHA-512/256 implementation available */
  223. /* Use local implementation */
  224. #define HAS_SHA512_256_IMPLEMENTATION 1
  225. /* ** This implementation of SHA-512/256 hash calculation was originally ** *
  226. * ** written by Evgeny Grin (Karlson2k) for GNU libmicrohttpd. ** *
  227. * ** The author ported the code to libcurl. The ported code is provided ** *
  228. * ** under curl license. ** *
  229. * ** This is a minimal version with minimal optimisations. Performance ** *
  230. * ** can be significantly improved. Big-endian store and load macros ** *
  231. * ** are obvious targets for optimisation. ** */
  232. #ifdef __GNUC__
  233. # if defined(__has_attribute) && defined(__STDC_VERSION__)
  234. # if __has_attribute(always_inline) && __STDC_VERSION__ >= 199901
  235. # define MHDX_INLINE inline __attribute__((always_inline))
  236. # endif
  237. # endif
  238. #endif
  239. #if !defined(MHDX_INLINE) && \
  240. defined(_MSC_VER) && !defined(__GNUC__) && !defined(__clang__)
  241. # if _MSC_VER >= 1400
  242. # define MHDX_INLINE __forceinline
  243. # endif
  244. #endif
  245. #if !defined(MHDX_INLINE)
  246. /* Assume that 'inline' keyword works or the
  247. * macro was already defined correctly. */
  248. # define MHDX_INLINE inline
  249. #endif
  250. /* Bits manipulation macros and functions.
  251. Can be moved to other headers to reuse. */
  252. #define MHDX_GET_64BIT_BE(ptr) \
  253. ( ((curl_uint64_t)(((const unsigned char*)(ptr))[0]) << 56) | \
  254. ((curl_uint64_t)(((const unsigned char*)(ptr))[1]) << 48) | \
  255. ((curl_uint64_t)(((const unsigned char*)(ptr))[2]) << 40) | \
  256. ((curl_uint64_t)(((const unsigned char*)(ptr))[3]) << 32) | \
  257. ((curl_uint64_t)(((const unsigned char*)(ptr))[4]) << 24) | \
  258. ((curl_uint64_t)(((const unsigned char*)(ptr))[5]) << 16) | \
  259. ((curl_uint64_t)(((const unsigned char*)(ptr))[6]) << 8) | \
  260. (curl_uint64_t)(((const unsigned char*)(ptr))[7]) )
  261. #define MHDX_PUT_64BIT_BE(ptr,val) do { \
  262. ((unsigned char*)(ptr))[7]=(unsigned char)((curl_uint64_t)(val)); \
  263. ((unsigned char*)(ptr))[6]=(unsigned char)(((curl_uint64_t)(val)) >> 8); \
  264. ((unsigned char*)(ptr))[5]=(unsigned char)(((curl_uint64_t)(val)) >> 16); \
  265. ((unsigned char*)(ptr))[4]=(unsigned char)(((curl_uint64_t)(val)) >> 24); \
  266. ((unsigned char*)(ptr))[3]=(unsigned char)(((curl_uint64_t)(val)) >> 32); \
  267. ((unsigned char*)(ptr))[2]=(unsigned char)(((curl_uint64_t)(val)) >> 40); \
  268. ((unsigned char*)(ptr))[1]=(unsigned char)(((curl_uint64_t)(val)) >> 48); \
  269. ((unsigned char*)(ptr))[0]=(unsigned char)(((curl_uint64_t)(val)) >> 56); \
  270. } while(0)
  271. /* Defined as a function. The macro version may duplicate the binary code
  272. * size as each argument is used twice, so if any calculation is used
  273. * as an argument, the calculation could be done twice. */
  274. static MHDX_INLINE curl_uint64_t
  275. MHDx_rotr64(curl_uint64_t value, unsigned int bits)
  276. {
  277. bits %= 64;
  278. if(0 == bits)
  279. return value;
  280. /* Defined in a form which modern compiler could optimise. */
  281. return (value >> bits) | (value << (64 - bits));
  282. }
  283. /* SHA-512/256 specific data */
  284. /**
  285. * Number of bits in a single SHA-512/256 word.
  286. */
  287. #define SHA512_256_WORD_SIZE_BITS 64
  288. /**
  289. * Number of bytes in a single SHA-512/256 word.
  290. */
  291. #define SHA512_256_BYTES_IN_WORD (SHA512_256_WORD_SIZE_BITS / 8)
  292. /**
  293. * Hash is kept internally as 8 64-bit words.
  294. * This is the intermediate hash size, used during computing the final digest.
  295. */
  296. #define SHA512_256_HASH_SIZE_WORDS 8
  297. /**
  298. * Size of the SHA-512/256 resulting digest in words.
  299. * This is the final digest size, not intermediate hash.
  300. */
  301. #define SHA512_256_DIGEST_SIZE_WORDS (SHA512_256_HASH_SIZE_WORDS / 2)
  302. /**
  303. * Size of the SHA-512/256 resulting digest in bytes
  304. * This is the final digest size, not intermediate hash.
  305. */
  306. #define SHA512_256_DIGEST_SIZE \
  307. (SHA512_256_DIGEST_SIZE_WORDS * SHA512_256_BYTES_IN_WORD)
  308. /**
  309. * Size of the SHA-512/256 single processing block in bits.
  310. */
  311. #define SHA512_256_BLOCK_SIZE_BITS 1024
  312. /**
  313. * Size of the SHA-512/256 single processing block in bytes.
  314. */
  315. #define SHA512_256_BLOCK_SIZE (SHA512_256_BLOCK_SIZE_BITS / 8)
  316. /**
  317. * Size of the SHA-512/256 single processing block in words.
  318. */
  319. #define SHA512_256_BLOCK_SIZE_WORDS \
  320. (SHA512_256_BLOCK_SIZE_BITS / SHA512_256_WORD_SIZE_BITS)
  321. /**
  322. * SHA-512/256 calculation context
  323. */
  324. struct mhdx_sha512_256ctx
  325. {
  326. /**
  327. * Intermediate hash value. The variable is properly aligned. Smart
  328. * compilers may automatically use fast load/store instruction for big
  329. * endian data on little endian machine.
  330. */
  331. curl_uint64_t H[SHA512_256_HASH_SIZE_WORDS];
  332. /**
  333. * SHA-512/256 input data buffer. The buffer is properly aligned. Smart
  334. * compilers may automatically use fast load/store instruction for big
  335. * endian data on little endian machine.
  336. */
  337. curl_uint64_t buffer[SHA512_256_BLOCK_SIZE_WORDS];
  338. /**
  339. * The number of bytes, lower part
  340. */
  341. curl_uint64_t count;
  342. /**
  343. * The number of bits, high part. Unlike lower part, this counts the number
  344. * of bits, not bytes.
  345. */
  346. curl_uint64_t count_bits_hi;
  347. };
  348. /**
  349. * Context type used for SHA-512/256 calculations
  350. */
  351. typedef struct mhdx_sha512_256ctx Curl_sha512_256_ctx;
  352. /**
  353. * Initialise structure for SHA-512/256 calculation.
  354. *
  355. * @param context the calculation context
  356. * @return always CURLE_OK
  357. */
  358. static CURLcode
  359. MHDx_sha512_256_init(void *context)
  360. {
  361. struct mhdx_sha512_256ctx *const ctx = (struct mhdx_sha512_256ctx *) context;
  362. /* Check whether the header and this file use the same numbers */
  363. DEBUGASSERT(SHA512_256_DIGEST_LENGTH == SHA512_256_DIGEST_SIZE);
  364. DEBUGASSERT(sizeof(curl_uint64_t) == 8);
  365. /* Initial hash values, see FIPS PUB 180-4 section 5.3.6.2 */
  366. /* Values generated by "IV Generation Function" as described in
  367. * section 5.3.6 */
  368. ctx->H[0] = CURL_UINT64_C(0x22312194FC2BF72C);
  369. ctx->H[1] = CURL_UINT64_C(0x9F555FA3C84C64C2);
  370. ctx->H[2] = CURL_UINT64_C(0x2393B86B6F53B151);
  371. ctx->H[3] = CURL_UINT64_C(0x963877195940EABD);
  372. ctx->H[4] = CURL_UINT64_C(0x96283EE2A88EFFE3);
  373. ctx->H[5] = CURL_UINT64_C(0xBE5E1E2553863992);
  374. ctx->H[6] = CURL_UINT64_C(0x2B0199FC2C85B8AA);
  375. ctx->H[7] = CURL_UINT64_C(0x0EB72DDC81C52CA2);
  376. /* Initialise number of bytes and high part of number of bits. */
  377. ctx->count = CURL_UINT64_C(0);
  378. ctx->count_bits_hi = CURL_UINT64_C(0);
  379. return CURLE_OK;
  380. }
  381. /**
  382. * Base of the SHA-512/256 transformation.
  383. * Gets a full 128 bytes block of data and updates hash values;
  384. * @param H hash values
  385. * @param data the data buffer with #SHA512_256_BLOCK_SIZE bytes block
  386. */
  387. static void
  388. MHDx_sha512_256_transform(curl_uint64_t H[SHA512_256_HASH_SIZE_WORDS],
  389. const void *data)
  390. {
  391. /* Working variables,
  392. see FIPS PUB 180-4 section 6.7, 6.4. */
  393. curl_uint64_t a = H[0];
  394. curl_uint64_t b = H[1];
  395. curl_uint64_t c = H[2];
  396. curl_uint64_t d = H[3];
  397. curl_uint64_t e = H[4];
  398. curl_uint64_t f = H[5];
  399. curl_uint64_t g = H[6];
  400. curl_uint64_t h = H[7];
  401. /* Data buffer, used as a cyclic buffer.
  402. See FIPS PUB 180-4 section 5.2.2, 6.7, 6.4. */
  403. curl_uint64_t W[16];
  404. /* 'Ch' and 'Maj' macro functions are defined with widely-used optimisation.
  405. See FIPS PUB 180-4 formulae 4.8, 4.9. */
  406. #define Sha512_Ch(x,y,z) ( (z) ^ ((x) & ((y) ^ (z))) )
  407. #define Sha512_Maj(x,y,z) ( ((x) & (y)) ^ ((z) & ((x) ^ (y))) )
  408. /* Four 'Sigma' macro functions.
  409. See FIPS PUB 180-4 formulae 4.10, 4.11, 4.12, 4.13. */
  410. #define SIG0(x) \
  411. ( MHDx_rotr64((x), 28) ^ MHDx_rotr64((x), 34) ^ MHDx_rotr64((x), 39) )
  412. #define SIG1(x) \
  413. ( MHDx_rotr64((x), 14) ^ MHDx_rotr64((x), 18) ^ MHDx_rotr64((x), 41) )
  414. #define sig0(x) \
  415. ( MHDx_rotr64((x), 1) ^ MHDx_rotr64((x), 8) ^ ((x) >> 7) )
  416. #define sig1(x) \
  417. ( MHDx_rotr64((x), 19) ^ MHDx_rotr64((x), 61) ^ ((x) >> 6) )
  418. if(1) {
  419. unsigned int t;
  420. /* K constants array.
  421. See FIPS PUB 180-4 section 4.2.3 for K values. */
  422. static const curl_uint64_t K[80] = {
  423. CURL_UINT64_C(0x428a2f98d728ae22), CURL_UINT64_C(0x7137449123ef65cd),
  424. CURL_UINT64_C(0xb5c0fbcfec4d3b2f), CURL_UINT64_C(0xe9b5dba58189dbbc),
  425. CURL_UINT64_C(0x3956c25bf348b538), CURL_UINT64_C(0x59f111f1b605d019),
  426. CURL_UINT64_C(0x923f82a4af194f9b), CURL_UINT64_C(0xab1c5ed5da6d8118),
  427. CURL_UINT64_C(0xd807aa98a3030242), CURL_UINT64_C(0x12835b0145706fbe),
  428. CURL_UINT64_C(0x243185be4ee4b28c), CURL_UINT64_C(0x550c7dc3d5ffb4e2),
  429. CURL_UINT64_C(0x72be5d74f27b896f), CURL_UINT64_C(0x80deb1fe3b1696b1),
  430. CURL_UINT64_C(0x9bdc06a725c71235), CURL_UINT64_C(0xc19bf174cf692694),
  431. CURL_UINT64_C(0xe49b69c19ef14ad2), CURL_UINT64_C(0xefbe4786384f25e3),
  432. CURL_UINT64_C(0x0fc19dc68b8cd5b5), CURL_UINT64_C(0x240ca1cc77ac9c65),
  433. CURL_UINT64_C(0x2de92c6f592b0275), CURL_UINT64_C(0x4a7484aa6ea6e483),
  434. CURL_UINT64_C(0x5cb0a9dcbd41fbd4), CURL_UINT64_C(0x76f988da831153b5),
  435. CURL_UINT64_C(0x983e5152ee66dfab), CURL_UINT64_C(0xa831c66d2db43210),
  436. CURL_UINT64_C(0xb00327c898fb213f), CURL_UINT64_C(0xbf597fc7beef0ee4),
  437. CURL_UINT64_C(0xc6e00bf33da88fc2), CURL_UINT64_C(0xd5a79147930aa725),
  438. CURL_UINT64_C(0x06ca6351e003826f), CURL_UINT64_C(0x142929670a0e6e70),
  439. CURL_UINT64_C(0x27b70a8546d22ffc), CURL_UINT64_C(0x2e1b21385c26c926),
  440. CURL_UINT64_C(0x4d2c6dfc5ac42aed), CURL_UINT64_C(0x53380d139d95b3df),
  441. CURL_UINT64_C(0x650a73548baf63de), CURL_UINT64_C(0x766a0abb3c77b2a8),
  442. CURL_UINT64_C(0x81c2c92e47edaee6), CURL_UINT64_C(0x92722c851482353b),
  443. CURL_UINT64_C(0xa2bfe8a14cf10364), CURL_UINT64_C(0xa81a664bbc423001),
  444. CURL_UINT64_C(0xc24b8b70d0f89791), CURL_UINT64_C(0xc76c51a30654be30),
  445. CURL_UINT64_C(0xd192e819d6ef5218), CURL_UINT64_C(0xd69906245565a910),
  446. CURL_UINT64_C(0xf40e35855771202a), CURL_UINT64_C(0x106aa07032bbd1b8),
  447. CURL_UINT64_C(0x19a4c116b8d2d0c8), CURL_UINT64_C(0x1e376c085141ab53),
  448. CURL_UINT64_C(0x2748774cdf8eeb99), CURL_UINT64_C(0x34b0bcb5e19b48a8),
  449. CURL_UINT64_C(0x391c0cb3c5c95a63), CURL_UINT64_C(0x4ed8aa4ae3418acb),
  450. CURL_UINT64_C(0x5b9cca4f7763e373), CURL_UINT64_C(0x682e6ff3d6b2b8a3),
  451. CURL_UINT64_C(0x748f82ee5defb2fc), CURL_UINT64_C(0x78a5636f43172f60),
  452. CURL_UINT64_C(0x84c87814a1f0ab72), CURL_UINT64_C(0x8cc702081a6439ec),
  453. CURL_UINT64_C(0x90befffa23631e28), CURL_UINT64_C(0xa4506cebde82bde9),
  454. CURL_UINT64_C(0xbef9a3f7b2c67915), CURL_UINT64_C(0xc67178f2e372532b),
  455. CURL_UINT64_C(0xca273eceea26619c), CURL_UINT64_C(0xd186b8c721c0c207),
  456. CURL_UINT64_C(0xeada7dd6cde0eb1e), CURL_UINT64_C(0xf57d4f7fee6ed178),
  457. CURL_UINT64_C(0x06f067aa72176fba), CURL_UINT64_C(0x0a637dc5a2c898a6),
  458. CURL_UINT64_C(0x113f9804bef90dae), CURL_UINT64_C(0x1b710b35131c471b),
  459. CURL_UINT64_C(0x28db77f523047d84), CURL_UINT64_C(0x32caab7b40c72493),
  460. CURL_UINT64_C(0x3c9ebe0a15c9bebc), CURL_UINT64_C(0x431d67c49c100d4c),
  461. CURL_UINT64_C(0x4cc5d4becb3e42b6), CURL_UINT64_C(0x597f299cfc657e2a),
  462. CURL_UINT64_C(0x5fcb6fab3ad6faec), CURL_UINT64_C(0x6c44198c4a475817)
  463. };
  464. /* One step of SHA-512/256 computation,
  465. see FIPS PUB 180-4 section 6.4.2 step 3.
  466. * Note: this macro updates working variables in-place, without rotation.
  467. * Note: the first (vH += SIG1(vE) + Ch(vE,vF,vG) + kt + wt) equals T1 in
  468. FIPS PUB 180-4 section 6.4.2 step 3.
  469. the second (vH += SIG0(vA) + Maj(vE,vF,vC) equals T1 + T2 in
  470. FIPS PUB 180-4 section 6.4.2 step 3.
  471. * Note: 'wt' must be used exactly one time in this macro as macro for
  472. 'wt' calculation may change other data as well every time when
  473. used. */
  474. #define SHA2STEP64(vA,vB,vC,vD,vE,vF,vG,vH,kt,wt) do { \
  475. (vD) += ((vH) += SIG1((vE)) + Sha512_Ch((vE),(vF),(vG)) + (kt) + (wt)); \
  476. (vH) += SIG0((vA)) + Sha512_Maj((vA),(vB),(vC)); } while (0)
  477. /* One step of SHA-512/256 computation with working variables rotation,
  478. see FIPS PUB 180-4 section 6.4.2 step 3. This macro version reassigns
  479. all working variables on each step. */
  480. #define SHA2STEP64RV(vA,vB,vC,vD,vE,vF,vG,vH,kt,wt) do { \
  481. curl_uint64_t tmp_h_ = (vH); \
  482. SHA2STEP64((vA),(vB),(vC),(vD),(vE),(vF),(vG),tmp_h_,(kt),(wt)); \
  483. (vH) = (vG); \
  484. (vG) = (vF); \
  485. (vF) = (vE); \
  486. (vE) = (vD); \
  487. (vD) = (vC); \
  488. (vC) = (vB); \
  489. (vB) = (vA); \
  490. (vA) = tmp_h_; } while(0)
  491. /* Get value of W(t) from input data buffer for 0 <= t <= 15,
  492. See FIPS PUB 180-4 section 6.2.
  493. Input data must be read in big-endian bytes order,
  494. see FIPS PUB 180-4 section 3.1.2. */
  495. #define SHA512_GET_W_FROM_DATA(buf,t) \
  496. MHDX_GET_64BIT_BE( \
  497. ((const unsigned char*) (buf)) + (t) * SHA512_256_BYTES_IN_WORD)
  498. /* During first 16 steps, before making any calculation on each step, the
  499. W element is read from the input data buffer as a big-endian value and
  500. stored in the array of W elements. */
  501. for(t = 0; t < 16; ++t) {
  502. SHA2STEP64RV(a, b, c, d, e, f, g, h, K[t], \
  503. W[t] = SHA512_GET_W_FROM_DATA(data, t));
  504. }
  505. /* 'W' generation and assignment for 16 <= t <= 79.
  506. See FIPS PUB 180-4 section 6.4.2.
  507. As only the last 16 'W' are used in calculations, it is possible to
  508. use 16 elements array of W as a cyclic buffer.
  509. Note: ((t-16) & 15) have same value as (t & 15) */
  510. #define Wgen(w,t) \
  511. (curl_uint64_t)( (w)[(t - 16) & 15] + sig1((w)[((t) - 2) & 15]) \
  512. + (w)[((t) - 7) & 15] + sig0((w)[((t) - 15) & 15]) )
  513. /* During the last 64 steps, before making any calculation on each step,
  514. current W element is generated from other W elements of the cyclic
  515. buffer and the generated value is stored back in the cyclic buffer. */
  516. for(t = 16; t < 80; ++t) {
  517. SHA2STEP64RV(a, b, c, d, e, f, g, h, K[t], \
  518. W[t & 15] = Wgen(W, t));
  519. }
  520. }
  521. /* Compute and store the intermediate hash.
  522. See FIPS PUB 180-4 section 6.4.2 step 4. */
  523. H[0] += a;
  524. H[1] += b;
  525. H[2] += c;
  526. H[3] += d;
  527. H[4] += e;
  528. H[5] += f;
  529. H[6] += g;
  530. H[7] += h;
  531. }
  532. /**
  533. * Process portion of bytes.
  534. *
  535. * @param context the calculation context
  536. * @param data bytes to add to hash
  537. * @param length number of bytes in @a data
  538. * @return always CURLE_OK
  539. */
  540. static CURLcode
  541. MHDx_sha512_256_update(void *context,
  542. const unsigned char *data,
  543. size_t length)
  544. {
  545. unsigned int bytes_have; /**< Number of bytes in the context buffer */
  546. struct mhdx_sha512_256ctx *const ctx = (struct mhdx_sha512_256ctx *)context;
  547. /* the void pointer here is required to mute Intel compiler warning */
  548. void *const ctx_buf = ctx->buffer;
  549. DEBUGASSERT((data != NULL) || (length == 0));
  550. if(0 == length)
  551. return CURLE_OK; /* Shortcut, do nothing */
  552. /* Note: (count & (SHA512_256_BLOCK_SIZE-1))
  553. equals (count % SHA512_256_BLOCK_SIZE) for this block size. */
  554. bytes_have = (unsigned int) (ctx->count & (SHA512_256_BLOCK_SIZE - 1));
  555. ctx->count += length;
  556. if(length > ctx->count)
  557. ctx->count_bits_hi += 1U << 3; /* Value wrap */
  558. ctx->count_bits_hi += ctx->count >> 61;
  559. ctx->count &= CURL_UINT64_C(0x1FFFFFFFFFFFFFFF);
  560. if(0 != bytes_have) {
  561. unsigned int bytes_left = SHA512_256_BLOCK_SIZE - bytes_have;
  562. if(length >= bytes_left) {
  563. /* Combine new data with data in the buffer and process the full
  564. block. */
  565. memcpy(((unsigned char *) ctx_buf) + bytes_have,
  566. data,
  567. bytes_left);
  568. data += bytes_left;
  569. length -= bytes_left;
  570. MHDx_sha512_256_transform(ctx->H, ctx->buffer);
  571. bytes_have = 0;
  572. }
  573. }
  574. while(SHA512_256_BLOCK_SIZE <= length) {
  575. /* Process any full blocks of new data directly,
  576. without copying to the buffer. */
  577. MHDx_sha512_256_transform(ctx->H, data);
  578. data += SHA512_256_BLOCK_SIZE;
  579. length -= SHA512_256_BLOCK_SIZE;
  580. }
  581. if(0 != length) {
  582. /* Copy incomplete block of new data (if any)
  583. to the buffer. */
  584. memcpy(((unsigned char *) ctx_buf) + bytes_have, data, length);
  585. }
  586. return CURLE_OK;
  587. }
  588. /**
  589. * Size of "length" insertion in bits.
  590. * See FIPS PUB 180-4 section 5.1.2.
  591. */
  592. #define SHA512_256_SIZE_OF_LEN_ADD_BITS 128
  593. /**
  594. * Size of "length" insertion in bytes.
  595. */
  596. #define SHA512_256_SIZE_OF_LEN_ADD (SHA512_256_SIZE_OF_LEN_ADD_BITS / 8)
  597. /**
  598. * Finalise SHA-512/256 calculation, return digest.
  599. *
  600. * @param context the calculation context
  601. * @param[out] digest set to the hash, must be #SHA512_256_DIGEST_SIZE bytes
  602. * @return always CURLE_OK
  603. */
  604. static CURLcode
  605. MHDx_sha512_256_finish(unsigned char *digest,
  606. void *context)
  607. {
  608. struct mhdx_sha512_256ctx *const ctx = (struct mhdx_sha512_256ctx *)context;
  609. curl_uint64_t num_bits; /**< Number of processed bits */
  610. unsigned int bytes_have; /**< Number of bytes in the context buffer */
  611. /* the void pointer here is required to mute Intel compiler warning */
  612. void *const ctx_buf = ctx->buffer;
  613. /* Memorise the number of processed bits.
  614. The padding and other data added here during the postprocessing must
  615. not change the amount of hashed data. */
  616. num_bits = ctx->count << 3;
  617. /* Note: (count & (SHA512_256_BLOCK_SIZE-1))
  618. equals (count % SHA512_256_BLOCK_SIZE) for this block size. */
  619. bytes_have = (unsigned int) (ctx->count & (SHA512_256_BLOCK_SIZE - 1));
  620. /* Input data must be padded with a single bit "1", then with zeros and
  621. the finally the length of data in bits must be added as the final bytes
  622. of the last block.
  623. See FIPS PUB 180-4 section 5.1.2. */
  624. /* Data is always processed in form of bytes (not by individual bits),
  625. therefore position of the first padding bit in byte is always
  626. predefined (0x80). */
  627. /* Buffer always have space at least for one byte (as full buffers are
  628. processed when formed). */
  629. ((unsigned char *) ctx_buf)[bytes_have++] = 0x80U;
  630. if(SHA512_256_BLOCK_SIZE - bytes_have < SHA512_256_SIZE_OF_LEN_ADD) {
  631. /* No space in the current block to put the total length of message.
  632. Pad the current block with zeros and process it. */
  633. if(bytes_have < SHA512_256_BLOCK_SIZE)
  634. memset(((unsigned char *) ctx_buf) + bytes_have, 0,
  635. SHA512_256_BLOCK_SIZE - bytes_have);
  636. /* Process the full block. */
  637. MHDx_sha512_256_transform(ctx->H, ctx->buffer);
  638. /* Start the new block. */
  639. bytes_have = 0;
  640. }
  641. /* Pad the rest of the buffer with zeros. */
  642. memset(((unsigned char *) ctx_buf) + bytes_have, 0,
  643. SHA512_256_BLOCK_SIZE - SHA512_256_SIZE_OF_LEN_ADD - bytes_have);
  644. /* Put high part of number of bits in processed message and then lower
  645. part of number of bits as big-endian values.
  646. See FIPS PUB 180-4 section 5.1.2. */
  647. /* Note: the target location is predefined and buffer is always aligned */
  648. MHDX_PUT_64BIT_BE(((unsigned char *) ctx_buf) \
  649. + SHA512_256_BLOCK_SIZE \
  650. - SHA512_256_SIZE_OF_LEN_ADD, \
  651. ctx->count_bits_hi);
  652. MHDX_PUT_64BIT_BE(((unsigned char *) ctx_buf) \
  653. + SHA512_256_BLOCK_SIZE \
  654. - SHA512_256_SIZE_OF_LEN_ADD \
  655. + SHA512_256_BYTES_IN_WORD, \
  656. num_bits);
  657. /* Process the full final block. */
  658. MHDx_sha512_256_transform(ctx->H, ctx->buffer);
  659. /* Put in BE mode the leftmost part of the hash as the final digest.
  660. See FIPS PUB 180-4 section 6.7. */
  661. MHDX_PUT_64BIT_BE((digest + 0 * SHA512_256_BYTES_IN_WORD), ctx->H[0]);
  662. MHDX_PUT_64BIT_BE((digest + 1 * SHA512_256_BYTES_IN_WORD), ctx->H[1]);
  663. MHDX_PUT_64BIT_BE((digest + 2 * SHA512_256_BYTES_IN_WORD), ctx->H[2]);
  664. MHDX_PUT_64BIT_BE((digest + 3 * SHA512_256_BYTES_IN_WORD), ctx->H[3]);
  665. /* Erase potentially sensitive data. */
  666. memset(ctx, 0, sizeof(struct mhdx_sha512_256ctx));
  667. return CURLE_OK;
  668. }
  669. /* Map to the local implementation */
  670. #define Curl_sha512_256_init MHDx_sha512_256_init
  671. #define Curl_sha512_256_update MHDx_sha512_256_update
  672. #define Curl_sha512_256_finish MHDx_sha512_256_finish
  673. #endif /* Local SHA-512/256 code */
  674. /**
  675. * Compute SHA-512/256 hash for the given data in one function call
  676. * @param[out] output the pointer to put the hash
  677. * @param[in] input the pointer to the data to process
  678. * @param input_size the size of the data pointed by @a input
  679. * @return always #CURLE_OK
  680. */
  681. CURLcode
  682. Curl_sha512_256it(unsigned char *output, const unsigned char *input,
  683. size_t input_size)
  684. {
  685. Curl_sha512_256_ctx ctx;
  686. CURLcode res;
  687. res = Curl_sha512_256_init(&ctx);
  688. if(res != CURLE_OK)
  689. return res;
  690. res = Curl_sha512_256_update(&ctx, (const void *) input, input_size);
  691. if(res != CURLE_OK) {
  692. (void) Curl_sha512_256_finish(output, &ctx);
  693. return res;
  694. }
  695. return Curl_sha512_256_finish(output, &ctx);
  696. }
  697. /* Wrapper function, takes 'unsigned int' as length type, returns void */
  698. static void
  699. Curl_sha512_256_update_i(void *context,
  700. const unsigned char *data,
  701. unsigned int length)
  702. {
  703. /* Hypothetically the function may fail, but assume it does not */
  704. (void) Curl_sha512_256_update(context, data, length);
  705. }
  706. /* Wrapper function, returns void */
  707. static void
  708. Curl_sha512_256_finish_v(unsigned char *result,
  709. void *context)
  710. {
  711. /* Hypothetically the function may fail, but assume it does not */
  712. (void) Curl_sha512_256_finish(result, context);
  713. }
  714. /* Wrapper function, takes 'unsigned int' as length type, returns void */
  715. const struct HMAC_params Curl_HMAC_SHA512_256[] = {
  716. {
  717. /* Initialize context procedure. */
  718. Curl_sha512_256_init,
  719. /* Update context with data. */
  720. Curl_sha512_256_update_i,
  721. /* Get final result procedure. */
  722. Curl_sha512_256_finish_v,
  723. /* Context structure size. */
  724. sizeof(Curl_sha512_256_ctx),
  725. /* Maximum key length (bytes). */
  726. SHA512_256_BLOCK_SIZE,
  727. /* Result length (bytes). */
  728. SHA512_256_DIGEST_SIZE
  729. }
  730. };
  731. #endif /* !CURL_DISABLE_DIGEST_AUTH && !CURL_DISABLE_SHA512_256 */