1*8fb009dcSAndroid Build Coastguard Worker /* Copyright (C) 1995-1998 Eric Young ([email protected])
2*8fb009dcSAndroid Build Coastguard Worker * All rights reserved.
3*8fb009dcSAndroid Build Coastguard Worker *
4*8fb009dcSAndroid Build Coastguard Worker * This package is an SSL implementation written
5*8fb009dcSAndroid Build Coastguard Worker * by Eric Young ([email protected]).
6*8fb009dcSAndroid Build Coastguard Worker * The implementation was written so as to conform with Netscapes SSL.
7*8fb009dcSAndroid Build Coastguard Worker *
8*8fb009dcSAndroid Build Coastguard Worker * This library is free for commercial and non-commercial use as long as
9*8fb009dcSAndroid Build Coastguard Worker * the following conditions are aheared to. The following conditions
10*8fb009dcSAndroid Build Coastguard Worker * apply to all code found in this distribution, be it the RC4, RSA,
11*8fb009dcSAndroid Build Coastguard Worker * lhash, DES, etc., code; not just the SSL code. The SSL documentation
12*8fb009dcSAndroid Build Coastguard Worker * included with this distribution is covered by the same copyright terms
13*8fb009dcSAndroid Build Coastguard Worker * except that the holder is Tim Hudson ([email protected]).
14*8fb009dcSAndroid Build Coastguard Worker *
15*8fb009dcSAndroid Build Coastguard Worker * Copyright remains Eric Young's, and as such any Copyright notices in
16*8fb009dcSAndroid Build Coastguard Worker * the code are not to be removed.
17*8fb009dcSAndroid Build Coastguard Worker * If this package is used in a product, Eric Young should be given attribution
18*8fb009dcSAndroid Build Coastguard Worker * as the author of the parts of the library used.
19*8fb009dcSAndroid Build Coastguard Worker * This can be in the form of a textual message at program startup or
20*8fb009dcSAndroid Build Coastguard Worker * in documentation (online or textual) provided with the package.
21*8fb009dcSAndroid Build Coastguard Worker *
22*8fb009dcSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
23*8fb009dcSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
24*8fb009dcSAndroid Build Coastguard Worker * are met:
25*8fb009dcSAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the copyright
26*8fb009dcSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
27*8fb009dcSAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
28*8fb009dcSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
29*8fb009dcSAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
30*8fb009dcSAndroid Build Coastguard Worker * 3. All advertising materials mentioning features or use of this software
31*8fb009dcSAndroid Build Coastguard Worker * must display the following acknowledgement:
32*8fb009dcSAndroid Build Coastguard Worker * "This product includes cryptographic software written by
33*8fb009dcSAndroid Build Coastguard Worker * Eric Young ([email protected])"
34*8fb009dcSAndroid Build Coastguard Worker * The word 'cryptographic' can be left out if the rouines from the library
35*8fb009dcSAndroid Build Coastguard Worker * being used are not cryptographic related :-).
36*8fb009dcSAndroid Build Coastguard Worker * 4. If you include any Windows specific code (or a derivative thereof) from
37*8fb009dcSAndroid Build Coastguard Worker * the apps directory (application code) you must include an acknowledgement:
38*8fb009dcSAndroid Build Coastguard Worker * "This product includes software written by Tim Hudson ([email protected])"
39*8fb009dcSAndroid Build Coastguard Worker *
40*8fb009dcSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY ERIC YOUNG ``AS IS'' AND
41*8fb009dcSAndroid Build Coastguard Worker * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
42*8fb009dcSAndroid Build Coastguard Worker * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
43*8fb009dcSAndroid Build Coastguard Worker * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
44*8fb009dcSAndroid Build Coastguard Worker * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
45*8fb009dcSAndroid Build Coastguard Worker * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
46*8fb009dcSAndroid Build Coastguard Worker * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
47*8fb009dcSAndroid Build Coastguard Worker * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
48*8fb009dcSAndroid Build Coastguard Worker * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
49*8fb009dcSAndroid Build Coastguard Worker * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
50*8fb009dcSAndroid Build Coastguard Worker * SUCH DAMAGE.
51*8fb009dcSAndroid Build Coastguard Worker *
52*8fb009dcSAndroid Build Coastguard Worker * The licence and distribution terms for any publically available version or
53*8fb009dcSAndroid Build Coastguard Worker * derivative of this code cannot be changed. i.e. this code cannot simply be
54*8fb009dcSAndroid Build Coastguard Worker * copied and put under another distribution licence
55*8fb009dcSAndroid Build Coastguard Worker * [including the GNU Public Licence.] */
56*8fb009dcSAndroid Build Coastguard Worker
57*8fb009dcSAndroid Build Coastguard Worker #include <openssl/bio.h>
58*8fb009dcSAndroid Build Coastguard Worker
59*8fb009dcSAndroid Build Coastguard Worker #if !defined(OPENSSL_NO_SOCK)
60*8fb009dcSAndroid Build Coastguard Worker
61*8fb009dcSAndroid Build Coastguard Worker #include <assert.h>
62*8fb009dcSAndroid Build Coastguard Worker #include <errno.h>
63*8fb009dcSAndroid Build Coastguard Worker #include <string.h>
64*8fb009dcSAndroid Build Coastguard Worker
65*8fb009dcSAndroid Build Coastguard Worker #if !defined(OPENSSL_WINDOWS)
66*8fb009dcSAndroid Build Coastguard Worker #include <sys/socket.h>
67*8fb009dcSAndroid Build Coastguard Worker #include <netinet/in.h>
68*8fb009dcSAndroid Build Coastguard Worker #include <arpa/inet.h>
69*8fb009dcSAndroid Build Coastguard Worker #include <unistd.h>
70*8fb009dcSAndroid Build Coastguard Worker #else
71*8fb009dcSAndroid Build Coastguard Worker OPENSSL_MSVC_PRAGMA(warning(push, 3))
72*8fb009dcSAndroid Build Coastguard Worker #include <winsock2.h>
73*8fb009dcSAndroid Build Coastguard Worker #include <ws2tcpip.h>
74*8fb009dcSAndroid Build Coastguard Worker OPENSSL_MSVC_PRAGMA(warning(pop))
75*8fb009dcSAndroid Build Coastguard Worker #endif
76*8fb009dcSAndroid Build Coastguard Worker
77*8fb009dcSAndroid Build Coastguard Worker #include <openssl/err.h>
78*8fb009dcSAndroid Build Coastguard Worker #include <openssl/mem.h>
79*8fb009dcSAndroid Build Coastguard Worker
80*8fb009dcSAndroid Build Coastguard Worker #include "internal.h"
81*8fb009dcSAndroid Build Coastguard Worker #include "../internal.h"
82*8fb009dcSAndroid Build Coastguard Worker
83*8fb009dcSAndroid Build Coastguard Worker
84*8fb009dcSAndroid Build Coastguard Worker enum {
85*8fb009dcSAndroid Build Coastguard Worker BIO_CONN_S_BEFORE,
86*8fb009dcSAndroid Build Coastguard Worker BIO_CONN_S_BLOCKED_CONNECT,
87*8fb009dcSAndroid Build Coastguard Worker BIO_CONN_S_OK,
88*8fb009dcSAndroid Build Coastguard Worker };
89*8fb009dcSAndroid Build Coastguard Worker
90*8fb009dcSAndroid Build Coastguard Worker typedef struct bio_connect_st {
91*8fb009dcSAndroid Build Coastguard Worker int state;
92*8fb009dcSAndroid Build Coastguard Worker
93*8fb009dcSAndroid Build Coastguard Worker char *param_hostname;
94*8fb009dcSAndroid Build Coastguard Worker char *param_port;
95*8fb009dcSAndroid Build Coastguard Worker int nbio;
96*8fb009dcSAndroid Build Coastguard Worker
97*8fb009dcSAndroid Build Coastguard Worker unsigned short port;
98*8fb009dcSAndroid Build Coastguard Worker
99*8fb009dcSAndroid Build Coastguard Worker struct sockaddr_storage them;
100*8fb009dcSAndroid Build Coastguard Worker socklen_t them_length;
101*8fb009dcSAndroid Build Coastguard Worker
102*8fb009dcSAndroid Build Coastguard Worker // the file descriptor is kept in bio->num in order to match the socket
103*8fb009dcSAndroid Build Coastguard Worker // BIO.
104*8fb009dcSAndroid Build Coastguard Worker
105*8fb009dcSAndroid Build Coastguard Worker // info_callback is called when the connection is initially made
106*8fb009dcSAndroid Build Coastguard Worker // callback(BIO,state,ret); The callback should return 'ret', state is for
107*8fb009dcSAndroid Build Coastguard Worker // compatibility with the SSL info_callback.
108*8fb009dcSAndroid Build Coastguard Worker int (*info_callback)(const BIO *bio, int state, int ret);
109*8fb009dcSAndroid Build Coastguard Worker } BIO_CONNECT;
110*8fb009dcSAndroid Build Coastguard Worker
111*8fb009dcSAndroid Build Coastguard Worker #if !defined(OPENSSL_WINDOWS)
closesocket(int sock)112*8fb009dcSAndroid Build Coastguard Worker static int closesocket(int sock) {
113*8fb009dcSAndroid Build Coastguard Worker return close(sock);
114*8fb009dcSAndroid Build Coastguard Worker }
115*8fb009dcSAndroid Build Coastguard Worker #endif
116*8fb009dcSAndroid Build Coastguard Worker
117*8fb009dcSAndroid Build Coastguard Worker // split_host_and_port sets |*out_host| and |*out_port| to the host and port
118*8fb009dcSAndroid Build Coastguard Worker // parsed from |name|. It returns one on success or zero on error. Even when
119*8fb009dcSAndroid Build Coastguard Worker // successful, |*out_port| may be NULL on return if no port was specified.
split_host_and_port(char ** out_host,char ** out_port,const char * name)120*8fb009dcSAndroid Build Coastguard Worker static int split_host_and_port(char **out_host, char **out_port,
121*8fb009dcSAndroid Build Coastguard Worker const char *name) {
122*8fb009dcSAndroid Build Coastguard Worker const char *host, *port = NULL;
123*8fb009dcSAndroid Build Coastguard Worker size_t host_len = 0;
124*8fb009dcSAndroid Build Coastguard Worker
125*8fb009dcSAndroid Build Coastguard Worker *out_host = NULL;
126*8fb009dcSAndroid Build Coastguard Worker *out_port = NULL;
127*8fb009dcSAndroid Build Coastguard Worker
128*8fb009dcSAndroid Build Coastguard Worker if (name[0] == '[') { // bracketed IPv6 address
129*8fb009dcSAndroid Build Coastguard Worker const char *close = strchr(name, ']');
130*8fb009dcSAndroid Build Coastguard Worker if (close == NULL) {
131*8fb009dcSAndroid Build Coastguard Worker return 0;
132*8fb009dcSAndroid Build Coastguard Worker }
133*8fb009dcSAndroid Build Coastguard Worker host = name + 1;
134*8fb009dcSAndroid Build Coastguard Worker host_len = close - host;
135*8fb009dcSAndroid Build Coastguard Worker if (close[1] == ':') { // [IP]:port
136*8fb009dcSAndroid Build Coastguard Worker port = close + 2;
137*8fb009dcSAndroid Build Coastguard Worker } else if (close[1] != 0) {
138*8fb009dcSAndroid Build Coastguard Worker return 0;
139*8fb009dcSAndroid Build Coastguard Worker }
140*8fb009dcSAndroid Build Coastguard Worker } else {
141*8fb009dcSAndroid Build Coastguard Worker const char *colon = strchr(name, ':');
142*8fb009dcSAndroid Build Coastguard Worker if (colon == NULL || strchr(colon + 1, ':') != NULL) { // IPv6 address
143*8fb009dcSAndroid Build Coastguard Worker host = name;
144*8fb009dcSAndroid Build Coastguard Worker host_len = strlen(name);
145*8fb009dcSAndroid Build Coastguard Worker } else { // host:port
146*8fb009dcSAndroid Build Coastguard Worker host = name;
147*8fb009dcSAndroid Build Coastguard Worker host_len = colon - name;
148*8fb009dcSAndroid Build Coastguard Worker port = colon + 1;
149*8fb009dcSAndroid Build Coastguard Worker }
150*8fb009dcSAndroid Build Coastguard Worker }
151*8fb009dcSAndroid Build Coastguard Worker
152*8fb009dcSAndroid Build Coastguard Worker *out_host = OPENSSL_strndup(host, host_len);
153*8fb009dcSAndroid Build Coastguard Worker if (*out_host == NULL) {
154*8fb009dcSAndroid Build Coastguard Worker return 0;
155*8fb009dcSAndroid Build Coastguard Worker }
156*8fb009dcSAndroid Build Coastguard Worker if (port == NULL) {
157*8fb009dcSAndroid Build Coastguard Worker *out_port = NULL;
158*8fb009dcSAndroid Build Coastguard Worker return 1;
159*8fb009dcSAndroid Build Coastguard Worker }
160*8fb009dcSAndroid Build Coastguard Worker *out_port = OPENSSL_strdup(port);
161*8fb009dcSAndroid Build Coastguard Worker if (*out_port == NULL) {
162*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(*out_host);
163*8fb009dcSAndroid Build Coastguard Worker *out_host = NULL;
164*8fb009dcSAndroid Build Coastguard Worker return 0;
165*8fb009dcSAndroid Build Coastguard Worker }
166*8fb009dcSAndroid Build Coastguard Worker return 1;
167*8fb009dcSAndroid Build Coastguard Worker }
168*8fb009dcSAndroid Build Coastguard Worker
conn_state(BIO * bio,BIO_CONNECT * c)169*8fb009dcSAndroid Build Coastguard Worker static int conn_state(BIO *bio, BIO_CONNECT *c) {
170*8fb009dcSAndroid Build Coastguard Worker int ret = -1, i;
171*8fb009dcSAndroid Build Coastguard Worker int (*cb)(const BIO *, int, int) = NULL;
172*8fb009dcSAndroid Build Coastguard Worker
173*8fb009dcSAndroid Build Coastguard Worker if (c->info_callback != NULL) {
174*8fb009dcSAndroid Build Coastguard Worker cb = c->info_callback;
175*8fb009dcSAndroid Build Coastguard Worker }
176*8fb009dcSAndroid Build Coastguard Worker
177*8fb009dcSAndroid Build Coastguard Worker for (;;) {
178*8fb009dcSAndroid Build Coastguard Worker switch (c->state) {
179*8fb009dcSAndroid Build Coastguard Worker case BIO_CONN_S_BEFORE:
180*8fb009dcSAndroid Build Coastguard Worker // If there's a hostname and a port, assume that both are
181*8fb009dcSAndroid Build Coastguard Worker // exactly what they say. If there is only a hostname, try
182*8fb009dcSAndroid Build Coastguard Worker // (just once) to split it into a hostname and port.
183*8fb009dcSAndroid Build Coastguard Worker
184*8fb009dcSAndroid Build Coastguard Worker if (c->param_hostname == NULL) {
185*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_NO_HOSTNAME_SPECIFIED);
186*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
187*8fb009dcSAndroid Build Coastguard Worker }
188*8fb009dcSAndroid Build Coastguard Worker
189*8fb009dcSAndroid Build Coastguard Worker if (c->param_port == NULL) {
190*8fb009dcSAndroid Build Coastguard Worker char *host, *port;
191*8fb009dcSAndroid Build Coastguard Worker if (!split_host_and_port(&host, &port, c->param_hostname) ||
192*8fb009dcSAndroid Build Coastguard Worker port == NULL) {
193*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(host);
194*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(port);
195*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_NO_PORT_SPECIFIED);
196*8fb009dcSAndroid Build Coastguard Worker ERR_add_error_data(2, "host=", c->param_hostname);
197*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
198*8fb009dcSAndroid Build Coastguard Worker }
199*8fb009dcSAndroid Build Coastguard Worker
200*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(c->param_port);
201*8fb009dcSAndroid Build Coastguard Worker c->param_port = port;
202*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(c->param_hostname);
203*8fb009dcSAndroid Build Coastguard Worker c->param_hostname = host;
204*8fb009dcSAndroid Build Coastguard Worker }
205*8fb009dcSAndroid Build Coastguard Worker
206*8fb009dcSAndroid Build Coastguard Worker if (!bio_ip_and_port_to_socket_and_addr(
207*8fb009dcSAndroid Build Coastguard Worker &bio->num, &c->them, &c->them_length, c->param_hostname,
208*8fb009dcSAndroid Build Coastguard Worker c->param_port)) {
209*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_UNABLE_TO_CREATE_SOCKET);
210*8fb009dcSAndroid Build Coastguard Worker ERR_add_error_data(4, "host=", c->param_hostname, ":", c->param_port);
211*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
212*8fb009dcSAndroid Build Coastguard Worker }
213*8fb009dcSAndroid Build Coastguard Worker
214*8fb009dcSAndroid Build Coastguard Worker if (c->nbio) {
215*8fb009dcSAndroid Build Coastguard Worker if (!bio_socket_nbio(bio->num, 1)) {
216*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_ERROR_SETTING_NBIO);
217*8fb009dcSAndroid Build Coastguard Worker ERR_add_error_data(4, "host=", c->param_hostname, ":",
218*8fb009dcSAndroid Build Coastguard Worker c->param_port);
219*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
220*8fb009dcSAndroid Build Coastguard Worker }
221*8fb009dcSAndroid Build Coastguard Worker }
222*8fb009dcSAndroid Build Coastguard Worker
223*8fb009dcSAndroid Build Coastguard Worker i = 1;
224*8fb009dcSAndroid Build Coastguard Worker ret = setsockopt(bio->num, SOL_SOCKET, SO_KEEPALIVE, (char *)&i,
225*8fb009dcSAndroid Build Coastguard Worker sizeof(i));
226*8fb009dcSAndroid Build Coastguard Worker if (ret < 0) {
227*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_SYSTEM_ERROR();
228*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_KEEPALIVE);
229*8fb009dcSAndroid Build Coastguard Worker ERR_add_error_data(4, "host=", c->param_hostname, ":", c->param_port);
230*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
231*8fb009dcSAndroid Build Coastguard Worker }
232*8fb009dcSAndroid Build Coastguard Worker
233*8fb009dcSAndroid Build Coastguard Worker BIO_clear_retry_flags(bio);
234*8fb009dcSAndroid Build Coastguard Worker ret = connect(bio->num, (struct sockaddr*) &c->them, c->them_length);
235*8fb009dcSAndroid Build Coastguard Worker if (ret < 0) {
236*8fb009dcSAndroid Build Coastguard Worker if (bio_socket_should_retry(ret)) {
237*8fb009dcSAndroid Build Coastguard Worker BIO_set_flags(bio, (BIO_FLAGS_IO_SPECIAL | BIO_FLAGS_SHOULD_RETRY));
238*8fb009dcSAndroid Build Coastguard Worker c->state = BIO_CONN_S_BLOCKED_CONNECT;
239*8fb009dcSAndroid Build Coastguard Worker bio->retry_reason = BIO_RR_CONNECT;
240*8fb009dcSAndroid Build Coastguard Worker } else {
241*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_SYSTEM_ERROR();
242*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_CONNECT_ERROR);
243*8fb009dcSAndroid Build Coastguard Worker ERR_add_error_data(4, "host=", c->param_hostname, ":",
244*8fb009dcSAndroid Build Coastguard Worker c->param_port);
245*8fb009dcSAndroid Build Coastguard Worker }
246*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
247*8fb009dcSAndroid Build Coastguard Worker } else {
248*8fb009dcSAndroid Build Coastguard Worker c->state = BIO_CONN_S_OK;
249*8fb009dcSAndroid Build Coastguard Worker }
250*8fb009dcSAndroid Build Coastguard Worker break;
251*8fb009dcSAndroid Build Coastguard Worker
252*8fb009dcSAndroid Build Coastguard Worker case BIO_CONN_S_BLOCKED_CONNECT:
253*8fb009dcSAndroid Build Coastguard Worker i = bio_sock_error(bio->num);
254*8fb009dcSAndroid Build Coastguard Worker if (i) {
255*8fb009dcSAndroid Build Coastguard Worker if (bio_socket_should_retry(ret)) {
256*8fb009dcSAndroid Build Coastguard Worker BIO_set_flags(bio, (BIO_FLAGS_IO_SPECIAL | BIO_FLAGS_SHOULD_RETRY));
257*8fb009dcSAndroid Build Coastguard Worker c->state = BIO_CONN_S_BLOCKED_CONNECT;
258*8fb009dcSAndroid Build Coastguard Worker bio->retry_reason = BIO_RR_CONNECT;
259*8fb009dcSAndroid Build Coastguard Worker ret = -1;
260*8fb009dcSAndroid Build Coastguard Worker } else {
261*8fb009dcSAndroid Build Coastguard Worker BIO_clear_retry_flags(bio);
262*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_SYSTEM_ERROR();
263*8fb009dcSAndroid Build Coastguard Worker OPENSSL_PUT_ERROR(BIO, BIO_R_NBIO_CONNECT_ERROR);
264*8fb009dcSAndroid Build Coastguard Worker ERR_add_error_data(4, "host=", c->param_hostname, ":", c->param_port);
265*8fb009dcSAndroid Build Coastguard Worker ret = 0;
266*8fb009dcSAndroid Build Coastguard Worker }
267*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
268*8fb009dcSAndroid Build Coastguard Worker } else {
269*8fb009dcSAndroid Build Coastguard Worker c->state = BIO_CONN_S_OK;
270*8fb009dcSAndroid Build Coastguard Worker }
271*8fb009dcSAndroid Build Coastguard Worker break;
272*8fb009dcSAndroid Build Coastguard Worker
273*8fb009dcSAndroid Build Coastguard Worker case BIO_CONN_S_OK:
274*8fb009dcSAndroid Build Coastguard Worker ret = 1;
275*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
276*8fb009dcSAndroid Build Coastguard Worker default:
277*8fb009dcSAndroid Build Coastguard Worker assert(0);
278*8fb009dcSAndroid Build Coastguard Worker goto exit_loop;
279*8fb009dcSAndroid Build Coastguard Worker }
280*8fb009dcSAndroid Build Coastguard Worker
281*8fb009dcSAndroid Build Coastguard Worker if (cb != NULL) {
282*8fb009dcSAndroid Build Coastguard Worker ret = cb((BIO *)bio, c->state, ret);
283*8fb009dcSAndroid Build Coastguard Worker if (ret == 0) {
284*8fb009dcSAndroid Build Coastguard Worker goto end;
285*8fb009dcSAndroid Build Coastguard Worker }
286*8fb009dcSAndroid Build Coastguard Worker }
287*8fb009dcSAndroid Build Coastguard Worker }
288*8fb009dcSAndroid Build Coastguard Worker
289*8fb009dcSAndroid Build Coastguard Worker exit_loop:
290*8fb009dcSAndroid Build Coastguard Worker if (cb != NULL) {
291*8fb009dcSAndroid Build Coastguard Worker ret = cb((BIO *)bio, c->state, ret);
292*8fb009dcSAndroid Build Coastguard Worker }
293*8fb009dcSAndroid Build Coastguard Worker
294*8fb009dcSAndroid Build Coastguard Worker end:
295*8fb009dcSAndroid Build Coastguard Worker return ret;
296*8fb009dcSAndroid Build Coastguard Worker }
297*8fb009dcSAndroid Build Coastguard Worker
BIO_CONNECT_new(void)298*8fb009dcSAndroid Build Coastguard Worker static BIO_CONNECT *BIO_CONNECT_new(void) {
299*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT *ret = OPENSSL_zalloc(sizeof(BIO_CONNECT));
300*8fb009dcSAndroid Build Coastguard Worker if (ret == NULL) {
301*8fb009dcSAndroid Build Coastguard Worker return NULL;
302*8fb009dcSAndroid Build Coastguard Worker }
303*8fb009dcSAndroid Build Coastguard Worker ret->state = BIO_CONN_S_BEFORE;
304*8fb009dcSAndroid Build Coastguard Worker return ret;
305*8fb009dcSAndroid Build Coastguard Worker }
306*8fb009dcSAndroid Build Coastguard Worker
BIO_CONNECT_free(BIO_CONNECT * c)307*8fb009dcSAndroid Build Coastguard Worker static void BIO_CONNECT_free(BIO_CONNECT *c) {
308*8fb009dcSAndroid Build Coastguard Worker if (c == NULL) {
309*8fb009dcSAndroid Build Coastguard Worker return;
310*8fb009dcSAndroid Build Coastguard Worker }
311*8fb009dcSAndroid Build Coastguard Worker
312*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(c->param_hostname);
313*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(c->param_port);
314*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(c);
315*8fb009dcSAndroid Build Coastguard Worker }
316*8fb009dcSAndroid Build Coastguard Worker
conn_new(BIO * bio)317*8fb009dcSAndroid Build Coastguard Worker static int conn_new(BIO *bio) {
318*8fb009dcSAndroid Build Coastguard Worker bio->init = 0;
319*8fb009dcSAndroid Build Coastguard Worker bio->num = -1;
320*8fb009dcSAndroid Build Coastguard Worker bio->flags = 0;
321*8fb009dcSAndroid Build Coastguard Worker bio->ptr = BIO_CONNECT_new();
322*8fb009dcSAndroid Build Coastguard Worker return bio->ptr != NULL;
323*8fb009dcSAndroid Build Coastguard Worker }
324*8fb009dcSAndroid Build Coastguard Worker
conn_close_socket(BIO * bio)325*8fb009dcSAndroid Build Coastguard Worker static void conn_close_socket(BIO *bio) {
326*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT *c = (BIO_CONNECT *) bio->ptr;
327*8fb009dcSAndroid Build Coastguard Worker
328*8fb009dcSAndroid Build Coastguard Worker if (bio->num == -1) {
329*8fb009dcSAndroid Build Coastguard Worker return;
330*8fb009dcSAndroid Build Coastguard Worker }
331*8fb009dcSAndroid Build Coastguard Worker
332*8fb009dcSAndroid Build Coastguard Worker // Only do a shutdown if things were established
333*8fb009dcSAndroid Build Coastguard Worker if (c->state == BIO_CONN_S_OK) {
334*8fb009dcSAndroid Build Coastguard Worker shutdown(bio->num, 2);
335*8fb009dcSAndroid Build Coastguard Worker }
336*8fb009dcSAndroid Build Coastguard Worker closesocket(bio->num);
337*8fb009dcSAndroid Build Coastguard Worker bio->num = -1;
338*8fb009dcSAndroid Build Coastguard Worker }
339*8fb009dcSAndroid Build Coastguard Worker
conn_free(BIO * bio)340*8fb009dcSAndroid Build Coastguard Worker static int conn_free(BIO *bio) {
341*8fb009dcSAndroid Build Coastguard Worker if (bio->shutdown) {
342*8fb009dcSAndroid Build Coastguard Worker conn_close_socket(bio);
343*8fb009dcSAndroid Build Coastguard Worker }
344*8fb009dcSAndroid Build Coastguard Worker
345*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT_free((BIO_CONNECT*) bio->ptr);
346*8fb009dcSAndroid Build Coastguard Worker
347*8fb009dcSAndroid Build Coastguard Worker return 1;
348*8fb009dcSAndroid Build Coastguard Worker }
349*8fb009dcSAndroid Build Coastguard Worker
conn_read(BIO * bio,char * out,int out_len)350*8fb009dcSAndroid Build Coastguard Worker static int conn_read(BIO *bio, char *out, int out_len) {
351*8fb009dcSAndroid Build Coastguard Worker int ret = 0;
352*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT *data;
353*8fb009dcSAndroid Build Coastguard Worker
354*8fb009dcSAndroid Build Coastguard Worker data = (BIO_CONNECT *)bio->ptr;
355*8fb009dcSAndroid Build Coastguard Worker if (data->state != BIO_CONN_S_OK) {
356*8fb009dcSAndroid Build Coastguard Worker ret = conn_state(bio, data);
357*8fb009dcSAndroid Build Coastguard Worker if (ret <= 0) {
358*8fb009dcSAndroid Build Coastguard Worker return ret;
359*8fb009dcSAndroid Build Coastguard Worker }
360*8fb009dcSAndroid Build Coastguard Worker }
361*8fb009dcSAndroid Build Coastguard Worker
362*8fb009dcSAndroid Build Coastguard Worker bio_clear_socket_error();
363*8fb009dcSAndroid Build Coastguard Worker ret = (int)recv(bio->num, out, out_len, 0);
364*8fb009dcSAndroid Build Coastguard Worker BIO_clear_retry_flags(bio);
365*8fb009dcSAndroid Build Coastguard Worker if (ret <= 0) {
366*8fb009dcSAndroid Build Coastguard Worker if (bio_socket_should_retry(ret)) {
367*8fb009dcSAndroid Build Coastguard Worker BIO_set_retry_read(bio);
368*8fb009dcSAndroid Build Coastguard Worker }
369*8fb009dcSAndroid Build Coastguard Worker }
370*8fb009dcSAndroid Build Coastguard Worker
371*8fb009dcSAndroid Build Coastguard Worker return ret;
372*8fb009dcSAndroid Build Coastguard Worker }
373*8fb009dcSAndroid Build Coastguard Worker
conn_write(BIO * bio,const char * in,int in_len)374*8fb009dcSAndroid Build Coastguard Worker static int conn_write(BIO *bio, const char *in, int in_len) {
375*8fb009dcSAndroid Build Coastguard Worker int ret;
376*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT *data;
377*8fb009dcSAndroid Build Coastguard Worker
378*8fb009dcSAndroid Build Coastguard Worker data = (BIO_CONNECT *)bio->ptr;
379*8fb009dcSAndroid Build Coastguard Worker if (data->state != BIO_CONN_S_OK) {
380*8fb009dcSAndroid Build Coastguard Worker ret = conn_state(bio, data);
381*8fb009dcSAndroid Build Coastguard Worker if (ret <= 0) {
382*8fb009dcSAndroid Build Coastguard Worker return ret;
383*8fb009dcSAndroid Build Coastguard Worker }
384*8fb009dcSAndroid Build Coastguard Worker }
385*8fb009dcSAndroid Build Coastguard Worker
386*8fb009dcSAndroid Build Coastguard Worker bio_clear_socket_error();
387*8fb009dcSAndroid Build Coastguard Worker ret = (int)send(bio->num, in, in_len, 0);
388*8fb009dcSAndroid Build Coastguard Worker BIO_clear_retry_flags(bio);
389*8fb009dcSAndroid Build Coastguard Worker if (ret <= 0) {
390*8fb009dcSAndroid Build Coastguard Worker if (bio_socket_should_retry(ret)) {
391*8fb009dcSAndroid Build Coastguard Worker BIO_set_retry_write(bio);
392*8fb009dcSAndroid Build Coastguard Worker }
393*8fb009dcSAndroid Build Coastguard Worker }
394*8fb009dcSAndroid Build Coastguard Worker
395*8fb009dcSAndroid Build Coastguard Worker return ret;
396*8fb009dcSAndroid Build Coastguard Worker }
397*8fb009dcSAndroid Build Coastguard Worker
conn_ctrl(BIO * bio,int cmd,long num,void * ptr)398*8fb009dcSAndroid Build Coastguard Worker static long conn_ctrl(BIO *bio, int cmd, long num, void *ptr) {
399*8fb009dcSAndroid Build Coastguard Worker int *ip;
400*8fb009dcSAndroid Build Coastguard Worker long ret = 1;
401*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT *data;
402*8fb009dcSAndroid Build Coastguard Worker
403*8fb009dcSAndroid Build Coastguard Worker data = (BIO_CONNECT *)bio->ptr;
404*8fb009dcSAndroid Build Coastguard Worker
405*8fb009dcSAndroid Build Coastguard Worker switch (cmd) {
406*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_RESET:
407*8fb009dcSAndroid Build Coastguard Worker ret = 0;
408*8fb009dcSAndroid Build Coastguard Worker data->state = BIO_CONN_S_BEFORE;
409*8fb009dcSAndroid Build Coastguard Worker conn_close_socket(bio);
410*8fb009dcSAndroid Build Coastguard Worker bio->flags = 0;
411*8fb009dcSAndroid Build Coastguard Worker break;
412*8fb009dcSAndroid Build Coastguard Worker case BIO_C_DO_STATE_MACHINE:
413*8fb009dcSAndroid Build Coastguard Worker // use this one to start the connection
414*8fb009dcSAndroid Build Coastguard Worker if (data->state != BIO_CONN_S_OK) {
415*8fb009dcSAndroid Build Coastguard Worker ret = (long)conn_state(bio, data);
416*8fb009dcSAndroid Build Coastguard Worker } else {
417*8fb009dcSAndroid Build Coastguard Worker ret = 1;
418*8fb009dcSAndroid Build Coastguard Worker }
419*8fb009dcSAndroid Build Coastguard Worker break;
420*8fb009dcSAndroid Build Coastguard Worker case BIO_C_SET_CONNECT:
421*8fb009dcSAndroid Build Coastguard Worker if (ptr != NULL) {
422*8fb009dcSAndroid Build Coastguard Worker bio->init = 1;
423*8fb009dcSAndroid Build Coastguard Worker if (num == 0) {
424*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(data->param_hostname);
425*8fb009dcSAndroid Build Coastguard Worker data->param_hostname = OPENSSL_strdup(ptr);
426*8fb009dcSAndroid Build Coastguard Worker if (data->param_hostname == NULL) {
427*8fb009dcSAndroid Build Coastguard Worker ret = 0;
428*8fb009dcSAndroid Build Coastguard Worker }
429*8fb009dcSAndroid Build Coastguard Worker } else if (num == 1) {
430*8fb009dcSAndroid Build Coastguard Worker OPENSSL_free(data->param_port);
431*8fb009dcSAndroid Build Coastguard Worker data->param_port = OPENSSL_strdup(ptr);
432*8fb009dcSAndroid Build Coastguard Worker if (data->param_port == NULL) {
433*8fb009dcSAndroid Build Coastguard Worker ret = 0;
434*8fb009dcSAndroid Build Coastguard Worker }
435*8fb009dcSAndroid Build Coastguard Worker } else {
436*8fb009dcSAndroid Build Coastguard Worker ret = 0;
437*8fb009dcSAndroid Build Coastguard Worker }
438*8fb009dcSAndroid Build Coastguard Worker }
439*8fb009dcSAndroid Build Coastguard Worker break;
440*8fb009dcSAndroid Build Coastguard Worker case BIO_C_SET_NBIO:
441*8fb009dcSAndroid Build Coastguard Worker data->nbio = (int)num;
442*8fb009dcSAndroid Build Coastguard Worker break;
443*8fb009dcSAndroid Build Coastguard Worker case BIO_C_GET_FD:
444*8fb009dcSAndroid Build Coastguard Worker if (bio->init) {
445*8fb009dcSAndroid Build Coastguard Worker ip = (int *)ptr;
446*8fb009dcSAndroid Build Coastguard Worker if (ip != NULL) {
447*8fb009dcSAndroid Build Coastguard Worker *ip = bio->num;
448*8fb009dcSAndroid Build Coastguard Worker }
449*8fb009dcSAndroid Build Coastguard Worker ret = bio->num;
450*8fb009dcSAndroid Build Coastguard Worker } else {
451*8fb009dcSAndroid Build Coastguard Worker ret = -1;
452*8fb009dcSAndroid Build Coastguard Worker }
453*8fb009dcSAndroid Build Coastguard Worker break;
454*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_GET_CLOSE:
455*8fb009dcSAndroid Build Coastguard Worker ret = bio->shutdown;
456*8fb009dcSAndroid Build Coastguard Worker break;
457*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_SET_CLOSE:
458*8fb009dcSAndroid Build Coastguard Worker bio->shutdown = (int)num;
459*8fb009dcSAndroid Build Coastguard Worker break;
460*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_PENDING:
461*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_WPENDING:
462*8fb009dcSAndroid Build Coastguard Worker ret = 0;
463*8fb009dcSAndroid Build Coastguard Worker break;
464*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_FLUSH:
465*8fb009dcSAndroid Build Coastguard Worker break;
466*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_GET_CALLBACK: {
467*8fb009dcSAndroid Build Coastguard Worker int (**fptr)(const BIO *bio, int state, int xret) = ptr;
468*8fb009dcSAndroid Build Coastguard Worker *fptr = data->info_callback;
469*8fb009dcSAndroid Build Coastguard Worker } break;
470*8fb009dcSAndroid Build Coastguard Worker default:
471*8fb009dcSAndroid Build Coastguard Worker ret = 0;
472*8fb009dcSAndroid Build Coastguard Worker break;
473*8fb009dcSAndroid Build Coastguard Worker }
474*8fb009dcSAndroid Build Coastguard Worker return ret;
475*8fb009dcSAndroid Build Coastguard Worker }
476*8fb009dcSAndroid Build Coastguard Worker
conn_callback_ctrl(BIO * bio,int cmd,bio_info_cb fp)477*8fb009dcSAndroid Build Coastguard Worker static long conn_callback_ctrl(BIO *bio, int cmd, bio_info_cb fp) {
478*8fb009dcSAndroid Build Coastguard Worker long ret = 1;
479*8fb009dcSAndroid Build Coastguard Worker BIO_CONNECT *data;
480*8fb009dcSAndroid Build Coastguard Worker
481*8fb009dcSAndroid Build Coastguard Worker data = (BIO_CONNECT *)bio->ptr;
482*8fb009dcSAndroid Build Coastguard Worker
483*8fb009dcSAndroid Build Coastguard Worker switch (cmd) {
484*8fb009dcSAndroid Build Coastguard Worker case BIO_CTRL_SET_CALLBACK:
485*8fb009dcSAndroid Build Coastguard Worker // This is the actual type signature of |fp|. The caller is expected to
486*8fb009dcSAndroid Build Coastguard Worker // cast it to |bio_info_cb| due to the |BIO_callback_ctrl| calling
487*8fb009dcSAndroid Build Coastguard Worker // convention.
488*8fb009dcSAndroid Build Coastguard Worker OPENSSL_MSVC_PRAGMA(warning(push))
489*8fb009dcSAndroid Build Coastguard Worker OPENSSL_MSVC_PRAGMA(warning(disable : 4191))
490*8fb009dcSAndroid Build Coastguard Worker OPENSSL_CLANG_PRAGMA("clang diagnostic push")
491*8fb009dcSAndroid Build Coastguard Worker OPENSSL_CLANG_PRAGMA("clang diagnostic ignored \"-Wunknown-warning-option\"")
492*8fb009dcSAndroid Build Coastguard Worker OPENSSL_CLANG_PRAGMA("clang diagnostic ignored \"-Wcast-function-type\"")
493*8fb009dcSAndroid Build Coastguard Worker data->info_callback = (int (*)(const struct bio_st *, int, int))fp;
494*8fb009dcSAndroid Build Coastguard Worker OPENSSL_CLANG_PRAGMA("clang diagnostic pop")
495*8fb009dcSAndroid Build Coastguard Worker OPENSSL_MSVC_PRAGMA(warning(pop))
496*8fb009dcSAndroid Build Coastguard Worker break;
497*8fb009dcSAndroid Build Coastguard Worker default:
498*8fb009dcSAndroid Build Coastguard Worker ret = 0;
499*8fb009dcSAndroid Build Coastguard Worker break;
500*8fb009dcSAndroid Build Coastguard Worker }
501*8fb009dcSAndroid Build Coastguard Worker return ret;
502*8fb009dcSAndroid Build Coastguard Worker }
503*8fb009dcSAndroid Build Coastguard Worker
BIO_new_connect(const char * hostname)504*8fb009dcSAndroid Build Coastguard Worker BIO *BIO_new_connect(const char *hostname) {
505*8fb009dcSAndroid Build Coastguard Worker BIO *ret;
506*8fb009dcSAndroid Build Coastguard Worker
507*8fb009dcSAndroid Build Coastguard Worker ret = BIO_new(BIO_s_connect());
508*8fb009dcSAndroid Build Coastguard Worker if (ret == NULL) {
509*8fb009dcSAndroid Build Coastguard Worker return NULL;
510*8fb009dcSAndroid Build Coastguard Worker }
511*8fb009dcSAndroid Build Coastguard Worker if (!BIO_set_conn_hostname(ret, hostname)) {
512*8fb009dcSAndroid Build Coastguard Worker BIO_free(ret);
513*8fb009dcSAndroid Build Coastguard Worker return NULL;
514*8fb009dcSAndroid Build Coastguard Worker }
515*8fb009dcSAndroid Build Coastguard Worker return ret;
516*8fb009dcSAndroid Build Coastguard Worker }
517*8fb009dcSAndroid Build Coastguard Worker
518*8fb009dcSAndroid Build Coastguard Worker static const BIO_METHOD methods_connectp = {
519*8fb009dcSAndroid Build Coastguard Worker BIO_TYPE_CONNECT, "socket connect", conn_write, conn_read,
520*8fb009dcSAndroid Build Coastguard Worker NULL /* puts */, NULL /* gets */, conn_ctrl, conn_new,
521*8fb009dcSAndroid Build Coastguard Worker conn_free, conn_callback_ctrl,
522*8fb009dcSAndroid Build Coastguard Worker };
523*8fb009dcSAndroid Build Coastguard Worker
BIO_s_connect(void)524*8fb009dcSAndroid Build Coastguard Worker const BIO_METHOD *BIO_s_connect(void) { return &methods_connectp; }
525*8fb009dcSAndroid Build Coastguard Worker
BIO_set_conn_hostname(BIO * bio,const char * name)526*8fb009dcSAndroid Build Coastguard Worker int BIO_set_conn_hostname(BIO *bio, const char *name) {
527*8fb009dcSAndroid Build Coastguard Worker return (int)BIO_ctrl(bio, BIO_C_SET_CONNECT, 0, (void*) name);
528*8fb009dcSAndroid Build Coastguard Worker }
529*8fb009dcSAndroid Build Coastguard Worker
BIO_set_conn_port(BIO * bio,const char * port_str)530*8fb009dcSAndroid Build Coastguard Worker int BIO_set_conn_port(BIO *bio, const char *port_str) {
531*8fb009dcSAndroid Build Coastguard Worker return (int)BIO_ctrl(bio, BIO_C_SET_CONNECT, 1, (void*) port_str);
532*8fb009dcSAndroid Build Coastguard Worker }
533*8fb009dcSAndroid Build Coastguard Worker
BIO_set_conn_int_port(BIO * bio,const int * port)534*8fb009dcSAndroid Build Coastguard Worker int BIO_set_conn_int_port(BIO *bio, const int *port) {
535*8fb009dcSAndroid Build Coastguard Worker char buf[DECIMAL_SIZE(int) + 1];
536*8fb009dcSAndroid Build Coastguard Worker snprintf(buf, sizeof(buf), "%d", *port);
537*8fb009dcSAndroid Build Coastguard Worker return BIO_set_conn_port(bio, buf);
538*8fb009dcSAndroid Build Coastguard Worker }
539*8fb009dcSAndroid Build Coastguard Worker
BIO_set_nbio(BIO * bio,int on)540*8fb009dcSAndroid Build Coastguard Worker int BIO_set_nbio(BIO *bio, int on) {
541*8fb009dcSAndroid Build Coastguard Worker return (int)BIO_ctrl(bio, BIO_C_SET_NBIO, on, NULL);
542*8fb009dcSAndroid Build Coastguard Worker }
543*8fb009dcSAndroid Build Coastguard Worker
BIO_do_connect(BIO * bio)544*8fb009dcSAndroid Build Coastguard Worker int BIO_do_connect(BIO *bio) {
545*8fb009dcSAndroid Build Coastguard Worker return (int)BIO_ctrl(bio, BIO_C_DO_STATE_MACHINE, 0, NULL);
546*8fb009dcSAndroid Build Coastguard Worker }
547*8fb009dcSAndroid Build Coastguard Worker
548*8fb009dcSAndroid Build Coastguard Worker #endif // OPENSSL_NO_SOCK
549