基于 LWIP 建立 TCP Server 与主机通信实验

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 网络实验,感谢

原文地址:https://www.cnblogs.com/GyForever1004/p/9004999.html

时间: 2024-10-11 14:02:44

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

基于Java的TCP Socket通信详解(计算机端/Android手机端)

TCP Socket通信是一种比较常用的基于连接的网络通信方式.本文通过Java实现TCP Socket通信,并将其用于计算机端.Android手机端,同时做到代码规范化,实现代码最大化复用. 本文代码可在GitHub下载,建议对照源码阅读文章 https://github.com/jzj1993/JavaTcpSocket TCP连接的建立 客户端和服务器间通过三次握手建立TCP连接.在Java中,连接建立完成后,服务器端和客户端分别获取到一个Socket实例,之后就可以通过这个Socket实

Android基础入门教程——7.6.2 基于TCP协议的Socket通信(1)

Android基础入门教程--7.6.2 基于TCP协议的Socket通信(1) 标签(空格分隔): Android基础入门教程 本节引言: 上一节的概念课枯燥无味是吧,不过总有点收获是吧,本节开始我们来研究基于TCP协议的Socket 通信,先来了解下Socket的概念,以及Socket通信的模型,实现Socket的步骤,以及作为Socket服务 端与客户端的两位各做要做什么事情!好的,我们由浅入深来扣这个Socket吧! 1.什么是Socket? 2.Socket通信模型: Socket通信

Android基础入门教程——7.6.3 基于TCP协议的Socket通信(2)

Android基础入门教程--7.6.3 基于TCP协议的Socket通信(2) 标签(空格分隔): Android基础入门教程 本节引言: 上节中我们给大家接触了Socket的一些基本概念以及使用方法,然后写了一个小猪简易聊天室的 Demo,相信大家对Socket有了初步的掌握,本节我们来学习下使用Socket来实现大文件的断点续传! 这里讲解的是别人写好的一个Socket上传大文件的例子,不要求我们自己可以写出来,需要的时候会用 就好! 1.运行效果图: 1.先把我们编写好的Socket服务

TCP server 为什么一个端口可以建立多个连接?

https://segmentfault.com/q/1010000003101541 如果是tcp client用同一个本地端口去连不同的两个服务器ip,连第二个时就会提示端口已被占用.但服务器的监听端口,可以accept多次,建立多个socket:我的问题是服务器一个端口为什么能建立多个连接而客户端却不行呢? 2015年08月16日提问 评论 邀请回答 编辑 更多 默认排序时间排序 5个回答 答案对人有帮助,有参考价值8答案没帮助,是错误的答案,答非所问 已采纳 TCP server 可以,

WIFI模块开发教程之W600网络篇2:AP模式下TCP Server通信

前言 本文研究如何在AP模式下进行TCP Server通信,所谓AP模式是说模块起来一个softAP热点,可以供其他WIFI设备连接,当其他设备连接成功后,另WIFI模块作为服务端,等待局域网中其他客户端连接后通信. 一. 理论基础 本节要处理的有两个问题,其一是如何利用RT_Thread连接路由器,其二是如何使用Socket套接字编程搞定TCP Server程序编写. 1.连接路由器 模块需要开启station,并且连接到一个路由器,RT_Thread中只需要调用wlan.mgnt.h中的函数

基于I/O的Server/Client实现

在前面的文章中讲了基于NIO实现的Server/Client.本文就讲讲基于同步堵塞式I/O实现的Server/Client好与前面的NIO中的Server/Client进行对照. 网络编程中须要解决的两个主要问题: 1.怎样准确的定位网络上的一台或多台主机. 2.找到主机后怎样可靠高效的进行传输数据. 而解决这两个问题的主要方式就是非常好的运用TCP/IP协议.所以我们所做的网络编程都是基于TCP/IP来实现的. 基于Socket的java网络编程的通信过程: server:使用ServerS

lwIP相关TCP/IP应用函数

lwIP为使用TCP/IP协议通信的应用程序编程提供了两种接口接口(APIs): * 低层次的称之为"core" / "callback" 或者 "raw" API * 高层次的称之为"sequential" API lwIP "sequential" API为使用TCP/IP协议栈编程提供符合常规的.通用的途径,它与BSD socket API非常相似.程序的执行过程同样是基于"open-rea

图说使用socket建立TCP连接

在网络应用如火如荼的今天,熟悉TCP/IP网络编程,那是最好不过.如果你并不非常熟悉,不妨花几分钟读一读. 为了帮助快速理解,先上个图,典型的使用socket建立和使用TCP/UDP连接过程为(截图来源戳这里): 下面仅讲述TCP连接建立的过程. (参考资料来自这里) 1. Initial State (初始阶段) o TCP is connection based, ... establishing it is a complex multistage process o initially

于TCP协议的socket通信

基于TCP协议Socket服务端和客户端的通信模型: Socket通信步骤: 1.建立服务端ServerSocket和客户端Socket 2.打开连接到Socket的输出输入流 3.按照协议进行读写操作 4.关闭相对应的资源 多线程服务器 多线程用来实现 服务器与多客户端之间的通信 基本步骤 1 服务器创建serverSocket,循环调用accept()等待客户端连接 2 客户端创建一个scoket并请求和服务器端连接 3 服务器端接收客户端请求,创建socket与该客户建立专线连接 4 建立