mbedtls_crypto.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. /*
  2. * Copyright (c) 2015-2023, Arm Limited and Contributors. All rights reserved.
  3. *
  4. * SPDX-License-Identifier: BSD-3-Clause
  5. */
  6. #include <assert.h>
  7. #include <stddef.h>
  8. #include <string.h>
  9. /* mbed TLS headers */
  10. #include <mbedtls/gcm.h>
  11. #include <mbedtls/md.h>
  12. #include <mbedtls/memory_buffer_alloc.h>
  13. #include <mbedtls/oid.h>
  14. #include <mbedtls/platform.h>
  15. #include <mbedtls/version.h>
  16. #include <mbedtls/x509.h>
  17. #include <common/debug.h>
  18. #include <drivers/auth/crypto_mod.h>
  19. #include <drivers/auth/mbedtls/mbedtls_common.h>
  20. #include <plat/common/platform.h>
  21. #define LIB_NAME "mbed TLS"
  22. #if CRYPTO_SUPPORT == CRYPTO_HASH_CALC_ONLY || \
  23. CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC
  24. /*
  25. * CRYPTO_MD_MAX_SIZE value is as per current stronger algorithm available
  26. * so make sure that mbed TLS MD maximum size must be lesser than this.
  27. */
  28. CASSERT(CRYPTO_MD_MAX_SIZE >= MBEDTLS_MD_MAX_SIZE,
  29. assert_mbedtls_md_size_overflow);
  30. #endif /* CRYPTO_SUPPORT == CRYPTO_HASH_CALC_ONLY || \
  31. CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC */
  32. /*
  33. * AlgorithmIdentifier ::= SEQUENCE {
  34. * algorithm OBJECT IDENTIFIER,
  35. * parameters ANY DEFINED BY algorithm OPTIONAL
  36. * }
  37. *
  38. * SubjectPublicKeyInfo ::= SEQUENCE {
  39. * algorithm AlgorithmIdentifier,
  40. * subjectPublicKey BIT STRING
  41. * }
  42. *
  43. * DigestInfo ::= SEQUENCE {
  44. * digestAlgorithm AlgorithmIdentifier,
  45. * digest OCTET STRING
  46. * }
  47. */
  48. /*
  49. * Initialize the library and export the descriptor
  50. */
  51. static void init(void)
  52. {
  53. /* Initialize mbed TLS */
  54. mbedtls_init();
  55. }
  56. #if CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_ONLY || \
  57. CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC
  58. /*
  59. * Verify a signature.
  60. *
  61. * Parameters are passed using the DER encoding format following the ASN.1
  62. * structures detailed above.
  63. */
  64. static int verify_signature(void *data_ptr, unsigned int data_len,
  65. void *sig_ptr, unsigned int sig_len,
  66. void *sig_alg, unsigned int sig_alg_len,
  67. void *pk_ptr, unsigned int pk_len)
  68. {
  69. mbedtls_asn1_buf sig_oid, sig_params;
  70. mbedtls_asn1_buf signature;
  71. mbedtls_md_type_t md_alg;
  72. mbedtls_pk_type_t pk_alg;
  73. mbedtls_pk_context pk = {0};
  74. int rc;
  75. void *sig_opts = NULL;
  76. const mbedtls_md_info_t *md_info;
  77. unsigned char *p, *end;
  78. unsigned char hash[MBEDTLS_MD_MAX_SIZE];
  79. /* Get pointers to signature OID and parameters */
  80. p = (unsigned char *)sig_alg;
  81. end = (unsigned char *)(p + sig_alg_len);
  82. rc = mbedtls_asn1_get_alg(&p, end, &sig_oid, &sig_params);
  83. if (rc != 0) {
  84. return CRYPTO_ERR_SIGNATURE;
  85. }
  86. /* Get the actual signature algorithm (MD + PK) */
  87. rc = mbedtls_x509_get_sig_alg(&sig_oid, &sig_params, &md_alg, &pk_alg, &sig_opts);
  88. if (rc != 0) {
  89. return CRYPTO_ERR_SIGNATURE;
  90. }
  91. /* Parse the public key */
  92. mbedtls_pk_init(&pk);
  93. p = (unsigned char *)pk_ptr;
  94. end = (unsigned char *)(p + pk_len);
  95. rc = mbedtls_pk_parse_subpubkey(&p, end, &pk);
  96. if (rc != 0) {
  97. rc = CRYPTO_ERR_SIGNATURE;
  98. goto end2;
  99. }
  100. /* Get the signature (bitstring) */
  101. p = (unsigned char *)sig_ptr;
  102. end = (unsigned char *)(p + sig_len);
  103. signature.tag = *p;
  104. rc = mbedtls_asn1_get_bitstring_null(&p, end, &signature.len);
  105. if ((rc != 0) || ((size_t)(end - p) != signature.len)) {
  106. rc = CRYPTO_ERR_SIGNATURE;
  107. goto end1;
  108. }
  109. signature.p = p;
  110. /* Calculate the hash of the data */
  111. md_info = mbedtls_md_info_from_type(md_alg);
  112. if (md_info == NULL) {
  113. rc = CRYPTO_ERR_SIGNATURE;
  114. goto end1;
  115. }
  116. p = (unsigned char *)data_ptr;
  117. rc = mbedtls_md(md_info, p, data_len, hash);
  118. if (rc != 0) {
  119. rc = CRYPTO_ERR_SIGNATURE;
  120. goto end1;
  121. }
  122. /* Verify the signature */
  123. rc = mbedtls_pk_verify_ext(pk_alg, sig_opts, &pk, md_alg, hash,
  124. mbedtls_md_get_size(md_info),
  125. signature.p, signature.len);
  126. if (rc != 0) {
  127. rc = CRYPTO_ERR_SIGNATURE;
  128. goto end1;
  129. }
  130. /* Signature verification success */
  131. rc = CRYPTO_SUCCESS;
  132. end1:
  133. mbedtls_pk_free(&pk);
  134. end2:
  135. mbedtls_free(sig_opts);
  136. return rc;
  137. }
  138. /*
  139. * Match a hash
  140. *
  141. * Digest info is passed in DER format following the ASN.1 structure detailed
  142. * above.
  143. */
  144. static int verify_hash(void *data_ptr, unsigned int data_len,
  145. void *digest_info_ptr, unsigned int digest_info_len)
  146. {
  147. mbedtls_asn1_buf hash_oid, params;
  148. mbedtls_md_type_t md_alg;
  149. const mbedtls_md_info_t *md_info;
  150. unsigned char *p, *end, *hash;
  151. unsigned char data_hash[MBEDTLS_MD_MAX_SIZE];
  152. size_t len;
  153. int rc;
  154. /*
  155. * Digest info should be an MBEDTLS_ASN1_SEQUENCE
  156. * and consume all bytes.
  157. */
  158. p = (unsigned char *)digest_info_ptr;
  159. end = p + digest_info_len;
  160. rc = mbedtls_asn1_get_tag(&p, end, &len, MBEDTLS_ASN1_CONSTRUCTED |
  161. MBEDTLS_ASN1_SEQUENCE);
  162. if (rc != 0 || ((size_t)(end - p) != len)) {
  163. return CRYPTO_ERR_HASH;
  164. }
  165. /* Get the hash algorithm */
  166. rc = mbedtls_asn1_get_alg(&p, end, &hash_oid, &params);
  167. if (rc != 0) {
  168. return CRYPTO_ERR_HASH;
  169. }
  170. rc = mbedtls_oid_get_md_alg(&hash_oid, &md_alg);
  171. if (rc != 0) {
  172. return CRYPTO_ERR_HASH;
  173. }
  174. md_info = mbedtls_md_info_from_type(md_alg);
  175. if (md_info == NULL) {
  176. return CRYPTO_ERR_HASH;
  177. }
  178. /* Hash should be octet string type and consume all bytes */
  179. rc = mbedtls_asn1_get_tag(&p, end, &len, MBEDTLS_ASN1_OCTET_STRING);
  180. if ((rc != 0) || ((size_t)(end - p) != len)) {
  181. return CRYPTO_ERR_HASH;
  182. }
  183. /* Length of hash must match the algorithm's size */
  184. if (len != mbedtls_md_get_size(md_info)) {
  185. return CRYPTO_ERR_HASH;
  186. }
  187. hash = p;
  188. /* Calculate the hash of the data */
  189. p = (unsigned char *)data_ptr;
  190. rc = mbedtls_md(md_info, p, data_len, data_hash);
  191. if (rc != 0) {
  192. return CRYPTO_ERR_HASH;
  193. }
  194. /* Compare values */
  195. rc = memcmp(data_hash, hash, mbedtls_md_get_size(md_info));
  196. if (rc != 0) {
  197. return CRYPTO_ERR_HASH;
  198. }
  199. return CRYPTO_SUCCESS;
  200. }
  201. #endif /* CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_ONLY || \
  202. CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC */
  203. #if CRYPTO_SUPPORT == CRYPTO_HASH_CALC_ONLY || \
  204. CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC
  205. /*
  206. * Map a generic crypto message digest algorithm to the corresponding macro used
  207. * by Mbed TLS.
  208. */
  209. static inline mbedtls_md_type_t md_type(enum crypto_md_algo algo)
  210. {
  211. switch (algo) {
  212. case CRYPTO_MD_SHA512:
  213. return MBEDTLS_MD_SHA512;
  214. case CRYPTO_MD_SHA384:
  215. return MBEDTLS_MD_SHA384;
  216. case CRYPTO_MD_SHA256:
  217. return MBEDTLS_MD_SHA256;
  218. default:
  219. /* Invalid hash algorithm. */
  220. return MBEDTLS_MD_NONE;
  221. }
  222. }
  223. /*
  224. * Calculate a hash
  225. *
  226. * output points to the computed hash
  227. */
  228. static int calc_hash(enum crypto_md_algo md_algo, void *data_ptr,
  229. unsigned int data_len,
  230. unsigned char output[CRYPTO_MD_MAX_SIZE])
  231. {
  232. const mbedtls_md_info_t *md_info;
  233. md_info = mbedtls_md_info_from_type(md_type(md_algo));
  234. if (md_info == NULL) {
  235. return CRYPTO_ERR_HASH;
  236. }
  237. /*
  238. * Calculate the hash of the data, it is safe to pass the
  239. * 'output' hash buffer pointer considering its size is always
  240. * bigger than or equal to MBEDTLS_MD_MAX_SIZE.
  241. */
  242. return mbedtls_md(md_info, data_ptr, data_len, output);
  243. }
  244. #endif /* CRYPTO_SUPPORT == CRYPTO_HASH_CALC_ONLY || \
  245. CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC */
  246. #if TF_MBEDTLS_USE_AES_GCM
  247. /*
  248. * Stack based buffer allocation for decryption operation. It could
  249. * be configured to balance stack usage vs execution speed.
  250. */
  251. #define DEC_OP_BUF_SIZE 128
  252. static int aes_gcm_decrypt(void *data_ptr, size_t len, const void *key,
  253. unsigned int key_len, const void *iv,
  254. unsigned int iv_len, const void *tag,
  255. unsigned int tag_len)
  256. {
  257. mbedtls_gcm_context ctx;
  258. mbedtls_cipher_id_t cipher = MBEDTLS_CIPHER_ID_AES;
  259. unsigned char buf[DEC_OP_BUF_SIZE];
  260. unsigned char tag_buf[CRYPTO_MAX_TAG_SIZE];
  261. unsigned char *pt = data_ptr;
  262. size_t dec_len;
  263. int diff, i, rc;
  264. mbedtls_gcm_init(&ctx);
  265. rc = mbedtls_gcm_setkey(&ctx, cipher, key, key_len * 8);
  266. if (rc != 0) {
  267. rc = CRYPTO_ERR_DECRYPTION;
  268. goto exit_gcm;
  269. }
  270. rc = mbedtls_gcm_starts(&ctx, MBEDTLS_GCM_DECRYPT, iv, iv_len, NULL, 0);
  271. if (rc != 0) {
  272. rc = CRYPTO_ERR_DECRYPTION;
  273. goto exit_gcm;
  274. }
  275. while (len > 0) {
  276. dec_len = MIN(sizeof(buf), len);
  277. rc = mbedtls_gcm_update(&ctx, dec_len, pt, buf);
  278. if (rc != 0) {
  279. rc = CRYPTO_ERR_DECRYPTION;
  280. goto exit_gcm;
  281. }
  282. memcpy(pt, buf, dec_len);
  283. pt += dec_len;
  284. len -= dec_len;
  285. }
  286. rc = mbedtls_gcm_finish(&ctx, tag_buf, sizeof(tag_buf));
  287. if (rc != 0) {
  288. rc = CRYPTO_ERR_DECRYPTION;
  289. goto exit_gcm;
  290. }
  291. /* Check tag in "constant-time" */
  292. for (diff = 0, i = 0; i < tag_len; i++)
  293. diff |= ((const unsigned char *)tag)[i] ^ tag_buf[i];
  294. if (diff != 0) {
  295. rc = CRYPTO_ERR_DECRYPTION;
  296. goto exit_gcm;
  297. }
  298. /* GCM decryption success */
  299. rc = CRYPTO_SUCCESS;
  300. exit_gcm:
  301. mbedtls_gcm_free(&ctx);
  302. return rc;
  303. }
  304. /*
  305. * Authenticated decryption of an image
  306. */
  307. static int auth_decrypt(enum crypto_dec_algo dec_algo, void *data_ptr,
  308. size_t len, const void *key, unsigned int key_len,
  309. unsigned int key_flags, const void *iv,
  310. unsigned int iv_len, const void *tag,
  311. unsigned int tag_len)
  312. {
  313. int rc;
  314. assert((key_flags & ENC_KEY_IS_IDENTIFIER) == 0);
  315. switch (dec_algo) {
  316. case CRYPTO_GCM_DECRYPT:
  317. rc = aes_gcm_decrypt(data_ptr, len, key, key_len, iv, iv_len,
  318. tag, tag_len);
  319. if (rc != 0)
  320. return rc;
  321. break;
  322. default:
  323. return CRYPTO_ERR_DECRYPTION;
  324. }
  325. return CRYPTO_SUCCESS;
  326. }
  327. #endif /* TF_MBEDTLS_USE_AES_GCM */
  328. /*
  329. * Register crypto library descriptor
  330. */
  331. #if CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC
  332. #if TF_MBEDTLS_USE_AES_GCM
  333. REGISTER_CRYPTO_LIB(LIB_NAME, init, verify_signature, verify_hash, calc_hash,
  334. auth_decrypt);
  335. #else
  336. REGISTER_CRYPTO_LIB(LIB_NAME, init, verify_signature, verify_hash, calc_hash,
  337. NULL);
  338. #endif
  339. #elif CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_ONLY
  340. #if TF_MBEDTLS_USE_AES_GCM
  341. REGISTER_CRYPTO_LIB(LIB_NAME, init, verify_signature, verify_hash,
  342. auth_decrypt);
  343. #else
  344. REGISTER_CRYPTO_LIB(LIB_NAME, init, verify_signature, verify_hash, NULL);
  345. #endif
  346. #elif CRYPTO_SUPPORT == CRYPTO_HASH_CALC_ONLY
  347. REGISTER_CRYPTO_LIB(LIB_NAME, init, calc_hash);
  348. #endif /* CRYPTO_SUPPORT == CRYPTO_AUTH_VERIFY_AND_HASH_CALC */