LWIP TCP滑动窗口为TCP ZeroWindow的解决方法

最近做项目使用到LWIP, 板子当作客户端,PC机当作服务器,当板子和PC机通信一段时间后,板子就接收不到数据。抓包分析为:

发现经过一段时间通信后,滑动窗口大小越来越小,最终显示为TCP WINDOW FULL,TCP ZeroWindow。 仔细分析了下LWIP源码,还以为是内存管理出了问题,跟踪了半天memp.c也没发现问题。后来发现问题出现自己的应用实现代码,下面是我这边的实现代码:

复制代码
err_t tcp_client_recv(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
{
    if (err != ERR_OK) {
        RLOG_ERROR("Error in tcp_client_recv: %d\n", err);
        tcp_client_close(tpcb); // 关闭连接
        return ERR_OK; // 释放 pbuf
    }

    if (p != NULL) {
        // 处理接收到的数据
        char *data = (char *)p->payload;
        int datalen = p->len;

        // 安全地复制数据
        char buff[datalen + 1];
        memcpy(buff, data, datalen);
        buff[datalen] = '\0';

        // 打印接收到的数据
        //RLOG_INFO("Received data: %s\n", buff);

        // 释放 pbuf
        pbuf_free(p);
        RLOG_INFO("pbuf freed\n");
    }
    tcp_output(tpcb);

    return ERR_OK;
}

上面的代码,需要在接受数据后,主动调用tcp_recved(tpcb, p->tot_len) ,修改代码后,再次验证功能正常。

修改后的代码:

复制代码
err_t tcp_client_recv(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
{
    if (err != ERR_OK) {
        RLOG_ERROR("Error in tcp_client_recv: %d\n", err);
        tcp_client_close(tpcb); // 关闭连接
        return ERR_OK; // 释放 pbuf
    }

    if (p != NULL) {
        // 处理接收到的数据
        char *data = (char *)p->payload;
        int datalen = p->len;

        // 安全地复制数据
        char buff[datalen + 1];
        memcpy(buff, data, datalen);
        buff[datalen] = '\0';

        // 打印接收到的数据
        //RLOG_INFO("Received data: %s\n", buff);
		
	tcp_recved(tpcb, p->tot_len);
        // 释放 pbuf
        pbuf_free(p);
        RLOG_INFO("pbuf freed\n");
    }
    tcp_output(tpcb);

    return ERR_OK;
}

原来,tcp_recved(tpcb, p->tot_len) 的主要作用是通知 TCP 层应用程序已经处理了指定长度的数据。这有助于 TCP 层调整接收窗口,避免接收缓冲区溢出,并确保数据传输的高效性。如果忘记调用这个函数,可能会导致接收窗口过小,影响数据传输效率。

相关推荐
liulilittle4 天前
回归物理本质:对拥塞控制实验室依赖与公平性误置的反思
网络·tcp/ip·计算机网络·算法·tcp·通信·拥塞控制
Molesidy5 天前
【Embedded Development】【ETH】【Lwip】基于STM32的ETH-Lwip以太网通信的学习记录
stm32cubemx·eth·lwip
东南门吹雪7 天前
JAVA TCP socket编程框架
java·高并发·socket·tcp·nio
十正7 天前
aiohttp.TCPConnector 连接池原理详解
网络·python·tcp·aiohttp
liulilittle8 天前
拥塞控制:排水终止的两种决策:OR 与 AND
网络·tcp/ip·计算机网络·算法·信息与通信·tcp·通信
liulilittle9 天前
删除 Inflight Bounds:为什么 KCC 放弃了 BDP 钳位
linux·网络·tcp/ip·计算机网络·信息与通信·tcp·通信
阿钱真强道10 天前
28 鸿蒙LiteOS RK2206 LwIP Raw API 实现无阻塞UDP双向通信
udp·harmonyos·鸿蒙·lwip·开源鸿蒙
liulilittle11 天前
TCP KCC v1.0(卡尔曼拥塞控制)
linux·服务器·网络·tcp/ip·计算机网络·tcp·通信
金融支付架构实战指南11 天前
Java Socket 全网read/write底层原理 + 避坑实战
socket·tcp·write·listen·connect
liulilittle11 天前
用户态 TCP 端口转发:对 CUBIC 友好,对 BBR/KCC 收益不大
运维·网络·tcp/ip·计算机网络·信息与通信·tcp·通信