mirror of
https://github.com/apache/nuttx.git
synced 2026-06-07 01:05:54 +08:00
uIP webserver now uses listen/accept
git-svn-id: svn://svn.code.sf.net/p/nuttx/code/trunk@386 42af7a65-404d-4744-a932-0658087f49c3
This commit is contained in:
+4
-2
@@ -39,12 +39,14 @@ MKDEP = $(TOPDIR)/tools/mkdeps.sh
|
||||
|
||||
ifeq ($(CONFIG_NET),y)
|
||||
include uiplib/Make.defs
|
||||
include dhcpc/Make.defs
|
||||
include resolv/Make.defs
|
||||
include smtp/Make.defs
|
||||
include telnetd/Make.defs
|
||||
include webclient/Make.defs
|
||||
include webserver/Make.defs
|
||||
ifeq ($(CONFIG_NET_UDP),y)
|
||||
include dhcpc/Make.defs
|
||||
include resolv/Make.defs
|
||||
endif
|
||||
include Make.str
|
||||
endif
|
||||
|
||||
|
||||
@@ -35,4 +35,4 @@
|
||||
|
||||
UIPLIB_ASRCS =
|
||||
UIPLIB_CSRCS = uiplib.c uip-setmacaddr.c uip-getmacaddr.c uip-sethostaddr.c \
|
||||
uip-gethostaddr.c uip-setdraddr.c uip-setnetmask.c
|
||||
uip-gethostaddr.c uip-setdraddr.c uip-setnetmask.c uip-server.c
|
||||
|
||||
@@ -81,7 +81,7 @@ int uip_gethostaddr(const char *ifname, struct in_addr *addr)
|
||||
int ret = ERROR;
|
||||
if (ifname && addr)
|
||||
{
|
||||
int sockfd = socket(PF_INET, SOCK_DGRAM, 0);
|
||||
int sockfd = socket(PF_INET, UIPLIB_SOCK_IOCTL, 0);
|
||||
if (sockfd >= 0)
|
||||
{
|
||||
struct ifreq req;
|
||||
|
||||
@@ -75,7 +75,7 @@ int uip_getmacaddr(const char *ifname, uint8 *macaddr)
|
||||
{
|
||||
/* Get a socket (only so that we get access to the INET subsystem) */
|
||||
|
||||
int sockfd = socket(PF_INET, SOCK_DGRAM, 0);
|
||||
int sockfd = socket(PF_INET, UIPLIB_SOCK_IOCTL, 0);
|
||||
if (sockfd >= 0)
|
||||
{
|
||||
struct ifreq req;
|
||||
|
||||
@@ -0,0 +1,191 @@
|
||||
/****************************************************************************
|
||||
* netutils/uiplib/uip-server.c
|
||||
*
|
||||
* Copyright (C) 2007 Gregory Nutt. All rights reserved.
|
||||
* Author: Gregory Nutt <spudmonkey@racsa.co.cr>
|
||||
*
|
||||
* 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 Gregory Nutt 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 COPYRIGHT HOLDERS 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
|
||||
* COPYRIGHT OWNER 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.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include <sys/types.h>
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
|
||||
#include <unistd.h>
|
||||
#include <sched.h>
|
||||
#include <errno.h>
|
||||
#include <debug.h>
|
||||
|
||||
#include <net/uip/uip-lib.h>
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
#define errno *get_errno_ptr()
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: uip_server
|
||||
*
|
||||
* Description:
|
||||
* Implement basic server logic
|
||||
*
|
||||
* Parameters:
|
||||
* portno The port to listen on (in network byte order)
|
||||
* handler The entrypoint of the task to spawn when a new connection is
|
||||
* accepted.
|
||||
* stacksize The stack size needed by the spawned task
|
||||
*
|
||||
* Return:
|
||||
* Does not return unless an error occurs.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
void uip_server(uint16 portno, main_t handler, int stacksize)
|
||||
{
|
||||
struct sockaddr_in myaddr;
|
||||
#ifdef CONFIG_NET_HAVE_SOLINGER
|
||||
struct linger ling;
|
||||
#endif
|
||||
struct sched_param param;
|
||||
socklen_t addrlen;
|
||||
const char *argv[2];
|
||||
int listensd;
|
||||
int acceptsd;
|
||||
#ifdef CONFIG_NET_HAVE_REUSEADDR
|
||||
int optval;
|
||||
#endif
|
||||
|
||||
/* Create a new TCP socket to use to listen for connections */
|
||||
|
||||
listensd = socket(PF_INET, SOCK_STREAM, 0);
|
||||
if (listensd < 0)
|
||||
{
|
||||
dbg("socket failure: %d\n", errno);
|
||||
return;
|
||||
}
|
||||
|
||||
/* Set socket to reuse address */
|
||||
|
||||
#ifdef CONFIG_NET_HAVE_REUSEADDR
|
||||
optval = 1;
|
||||
if (setsockopt(listensd, SOL_SOCKET, SO_REUSEADDR, (void*)&optval, sizeof(int)) < 0)
|
||||
{
|
||||
dbg("setsockopt SO_REUSEADDR failure: %d\n", errno);
|
||||
goto errout_with_socket;
|
||||
}
|
||||
#endif
|
||||
|
||||
/* Bind the socket to a local address */
|
||||
|
||||
myaddr.sin_family = AF_INET;
|
||||
myaddr.sin_port = portno;
|
||||
myaddr.sin_addr.s_addr = INADDR_ANY;
|
||||
|
||||
if (bind(listensd, (struct sockaddr*)&myaddr, sizeof(struct sockaddr_in)) < 0)
|
||||
{
|
||||
dbg("bind failure: %d\n", errno);
|
||||
goto errout_with_socket;
|
||||
}
|
||||
|
||||
/* Listen for connections on the bound TCP socket */
|
||||
|
||||
if (listen(listensd, 5) < 0)
|
||||
{
|
||||
dbg("listen failure %d\n", errno);
|
||||
goto errout_with_socket;
|
||||
}
|
||||
|
||||
/* Begin accepting connections */
|
||||
|
||||
dbg("Accepting connections on port %d\n", ntohs(portno));
|
||||
for (;;)
|
||||
{
|
||||
addrlen = sizeof(struct sockaddr_in);
|
||||
acceptsd = accept(listensd, (struct sockaddr*)&myaddr, &addrlen);
|
||||
if (acceptsd < 0)
|
||||
{
|
||||
dbg("accept failure: %d\n", errno);
|
||||
break;;
|
||||
}
|
||||
dbg("Connection accepted -- spawning\n");
|
||||
|
||||
/* Configure to "linger" until all data is sent when the socket is closed */
|
||||
|
||||
#ifdef CONFIG_NET_HAVE_SOLINGER
|
||||
ling.l_onoff = 1;
|
||||
ling.l_linger = 30; /* timeout is seconds */
|
||||
if (setsockopt(acceptsd, SOL_SOCKET, SO_LINGER, &ling, sizeof(struct linger)) < 0)
|
||||
{
|
||||
close(acceptsd);
|
||||
dbg("setsockopt SO_LINGER failure: %d\n", errno);
|
||||
break;;
|
||||
}
|
||||
#endif
|
||||
|
||||
/* Spawn a thread to handle the connection. The socket descriptor +1 is
|
||||
* provided in as the single argument to the new thread. (The +1 is intended
|
||||
* to handle the valid, zero file descriptor).
|
||||
*/
|
||||
|
||||
if (sched_getparam(0, ¶m) < 0)
|
||||
{
|
||||
close(acceptsd);
|
||||
dbg("sched_getparam failed: %d\n", errno);
|
||||
break;;
|
||||
}
|
||||
|
||||
argv[0] = (char*)(acceptsd + 1);
|
||||
argv[1] = NULL;
|
||||
|
||||
if (task_create("", param.sched_priority, stacksize, handler, argv) < 0)
|
||||
{
|
||||
close(acceptsd);
|
||||
dbg("task_create failed: %d\n", errno);
|
||||
break;;
|
||||
}
|
||||
|
||||
/* We can close our copy of acceptsd now. This file descriptor was dup'ed
|
||||
* by task_create and we no longer need to retain the reference.
|
||||
*/
|
||||
|
||||
close(acceptsd);
|
||||
}
|
||||
|
||||
errout_with_socket:
|
||||
close(listensd);
|
||||
}
|
||||
@@ -78,7 +78,7 @@ int uip_setdraddr(const char *ifname, const struct in_addr *addr)
|
||||
int ret = ERROR;
|
||||
if (ifname && addr)
|
||||
{
|
||||
int sockfd = socket(PF_INET, SOCK_DGRAM, 0);
|
||||
int sockfd = socket(PF_INET, UIPLIB_SOCK_IOCTL, 0);
|
||||
if (sockfd >= 0)
|
||||
{
|
||||
struct ifreq req;
|
||||
|
||||
@@ -78,7 +78,7 @@ int uip_sethostaddr(const char *ifname, const struct in_addr *addr)
|
||||
int ret = ERROR;
|
||||
if (ifname && addr)
|
||||
{
|
||||
int sockfd = socket(PF_INET, SOCK_DGRAM, 0);
|
||||
int sockfd = socket(PF_INET, UIPLIB_SOCK_IOCTL, 0);
|
||||
if (sockfd >= 0)
|
||||
{
|
||||
struct ifreq req;
|
||||
|
||||
@@ -86,7 +86,7 @@ int uip_setmacaddr(const char *ifname, const uint8 *macaddr)
|
||||
{
|
||||
/* Get a socket (only so that we get access to the INET subsystem) */
|
||||
|
||||
int sockfd = socket(PF_INET, SOCK_DGRAM, 0);
|
||||
int sockfd = socket(PF_INET, UIPLIB_SOCK_IOCTL, 0);
|
||||
if (sockfd >= 0)
|
||||
{
|
||||
struct ifreq req;
|
||||
|
||||
@@ -78,7 +78,7 @@ int uip_setnetmask(const char *ifname, const struct in_addr *addr)
|
||||
int ret = ERROR;
|
||||
if (ifname && addr)
|
||||
{
|
||||
int sockfd = socket(PF_INET, SOCK_DGRAM, 0);
|
||||
int sockfd = socket(PF_INET, UIPLIB_SOCK_IOCTL, 0);
|
||||
if (sockfd >= 0)
|
||||
{
|
||||
struct ifreq req;
|
||||
|
||||
@@ -125,7 +125,7 @@ static void file_stats(struct httpd_state *pstate, char *ptr)
|
||||
char buffer[16];
|
||||
char *pcount = strchr(ptr, ' ') + 1;
|
||||
snprintf(buffer, 16, "%5u", httpd_fs_count(pcount));
|
||||
(void)send(pstate->sockout, buffer, strlen(buffer), 0);
|
||||
(void)send(pstate->sockfd, buffer, strlen(buffer), 0);
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
+147
-79
@@ -45,11 +45,17 @@
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <nuttx/config.h>
|
||||
|
||||
#include <sys/socket.h>
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <debug.h>
|
||||
|
||||
#include <net/uip/uip.h>
|
||||
#include <net/uip/uip-lib.h>
|
||||
#include <net/uip/httpd.h>
|
||||
|
||||
#include "httpd.h"
|
||||
@@ -60,9 +66,6 @@
|
||||
* Definitions
|
||||
****************************************************************************/
|
||||
|
||||
#define STATE_WAITING 0
|
||||
#define STATE_OUTPUT 1
|
||||
|
||||
#define ISO_nl 0x0a
|
||||
#define ISO_space 0x20
|
||||
#define ISO_bang 0x21
|
||||
@@ -71,21 +74,50 @@
|
||||
#define ISO_slash 0x2f
|
||||
#define ISO_colon 0x3a
|
||||
|
||||
#define SEND_STR(psock, str) psock_send(psock, str, strlen(str))
|
||||
#define CONFIG_NETUTILS_HTTPD_DUMPBUFFER 1
|
||||
|
||||
/****************************************************************************
|
||||
* Private Functions
|
||||
****************************************************************************/
|
||||
|
||||
static inline int send_file(struct httpd_state *pstate)
|
||||
#ifdef CONFIG_NETUTILS_HTTPD_DUMPBUFFER
|
||||
static void httpd_dumpbuffer(struct httpd_state *pstate, ssize_t nbytes)
|
||||
{
|
||||
return send(pstate->sockout, pstate->file.data, pstate->file.len, 0);
|
||||
}
|
||||
#ifdef CONFIG_DEBUG
|
||||
char line[128];
|
||||
int ch;
|
||||
int i;
|
||||
int j;
|
||||
|
||||
static inline int send_part_of_file(struct httpd_state *pstate)
|
||||
{
|
||||
return send(pstate->sockout, pstate->file.data, pstate->len, 0);
|
||||
for (i = 0; i < nbytes; i += 16)
|
||||
{
|
||||
sprintf(line, "%04x: ", i);
|
||||
for ( j = 0; j < 16; j++)
|
||||
{
|
||||
if (i + j < nbytes)
|
||||
{
|
||||
sprintf(&line[strlen(line)], "%02x ", pstate->ht_buffer[i+j] );
|
||||
}
|
||||
else
|
||||
{
|
||||
strcpy(&line[strlen(line)], " ");
|
||||
}
|
||||
}
|
||||
for ( j = 0; j < 16; j++)
|
||||
{
|
||||
if (i + j < nbytes)
|
||||
{
|
||||
ch = pstate->ht_buffer[i+j];
|
||||
sprintf(&line[strlen(line)], "%c", ch >= 0x20 && ch <= 0x7e ? ch : '.');
|
||||
}
|
||||
}
|
||||
dbg("%s", line);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#else
|
||||
# define httpd_dumpbuffer(pstate,nbytes)
|
||||
#endif
|
||||
|
||||
static void next_scriptstate(struct httpd_state *pstate)
|
||||
{
|
||||
@@ -95,35 +127,40 @@ static void next_scriptstate(struct httpd_state *pstate)
|
||||
pstate->scriptptr = p;
|
||||
}
|
||||
|
||||
static void handle_script(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
static void handle_script(struct httpd_state *pstate)
|
||||
{
|
||||
char *ptr;
|
||||
|
||||
while(pstate->file.len > 0) {
|
||||
|
||||
/* Check if we should start executing a script. */
|
||||
/* Check if we should start executing a script */
|
||||
|
||||
if (*pstate->file.data == ISO_percent &&
|
||||
*(pstate->file.data + 1) == ISO_bang) {
|
||||
pstate->scriptptr = pstate->file.data + 3;
|
||||
pstate->scriptlen = pstate->file.len - 3;
|
||||
if (*(pstate->scriptptr - 1) == ISO_colon) {
|
||||
httpd_fs_open(pstate->scriptptr + 1, &pstate->file);
|
||||
send_file(pstate);
|
||||
} else {
|
||||
httpd_cgi(pstate->scriptptr)(pstate, pstate->scriptptr);
|
||||
}
|
||||
if (*(pstate->scriptptr - 1) == ISO_colon)
|
||||
{
|
||||
httpd_fs_open(pstate->scriptptr + 1, &pstate->file);
|
||||
send(pstate->sockfd, pstate->file.data, pstate->file.len, 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
httpd_cgi(pstate->scriptptr)(pstate, pstate->scriptptr);
|
||||
}
|
||||
next_scriptstate(pstate);
|
||||
|
||||
/* The script is over, so we reset the pointers and continue
|
||||
sending the rest of the file. */
|
||||
sending the rest of the file */
|
||||
pstate->file.data = pstate->scriptptr;
|
||||
pstate->file.len = pstate->scriptlen;
|
||||
|
||||
} else {
|
||||
/* See if we find the start of script marker in the block of HTML
|
||||
to be sent. */
|
||||
to be sent */
|
||||
|
||||
if (pstate->file.len > uip_mss(conn)) {
|
||||
pstate->len = uip_mss(conn);
|
||||
if (pstate->file.len > HTTPD_IOBUFFER_SIZE) {
|
||||
pstate->len = HTTPD_IOBUFFER_SIZE;
|
||||
} else {
|
||||
pstate->len = pstate->file.len;
|
||||
}
|
||||
@@ -136,11 +173,11 @@ static void handle_script(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
if (ptr != NULL &&
|
||||
ptr != pstate->file.data) {
|
||||
pstate->len = (int)(ptr - pstate->file.data);
|
||||
if (pstate->len >= uip_mss(conn)) {
|
||||
pstate->len = uip_mss(conn);
|
||||
if (pstate->len >= HTTPD_IOBUFFER_SIZE) {
|
||||
pstate->len = HTTPD_IOBUFFER_SIZE;
|
||||
}
|
||||
}
|
||||
send_part_of_file(pstate);
|
||||
send(pstate->sockfd, pstate->file.data, pstate->len, 0);
|
||||
pstate->file.data += pstate->len;
|
||||
pstate->file.len -= pstate->len;
|
||||
}
|
||||
@@ -152,41 +189,41 @@ static int send_headers(struct httpd_state *pstate, const char *statushdr)
|
||||
char *ptr;
|
||||
int ret;
|
||||
|
||||
ret = send(pstate->sockout, statushdr, strlen(statushdr), 0);
|
||||
ret = send(pstate->sockfd, statushdr, strlen(statushdr), 0);
|
||||
|
||||
ptr = strrchr(pstate->filename, ISO_period);
|
||||
if (ptr == NULL)
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_binary, strlen(http_content_type_binary), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_binary, strlen(http_content_type_binary), 0);
|
||||
}
|
||||
else if (strncmp(http_html, ptr, 5) == 0 || strncmp(http_shtml, ptr, 6) == 0)
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_html, strlen(http_content_type_html), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_html, strlen(http_content_type_html), 0);
|
||||
}
|
||||
else if (strncmp(http_css, ptr, 4) == 0)
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_css, strlen(http_content_type_css), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_css, strlen(http_content_type_css), 0);
|
||||
}
|
||||
else if (strncmp(http_png, ptr, 4) == 0)
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_png, strlen(http_content_type_png), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_png, strlen(http_content_type_png), 0);
|
||||
}
|
||||
else if (strncmp(http_gif, ptr, 4) == 0)
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_gif, strlen(http_content_type_gif), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_gif, strlen(http_content_type_gif), 0);
|
||||
}
|
||||
else if (strncmp(http_jpg, ptr, 4) == 0)
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_jpg, strlen(http_content_type_jpg), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_jpg, strlen(http_content_type_jpg), 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = send(pstate->sockout, http_content_type_plain, strlen(http_content_type_plain), 0);
|
||||
ret = send(pstate->sockfd, http_content_type_plain, strlen(http_content_type_plain), 0);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void handle_output(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
static void httpd_sendfile(struct httpd_state *pstate)
|
||||
{
|
||||
char *ptr;
|
||||
|
||||
@@ -195,7 +232,7 @@ static void handle_output(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
httpd_fs_open(http_404_html, &pstate->file);
|
||||
strcpy(pstate->filename, http_404_html);
|
||||
send_headers(pstate, http_header_404);
|
||||
send_file(pstate);
|
||||
send(pstate->sockfd, pstate->file.data, pstate->file.len, 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
@@ -203,75 +240,102 @@ static void handle_output(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
ptr = strchr(pstate->filename, ISO_period);
|
||||
if (ptr != NULL && strncmp(ptr, http_shtml, 6) == 0)
|
||||
{
|
||||
handle_script(pstate, conn);
|
||||
handle_script(pstate);
|
||||
}
|
||||
else
|
||||
{
|
||||
send_file(pstate);
|
||||
send(pstate->sockfd, pstate->file.data, pstate->file.len, 0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_input(struct httpd_state *pstate)
|
||||
static inline int httpd_cmd(struct httpd_state *pstate)
|
||||
{
|
||||
ssize_t recvlen;
|
||||
int i;
|
||||
|
||||
if (recv(pstate->sockin, pstate->inputbuf, HTTPD_INBUFFER_SIZE, 0) < 0)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
/* Get the next HTTP command. We will handle only GET */
|
||||
|
||||
if (strncmp(pstate->inputbuf, http_get, 4) != 0)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
recvlen = recv(pstate->sockin, pstate->inputbuf, HTTPD_INBUFFER_SIZE, 0);
|
||||
recvlen = recv(pstate->sockfd, pstate->ht_buffer, HTTPD_IOBUFFER_SIZE, 0);
|
||||
if (recvlen < 0)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
httpd_dumpbuffer(pstate, recvlen);
|
||||
|
||||
if (pstate->inputbuf[0] != ISO_slash)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
/* We will handle only GET */
|
||||
|
||||
if (pstate->inputbuf[1] == ISO_space)
|
||||
if (strncmp(pstate->ht_buffer, http_get, 4) != 0)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
/* Get the name of the file to provide */
|
||||
|
||||
if (pstate->ht_buffer[4] != ISO_slash)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
else if (pstate->ht_buffer[5] == ISO_space)
|
||||
{
|
||||
strncpy(pstate->filename, http_index_html, sizeof(pstate->filename));
|
||||
}
|
||||
else
|
||||
{
|
||||
pstate->inputbuf[recvlen - 1] = 0;
|
||||
strncpy(pstate->filename, &pstate->inputbuf[0], sizeof(pstate->filename));
|
||||
}
|
||||
|
||||
pstate->state = STATE_OUTPUT;
|
||||
|
||||
while(1)
|
||||
{
|
||||
recvlen = recv(pstate->sockin, pstate->inputbuf, HTTPD_INBUFFER_SIZE, 0);
|
||||
if (recvlen < 0)
|
||||
for (i = 5; i < sizeof(pstate->filename) + 5 && pstate->ht_buffer[5] != ISO_space; i++)
|
||||
{
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
if (strncmp(pstate->inputbuf, http_referer, 8) == 0)
|
||||
{
|
||||
pstate->inputbuf[recvlen - 2] = 0;
|
||||
pstate->filename[i] = pstate->ht_buffer[i+5];
|
||||
}
|
||||
}
|
||||
|
||||
/* Then send the file */
|
||||
|
||||
httpd_sendfile(pstate);
|
||||
return OK;
|
||||
}
|
||||
|
||||
static void handle_connection(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
/****************************************************************************
|
||||
* Name: httpd_handler
|
||||
*
|
||||
* Description:
|
||||
* Each time a new connection to port 80 is made, a new thread is created
|
||||
* that begins at this entry point. There should be exactly one argument
|
||||
* and it should be the socket descriptor (+1).
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
static int httpd_handler(int argc, char *argv[])
|
||||
{
|
||||
handle_input(pstate);
|
||||
if (pstate->state == STATE_OUTPUT) {
|
||||
handle_output(pstate, conn);
|
||||
struct httpd_state *pstate = (struct httpd_state *)malloc(sizeof(struct httpd_state));
|
||||
int sockfd = (int)argv[1] - 1;
|
||||
int ret = ERROR;
|
||||
|
||||
/* Verify that the state structure was successfully allocated */
|
||||
|
||||
if (pstate)
|
||||
{
|
||||
/* Loop processing each HTTP command */
|
||||
do
|
||||
{
|
||||
/* Re-initialize the thread state structure */
|
||||
|
||||
memset(pstate, 0, sizeof(struct httpd_state));
|
||||
pstate->sockfd = sockfd;
|
||||
|
||||
/* Then handle the next httpd command */
|
||||
|
||||
ret = httpd_cmd(pstate);
|
||||
}
|
||||
while (ret == OK);
|
||||
|
||||
/* End of command processing -- Clean up and exit */
|
||||
|
||||
free(pstate);
|
||||
}
|
||||
|
||||
/* Exit the task */
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
@@ -288,20 +352,24 @@ static void handle_connection(struct httpd_state *pstate, struct uip_conn *conn)
|
||||
|
||||
int httpd_listen(void)
|
||||
{
|
||||
#warning "this is all very broken at the moment"
|
||||
return OK;
|
||||
/* Execute httpd_handler on each connection to port 80 */
|
||||
|
||||
uip_server(HTONS(80), httpd_handler, CONFIG_EXAMPLES_UIP_HTTPDSTACKSIZE);
|
||||
|
||||
/* uip_server only returns on errors */
|
||||
|
||||
return ERROR;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: httpd_init
|
||||
*
|
||||
* Description:
|
||||
* This function initializes the web server and should be called at system
|
||||
* boot-up.
|
||||
* This function initializes the web server and should be called at system
|
||||
* boot-up.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
void httpd_init(void)
|
||||
{
|
||||
uip_listen(HTONS(80));
|
||||
}
|
||||
|
||||
+39
-14
@@ -1,11 +1,19 @@
|
||||
/* httpd.h
|
||||
/****************************************************************************
|
||||
* netutils/webserver/httpd.h
|
||||
*
|
||||
* Copyright (c) 2001-2005, Adam Dunkels.
|
||||
* All rights reserved.
|
||||
* Copyright (C) 2007 Gregory Nutt. All rights reserved.
|
||||
* Author: Gregory Nutt <spudmonkey@racsa.co.cr>
|
||||
*
|
||||
* Based on uIP which also has a BSD style license:
|
||||
*
|
||||
* Author: Adam Dunkels <adam@sics.se>
|
||||
* Copyright (c) 2001-2005, Adam Dunkels.
|
||||
* 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
|
||||
@@ -26,15 +34,33 @@
|
||||
* 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.
|
||||
*/
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
#ifndef _NETUTILS_WEBSERVER_HTTPD_H
|
||||
#define _NETUTILS_WEBSERVER_HTTPD_H
|
||||
|
||||
/****************************************************************************
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include <nuttx/config.h>
|
||||
#include <sys/types.h>
|
||||
|
||||
/****************************************************************************
|
||||
* Definitions
|
||||
****************************************************************************/
|
||||
|
||||
#define HTTPD_FS_STATISTICS 1
|
||||
#define HTTPD_INBUFFER_SIZE 50
|
||||
#define HTTPD_IOBUFFER_SIZE 512
|
||||
|
||||
#ifndef CONFIG_EXAMPLES_UIP_HTTPDSTACKSIZE
|
||||
# define CONFIG_EXAMPLES_UIP_HTTPDSTACKSIZE 4096
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Public Types
|
||||
****************************************************************************/
|
||||
|
||||
struct httpd_fs_file
|
||||
{
|
||||
@@ -44,13 +70,10 @@ struct httpd_fs_file
|
||||
|
||||
struct httpd_state
|
||||
{
|
||||
unsigned char timer;
|
||||
int sockin;
|
||||
int sockout;
|
||||
char inputbuf[HTTPD_INBUFFER_SIZE];
|
||||
char ht_buffer[HTTPD_IOBUFFER_SIZE];
|
||||
char filename[20];
|
||||
char state;
|
||||
struct httpd_fs_file file;
|
||||
int sockfd; /* The socket descriptor from accept() */
|
||||
int len;
|
||||
char *scriptptr;
|
||||
int scriptlen;
|
||||
@@ -58,17 +81,19 @@ struct httpd_state
|
||||
unsigned short count;
|
||||
};
|
||||
|
||||
/****************************************************************************
|
||||
* Public Function Prototypes
|
||||
****************************************************************************/
|
||||
|
||||
#ifdef HTTPD_FS_STATISTICS
|
||||
#if HTTPD_FS_STATISTICS == 1
|
||||
extern uint16 httpd_fs_count(char *name);
|
||||
#endif /* HTTPD_FS_STATISTICS */
|
||||
#endif /* HTTPD_FS_STATISTICS */
|
||||
|
||||
/* file must be allocated by caller and will be filled in
|
||||
* by the function.
|
||||
*/
|
||||
/* file must be allocated by caller and will be filled in by the function. */
|
||||
|
||||
int httpd_fs_open(const char *name, struct httpd_fs_file *file);
|
||||
int httpd_fs_open(const char *name, struct httpd_fs_file *file);
|
||||
void httpd_fs_init(void);
|
||||
|
||||
#endif /* _NETUTILS_WEBSERVER_HTTPD_H */
|
||||
|
||||
Reference in New Issue
Block a user