如何使用 lwIP 堆栈发送简单的 HTTP 请求?
Posted
技术标签:
【中文标题】如何使用 lwIP 堆栈发送简单的 HTTP 请求?【英文标题】:How can I send a simple HTTP request with a lwIP stack? 【发布时间】:2014-11-29 07:51:14 【问题描述】:如果问题不相关,请移动/关闭它。
核心:Cortex-M4
微处理器:TI TM4C1294NCPDT。
IP 堆栈:lwIP 1.4.1
我正在使用这个微处理器进行一些数据记录,我想通过 HTTP 请求以以下形式将一些信息发送到单独的 Web 服务器:
http://123.456.789.012:8800/process.php?data1=foo&data2=bar&time=1234568789
并且我希望处理器能够看到响应标头(即是否为 200 OK 或出现问题)——它不必显示/接收实际内容。
lwIP 有一个用于微处理器的 http 服务器,但我追求的是相反的(微处理器是客户端)。
我不确定数据包如何与请求/响应标头相关联,因此我不确定我应该如何实际发送/接收信息。
【问题讨论】:
【参考方案1】:这最终实现起来非常简单,忘记更新这个问题。
我几乎按照this 网站上给出的说明进行操作,这是 Raw/TCP 的“文档”。
基本上,HTTP 请求被编码在 TCP 数据包中,所以为了向我的 PHP 服务器发送数据,我使用 TCP 数据包发送了一个 HTTP 请求(lwIP 完成了所有工作)。
我要发送的 HTTP 数据包如下所示:
HEAD /process.php?data1=12&data2=5 HTTP/1.0
主机:mywebsite.com
要将其“翻译”为 HTTP 服务器可以理解的文本,您必须在代码中添加“\r\n”回车符/换行符。所以它看起来像这样:
char *string = "HEAD /process.php?data1=12&data2=5 HTTP/1.0\r\nHost: mywebsite.com\r\n\r\n ";
注意末尾有两个“\r\n”
您可以使用 GET 或 HEAD,但因为我不关心我的 PHP 服务器返回的 html 站点,所以我使用了 HEAD(它在成功时返回 200 OK,或者在失败时返回不同的代码)。
lwIP raw/tcp 适用于回调。您基本上设置了所有回调函数,然后将您想要的数据推送到 TCP 缓冲区(在本例中为上面指定的 TCP 字符串),然后您告诉 lwIP 发送数据包。
建立TCP连接的函数(我的应用程序每次要发送TCP数据包时都会直接调用该函数):
void tcp_setup(void)
uint32_t data = 0xdeadbeef;
/* create an ip */
struct ip_addr ip;
IP4_ADDR(&ip, 110,777,888,999); //IP of my PHP server
/* create the control block */
testpcb = tcp_new(); //testpcb is a global struct tcp_pcb
// as defined by lwIP
/* dummy data to pass to callbacks*/
tcp_arg(testpcb, &data);
/* register callbacks with the pcb */
tcp_err(testpcb, tcpErrorHandler);
tcp_recv(testpcb, tcpRecvCallback);
tcp_sent(testpcb, tcpSendCallback);
/* now connect */
tcp_connect(testpcb, &ip, 80, connectCallback);
与我的 PHP 服务器建立连接后,lwIP 将调用“connectCallback”函数:
/* connection established callback, err is unused and only return 0 */
err_t connectCallback(void *arg, struct tcp_pcb *tpcb, err_t err)
UARTprintf("Connection Established.\n");
UARTprintf("Now sending a packet\n");
tcp_send_packet();
return 0;
该函数调用实际发送HTTP请求的函数tcp_send_packet(),如下:
uint32_t tcp_send_packet(void)
char *string = "HEAD /process.php?data1=12&data2=5 HTTP/1.0\r\nHost: mywebsite.com\r\n\r\n ";
uint32_t len = strlen(string);
/* push to buffer */
error = tcp_write(testpcb, string, strlen(string), TCP_WRITE_FLAG_COPY);
if (error)
UARTprintf("ERROR: Code: %d (tcp_send_packet :: tcp_write)\n", error);
return 1;
/* now send */
error = tcp_output(testpcb);
if (error)
UARTprintf("ERROR: Code: %d (tcp_send_packet :: tcp_output)\n", error);
return 1;
return 0;
一旦 TCP 数据包被发送(如果你想“希望最好”并且不关心数据是否实际发送,这都是必需的),PHP 服务器返回一个 TCP 数据包(带有 200 OK,等以及 HTML 代码(如果您使用 GET 而不是 HEAD)。这段代码可以在下面的代码中阅读和验证:
err_t tcpRecvCallback(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
UARTprintf("Data recieved.\n");
if (p == NULL)
UARTprintf("The remote host closed the connection.\n");
UARTprintf("Now I'm closing the connection.\n");
tcp_close_con();
return ERR_ABRT;
else
UARTprintf("Number of pbufs %d\n", pbuf_clen(p));
UARTprintf("Contents of pbuf %s\n", (char *)p->payload);
return 0;
p->payload 包含实际的“200 OK”等信息。希望这对某人有所帮助。
我在上面的代码中省略了一些错误检查以简化答案。
【讨论】:
太棒了!非常感谢你的帮助。 lwIP 文档缺少这样的基本示例。不过要小心,我猜用户代码必须在 tcpRecvCallback 中手动调用 pbuf_free(p),否则内存不会从 PBUF_POOL 中释放。我在 tcp_close(pcb) 上也遇到了一些问题,它返回 ERR_OK 但 pcb 从未被释放并无限期地保持在 CLOSE_WAIT 状态。我必须在调用 tcp_close 之前设置 tcp_input_pcb = NULL,作为一种解决方法.... 这对我有用。谢谢@tgun926。一个查询是,我使用它每 5 分钟向服务器发送一次数据。它在 tcp_connect 调用时停留在分段部分。你能帮我解决这个问题吗?我只是想使用此功能将数据发送到服务器。我有多个服务器。所以我每次都要用不同的IP连接tcp。 在调用 sys_init() 和 lwip_init() 之后,我尝试在主循环中使用您的代码,但我似乎无法发出 HTTP 请求,除非我明确提供(并绑定到) 我的本地 IP 地址。你知道为什么会这样吗?在您提供的文档链接中,它说调用 tcp_bind() 是可选的(在我的情况下不是)。【参考方案2】:查看***中的HTTP example。客户端将发送 GET 和 HOST 行。服务器将响应多行响应。第一行会有响应码。
【讨论】:
是的,我已经看过很多次了。我正在通过 lwIP 库获得更具体的答案。我想我几乎已经通过 raw/tcp api 弄清楚了——很快就会测试出来。以上是关于如何使用 lwIP 堆栈发送简单的 HTTP 请求?的主要内容,如果未能解决你的问题,请参考以下文章