1 /** 2 * @file 3 * Application layered TCP connection API that executes a proxy-connect. 4 * 5 * This file provides a starting layer that executes a proxy-connect e.g. to 6 * set up TLS connections through a http proxy. 7 */ 8 9 /* 10 * Copyright (c) 2018 Simon Goldschmidt 11 * All rights reserved. 12 * 13 * Redistribution and use in source and binary forms, with or without modification, 14 * are permitted provided that the following conditions are met: 15 * 16 * 1. Redistributions of source code must retain the above copyright notice, 17 * this list of conditions and the following disclaimer. 18 * 2. Redistributions in binary form must reproduce the above copyright notice, 19 * this list of conditions and the following disclaimer in the documentation 20 * and/or other materials provided with the distribution. 21 * 3. The name of the author may not be used to endorse or promote products 22 * derived from this software without specific prior written permission. 23 * 24 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED 25 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 26 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT 27 * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 28 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT 29 * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 30 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 31 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 32 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY 33 * OF SUCH DAMAGE. 34 * 35 * This file is part of the lwIP TCP/IP stack. 36 * 37 * Author: Simon Goldschmidt <goldsimon@gmx.de> 38 * 39 */ 40 41 #include "lwip/apps/altcp_proxyconnect.h" 42 43 #if LWIP_ALTCP /* don't build if not configured for use in lwipopts.h */ 44 45 #include "lwip/altcp.h" 46 #include "lwip/priv/altcp_priv.h" 47 48 #include "lwip/altcp_tcp.h" 49 #include "lwip/altcp_tls.h" 50 51 #include "lwip/mem.h" 52 #include "lwip/init.h" 53 54 #include <stdio.h> 55 56 /** This string is passed in the HTTP header as "User-Agent: " */ 57 #ifndef ALTCP_PROXYCONNECT_CLIENT_AGENT 58 #define ALTCP_PROXYCONNECT_CLIENT_AGENT "lwIP/" LWIP_VERSION_STRING " (http://savannah.nongnu.org/projects/lwip)" 59 #endif 60 61 #define ALTCP_PROXYCONNECT_FLAGS_CONNECT_STARTED 0x01 62 #define ALTCP_PROXYCONNECT_FLAGS_HANDSHAKE_DONE 0x02 63 64 typedef struct altcp_proxyconnect_state_s 65 { 66 ip_addr_t outer_addr; 67 u16_t outer_port; 68 struct altcp_proxyconnect_config *conf; 69 u8_t flags; 70 } altcp_proxyconnect_state_t; 71 72 /* Variable prototype, the actual declaration is at the end of this file 73 since it contains pointers to static functions declared here */ 74 extern const struct altcp_functions altcp_proxyconnect_functions; 75 76 /* memory management functions: */ 77 78 static altcp_proxyconnect_state_t * 79 altcp_proxyconnect_state_alloc(void) 80 { 81 altcp_proxyconnect_state_t *ret = (altcp_proxyconnect_state_t *)mem_calloc(1, sizeof(altcp_proxyconnect_state_t)); 82 return ret; 83 } 84 85 static void 86 altcp_proxyconnect_state_free(altcp_proxyconnect_state_t *state) 87 { 88 LWIP_ASSERT("state != NULL", state != NULL); 89 mem_free(state); 90 } 91 92 /* helper functions */ 93 94 #define PROXY_CONNECT "CONNECT %s:%d HTTP/1.1\r\n" /* HOST, PORT */ \ 95 "User-Agent: %s\r\n" /* User-Agent */\ 96 "Proxy-Connection: keep-alive\r\n" \ 97 "Connection: keep-alive\r\n" \ 98 "\r\n" 99 #define PROXY_CONNECT_FORMAT(host, port) PROXY_CONNECT, host, port, ALTCP_PROXYCONNECT_CLIENT_AGENT 100 101 /* Format the http proxy connect request via snprintf */ 102 static int 103 altcp_proxyconnect_format_request(char *buffer, size_t bufsize, const char *host, int port) 104 { 105 return snprintf(buffer, bufsize, PROXY_CONNECT_FORMAT(host, port)); 106 } 107 108 /* Create and send the http proxy connect request */ 109 static err_t 110 altcp_proxyconnect_send_request(struct altcp_pcb *conn) 111 { 112 int len, len2; 113 mem_size_t alloc_len; 114 char *buffer, *host; 115 altcp_proxyconnect_state_t *state = (altcp_proxyconnect_state_t *)conn->state; 116 117 if (!state) { 118 return ERR_VAL; 119 } 120 /* Use printf with zero length to get the required allocation size */ 121 len = altcp_proxyconnect_format_request(NULL, 0, "", state->outer_port); 122 if (len < 0) { 123 return ERR_VAL; 124 } 125 /* add allocation size for IP address strings */ 126 #if LWIP_IPV6 127 len += 40; /* worst-case IPv6 address length */ 128 #else 129 len += 16; /* worst-case IPv4 address length */ 130 #endif 131 alloc_len = (mem_size_t)len; 132 if ((len < 0) || (int)alloc_len != len) { 133 /* overflow */ 134 return ERR_MEM; 135 } 136 /* Allocate a buffer for the request string */ 137 buffer = (char *)mem_malloc(alloc_len); 138 if (buffer == NULL) { 139 return ERR_MEM; 140 } 141 host = ipaddr_ntoa(&state->outer_addr); 142 len2 = altcp_proxyconnect_format_request(buffer, alloc_len, host, state->outer_port); 143 if ((len2 > 0) && (len2 <= len) && (len2 <= 0xFFFF)) { 144 err_t err = altcp_write(conn->inner_conn, buffer, (u16_t)len2, TCP_WRITE_FLAG_COPY); 145 if (err != ERR_OK) { 146 /* @todo: abort? */ 147 mem_free(buffer); 148 return err; 149 } 150 } 151 mem_free(buffer); 152 return ERR_OK; 153 } 154 155 /* callback functions from inner/lower connection: */ 156 157 /** Connected callback from lower connection (i.e. TCP). 158 * Not really implemented/tested yet... 159 */ 160 static err_t 161 altcp_proxyconnect_lower_connected(void *arg, struct altcp_pcb *inner_conn, err_t err) 162 { 163 struct altcp_pcb *conn = (struct altcp_pcb *)arg; 164 if (conn && conn->state) { 165 LWIP_ASSERT("pcb mismatch", conn->inner_conn == inner_conn); 166 LWIP_UNUSED_ARG(inner_conn); /* for LWIP_NOASSERT */ 167 /* upper connected is called when handshake is done */ 168 if (err != ERR_OK) { 169 if (conn->connected) { 170 if (conn->connected(conn->arg, conn, err) == ERR_ABRT) { 171 return ERR_ABRT; 172 } 173 return ERR_OK; 174 } 175 } 176 /* send proxy connect request here */ 177 return altcp_proxyconnect_send_request(conn); 178 } 179 return ERR_VAL; 180 } 181 182 /** Recv callback from lower connection (i.e. TCP) 183 * This one mainly differs between connection setup (wait for proxy OK string) 184 * and application phase (data is passed on to the application). 185 */ 186 static err_t 187 altcp_proxyconnect_lower_recv(void *arg, struct altcp_pcb *inner_conn, struct pbuf *p, err_t err) 188 { 189 altcp_proxyconnect_state_t *state; 190 struct altcp_pcb *conn = (struct altcp_pcb *)arg; 191 192 LWIP_ASSERT("no err expected", err == ERR_OK); 193 LWIP_UNUSED_ARG(err); 194 195 if (!conn) { 196 /* no connection given as arg? should not happen, but prevent pbuf/conn leaks */ 197 if (p != NULL) { 198 pbuf_free(p); 199 } 200 altcp_close(inner_conn); 201 return ERR_CLSD; 202 } 203 state = (altcp_proxyconnect_state_t *)conn->state; 204 LWIP_ASSERT("pcb mismatch", conn->inner_conn == inner_conn); 205 if (!state) { 206 /* already closed */ 207 if (p != NULL) { 208 pbuf_free(p); 209 } 210 altcp_close(inner_conn); 211 return ERR_CLSD; 212 } 213 if (state->flags & ALTCP_PROXYCONNECT_FLAGS_HANDSHAKE_DONE) { 214 /* application phase, just pass this through */ 215 if (conn->recv) { 216 return conn->recv(conn->arg, conn, p, err); 217 } 218 pbuf_free(p); 219 return ERR_OK; 220 } else { 221 /* setup phase */ 222 /* handle NULL pbuf (inner connection closed) */ 223 if (p == NULL) { 224 if (altcp_close(conn) != ERR_OK) { 225 altcp_abort(conn); 226 return ERR_ABRT; 227 } 228 return ERR_OK; 229 } else { 230 /* @todo: parse setup phase rx data 231 for now, we just wait for the end of the header... */ 232 u16_t idx = pbuf_memfind(p, "\r\n\r\n", 4, 0); 233 altcp_recved(inner_conn, p->tot_len); 234 pbuf_free(p); 235 if (idx != 0xFFFF) { 236 state->flags |= ALTCP_PROXYCONNECT_FLAGS_HANDSHAKE_DONE; 237 if (conn->connected) { 238 return conn->connected(conn->arg, conn, ERR_OK); 239 } 240 } 241 return ERR_OK; 242 } 243 } 244 } 245 246 /** Sent callback from lower connection (i.e. TCP) 247 * This only informs the upper layer to try to send more, not about 248 * the number of ACKed bytes. 249 */ 250 static err_t 251 altcp_proxyconnect_lower_sent(void *arg, struct altcp_pcb *inner_conn, u16_t len) 252 { 253 struct altcp_pcb *conn = (struct altcp_pcb *)arg; 254 LWIP_UNUSED_ARG(len); 255 if (conn) { 256 altcp_proxyconnect_state_t *state = (altcp_proxyconnect_state_t *)conn->state; 257 LWIP_ASSERT("pcb mismatch", conn->inner_conn == inner_conn); 258 LWIP_UNUSED_ARG(inner_conn); /* for LWIP_NOASSERT */ 259 if (!state || !(state->flags & ALTCP_PROXYCONNECT_FLAGS_HANDSHAKE_DONE)) { 260 /* @todo: do something here? */ 261 return ERR_OK; 262 } 263 /* pass this on to upper sent */ 264 if (conn->sent) { 265 return conn->sent(conn->arg, conn, len); 266 } 267 } 268 return ERR_OK; 269 } 270 271 /** Poll callback from lower connection (i.e. TCP) 272 * Just pass this on to the application. 273 * @todo: retry sending? 274 */ 275 static err_t 276 altcp_proxyconnect_lower_poll(void *arg, struct altcp_pcb *inner_conn) 277 { 278 struct altcp_pcb *conn = (struct altcp_pcb *)arg; 279 if (conn) { 280 LWIP_ASSERT("pcb mismatch", conn->inner_conn == inner_conn); 281 LWIP_UNUSED_ARG(inner_conn); /* for LWIP_NOASSERT */ 282 if (conn->poll) { 283 return conn->poll(conn->arg, conn); 284 } 285 } 286 return ERR_OK; 287 } 288 289 static void 290 altcp_proxyconnect_lower_err(void *arg, err_t err) 291 { 292 struct altcp_pcb *conn = (struct altcp_pcb *)arg; 293 if (conn) { 294 conn->inner_conn = NULL; /* already freed */ 295 if (conn->err) { 296 conn->err(conn->arg, err); 297 } 298 altcp_free(conn); 299 } 300 } 301 302 303 /* setup functions */ 304 305 static void 306 altcp_proxyconnect_setup_callbacks(struct altcp_pcb *conn, struct altcp_pcb *inner_conn) 307 { 308 altcp_arg(inner_conn, conn); 309 altcp_recv(inner_conn, altcp_proxyconnect_lower_recv); 310 altcp_sent(inner_conn, altcp_proxyconnect_lower_sent); 311 altcp_err(inner_conn, altcp_proxyconnect_lower_err); 312 /* tcp_poll is set when interval is set by application */ 313 /* listen is set totally different :-) */ 314 } 315 316 static err_t 317 altcp_proxyconnect_setup(struct altcp_proxyconnect_config *config, struct altcp_pcb *conn, struct altcp_pcb *inner_conn) 318 { 319 altcp_proxyconnect_state_t *state; 320 if (!config) { 321 return ERR_ARG; 322 } 323 LWIP_ASSERT("invalid inner_conn", conn != inner_conn); 324 325 /* allocate proxyconnect context */ 326 state = altcp_proxyconnect_state_alloc(); 327 if (state == NULL) { 328 return ERR_MEM; 329 } 330 state->flags = 0; 331 state->conf = config; 332 altcp_proxyconnect_setup_callbacks(conn, inner_conn); 333 conn->inner_conn = inner_conn; 334 conn->fns = &altcp_proxyconnect_functions; 335 conn->state = state; 336 return ERR_OK; 337 } 338 339 /** Allocate a new altcp layer connecting through a proxy. 340 * This function gets the inner pcb passed. 341 * 342 * @param config struct altcp_proxyconnect_config that contains the proxy settings 343 * @param inner_pcb pcb that makes the connection to the proxy (i.e. tcp pcb) 344 */ 345 struct altcp_pcb * 346 altcp_proxyconnect_new(struct altcp_proxyconnect_config *config, struct altcp_pcb *inner_pcb) 347 { 348 struct altcp_pcb *ret; 349 if (inner_pcb == NULL) { 350 return NULL; 351 } 352 ret = altcp_alloc(); 353 if (ret != NULL) { 354 if (altcp_proxyconnect_setup(config, ret, inner_pcb) != ERR_OK) { 355 altcp_free(ret); 356 return NULL; 357 } 358 } 359 return ret; 360 } 361 362 /** Allocate a new altcp layer connecting through a proxy. 363 * This function allocates the inner pcb as tcp pcb, resulting in a direct tcp 364 * connection to the proxy. 365 * 366 * @param config struct altcp_proxyconnect_config that contains the proxy settings 367 * @param ip_type IP type of the connection (@ref lwip_ip_addr_type) 368 */ 369 struct altcp_pcb * 370 altcp_proxyconnect_new_tcp(struct altcp_proxyconnect_config *config, u8_t ip_type) 371 { 372 struct altcp_pcb *inner_pcb, *ret; 373 374 /* inner pcb is tcp */ 375 inner_pcb = altcp_tcp_new_ip_type(ip_type); 376 if (inner_pcb == NULL) { 377 return NULL; 378 } 379 ret = altcp_proxyconnect_new(config, inner_pcb); 380 if (ret == NULL) { 381 altcp_close(inner_pcb); 382 } 383 return ret; 384 } 385 386 /** Allocator function to allocate a proxy connect altcp pcb connecting directly 387 * via tcp to the proxy. 388 * 389 * The returned pcb is a chain: altcp_proxyconnect - altcp_tcp - tcp pcb 390 * 391 * This function is meant for use with @ref altcp_new. 392 * 393 * @param arg struct altcp_proxyconnect_config that contains the proxy settings 394 * @param ip_type IP type of the connection (@ref lwip_ip_addr_type) 395 */ 396 struct altcp_pcb * 397 altcp_proxyconnect_alloc(void *arg, u8_t ip_type) 398 { 399 return altcp_proxyconnect_new_tcp((struct altcp_proxyconnect_config *)arg, ip_type); 400 } 401 402 403 #if LWIP_ALTCP_TLS 404 405 /** Allocator function to allocate a TLS connection through a proxy. 406 * 407 * The returned pcb is a chain: altcp_tls - altcp_proxyconnect - altcp_tcp - tcp pcb 408 * 409 * This function is meant for use with @ref altcp_new. 410 * 411 * @param arg struct altcp_proxyconnect_tls_config that contains the proxy settings 412 * and tls settings 413 * @param ip_type IP type of the connection (@ref lwip_ip_addr_type) 414 */ 415 struct altcp_pcb * 416 altcp_proxyconnect_tls_alloc(void *arg, u8_t ip_type) 417 { 418 struct altcp_proxyconnect_tls_config *cfg = (struct altcp_proxyconnect_tls_config *)arg; 419 struct altcp_pcb *proxy_pcb; 420 struct altcp_pcb *tls_pcb; 421 422 proxy_pcb = altcp_proxyconnect_new_tcp(&cfg->proxy, ip_type); 423 tls_pcb = altcp_tls_wrap(cfg->tls_config, proxy_pcb); 424 425 if (tls_pcb == NULL) { 426 altcp_close(proxy_pcb); 427 } 428 return tls_pcb; 429 } 430 #endif /* LWIP_ALTCP_TLS */ 431 432 /* "virtual" functions */ 433 static void 434 altcp_proxyconnect_set_poll(struct altcp_pcb *conn, u8_t interval) 435 { 436 if (conn != NULL) { 437 altcp_poll(conn->inner_conn, altcp_proxyconnect_lower_poll, interval); 438 } 439 } 440 441 static void 442 altcp_proxyconnect_recved(struct altcp_pcb *conn, u16_t len) 443 { 444 altcp_proxyconnect_state_t *state; 445 if (conn == NULL) { 446 return; 447 } 448 state = (altcp_proxyconnect_state_t *)conn->state; 449 if (state == NULL) { 450 return; 451 } 452 if (!(state->flags & ALTCP_PROXYCONNECT_FLAGS_HANDSHAKE_DONE)) { 453 return; 454 } 455 altcp_recved(conn->inner_conn, len); 456 } 457 458 static err_t 459 altcp_proxyconnect_connect(struct altcp_pcb *conn, const ip_addr_t *ipaddr, u16_t port, altcp_connected_fn connected) 460 { 461 altcp_proxyconnect_state_t *state; 462 463 if ((conn == NULL) || (ipaddr == NULL)) { 464 return ERR_VAL; 465 } 466 state = (altcp_proxyconnect_state_t *)conn->state; 467 if (state == NULL) { 468 return ERR_VAL; 469 } 470 if (state->flags & ALTCP_PROXYCONNECT_FLAGS_CONNECT_STARTED) { 471 return ERR_VAL; 472 } 473 state->flags |= ALTCP_PROXYCONNECT_FLAGS_CONNECT_STARTED; 474 475 conn->connected = connected; 476 /* connect to our proxy instead, but store the requested address and port */ 477 ip_addr_copy(state->outer_addr, *ipaddr); 478 state->outer_port = port; 479 480 return altcp_connect(conn->inner_conn, &state->conf->proxy_addr, state->conf->proxy_port, altcp_proxyconnect_lower_connected); 481 } 482 483 static struct altcp_pcb * 484 altcp_proxyconnect_listen(struct altcp_pcb *conn, u8_t backlog, err_t *err) 485 { 486 LWIP_UNUSED_ARG(conn); 487 LWIP_UNUSED_ARG(backlog); 488 LWIP_UNUSED_ARG(err); 489 /* listen not supported! */ 490 return NULL; 491 } 492 493 static void 494 altcp_proxyconnect_abort(struct altcp_pcb *conn) 495 { 496 if (conn != NULL) { 497 if (conn->inner_conn != NULL) { 498 altcp_abort(conn->inner_conn); 499 } 500 altcp_free(conn); 501 } 502 } 503 504 static err_t 505 altcp_proxyconnect_close(struct altcp_pcb *conn) 506 { 507 if (conn == NULL) { 508 return ERR_VAL; 509 } 510 if (conn->inner_conn != NULL) { 511 err_t err = altcp_close(conn->inner_conn); 512 if (err != ERR_OK) { 513 /* closing inner conn failed, return the error */ 514 return err; 515 } 516 } 517 /* no inner conn or closing it succeeded, deallocate myself */ 518 altcp_free(conn); 519 return ERR_OK; 520 } 521 522 static err_t 523 altcp_proxyconnect_write(struct altcp_pcb *conn, const void *dataptr, u16_t len, u8_t apiflags) 524 { 525 altcp_proxyconnect_state_t *state; 526 527 LWIP_UNUSED_ARG(apiflags); 528 529 if (conn == NULL) { 530 return ERR_VAL; 531 } 532 533 state = (altcp_proxyconnect_state_t *)conn->state; 534 if (state == NULL) { 535 /* @todo: which error? */ 536 return ERR_CLSD; 537 } 538 if (!(state->flags & ALTCP_PROXYCONNECT_FLAGS_HANDSHAKE_DONE)) { 539 /* @todo: which error? */ 540 return ERR_VAL; 541 } 542 return altcp_write(conn->inner_conn, dataptr, len, apiflags); 543 } 544 545 static void 546 altcp_proxyconnect_dealloc(struct altcp_pcb *conn) 547 { 548 /* clean up and free tls state */ 549 if (conn) { 550 altcp_proxyconnect_state_t *state = (altcp_proxyconnect_state_t *)conn->state; 551 if (state) { 552 altcp_proxyconnect_state_free(state); 553 conn->state = NULL; 554 } 555 } 556 } 557 const struct altcp_functions altcp_proxyconnect_functions = { 558 altcp_proxyconnect_set_poll, 559 altcp_proxyconnect_recved, 560 altcp_default_bind, 561 altcp_proxyconnect_connect, 562 altcp_proxyconnect_listen, 563 altcp_proxyconnect_abort, 564 altcp_proxyconnect_close, 565 altcp_default_shutdown, 566 altcp_proxyconnect_write, 567 altcp_default_output, 568 altcp_default_mss, 569 altcp_default_sndbuf, 570 altcp_default_sndqueuelen, 571 altcp_default_nagle_disable, 572 altcp_default_nagle_enable, 573 altcp_default_nagle_disabled, 574 altcp_default_setprio, 575 altcp_proxyconnect_dealloc, 576 altcp_default_get_tcp_addrinfo, 577 altcp_default_get_ip, 578 altcp_default_get_port 579 #ifdef LWIP_DEBUG 580 , altcp_default_dbg_get_tcp_state 581 #endif 582 }; 583 584 #endif /* LWIP_ALTCP */ 585