skeymgmt_meth.c 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. /*
  2. * Copyright 2025 The OpenSSL Project Authors. All Rights Reserved.
  3. *
  4. * Licensed under the Apache License 2.0 (the "License"). You may not use
  5. * this file except in compliance with the License. You can obtain a copy
  6. * in the file LICENSE in the source distribution or at
  7. * https://www.openssl.org/source/license.html
  8. */
  9. #include <openssl/crypto.h>
  10. #include <openssl/core_dispatch.h>
  11. #include <openssl/evp.h>
  12. #include <openssl/err.h>
  13. #include "internal/core.h"
  14. #include "internal/provider.h"
  15. #include "internal/refcount.h"
  16. #include "crypto/evp.h"
  17. #include "evp_local.h"
  18. void *evp_skeymgmt_generate(const EVP_SKEYMGMT *skeymgmt, const OSSL_PARAM params[])
  19. {
  20. void *provctx = ossl_provider_ctx(EVP_SKEYMGMT_get0_provider(skeymgmt));
  21. return (skeymgmt->generate != NULL) ? skeymgmt->generate(provctx, params) : NULL;
  22. }
  23. void *evp_skeymgmt_import(const EVP_SKEYMGMT *skeymgmt, int selection, const OSSL_PARAM params[])
  24. {
  25. void *provctx = ossl_provider_ctx(EVP_SKEYMGMT_get0_provider(skeymgmt));
  26. /* This is mandatory, no need to check for its presence */
  27. return skeymgmt->import(provctx, selection, params);
  28. }
  29. int evp_skeymgmt_export(const EVP_SKEYMGMT *skeymgmt, void *keydata,
  30. int selection, OSSL_CALLBACK *param_cb, void *cbarg)
  31. {
  32. /* This is mandatory, no need to check for its presence */
  33. return skeymgmt->export(keydata, selection, param_cb, cbarg);
  34. }
  35. void evp_skeymgmt_freedata(const EVP_SKEYMGMT *skeymgmt, void *keydata)
  36. {
  37. /* This is mandatory, no need to check for its presence */
  38. skeymgmt->free(keydata);
  39. }
  40. static void *skeymgmt_new(void)
  41. {
  42. EVP_SKEYMGMT *skeymgmt = NULL;
  43. if ((skeymgmt = OPENSSL_zalloc(sizeof(*skeymgmt))) == NULL)
  44. return NULL;
  45. if (!CRYPTO_NEW_REF(&skeymgmt->refcnt, 1)) {
  46. EVP_SKEYMGMT_free(skeymgmt);
  47. return NULL;
  48. }
  49. return skeymgmt;
  50. }
  51. static void *skeymgmt_from_algorithm(int name_id,
  52. const OSSL_ALGORITHM *algodef,
  53. OSSL_PROVIDER *prov)
  54. {
  55. const OSSL_DISPATCH *fns = algodef->implementation;
  56. EVP_SKEYMGMT *skeymgmt = NULL;
  57. if ((skeymgmt = skeymgmt_new()) == NULL)
  58. return NULL;
  59. skeymgmt->name_id = name_id;
  60. if ((skeymgmt->type_name = ossl_algorithm_get1_first_name(algodef)) == NULL) {
  61. EVP_SKEYMGMT_free(skeymgmt);
  62. return NULL;
  63. }
  64. skeymgmt->description = algodef->algorithm_description;
  65. for (; fns->function_id != 0; fns++) {
  66. switch (fns->function_id) {
  67. case OSSL_FUNC_SKEYMGMT_FREE:
  68. if (skeymgmt->free == NULL)
  69. skeymgmt->free = OSSL_FUNC_skeymgmt_free(fns);
  70. break;
  71. case OSSL_FUNC_SKEYMGMT_IMPORT:
  72. if (skeymgmt->import == NULL)
  73. skeymgmt->import = OSSL_FUNC_skeymgmt_import(fns);
  74. break;
  75. case OSSL_FUNC_SKEYMGMT_EXPORT:
  76. if (skeymgmt->export == NULL)
  77. skeymgmt->export = OSSL_FUNC_skeymgmt_export(fns);
  78. break;
  79. case OSSL_FUNC_SKEYMGMT_GENERATE:
  80. if (skeymgmt->generate == NULL)
  81. skeymgmt->generate = OSSL_FUNC_skeymgmt_generate(fns);
  82. break;
  83. case OSSL_FUNC_SKEYMGMT_GET_KEY_ID:
  84. if (skeymgmt->get_key_id == NULL)
  85. skeymgmt->get_key_id = OSSL_FUNC_skeymgmt_get_key_id(fns);
  86. break;
  87. case OSSL_FUNC_SKEYMGMT_IMP_SETTABLE_PARAMS:
  88. if (skeymgmt->imp_params == NULL)
  89. skeymgmt->imp_params = OSSL_FUNC_skeymgmt_imp_settable_params(fns);
  90. break;
  91. case OSSL_FUNC_SKEYMGMT_GEN_SETTABLE_PARAMS:
  92. if (skeymgmt->gen_params == NULL)
  93. skeymgmt->gen_params = OSSL_FUNC_skeymgmt_gen_settable_params(fns);
  94. break;
  95. }
  96. }
  97. /* Check that the provider is sensible */
  98. if (skeymgmt->free == NULL
  99. || skeymgmt->import == NULL
  100. || skeymgmt->export == NULL) {
  101. EVP_SKEYMGMT_free(skeymgmt);
  102. ERR_raise(ERR_LIB_EVP, EVP_R_INVALID_PROVIDER_FUNCTIONS);
  103. return NULL;
  104. }
  105. if (!ossl_provider_up_ref(prov)) {
  106. EVP_SKEYMGMT_free(skeymgmt);
  107. ERR_raise(ERR_LIB_EVP, EVP_R_INITIALIZATION_ERROR);
  108. return NULL;
  109. }
  110. skeymgmt->prov = prov;
  111. return skeymgmt;
  112. }
  113. EVP_SKEYMGMT *evp_skeymgmt_fetch_from_prov(OSSL_PROVIDER *prov,
  114. const char *name,
  115. const char *properties)
  116. {
  117. return evp_generic_fetch_from_prov(prov,
  118. OSSL_OP_SKEYMGMT,
  119. name, properties,
  120. skeymgmt_from_algorithm,
  121. (int (*)(void *))EVP_SKEYMGMT_up_ref,
  122. (void (*)(void *))EVP_SKEYMGMT_free);
  123. }
  124. EVP_SKEYMGMT *EVP_SKEYMGMT_fetch(OSSL_LIB_CTX *ctx, const char *algorithm,
  125. const char *properties)
  126. {
  127. return evp_generic_fetch(ctx, OSSL_OP_SKEYMGMT, algorithm, properties,
  128. skeymgmt_from_algorithm,
  129. (int (*)(void *))EVP_SKEYMGMT_up_ref,
  130. (void (*)(void *))EVP_SKEYMGMT_free);
  131. }
  132. int EVP_SKEYMGMT_up_ref(EVP_SKEYMGMT *skeymgmt)
  133. {
  134. int ref = 0;
  135. CRYPTO_UP_REF(&skeymgmt->refcnt, &ref);
  136. return 1;
  137. }
  138. void EVP_SKEYMGMT_free(EVP_SKEYMGMT *skeymgmt)
  139. {
  140. int ref = 0;
  141. if (skeymgmt == NULL)
  142. return;
  143. CRYPTO_DOWN_REF(&skeymgmt->refcnt, &ref);
  144. if (ref > 0)
  145. return;
  146. OPENSSL_free(skeymgmt->type_name);
  147. ossl_provider_free(skeymgmt->prov);
  148. CRYPTO_FREE_REF(&skeymgmt->refcnt);
  149. OPENSSL_free(skeymgmt);
  150. }
  151. const OSSL_PROVIDER *EVP_SKEYMGMT_get0_provider(const EVP_SKEYMGMT *skeymgmt)
  152. {
  153. return (skeymgmt != NULL) ? skeymgmt->prov : NULL;
  154. }
  155. const char *EVP_SKEYMGMT_get0_description(const EVP_SKEYMGMT *skeymgmt)
  156. {
  157. return (skeymgmt != NULL) ? skeymgmt->description : NULL;
  158. }
  159. const char *EVP_SKEYMGMT_get0_name(const EVP_SKEYMGMT *skeymgmt)
  160. {
  161. return (skeymgmt != NULL) ? skeymgmt->type_name : NULL;
  162. }
  163. int EVP_SKEYMGMT_is_a(const EVP_SKEYMGMT *skeymgmt, const char *name)
  164. {
  165. return skeymgmt != NULL
  166. && evp_is_a(skeymgmt->prov, skeymgmt->name_id, NULL, name);
  167. }
  168. void EVP_SKEYMGMT_do_all_provided(OSSL_LIB_CTX *libctx,
  169. void (*fn)(EVP_SKEYMGMT *skeymgmt, void *arg),
  170. void *arg)
  171. {
  172. evp_generic_do_all(libctx, OSSL_OP_SKEYMGMT,
  173. (void (*)(void *, void *))fn, arg,
  174. skeymgmt_from_algorithm,
  175. (int (*)(void *))EVP_SKEYMGMT_up_ref,
  176. (void (*)(void *))EVP_SKEYMGMT_free);
  177. }
  178. int EVP_SKEYMGMT_names_do_all(const EVP_SKEYMGMT *skeymgmt,
  179. void (*fn)(const char *name, void *data),
  180. void *data)
  181. {
  182. if (skeymgmt == NULL)
  183. return 0;
  184. if (skeymgmt->prov != NULL)
  185. return evp_names_do_all(skeymgmt->prov, skeymgmt->name_id, fn, data);
  186. return 1;
  187. }
  188. const OSSL_PARAM *EVP_SKEYMGMT_get0_gen_settable_params(const EVP_SKEYMGMT *skeymgmt)
  189. {
  190. void *provctx = NULL;
  191. if (skeymgmt == NULL)
  192. return 0;
  193. provctx = ossl_provider_ctx(EVP_SKEYMGMT_get0_provider(skeymgmt));
  194. return (skeymgmt->gen_params != NULL) ? skeymgmt->gen_params(provctx) : NULL;
  195. }
  196. const OSSL_PARAM *EVP_SKEYMGMT_get0_imp_settable_params(const EVP_SKEYMGMT *skeymgmt)
  197. {
  198. void *provctx = NULL;
  199. if (skeymgmt == NULL)
  200. return 0;
  201. provctx = ossl_provider_ctx(EVP_SKEYMGMT_get0_provider(skeymgmt));
  202. return (skeymgmt->imp_params != NULL) ? skeymgmt->imp_params(provctx) : NULL;
  203. }