OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [rtos/] [ecos-3.0/] [packages/] [net/] [lwip_tcpip/] [current/] [tests/] [httpd_simple.c] - Rev 817

Go to most recent revision | Compare with Previous | Blame | View Log

//==========================================================================
//
//      httpd_simple.c
//
//      A simple HTTP server using raw tcp pcbs in 'Simple' mode.
//
//==========================================================================
//####ECOSGPLCOPYRIGHTBEGIN####
// -------------------------------------------
// This file is part of eCos, the Embedded Configurable Operating System.
// Copyright (C) 2008 Free Software Foundation
//
// eCos is free software; you can redistribute it and/or modify it under
// the terms of the GNU General Public License as published by the Free
// Software Foundation; either version 2 or (at your option) any later version.
//
// eCos is distributed in the hope that it will be useful, but WITHOUT ANY
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
// FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
// for more details.
//
// You should have received a copy of the GNU General Public License along
// with eCos; if not, write to the Free Software Foundation, Inc.,
// 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
//
// As a special exception, if other files instantiate templates or use macros
// or inline functions from this file, or you compile this file and link it
// with other works to produce a work based on this file, this file does not
// by itself cause the resulting work to be covered by the GNU General Public
// License. However the source code for this file must still be made available
// in accordance with section (3) of the GNU General Public License.
//
// This exception does not invalidate any other reasons why a work based on
// this file might be covered by the GNU General Public License.
// -------------------------------------------
//####ECOSGPLCOPYRIGHTEND####
//==========================================================================
//#####DESCRIPTIONBEGIN####
//
// Author(s):    Simon Kallweit
// Contributors:
// Date:         2008-12-01
// Purpose:
// Description:  A simple HTTP server using raw tcp pcbs in 'Simple' mode.
//
//####DESCRIPTIONEND####
//
//==========================================================================
 
#include <cyg/hal/hal_arch.h>
#include <cyg/infra/diag.h>
#include <cyg/infra/testcase.h>
#include <cyg/kernel/kapi.h>
 
#include <lwip.h>
#include <lwip/init.h>
 
#ifdef CYGFUN_LWIP_MODE_SIMPLE
#if LWIP_TCP
 
struct http_state {
    char buf[256];
    char *file;
    u32_t left;
    u8_t retries;
};
 
static int request_counter = 1;
 
#define STACK_SIZE CYGNUM_HAL_STACK_SIZE_TYPICAL
 
static char main_stack[STACK_SIZE];
static cyg_thread main_thread;
static cyg_handle_t main_handle;
 
 
// Appends text to the buffer.
static void
buf_append_text(char *buf, const char *text)
{
    char *dst = buf;
    const char *src = text;
 
    while (*dst != '\0')
        dst++;
 
    while (*src != '\0')
        *dst++ = *src++;
 
    *dst = '\0';
}
 
// Appends a decimal value to the buffer.
static void
buf_append_dec(char *buf, int value)
{
    char *dst = buf;
    char tmp[16];
    int i = 0;
 
    while (*dst != '\0')
        dst++;
 
    do {
        tmp[i++] = '0' + value % 10; 
        value /= 10;
    } while (value > 0);
 
    while (i-- > 0)
        *dst++ = tmp[i];
 
    *dst = '\0';
}
 
// Returns the length of the buffer.
static size_t
buf_length(char *buf)
{
    size_t len = 0;
 
    while (*buf++ != '\0')
        len++;
 
    return len;
}
 
// Prepares the HTTP page.
static void
prepare_page(struct http_state *hs)
{
    hs->buf[0] = '\0';
    buf_append_text(hs->buf, "HTTP/1.0 200 OK\r\n");
    buf_append_text(hs->buf, "Content-Type: text/html\r\n");
    buf_append_text(hs->buf, "\r\n");
    buf_append_text(hs->buf, "<html>\r\n");
    buf_append_text(hs->buf, "<h1>eCos - HTTP test server running on lwIP</h1>\r\n");
    buf_append_text(hs->buf, "<table><tr><td>Version:</td><td>");
    buf_append_dec(hs->buf, LWIP_VERSION_MAJOR);
    buf_append_text(hs->buf, ".");
    buf_append_dec(hs->buf, LWIP_VERSION_MINOR);
    buf_append_text(hs->buf, ".");
    buf_append_dec(hs->buf, LWIP_VERSION_REVISION);
    if (LWIP_VERSION_IS_RELEASE)
        buf_append_text(hs->buf, " (release)");
    if (LWIP_VERSION_IS_DEVELOPMENT)
        buf_append_text(hs->buf, " (development)");
    if (LWIP_VERSION_IS_RC)
        buf_append_text(hs->buf, " (rc)");
    buf_append_text(hs->buf, "</td></tr><tr><td>Requests:</td><td>");
    buf_append_dec(hs->buf, request_counter++);
    buf_append_text(hs->buf, "</td></tr></table>\r\n");
    buf_append_text(hs->buf, "</html>");
 
    hs->file = hs->buf;
    hs->left = buf_length(hs->buf);
}
 
// Called when an error occured. Frees the allocated resources.
static void
http_err(void *arg, err_t err)
{
    struct http_state *hs;
 
    CYG_TEST_INFO("Connection error");
 
    hs = arg;
    mem_free(hs);
}
 
// Closes the TCP connection and frees the allocated resources.
static void
close_conn(struct tcp_pcb *pcb, struct http_state *hs)
{
    CYG_TEST_INFO("Closing connection");
 
    tcp_arg(pcb, NULL);
    tcp_sent(pcb, NULL);
    tcp_recv(pcb, NULL);
    mem_free(hs);
    tcp_close(pcb);
}
 
// Sends the next chunk of data.
static void
send_data(struct tcp_pcb *pcb, struct http_state *hs)
{
    err_t err;
    u16_t len;
 
    CYG_TEST_INFO("Sending data");
 
    // We cannot send more data than space available in the send buffer
    len = tcp_sndbuf(pcb) < hs->left ? tcp_sndbuf(pcb) : hs->left;
 
    do {
        err = tcp_write(pcb, hs->file, len, 0);
        if (err == ERR_MEM) {
            len /= 2;
        }
    } while (err == ERR_MEM && len > 1);
 
    if (err == ERR_OK) {
        hs->file += len;
        hs->left -= len;
    }
}
 
// Called in a regular interval. Implements retries.
static err_t
http_poll(void *arg, struct tcp_pcb *pcb)
{
    struct http_state *hs;
 
    CYG_TEST_INFO("Polling");
 
    hs = arg;
 
    if (hs == NULL) {
        tcp_abort(pcb);
        return ERR_ABRT;
    } else {
        ++hs->retries;
        if (hs->retries == 4) {
            tcp_abort(pcb);
            return ERR_ABRT;
        }
        send_data(pcb, hs);
    }
 
    return ERR_OK;
}
 
// Called when data has been sent.
static err_t
http_sent(void *arg, struct tcp_pcb *pcb, u16_t len)
{
    struct http_state *hs;
 
    CYG_TEST_INFO("Sent data");
 
    hs = arg;
 
    hs->retries = 0;
 
    if (hs->left > 0) {
        send_data(pcb, hs);
    } else {
        close_conn(pcb, hs);
    }
 
    return ERR_OK;
}
 
// Called when data has been received. Checks for HTTP requests.
static err_t
http_recv(void *arg, struct tcp_pcb *pcb, struct pbuf *p, err_t err)
{
    int i;
    char *data;
    struct http_state *hs;
 
    CYG_TEST_INFO("Received data");
 
    hs = arg;
 
    if (err == ERR_OK && p != NULL) {
        // Inform TCP that we have taken the data
        tcp_recved(pcb, p->tot_len);
 
        // Check for HTTP requests
        if (hs->file == NULL) {
            data = p->payload;
 
            if (*data == 'G') {
                for (i = 0; i < 40; i++) {
                    if (((char *) data + 4)[i] == ' ' ||
                        ((char *) data + 4)[i] == '\r' ||
                        ((char *) data + 4)[i] == '\n') {
                        ((char *)data + 4)[i] = 0;
                    }
                }
 
                prepare_page(hs);
 
                pbuf_free(p);
                send_data(pcb, hs);
 
                // Tell TCP that we wish be to informed of data that has been
                // successfully sent by a call to the http_sent() function.
                tcp_sent(pcb, http_sent);
            } else {
                pbuf_free(p);
                close_conn(pcb, hs);
            }
        } else {
            pbuf_free(p);
        }
    }
 
    if (err == ERR_OK && p == NULL) {
        close_conn(pcb, hs);
    }
 
    return ERR_OK;
}
 
// Called when a new connection was accepted.
static err_t
http_accept(void *arg, struct tcp_pcb *pcb, err_t err)
{
    struct http_state *hs;
 
    CYG_TEST_INFO("Incoming connection");
 
    tcp_setprio(pcb, TCP_PRIO_MIN);
 
    // Allocate memory for the structure that holds the state of the connection
    hs = mem_malloc(sizeof(struct http_state));
    if (hs == NULL)
        return ERR_MEM;
 
    // Initialize the structure
    hs->file = NULL;
    hs->left = 0;
    hs->retries = 0;
 
    // Tell TCP that this is the structure we wish to be passed for our
    // callbacks
    tcp_arg(pcb, hs);
 
    // Register callbacks
    tcp_recv(pcb, http_recv);
    tcp_err(pcb, http_err);
    tcp_poll(pcb, http_poll, 4);
 
    CYG_TEST_INFO("Connection accepted");
 
    return ERR_OK;
}
 
// Main thread.
void
main_thread_entry(cyg_addrword_t p)
{
    struct tcp_pcb *pcb;
 
    CYG_TEST_INFO("Initializing lwIP");
    cyg_lwip_simple_init();
 
    CYG_TEST_INFO("Initializing tcb");
    pcb = tcp_new();
    tcp_bind(pcb, IP_ADDR_ANY, 80);
    pcb = tcp_listen(pcb);
    tcp_accept(pcb, http_accept);
 
    CYG_TEST_INFO("Running");
    while (1) {
        cyg_lwip_simple_poll();
        cyg_thread_delay(1);
    }
}
 
void
httpd_main(void)
{
    CYG_TEST_INIT();
    CYG_TEST_INFO("Testing httpd");
 
    cyg_thread_create(
        10,                 // Priority
        main_thread_entry,  // Entry
        0,                  // Entry parameter
        "main",             // Name
        main_stack,         // Stack
        STACK_SIZE,         // Size
        &main_handle,       // Handle
        &main_thread        // Thread data structure
    );
    cyg_thread_resume(main_handle);
    cyg_scheduler_start();
 
    CYG_TEST_FAIL_FINISH("Not reached");
}
 
externC void
cyg_start(void)
{
    httpd_main();
}
 
#else // LWIP_TCP
#define N_A_MSG "TCP support disabled"
#endif // LWIP_TCP
 
#else // CYGFUN_LWIP_MODE_SIMPLE
#define N_A_MSG "Not configured in 'Simple' mode"
#endif // CYGFUN_LWIP_MODE_SIMPLE
 
#ifdef N_A_MSG
externC void
cyg_start(void)
{
    CYG_TEST_INIT();
    CYG_TEST_NA(N_A_MSG);
}
#endif // N_A_MSG
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.