@[TOC]
前提
环境我的是Ubuntu20.04.板子这里我用的是mx6ull,正点原子的alpha开发板,不过这是个纯软的东西,硬件平台不重要。然后交叉编译链我相信这些你都弄好了。
<font color=red>我初学这个,本文主要是我的个人记录,一些错误啥的难免,望各位大佬轻喷</font>
网上找到的fastcgi基本是php相关的,发现用c/c++文章还挺少,所以写一篇记录下过程。
lighttpd交叉编译安装
源码下载
下载源码没啥好说的,直接官网走起。我这里下载的是lighttpd-1.4.55.tar.gz这个版本。
除了下载lighttpd的源码,还需要pcre这个东西(我也初学者,不知道原因,大概是依赖这个吧)。pcre同样官网下一个。顺带一提 pcre和pcre2是两个东西。我开始下的时候没注意,折腾了半天。这里我用的pcre-8.42.tar.bz2这个版本。
交叉编译
首先编译pcre这个东西,先解压:
tar -xjvf pcre-8.42.tar.bz2
cd pcre-8.42
然后配置一下,arm-linux-gnueabihf-gcc这个是我的交叉编译链,然后最后设置安装目录为当前文件夹下install文件夹。
./configure CC=arm-linux-gnueabihf-gcc --host=arm-linux-gnueabihf target=arm-linux --prefix=/ --enable-utf8 --enable-unicode-properties --prefix=$(pwd)/install
配置完成后编译然后安装,因为配置时候制定了安装目录,所以这里直接make install就好了。
make
make install
印象中我第一次编译时候中间有报错来着,但是现在重做没报错,可能该装的依赖上次都装好了(写文章不能拖着呀,不然有些细节都忘了)。
接下来就编译lighttpd,同样先解压
tar -zxvf lighttpd-1.4.55.tar.gz
cd lighttpd-1.4.55/
lighttpd的配置就有一些长了,这个地方第一次弄的时候卡了好久,CC,RANLIB这些你用其他lighttpd版本时候可能还需要CXX啥的,反正缺啥加啥,是在不想看就把这些都配。然后我遇到另个坑是一直找不到pcre,最后问了googl,要手动指明库路径。然后下面就是我试过的可用版本。
./configure -prefix=$(pwd)/install \
--host=arm-linux-gnueabihf CC=arm-linux-gnueabihf-gcc \
RANLIB=arm-linux-gnueabihf-ranlib STRIP=arm-linux-gnueabihf-strip\
--enable-shared --without-mysql --without-zlib --without-bzip2 \
--disable-ipv6 \
PCRECONFIG=/home/shi/linux/web/new/pcre-8.42/install/bin/pcre-config \
PCRE_LIB=/home/shi/linux/web/new/pcre-8.42/install/lib/libpcre.a \
CFLAGS="$CFLAGS -DHAVE_PCRE_H=1 -DHAVE_LIBPCRE=1 -I/home/shi/linux/web/new/pcre-8.42/install/include"
然后进行编译和安装
make
make install
一切顺利的话,在当前目录下生成了install文件,并且里面有如下文件。
然后在install目录下载新建一些文件夹。cache、cgi-bin、config、log、sockets、upload、vhosts、webpages。
mkdir cache cgi-bin config log sockets upload vhosts webpages
接下来把需要的配置文件拷贝过来,路径如下,全部拷贝到刚刚新建的config文件夹
doc/config/conf.d
doc/config/lighttpd.conf
doc/config/modules.conf
接下来简单修改一下配置。修改的思路是这样的,lighttpd的默认配置他的配置文件,日志,网页文件等都是按着linux标准方式来的,这里我把这一些文件都放在了一个目录下面,所以要修改一下配置,告诉程序这里对应文件分别放在哪。第二点就是在嵌入式上资源受限,减少一些资源开销。
我的lighttpd.conf配置文件修改部分如下,完成的文件有点大,直接丢码云了(直接上传csdn又要积分啥的,不友好),地址见文末
## 我最终移植到板子上的时候把install文件夹名字改成了lighttpd,文件夹放
## 在根目录"/"下,所以下面几个目录对应如下
var.log_root = "/lighttpd/log"
var.server_root = "/lighttpd"
var.state_dir = "/lighttpd"
var.home_dir = "/lighttpd"
var.conf_dir = "/lighttpd/config"
## 这里server_root和"/vhost",两个字符串加起来正好"/lighttpd/vhost",正
## 好对应刚刚新建的vhosts
var.vhosts_dir = server_root + "/vhosts"
## 同vhosts
var.cache_dir = server_root + "/cache"
## 禁用ipv6,我们编译的时候禁用ipv6了。可以翻回去看编译配置
server.use-ipv6 = "disable"
## 看翻译这里是用一个不同的身份启动运行服务器,我这里注释了,直接root干就好
## server.username = "lighttpd"
## server.groupname = "lighttpd"
## 同vhosts,这个文件夹存放html文件
server.document-root = server_root + "/webpages"
## 同vhosts
server.pid-file = state_dir + "/lighttpd.pid"
## 访问日志调试时候先开着,最终部署时候看需求开关(据说生成的日志文件比
## 较大,如果部署开启这个考虑要定时删除)
include "conf.d/access_log.conf"
## 这里只是一个简单的服务器,用不着2048,我这里用256,
server.max-fds = 256
## 这个和max-fds 这个数值有关系,注意上面官方英语注释
server.max-connections = 128
## 上传文件夹路径
server.upload-dirs = ( "/lighttpd/upload" )
到现在为止,应该可以运行然后访问静态html文件了。
在webpages下面随便新建一个index.html文件,内容随便,名字先不随便,名字随便的话就要在地址栏指定名字了。
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>test</title>
</head>
<body>
<h1>我的第一个标题</h1>
<p>我的第一个段落。</p>
</body>
</html>
简单测试
接来下吧install整个文件夹复制到开发板的/
目录下面,然后重命名为lighttpd。
然后启动lighttpd试试看先。
cd /lighttpd/sbin
./lighttpd -f ../config/lighttpd.conf -m ../lib
如果你板子的80端口没有被占用,现在服务器应该是起来了
在电脑浏览器输入板子ip访问试试.显示如下,到此已经成功了一半。
fastcgi编译配置
源码下载
fastcgi的源码在github上面,下载速度可能有点感人,gitee好像有个什么搬运计划还是啥 ,反正大概就是把github上面一些项目搬运过来提速下载,我懒得去看这个在不在计划之列,我把我用的这个版本一并放在gitee,需要的话自取。地址在文末。
我这里用的是fcgi2-2.4.2.tar.gz版本。
交叉编译生成动态库
先解压,
tar -xzvf fcgi2-2.4.2.tar.gz
cd fcgi2-2.4.2
然后配置,这个配置我没找到完整参考,我猜是这样的
./autogen.sh
./configure CC=arm-linux-gnueabihf-gcc CXX=arm-linux-gnueabihf-g++ --host=arm-linux-gnueabihf --prefix=$(pwd)/install
然后编译安装
make
make install
执行完成后再install文件夹会生成三个文件夹,分别是bin,include,lib,暂且放一边,等会再用他们。
修改lighttpd配置
接下来修改一下lighttpd的配置,开启lighttpd对factcgi的支持。
首先在modules.conf找到如下行,并把include前面的#删除。
##
## FastCGI (mod_fastcgi)
##
include "conf.d/fastcgi.conf"
上面这行配置意思很简单,就是包含conf.d文件夹下的fastcgi.conf配置文件,所以接下来修改fastcgi.conf这个文件
在这个文件中添加如下行:
fastcgi.debug = 1
fastcgi.server = (
"/test" =>(
"test.fastcgi.handler" =>(
"socket" => socket_dir + "/test/test.fastcgi.socket",
"check-local" => "disable",
"bin-path" => home_dir + "/cgi-bin/test.fcgi",
"max-procs" => 3,
)
)
)
把fastcgi编译生成的install文件夹拷贝到板子里,然后修改你的.bashrc文件PATH路径加入install/bin,LD_LIBRARY_PATH路径加入install/lib,或者直接把install/bin目录下文件复制到板子/bin,把install/lib目录下文件全部复制到板子/etc。
简单测试
c语言fcgi程序
这块没啥好说的 直接上代码
#include <fcgi_stdio.h>
int main (void) {
while (FCGI_Accept() >= 0) {
printf("Status: 200 OK\r\n\r\nHello World!\n");
}
return 0;
}
编译的时候就需要用到刚刚生成的那个install文件夹下东西了。注意这个不能直接复制,{your path}要换成你自己的路径
arm-linux-gnueabihf-gcc -o test.fcgi test.c -L{your path}/install/lib \
-lfcgi -I{your path}/install/include
把刚刚生成的test.fcgi复制到板子/lighttpd/cgi-bin目录下。然后重启服务器,在浏览器输入ip/test,可以看到返回了hello world。
c++ fcgi程序
老规矩,先上代码
#include <string>
#include <iostream>
#include <string.h>
#include <fcgi_stdio.h>
#include <fcgiapp.h>
using namespace std;
int main(int argc, char **argv)
{
FCGX_Stream *in, *out, *err;
FCGX_ParamArray evnp;
while(FCGX_Accept(&in, &out, &err, &evnp) >= 0)
{
string query;
do
{
char *pRequestMoethod = FCGX_GetParam("REQUEST_METHOD", evnp);
if(pRequestMoethod == NULL)
{
cout << "ERROR: getevn REQUEST_METHOD is failed." << endl;
break;
}
//判断请求方法
if(strcmp(pRequestMoethod, "POST") == 0)
{
//获取请求中URL携带的参数,POST也会在URL中携带参数
query = FCGX_GetParam("QUERY_STRING", evnp);
//获取POST请求携带的参数长度
char *pLenstr = FCGX_GetParam("CONTENT_LENGTH", evnp);
if(pLenstr != NULL)
{
long len = atoi(pLenstr);
string postData;
//获取POST携带的数据
for(int i = 0; i < (int)len; i++)
{
int ch = FCGX_GetChar(in);
if(ch < 0)
{
break;
}
postData += (char)ch;
}
//处理请求,具体逻辑忽略
// processRequest(query, postData);
}
else
{
cout << "getenv failed." << endl;
}
}
else if(strcmp(pRequestMoethod, "GET") == 0)
{
//获取请求中URL携带的参数
query = FCGX_GetParam("QUERY_STRING", evnp);
//处理GET请求,代码隐藏
// processRequest(query);
break;
}
else
{
cout << "This is unknown quest." << endl;
}
}
while(false);
string reply = "hello world";
// reply = getReply(); //获取返回数据
//向客户端返回用到数据
FCGX_FPrintF(out, "Content-type: text/html;\r\n"
"Content-Length: %d\r\n"
"\r\n"
"%s",reply.length(), reply.c_str());
}
return 0;
}
编译类似,
arm-linux-gnueabihf-g++ -o test.fcgi test.cpp -L{your path}/install/lib \
-lfcgi -I{your path}/install/include
gitee仓库链接
https://gitee.com/sutichunxiao/lighttpd
参考
[1] https://blog.flowlore.com/passages/cc-fastcgi-http/
[2] https://redmine.lighttpd.net/projects/lighttpd/wiki/Docs_ModFastCGI
[3] https://www.huaweicloud.com/articles/2640eb7232b1794193263975ee0c65bc.html
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。