xref: /freebsd/crypto/openssl/crypto/bio/bio_meth.c (revision b077aed33b7b6aefca7b17ddb250cf521f938613)
1 /*
2  * Copyright 2016-2020 The OpenSSL Project Authors. All Rights Reserved.
3  *
4  * Licensed under the Apache License 2.0 (the "License").  You may not use
5  * this file except in compliance with the License.  You can obtain a copy
6  * in the file LICENSE in the source distribution or at
7  * https://www.openssl.org/source/license.html
8  */
9 
10 #include "bio_local.h"
11 #include "internal/thread_once.h"
12 
13 CRYPTO_RWLOCK *bio_type_lock = NULL;
14 static CRYPTO_ONCE bio_type_init = CRYPTO_ONCE_STATIC_INIT;
15 
DEFINE_RUN_ONCE_STATIC(do_bio_type_init)16 DEFINE_RUN_ONCE_STATIC(do_bio_type_init)
17 {
18     bio_type_lock = CRYPTO_THREAD_lock_new();
19     return bio_type_lock != NULL;
20 }
21 
BIO_get_new_index(void)22 int BIO_get_new_index(void)
23 {
24     static CRYPTO_REF_COUNT bio_count = BIO_TYPE_START;
25     int newval;
26 
27     if (!RUN_ONCE(&bio_type_init, do_bio_type_init)) {
28         ERR_raise(ERR_LIB_BIO, ERR_R_MALLOC_FAILURE);
29         return -1;
30     }
31     if (!CRYPTO_UP_REF(&bio_count, &newval, bio_type_lock))
32         return -1;
33     return newval;
34 }
35 
BIO_meth_new(int type,const char * name)36 BIO_METHOD *BIO_meth_new(int type, const char *name)
37 {
38     BIO_METHOD *biom = OPENSSL_zalloc(sizeof(BIO_METHOD));
39 
40     if (biom == NULL
41             || (biom->name = OPENSSL_strdup(name)) == NULL) {
42         OPENSSL_free(biom);
43         ERR_raise(ERR_LIB_BIO, ERR_R_MALLOC_FAILURE);
44         return NULL;
45     }
46     biom->type = type;
47     return biom;
48 }
49 
BIO_meth_free(BIO_METHOD * biom)50 void BIO_meth_free(BIO_METHOD *biom)
51 {
52     if (biom != NULL) {
53         OPENSSL_free(biom->name);
54         OPENSSL_free(biom);
55     }
56 }
57 
BIO_meth_get_write(const BIO_METHOD * biom)58 int (*BIO_meth_get_write(const BIO_METHOD *biom)) (BIO *, const char *, int)
59 {
60     return biom->bwrite_old;
61 }
62 
BIO_meth_get_write_ex(const BIO_METHOD * biom)63 int (*BIO_meth_get_write_ex(const BIO_METHOD *biom)) (BIO *, const char *, size_t,
64                                                 size_t *)
65 {
66     return biom->bwrite;
67 }
68 
69 /* Conversion for old style bwrite to new style */
bwrite_conv(BIO * bio,const char * data,size_t datal,size_t * written)70 int bwrite_conv(BIO *bio, const char *data, size_t datal, size_t *written)
71 {
72     int ret;
73 
74     if (datal > INT_MAX)
75         datal = INT_MAX;
76 
77     ret = bio->method->bwrite_old(bio, data, (int)datal);
78 
79     if (ret <= 0) {
80         *written = 0;
81         return ret;
82     }
83 
84     *written = (size_t)ret;
85 
86     return 1;
87 }
88 
BIO_meth_set_write(BIO_METHOD * biom,int (* bwrite)(BIO *,const char *,int))89 int BIO_meth_set_write(BIO_METHOD *biom,
90                        int (*bwrite) (BIO *, const char *, int))
91 {
92     biom->bwrite_old = bwrite;
93     biom->bwrite = bwrite_conv;
94     return 1;
95 }
96 
BIO_meth_set_write_ex(BIO_METHOD * biom,int (* bwrite)(BIO *,const char *,size_t,size_t *))97 int BIO_meth_set_write_ex(BIO_METHOD *biom,
98                        int (*bwrite) (BIO *, const char *, size_t, size_t *))
99 {
100     biom->bwrite_old = NULL;
101     biom->bwrite = bwrite;
102     return 1;
103 }
104 
BIO_meth_get_read(const BIO_METHOD * biom)105 int (*BIO_meth_get_read(const BIO_METHOD *biom)) (BIO *, char *, int)
106 {
107     return biom->bread_old;
108 }
109 
BIO_meth_get_read_ex(const BIO_METHOD * biom)110 int (*BIO_meth_get_read_ex(const BIO_METHOD *biom)) (BIO *, char *, size_t, size_t *)
111 {
112     return biom->bread;
113 }
114 
115 /* Conversion for old style bread to new style */
bread_conv(BIO * bio,char * data,size_t datal,size_t * readbytes)116 int bread_conv(BIO *bio, char *data, size_t datal, size_t *readbytes)
117 {
118     int ret;
119 
120     if (datal > INT_MAX)
121         datal = INT_MAX;
122 
123     ret = bio->method->bread_old(bio, data, (int)datal);
124 
125     if (ret <= 0) {
126         *readbytes = 0;
127         return ret;
128     }
129 
130     *readbytes = (size_t)ret;
131 
132     return 1;
133 }
134 
BIO_meth_set_read(BIO_METHOD * biom,int (* bread)(BIO *,char *,int))135 int BIO_meth_set_read(BIO_METHOD *biom,
136                       int (*bread) (BIO *, char *, int))
137 {
138     biom->bread_old = bread;
139     biom->bread = bread_conv;
140     return 1;
141 }
142 
BIO_meth_set_read_ex(BIO_METHOD * biom,int (* bread)(BIO *,char *,size_t,size_t *))143 int BIO_meth_set_read_ex(BIO_METHOD *biom,
144                          int (*bread) (BIO *, char *, size_t, size_t *))
145 {
146     biom->bread_old = NULL;
147     biom->bread = bread;
148     return 1;
149 }
150 
BIO_meth_get_puts(const BIO_METHOD * biom)151 int (*BIO_meth_get_puts(const BIO_METHOD *biom)) (BIO *, const char *)
152 {
153     return biom->bputs;
154 }
155 
BIO_meth_set_puts(BIO_METHOD * biom,int (* bputs)(BIO *,const char *))156 int BIO_meth_set_puts(BIO_METHOD *biom,
157                       int (*bputs) (BIO *, const char *))
158 {
159     biom->bputs = bputs;
160     return 1;
161 }
162 
BIO_meth_get_gets(const BIO_METHOD * biom)163 int (*BIO_meth_get_gets(const BIO_METHOD *biom)) (BIO *, char *, int)
164 {
165     return biom->bgets;
166 }
167 
BIO_meth_set_gets(BIO_METHOD * biom,int (* bgets)(BIO *,char *,int))168 int BIO_meth_set_gets(BIO_METHOD *biom,
169                       int (*bgets) (BIO *, char *, int))
170 {
171     biom->bgets = bgets;
172     return 1;
173 }
174 
BIO_meth_get_ctrl(const BIO_METHOD * biom)175 long (*BIO_meth_get_ctrl(const BIO_METHOD *biom)) (BIO *, int, long, void *)
176 {
177     return biom->ctrl;
178 }
179 
BIO_meth_set_ctrl(BIO_METHOD * biom,long (* ctrl)(BIO *,int,long,void *))180 int BIO_meth_set_ctrl(BIO_METHOD *biom,
181                       long (*ctrl) (BIO *, int, long, void *))
182 {
183     biom->ctrl = ctrl;
184     return 1;
185 }
186 
BIO_meth_get_create(const BIO_METHOD * biom)187 int (*BIO_meth_get_create(const BIO_METHOD *biom)) (BIO *)
188 {
189     return biom->create;
190 }
191 
BIO_meth_set_create(BIO_METHOD * biom,int (* create)(BIO *))192 int BIO_meth_set_create(BIO_METHOD *biom, int (*create) (BIO *))
193 {
194     biom->create = create;
195     return 1;
196 }
197 
BIO_meth_get_destroy(const BIO_METHOD * biom)198 int (*BIO_meth_get_destroy(const BIO_METHOD *biom)) (BIO *)
199 {
200     return biom->destroy;
201 }
202 
BIO_meth_set_destroy(BIO_METHOD * biom,int (* destroy)(BIO *))203 int BIO_meth_set_destroy(BIO_METHOD *biom, int (*destroy) (BIO *))
204 {
205     biom->destroy = destroy;
206     return 1;
207 }
208 
BIO_meth_get_callback_ctrl(const BIO_METHOD * biom)209 long (*BIO_meth_get_callback_ctrl(const BIO_METHOD *biom)) (BIO *, int, BIO_info_cb *)
210 {
211     return biom->callback_ctrl;
212 }
213 
BIO_meth_set_callback_ctrl(BIO_METHOD * biom,long (* callback_ctrl)(BIO *,int,BIO_info_cb *))214 int BIO_meth_set_callback_ctrl(BIO_METHOD *biom,
215                                long (*callback_ctrl) (BIO *, int,
216                                                       BIO_info_cb *))
217 {
218     biom->callback_ctrl = callback_ctrl;
219     return 1;
220 }
221