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