1e1051a39Sopenharmony_ci/* 2e1051a39Sopenharmony_ci * Copyright 2019-2021 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 <assert.h> 11e1051a39Sopenharmony_ci#include <openssl/core_dispatch.h> 12e1051a39Sopenharmony_ci#include "internal/cryptlib.h" 13e1051a39Sopenharmony_ci#include "prov/bio.h" 14e1051a39Sopenharmony_ci 15e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_new_file_fn *c_bio_new_file = NULL; 16e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_new_membuf_fn *c_bio_new_membuf = NULL; 17e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_read_ex_fn *c_bio_read_ex = NULL; 18e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_write_ex_fn *c_bio_write_ex = NULL; 19e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_gets_fn *c_bio_gets = NULL; 20e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_puts_fn *c_bio_puts = NULL; 21e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_ctrl_fn *c_bio_ctrl = NULL; 22e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_up_ref_fn *c_bio_up_ref = NULL; 23e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_free_fn *c_bio_free = NULL; 24e1051a39Sopenharmony_cistatic OSSL_FUNC_BIO_vprintf_fn *c_bio_vprintf = NULL; 25e1051a39Sopenharmony_ci 26e1051a39Sopenharmony_ciint ossl_prov_bio_from_dispatch(const OSSL_DISPATCH *fns) 27e1051a39Sopenharmony_ci{ 28e1051a39Sopenharmony_ci for (; fns->function_id != 0; fns++) { 29e1051a39Sopenharmony_ci switch (fns->function_id) { 30e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_NEW_FILE: 31e1051a39Sopenharmony_ci if (c_bio_new_file == NULL) 32e1051a39Sopenharmony_ci c_bio_new_file = OSSL_FUNC_BIO_new_file(fns); 33e1051a39Sopenharmony_ci break; 34e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_NEW_MEMBUF: 35e1051a39Sopenharmony_ci if (c_bio_new_membuf == NULL) 36e1051a39Sopenharmony_ci c_bio_new_membuf = OSSL_FUNC_BIO_new_membuf(fns); 37e1051a39Sopenharmony_ci break; 38e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_READ_EX: 39e1051a39Sopenharmony_ci if (c_bio_read_ex == NULL) 40e1051a39Sopenharmony_ci c_bio_read_ex = OSSL_FUNC_BIO_read_ex(fns); 41e1051a39Sopenharmony_ci break; 42e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_WRITE_EX: 43e1051a39Sopenharmony_ci if (c_bio_write_ex == NULL) 44e1051a39Sopenharmony_ci c_bio_write_ex = OSSL_FUNC_BIO_write_ex(fns); 45e1051a39Sopenharmony_ci break; 46e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_GETS: 47e1051a39Sopenharmony_ci if (c_bio_gets == NULL) 48e1051a39Sopenharmony_ci c_bio_gets = OSSL_FUNC_BIO_gets(fns); 49e1051a39Sopenharmony_ci break; 50e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_PUTS: 51e1051a39Sopenharmony_ci if (c_bio_puts == NULL) 52e1051a39Sopenharmony_ci c_bio_puts = OSSL_FUNC_BIO_puts(fns); 53e1051a39Sopenharmony_ci break; 54e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_CTRL: 55e1051a39Sopenharmony_ci if (c_bio_ctrl == NULL) 56e1051a39Sopenharmony_ci c_bio_ctrl = OSSL_FUNC_BIO_ctrl(fns); 57e1051a39Sopenharmony_ci break; 58e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_UP_REF: 59e1051a39Sopenharmony_ci if (c_bio_up_ref == NULL) 60e1051a39Sopenharmony_ci c_bio_up_ref = OSSL_FUNC_BIO_up_ref(fns); 61e1051a39Sopenharmony_ci break; 62e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_FREE: 63e1051a39Sopenharmony_ci if (c_bio_free == NULL) 64e1051a39Sopenharmony_ci c_bio_free = OSSL_FUNC_BIO_free(fns); 65e1051a39Sopenharmony_ci break; 66e1051a39Sopenharmony_ci case OSSL_FUNC_BIO_VPRINTF: 67e1051a39Sopenharmony_ci if (c_bio_vprintf == NULL) 68e1051a39Sopenharmony_ci c_bio_vprintf = OSSL_FUNC_BIO_vprintf(fns); 69e1051a39Sopenharmony_ci break; 70e1051a39Sopenharmony_ci } 71e1051a39Sopenharmony_ci } 72e1051a39Sopenharmony_ci 73e1051a39Sopenharmony_ci return 1; 74e1051a39Sopenharmony_ci} 75e1051a39Sopenharmony_ci 76e1051a39Sopenharmony_ciOSSL_CORE_BIO *ossl_prov_bio_new_file(const char *filename, const char *mode) 77e1051a39Sopenharmony_ci{ 78e1051a39Sopenharmony_ci if (c_bio_new_file == NULL) 79e1051a39Sopenharmony_ci return NULL; 80e1051a39Sopenharmony_ci return c_bio_new_file(filename, mode); 81e1051a39Sopenharmony_ci} 82e1051a39Sopenharmony_ci 83e1051a39Sopenharmony_ciOSSL_CORE_BIO *ossl_prov_bio_new_membuf(const char *filename, int len) 84e1051a39Sopenharmony_ci{ 85e1051a39Sopenharmony_ci if (c_bio_new_membuf == NULL) 86e1051a39Sopenharmony_ci return NULL; 87e1051a39Sopenharmony_ci return c_bio_new_membuf(filename, len); 88e1051a39Sopenharmony_ci} 89e1051a39Sopenharmony_ci 90e1051a39Sopenharmony_ciint ossl_prov_bio_read_ex(OSSL_CORE_BIO *bio, void *data, size_t data_len, 91e1051a39Sopenharmony_ci size_t *bytes_read) 92e1051a39Sopenharmony_ci{ 93e1051a39Sopenharmony_ci if (c_bio_read_ex == NULL) 94e1051a39Sopenharmony_ci return 0; 95e1051a39Sopenharmony_ci return c_bio_read_ex(bio, data, data_len, bytes_read); 96e1051a39Sopenharmony_ci} 97e1051a39Sopenharmony_ci 98e1051a39Sopenharmony_ciint ossl_prov_bio_write_ex(OSSL_CORE_BIO *bio, const void *data, size_t data_len, 99e1051a39Sopenharmony_ci size_t *written) 100e1051a39Sopenharmony_ci{ 101e1051a39Sopenharmony_ci if (c_bio_write_ex == NULL) 102e1051a39Sopenharmony_ci return 0; 103e1051a39Sopenharmony_ci return c_bio_write_ex(bio, data, data_len, written); 104e1051a39Sopenharmony_ci} 105e1051a39Sopenharmony_ci 106e1051a39Sopenharmony_ciint ossl_prov_bio_gets(OSSL_CORE_BIO *bio, char *buf, int size) 107e1051a39Sopenharmony_ci{ 108e1051a39Sopenharmony_ci if (c_bio_gets == NULL) 109e1051a39Sopenharmony_ci return -1; 110e1051a39Sopenharmony_ci return c_bio_gets(bio, buf, size); 111e1051a39Sopenharmony_ci} 112e1051a39Sopenharmony_ci 113e1051a39Sopenharmony_ciint ossl_prov_bio_puts(OSSL_CORE_BIO *bio, const char *str) 114e1051a39Sopenharmony_ci{ 115e1051a39Sopenharmony_ci if (c_bio_puts == NULL) 116e1051a39Sopenharmony_ci return -1; 117e1051a39Sopenharmony_ci return c_bio_puts(bio, str); 118e1051a39Sopenharmony_ci} 119e1051a39Sopenharmony_ci 120e1051a39Sopenharmony_ciint ossl_prov_bio_ctrl(OSSL_CORE_BIO *bio, int cmd, long num, void *ptr) 121e1051a39Sopenharmony_ci{ 122e1051a39Sopenharmony_ci if (c_bio_ctrl == NULL) 123e1051a39Sopenharmony_ci return -1; 124e1051a39Sopenharmony_ci return c_bio_ctrl(bio, cmd, num, ptr); 125e1051a39Sopenharmony_ci} 126e1051a39Sopenharmony_ci 127e1051a39Sopenharmony_ciint ossl_prov_bio_up_ref(OSSL_CORE_BIO *bio) 128e1051a39Sopenharmony_ci{ 129e1051a39Sopenharmony_ci if (c_bio_up_ref == NULL) 130e1051a39Sopenharmony_ci return 0; 131e1051a39Sopenharmony_ci return c_bio_up_ref(bio); 132e1051a39Sopenharmony_ci} 133e1051a39Sopenharmony_ci 134e1051a39Sopenharmony_ciint ossl_prov_bio_free(OSSL_CORE_BIO *bio) 135e1051a39Sopenharmony_ci{ 136e1051a39Sopenharmony_ci if (c_bio_free == NULL) 137e1051a39Sopenharmony_ci return 0; 138e1051a39Sopenharmony_ci return c_bio_free(bio); 139e1051a39Sopenharmony_ci} 140e1051a39Sopenharmony_ci 141e1051a39Sopenharmony_ciint ossl_prov_bio_vprintf(OSSL_CORE_BIO *bio, const char *format, va_list ap) 142e1051a39Sopenharmony_ci{ 143e1051a39Sopenharmony_ci if (c_bio_vprintf == NULL) 144e1051a39Sopenharmony_ci return -1; 145e1051a39Sopenharmony_ci return c_bio_vprintf(bio, format, ap); 146e1051a39Sopenharmony_ci} 147e1051a39Sopenharmony_ci 148e1051a39Sopenharmony_ciint ossl_prov_bio_printf(OSSL_CORE_BIO *bio, const char *format, ...) 149e1051a39Sopenharmony_ci{ 150e1051a39Sopenharmony_ci va_list ap; 151e1051a39Sopenharmony_ci int ret; 152e1051a39Sopenharmony_ci 153e1051a39Sopenharmony_ci va_start(ap, format); 154e1051a39Sopenharmony_ci ret = ossl_prov_bio_vprintf(bio, format, ap); 155e1051a39Sopenharmony_ci va_end(ap); 156e1051a39Sopenharmony_ci 157e1051a39Sopenharmony_ci return ret; 158e1051a39Sopenharmony_ci} 159e1051a39Sopenharmony_ci 160e1051a39Sopenharmony_ci#ifndef FIPS_MODULE 161e1051a39Sopenharmony_ci 162e1051a39Sopenharmony_ci/* No direct BIO support in the FIPS module */ 163e1051a39Sopenharmony_ci 164e1051a39Sopenharmony_cistatic int bio_core_read_ex(BIO *bio, char *data, size_t data_len, 165e1051a39Sopenharmony_ci size_t *bytes_read) 166e1051a39Sopenharmony_ci{ 167e1051a39Sopenharmony_ci return ossl_prov_bio_read_ex(BIO_get_data(bio), data, data_len, bytes_read); 168e1051a39Sopenharmony_ci} 169e1051a39Sopenharmony_ci 170e1051a39Sopenharmony_cistatic int bio_core_write_ex(BIO *bio, const char *data, size_t data_len, 171e1051a39Sopenharmony_ci size_t *written) 172e1051a39Sopenharmony_ci{ 173e1051a39Sopenharmony_ci return ossl_prov_bio_write_ex(BIO_get_data(bio), data, data_len, written); 174e1051a39Sopenharmony_ci} 175e1051a39Sopenharmony_ci 176e1051a39Sopenharmony_cistatic long bio_core_ctrl(BIO *bio, int cmd, long num, void *ptr) 177e1051a39Sopenharmony_ci{ 178e1051a39Sopenharmony_ci return ossl_prov_bio_ctrl(BIO_get_data(bio), cmd, num, ptr); 179e1051a39Sopenharmony_ci} 180e1051a39Sopenharmony_ci 181e1051a39Sopenharmony_cistatic int bio_core_gets(BIO *bio, char *buf, int size) 182e1051a39Sopenharmony_ci{ 183e1051a39Sopenharmony_ci return ossl_prov_bio_gets(BIO_get_data(bio), buf, size); 184e1051a39Sopenharmony_ci} 185e1051a39Sopenharmony_ci 186e1051a39Sopenharmony_cistatic int bio_core_puts(BIO *bio, const char *str) 187e1051a39Sopenharmony_ci{ 188e1051a39Sopenharmony_ci return ossl_prov_bio_puts(BIO_get_data(bio), str); 189e1051a39Sopenharmony_ci} 190e1051a39Sopenharmony_ci 191e1051a39Sopenharmony_cistatic int bio_core_new(BIO *bio) 192e1051a39Sopenharmony_ci{ 193e1051a39Sopenharmony_ci BIO_set_init(bio, 1); 194e1051a39Sopenharmony_ci 195e1051a39Sopenharmony_ci return 1; 196e1051a39Sopenharmony_ci} 197e1051a39Sopenharmony_ci 198e1051a39Sopenharmony_cistatic int bio_core_free(BIO *bio) 199e1051a39Sopenharmony_ci{ 200e1051a39Sopenharmony_ci BIO_set_init(bio, 0); 201e1051a39Sopenharmony_ci ossl_prov_bio_free(BIO_get_data(bio)); 202e1051a39Sopenharmony_ci 203e1051a39Sopenharmony_ci return 1; 204e1051a39Sopenharmony_ci} 205e1051a39Sopenharmony_ci 206e1051a39Sopenharmony_ciBIO_METHOD *ossl_bio_prov_init_bio_method(void) 207e1051a39Sopenharmony_ci{ 208e1051a39Sopenharmony_ci BIO_METHOD *corebiometh = NULL; 209e1051a39Sopenharmony_ci 210e1051a39Sopenharmony_ci corebiometh = BIO_meth_new(BIO_TYPE_CORE_TO_PROV, "BIO to Core filter"); 211e1051a39Sopenharmony_ci if (corebiometh == NULL 212e1051a39Sopenharmony_ci || !BIO_meth_set_write_ex(corebiometh, bio_core_write_ex) 213e1051a39Sopenharmony_ci || !BIO_meth_set_read_ex(corebiometh, bio_core_read_ex) 214e1051a39Sopenharmony_ci || !BIO_meth_set_puts(corebiometh, bio_core_puts) 215e1051a39Sopenharmony_ci || !BIO_meth_set_gets(corebiometh, bio_core_gets) 216e1051a39Sopenharmony_ci || !BIO_meth_set_ctrl(corebiometh, bio_core_ctrl) 217e1051a39Sopenharmony_ci || !BIO_meth_set_create(corebiometh, bio_core_new) 218e1051a39Sopenharmony_ci || !BIO_meth_set_destroy(corebiometh, bio_core_free)) { 219e1051a39Sopenharmony_ci BIO_meth_free(corebiometh); 220e1051a39Sopenharmony_ci return NULL; 221e1051a39Sopenharmony_ci } 222e1051a39Sopenharmony_ci 223e1051a39Sopenharmony_ci return corebiometh; 224e1051a39Sopenharmony_ci} 225e1051a39Sopenharmony_ci 226e1051a39Sopenharmony_ciBIO *ossl_bio_new_from_core_bio(PROV_CTX *provctx, OSSL_CORE_BIO *corebio) 227e1051a39Sopenharmony_ci{ 228e1051a39Sopenharmony_ci BIO *outbio; 229e1051a39Sopenharmony_ci BIO_METHOD *corebiometh = ossl_prov_ctx_get0_core_bio_method(provctx); 230e1051a39Sopenharmony_ci 231e1051a39Sopenharmony_ci if (corebiometh == NULL) 232e1051a39Sopenharmony_ci return NULL; 233e1051a39Sopenharmony_ci 234e1051a39Sopenharmony_ci if ((outbio = BIO_new(corebiometh)) == NULL) 235e1051a39Sopenharmony_ci return NULL; 236e1051a39Sopenharmony_ci if (!ossl_prov_bio_up_ref(corebio)) { 237e1051a39Sopenharmony_ci BIO_free(outbio); 238e1051a39Sopenharmony_ci return NULL; 239e1051a39Sopenharmony_ci } 240e1051a39Sopenharmony_ci BIO_set_data(outbio, corebio); 241e1051a39Sopenharmony_ci return outbio; 242e1051a39Sopenharmony_ci} 243e1051a39Sopenharmony_ci 244e1051a39Sopenharmony_ci#endif 245