Как я могу отправить простой HTTP-запрос со стеком lwIP?

пожалуйста, переместите / закройте это, если вопрос не имеет значения.

Ядро: Cortex-M4

микропроцессор: TI TM4C1294NCPDT.

стек IP: lwIP 1.4.1

Я использую этот микропроцессор для ведения журнала данных, и я хочу отправить некоторую информацию на отдельный веб-сервер через HTTP-запрос в виде:

http://123.456.789.012:8800/process.php?data1=foo&data2=bar&time=1234568789

и я хочу, чтобы процессор мог видеть заголовок ответа (i.e если это было 200 OK или что - то пошло не так) - он не должен отображать/получать фактическое содержимое.

lwIP имеет http-сервер для микропроцессора, но я после противоположного (микропроцессор является клиентом).

Я не уверен, как пакеты коррелируют с заголовками запроса/ответа, поэтому я не уверен, как я должен фактически отправлять/получать информацию.

2 ответов


это оказалось довольно простым в реализации, забыл обновить этот вопрос.

Я в значительной степени следовал инструкциям, данным на этой сайт, который является Raw / TCP "документация".

в основном, HTTP-запрос кодируется в TCP-пакетах, поэтому для отправки данных на мой PHP-сервер я отправил HTTP-запрос с использованием TCP-пакетов (lwIP выполняет всю работу).

HTTP-пакет, который я хочу отправить, выглядит так:

глава /процесс.РНР?data1=12 & data2=5 HTTP / 1.0

хост: mywebsite.com

чтобы " перевести "это в текст, который понимается HTTP-сервером, вы должны добавить" \r\n " Carriage return/newline в свой код. Так это выглядит так:

char *string = "HEAD /process.php?data1=12&data2=5 HTTP/1.0\r\nHost: mywebsite.com\r\n\r\n ";

обратите внимание, что конец имеет два лота "\r\n"

вы можете использовать GET или HEAD, но поскольку я не заботился о HTML-сайте, Мой PHP-сервер вернулся, я использовал 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-сервером установлено, 'connectCallback' функция вызывается lwIP:

/* 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;
}

эта функция вызывает фактическую функцию tcp_send_packet (), которая отправляет HTTP-запрос следующим образом:

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->полезная нагрузка содержит фактическое "200 OK" и т. д. информация. Надеюсь, это кому-то поможет.

я пропустил некоторую проверку ошибок в моем коде выше, чтобы упростить ответ.


посмотри пример HTTP в Википедии. Клиент отправит строки GET и HOST. Сервер ответит множеством строк для ответа. В первой строке будет код ответа.