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