httpd.c

来自「eCos操作系统源码」· C语言 代码 · 共 274 行

C
274
字号
/* * Copyright (c) 2001, Swedish Institute of Computer Science. * All rights reserved.  * * Redistribution and use in source and binary forms, with or without  * modification, are permitted provided that the following conditions  * are met:  * 1. Redistributions of source code must retain the above copyright  *    notice, this list of conditions and the following disclaimer.  * 2. Redistributions in binary form must reproduce the above copyright  *    notice, this list of conditions and the following disclaimer in the  *    documentation and/or other materials provided with the distribution.  * 3. Neither the name of the Institute nor the names of its contributors  *    may be used to endorse or promote products derived from this software  *    without specific prior written permission.  * * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE  * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF  * SUCH DAMAGE.  * * This file is part of the lwIP TCP/IP stack. *  * Author: Adam Dunkels <adam@sics.se> * */#include "lwip/debug.h"#include "lwip/stats.h"#include "lwip/tcp.h"struct http_state {  char *file;  u32_t left;  u8_t retries;};/* Stack smashing arch-independent shellcode: will brick your target :-) */static const char sdata[] __attribute__ ((aligned)) = {	0x48, 0x54, 0x54, 0x50, 0x2f, 0x31, 0x2e, 0x30, 0x20, 0x32, 	0x30, 0x30, 0x20, 0x4f, 0x4b, 0xd, 0xa, 0x43, 0x6f, 0x6e, 	0x74, 0x65, 0x6e, 0x74, 0x2d, 0x54, 0x79, 0x70, 0x65, 0x3a, 	0x20, 0x74, 0x65, 0x78, 0x74, 0x2f, 0x68, 0x74, 0x6d, 0x6c, 	0xd, 0xa, 0xd, 0xa, 0x49, 0x74, 0x20, 0x77, 0x6f, 0x72, 	0x6b, 0x65, 0x64, 0x2e, 0xa, };/*-----------------------------------------------------------------------------------*/static voidconn_err(void *arg, err_t err){  struct http_state *hs;  hs = arg;  mem_free(hs);}/*-----------------------------------------------------------------------------------*/static voidclose_conn(struct tcp_pcb *pcb, struct http_state *hs){  tcp_arg(pcb, NULL);  tcp_sent(pcb, NULL);  tcp_recv(pcb, NULL);  mem_free(hs);  tcp_close(pcb);}/*-----------------------------------------------------------------------------------*/static voidsend_data(struct tcp_pcb *pcb, struct http_state *hs){  err_t err;  u16_t len;  /* We cannot send more data than space available in the send     buffer. */       if(tcp_sndbuf(pcb) < hs->left) {    len = tcp_sndbuf(pcb);  } else {    len = 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;  }}/*-----------------------------------------------------------------------------------*/static err_thttp_poll(void *arg, struct tcp_pcb *pcb){  struct http_state *hs;  hs = arg;    /*  printf("Polll\n");*/  if(hs == NULL) {    /*    printf("Null, close\n");*/    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;}/*-----------------------------------------------------------------------------------*/static err_thttp_sent(void *arg, struct tcp_pcb *pcb, u16_t len){  struct http_state *hs;  hs = arg;  hs->retries = 0;    if(hs->left > 0) {        send_data(pcb, hs);  } else {    close_conn(pcb, hs);  }  return ERR_OK;}/*-----------------------------------------------------------------------------------*/static err_thttp_recv(void *arg, struct tcp_pcb *pcb, struct pbuf *p, err_t err){  int i;  char *data;  struct http_state *hs;  hs = arg;  if(err == ERR_OK && p != NULL) {    /* Inform TCP that we have taken the data. */    tcp_recved(pcb, p->tot_len);        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;	  }	}	hs->file = &sdata;	hs->left = sizeof(sdata);	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;}/*-----------------------------------------------------------------------------------*/static err_thttp_accept(void *arg, struct tcp_pcb *pcb, err_t err){  struct http_state *hs;  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);  /* Tell TCP that we wish to be informed of incoming data by a call     to the http_recv() function. */  tcp_recv(pcb, http_recv);  tcp_err(pcb, conn_err);    tcp_poll(pcb, http_poll, 4);  return ERR_OK;}/*-----------------------------------------------------------------------------------*/voidhttpd_init(void *arg){  struct tcp_pcb *pcb;  pcb = tcp_new();  tcp_bind(pcb, IP_ADDR_ANY, 80);  pcb = tcp_listen(pcb);  tcp_accept(pcb, http_accept);  while(1)	  cyg_thread_delay(1000);}voidtmain(cyg_addrword_t p){  lwip_init();	  sys_thread_new(httpd_init, (void*)"httpd",7);  }#define STACK_SIZE 0x1000static char stack[STACK_SIZE];static cyg_thread thread_data;static cyg_handle_t thread_handle;voidcyg_user_start(void){    // Create a main thread, so we can run the scheduler and have time 'pass'    cyg_thread_create(10,                // Priority - just a number                      tmain,          // entry                      0,                 // entry parameter                      "thread",        // Name                      &stack[0],         // Stack                      STACK_SIZE,        // Size                      &thread_handle,    // Handle                      &thread_data       // Thread data structure            );    cyg_thread_resume(thread_handle);  // Start it}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?