LWIP 版本:2.0.3

上一篇文章是写如何将 LWIP 移植到板子上,今天晚上记录基于 LWIP 实现与主机的网络通信。

先是打开了原子的实验例程,大概浏览了一遍,觉得 TCP 网络网络通信也就是那么一些套路。什么 创建、配置、绑定、监听、accept ....,果断复制源文件到工程路径下,调整头文件包含直至编译无误。将 tcp_server_init( ) 加入到 main 中,下载测试,果然出现问题。 ping 都 ping 不通了,尴尬.....

问题解决过程:

出问题了是好事,可以更加深入的了解 LWIP(强行安慰自己一波)。遂加入 printf 输出查找问题根源,最后追踪到 netconn_new 内的 netconn_alloc 函数处,解决办法如下:

本以为是其内部调用的函数 memp_malloc 申请内存失败。进入函数内部开始阅读源码,发现 MEMP_MEM_MALLOC 宏没有被打开,误理解为没有启用该宏就不会成功申请到内存,后来经测试发现并不是它的问题,官方注释如下:

/**
* MEMP_MEM_MALLOC==1: Use mem_malloc/mem_free instead of the lwip pool allocator.
* Especially useful with MEM_LIBC_MALLOC but handle with care regarding execution
* speed (heap alloc can be much slower than pool alloc) and usage from interrupts
* (especially if your netif driver allocates PBUF_POOL pbufs for received frames
* from interrupt)!
* ATTENTION: Currently, this uses the heap for ALL pools (also for private pools,
* not only for internal pools defined in memp_std.h)!
*/

如果不启用该宏,LWIP 内核会直接从 lwip pool 中取出相应的结构,pool 的内存已经在 LWIP 内核运行前申请完毕,官方推荐不启用该宏,因为 speed (heap alloc can be much slower than pool alloc)。

后来继续查找问题所在,发现一个致命问题:netconn_alloc 函数内调用的创建邮箱函数 sys_mbox_new 没有为其参数(一个邮箱指针)分配内存,这是移植时残留的问题,终于找上头来。通过 LWIP 内核中提供的 mem_malloc/mem_free 函数解决了这个问题,如下所示:

err_t sys_mbox_new( sys_mbox_t *mbox, int size)
{
*mbox = mem_malloc(sizeof(TQ_DESCR));
.....
} void sys_mbox_free(sys_mbox_t * mbox)
{
.......
mem_free(*mbox);
*mbox=NULL;
}

重新编译工程,下载到板子,完美运行!

运行结果:

打开网络助手,配置好 tcp server 与端口,点击连接,显示如下:

串口助手显示如下:

附:lwip_server.c 源码

#include "tcp_server.h"
#include "lwip/opt.h"
#include "lwip_app.h"
#include "lwip/sys.h"
#include "lwip/api.h"
#include "delay.h"
#include "string.h" u8 tcp_server_recvbuf[TCP_SERVER_RX_BUFSIZE];
static const u8 tcp_server_sendbuf[] = "Welcome to connect to the LWIP server.\r\n";
u8 tcp_server_flag = 0xff;
#define TCPSERVER_PRIO 6
#define TCPSERVER_STK_SIZE 300
OS_STK TCPSERVER_TASK_STK[TCPSERVER_STK_SIZE]; static void tcp_server_thread(void *arg)
{
OS_CPU_SR cpu_sr;
u32 data_len = 0;
struct pbuf *q;
err_t err,recv_err;
u8 remot_addr[4];
struct netconn *conn, *newconn;
static ip_addr_t ipaddr;
static u16_t port; LWIP_UNUSED_ARG(arg); /* 创建 TCP、绑定、监听 */
conn = netconn_new(NETCONN_TCP);
netconn_bind(conn,IP_ADDR_ANY,TCP_SERVER_PORT);
netconn_listen(conn); /* 禁止阻塞线程,等待时间为 10ms */
conn->recv_timeout = 10; while (1)
{
err = netconn_accept(conn,&newconn);
if(err==ERR_OK) newconn->recv_timeout = 10; if (err == ERR_OK)
{
struct netbuf *recvbuf; /* 获取到客户端的 IP 及端口号 */
netconn_getaddr(newconn,&ipaddr,&port,0); remot_addr[3] = (uint8_t)(ipaddr.addr >> 24);
remot_addr[2] = (uint8_t)(ipaddr.addr >> 16);
remot_addr[1] = (uint8_t)(ipaddr.addr >> 8);
remot_addr[0] = (uint8_t)(ipaddr.addr);
printf("主机:%d.%d.%d.%d 连接上服务器,主机端口号为:%d\r\n",remot_addr[0], remot_addr[1],remot_addr[2],remot_addr[3],port); while(1)
{
if((tcp_server_flag & LWIP_SEND_DATA) == LWIP_SEND_DATA)
{
/* 向客户端发送消息 */
err = netconn_write(newconn,tcp_server_sendbuf,strlen((char*)tcp_server_sendbuf),NETCONN_COPY);
if(err != ERR_OK)
{
printf("发送失败\r\n");
}
tcp_server_flag &= ~LWIP_SEND_DATA;
} /* 接收消息 */
if((recv_err = netconn_recv(newconn,&recvbuf)) == ERR_OK)
{
OS_ENTER_CRITICAL();
memset(tcp_server_recvbuf,0,TCP_SERVER_RX_BUFSIZE); /* 将缓冲区信息转存到自定义空间 */
for(q=recvbuf->p;q!=NULL;q=q->next)
{
if(q->len > (TCP_SERVER_RX_BUFSIZE-data_len))
memcpy(tcp_server_recvbuf+data_len,q->payload,(TCP_SERVER_RX_BUFSIZE-data_len));
else memcpy(tcp_server_recvbuf+data_len,q->payload,q->len);
data_len += q->len;
if(data_len > TCP_SERVER_RX_BUFSIZE) break;
}
OS_EXIT_CRITICAL();
data_len=0;
printf("%s\r\n",tcp_server_recvbuf);
netbuf_delete(recvbuf); /* 断开连接 */
}else if(recv_err == ERR_CLSD)
{
netconn_close(newconn);
netconn_delete(newconn);
tcp_server_flag = 0xff;
printf("主机:%d.%d.%d.%d 断开与服务器的连接\r\n",remot_addr[0], remot_addr[1],remot_addr[2],remot_addr[3]);
break;
}
}
}
delay_ms(10);
}
} INT8U tcp_server_init(void)
{
INT8U res=0;
OS_CPU_SR cpu_sr; OS_ENTER_CRITICAL(); /* 创建 TCP 服务器任务 */
res = OSTaskCreate(tcp_server_thread,(void*)0,(OS_STK*)&TCPSERVER_TASK_STK[TCPSERVER_STK_SIZE-1],TCPSERVER_PRIO);
OS_EXIT_CRITICAL(); return res;
}

本篇文章参考自原子的 LWIP 网络实验,感谢

基于 LWIP 建立 TCP Server 与主机通信实验的更多相关文章

  1. Python之tcp server模拟Http通信

    1.python tcp server代码: import socket def main(): tcp_server_socket = socket.socket(socket.AF_INET, s ...

  2. swoole之建立 tcp server

    一.swoole的安装 参照官网:https://wiki.swoole.com/wiki/page/6.html 二.代码部分 服务端: <?php $host = "127.0.0 ...

  3. 【iCore3 双核心板_FPGA】实验十六:基于SPI总线的ARM与FPGA通信实验

    实验指导书及代码包下载: http://pan.baidu.com/s/1hs6lDdi iCore3 购买链接: https://item.taobao.com/item.htm?id=524229 ...

  4. 【iCore3 双核心板_FPGA】实验十七:基于I2C总线的ARM与FPGA通信实验

    实验指导书及代码包下载: http://pan.baidu.com/s/1dFqddMp iCore3 购买链接: https://item.taobao.com/item.htm?id=524229 ...

  5. ESP8266开发之旅 网络篇⑦ TCP Server & TCP Client

    授人以鱼不如授人以渔,目的不是为了教会你具体项目开发,而是学会学习的能力.希望大家分享给你周边需要的朋友或者同学,说不定大神成长之路有博哥的奠基石... QQ技术互动交流群:ESP8266&3 ...

  6. 基于Linux的TCP网络聊天室

    1.实验项目名称:基于Linux的TCP网络聊天室 2.实验目的:通过TCP完成多用户群聊和私聊功能. 3.实验过程: 通过socket建立用户连接并传送用户输入的信息,分别来写客户端和服务器端,利用 ...

  7. CentOS7 基于 subversion 配置 SVN server

    由于 Window Server 环境下,VisualSVN Server Community 版本只支持 15 个同时在线用户,所以彻底放弃 Windows Server,在 Linux Serve ...

  8. 【Java TCP/IP Socket】基于NIO的TCP通信(含代码)

    NIO主要原理及使用 NIO采取通道(Channel)和缓冲区(Buffer)来传输和保存数据,它是非阻塞式的I/O,即在等待连接.读写数据(这些都是在一线程以客户端的程序中会阻塞线程的操作)的时候, ...

  9. Python的网络编程[0] -> socket[2] -> 利用 socket 建立 TCP/UDP 通信

    Socket 目录 socket 的 TCP/IP 通信基本建立过程 socket 的 UDP 通信基本建立过程 socket 的 UDP 广播式通信基本建立过程 socket 的多线程通信建立过程 ...

随机推荐

  1. weblogic获取应用目录路径(war包)

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletExcepti ...

  2. EmEditor注册码

    Emeditor 是一款非常好用的文本编辑器 Emeditor 注册码 姓 名:ttrar.com 序 列 号:DKAZQ-R9TYP-5SM2A-9Z8KD-3E2RK 姓 名:whyida 序 列 ...

  3. wincc 用脚本记录日志

    方法1: 在vbs脚本库里面添加 Sub LogTxt(message)    Const ForReading = 1, ForWriting = 2, ForAppending = 8    Di ...

  4. openXML向Word插入表

    表是 Word 中的另一类型的块级内容,它是以行和列排列的一组段落(以及其他块级内容). Word 中的表格通过 tbl 元素定义,该元素类似于 HTML <表格>标记. 表元素指定文档中 ...

  5. vs 2017 打开 iis express问题

    问题: 更新vs2017 15.6.4后,首次打开网站 iis express 一直报 无法连接到web服务器. 解决办法: 关闭防火墙,在次启动即可,启动成功后,在次打开防火墙也无影响.

  6. 小白看过来runtinme

    RunTime 概述 RunTime消息机制 RunTime交换方法 RunTime消息转发 RunTime关联对象 RunTime实现字典与模型互转 1.RunTime 概述 我们在面试的时候,经常 ...

  7. [转]语言模型训练工具SRILM

    SRILM是一个建立和使用统计语言模型的开源工具包,从1995年开始由SRI 口语技术与研究实验室(SRI Speech Technology and Research Laboratory)开发,现 ...

  8. 脑残式网络编程入门(三):HTTP协议必知必会的一些知识

    本文原作者:“竹千代”,原文由“玉刚说”写作平台提供写作赞助,原文版权归“玉刚说”微信公众号所有,即时通讯网收录时有改动. 1.前言 无论是即时通讯应用还是传统的信息系统,Http协议都是我们最常打交 ...

  9. Javascript高级编程学习笔记(29)—— BOM(3)location对象

    在JS中location是一个神奇的对象 它既是window对象的属性,也是document对象的属性 它的作用主要在于保存当前文档页面的信息,以及将 url 解析为独立的片段 location对象属 ...

  10. 吴恩达机器学习笔记38-决策下一步做什么(Deciding What to Do Next Revisited)

    我们已经讨论了模型选择问题,偏差和方差的问题.那么这些诊断法则怎样帮助我们判断,哪些方法可能有助于改进学习算法的效果,而哪些可能是徒劳的呢? 让我们再次回到最开始的例子,在那里寻找答案,这就是我们之前 ...