• 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为表单输入和输出的两个页面。

  • 相关阅读:
    PHP简单模拟登录功能实例分享
    一个form表单,多个提交按钮
    jquery validation验证身份证号、护照、电话号码、email
    MockMvc和Mockito之酷炫使用
    Java8 Stream API
    第一章 Lambda表达式
    Java中线程顺序执行
    单元测试之获取Spring下所有Bean
    iBatis之type
    json解析之jackson ObjectMapper
  • 原文地址:https://www.cnblogs.com/xiangwengao/p/3051409.html
Copyright © 2020-2023  润新知