zoukankan      html  css  js  c++  java
  • lwip 使用记录(1)

    原子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.

    这一点对应网络的断线重连右转着很重要的作用.

    如果问题解决起来不妥或者有更好的解决办法,麻烦请告知,帮助曾经和你一样的入门者,谢谢。
  • 相关阅读:
    shell 数组遍历加引号和不加引号的区别?
    shell map数据结构的实现
    PyCharm 项目打开窗口设置为当前还是新开一个怎么办?
    python 模拟ssh 登录远程服务器
    python 字节码死磕
    docker相关内容
    Windows7安装 docker-compose的过程
    史上最简单的Docker入门教程
    MySQL触发器使用详解
    存储过程
  • 原文地址:https://www.cnblogs.com/ourran/p/5914170.html
Copyright © 2011-2022 走看看