欢迎访问 生活随笔!

生活随笔

当前位置: 首页 > 编程资源 > 编程问答 >内容正文

编程问答

LwIP之协议栈接口

发布时间:2025/3/15 编程问答 48 豆豆
生活随笔 收集整理的这篇文章主要介绍了 LwIP之协议栈接口 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

先了解应用层和协议栈是怎么交互消息的

先看内核消息数据结构

/* tcp/ip内核消息类型 */ enum tcpip_msg_type {TCPIP_MSG_API, //调用APITCPIP_MSG_INPKT, //底层数据包输入TCPIP_MSG_TIMEOUT, //注册一个定时事件TCPIP_MSG_UNTIMEOUT, //删除一个定时事件TCPIP_MSG_CALLBACK, //上层回调函数TCPIP_MSG_CALLBACK_STATIC //上层回调函数(消息为静态) };/* tcp/ip内核消息 */ struct tcpip_msg {enum tcpip_msg_type type; //消息类型sys_sem_t *sem; //信号量指针union {/* API消息 */struct api_msg *apimsg;/* 底层数据包输入消息 */struct {struct pbuf *p; //数据包pbuf指针struct netif *netif; //网络接口指针}inp;/* 上层回调函数消息 */struct {tcpip_callback_fn function; //回调函数void *ctx; //回调函数参数}cb;/* 定时事件 */struct {u32_t msecs; //时间sys_timeout_handler h; //回调函数void *arg; //回调函数参数}tmo;}msg; };

在看一下,内核是怎么分类处理应用层消息的

static tcpip_init_done_fn tcpip_init_done; static void *tcpip_init_done_arg; static sys_mbox_t mbox;/* tcp/ip线程 */ static void tcpip_thread(void *arg) {struct tcpip_msg *msg;/* 用户自定义初始化函数 */if(tcpip_init_done != NULL) {tcpip_init_done(tcpip_init_done_arg);}while(1) {/* 所有的超时事件和邮箱消息都在这里处理 */sys_timeouts_mbox_fetch(&mbox, (void **)&msg);/* 接收到消息 */switch(msg->type) {/* API消息 */case TCPIP_MSG_API:/* 调用API */msg->msg.apimsg->function(&(msg->msg.apimsg->msg));break;/* 底层数据包输入 */case TCPIP_MSG_INPKT:/* 以太网设备 *//* 支持ARP协议 */if(msg->msg.inp.netif->flags & (NETIF_FLAG_ETHARP | NETIF_FLAG_ETHERNET)) {/* 以太网数据包输入处理 */ethernet_input(msg->msg.inp.p, msg->msg.inp.netif);}/* 不支持ARP协议 */else{/* IP数据包输入处理 */ip_input(msg->msg.inp.p, msg->msg.inp.netif);}/* 释放该消息内存空间 */memp_free(MEMP_TCPIP_MSG_INPKT, msg);break;/* 注册一个定时事件 */case TCPIP_MSG_TIMEOUT:/* 启动定时事件 */sys_timeout(msg->msg.tmo.msecs, msg->msg.tmo.h, msg->msg.tmo.arg);/* 释放该消息内存空间 */memp_free(MEMP_TCPIP_MSG_API, msg);break;/* 删除一个定时事件 */case TCPIP_MSG_UNTIMEOUT:/* 取消定时事件 */sys_untimeout(msg->msg.tmo.h, msg->msg.tmo.arg);/* 释放该消息内存空间 */memp_free(MEMP_TCPIP_MSG_API, msg);break;/* 上层回调函数 */case TCPIP_MSG_CALLBACK:/* 调用回调函数 */msg->msg.cb.function(msg->msg.cb.ctx);/* 释放该消息内存空间 */memp_free(MEMP_TCPIP_MSG_API, msg);break;/* 上层回调函数(消息为静态) */case TCPIP_MSG_CALLBACK_STATIC:/* 调用回调函数 */msg->msg.cb.function(msg->msg.cb.ctx);break;default:break;}} }/* 底层数据包输入处理函数 */ err_t tcpip_input(struct pbuf *p, struct netif *inp) {struct tcpip_msg *msg;/* 邮箱无效 */if(!sys_mbox_valid(&mbox)) {return ERR_VAL;}/* 为底层数据包输入消息从内存池申请内存 */msg = (struct tcpip_msg *)memp_malloc(MEMP_TCPIP_MSG_INPKT);if(msg == NULL) {return ERR_MEM;}/* 底层数据包输入消息 */msg->type = TCPIP_MSG_INPKT;/* 数据包内容 */msg->msg.inp.p = p;/* 网络接口指针 */msg->msg.inp.netif = inp;/* 发送消息 */if(sys_mbox_trypost(&mbox, msg) != ERR_OK) {memp_free(MEMP_TCPIP_MSG_INPKT, msg);return ERR_MEM;}return ERR_OK; }/* 上层回调函数 */ err_t tcpip_callback_with_block(tcpip_callback_fn function, void *ctx, u8_t block) {struct tcpip_msg *msg;/* 邮箱有效 */if(sys_mbox_valid(&mbox)) {/* 为上层回调函数消息从内存池申请内存 */msg = (struct tcpip_msg *)memp_malloc(MEMP_TCPIP_MSG_API);if(msg == NULL) {return ERR_MEM;}/* 上层回调函数 */msg->type = TCPIP_MSG_CALLBACK;/* 回调函数 */msg->msg.cb.function = function;/* 回调函数参数 */msg->msg.cb.ctx = ctx;/* 发送消息(阻塞) */if(block) {sys_mbox_post(&mbox, msg);}/* 发送消息(非阻塞) */else {if(sys_mbox_trypost(&mbox, msg) != ERR_OK) {memp_free(MEMP_TCPIP_MSG_API, msg);return ERR_MEM;}}return ERR_OK;}return ERR_VAL; }/* 注册一个定时事件 */ err_t tcpip_timeout(u32_t msecs, sys_timeout_handler h, void *arg) {struct tcpip_msg *msg;/* 邮箱为有效 */if(sys_mbox_valid(&mbox)) {/* 为注册一个定时事件消息从内存池申请内存 */msg = (struct tcpip_msg *)memp_malloc(MEMP_TCPIP_MSG_API);if(msg == NULL) {return ERR_MEM;}/* 注册一个定时事件 */msg->type = TCPIP_MSG_TIMEOUT;/* 定时事件 */msg->msg.tmo.msecs = msecs;/* 回调函数 */msg->msg.tmo.h = h;/* 回调函数参数 */msg->msg.tmo.arg = arg;/* 发送消息 */sys_mbox_post(&mbox, msg);return ERR_OK;}return ERR_VAL; }/* 删除一个定时事件 */ err_t tcpip_untimeout(sys_timeout_handler h, void *arg) {struct tcpip_msg *msg;/* 邮箱为有效 */if(sys_mbox_valid(&mbox)) {/* 为删除一个定时事件消息从内存池申请内存 */msg = (struct tcpip_msg *)memp_malloc(MEMP_TCPIP_MSG_API);if(msg == NULL) {return ERR_MEM;}/* 删除一个定时事件 */msg->type = TCPIP_MSG_UNTIMEOUT;/* 回调函数 */msg->msg.tmo.h = h;/* 回调函数参数 */msg->msg.tmo.arg = arg;/* 发送消息 */sys_mbox_post(&mbox, msg);return ERR_OK;}return ERR_VAL; }/* API调用 */ err_t tcpip_apimsg(struct api_msg *apimsg) {struct tcpip_msg msg;/* 邮箱为有效 */if(sys_mbox_valid(&mbox)) {/* 调用API */msg.type = TCPIP_MSG_API;/* API消息 */msg.msg.apimsg = apimsg;/* 发送消息 */sys_mbox_post(&mbox, &msg);/* 等待API调用完成 */sys_arch_sem_wait(&apimsg->msg.conn->op_completed, 0);return apimsg->msg.err;}return ERR_VAL; }/* 创建上层回调函数消息 */ struct tcpip_callback_msg *tcpip_callbackmsg_new(tcpip_callback_fn function, void *ctx) {/* 为上层回调函数消息从内存池申请内存 */struct tcpip_msg *msg = (struct tcpip_msg *)memp_malloc(MEMP_TCPIP_MSG_API);if(msg == NULL) {return NULL;}/* 上层回调函数 */msg->type = TCPIP_MSG_CALLBACK_STATIC;/* 回调函数 */msg->msg.cb.function = function;/* 回调函数参数 */msg->msg.cb.ctx = ctx;/* 返回消息指针 */return (struct tcpip_callback_msg *)msg; }/* 释放上层回调函数消息 */ void tcpip_callbackmsg_delete(struct tcpip_callback_msg* msg) {memp_free(MEMP_TCPIP_MSG_API, msg); }/* 发送上层回调函数消息 */ err_t tcpip_trycallback(struct tcpip_callback_msg* msg) {if(!sys_mbox_valid(&mbox)) {return ERR_VAL;}return sys_mbox_trypost(&mbox, msg); }/* tcp/ip初始化 */ void tcpip_init(tcpip_init_done_fn initfunc, void *arg) {/* 初始化协议栈 */lwip_init();/* 自定义初始化函数 */tcpip_init_done = initfunc;tcpip_init_done_arg = arg;/* 创建邮箱 */if(sys_mbox_new(&mbox, TCPIP_MBOX_SIZE) != ERR_OK) {}/* 创建线程 */sys_thread_new(TCPIP_THREAD_NAME, tcpip_thread, NULL, TCPIP_THREAD_STACKSIZE, TCPIP_THREAD_PRIO); }/* 释放pbuf上层回调函数 */ static void pbuf_free_int(void *p) {struct pbuf *q = (struct pbuf *)p;pbuf_free(q); }/* 使用上层回调函数消息释放pbuf */ err_t pbuf_free_callback(struct pbuf *p) {/* 使用上层回调函数消息释放pbuf */return tcpip_callback_with_block(pbuf_free_int, p, 0); }/* 使用上层回调函数消息释放到内存堆 */ err_t mem_free_callback(void *m) {/* 使用上层回调函数消息释放到内存堆 */return tcpip_callback_with_block(mem_free, m, 0); }

接下来看一下内核收到消息后具体是怎么工作的

/* API消息 */ struct api_msg_msg {/* 连接结构 */struct netconn *conn;/* 错误 */err_t err;union {/* do_send时需要的参数 */struct netbuf *b;/* do_newconn时需要的参数 */struct {u8_t proto; //协议类型}n;/* do_bind和do_connect时需要的参数 */struct {ip_addr_t *ipaddr; //IP地址u16_t port; //端口号}bc;/* do_getaddr时需要的参数 */struct {ip_addr_t *ipaddr; //IP地址u16_t *port; //端口号u8_t local; //1本地IP、0远程IP}ad;/* do_write时需要的参数 */struct {const void *dataptr; //数据size_t len; //长度u8_t apiflags; //标志位}w;/* do_recv时需要的参数 */struct {u32_t len; //长度}r;/* do_close时需要的参数 */struct {u8_t shut; //关闭方向}sd;}msg; };/* API消息 */ struct api_msg {void (*function)(struct api_msg_msg *msg); //回调函数struct api_msg_msg msg; //回调函数参数 }; /* 设置是否为阻塞态 */ #define SET_NONBLOCKING_CONNECT(conn, val) do { if(val) { \(conn)->flags |= NETCONN_FLAG_IN_NONBLOCKING_CONNECT; \ } else { \(conn)->flags &= ~ NETCONN_FLAG_IN_NONBLOCKING_CONNECT; }} while(0) /* 非阻塞态 */ #define IN_NONBLOCKING_CONNECT(conn) (((conn)->flags & NETCONN_FLAG_IN_NONBLOCKING_CONNECT) != 0)static err_t do_writemore(struct netconn *conn); static void do_close_internal(struct netconn *conn);/* 原始IP接口接收回调函数 */ static u8_t recv_raw(void *arg, struct raw_pcb *pcb, struct pbuf *p, ip_addr_t *addr) {struct pbuf *q;struct netbuf *buf;struct netconn *conn;/* 参数 */conn = (struct netconn *)arg;/* 接收邮箱有效 */if((conn != NULL) && sys_mbox_valid(&conn->recvmbox)) {/* 将数据放到netbuf中 */q = pbuf_alloc(PBUF_RAW, p->tot_len, PBUF_RAM);if(q != NULL) {if(pbuf_copy(q, p) != ERR_OK) {pbuf_free(q);q = NULL;}}if(q != NULL) {u16_t len;buf = (struct netbuf *)memp_malloc(MEMP_NETBUF);if(buf == NULL) {pbuf_free(q);return 0;}buf->p = q;buf->ptr = q;ip_addr_copy(buf->addr, *ip_current_src_addr());buf->port = pcb->protocol;len = q->tot_len;/* 发送接收邮箱消息 */if(sys_mbox_trypost(&conn->recvmbox, buf) != ERR_OK) {netbuf_delete(buf);return 0;} else {/* 调用连接回调函数 */API_EVENT(conn, NETCONN_EVT_RCVPLUS, len);}}}return 0; }/* UDP接口接收回调函数 */ static void recv_udp(void *arg, struct udp_pcb *pcb, struct pbuf *p, ip_addr_t *addr, u16_t port) {struct netbuf *buf;struct netconn *conn;u16_t len;conn = (struct netconn *)arg;/* 邮箱无效 */if((conn == NULL) || !sys_mbox_valid(&conn->recvmbox)) {pbuf_free(p);return;}/* 将数据放到netbuf中 */buf = (struct netbuf *)memp_malloc(MEMP_NETBUF);if(buf == NULL) {pbuf_free(p);return;} else {buf->p = p;buf->ptr = p;ip_addr_set(&buf->addr, addr);buf->port = port;}len = p->tot_len;/* 发送接收邮箱消息 */if(sys_mbox_trypost(&conn->recvmbox, buf) != ERR_OK) {netbuf_delete(buf);return;} else {/* 调用连接回调函数 */API_EVENT(conn, NETCONN_EVT_RCVPLUS, len);} }/* 接收回调函数 */ static err_t recv_tcp(void *arg, struct tcp_pcb *pcb, struct pbuf *p, err_t err) {struct netconn *conn;u16_t len;conn = (struct netconn *)arg;if(conn == NULL) {return ERR_VAL;}/* 邮箱无效 */if(!sys_mbox_valid(&conn->recvmbox)) {if(p != NULL) {tcp_recved(pcb, p->tot_len);pbuf_free(p);}return ERR_OK;}/* 设置致命错误 */NETCONN_SET_SAFE_ERR(conn, err);if(p != NULL) {len = p->tot_len;} else {len = 0;}/* 发送接收邮箱消息 */if(sys_mbox_trypost(&conn->recvmbox, p) != ERR_OK) {return ERR_MEM;}else {/* 调用连接回调函数 */API_EVENT(conn, NETCONN_EVT_RCVPLUS, len);}return ERR_OK; }/* poll回调函数 */ static err_t poll_tcp(void *arg, struct tcp_pcb *pcb) {struct netconn *conn = (struct netconn *)arg;/* 正在发送数据 */if(conn->state == NETCONN_WRITE) {do_writemore(conn);}/* 关闭状态 */else if(conn->state == NETCONN_CLOSE) {do_close_internal(conn);}/* 需要再次检查是否可写 */if(conn->flags & NETCONN_FLAG_CHECK_WRITESPACE) {/* 判断是否可写 */if((conn->pcb.tcp != NULL) && (tcp_sndbuf(conn->pcb.tcp) > TCP_SNDLOWAT) &&(tcp_sndqueuelen(conn->pcb.tcp) < TCP_SNDQUEUELOWAT)) {conn->flags &= ~NETCONN_FLAG_CHECK_WRITESPACE;/* 调用连接回调函数 */API_EVENT(conn, NETCONN_EVT_SENDPLUS, 0);}}return ERR_OK; }/* 发送回调函数 */ static err_t sent_tcp(void *arg, struct tcp_pcb *pcb, u16_t len) {struct netconn *conn = (struct netconn *)arg;/* 正在发送数据 */if(conn->state == NETCONN_WRITE) {do_writemore(conn);} /* 关闭状态 */else if(conn->state == NETCONN_CLOSE) {do_close_internal(conn);}if(conn) {/* 需要再次检查是否可写 */if((conn->pcb.tcp != NULL) && (tcp_sndbuf(conn->pcb.tcp) > TCP_SNDLOWAT) &&(tcp_sndqueuelen(conn->pcb.tcp) < TCP_SNDQUEUELOWAT)) {conn->flags &= ~NETCONN_FLAG_CHECK_WRITESPACE;/* 调用连接回调函数 */API_EVENT(conn, NETCONN_EVT_SENDPLUS, len);}}return ERR_OK; }/* 错误回调函数 */ static void err_tcp(void *arg, err_t err) {struct netconn *conn;enum netconn_state old_state;conn = (struct netconn *)arg;conn->pcb.tcp = NULL;/* 记录错误 */conn->last_err = err;/* 复位连接 */old_state = conn->state;conn->state = NETCONN_NONE;/* 调用连接回调函数 */API_EVENT(conn, NETCONN_EVT_ERROR, 0);API_EVENT(conn, NETCONN_EVT_RCVPLUS, 0);API_EVENT(conn, NETCONN_EVT_SENDPLUS, 0);/* 发送接收邮箱消息 */if(sys_mbox_valid(&conn->recvmbox)) {sys_mbox_trypost(&conn->recvmbox, NULL);}/* 发送接受连接邮箱消息 */if(sys_mbox_valid(&conn->acceptmbox)) {sys_mbox_trypost(&conn->acceptmbox, NULL);}if((old_state == NETCONN_WRITE) || (old_state == NETCONN_CLOSE) ||(old_state == NETCONN_CONNECT)) {/* 清空阻塞态 */int was_nonblocking_connect = IN_NONBLOCKING_CONNECT(conn);SET_NONBLOCKING_CONNECT(conn, 0);/* 阻塞态 */if(!was_nonblocking_connect) {conn->current_msg->err = err;conn->current_msg = NULL;/* 释放操作完成信号量 */sys_sem_signal(&conn->op_completed);}} }/* 设置TCP回调函数 */ static void setup_tcp(struct netconn *conn) {struct tcp_pcb *pcb;pcb = conn->pcb.tcp;/* 设置用户自定义参数 */ tcp_arg(pcb, conn);/* 设置接收回调函数 */ tcp_recv(pcb, recv_tcp);/* 设置发送成功回调函数 */tcp_sent(pcb, sent_tcp);/* 设置poll回调函数和周期 */tcp_poll(pcb, poll_tcp, 4);/* 设置连接错误回调函数 */ tcp_err(pcb, err_tcp); }/* 接受连接回调函数 */ static err_t accept_function(void *arg, struct tcp_pcb *newpcb, err_t err) {struct netconn *newconn;struct netconn *conn = (struct netconn *)arg;if(!sys_mbox_valid(&conn->acceptmbox)) {return ERR_VAL;}/* 为新连接创建一个连接结构体,设置TCP回调函数 */newconn = netconn_alloc(conn->type, conn->callback);if(newconn == NULL) {return ERR_MEM;}newconn->pcb.tcp = newpcb;setup_tcp(newconn);newconn->last_err = err;/* 发送接受连接消息 */if(sys_mbox_trypost(&conn->acceptmbox, newconn) != ERR_OK) {struct tcp_pcb* pcb = newconn->pcb.tcp;tcp_arg(pcb, NULL);tcp_recv(pcb, NULL);tcp_sent(pcb, NULL);tcp_poll(pcb, NULL, 4);tcp_err(pcb, NULL);newconn->pcb.tcp = NULL;sys_mbox_free(&newconn->recvmbox);sys_mbox_set_invalid(&newconn->recvmbox);netconn_free(newconn);return ERR_MEM;} else {/* 调用连接事件回调函数 */API_EVENT(conn, NETCONN_EVT_RCVPLUS, 0);}return ERR_OK; }/* 创建控制块 */ static void pcb_new(struct api_msg_msg *msg) {LWIP_ASSERT("pcb_new: pcb already allocated", msg->conn->pcb.tcp == NULL);/* 判断协议类型 */switch(NETCONNTYPE_GROUP(msg->conn->type)) {/* 原始IP连接 */case NETCONN_RAW:/* 创建IP原始接口控制块 */msg->conn->pcb.raw = raw_new(msg->msg.n.proto);if(msg->conn->pcb.raw == NULL) {msg->err = ERR_MEM;break;}/* IP原始接口设置接收回调函数 */raw_recv(msg->conn->pcb.raw, recv_raw, msg->conn);break;/* UDP */case NETCONN_UDP:/* 创建一个UDP控制块 */msg->conn->pcb.udp = udp_new();if(msg->conn->pcb.udp == NULL) {msg->err = ERR_MEM;break;}/* 无校验UDP */if(msg->conn->type == NETCONN_UDPNOCHKSUM) {udp_setflags(msg->conn->pcb.udp, UDP_FLAGS_NOCHKSUM);}/* 设置UDP控制块回调函数 */udp_recv(msg->conn->pcb.udp, recv_udp, msg->conn);break;/* TCP */case NETCONN_TCP:/* 新建TCP控制块 */msg->conn->pcb.tcp = tcp_new();if(msg->conn->pcb.tcp == NULL) {msg->err = ERR_MEM;break;}/* 设置TCP回调函数 */setup_tcp(msg->conn);break;/* 不支持 */default:msg->err = ERR_VAL;break;} }/* 创建一个连接PCB */ void do_newconn(struct api_msg_msg *msg) {msg->err = ERR_OK;/* 控制块还未创建 */if(msg->conn->pcb.tcp == NULL) {/* 创建控制块 */pcb_new(msg);}/* 释放API调用完成信号量 */TCPIP_APIMSG_ACK(msg); }/* 创建一个连接结构体 */ struct netconn *netconn_alloc(enum netconn_type t, netconn_callback callback) {struct netconn *conn;int size;/* 为连接申请内存空间 */conn = (struct netconn *)memp_malloc(MEMP_NETCONN);if(conn == NULL) {return NULL;}/* 清空最近一个错误 */conn->last_err = ERR_OK;/* 连接类型 */conn->type = t;/* 清空控制块指针 */conn->pcb.tcp = NULL;size = DEFAULT_RAW_RECVMBOX_SIZE;/* 创建API调用完成信号量 */if(sys_sem_new(&conn->op_completed, 0) != ERR_OK) {goto free_and_return;}/* 创建接收邮箱 */if(sys_mbox_new(&conn->recvmbox, size) != ERR_OK) {sys_sem_free(&conn->op_completed);goto free_and_return;}/* 接受连接邮箱设置为无效 */ sys_mbox_set_invalid(&conn->acceptmbox);/* 不处于任何连接状态 */conn->state = NETCONN_NONE;/* 套接字描述符 */conn->socket = -1;/* 回调函数 */conn->callback = callback;/* 写数据缓存不足时,数据暂时封装在current_msg中,write_offset是下一次发送的索引 */conn->current_msg = NULL;conn->write_offset = 0;/* 清空标志位 */conn->flags = 0;return conn;free_and_return:memp_free(MEMP_NETCONN, conn);return NULL; }/* 释放连接结构体 */ void netconn_free(struct netconn *conn) {/* 删除API调用完成信号量 */sys_sem_free(&conn->op_completed);/* API信号量设置为无效 */ sys_sem_set_invalid(&conn->op_completed);/* 释放连接内存空间 */memp_free(MEMP_NETCONN, conn); }/* 删除接收邮箱和接受连接邮箱 */ static void netconn_drain(struct netconn *conn) {void *mem;struct pbuf *p;/* 接收邮箱有效 */if(sys_mbox_valid(&conn->recvmbox)) {/* 接收消息并处理 */while(sys_mbox_tryfetch(&conn->recvmbox, &mem) != SYS_MBOX_EMPTY) {/* TCP连接 */if(conn->type == NETCONN_TCP) {/* pbuf处理 */if(mem != NULL) {p = (struct pbuf *)mem;if(conn->pcb.tcp != NULL) {/* 应用层接收对方传来的数据 */tcp_recved(conn->pcb.tcp, p->tot_len);}/* 释放数据pbuf */pbuf_free(p);}}else{/* 删除netbuf */netbuf_delete((struct netbuf *)mem);}}/* 删除接收邮箱 */sys_mbox_free(&conn->recvmbox);sys_mbox_set_invalid(&conn->recvmbox);}/* 接受连接邮箱有效 */if(sys_mbox_valid(&conn->acceptmbox)) {/* 接收消息并处理 */while(sys_mbox_tryfetch(&conn->acceptmbox, &mem) != SYS_MBOX_EMPTY) {/* 新连接处理 */struct netconn *newconn = (struct netconn *)mem;if(conn->pcb.tcp != NULL) {/* 接受新连接 */tcp_accepted(conn->pcb.tcp);}/* 删除每一个新连接接收邮箱 */netconn_drain(newconn);/* 删除TCP控制块,并发送TCP复位报文 */if(newconn->pcb.tcp != NULL) {tcp_abort(newconn->pcb.tcp);newconn->pcb.tcp = NULL;}/* 释放连接结构体 */netconn_free(newconn);}/* 删除接受连接邮箱 */sys_mbox_free(&conn->acceptmbox);sys_mbox_set_invalid(&conn->acceptmbox);} }/* 关闭连接 */ static void do_close_internal(struct netconn *conn) {err_t err;u8_t shut, shut_rx, shut_tx, close;/* 关闭方向 */shut = conn->current_msg->msg.sd.shut;shut_rx = shut & NETCONN_SHUT_RD;shut_tx = shut & NETCONN_SHUT_WR;/* 关闭 */close = shut == NETCONN_SHUT_RDWR;/* 清空用户自定义参数 */ if(close) {tcp_arg(conn->pcb.tcp, NULL);}/* 侦听状态,清空接受连接回调函数 */if(conn->pcb.tcp->state == LISTEN) {tcp_accept(conn->pcb.tcp, NULL);} /* 非侦听状态 */else{/* 关闭接收方向,清空接收和接受连接回调函数 */if(shut_rx) {tcp_recv(conn->pcb.tcp, NULL);tcp_accept(conn->pcb.tcp, NULL);}/* 关闭发送方向,清空发送回调函数 */if(shut_tx) {tcp_sent(conn->pcb.tcp, NULL);}/* 清空poll和错误回调函数 */if(close) {tcp_poll(conn->pcb.tcp, NULL, 4);tcp_err(conn->pcb.tcp, NULL);}}/* 关闭连接 */if(close) {err = tcp_close(conn->pcb.tcp);} else {err = tcp_shutdown(conn->pcb.tcp, shut_rx, shut_tx);}/* 调用连接API回调函数 */if(err == ERR_OK) {conn->current_msg->err = ERR_OK;conn->current_msg = NULL;conn->state = NETCONN_NONE;if(close) {conn->pcb.tcp = NULL;API_EVENT(conn, NETCONN_EVT_ERROR, 0);}if(shut_rx) {API_EVENT(conn, NETCONN_EVT_RCVPLUS, 0);}if(shut_tx) {API_EVENT(conn, NETCONN_EVT_SENDPLUS, 0);}sys_sem_signal(&conn->op_completed);}/* 关闭失败 */else {tcp_sent(conn->pcb.tcp, sent_tcp);tcp_poll(conn->pcb.tcp, poll_tcp, 4);tcp_err(conn->pcb.tcp, err_tcp);tcp_arg(conn->pcb.tcp, conn);} }/* 删除控制块 */ void do_delconn(struct api_msg_msg *msg) {if((msg->conn->state != NETCONN_NONE) && (msg->conn->state != NETCONN_LISTEN) && (msg->conn->state != NETCONN_CONNECT)) {msg->err = ERR_INPROGRESS;} else {/* 删除接收邮箱和接受连接邮箱 */netconn_drain(msg->conn);/* 判断连接类型 */if(msg->conn->pcb.tcp != NULL) {switch(NETCONNTYPE_GROUP(msg->conn->type)) {/* 原始IP连接 */case NETCONN_RAW:/* 移除IP原始接口控制块 */raw_remove(msg->conn->pcb.raw);break;/* UDP */case NETCONN_UDP:/* 删除UDP控制块 */msg->conn->pcb.udp->recv_arg = NULL;udp_remove(msg->conn->pcb.udp);break;/* TCP */case NETCONN_TCP:msg->conn->state = NETCONN_CLOSE;msg->msg.sd.shut = NETCONN_SHUT_RDWR;msg->conn->current_msg = msg;/* 关闭连接 */do_close_internal(msg->conn);return;default:break;}msg->conn->pcb.tcp = NULL;}/* 调用连接API回调函数 */API_EVENT(msg->conn, NETCONN_EVT_RCVPLUS, 0);API_EVENT(msg->conn, NETCONN_EVT_SENDPLUS, 0);}/* 释放操作完成信号量 */if(sys_sem_valid(&msg->conn->op_completed)) {sys_sem_signal(&msg->conn->op_completed);} }/* 绑定本地端口 */ void do_bind(struct api_msg_msg *msg) {/* 根据连接类型调用绑定 */if(ERR_IS_FATAL(msg->conn->last_err)) {msg->err = msg->conn->last_err;} else {msg->err = ERR_VAL;if(msg->conn->pcb.tcp != NULL) {switch(NETCONNTYPE_GROUP(msg->conn->type)) {case NETCONN_RAW:msg->err = raw_bind(msg->conn->pcb.raw, msg->msg.bc.ipaddr);break;case NETCONN_UDP:msg->err = udp_bind(msg->conn->pcb.udp, msg->msg.bc.ipaddr, msg->msg.bc.port);break;case NETCONN_TCP:msg->err = tcp_bind(msg->conn->pcb.tcp, msg->msg.bc.ipaddr, msg->msg.bc.port);break;default:break;}}}/* 释放API调用完成信号量 */TCPIP_APIMSG_ACK(msg); }/* 连接完成 */ static err_t do_connected(void *arg, struct tcp_pcb *pcb, err_t err) {struct netconn *conn;int was_blocking;conn = (struct netconn *)arg;if(conn == NULL) {return ERR_VAL;}if(conn->current_msg != NULL) {conn->current_msg->err = err;}/* 设置TCP回调函数 */if((conn->type == NETCONN_TCP) && (err == ERR_OK)){setup_tcp(conn);}/* 是否为阻塞态 */was_blocking = !IN_NONBLOCKING_CONNECT(conn);/* 清空非阻塞态 */SET_NONBLOCKING_CONNECT(conn, 0);/* 清空消息 */conn->current_msg = NULL;/* 清空状态 */conn->state = NETCONN_NONE;if(!was_blocking) {NETCONN_SET_SAFE_ERR(conn, ERR_OK);}/* 调用连接API回调函数 */API_EVENT(conn, NETCONN_EVT_SENDPLUS, 0);/* 阻塞态,释放操作完成信号量 */if(was_blocking) {sys_sem_signal(&conn->op_completed);}return ERR_OK; }/* 连接远程端口 */ void do_connect(struct api_msg_msg *msg) {/* 根据连接类型调用连接 */if(msg->conn->pcb.tcp == NULL) {msg->err = ERR_CLSD;} else {switch(NETCONNTYPE_GROUP(msg->conn->type)) {case NETCONN_RAW:msg->err = raw_connect(msg->conn->pcb.raw, msg->msg.bc.ipaddr);break;case NETCONN_UDP:msg->err = udp_connect(msg->conn->pcb.udp, msg->msg.bc.ipaddr, msg->msg.bc.port);break;case NETCONN_TCP:if(msg->conn->state != NETCONN_NONE) {msg->err = ERR_ISCONN;} else {setup_tcp(msg->conn);msg->err = tcp_connect(msg->conn->pcb.tcp, msg->msg.bc.ipaddr, msg->msg.bc.port, do_connected);if(msg->err == ERR_OK) {u8_t non_blocking = netconn_is_nonblocking(msg->conn);msg->conn->state = NETCONN_CONNECT;SET_NONBLOCKING_CONNECT(msg->conn, non_blocking);if(non_blocking) {msg->err = ERR_INPROGRESS;} else {msg->conn->current_msg = msg;return;}}}break;default:break;}}/* 释放操作完成信号量 */sys_sem_signal(&msg->conn->op_completed); }/* 断开连接 */ void do_disconnect(struct api_msg_msg *msg) {if(NETCONNTYPE_GROUP(msg->conn->type) == NETCONN_UDP) {udp_disconnect(msg->conn->pcb.udp);msg->err = ERR_OK;} else{msg->err = ERR_VAL;}/* 释放操作完成信号量 */TCPIP_APIMSG_ACK(msg); }/* 侦听 */ void do_listen(struct api_msg_msg *msg) {if(ERR_IS_FATAL(msg->conn->last_err)) {msg->err = msg->conn->last_err;} else {msg->err = ERR_CONN;if(msg->conn->pcb.tcp != NULL) {if(msg->conn->type == NETCONN_TCP) {if(msg->conn->state == NETCONN_NONE) {struct tcp_pcb *lpcb = tcp_listen(msg->conn->pcb.tcp);/* 删除接收邮箱,创建接受连接邮箱 */if(lpcb == NULL) {msg->err = ERR_MEM;}else {if(sys_mbox_valid(&msg->conn->recvmbox)) {sys_mbox_free(&msg->conn->recvmbox);sys_mbox_set_invalid(&msg->conn->recvmbox);}msg->err = ERR_OK;if(!sys_mbox_valid(&msg->conn->acceptmbox)) {msg->err = sys_mbox_new(&msg->conn->acceptmbox, DEFAULT_ACCEPTMBOX_SIZE);}if(msg->err == ERR_OK) {msg->conn->state = NETCONN_LISTEN;msg->conn->pcb.tcp = lpcb;tcp_arg(msg->conn->pcb.tcp, msg->conn);tcp_accept(msg->conn->pcb.tcp, accept_function);} else {tcp_close(lpcb);msg->conn->pcb.tcp = NULL;}}}} else {msg->err = ERR_ARG;}}}/* 释放操作完成信号量 */TCPIP_APIMSG_ACK(msg); }/* 非TCP发送数据 */ void do_send(struct api_msg_msg *msg) {if(ERR_IS_FATAL(msg->conn->last_err)) {msg->err = msg->conn->last_err;} else {msg->err = ERR_CONN;if(msg->conn->pcb.tcp != NULL) {switch(NETCONNTYPE_GROUP(msg->conn->type)) {case NETCONN_RAW:if(ip_addr_isany(&msg->msg.b->addr)) {msg->err = raw_send(msg->conn->pcb.raw, msg->msg.b->p);} else {msg->err = raw_sendto(msg->conn->pcb.raw, msg->msg.b->p, &msg->msg.b->addr);}break;case NETCONN_UDP:if(ip_addr_isany(&msg->msg.b->addr)) {msg->err = udp_send(msg->conn->pcb.udp, msg->msg.b->p);} else {msg->err = udp_sendto(msg->conn->pcb.udp, msg->msg.b->p, &msg->msg.b->addr, msg->msg.b->port);}break;default:break;}}}/* 释放操作完成信号量 */TCPIP_APIMSG_ACK(msg); }#if LWIP_TCP /* TCP接收数据 */ void do_recv(struct api_msg_msg *msg) {msg->err = ERR_OK;if(msg->conn->pcb.tcp != NULL) {if(msg->conn->type == NETCONN_TCP) {u32_t remaining = msg->msg.r.len;do {u16_t recved = (remaining > 0xffff) ? 0xffff : (u16_t)remaining;tcp_recved(msg->conn->pcb.tcp, recved);remaining -= recved;}while(remaining != 0);}}/* 释放操作完成信号量 */TCPIP_APIMSG_ACK(msg); }/* TCP发送没发完的数据 */ static err_t do_writemore(struct netconn *conn) {err_t err;void *dataptr;u16_t len, available;u8_t write_finished = 0;size_t diff;u8_t dontblock = netconn_is_nonblocking(conn) || (conn->current_msg->msg.w.apiflags & NETCONN_DONTBLOCK);u8_t apiflags = conn->current_msg->msg.w.apiflags;{/* 判断一次能不能写完 */dataptr = (u8_t*)conn->current_msg->msg.w.dataptr + conn->write_offset;diff = conn->current_msg->msg.w.len - conn->write_offset;if(diff > 0xffffUL) {len = 0xffff;apiflags |= TCP_WRITE_FLAG_MORE;} else {len = (u16_t)diff;}available = tcp_sndbuf(conn->pcb.tcp);/* 写不完 */if(available < len) {len = available;/* 非阻塞,报错 */if(dontblock){if(!len) {err = ERR_WOULDBLOCK;goto err_mem;}} /* 阻塞,还有剩余数据 */else {apiflags |= TCP_WRITE_FLAG_MORE;}}/* 组建发送数据 */err = tcp_write(conn->pcb.tcp, dataptr, len, apiflags);if((err == ERR_OK) || (err == ERR_MEM)) { err_mem:/* 非阻塞态,并且没写完 */if(dontblock && (len < conn->current_msg->msg.w.len)) {/* 调用连接API回调函数 */API_EVENT(conn, NETCONN_EVT_SENDMINUS, len);/* 需要再次检查是否可写 */conn->flags |= NETCONN_FLAG_CHECK_WRITESPACE;} /* 发送数据成功 */else if((tcp_sndbuf(conn->pcb.tcp) <= TCP_SNDLOWAT) || (tcp_sndqueuelen(conn->pcb.tcp) >= TCP_SNDQUEUELOWAT)) {/* 调用连接API回调函数 */API_EVENT(conn, NETCONN_EVT_SENDMINUS, len);}}/* 将数据发送出去 */if(err == ERR_OK) {conn->write_offset += len;if((conn->write_offset == conn->current_msg->msg.w.len) || dontblock) {conn->current_msg->msg.w.len = conn->write_offset;write_finished = 1;conn->write_offset = 0;}tcp_output(conn->pcb.tcp);} else if((err == ERR_MEM) && !dontblock) {tcp_output(conn->pcb.tcp);} else {write_finished = 1;conn->current_msg->msg.w.len = 0;}}/* 发送完成 */if(write_finished) {conn->current_msg->err = err;conn->current_msg = NULL;conn->state = NETCONN_NONE;{/* 释放操作完成信号量 */sys_sem_signal(&conn->op_completed);}}return ERR_OK; }/* TCP发送数据 */ void do_write(struct api_msg_msg *msg) {if(ERR_IS_FATAL(msg->conn->last_err)) {msg->err = msg->conn->last_err;} else {/* TCP连接 */if(msg->conn->type == NETCONN_TCP) {if(msg->conn->state != NETCONN_NONE) {msg->err = ERR_INPROGRESS;}else if(msg->conn->pcb.tcp != NULL) {msg->conn->state = NETCONN_WRITE;msg->conn->current_msg = msg;msg->conn->write_offset = 0;do_writemore(msg->conn);return;} else {msg->err = ERR_CONN;}} /* 非TCP连接 */else {msg->err = ERR_VAL;}}/* 释放操作完成信号量 */TCPIP_APIMSG_ACK(msg); }/* 获取本地或者远程IP和端口号 */ void do_getaddr(struct api_msg_msg *msg) {if(msg->conn->pcb.ip != NULL) {*(msg->msg.ad.ipaddr) = (msg->msg.ad.local ? msg->conn->pcb.ip->local_ip : msg->conn->pcb.ip->remote_ip);msg->err = ERR_OK;switch(NETCONNTYPE_GROUP(msg->conn->type)) {case NETCONN_RAW:if(msg->msg.ad.local) {*(msg->msg.ad.port) = msg->conn->pcb.raw->protocol;} else {msg->err = ERR_CONN;}break;case NETCONN_UDP:if(msg->msg.ad.local) {*(msg->msg.ad.port) = msg->conn->pcb.udp->local_port;} else {if((msg->conn->pcb.udp->flags & UDP_FLAGS_CONNECTED) == 0) {msg->err = ERR_CONN;} else {*(msg->msg.ad.port) = msg->conn->pcb.udp->remote_port;}}break;case NETCONN_TCP:*(msg->msg.ad.port) = (msg->msg.ad.local ? msg->conn->pcb.tcp->local_port : msg->conn->pcb.tcp->remote_port);break;default:break;}} else {msg->err = ERR_CONN;}/* 释放操作完成信号量 */TCPIP_APIMSG_ACK(msg); }/* 关闭连接 */ void do_close(struct api_msg_msg *msg) {if((msg->conn->state != NETCONN_NONE) && (msg->conn->state != NETCONN_LISTEN)) {msg->err = ERR_INPROGRESS;} /* TCP连接 */else if((msg->conn->pcb.tcp != NULL) && (msg->conn->type == NETCONN_TCP)) {if((msg->msg.sd.shut != NETCONN_SHUT_RDWR) && (msg->conn->state == NETCONN_LISTEN)) {msg->err = ERR_CONN;} else {/* 关闭接收方向 */if(msg->msg.sd.shut & NETCONN_SHUT_RD) {/* 删除接收邮箱和接受连接邮箱 */netconn_drain(msg->conn);}msg->conn->state = NETCONN_CLOSE;msg->conn->current_msg = msg;/* 关闭连接 */do_close_internal(msg->conn);return;}} else{msg->err = ERR_VAL;}/* 释放操作完成信号量 */sys_sem_signal(&msg->conn->op_completed); }

最后,看一下应用程序接口

/* 创建一个新的连接并且设置回调函数 */ struct netconn *netconn_new_with_proto_and_callback(enum netconn_type t, u8_t proto, netconn_callback callback) {struct netconn *conn;struct api_msg msg;/* 创建一个连接结构体 */conn = netconn_alloc(t, callback);if(conn != NULL) {/* 请求创建一个连接PCB */msg.function = do_newconn;msg.msg.msg.n.proto = proto;msg.msg.conn = conn;if(TCPIP_APIMSG(&msg) != ERR_OK) {sys_sem_free(&conn->op_completed);sys_mbox_free(&conn->recvmbox);memp_free(MEMP_NETCONN, conn);return NULL;}}return conn; }/* 删除连接 */ err_t netconn_delete(struct netconn *conn) {struct api_msg msg;if(conn == NULL) {return ERR_OK;}/* 请求删除控制块 */msg.function = do_delconn;msg.msg.conn = conn;tcpip_apimsg(&msg);/* 释放连接结构体 */netconn_free(conn);return ERR_OK; }/* 获取本地或者远程IP和端口号 */ err_t netconn_getaddr(struct netconn *conn, ip_addr_t *addr, u16_t *port, u8_t local) {struct api_msg msg;err_t err;/* 请求获取本地或者远程IP和端口号 */msg.function = do_getaddr;msg.msg.conn = conn;msg.msg.msg.ad.ipaddr = addr;msg.msg.msg.ad.port = port;msg.msg.msg.ad.local = local;err = TCPIP_APIMSG(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 绑定本地端口 */ err_t netconn_bind(struct netconn *conn, ip_addr_t *addr, u16_t port) {struct api_msg msg;err_t err;/* 请求绑定本地端口 */msg.function = do_bind;msg.msg.conn = conn;msg.msg.msg.bc.ipaddr = addr;msg.msg.msg.bc.port = port;err = TCPIP_APIMSG(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 连接远程端口 */ err_t netconn_connect(struct netconn *conn, ip_addr_t *addr, u16_t port) {struct api_msg msg;err_t err;/* 请求连接远程端口 */msg.function = do_connect;msg.msg.conn = conn;msg.msg.msg.bc.ipaddr = addr;msg.msg.msg.bc.port = port;err = tcpip_apimsg(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 断开连接 */ err_t netconn_disconnect(struct netconn *conn) {struct api_msg msg;err_t err;/* 请求断开连接 */msg.function = do_disconnect;msg.msg.conn = conn;err = TCPIP_APIMSG(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 开始侦听 */ err_t netconn_listen_with_backlog(struct netconn *conn, u8_t backlog) {struct api_msg msg;err_t err;/* 请求侦听 */msg.function = do_listen;msg.msg.conn = conn;err = TCPIP_APIMSG(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 接受连接 */ err_t netconn_accept(struct netconn *conn, struct netconn **new_conn) {struct netconn *newconn;err_t err;*new_conn = NULL;err = conn->last_err;if (ERR_IS_FATAL(err)) {return err;}/* 等待连接 */sys_arch_mbox_fetch(&conn->acceptmbox, (void **)&newconn, 0);/* 调用连接API回调函数 */API_EVENT(conn, NETCONN_EVT_RCVMINUS, 0);if(newconn == NULL) {NETCONN_SET_SAFE_ERR(conn, ERR_ABRT);return ERR_ABRT;}*new_conn = newconn;return ERR_OK; }/* 接收数据 */ static err_t netconn_recv_data(struct netconn *conn, void **new_buf) {void *buf = NULL;u16_t len;err_t err;struct api_msg msg;*new_buf = NULL;err = conn->last_err;if(ERR_IS_FATAL(err)) {return err;}/* 等待数据 */sys_arch_mbox_fetch(&conn->recvmbox, &buf, 0);/* TCP */if(conn->type == NETCONN_TCP){/* 自动接收 */if(!netconn_get_noautorecved(conn) || (buf == NULL)) {/* 请求接收,更新窗口 */msg.function = do_recv;msg.msg.conn = conn;if(buf != NULL) {msg.msg.msg.r.len = ((struct pbuf *)buf)->tot_len;} else {msg.msg.msg.r.len = 1;}TCPIP_APIMSG(&msg);}if(buf == NULL) {API_EVENT(conn, NETCONN_EVT_RCVMINUS, 0);NETCONN_SET_SAFE_ERR(conn, ERR_CLSD);return ERR_CLSD;}len = ((struct pbuf *)buf)->tot_len;}/* 非TCP */else{len = netbuf_len((struct netbuf *)buf);}/* 调用连接API回调函数 */API_EVENT(conn, NETCONN_EVT_RCVMINUS, len);*new_buf = buf;return ERR_OK; }/* 接收数据 */ err_t netconn_recv_tcp_pbuf(struct netconn *conn, struct pbuf **new_buf) {return netconn_recv_data(conn, (void **)new_buf); }/* 接收数据并将数据封装到netbuf */ err_t netconn_recv(struct netconn *conn, struct netbuf **new_buf) {struct netbuf *buf = NULL;err_t err;*new_buf = NULL;if (conn->type == NETCONN_TCP){struct pbuf *p = NULL;buf = (struct netbuf *)memp_malloc(MEMP_NETBUF);if(buf == NULL) {NETCONN_SET_SAFE_ERR(conn, ERR_MEM);return ERR_MEM;}err = netconn_recv_data(conn, (void **)&p);if(err != ERR_OK) {memp_free(MEMP_NETBUF, buf);return err;}buf->p = p;buf->ptr = p;buf->port = 0;ip_addr_set_any(&buf->addr);*new_buf = buf;return ERR_OK;}else{return netconn_recv_data(conn, (void **)new_buf);} }/* 更新接收窗口 */ void netconn_recved(struct netconn *conn, u32_t length) {if((conn != NULL) && (conn->type == NETCONN_TCP) && (netconn_get_noautorecved(conn))) {struct api_msg msg;/* 请求接收,更新窗口 */msg.function = do_recv;msg.msg.conn = conn;msg.msg.msg.r.len = length;TCPIP_APIMSG(&msg);} }/* UDP或RAW发送数据 */ err_t netconn_sendto(struct netconn *conn, struct netbuf *buf, ip_addr_t *addr, u16_t port) {if(buf != NULL) {ip_addr_set(&buf->addr, addr);buf->port = port;return netconn_send(conn, buf);}return ERR_VAL; }/* UDP或RAW发送数据 */ err_t netconn_send(struct netconn *conn, struct netbuf *buf) {struct api_msg msg;err_t err;/* 请求发送数据 */msg.function = do_send;msg.msg.conn = conn;msg.msg.msg.b = buf;err = TCPIP_APIMSG(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* TCP发送数据 */ err_t netconn_write_partly(struct netconn *conn, const void *dataptr, size_t size, u8_t apiflags, size_t *bytes_written) {struct api_msg msg;err_t err;u8_t dontblock;if(size == 0) {return ERR_OK;}dontblock = netconn_is_nonblocking(conn) || (apiflags & NETCONN_DONTBLOCK);if(dontblock && !bytes_written) {return ERR_VAL;}/* 请求TCP发送数据 */msg.function = do_write;msg.msg.conn = conn;msg.msg.msg.w.dataptr = dataptr;msg.msg.msg.w.apiflags = apiflags;msg.msg.msg.w.len = size;err = TCPIP_APIMSG(&msg);if((err == ERR_OK) && (bytes_written != NULL)) {/* 非阻塞 */if(dontblock) {*bytes_written = msg.msg.msg.w.len;} /* 阻塞型 */else {*bytes_written = size;}}NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 关闭连接 */ static err_t netconn_close_shutdown(struct netconn *conn, u8_t how) {struct api_msg msg;err_t err;/* 请求关闭连接 */msg.function = do_close;msg.msg.conn = conn;msg.msg.msg.sd.shut = how;err = tcpip_apimsg(&msg);NETCONN_SET_SAFE_ERR(conn, err);return err; }/* 完全关闭连接 */ err_t netconn_close(struct netconn *conn) {return netconn_close_shutdown(conn, NETCONN_SHUT_RDWR); }/* 关闭连接方向 */ err_t netconn_shutdown(struct netconn *conn, u8_t shut_rx, u8_t shut_tx) {return netconn_close_shutdown(conn, (shut_rx ? NETCONN_SHUT_RD : 0) | (shut_tx ? NETCONN_SHUT_WR : 0)); }

 

总结

以上是生活随笔为你收集整理的LwIP之协议栈接口的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得生活随笔网站内容还不错,欢迎将生活随笔推荐给好友。