如何使用lwIP堆栈发送简单的HTTP请求?

如何使用lwIP堆栈发送简单的HTTP请求?,http,ip,embedded,packet,lwip,Http,Ip,Embedded,Packet,Lwip,如果问题不相关,请移动/关闭此项 核心: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有

如果问题不相关,请移动/关闭此项

核心: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服务器,但我的目标是相反的(微处理器是客户端)


我不确定数据包如何与请求/响应头关联,因此我不确定我是如何实际发送/接收信息的。

看看维基百科中的。客户端将发送GET和HOST行。服务器将用许多行响应。第一行将有响应代码。

这非常容易实现,忘记了更新这个问题

我基本上遵循了现场给出的说明,即原始/TCP“文档”

基本上,HTTP请求是用TCP数据包编码的,所以为了向我的PHP服务器发送数据,我使用TCP数据包发送了一个HTTP请求(lwIP完成所有工作)

我要发送的HTTP数据包如下所示:

char *string = "HEAD /process.php?data1=12&data2=5 HTTP/1.0\r\nHost: mywebsite.com\r\n\r\n ";
HEAD/process.php?data1=12&data2=5http/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数据包时,应用程序都会直接调用此函数):

与我的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数据包(如果使用GET而不是HEAD,则返回200 OK等,以及HTML代码)。可在以下代码中读取和验证此代码:

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->有效载荷包含实际的“200 OK”等信息。希望这对某人有所帮助


为了简化答案,我在上面的代码中省略了一些错误检查。

是的,我已经看过很多次了。我想要的是一个关于通过lwIP库实现这一点的更具体的答案。我想我已经通过raw/TCPAPI找到了答案-很快就会测试出来。太棒了!非常感谢你的帮助。lwIP文档缺少这样一个基本示例。不过要小心,我猜用户代码必须在tcpRecvCallback中手动调用pbuf_free(p),否则内存不会从pbuf_池中释放。我在tcp_close(pcb)上也遇到了一些问题,它返回ERR_OK,但pcb永远不会被释放,并且无限期地处于close_WAIT状态。在调用tcp\u close之前,我必须将tcp\u input\u pcb=NULL设置为一种变通方法……这对我来说很有效。谢谢@tgun926。一个问题是,我使用它每5分钟向服务器发送一次数据。当tcp_connect调用时,它在分段部分阻塞。你能帮我吗?我想用这个函数向服务器发送数据。我有多台服务器。所以每次我都要用不同的IP连接tcp。在调用sys_init()和lwip_init()之后,我尝试在主循环中使用您的代码,但除非我显式地提供(并绑定)我的本地IP地址,否则我似乎无法发出HTTP请求。你知道为什么会这样吗?在您提供的文档链接中,它说调用tcp_bind()是可选的(在我的例子中不是这样)。