curl_sha512_256.c 29 KB

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