如何使用 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 请求?的主要内容,如果未能解决你的问题,请参考以下文章

lwip stm32 - http请求失败

如何通过lwip协议栈发送数据给上位机

使用 LWIP 发送原始 IP 数据包

SVG堆栈精灵没有减少发送请求的数量

lwip 堆栈 netconn api 保持连接“保持活动”

lwip组件的SNTP app如何设置接收时间?