• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  * @file
3  * @defgroup altcp Application layered TCP Functions
4  * @ingroup altcp_api
5  *
6  * This file contains the common functions for altcp to work.
7  * For more details see @ref altcp_api.
8  */
9 
10 /**
11  * @defgroup altcp_api Application layered TCP Introduction
12  * @ingroup callbackstyle_api
13  *
14  * Overview
15  * --------
16  * altcp (application layered TCP connection API; to be used from TCPIP thread)
17  * is an abstraction layer that prevents applications linking hard against the
18  * @ref tcp.h functions while providing the same functionality. It is used to
19  * e.g. add SSL/TLS (see LWIP_ALTCP_TLS) or proxy-connect support to an application
20  * written for the tcp callback API without that application knowing the
21  * protocol details.
22  *
23  * * This interface mimics the tcp callback API to the application while preventing
24  *   direct linking (much like virtual functions).
25  * * This way, an application can make use of other application layer protocols
26  *   on top of TCP without knowing the details (e.g. TLS, proxy connection).
27  * * This is achieved by simply including "lwip/altcp.h" instead of "lwip/tcp.h",
28  *   replacing "struct tcp_pcb" with "struct altcp_pcb" and prefixing all functions
29  *   with "altcp_" instead of "tcp_".
30  *
31  * With altcp support disabled (LWIP_ALTCP==0), applications written against the
32  * altcp API can still be compiled but are directly linked against the tcp.h
33  * callback API and then cannot use layered protocols. To minimize code changes
34  * in this case, the use of altcp_allocators is strongly suggested.
35  *
36  * Usage
37  * -----
38  * To make use of this API from an existing tcp raw API application:
39  * * Include "lwip/altcp.h" instead of "lwip/tcp.h"
40  * * Replace "struct tcp_pcb" with "struct altcp_pcb"
41  * * Prefix all called tcp API functions with "altcp_" instead of "tcp_" to link
42  *   against the altcp functions
43  * * @ref altcp_new (and @ref altcp_new_ip_type/@ref altcp_new_ip6) take
44  *   an @ref altcp_allocator_t as an argument, whereas the original tcp API
45  *   functions take no arguments.
46  * * An @ref altcp_allocator_t allocator is an object that holds a pointer to an
47  *   allocator object and a corresponding state (e.g. for TLS, the corresponding
48  *   state may hold certificates or keys). This way, the application does not
49  *   even need to know if it uses TLS or pure TCP, this is handled at runtime
50  *   by passing a specific allocator.
51  * * An application can alternatively bind hard to the altcp_tls API by calling
52  *   @ref altcp_tls_new or @ref altcp_tls_wrap.
53  * * The TLS layer is not directly implemented by lwIP, but a port to mbedTLS is
54  *   provided.
55  * * Another altcp layer is proxy-connect to use TLS behind a HTTP proxy (see
56  *   @ref altcp_proxyconnect.h)
57  *
58  * altcp_allocator_t
59  * -----------------
60  * An altcp allocator is created by the application by combining an allocator
61  * callback function and a corresponding state, e.g.:\code{.c}
62  * static const unsigned char cert[] = {0x2D, ... (see mbedTLS doc for how to create this)};
63  * struct altcp_tls_config * conf = altcp_tls_create_config_client(cert, sizeof(cert));
64  * altcp_allocator_t tls_allocator = {
65  *   altcp_tls_alloc, conf
66  * };
67  * \endcode
68  *
69  *
70  * struct altcp_tls_config
71  * -----------------------
72  * The struct altcp_tls_config holds state that is needed to create new TLS client
73  * or server connections (e.g. certificates and private keys).
74  *
75  * It is not defined by lwIP itself but by the TLS port (e.g. altcp_tls to mbedTLS
76  * adaption). However, the parameters used to create it are defined in @ref
77  * altcp_tls.h (see @ref altcp_tls_create_config_server_privkey_cert for servers
78  * and @ref altcp_tls_create_config_client/@ref altcp_tls_create_config_client_2wayauth
79  * for clients).
80  *
81  * For mbedTLS, ensure that certificates can be parsed by 'mbedtls_x509_crt_parse()' and
82  * private keys can be parsed by 'mbedtls_pk_parse_key()'.
83  */
84 
85 /*
86  * Copyright (c) 2017 Simon Goldschmidt
87  * All rights reserved.
88  *
89  * Redistribution and use in source and binary forms, with or without modification,
90  * are permitted provided that the following conditions are met:
91  *
92  * 1. Redistributions of source code must retain the above copyright notice,
93  *    this list of conditions and the following disclaimer.
94  * 2. Redistributions in binary form must reproduce the above copyright notice,
95  *    this list of conditions and the following disclaimer in the documentation
96  *    and/or other materials provided with the distribution.
97  * 3. The name of the author may not be used to endorse or promote products
98  *    derived from this software without specific prior written permission.
99  *
100  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
101  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
102  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
103  * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
104  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
105  * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
106  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
107  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
108  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
109  * OF SUCH DAMAGE.
110  *
111  * This file is part of the lwIP TCP/IP stack.
112  *
113  * Author: Simon Goldschmidt <goldsimon@gmx.de>
114  *
115  */
116 
117 #include "lwip/opt.h"
118 
119 #if LWIP_ALTCP /* don't build if not configured for use in lwipopts.h */
120 
121 #include "lwip/altcp.h"
122 #include "lwip/priv/altcp_priv.h"
123 #include "lwip/altcp_tcp.h"
124 #include "lwip/tcp.h"
125 #include "lwip/mem.h"
126 
127 #include <string.h>
128 
129 extern const struct altcp_functions altcp_tcp_functions;
130 
131 /**
132  * For altcp layer implementations only: allocate a new struct altcp_pcb from the pool
133  * and zero the memory
134  */
135 struct altcp_pcb *
altcp_alloc(void)136 altcp_alloc(void)
137 {
138   struct altcp_pcb *ret = (struct altcp_pcb *)memp_malloc(MEMP_ALTCP_PCB);
139   if (ret != NULL) {
140     memset(ret, 0, sizeof(struct altcp_pcb));
141   }
142   return ret;
143 }
144 
145 /**
146  * For altcp layer implementations only: return a struct altcp_pcb to the pool
147  */
148 void
altcp_free(struct altcp_pcb * conn)149 altcp_free(struct altcp_pcb *conn)
150 {
151   if (conn) {
152     if (conn->fns && conn->fns->dealloc) {
153       conn->fns->dealloc(conn);
154     }
155     memp_free(MEMP_ALTCP_PCB, conn);
156   }
157 }
158 
159 /**
160  * @ingroup altcp
161  * altcp_new_ip6: @ref altcp_new for IPv6
162  */
163 struct altcp_pcb *
altcp_new_ip6(altcp_allocator_t * allocator)164 altcp_new_ip6(altcp_allocator_t *allocator)
165 {
166   return altcp_new_ip_type(allocator, IPADDR_TYPE_V6);
167 }
168 
169 /**
170  * @ingroup altcp
171  * altcp_new: @ref altcp_new for IPv4
172  */
173 struct altcp_pcb *
altcp_new(altcp_allocator_t * allocator)174 altcp_new(altcp_allocator_t *allocator)
175 {
176   return altcp_new_ip_type(allocator, IPADDR_TYPE_V4);
177 }
178 
179 /**
180  * @ingroup altcp
181  * altcp_new_ip_type: called by applications to allocate a new pcb with the help of an
182  * allocator function.
183  *
184  * @param allocator allocator function and argument
185  * @param ip_type IP version of the pcb (@ref lwip_ip_addr_type)
186  * @return a new altcp_pcb or NULL on error
187  */
188 struct altcp_pcb *
altcp_new_ip_type(altcp_allocator_t * allocator,u8_t ip_type)189 altcp_new_ip_type(altcp_allocator_t *allocator, u8_t ip_type)
190 {
191   struct altcp_pcb *conn;
192   if (allocator == NULL) {
193     /* no allocator given, create a simple TCP connection */
194     return altcp_tcp_new_ip_type(ip_type);
195   }
196   if (allocator->alloc == NULL) {
197     /* illegal allocator */
198     return NULL;
199   }
200   conn = allocator->alloc(allocator->arg, ip_type);
201   if (conn == NULL) {
202     /* allocation failed */
203     return NULL;
204   }
205   return conn;
206 }
207 
208 /**
209  * @ingroup altcp
210  * @see tcp_arg()
211  */
212 void
altcp_arg(struct altcp_pcb * conn,void * arg)213 altcp_arg(struct altcp_pcb *conn, void *arg)
214 {
215   if (conn) {
216     conn->arg = arg;
217   }
218 }
219 
220 /**
221  * @ingroup altcp
222  * @see tcp_accept()
223  */
224 void
altcp_accept(struct altcp_pcb * conn,altcp_accept_fn accept)225 altcp_accept(struct altcp_pcb *conn, altcp_accept_fn accept)
226 {
227   if (conn != NULL) {
228     conn->accept = accept;
229   }
230 }
231 
232 /**
233  * @ingroup altcp
234  * @see tcp_recv()
235  */
236 void
altcp_recv(struct altcp_pcb * conn,altcp_recv_fn recv)237 altcp_recv(struct altcp_pcb *conn, altcp_recv_fn recv)
238 {
239   if (conn) {
240     conn->recv = recv;
241   }
242 }
243 
244 /**
245  * @ingroup altcp
246  * @see tcp_sent()
247  */
248 void
altcp_sent(struct altcp_pcb * conn,altcp_sent_fn sent)249 altcp_sent(struct altcp_pcb *conn, altcp_sent_fn sent)
250 {
251   if (conn) {
252     conn->sent = sent;
253   }
254 }
255 
256 /**
257  * @ingroup altcp
258  * @see tcp_poll()
259  */
260 void
altcp_poll(struct altcp_pcb * conn,altcp_poll_fn poll,u8_t interval)261 altcp_poll(struct altcp_pcb *conn, altcp_poll_fn poll, u8_t interval)
262 {
263   if (conn) {
264     conn->poll = poll;
265     conn->pollinterval = interval;
266     if (conn->fns && conn->fns->set_poll) {
267       conn->fns->set_poll(conn, interval);
268     }
269   }
270 }
271 
272 /**
273  * @ingroup altcp
274  * @see tcp_err()
275  */
276 void
altcp_err(struct altcp_pcb * conn,altcp_err_fn err)277 altcp_err(struct altcp_pcb *conn, altcp_err_fn err)
278 {
279   if (conn) {
280     conn->err = err;
281   }
282 }
283 
284 /* Generic functions calling the "virtual" ones */
285 
286 /**
287  * @ingroup altcp
288  * @see tcp_recved()
289  */
290 void
altcp_recved(struct altcp_pcb * conn,u16_t len)291 altcp_recved(struct altcp_pcb *conn, u16_t len)
292 {
293   if (conn && conn->fns && conn->fns->recved) {
294     conn->fns->recved(conn, len);
295   }
296 }
297 
298 /**
299  * @ingroup altcp
300  * @see tcp_bind()
301  */
302 err_t
altcp_bind(struct altcp_pcb * conn,const ip_addr_t * ipaddr,u16_t port)303 altcp_bind(struct altcp_pcb *conn, const ip_addr_t *ipaddr, u16_t port)
304 {
305   if (conn && conn->fns && conn->fns->bind) {
306     return conn->fns->bind(conn, ipaddr, port);
307   }
308   return ERR_VAL;
309 }
310 
311 /**
312  * @ingroup altcp
313  * @see tcp_connect()
314  */
315 err_t
altcp_connect(struct altcp_pcb * conn,const ip_addr_t * ipaddr,u16_t port,altcp_connected_fn connected)316 altcp_connect(struct altcp_pcb *conn, const ip_addr_t *ipaddr, u16_t port, altcp_connected_fn connected)
317 {
318   if (conn && conn->fns && conn->fns->connect) {
319     return conn->fns->connect(conn, ipaddr, port, connected);
320   }
321   return ERR_VAL;
322 }
323 
324 /**
325  * @ingroup altcp
326  * @see tcp_listen_with_backlog_and_err()
327  */
328 struct altcp_pcb *
altcp_listen_with_backlog_and_err(struct altcp_pcb * conn,u8_t backlog,err_t * err)329 altcp_listen_with_backlog_and_err(struct altcp_pcb *conn, u8_t backlog, err_t *err)
330 {
331   if (conn && conn->fns && conn->fns->listen) {
332     return conn->fns->listen(conn, backlog, err);
333   }
334   return NULL;
335 }
336 
337 /**
338  * @ingroup altcp
339  * @see tcp_abort()
340  */
341 void
altcp_abort(struct altcp_pcb * conn)342 altcp_abort(struct altcp_pcb *conn)
343 {
344   if (conn && conn->fns && conn->fns->abort) {
345     conn->fns->abort(conn);
346   }
347 }
348 
349 /**
350  * @ingroup altcp
351  * @see tcp_close()
352  */
353 err_t
altcp_close(struct altcp_pcb * conn)354 altcp_close(struct altcp_pcb *conn)
355 {
356   if (conn && conn->fns && conn->fns->close) {
357     return conn->fns->close(conn);
358   }
359   return ERR_VAL;
360 }
361 
362 /**
363  * @ingroup altcp
364  * @see tcp_shutdown()
365  */
366 err_t
altcp_shutdown(struct altcp_pcb * conn,int shut_rx,int shut_tx)367 altcp_shutdown(struct altcp_pcb *conn, int shut_rx, int shut_tx)
368 {
369   if (conn && conn->fns && conn->fns->shutdown) {
370     return conn->fns->shutdown(conn, shut_rx, shut_tx);
371   }
372   return ERR_VAL;
373 }
374 
375 /**
376  * @ingroup altcp
377  * @see tcp_write()
378  */
379 err_t
altcp_write(struct altcp_pcb * conn,const void * dataptr,u16_t len,u8_t apiflags)380 altcp_write(struct altcp_pcb *conn, const void *dataptr, u16_t len, u8_t apiflags)
381 {
382   if (conn && conn->fns && conn->fns->write) {
383     return conn->fns->write(conn, dataptr, len, apiflags);
384   }
385   return ERR_VAL;
386 }
387 
388 /**
389  * @ingroup altcp
390  * @see tcp_output()
391  */
392 err_t
altcp_output(struct altcp_pcb * conn)393 altcp_output(struct altcp_pcb *conn)
394 {
395   if (conn && conn->fns && conn->fns->output) {
396     return conn->fns->output(conn);
397   }
398   return ERR_VAL;
399 }
400 
401 /**
402  * @ingroup altcp
403  * @see tcp_mss()
404  */
405 u16_t
altcp_mss(struct altcp_pcb * conn)406 altcp_mss(struct altcp_pcb *conn)
407 {
408   if (conn && conn->fns && conn->fns->mss) {
409     return conn->fns->mss(conn);
410   }
411   return 0;
412 }
413 
414 /**
415  * @ingroup altcp
416  * @see tcp_sndbuf()
417  */
418 u16_t
altcp_sndbuf(struct altcp_pcb * conn)419 altcp_sndbuf(struct altcp_pcb *conn)
420 {
421   if (conn && conn->fns && conn->fns->sndbuf) {
422     return conn->fns->sndbuf(conn);
423   }
424   return 0;
425 }
426 
427 /**
428  * @ingroup altcp
429  * @see tcp_sndqueuelen()
430  */
431 u16_t
altcp_sndqueuelen(struct altcp_pcb * conn)432 altcp_sndqueuelen(struct altcp_pcb *conn)
433 {
434   if (conn && conn->fns && conn->fns->sndqueuelen) {
435     return conn->fns->sndqueuelen(conn);
436   }
437   return 0;
438 }
439 
440 void
altcp_nagle_disable(struct altcp_pcb * conn)441 altcp_nagle_disable(struct altcp_pcb *conn)
442 {
443   if (conn && conn->fns && conn->fns->nagle_disable) {
444     conn->fns->nagle_disable(conn);
445   }
446 }
447 
448 void
altcp_nagle_enable(struct altcp_pcb * conn)449 altcp_nagle_enable(struct altcp_pcb *conn)
450 {
451   if (conn && conn->fns && conn->fns->nagle_enable) {
452     conn->fns->nagle_enable(conn);
453   }
454 }
455 
456 int
altcp_nagle_disabled(struct altcp_pcb * conn)457 altcp_nagle_disabled(struct altcp_pcb *conn)
458 {
459   if (conn && conn->fns && conn->fns->nagle_disabled) {
460     return conn->fns->nagle_disabled(conn);
461   }
462   return 0;
463 }
464 
465 /**
466  * @ingroup altcp
467  * @see tcp_setprio()
468  */
469 void
altcp_setprio(struct altcp_pcb * conn,u8_t prio)470 altcp_setprio(struct altcp_pcb *conn, u8_t prio)
471 {
472   if (conn && conn->fns && conn->fns->setprio) {
473     conn->fns->setprio(conn, prio);
474   }
475 }
476 
477 err_t
altcp_get_tcp_addrinfo(struct altcp_pcb * conn,int local,ip_addr_t * addr,u16_t * port)478 altcp_get_tcp_addrinfo(struct altcp_pcb *conn, int local, ip_addr_t *addr, u16_t *port)
479 {
480   if (conn && conn->fns && conn->fns->addrinfo) {
481     return conn->fns->addrinfo(conn, local, addr, port);
482   }
483   return ERR_VAL;
484 }
485 
486 ip_addr_t *
altcp_get_ip(struct altcp_pcb * conn,int local)487 altcp_get_ip(struct altcp_pcb *conn, int local)
488 {
489   if (conn && conn->fns && conn->fns->getip) {
490     return conn->fns->getip(conn, local);
491   }
492   return NULL;
493 }
494 
495 u16_t
altcp_get_port(struct altcp_pcb * conn,int local)496 altcp_get_port(struct altcp_pcb *conn, int local)
497 {
498   if (conn && conn->fns && conn->fns->getport) {
499     return conn->fns->getport(conn, local);
500   }
501   return 0;
502 }
503 
504 #ifdef LWIP_DEBUG
505 enum tcp_state
altcp_dbg_get_tcp_state(struct altcp_pcb * conn)506 altcp_dbg_get_tcp_state(struct altcp_pcb *conn)
507 {
508   if (conn && conn->fns && conn->fns->dbg_get_tcp_state) {
509     return conn->fns->dbg_get_tcp_state(conn);
510   }
511   return CLOSED;
512 }
513 #endif
514 
515 /* Default implementations for the "virtual" functions */
516 
517 void
altcp_default_set_poll(struct altcp_pcb * conn,u8_t interval)518 altcp_default_set_poll(struct altcp_pcb *conn, u8_t interval)
519 {
520   if (conn && conn->inner_conn) {
521     altcp_poll(conn->inner_conn, conn->poll, interval);
522   }
523 }
524 
525 void
altcp_default_recved(struct altcp_pcb * conn,u16_t len)526 altcp_default_recved(struct altcp_pcb *conn, u16_t len)
527 {
528   if (conn && conn->inner_conn) {
529     altcp_recved(conn->inner_conn, len);
530   }
531 }
532 
533 err_t
altcp_default_bind(struct altcp_pcb * conn,const ip_addr_t * ipaddr,u16_t port)534 altcp_default_bind(struct altcp_pcb *conn, const ip_addr_t *ipaddr, u16_t port)
535 {
536   if (conn && conn->inner_conn) {
537     return altcp_bind(conn->inner_conn, ipaddr, port);
538   }
539   return ERR_VAL;
540 }
541 
542 err_t
altcp_default_shutdown(struct altcp_pcb * conn,int shut_rx,int shut_tx)543 altcp_default_shutdown(struct altcp_pcb *conn, int shut_rx, int shut_tx)
544 {
545   if (conn) {
546     if (shut_rx && shut_tx && conn->fns && conn->fns->close) {
547       /* default shutdown for both sides is close */
548       return conn->fns->close(conn);
549     }
550     if (conn->inner_conn) {
551       return altcp_shutdown(conn->inner_conn, shut_rx, shut_tx);
552     }
553   }
554   return ERR_VAL;
555 }
556 
557 err_t
altcp_default_write(struct altcp_pcb * conn,const void * dataptr,u16_t len,u8_t apiflags)558 altcp_default_write(struct altcp_pcb *conn, const void *dataptr, u16_t len, u8_t apiflags)
559 {
560   if (conn && conn->inner_conn) {
561     return altcp_write(conn->inner_conn, dataptr, len, apiflags);
562   }
563   return ERR_VAL;
564 }
565 
566 err_t
altcp_default_output(struct altcp_pcb * conn)567 altcp_default_output(struct altcp_pcb *conn)
568 {
569   if (conn && conn->inner_conn) {
570     return altcp_output(conn->inner_conn);
571   }
572   return ERR_VAL;
573 }
574 
575 u16_t
altcp_default_mss(struct altcp_pcb * conn)576 altcp_default_mss(struct altcp_pcb *conn)
577 {
578   if (conn && conn->inner_conn) {
579     return altcp_mss(conn->inner_conn);
580   }
581   return 0;
582 }
583 
584 u16_t
altcp_default_sndbuf(struct altcp_pcb * conn)585 altcp_default_sndbuf(struct altcp_pcb *conn)
586 {
587   if (conn && conn->inner_conn) {
588     return altcp_sndbuf(conn->inner_conn);
589   }
590   return 0;
591 }
592 
593 u16_t
altcp_default_sndqueuelen(struct altcp_pcb * conn)594 altcp_default_sndqueuelen(struct altcp_pcb *conn)
595 {
596   if (conn && conn->inner_conn) {
597     return altcp_sndqueuelen(conn->inner_conn);
598   }
599   return 0;
600 }
601 
602 void
altcp_default_nagle_disable(struct altcp_pcb * conn)603 altcp_default_nagle_disable(struct altcp_pcb *conn)
604 {
605   if (conn && conn->inner_conn) {
606     altcp_nagle_disable(conn->inner_conn);
607   }
608 }
609 
610 void
altcp_default_nagle_enable(struct altcp_pcb * conn)611 altcp_default_nagle_enable(struct altcp_pcb *conn)
612 {
613   if (conn && conn->inner_conn) {
614     altcp_nagle_enable(conn->inner_conn);
615   }
616 }
617 
618 int
altcp_default_nagle_disabled(struct altcp_pcb * conn)619 altcp_default_nagle_disabled(struct altcp_pcb *conn)
620 {
621   if (conn && conn->inner_conn) {
622     return altcp_nagle_disabled(conn->inner_conn);
623   }
624   return 0;
625 }
626 
627 void
altcp_default_setprio(struct altcp_pcb * conn,u8_t prio)628 altcp_default_setprio(struct altcp_pcb *conn, u8_t prio)
629 {
630   if (conn && conn->inner_conn) {
631     altcp_setprio(conn->inner_conn, prio);
632   }
633 }
634 
635 void
altcp_default_dealloc(struct altcp_pcb * conn)636 altcp_default_dealloc(struct altcp_pcb *conn)
637 {
638   LWIP_UNUSED_ARG(conn);
639   /* nothing to do */
640 }
641 
642 err_t
altcp_default_get_tcp_addrinfo(struct altcp_pcb * conn,int local,ip_addr_t * addr,u16_t * port)643 altcp_default_get_tcp_addrinfo(struct altcp_pcb *conn, int local, ip_addr_t *addr, u16_t *port)
644 {
645   if (conn && conn->inner_conn) {
646     return altcp_get_tcp_addrinfo(conn->inner_conn, local, addr, port);
647   }
648   return ERR_VAL;
649 }
650 
651 ip_addr_t *
altcp_default_get_ip(struct altcp_pcb * conn,int local)652 altcp_default_get_ip(struct altcp_pcb *conn, int local)
653 {
654   if (conn && conn->inner_conn) {
655     return altcp_get_ip(conn->inner_conn, local);
656   }
657   return NULL;
658 }
659 
660 u16_t
altcp_default_get_port(struct altcp_pcb * conn,int local)661 altcp_default_get_port(struct altcp_pcb *conn, int local)
662 {
663   if (conn && conn->inner_conn) {
664     return altcp_get_port(conn->inner_conn, local);
665   }
666   return 0;
667 }
668 
669 #ifdef LWIP_DEBUG
670 enum tcp_state
altcp_default_dbg_get_tcp_state(struct altcp_pcb * conn)671 altcp_default_dbg_get_tcp_state(struct altcp_pcb *conn)
672 {
673   if (conn && conn->inner_conn) {
674     return altcp_dbg_get_tcp_state(conn->inner_conn);
675   }
676   return CLOSED;
677 }
678 #endif
679 
680 
681 #endif /* LWIP_ALTCP */
682