实践操作整理:嵌入式linux上通过boa服务器实现cgi的web上网

[复制链接]
5030|6
 楼主| db10 发表于 2008-8-5 17:10 | 显示全部楼层 |阅读模式
&nbsp;&nbsp;&nbsp;&nbsp;嵌入式linux上通过boa服务器实现cgi/html的web上网<br /><br />简介&nbsp;&nbsp;&nbsp;&nbsp;2<br />第一步&nbsp;Boa程序的移植&nbsp;&nbsp;&nbsp;&nbsp;2<br />1.下载Boa源码&nbsp;&nbsp;&nbsp;&nbsp;3<br />2、生成Makefile文件&nbsp;&nbsp;&nbsp;&nbsp;3<br />3、修改Makefile文件&nbsp;&nbsp;&nbsp;&nbsp;3<br />4、编译&nbsp;&nbsp;&nbsp;&nbsp;3<br />第二步&nbsp;Boa的配置&nbsp;&nbsp;&nbsp;&nbsp;3<br />第三步boa的执行&nbsp;&nbsp;&nbsp;&nbsp;6<br />1.将根文系统ramdisk在电脑主机上mount&nbsp;–o&nbsp;loop&nbsp;ramdisk&nbsp;目标&nbsp;&nbsp;&nbsp;&nbsp;目录进行展开&nbsp;&nbsp;&nbsp;&nbsp;6<br />2.测试操作&nbsp;&nbsp;&nbsp;&nbsp;7<br />3.将ramdisk文件重新打包压缩下载到嵌入式板子上。&nbsp;&nbsp;&nbsp;&nbsp;7<br />4.CGI脚本测试&nbsp;&nbsp;&nbsp;&nbsp;7<br /><br /><br />简介<br />随着Internet技术的兴起,在嵌入式设备的管理与交互中,基于Web方式的应用成为目前的主流,这种程序结构也就是大家非常熟悉的B/S结构,即在&nbsp;嵌入式设备上运行一个支持脚本或CGI功能的Web服务器,能够生成动态页面,在用户端只需要通过Web浏览器就可以对嵌入式设备进行管理和监控,非常方&nbsp;便实用。本节主要介绍这种应用的开发和移植工作。<br />嵌入式Web服务器移植&nbsp;&nbsp;由于嵌入式设备资源一般都比较有限,并且也不需要能同时处理很多用户的请求,因此不会使用Linux下最常用的如Apache&nbsp;等服务器,而需要使用一些专门为嵌入式设备设计的Web服务器,这些Web服务器在存贮空间和运行时所占有的内存空间上都会非常适合于嵌入式应用场合。<br />&nbsp;&nbsp;&nbsp;典型的嵌入式Web服务器有Boa&nbsp;(http://www.boa.org/)和thttpd&nbsp;(http://www.acme.com/software/thttpd/)等,它们和Apache等高性能的Web服务器主要的区别在于它们一般是&nbsp;单进程服务器,只有在完成一个用户请求后才能响应另一个用户的请求,而无法并发响应,但这在嵌入式设备的应用场合里已经足够了。<br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Boa是一个非常小巧的Web服务器,可执行代码只有约60KB。它是一个单任务Web服务器,只能依次完成用户的请求,而不会fork出新的进程来处理&nbsp;并发连接请求。但Boa支持CGI,能够为CGI程序fork出一个进程来执行。Boa的设计目标是速度和安全,在其站点公布的性能测试中,Boa的性能要好于Apache服务器。<br />说明:下面的移植和操作是在at91rm9200芯片linux2.4.27上进行的。<br />第一步&nbsp;Boa程序的移植<br />&nbsp;这一步的工作在电脑主机上完成。<br />对于uClinux-dist而言,就是make&nbsp;menuconfig在应用程序选单中,在network&nbsp;application下面选择boa,然后到uClinux-dist目录下执行make编译即可。<br />对于有MMU的平台,如armlinux和ppclinux,都是到网上下载一个boa的tar包,随便找个主流的可用的版本即可,不要最新的也不要最旧的。下载到REDHAT宿主机上后,解压到任意目录,然后修改boa/src/Makefile里面的编译器,就只要加入两三行即可,如下几步操作:<br />1.下载Boa源码<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;下载地址:&nbsp;http://www.boa.org/,&nbsp;或者http://sourceforge.net/<br />&nbsp;&nbsp;&nbsp;&nbsp;最新发行版本:&nbsp;0.94.13<br />&nbsp;&nbsp;&nbsp;&nbsp;下载&nbsp;boa-0.94.13.tar.gz,注意:从boa上下载的是boa-0.94.13.tar.tar,解压方式一样<br />&nbsp;&nbsp;&nbsp;&nbsp;解压:#&nbsp;tar&nbsp;xzf&nbsp;boa-0.94.13.tar.gz<br /><br />2、生成Makefile文件<br />boa-0.94.13.tar.gz解压后生成boa-0.94.13目录,直接运行./电脑主机目录/boa-0.94.13/src/configure文件<br /><br /><br />3、修改Makefile文件<br />a、&nbsp;修改&nbsp;CC&nbsp;=&nbsp;gcc<br />为&nbsp;CC&nbsp;=&nbsp;/usr/local/arm/2.95.3/bin/arm-linux-gcc&nbsp;(根据自己电脑主机的编译器位置定)<br />&nbsp;&nbsp;&nbsp;b、&nbsp;修改&nbsp;CPP&nbsp;=&nbsp;gcc&nbsp;-&nbsp;E<br />为&nbsp;CPP&nbsp;=&nbsp;/usr/local/arm/2.95.3/bin/arm-linux-gcc&nbsp;-E<br />&nbsp;&nbsp;注意:boa-0.94.13用3.3.2和3.4.1编译器编译不通过。没有具体研究。<br /><br />4、编译<br />进入:/电脑主机目录/boa-0.94.13/src目录<br />&nbsp;&nbsp;&nbsp;&nbsp;#&nbsp;make<br />&nbsp;&nbsp;&nbsp;&nbsp;#&nbsp;/usr/local/arm/2.95.3/bin/arm-linux-strip&nbsp;boa&nbsp;&nbsp;(这是为了进行去除boa中生成的编译信息,让文件变小)<br />第二步&nbsp;Boa的配置<br />这一步的工作也在电脑主机上完成。<br />在电脑主机上的Boa源码boa-0.94.13目录下已有一个示例boa.conf,可以在其基础上进行修改。如下:<br />1、Group的修改<br />&nbsp;&nbsp;&nbsp;&nbsp;修改&nbsp;Group&nbsp;nogroup<br />为&nbsp;Group&nbsp;0<br />&nbsp;&nbsp;&nbsp;&nbsp;由于在/etc/group文件中没有nogroup组,所以设成0。另外在/etc/passwd中有nobody用户,所以User&nbsp;nobody不用修改。<br /><br />2、ScriptAlias的修改<br />&nbsp;&nbsp;&nbsp;&nbsp;修改&nbsp;ScriptAlias&nbsp;/cgi-bin/&nbsp;&nbsp;/usr/lib/cgi-bin/<br />为&nbsp;ScriptAlias&nbsp;/cgi-bin/&nbsp;&nbsp;/var/www/cgi-bin/<br /><br />3、ServerName的设置<br />修改&nbsp;ServerName&nbsp;www.your.org.here<br />为&nbsp;ServerName&nbsp;www.your.org.here<br />&nbsp;&nbsp;&nbsp;&nbsp;注意:该项默认为未打开,执行Boa会异常退出,提示“gethostbyname::No&nbsp;such&nbsp;file&nbsp;or&nbsp;directory”,所以必须打开。其它默认设置即可。<br />注释:boa.conf,下面解释一下该文件的含义:<br />#监听的端口号,缺省都是80,一般无需修改<br />Port&nbsp;80<br />#&nbsp;bind调用的IP地址,一般注释掉,表明绑定到INADDR_ANY,通配于服务器的所有IP地址<br />#Listen&nbsp;192.68.0.5<br />#作为哪个用户运行,即它拥有该用户的权限,一般都是nobody,需要/etc/passwd中有<br />#nobody用户<br />User&nbsp;nobody<br />#作为哪个用户组运行,即它拥有该用户组的权限,一般都是nogroup,需要在/etc/group文<br />#件中有nogroup组<br />Group&nbsp;nogroup<br />#当服务器发生问题时发送报警的email地址,目前未用,注释掉<br />#ServerAdmin&nbsp;root@localhost<br />#错误日志文件。如果没有以/开始,则表示从服务器的根路径开始。如果不需要错误日志,<br />则用#/dev/null。在下面设置时,注意一定要建立/var/log/boa目录<br />ErrorLog&nbsp;/var/log/boa/error_log<br />#访问日志文件。如果没有以/开始,则表示从服务器的根路径开始。如果不需要错误日志,<br />则用#/dev/null或直接注释掉。在下面设置时,注意一定要建立/var/log/boa目录<br />#AccessLog&nbsp;/var/log/boa/access_log<br />#是否使用本地时间。如果没有注释掉,则使用本地时间。注释掉则使用UTC时间<br />#UseLocaltime<br />#是否记录CGI运行信息,如果没有注释掉,则记录,注释掉则不记录<br />#VerboseCGILogs<br />#服务器名字<br />ServerName&nbsp;http://www.hyesco.com/<br />#是否启动虚拟主机功能,即设备可以有多个网络接口,每个接口都可以拥有一个虚拟的Web服<br />#务器。一般注释掉,即不需要启动<br />#VirtualHost<br />#非常重要,HTML文档的主目录。如果没有以/开始,则表示从服务器的根路径开始。<br />DocumentRoot&nbsp;/var/www<br />#如果收到一个用户请求的话,在用户主目录后再增加的目录名<br />UserDir&nbsp;public_html<br />#HTML目录索引的文件名,也是没有用户只指明访问目录时返回的文件名<br />DirectoryIndex&nbsp;index.html<br />#当HTML目录没有索引文件时,用户只指明访问目录时,boa会调用该程序生成索引文件然后<br />#返回给用户,因为该过程比较慢最好不执行,可以注释掉或者给每个HTML目录加上#DirectoryIndex指明的文件<br />#DirectoryMaker&nbsp;/usr/lib/boa/boa_indexer<br />#如果DirectoryIndex不存在,并且DirectoryMaker被注释,那么就用Boa自带的索引<br />#生成程序来生成目录的索引文件并输出到下面目录,该目录必须是Boa能读写<br />#&nbsp;DirectoryCache&nbsp;/var/spool/boa/dircache<br />#一个连接所允许的HTTP持续作用请求最大数目,注释或设为0都将关闭HTTP持续作用<br />KeepAliveMax&nbsp;1000<br />#HTTP持续作用中服务器在两次请求之间等待的时间数,以秒为单位,超时将关闭连接<br />KeepAliveTimeout&nbsp;10<br />#指明mime.types文件位置。如果没有以/开始,则表示从服务器的根路径开始。可以注释掉<br />#避免使用mime.types文件,此时需要用AddType在本文件里指明<br />MimeTypes&nbsp;/etc/mime.types<br />#文件扩展名没有或未知的话,使用的缺省MIME类型<br />DefaultType&nbsp;text/plain<br />#提供CGI程序的PATH环境变量值<br />CGIPath&nbsp;/bin:/usr/bin:/usr/local/bin<br />#将文件扩展名和MIME类型关联起来,和mime.types文件作用一样。如果用mime.types<br />#文件,则注释掉,如果不使用mime.types文件,则必须使用<br />#AddType&nbsp;application/x-httpd-cgi&nbsp;cgi<br />#指明文档重定向路径<br />#Redirect&nbsp;/bar&nbsp;http://elsewhere/feh/bar<br />#为路径加上别名<br />Alias&nbsp;/doc&nbsp;/usr/doc<br />#非常重要,指明CGI脚本的虚拟路径对应的实际路径。一般所有的CGI脚本都要放在实际路径<br />#里,用户访问执行时输入站点+虚拟路径+CGI脚本名<br />ScriptAlias&nbsp;/cgi-bin/&nbsp;/var/www/cgi-bin/<br />第三步boa的执行<br />这一步的工作也在电脑主机上完成。但是操作到的目录跟嵌入式板子上的目录有关。<br />1.将根文系统ramdisk在电脑主机上mount&nbsp;–o&nbsp;loop&nbsp;ramdisk&nbsp;目标&nbsp;&nbsp;&nbsp;&nbsp;目录进行展开<br />进入展开后ramdisk的根文件夹,然后执行下面:<br />#&nbsp;mkdir&nbsp;/var/log<br />#&nbsp;mkdir&nbsp;/var/log/boa(创建日志文件目录)<br />#&nbsp;mkdir&nbsp;/var/www&nbsp;(创建HTML文档的主目录)<br />#&nbsp;mkdir&nbsp;/etc/boa<br /><br />将第二步中修改过的boa.conf文件从电脑主机上别的目录拷贝到ramdsik的相关目录下<br />#&nbsp;cp&nbsp;/电脑主机目录/&nbsp;boa-0.94.13/boa.conf&nbsp;&nbsp;/etc/boa/<br />将mime.types文件复制到嵌入式板子ramdsik的/etc目录下,通常mime.types可以从linux电脑主机的&nbsp;/etc目录下直接复制即可。<br />#&nbsp;cp&nbsp;/&nbsp;etc/mime.type&nbsp;&nbsp;&nbsp;/etc/<br />上面第一步骤中编译生成的boa可执行文件,将其复制到展开后ramdsik的bin目录里面。<br />#&nbsp;cp&nbsp;/电脑主机目录/&nbsp;boa-0.94.13/src/boa&nbsp;&nbsp;&nbsp;/bin/<br />2.测试操作<br />&nbsp;&nbsp;可以将主机&nbsp;/usr/share/doc/HTML/目录下的index.html(这个是静态的html网页显示文件,是redhat9的欢迎主页,当在局域网的PC的IE上输入http://嵌入式板子的IP地址&nbsp;时会显示这个默认的静态网页)文件和img目录复制到嵌入式ramdsik的/var/www目录下<br />#&nbsp;cp&nbsp;/usr/share/doc/HTML/index.html&nbsp;&nbsp;/var/www/<br />#&nbsp;cp&nbsp;/usr/share/doc/HTML/img&nbsp;&nbsp;/var/www/<br />#&nbsp;cp&nbsp;/usr/share/doc/HTML/index-ZH_CH.html&nbsp;&nbsp;/var/www/<br />3.将ramdisk文件重新打包压缩下载到嵌入式板子上。<br />&nbsp;&nbsp;Ramdisk重新大包压缩下载到板子上后,在板子的超级终端的根目录下输入./boa,看boa是否可以执行。<br />&nbsp;可能提示错误如:libso.0.6&nbsp;&nbsp;GLIB2.2版本问题,这个一般是boa执行时需要动态共享库,解决的办法是,在编译boa的时候用静态的方式进行编译,即<br />进入:/电脑主机目录/boa-0.94.13/src目录下,修改makefile文件中:<br />LDFLAGS&nbsp;=&nbsp;-g<br />修改为:LDFLAGS&nbsp;=&nbsp;-static&nbsp;就可以了。<br /><br />执行boa时,可能出现其他的错误,则可以在开发板的/var/log/boa/error_log文件中找原因。大部分是跟boa.conf配置文件的正确与否有关。如:<br />问题:在执行./boa时出现错误:boa执行时,出错&nbsp;log.c:73&nbsp;unable&nbsp;to&nbsp;dup2&nbsp;the&nbsp;error&nbsp;log:bad&nbsp;file&nbsp;descriptor<br />解决方法:在log.c里注释掉那个if语句。<br />又出现了另个问题:unable&nbsp;to&nbsp;dup2&nbsp;the&nbsp;error&nbsp;log:&nbsp;Bad&nbsp;file&nbsp;descriptor<br />解决方法:在boa.conf里注释掉#AccessLog&nbsp;/var/log/boa/access_log<br />4.CGI脚本测试<br />书写下面的test.c测试程序,<br /><br />#include&nbsp;&ltstdio.h&gt<br />#include&nbsp;&ltstring.h&gt<br /><br />int&nbsp;main(void)<br />{<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;Contenttype:text/html

&quot;);&nbsp;&nbsp;&nbsp;&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lthtml&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lthead&gt&lttitle&gtAn&nbsp;HTML&nbsp;Page&nbsp;From&nbsp;a&nbsp;CGI&lt/title&gt&lt/head&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&ltbody&gt&ltbr&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lth2&gt&nbsp;This&nbsp;is&nbsp;an&nbsp;HTML&nbsp;page&nbsp;generated&nbsp;from&nbsp;with&nbsp;in&nbsp;a&nbsp;CGI&nbsp;program..&lt/h2&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lthr&gt&ltp&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lta&nbsp;href=&quot;output.html#two&quot;&gt&ltb&gt&nbsp;Go&nbsp;back&nbsp;to&nbsp;out&nbsp;put.html&nbsp;page&nbsp;&lt/b&gt&lt/a&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lt/body&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;printf(&quot;&lt/html&gt
&quot;);&nbsp;<br />&nbsp;&nbsp;&nbsp;&nbsp;fflush(stdout);&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<br />}<br />交叉编译生成test.cgi<br />#&nbsp;/usr/local/arm/2.95.3/bin/arm-linux-gcc&nbsp;-o&nbsp;test.cgi&nbsp;test.c<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;将test.cgi拷贝至嵌入式板子的/var/www/cgi-bin/下,当在局域网PC的IE上输入&nbsp;http://&nbsp;嵌入式板子的IP地址/cgi-bin/test.cgi<br />可以显示出test.cgi的网页。<br />&nbsp;&nbsp;有关CGI的知识可以学习:用C语言实现CGI相关资料。<br />注意:上面代码执行时,可能会出现错误提示:<br /><br />502&nbsp;Bad&nbsp;Gateway<br />The&nbsp;CGI&nbsp;was&nbsp;not&nbsp;CGI/1.1&nbsp;compliant.<br /><br />可能出现的原因:<br />1.测试下看是否能浏览静态网页,以保证网络是正确的。<br />&nbsp;&nbsp;&nbsp;在上面“第三步的&nbsp;2.测试操作”后局域网的PC的IE上输入http://嵌入式板子的IP地址,如果能显示redhat9的欢迎主页,说明静态访问是正确的了。<br />2.将.cgi文件拷贝至目标板上后,必须改变其权限&nbsp;<br />chmod&nbsp;755&nbsp;&nbsp;test.cgi<br />否则,上位机浏览时会提示上面错误<br />3.最有可能是CGI程序本身的问题。<br />我们已经看到test.c的打印开头有使用MIME头信息″Content&nbsp;type&nbsp;:text/html

″来表示输出HTML源代码给Web服务器。请注意任何MIME头信息后必须有一个空行,就是加了

,<br />所以这点必须保证。<br />4.Test.c的原文件是在文本模式下编辑的,在linux下不能执行<br />&nbsp;&nbsp;&nbsp;可以改动:dos2unix&nbsp;test.c&nbsp;改动后再进行编译<br />5.还有一种可能,test.cgi被编译成了动态形式,然而程序执行时在嵌入式中找不到动态库文件,所以在编译程序时加上-static,编译成静态的形式,但是这样编译出来的文件很大,在嵌入式中不适宜采用这种方式。<br />
 楼主| db10 发表于 2008-8-5 17:14 | 显示全部楼层

。。。

&nbsp;上面是我最近实现的,并写的资料。其中有不少可能碰到的问题和较好的解决方法,希望对正在进行web嵌入式开发的人员有直接指导意义。
发表于 2008-8-5 17:20 | 显示全部楼层

好.顶...

  
brandnew 发表于 2008-8-5 20:29 | 显示全部楼层

。。。

busybox里面不是有httpd吗?<br /><br />为何还要配boa?<br />
 楼主| db10 发表于 2008-8-5 23:04 | 显示全部楼层

brandnew

可以用啊,但是你要看看优势啊:<br /><br />Httpd是最简单的一个web服务器,功能也最弱,不支持认证、CGI(动态网页)。&nbsp;thttpd&nbsp;web服务器和boa&nbsp;web服务器都支持认证、CGI等,功能都比较全。boa是一个单任务的http服务器,源代码开放、性能高。它和传统服务器的主要区别是:它是单进程的,在接到新的httpd请求时并不&nbsp;FORK出一个新的进程来响应,不能对每个到来的连接处理,而且也不能将自己拷贝很多份来处理多连接。它在内部处理所有正在进行的http连接请求,只对单独的CGI程序、自动文件的产生、自动文件的解压等请求FORK出新进程。它的优点是代码简单、速度快、适合于嵌入式应用。
MCUJunior 发表于 2008-8-5 23:47 | 显示全部楼层

小声问一句

&nbsp;&nbsp;BoA可以用在ucos下吗?
后学 发表于 2008-8-6 08:09 | 显示全部楼层

不错!

我项目里面也有一个boa,一直没用心研究过,<br />如果把最终代码打包提供下载,很多人会喜欢。。。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

77

主题

2230

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部