| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344 |
- /*
- * Copyright 2020-2024 The OpenSSL Project Authors. All Rights Reserved.
- *
- * Licensed under the Apache License 2.0 (the "License"). You may not use
- * this file except in compliance with the License. You can obtain a copy
- * in the file LICENSE in the source distribution or at
- * https://www.openssl.org/source/license.html
- */
- /*
- * This file uses the low level AES functions (which are deprecated for
- * non-internal use) in order to implement provider AES ciphers.
- */
- #include "internal/deprecated.h"
- #include <openssl/core.h>
- #include <openssl/core_names.h>
- #include <openssl/params.h>
- #include <openssl/proverr.h>
- #include "prov/providercommon.h"
- #include "prov/ciphercommon.h"
- #include "prov/ciphercommon_aead.h"
- #include "testutil.h"
- #include "fake_pipelineprov.h"
- /*
- * This file provides a fake provider that implements a pipeline cipher
- * for AES GCM.
- */
- typedef struct fake_pipeline_ctx_st {
- size_t keylen;
- size_t ivlen;
- size_t numpipes;
- EVP_CIPHER *cipher;
- EVP_CIPHER_CTX *cipher_ctxs[EVP_MAX_PIPES];
- } CIPHER_PIPELINE_CTX;
- static void *fake_pipeline_newctx(void *provctx, char *ciphername,
- size_t kbits, size_t ivbits)
- {
- CIPHER_PIPELINE_CTX *ctx;
- if (!ossl_prov_is_running())
- return NULL;
- ctx = OPENSSL_zalloc(sizeof(*ctx));
- if (ctx == NULL)
- return NULL;
- ctx->keylen = kbits / 8;
- ctx->ivlen = ivbits / 8;
- ctx->numpipes = 0;
- ctx->cipher = EVP_CIPHER_fetch(provctx, ciphername, "provider=default");
- return ctx;
- }
- static OSSL_FUNC_cipher_freectx_fn fake_pipeline_freectx;
- static void fake_pipeline_freectx(void *vctx)
- {
- CIPHER_PIPELINE_CTX *ctx = (CIPHER_PIPELINE_CTX *)vctx;
- size_t i;
- EVP_CIPHER_free(ctx->cipher);
- for (i = 0; i < ctx->numpipes; i++)
- EVP_CIPHER_CTX_free(ctx->cipher_ctxs[i]);
- OPENSSL_clear_free(ctx, sizeof(*ctx));
- }
- OSSL_FUNC_cipher_pipeline_encrypt_init_fn fake_pipeline_einit;
- OSSL_FUNC_cipher_pipeline_decrypt_init_fn fake_pipeline_dinit;
- OSSL_FUNC_cipher_pipeline_update_fn fake_pipeline_update;
- OSSL_FUNC_cipher_pipeline_final_fn fake_pipeline_final;
- OSSL_FUNC_cipher_gettable_ctx_params_fn fake_pipeline_aead_gettable_ctx_params;
- OSSL_FUNC_cipher_get_ctx_params_fn fake_pipeline_aead_get_ctx_params;
- OSSL_FUNC_cipher_settable_ctx_params_fn fake_pipeline_aead_settable_ctx_params;
- OSSL_FUNC_cipher_set_ctx_params_fn fake_pipeline_aead_set_ctx_params;
- static int fake_pipeline_init(void *vctx,
- const unsigned char *key, size_t keylen,
- size_t numpipes, const unsigned char **iv,
- size_t ivlen, int enc)
- {
- CIPHER_PIPELINE_CTX *ctx = (CIPHER_PIPELINE_CTX *)vctx;
- size_t i = 0;
- ctx->numpipes = numpipes;
- for (i = 0; i < numpipes; i++) {
- ctx->cipher_ctxs[i] = EVP_CIPHER_CTX_new();
- if (ctx->cipher_ctxs[i] == NULL)
- return 0;
- if (!EVP_CipherInit(ctx->cipher_ctxs[i], ctx->cipher, key, iv[i], enc))
- return 0;
- }
- return 1;
- }
- int fake_pipeline_einit(void *vctx,
- const unsigned char *key, size_t keylen,
- size_t numpipes, const unsigned char **iv,
- size_t ivlen, const OSSL_PARAM params[])
- {
- return fake_pipeline_init(vctx, key, keylen, numpipes, iv, ivlen, 1);
- }
- int fake_pipeline_dinit(void *vctx,
- const unsigned char *key, size_t keylen,
- size_t numpipes, const unsigned char **iv,
- size_t ivlen, const OSSL_PARAM params[])
- {
- return fake_pipeline_init(vctx, key, keylen, numpipes, iv, ivlen, 0);
- }
- int fake_pipeline_update(void *vctx, size_t numpipes,
- unsigned char **out, size_t *outl,
- const size_t *outsize,
- const unsigned char **in, const size_t *inl)
- {
- CIPHER_PIPELINE_CTX *ctx = (CIPHER_PIPELINE_CTX *)vctx;
- int ioutl, inl_;
- size_t i = 0;
- for (i = 0; i < numpipes; i++) {
- inl_ = (int)inl[i];
- if (!EVP_CipherUpdate(ctx->cipher_ctxs[i],
- (out != NULL) ? out[i] : NULL,
- &ioutl,
- in[i], inl_))
- return 0;
- outl[i] = (size_t)ioutl;
- }
- return 1;
- }
- int fake_pipeline_final(void *vctx, size_t numpipes,
- unsigned char **out, size_t *outl,
- const size_t *outsize)
- {
- CIPHER_PIPELINE_CTX *ctx = (CIPHER_PIPELINE_CTX *)vctx;
- int ioutl;
- size_t i = 0;
- for (i = 0; i < numpipes; i++) {
- if (!EVP_CipherFinal(ctx->cipher_ctxs[i], out[i], &ioutl))
- return 0;
- outl[i] = (size_t)ioutl;
- }
- return 1;
- }
- static const OSSL_PARAM fake_pipeline_aead_known_gettable_ctx_params[] = {
- OSSL_PARAM_size_t(OSSL_CIPHER_PARAM_KEYLEN, NULL),
- OSSL_PARAM_size_t(OSSL_CIPHER_PARAM_IVLEN, NULL),
- OSSL_PARAM_size_t(OSSL_CIPHER_PARAM_AEAD_TAGLEN, NULL),
- OSSL_PARAM_octet_ptr(OSSL_CIPHER_PARAM_PIPELINE_AEAD_TAG, NULL, 0),
- OSSL_PARAM_END
- };
- const OSSL_PARAM *fake_pipeline_aead_gettable_ctx_params(ossl_unused void *cctx,
- ossl_unused void *provctx)
- {
- return fake_pipeline_aead_known_gettable_ctx_params;
- }
- static const OSSL_PARAM fake_pipeline_aead_known_settable_ctx_params[] = {
- OSSL_PARAM_octet_ptr(OSSL_CIPHER_PARAM_PIPELINE_AEAD_TAG, NULL, 0),
- OSSL_PARAM_END
- };
- const OSSL_PARAM *fake_pipeline_aead_settable_ctx_params(ossl_unused void *cctx,
- ossl_unused void *provctx)
- {
- return fake_pipeline_aead_known_settable_ctx_params;
- }
- int fake_pipeline_aead_get_ctx_params(void *vctx, OSSL_PARAM params[])
- {
- CIPHER_PIPELINE_CTX *ctx = (CIPHER_PIPELINE_CTX *)vctx;
- OSSL_PARAM *p;
- size_t taglen, i;
- unsigned char **aead_tags = NULL;
- OSSL_PARAM aead_params[2] = { OSSL_PARAM_END, OSSL_PARAM_END };
- if (ossl_param_is_empty(params))
- return 1;
- p = OSSL_PARAM_locate(params, OSSL_CIPHER_PARAM_IVLEN);
- if (p != NULL) {
- if (!OSSL_PARAM_set_size_t(p, ctx->ivlen)) {
- ERR_raise(ERR_LIB_PROV, PROV_R_FAILED_TO_SET_PARAMETER);
- return 0;
- }
- }
- p = OSSL_PARAM_locate(params, OSSL_CIPHER_PARAM_KEYLEN);
- if (p != NULL) {
- if (!OSSL_PARAM_set_size_t(p, ctx->keylen)) {
- ERR_raise(ERR_LIB_PROV, PROV_R_FAILED_TO_SET_PARAMETER);
- return 0;
- }
- }
- p = OSSL_PARAM_locate(params, OSSL_CIPHER_PARAM_PIPELINE_AEAD_TAG);
- if (p != NULL) {
- if (!OSSL_PARAM_get_octet_ptr(p, (const void **)&aead_tags, &taglen)) {
- ERR_raise(ERR_LIB_PROV, PROV_R_FAILED_TO_SET_PARAMETER);
- return 0;
- }
- for (i = 0; i < ctx->numpipes; i++) {
- aead_params[0] = OSSL_PARAM_construct_octet_string(OSSL_CIPHER_PARAM_AEAD_TAG,
- (void *)aead_tags[i],
- taglen);
- if (!EVP_CIPHER_CTX_get_params(ctx->cipher_ctxs[i], aead_params)) {
- ERR_raise(ERR_LIB_PROV, PROV_R_FAILED_TO_SET_PARAMETER);
- return 0;
- }
- }
- }
- return 1;
- }
- int fake_pipeline_aead_set_ctx_params(void *vctx, const OSSL_PARAM params[])
- {
- CIPHER_PIPELINE_CTX *ctx = (CIPHER_PIPELINE_CTX *)vctx;
- const OSSL_PARAM *p;
- size_t taglen, i;
- unsigned char **aead_tags = NULL;
- OSSL_PARAM aead_params[2] = { OSSL_PARAM_END, OSSL_PARAM_END };
- p = OSSL_PARAM_locate_const(params, OSSL_CIPHER_PARAM_PIPELINE_AEAD_TAG);
- if (p != NULL) {
- if (!OSSL_PARAM_get_octet_ptr(p, (const void **)&aead_tags, &taglen)) {
- ERR_raise(ERR_LIB_PROV, PROV_R_FAILED_TO_SET_PARAMETER);
- return 0;
- }
- for (i = 0; i < ctx->numpipes; i++) {
- aead_params[0] = OSSL_PARAM_construct_octet_string(OSSL_CIPHER_PARAM_AEAD_TAG,
- (void *)aead_tags[i],
- taglen);
- if (!EVP_CIPHER_CTX_set_params(ctx->cipher_ctxs[i], aead_params)) {
- ERR_raise(ERR_LIB_PROV, PROV_R_FAILED_TO_SET_PARAMETER);
- return 0;
- }
- }
- }
- /* No other settable ctx param */
- return 1;
- }
- #define IMPLEMENT_aead_cipher_pipeline(alg, lc, UCMODE, flags, kbits, blkbits, \
- ivbits, ciphername) \
- static OSSL_FUNC_cipher_get_params_fn alg##_##kbits##_##lc##_get_params; \
- static int alg##_##kbits##_##lc##_get_params(OSSL_PARAM params[]) \
- { \
- return ossl_cipher_generic_get_params(params, EVP_CIPH_##UCMODE##_MODE, \
- flags, kbits, blkbits, ivbits); \
- } \
- static OSSL_FUNC_cipher_newctx_fn fake_pipeline_##alg##_##kbits##_##lc##_newctx; \
- static void * fake_pipeline_##alg##_##kbits##_##lc##_newctx(void *provctx) \
- { \
- return fake_pipeline_newctx(provctx, ciphername, kbits, ivbits); \
- } \
- static const OSSL_DISPATCH fake_pipeline_##alg##kbits##lc##_functions[] = { \
- { OSSL_FUNC_CIPHER_NEWCTX, \
- (void (*)(void))fake_pipeline_##alg##_##kbits##_##lc##_newctx }, \
- { OSSL_FUNC_CIPHER_FREECTX, \
- (void (*)(void))fake_pipeline_freectx }, \
- { OSSL_FUNC_CIPHER_PIPELINE_ENCRYPT_INIT, \
- (void (*)(void))fake_pipeline_einit }, \
- { OSSL_FUNC_CIPHER_PIPELINE_DECRYPT_INIT, \
- (void (*)(void))fake_pipeline_dinit }, \
- { OSSL_FUNC_CIPHER_PIPELINE_UPDATE, \
- (void (*)(void))fake_pipeline_update }, \
- { OSSL_FUNC_CIPHER_PIPELINE_FINAL, \
- (void (*)(void))fake_pipeline_final }, \
- { OSSL_FUNC_CIPHER_GET_PARAMS, \
- (void (*)(void)) alg##_##kbits##_##lc##_get_params }, \
- { OSSL_FUNC_CIPHER_GET_CTX_PARAMS, \
- (void (*)(void)) fake_pipeline_aead_get_ctx_params }, \
- { OSSL_FUNC_CIPHER_SET_CTX_PARAMS, \
- (void (*)(void)) fake_pipeline_aead_set_ctx_params }, \
- { OSSL_FUNC_CIPHER_GETTABLE_PARAMS, \
- (void (*)(void)) ossl_cipher_generic_gettable_params }, \
- { OSSL_FUNC_CIPHER_GETTABLE_CTX_PARAMS, \
- (void (*)(void)) fake_pipeline_aead_gettable_ctx_params }, \
- { OSSL_FUNC_CIPHER_SETTABLE_CTX_PARAMS, \
- (void (*)(void)) fake_pipeline_aead_settable_ctx_params }, \
- OSSL_DISPATCH_END \
- }
- IMPLEMENT_aead_cipher_pipeline(aes, gcm, GCM, AEAD_FLAGS, 256, 8, 96, "AES-256-GCM");
- static const OSSL_ALGORITHM fake_ciphers[] = {
- {"AES-256-GCM", "provider=fake-pipeline", fake_pipeline_aes256gcm_functions},
- {NULL, NULL, NULL}
- };
- static const OSSL_ALGORITHM *fake_pipeline_query(OSSL_PROVIDER *prov,
- int operation_id,
- int *no_cache)
- {
- *no_cache = 0;
- switch (operation_id) {
- case OSSL_OP_CIPHER:
- return fake_ciphers;
- }
- return NULL;
- }
- /* Functions we provide to the core */
- static const OSSL_DISPATCH fake_pipeline_method[] = {
- { OSSL_FUNC_PROVIDER_TEARDOWN, (void (*)(void))OSSL_LIB_CTX_free },
- { OSSL_FUNC_PROVIDER_QUERY_OPERATION, (void (*)(void))fake_pipeline_query },
- OSSL_DISPATCH_END
- };
- static int fake_pipeline_provider_init(const OSSL_CORE_HANDLE *handle,
- const OSSL_DISPATCH *in,
- const OSSL_DISPATCH **out, void **provctx)
- {
- if (!TEST_ptr(*provctx = OSSL_LIB_CTX_new()))
- return 0;
- *out = fake_pipeline_method;
- return 1;
- }
- OSSL_PROVIDER *fake_pipeline_start(OSSL_LIB_CTX *libctx)
- {
- OSSL_PROVIDER *p;
- if (!TEST_true(OSSL_PROVIDER_add_builtin(libctx, "fake-pipeline",
- fake_pipeline_provider_init))
- || !TEST_ptr(p = OSSL_PROVIDER_try_load(libctx, "fake-pipeline", 1)))
- return NULL;
- return p;
- }
- void fake_pipeline_finish(OSSL_PROVIDER *p)
- {
- OSSL_PROVIDER_unload(p);
- }
|