web server大全之GoAhead移植(转载)

转自:http://linux.chinaunix.net/techdoc/develop/2009/06/19/1119124.shtml

注:最近在做goAhead web server和移植其到TI芯片+linux上,这里先转一篇相关的文章来学习下,希望有所帮助。。。
*******************************
* web server大全之GoAhead移植 *
*******************************
    2009/02/14  asdjf@163.com  www.armecos.com
    很多人希望在产品中使用Web Server,为此,我们总结了十几种各式各样的Web Server任君选择。Web Server开发再也不是困难的事情了。
    本文档介绍强大的嵌入式Web服务器GoAhead!!!
    它的主要特点是:

    1、支持ASP。
    2、嵌入式JavaScript---Ejscript。
    3、支持标准的CGI。
    4、支持内存中的CGI处理。
    5、快速响应,每秒可处理超过65个请求。
    6、符合HTTP 1.0/HTTP 1.1标准。
    7、拥有众多扩展API,方便用户开发。
    8、支持SSL 3.0.
    9、支持用户群组管理。
    10、支持DAA访问认证。
    11、小内存,如果不包含SSL,仅要求60K的内存:包含SSL,要求500K内存。
    12、Web页面可以存在于ROM或文件系统中。
    13、支持多种操作系统,如:eCos、Linux、LynxOS、QNX、VxWorks、WinCE、pSOS等。

图1 GoAhead运行效果。

图2 GoAhead源程序结构框图

GoAhead Web服务器是GoAhead公司早期推出的一种可以运行于多种平台的小巧而精致的Web服务器,它具有移植性好、开放源代码、代码量小的特点。GoAhead Web服务器特别适合于嵌入式系统。
    GoAhead Web服务器的详细说明文档位于GoAhead源码中的webs/docs目录下,源码可以从http://www.goahead.com下载。注意:由于目前eCos不支持用户群组,因此eCos不支持GoAhead的用户管理和访问控制功能。
    解压缩GoAhead源码到/g目录,可以看到GoAhead源码组织结构如下:
    /g
     |
     |\______各种OS移植子目录(如:ecos子目录)
     |\______Web自目录(用来保存自己设计的网页)
     |\______GoAhead服务器源码(C程序)
      \______webcomp.c网页编译器
    和通常的Web Server不同,我们设计的网页(ASP、html等)在编译阶段就被解析并和服务器源码编译到了一起,而不是象其他服务器那样在运行阶段读取网页并解析内容。
    GoAhead根目录下的webcomp.c网页编译器负责把Web子目录下的所有Web网页进行转换,使其能够与GoAhead Web服务器源码以及eCos其他应用代码一起编译。
    web子目录下是所有Web网页内容。Web服务器的所有网页都必须放置在该目录下。
    eCos子目录包含了与eCos的接口,包括main.c文件和makefile文件。用户根据实际需要可以对main.c和makefile文件进行修改。
    通过阅读ecos目录下的makefile文件可知,GoAhead Web服务器编译过程主要有三个步骤:
    1、编译webcomp.c文件,生成网页编译器webcomp.exe。webcomp.c使用本地编译器gcc进行编译,编译后的网页编译器位于ecos子目录下。网页编译器将web子目录下的所有网页进行转换并生成webcomp.c文件。webcomp.c文件将存放于ecos子目录下。
    2、交叉编译器对GoAhead根目录下的Web服务器源码和网页文件webcomp.c进行编译,生成库文件libwebs.a。
    3、eCos应用程序在编译时与库文件libwebs.a进行链接,生成可运行于目标平台的可执行文件。
    以下是ecos子目录下的makefile文件,从中可以看出这三个步骤的执行过程。

# eCos makefile
all: compile
#
# These definitions come from your eCos install tree
#
DEBUG := -g -Wall -O2
# For Cirrus Logic EDB72xx board
PKG_INSTALL_DIR := /tmp/untitled_install
COMMAND_PREFIX := arm-elf-
CFLAGS := -mcpu=arm7tdmi $(DEBUG)
# For Motorola PowerPC MBX/860
##PKG_INSTALL_DIR := /work/net_mbx/install
##COMMAND_PREFIX := powerpc-eabi-
##CFLAGS := -mcpu=860 -msoft-float $(DEBUG)
# 
# These should not need to be changed
#
CC := $(COMMAND_PREFIX)gcc
OBJCOPY := $(COMMAND_PREFIX)objcopy
AR := $(COMMAND_PREFIX)ar
LDFLAGS = -nostartfiles -L$(PKG_INSTALL_DIR)/lib -Wl,--gc-sections $(LIBS)
LIBS = -Ttarget.ld -nostdlib
CXXFLAGS      = $(CFLAGS)
EXTRACFLAGS   = -Wall -I$(PKG_INSTALL_DIR)/include -ffunction-sections -fdata-sections
EXTRACXXFLAGS = $(EXTRACFLAGS) -fno-exceptions -fno-rtti -fvtable-gc -finit-priority
# eCos build rules
%.o: %.c
$(CC) -c -o $*.o $(CFLAGS) $(EXTRACFLAGS) -Wp,-MD,$*.d $/dev/null
%.o: %.cxx
$(CXX) -c -o $*.o $(CXXFLAGS) $(EXTRACXXFLAGS) $.depend
#
# Build archive of objects
#
$(ARCH): $(OBJ_FILES)
$(AR) $(ARFLAGS) $(ARCH) $?
#
# Primary link
#
$(NAME): Makefile  main.o $(ARCH)
$(CC) -o $(NAME) $(CFLAGS) $(IFLAGS) \
main.o $(ARCH) $(LDFLAGS) 
clean:
rm -f $(NAME) $(ARCH) $(DEPEND_FILES) $(OBJ_FILES) 
rm -f main.o webrom.c webcomp web_files .depend
# 
# This tool needs to be built using the native C compiler
#
webcomp:
gcc -o webcomp -O2 -DWEBS -DUEMF -DOS="Linux" -DLINUX -D_STRUCT_TIMEVAL -I.. ../webcomp.c
#
# Build a set of ROMable pages
#
webrom.c: webcomp
find ../web -name "*.*" >web_files
./webcomp ../web web_files >webrom.c
# Dependencies
-include .depend

    我们修改了makefile中的几个定义:
    1、PKG_INSTALL_DIR := /tmp/untitled_install 指向《ecos增值包》提供的系统库文件。
    2、修改cc为gcc,cygwin环境下编译器为gcc。增加-D_STRUCT_TIMEVAL定义,以避免uemf.h中的struct timeval结构体定义和ecos库中的已有定义冲突。
        gcc -o webcomp -O2 -DWEBS -DUEMF -DOS="Linux" -DLINUX -D_STRUCT_TIMEVAL -I.. ../webcomp.c
    除了makefile需要修改外,main.c文件需要将最后的send()和recv()函数定义注释掉,因为和ecos库里已有的定义冲突。
/******************************************************************************/

/*
* Wrappers for depreciated socket I/O functions
*/
/*
int send(int s, const void *buf, size_t len, int flags)
{
    return write(s, buf, len);
}
int recv(int s, void *buf, size_t len, int flags)
{
    return read(s, buf, len);
}
*/

/******************************************************************************/
    根目录下sockGen.c文件中需要增加如下定义,以避免编译错误。
    #include "sys/select.h"
    #define NFDBITS __NFDBITS
    根目录下uemf.h中的下列定义冲突,注释掉即可。
    //#define O_RDONLY 1
    根目录下wsIntrn.h中增加下列引用,以避免编译错误。

#ifdef ECOS
#include 
#include 
#endif

    上面讲的都是GoAhead本身的修改,对于我们应用来说,还需要修改ecos目录下的main.c文件。移植时通常需要修改两个地方:ecos的入口点函数main()和Web服务器的初始化函数initWebs()。
    1、Web服务器的启动。main.c文件主要用于对Web服务器进行独立的测试和调试,因此可以直接使用main()函数来启动Web服务器,但是在实际项目开发中,GoAhead Web服务器通常只是eCos应用软件的一个功能模块,这种情况下,可以将Web服务器当成一个线程来启动。
    下面的代码就是把Web服务器当成线程启动的一个实例。线程入口函数goahead_program()就是原来main.c文件中的main()入口函数。代码中的Web服务器的线程优先级为16,线程名为"GoAhead Web Server"。eCos应用程序通过调用do_webs()函数来启动Web服务器线程。这种情况下,最好是修改main.c文件名并将其加入到eCos应用程序项目中,与其他源码程序一起编译。此时使用的makefile文件可参考《第十二讲 多目录下makefile的通用写法》文档。

#include  "../uemf.h"
#include  "../wsIntrn.h"
#include  
cyg_handle_t webs_thread_handle;
cyg_thread webs_thread_s; //space for web thread objects
char webs_stack[4096];    //space for 4K stacks
cyg_thread_entry_t goahead_program;
void do_webs(int argc, char *argv[])
{
    cyg_thread_create(16, goahead_program, (cyg_addrword_t) 0,
                      "GoAhead Web Server", (void *)webs_stack, 163840,
                      &webs_thread_handle, &webs_thread_s);
    cyg_thread_resume(webs_thread_handle);
}
void goahead_program(cyg_addrword_t data)
{
    bopen(NULL, (60 * 1024), B_USE_MALLOC);
    if (initWebs() Expanded ASP data:

    在对eCos子目录下的makefile文件、main.c文件、web子目录的网页内容及根目录下的若干文件进行修改后,在Cygwin环境下首先进入到Web服务器源码的eCos子目录,然后直接使用make命令就可以完成GoAhead的编译过程。使用make clean可以清除编译垃圾,当修改了Web服务器源码、网页内容和main.c文件后,都必须使用该命令清除前一次的编译结果和编译中间文件,否则,程序运行可能不正常。
    如果只是修改了main.c文件,那么可以直接使用下面命令进行eCos应用程序编译:
    $arm-elf-gcc main.c -o webs -g -DWEBS -DUEMF -DWEBVS_PAGE_ROM
                 -DOS="eCos" -DECOS -D__ECOS -D__NO_FCNTL=1
                 -I.. -I/h/ecos-work/mywork_install/include
                 -L/h/ecos-work/mywork_install/lib libwebs.a
                 -Ttarget.ld -nostdlib -Wall -Wl,--gc-sections
    该应用程序直接使用了前面已经编译好的库文件libwebs.a。当eCos应用程序包含多个源码文件时,可参考《第十二讲 多目录下makefile的通用写法》文档。命令中第三行的两个“-I”分别指定了GoAhead和eCos的头文件路径,第四行指定了eCos的库文件路径和GoAhead Web服务器库文件libwebs.a。实际使用时要根据具体路径进行修改。
图3 JavaScript测试

图4 ASP表单Form测试输入

图5 ASP表单Form测试输出结果

网页设计举例
    前面已经对ASP网页的内嵌函数进行了说明。这里我们讨论ASP网页的设计方法。下面是GoAhead Web服务器源码中的一个表单网页forms.asp:
GoForm Test
Name:
Address:
forms.asp是一个提交姓名和地址的页面,它调用CGI程序formTest对表单进行处理。formTest是一个内存CGI程序,它必须在initWebs()函数中使用websFormdefine()进行定义。main.c中提供了一个表单处理函数的例子:

static void formTest(webs_t wp, char_t *path, char_t *query)
{
    char_t *name, *address;
    name = websGetVar(wp, T("name"), T("Joe Smith")); 
    address = websGetVar(wp, T("address"), T("1212 Milky Way Ave.")); 
    websHeader(wp);
    websWrite(wp, T(" 
Name: %s, Address: %s\n"), name, address);
    websFooter(wp);
    websDone(wp, 200);
}

    该表单处理CGI程序首先获取name和address两个变量值,然后再将表单输入的内容以单独的一个页面进行输出。从formTest函数的最后四行还可以看出页面输出的四个基本函数。图4和图5为表单输入和输出的两个页面。

原文地址:https://www.cnblogs.com/xiangwengao/p/3051409.html