1e1051a39Sopenharmony_ci/* 2e1051a39Sopenharmony_ci * Copyright 2019-2022 The OpenSSL Project Authors. All Rights Reserved. 3e1051a39Sopenharmony_ci * 4e1051a39Sopenharmony_ci * Licensed under the Apache License 2.0 (the "License"). You may not use 5e1051a39Sopenharmony_ci * this file except in compliance with the License. You can obtain a copy 6e1051a39Sopenharmony_ci * in the file LICENSE in the source distribution or at 7e1051a39Sopenharmony_ci * https://www.openssl.org/source/license.html 8e1051a39Sopenharmony_ci */ 9e1051a39Sopenharmony_ci 10e1051a39Sopenharmony_ci#include <openssl/core_dispatch.h> 11e1051a39Sopenharmony_ci#include "bio_local.h" 12e1051a39Sopenharmony_ci#include "internal/cryptlib.h" 13e1051a39Sopenharmony_ci 14e1051a39Sopenharmony_citypedef struct { 15e1051a39Sopenharmony_ci OSSL_FUNC_BIO_read_ex_fn *c_bio_read_ex; 16e1051a39Sopenharmony_ci OSSL_FUNC_BIO_write_ex_fn *c_bio_write_ex; 17e1051a39Sopenharmony_ci OSSL_FUNC_BIO_gets_fn *c_bio_gets; 18e1051a39Sopenharmony_ci OSSL_FUNC_BIO_puts_fn *c_bio_puts; 19e1051a39Sopenharmony_ci OSSL_FUNC_BIO_ctrl_fn *c_bio_ctrl; 20e1051a39Sopenharmony_ci OSSL_FUNC_BIO_up_ref_fn *c_bio_up_ref; 21e1051a39Sopenharmony_ci OSSL_FUNC_BIO_free_fn *c_bio_free; 22e1051a39Sopenharmony_ci} BIO_CORE_GLOBALS; 23e1051a39Sopenharmony_ci 24e1051a39Sopenharmony_cistatic void bio_core_globals_free(void *vbcg) 25e1051a39Sopenharmony_ci{ 26e1051a39Sopenharmony_ci OPENSSL_free(vbcg); 27e1051a39Sopenharmony_ci} 28e1051a39Sopenharmony_ci 29e1051a39Sopenharmony_cistatic void *bio_core_globals_new(OSSL_LIB_CTX *ctx) 30e1051a39Sopenharmony_ci{ 31e1051a39Sopenharmony_ci return OPENSSL_zalloc(sizeof(BIO_CORE_GLOBALS)); 32e1051a39Sopenharmony_ci} 33e1051a39Sopenharmony_ci 34e1051a39Sopenharmony_cistatic const OSSL_LIB_CTX_METHOD bio_core_globals_method = { 35e1051a39Sopenharmony_ci OSSL_LIB_CTX_METHOD_DEFAULT_PRIORITY, 36e1051a39Sopenharmony_ci bio_core_globals_new, 37e1051a39Sopenharmony_ci bio_core_globals_free, 38e1051a39Sopenharmony_ci}; 39e1051a39Sopenharmony_ci 40e1051a39Sopenharmony_cistatic ossl_inline BIO_CORE_GLOBALS *get_globals(OSSL_LIB_CTX *libctx) 41e1051a39Sopenharmony_ci{ 42e1051a39Sopenharmony_ci return ossl_lib_ctx_get_data(libctx, OSSL_LIB_CTX_BIO_CORE_INDEX, 43e1051a39Sopenharmony_ci &bio_core_globals_method); 44e1051a39Sopenharmony_ci} 45e1051a39Sopenharmony_ci 46e1051a39Sopenharmony_cistatic int bio_core_read_ex(BIO *bio, char *data, size_t data_len, 47e1051a39Sopenharmony_ci size_t *bytes_read) 48e1051a39Sopenharmony_ci{ 49e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(bio->libctx); 50e1051a39Sopenharmony_ci 51e1051a39Sopenharmony_ci if (bcgbl == NULL || bcgbl->c_bio_read_ex == NULL) 52e1051a39Sopenharmony_ci return 0; 53e1051a39Sopenharmony_ci return bcgbl->c_bio_read_ex(BIO_get_data(bio), data, data_len, bytes_read); 54e1051a39Sopenharmony_ci} 55e1051a39Sopenharmony_ci 56e1051a39Sopenharmony_cistatic int bio_core_write_ex(BIO *bio, const char *data, size_t data_len, 57e1051a39Sopenharmony_ci size_t *written) 58e1051a39Sopenharmony_ci{ 59e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(bio->libctx); 60e1051a39Sopenharmony_ci 61e1051a39Sopenharmony_ci if (bcgbl == NULL || bcgbl->c_bio_write_ex == NULL) 62e1051a39Sopenharmony_ci return 0; 63e1051a39Sopenharmony_ci return bcgbl->c_bio_write_ex(BIO_get_data(bio), data, data_len, written); 64e1051a39Sopenharmony_ci} 65e1051a39Sopenharmony_ci 66e1051a39Sopenharmony_cistatic long bio_core_ctrl(BIO *bio, int cmd, long num, void *ptr) 67e1051a39Sopenharmony_ci{ 68e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(bio->libctx); 69e1051a39Sopenharmony_ci 70e1051a39Sopenharmony_ci if (bcgbl == NULL || bcgbl->c_bio_ctrl == NULL) 71e1051a39Sopenharmony_ci return -1; 72e1051a39Sopenharmony_ci return bcgbl->c_bio_ctrl(BIO_get_data(bio), cmd, num, ptr); 73e1051a39Sopenharmony_ci} 74e1051a39Sopenharmony_ci 75e1051a39Sopenharmony_cistatic int bio_core_gets(BIO *bio, char *buf, int size) 76e1051a39Sopenharmony_ci{ 77e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(bio->libctx); 78e1051a39Sopenharmony_ci 79e1051a39Sopenharmony_ci if (bcgbl == NULL || bcgbl->c_bio_gets == NULL) 80e1051a39Sopenharmony_ci return -1; 81e1051a39Sopenharmony_ci return bcgbl->c_bio_gets(BIO_get_data(bio), buf, size); 82e1051a39Sopenharmony_ci} 83e1051a39Sopenharmony_ci 84e1051a39Sopenharmony_cistatic int bio_core_puts(BIO *bio, const char *str) 85e1051a39Sopenharmony_ci{ 86e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(bio->libctx); 87e1051a39Sopenharmony_ci 88e1051a39Sopenharmony_ci if (bcgbl == NULL || bcgbl->c_bio_puts == NULL) 89e1051a39Sopenharmony_ci return -1; 90e1051a39Sopenharmony_ci return bcgbl->c_bio_puts(BIO_get_data(bio), str); 91e1051a39Sopenharmony_ci} 92e1051a39Sopenharmony_ci 93e1051a39Sopenharmony_cistatic int bio_core_new(BIO *bio) 94e1051a39Sopenharmony_ci{ 95e1051a39Sopenharmony_ci BIO_set_init(bio, 1); 96e1051a39Sopenharmony_ci 97e1051a39Sopenharmony_ci return 1; 98e1051a39Sopenharmony_ci} 99e1051a39Sopenharmony_ci 100e1051a39Sopenharmony_cistatic int bio_core_free(BIO *bio) 101e1051a39Sopenharmony_ci{ 102e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(bio->libctx); 103e1051a39Sopenharmony_ci 104e1051a39Sopenharmony_ci if (bcgbl == NULL) 105e1051a39Sopenharmony_ci return 0; 106e1051a39Sopenharmony_ci 107e1051a39Sopenharmony_ci BIO_set_init(bio, 0); 108e1051a39Sopenharmony_ci bcgbl->c_bio_free(BIO_get_data(bio)); 109e1051a39Sopenharmony_ci 110e1051a39Sopenharmony_ci return 1; 111e1051a39Sopenharmony_ci} 112e1051a39Sopenharmony_ci 113e1051a39Sopenharmony_cistatic const BIO_METHOD corebiometh = { 114e1051a39Sopenharmony_ci BIO_TYPE_CORE_TO_PROV, 115e1051a39Sopenharmony_ci "BIO to Core filter", 116e1051a39Sopenharmony_ci bio_core_write_ex, 117e1051a39Sopenharmony_ci NULL, 118e1051a39Sopenharmony_ci bio_core_read_ex, 119e1051a39Sopenharmony_ci NULL, 120e1051a39Sopenharmony_ci bio_core_puts, 121e1051a39Sopenharmony_ci bio_core_gets, 122e1051a39Sopenharmony_ci bio_core_ctrl, 123e1051a39Sopenharmony_ci bio_core_new, 124e1051a39Sopenharmony_ci bio_core_free, 125e1051a39Sopenharmony_ci NULL, 126e1051a39Sopenharmony_ci}; 127e1051a39Sopenharmony_ci 128e1051a39Sopenharmony_ciconst BIO_METHOD *BIO_s_core(void) 129e1051a39Sopenharmony_ci{ 130e1051a39Sopenharmony_ci return &corebiometh; 131e1051a39Sopenharmony_ci} 132e1051a39Sopenharmony_ci 133e1051a39Sopenharmony_ciBIO *BIO_new_from_core_bio(OSSL_LIB_CTX *libctx, OSSL_CORE_BIO *corebio) 134e1051a39Sopenharmony_ci{ 135e1051a39Sopenharmony_ci BIO *outbio; 136e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(libctx); 137e1051a39Sopenharmony_ci 138e1051a39Sopenharmony_ci /* Check the library context has been initialised with the callbacks */ 139e1051a39Sopenharmony_ci if (bcgbl == NULL || (bcgbl->c_bio_write_ex == NULL && bcgbl->c_bio_read_ex == NULL)) 140e1051a39Sopenharmony_ci return NULL; 141e1051a39Sopenharmony_ci 142e1051a39Sopenharmony_ci if ((outbio = BIO_new_ex(libctx, BIO_s_core())) == NULL) 143e1051a39Sopenharmony_ci return NULL; 144e1051a39Sopenharmony_ci 145e1051a39Sopenharmony_ci if (!bcgbl->c_bio_up_ref(corebio)) { 146e1051a39Sopenharmony_ci BIO_free(outbio); 147e1051a39Sopenharmony_ci return NULL; 148e1051a39Sopenharmony_ci } 149e1051a39Sopenharmony_ci BIO_set_data(outbio, corebio); 150e1051a39Sopenharmony_ci return outbio; 151e1051a39Sopenharmony_ci} 152e1051a39Sopenharmony_ci 153e1051a39Sopenharmony_ciint ossl_bio_init_core(OSSL_LIB_CTX *libctx, const OSSL_DISPATCH *fns) 154e1051a39Sopenharmony_ci{ 155e1051a39Sopenharmony_ci BIO_CORE_GLOBALS *bcgbl = get_globals(libctx); 156e1051a39Sopenharmony_ci 157e1051a39Sopenharmony_ci if (bcgbl == NULL) 158e1051a39Sopenharmony_ci return 0; 159e1051a39Sopenharmony_ci 160e1051a39Sopenharmony_ci for (; fns->function_id != 0; fns++) { 161e1051a39Sopenharmony_ci switch (fns->function_id) { 162e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_READ_EX: 163e1051a39Sopenharmony_ci if (bcgbl->c_bio_read_ex == NULL) 164e1051a39Sopenharmony_ci bcgbl->c_bio_read_ex = OSSL_FUNC_BIO_read_ex(fns); 165e1051a39Sopenharmony_ci break; 166e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_WRITE_EX: 167e1051a39Sopenharmony_ci if (bcgbl->c_bio_write_ex == NULL) 168e1051a39Sopenharmony_ci bcgbl->c_bio_write_ex = OSSL_FUNC_BIO_write_ex(fns); 169e1051a39Sopenharmony_ci break; 170e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_GETS: 171e1051a39Sopenharmony_ci if (bcgbl->c_bio_gets == NULL) 172e1051a39Sopenharmony_ci bcgbl->c_bio_gets = OSSL_FUNC_BIO_gets(fns); 173e1051a39Sopenharmony_ci break; 174e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_PUTS: 175e1051a39Sopenharmony_ci if (bcgbl->c_bio_puts == NULL) 176e1051a39Sopenharmony_ci bcgbl->c_bio_puts = OSSL_FUNC_BIO_puts(fns); 177e1051a39Sopenharmony_ci break; 178e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_CTRL: 179e1051a39Sopenharmony_ci if (bcgbl->c_bio_ctrl == NULL) 180e1051a39Sopenharmony_ci bcgbl->c_bio_ctrl = OSSL_FUNC_BIO_ctrl(fns); 181e1051a39Sopenharmony_ci break; 182e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_UP_REF: 183e1051a39Sopenharmony_ci if (bcgbl->c_bio_up_ref == NULL) 184e1051a39Sopenharmony_ci bcgbl->c_bio_up_ref = OSSL_FUNC_BIO_up_ref(fns); 185e1051a39Sopenharmony_ci break; 186e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_FREE: 187e1051a39Sopenharmony_ci if (bcgbl->c_bio_free == NULL) 188e1051a39Sopenharmony_ci bcgbl->c_bio_free = OSSL_FUNC_BIO_free(fns); 189e1051a39Sopenharmony_ci break; 190e1051a39Sopenharmony_ci } 191e1051a39Sopenharmony_ci } 192e1051a39Sopenharmony_ci 193e1051a39Sopenharmony_ci return 1; 194e1051a39Sopenharmony_ci} 195