关于LWIP协议栈连续多次tcp_write后失败的解决过程
(2014-04-02 11:37:01)
标签:
it |
分类: ARM |
好像好久不发日志了,最近比较闲,更新一篇关于lwip协议栈的文章。
前段时间一直在调试lwip协议栈的问题,在stm32F107上实现一个C/S
{
}
然后在Server_accept中也主要是初始化一些回调函数,
{
}
最重要的函数就是Server_recv()了,在这个函数中,根据客户端不同的命令,然后处理相应的数据发送给客户端,但这是问题就暴露出来了。截取一段发送数据的简化代码:
for(i = 0; i < WMFlag.WM_Record_Num; ++i) {
}
其中SendCharBuff主要是调用tcp_write函数,这个当WM_Record_Num这个数值很大时,客户端总是接收不全,后来经过反复地进行实验发现,然来是tcp_write这个函数在循环到12次的时候会返回ERR_MEM的内存错误,这个问题让我百思不得其解,然后通过网上的一些资料,很多人说是lwip协议栈有BUG,然后我姑且相信了这个结论,但是有BUG也得继续调呀。于是便想到了winsock里面有个WSAGetLastError()这个函数,但是lwip里面却没有,网上找了找说可以开启lwip的调试功能,于是乎就开始设置lwip的调试功能了。
关于开启LWIP的调试功能主要的设置如下:
1、
// Add
By
//
//
extern void USART2_Printf(const char *format, ...);
//
//
PS
// “aaaa” “bbb” == “aaaabbb”
#define U8_F "c"
#define S8_F "c"
#define X8_F "x"
#define U16_F "u"
#define S16_F "d"
#define X16_F "x"
#define U32_F "u"
#define S32_F "d"
#define X32_F "x"
//
#define LWIP_DEBUG
#define
LWIP_PLATFORM_DIAG(x)
// Add End
2、
#ifndef TCP_OUTPUT_DEBUG
#define
TCP_OUTPUT_DEBUG
#endif
3、
void USART2_Printf(const char *format,...)
{
}
开启调试后得到如下的调试输出信息:
tcp_write(pcb=20009c1c, data=08003a24, len=6, apiflags=1)
tcp_enqueue(pcb=20009c1c, arg=08003a24, len=6, flags=0, apiflags=1)
tcp_enqueue: queuelen: 12
tcp_enqueue: too long queue 12 (max 12)
tcp_output_segment: 6848:6920
State: ESTABLISHED
可以看出在tcp_enqueue第12次的时候输出了too long queue 12 (max 12),超出最大的列队次数,于是在工程中搜索too long queue这句话,在tcp_out.c文件中找到了如下代码:
if ((queuelen > TCP_SND_QUEUELEN) || (queuelen > TCP_SNDQUEUELEN_OVERFLOW)) {
LWIP_DEBUGF(TCP_OUTPUT_DEBUG | 2, ("tcp_enqueue: queue too long %"U16_F" (%"U16_F")\n", queuelen, TCP_SND_QUEUELEN));
goto memerr;
}
然后发现TCP_SND_QUEUELEN的值为12,跟到TCP_SND_QUEUELEN的定义代码:
#define
TCP_SND_BUF
//
#define
TCP_SND_QUEUELEN
于是将那个默认的系数由6改为3000,再次测试时发现可以连续tcp_write超过12次了,于是解决了tcp_write的连续多次调用后失败的问题。
最后补充一下tcp_write这个函数的最后一个参数的说明,该函数的声明如下
err_t tcp_write(struct tcp_pcb *pcb, void *dataptr, u16_t len, u8_t copy);
其中第四个参数是一个copy参数,当为0时为不拷贝数据,也就是在dataptr所指的缓冲区里面发送数据,因为调用tcp_write成功后数据并不会立即发送,所以要确保dataptr所指的缓冲区内容保持不变,如果调用tcp_write成功后,再改变dataptr缓冲区可能就会和预期发送的数据不相符,当时我也碰到过这个问题,后来将最后的参数改为1,为1时即拷贝缓冲区内容,当执行tcp_write时,会将dataptr所指向的缓冲区内容先拷贝到发送的缓冲区中,这样的话执行tcp_write之后再改变dataptr所指的内容是不影响数据的正确发送的。
好像好久不发日志了,最近比较闲,更新一篇关于lwip协议栈的文章。
前段时间一直在调试lwip协议栈的问题,在stm32F107上实现一个C/S
{
}
然后在Server_accept中也主要是初始化一些回调函数,
{
}
最重要的函数就是Server_recv()了,在这个函数中,根据客户端不同的命令,然后处理相应的数据发送给客户端,但这是问题就暴露出来了。截取一段发送数据的简化代码:
for(i = 0; i < WMFlag.WM_Record_Num; ++i) {
}
其中SendCharBuff主要是调用tcp_write函数,这个当WM_Record_Num这个数值很大时,客户端总是接收不全,后来经过反复地进行实验发现,然来是tcp_write这个函数在循环到12次的时候会返回ERR_MEM的内存错误,这个问题让我百思不得其解,然后通过网上的一些资料,很多人说是lwip协议栈有BUG,然后我姑且相信了这个结论,但是有BUG也得继续调呀。于是便想到了winsock里面有个WSAGetLastError()这个函数,但是lwip里面却没有,网上找了找说可以开启lwip的调试功能,于是乎就开始设置lwip的调试功能了。
关于开启LWIP的调试功能主要的设置如下:
1、
// Add
By
//
//
extern void USART2_Printf(const char *format, ...);
//
//
PS
// “aaaa” “bbb” == “aaaabbb”
#define U8_F "c"
#define S8_F "c"
#define X8_F "x"
#define U16_F "u"
#define S16_F "d"
#define X16_F "x"
#define U32_F "u"
#define S32_F "d"
#define X32_F "x"
//
#define LWIP_DEBUG
#define
LWIP_PLATFORM_DIAG(x)
// Add End
2、
#ifndef TCP_OUTPUT_DEBUG
#define
TCP_OUTPUT_DEBUG
#endif
3、
void USART2_Printf(const char *format,...)
{
}
开启调试后得到如下的调试输出信息:
tcp_write(pcb=20009c1c, data=08003a24, len=6, apiflags=1)
tcp_enqueue(pcb=20009c1c, arg=08003a24, len=6, flags=0, apiflags=1)
tcp_enqueue: queuelen: 12
tcp_enqueue: too long queue 12 (max 12)
tcp_output_segment: 6848:6920
State: ESTABLISHED
可以看出在tcp_enqueue第12次的时候输出了too long queue 12 (max 12),超出最大的列队次数,于是在工程中搜索too long queue这句话,在tcp_out.c文件中找到了如下代码:
if ((queuelen > TCP_SND_QUEUELEN) || (queuelen > TCP_SNDQUEUELEN_OVERFLOW)) {
LWIP_DEBUGF(TCP_OUTPUT_DEBUG | 2, ("tcp_enqueue: queue too long %"U16_F" (%"U16_F")\n", queuelen, TCP_SND_QUEUELEN));
goto memerr;
}
然后发现TCP_SND_QUEUELEN的值为12,跟到TCP_SND_QUEUELEN的定义代码:
#define
TCP_SND_BUF
//
#define
TCP_SND_QUEUELEN
于是将那个默认的系数由6改为3000,再次测试时发现可以连续tcp_write超过12次了,于是解决了tcp_write的连续多次调用后失败的问题。
最后补充一下tcp_write这个函数的最后一个参数的说明,该函数的声明如下
err_t tcp_write(struct tcp_pcb *pcb, void *dataptr, u16_t len, u8_t copy);
其中第四个参数是一个copy参数,当为0时为不拷贝数据,也就是在dataptr所指的缓冲区里面发送数据,因为调用tcp_write成功后数据并不会立即发送,所以要确保dataptr所指的缓冲区内容保持不变,如果调用tcp_write成功后,再改变dataptr缓冲区可能就会和预期发送的数据不相符,当时我也碰到过这个问题,后来将最后的参数改为1,为1时即拷贝缓冲区内容,当执行tcp_write时,会将dataptr所指向的缓冲区内容先拷贝到发送的缓冲区中,这样的话执行tcp_write之后再改变dataptr所指的内容是不影响数据的正确发送的。