原子F429的lwip实验:网络实验8 NETCONN_TCP客户端实验 代码
//tcp客户端任务函数 static void tcp_client_thread(void *arg) { OS_CPU_SR cpu_sr; u32 data_len = 0; struct pbuf *q; err_t err,recv_err; static ip_addr_t server_ipaddr,loca_ipaddr; static u16_t server_port,loca_port; LWIP_UNUSED_ARG(arg); server_port = REMOTE_PORT; IP4_ADDR(&server_ipaddr, lwipdev.remoteip[0],lwipdev.remoteip[1], lwipdev.remoteip[2],lwipdev.remoteip[3]); while (1) { tcp_clientconn = netconn_new(NETCONN_TCP); //创建一个TCP链接 err = netconn_connect(tcp_clientconn,&server_ipaddr,server_port);//连接服务器 if(err != ERR_OK) netconn_delete(tcp_clientconn); //返回值不等于ERR_OK,删除tcp_clientconn连接 else if (err == ERR_OK) //处理新连接的数据 { struct netbuf *recvbuf; tcp_clientconn->recv_timeout = 10; netconn_getaddr(tcp_clientconn,&loca_ipaddr,&loca_port,1); //获取本地IP主机IP地址和端口号 printf("连接上服务器%d.%d.%d.%d,本机端口号为:%d ",lwipdev.remoteip[0],lwipdev.remoteip[1], lwipdev.remoteip[2],lwipdev.remoteip[3],loca_port); while(1) { if((tcp_client_flag & LWIP_SEND_DATA) == LWIP_SEND_DATA) //有数据要发送 { err = netconn_write(tcp_clientconn ,tcp_client_sendbuf,strlen((char*)tcp_client_sendbuf),NETCONN_COPY); //发送tcp_server_sentbuf中的数据 if(err != ERR_OK) { printf("发送失败 "); } tcp_client_flag &= ~LWIP_SEND_DATA; } if((recv_err = netconn_recv(tcp_clientconn,&recvbuf)) == ERR_OK) //接收到数据 { OS_ENTER_CRITICAL(); //关中断 memset(tcp_client_recvbuf,0,TCP_CLIENT_RX_BUFSIZE); //数据接收缓冲区清零 for(q=recvbuf->p;q!=NULL;q=q->next) //遍历完整个pbuf链表 { //判断要拷贝到TCP_CLIENT_RX_BUFSIZE中的数据是否大于TCP_CLIENT_RX_BUFSIZE的剩余空间,如果大于 //的话就只拷贝TCP_CLIENT_RX_BUFSIZE中剩余长度的数据,否则的话就拷贝所有的数据 if(q->len > (TCP_CLIENT_RX_BUFSIZE-data_len)) memcpy(tcp_client_recvbuf+data_len,q->payload,(TCP_CLIENT_RX_BUFSIZE-data_len));//拷贝数据 else memcpy(tcp_client_recvbuf+data_len,q->payload,q->len); data_len += q->len; if(data_len > TCP_CLIENT_RX_BUFSIZE) break; //超出TCP客户端接收数组,跳出 } OS_EXIT_CRITICAL(); //开中断 data_len=0; //复制完成后data_len要清零。 printf("%s ",tcp_client_recvbuf); netbuf_delete(recvbuf); }else if(recv_err == ERR_CLSD) //关闭连接 { netconn_close(tcp_clientconn); netconn_delete(tcp_clientconn); printf("服务器%d.%d.%d.%d断开连接 ",lwipdev.remoteip[0],lwipdev.remoteip[1], lwipdev.remoteip[2],lwipdev.remoteip[3]); break; } } } } }
在使用lwip中出现这种值得注意的情况:
1.上边标红的地方:
tcp_clientconn = netconn_new(NETCONN_TCP);
err = netconn_connect(tcp_clientconn,&server_ipaddr,server_port);//连接服务器
if(err != ERR_OK) netconn_delete(tcp_clientconn); //返回值不等于ERR_OK,删除tcp_clientconn连接
这里 netconn_connect 连接不上 的话,后续处理是直接是 netconn_delete 这是没的说的
2.上边标红的地方:
netconn_close(tcp_clientconn); netconn_delete(tcp_clientconn);
这里 使用完 netconn_close 之后会随着调用 netconn_delete 这是必须的
结合1/2,就是 使用netconn_delete之后就可以再次使用 netconn_connect,
在实际过程中,试过使用 netconn_close 后没有调用 netconn_delete, 会出现大约5次 netconn_new 后,
返回的指针均为 0x00; 追进去看后发现在 memp.c的(st1.13中的lwip)L320的memp_pool[type],
对应memp_pool[type].tab 的内容为空,剩下的就没有再去追踪,应该是内存那块有问题吧,具体原因没有细查.有知道的麻烦回复下,谢谢.
于是在工程在netconn_close后将 netconn_delete 补上,发现delete后可以正常的new和connect.
这一点对应网络的断线重连右转着很重要的作用.