20155110王一帆 《远程安防监控系统》课程设计个人报告

一、个人所做的工作

  1. 编译linux内核,制作文件系统镜像
  2. 交叉编译项目代码
  3. 配合组长完成SD卡的烧写
  4. 调试运行开发板的各项功能

二、遇到的问题的及解决方法

问题1:fastboot的驱动安装

首先,我们在设备管理器中找到Android设备

安防监视系统架构 安防监控系统方案设计_数据库

但是安装fastboot驱动时,却失败了。

安防监视系统架构 安防监控系统方案设计_数据库_02

问题1的解决方法

在windows设置中找到“更新与安全”

安防监视系统架构 安防监控系统方案设计_安防监视系统架构_03

找到恢复,在高级启动中找到“立即重启”

安防监视系统架构 安防监控系统方案设计_c/c++_04

重启之后,选择“疑难解答”

安防监视系统架构 安防监控系统方案设计_嵌入式_05

接着选择“高级选项”

安防监视系统架构 安防监控系统方案设计_c/c++_06

然后选择“启动设置”

安防监视系统架构 安防监控系统方案设计_嵌入式_07

重启以后,按下数字键“7”禁用驱动的强制签名

安防监视系统架构 安防监控系统方案设计_数据库_08

之后,我们就可以成功安装驱动了。

安防监视系统架构 安防监控系统方案设计_安防监视系统架构_09

安防监视系统架构 安防监控系统方案设计_c/c++_10

我们这时再观察设备管理器,我们就可以发现Android设备从未知变成已知了。

安防监视系统架构 安防监控系统方案设计_嵌入式_11

问题2:open /dev/ttyUSB0 error: No such file or directory

在进行串口通讯的时候,打开开发板后,一直会提示这样的消息。

安防监视系统架构 安防监控系统方案设计_c/c++_12

原因是/dev目录下没有ttyUSB0这个设备文件。

安防监视系统架构 安防监控系统方案设计_安防监视系统架构_13

问题2的解决方法

事实上,我们依旧可以输入linux命令的,要想不看到这个信息,需要在/dev下建立一个软链接。

输入ln -s /dev/null /dev/ttyUSB0,用“空设备”做个软链接。

安防监视系统架构 安防监控系统方案设计_c/c++_14

这样就不会一直出现open /dev/ttyUSB0 error: No such file or directory这样的提示了

而且这样做不会影响正常的操作。

三、剖析CGI源码时的疑问与解答

我自己有过Web编程的经验,也了解早期的Web编程是靠CGI来完成的,用的语言也是五花八门——C/C++、perl、bash……

我选择研究该项目的CGI源码,也是因为自己对Web编程有一些了解,而对系统编程就完全不懂,也搞不清什么线程、锁、同步的概念。

(一)登录表单处理——login.cgi

我们进入文件系统rootfs的www目录下,这里就是存放web服务器html,css,js和cgi程序的地方。我们先看看index.html.

我们直接看登录的表单部分

<form name="form1" method="post"  action="cgi-bin/login.cgi">
  <table width="100%" border="0" cellspacing="9" cellpadding="0">
  <tbody>
  <tr>
    <td width="92">用户帐号:</td>
    <td width="130"><label>
    <input name="username" type="text" id="username" value="user"></label></td>
  </tr>
  <tr>
    <td>登录密码:</td>
    <td>
    <label>
      <input name="password" type="password" id="password" value="123456">
    </label>
    </td>
  </tr>
  <tr>
    <td height="25"></td>
    <td><input type="image" name="submit" style="width:97px;height:25px;" src="images/login/go.gif"></td>
  </tr>
  </tbody> 
  </table>
</form>
<form name="form1" method="post"  action="cgi-bin/login.cgi">
  <table width="100%" border="0" cellspacing="9" cellpadding="0">
  <tbody>
  <tr>
    <td width="92">用户帐号:</td>
    <td width="130"><label>
    <input name="username" type="text" id="username" value="user"></label></td>
  </tr>
  <tr>
    <td>登录密码:</td>
    <td>
    <label>
      <input name="password" type="password" id="password" value="123456">
    </label>
    </td>
  </tr>
  <tr>
    <td height="25"></td>
    <td><input type="image" name="submit" style="width:97px;height:25px;" src="images/login/go.gif"></td>
  </tr>
  </tbody> 
  </table>
</form>

这个表单的数据会提交给login.cgi这个程序去处理。学过java servlet的同学会发现,CGI程序和servlet比较接近。

我们来分析一下login.cgi的源码。

//login.c

cgiFormStringNoNewlines("username", name, N);
cgiFormStringNoNewlines("password", pw, N);

这两条语句将username和password分别放到char数组name和pw中,接下来肯定是要到数据库里面去查询了。这里的数据库用的是sqlite,非常轻量级的一个数据库。

//login.c

if(sqlite3_open("/user.db", &db) != SQLITE_OK)
{
    fprintf(cgiOut, "<BODY>");
    fprintf(cgiOut, "<H1>%s</H1>", "Server is busy...");        
    fprintf(cgiOut, "<meta http-equiv=\"refresh\" content=\"1;url=../index.html\">");
    return -1;
}

sprintf(sql, "select * from usr where name='%s' and password='%s'", name, pw);

if(sqlite3_get_table(db, sql, &result, &row, &column, NULL) != SQLITE_OK)
{
    fprintf(cgiOut, "<BODY>");
    fprintf(cgiOut, "<H1>%s</H1>", "Server is busy...");        
    fprintf(cgiOut, "<meta http-equiv=\"refresh\" content=\"1;url=../index.html\">");
    sqlite3_close(db);
    return -1;
}

非常容易理解,sqlite3_open()函数用来打开数据user.db,如果打开失败就会看到网页上有Server is busy...字样。

char数组sql中存放了一条select语句,sqlite3_get_table()函数就是用来查询的,如果数据库出现问题,依然在网页上显示有Server is busy...

如果在数据库里找不到对应的用户,变量row的值就是0,网页上显示Name or password error

//login.c

if(row == 0)
{
    fprintf(cgiOut, "<BODY>");
    fprintf(cgiOut, "<H1>%s</H1>", "Name or password error");       
    fprintf(cgiOut, "<meta http-equiv=\"refresh\" content=\"1;url=../index.html\">");
    sqlite3_close(db);
    return 0;
}

查看user.db数据库,只有一条记录

安防监视系统架构 安防监控系统方案设计_c/c++_15

代码里的cgiOut是什么呢?,我们在cgic.h头文件的实现文件cgic.h中找到了答案。

//cgic.c

    cgiIn = stdin;
    cgiOut = stdout;

cgiOut就是标准输出。登录成功后,会跳转到主页面main.html。

(二)环境信息获取——env_1_a9_info.cgi

main.html页面是一个frameset框架,由left.html,top.html,right.html三个页面构成。

其中只要left.html导航栏具有实际的功能选项。

安防监视系统架构 安防监控系统方案设计_c/c++_16

我们先看看环境信息env1.html页面中的cgi程序

<iframe  frameborder="0"  border=0 scrolling="no" src="cgi-bin/env_1_a9_info.cgi" width="100%" height="100%"></iframe>

这个iframe告诉我们env_1_a9_info.cgi程序会处理当前的实时数据,并返回结果。

分析一下env_1_a9_info.cgi的源码,来看看这些数据是怎么得到的。

get_env()函数是用来获取环境信息的。

//env_1_a9_info.c

void get_env()
{
    sqlite3 *db;
    char sql1[N] = {0}, sql2[N] = {0};
    char **result1, **result2;
    int row1, colunm1, row2, colunm2;
    if  (sqlite3_open("/smartfarm.db", &db) != SQLITE_OK) {
        fprintf(cgiOut, "<H2>sqlit smartfarm.db open err</H2>");    
        errflag = 1;
        return ;
    }
    sprintf(sql1, "select * from env where farm_no=1");
    if (sqlite3_get_table(db, sql1, &result1, &row1, &colunm1, NULL) != SQLITE_OK) {
        fprintf(cgiOut, "<H2>sqlite3_get_table  err</H2>"); 
        errflag = 1;
        return ;
    }
    strncpy(temp_max, result1[colunm1 + 1], 9);
    strncpy(temp_min, result1[colunm1 + 2], 9);
    strncpy(hum_max, result1[colunm1 + 3], 9);
    strncpy(hum_min, result1[colunm1 + 4], 9);
    strncpy(light_max, result1[colunm1 +5], 9);
    strncpy(light_min, result1[colunm1 + 6], 9);

    .......

    sqlite3_close(db);
}

不难发现,这些数据都是从数据库smartfarm.db中获取的。

安防监视系统架构 安防监控系统方案设计_数据库_17

最高/最低气温,湿度,光照,都在env表中。

还有一张collect_env表,显示的是实时信息,但是好像没有用到。

安防监视系统架构 安防监控系统方案设计_#define_18

//env_1_a9_info.c

void get_env()
{
    ........

    sprintf(sql2, "select * from collect_env where farm_no=-1");
    if (sqlite3_get_table(db, sql2, &result2, &row2, &colunm2, NULL) != SQLITE_OK) {
        errflag = 1;
        return ;
    }
#if 0
    strncpy(temp, result2[colunm2 + 1], 9);
    strncpy(hum, result2[colunm2 + 2], 9);
    strncpy(light, result2[colunm2 + 3], 9);
#endif

}

很明显,temp,hum,light分别存放当前的温度、湿度和光照,但是被注释掉了

(三)实时监控的图像拍照处理——take_photo.cgi

实现监控的CGI程序是take_photo.cgi,我们还是直接分析它的源代码。

抛开cgi程序的fprintf不看,我们直接看核心处理的部分

//take_photo.c

    key_t key; 
    int msgid; 
    char buf[2] = {0};

    if((key = ftok("/lib", 'a')) < 0)
    {
        perror("ftok");
        exit(1);
    }
    if ((msgid = msgget(key, IPC_CREAT | IPC_EXCL | 0666)) < 0)
    {
        if (errno == EEXIST) {
            msgid = msgget(key, 0666);
        }else{
            perror("msgget msgid");
            return 0;
        }
    }

    struct message msgbuf;
    msgbuf.type = 1L;
    msgbuf.msg_type = 2L;

    cgiFormString("mode", buf, 2);
    if(buf[0] <= '0' || buf[0] > '9')
        goto err;
    
    msgbuf.text.camera = buf[0] - '0';
    msgsnd (msgid, &msgbuf, sizeof (msgbuf) - sizeof (long), 0);

这是一段linux的进程间通信,我们需要关注以下三个系统调用:

  • key_t ftok(const char *pathname, int proj_id)
  • int msgget(key_t key, int msgflg)
  • int msgsnd(int msgid, const void *msg_ptr, size_t msg_sz, int msgflg)

这些调用的细节不去管它,只要知道该进程需要收发其他进程的消息就可以了。

我们关注这段代码,看看进程发送的消息msgbuf是什么。

//take_photo.c

    struct message msgbuf;
    msgbuf.type = 1L;
    msgbuf.msg_type = 2L;

    cgiFormString("mode", buf, 2);
    if(buf[0] <= '0' || buf[0] > '9')
        goto err;
    
    msgbuf.text.camera = buf[0] - '0';

注意到这里的cgiFormStirng函数,我们的buf存的是表单中收集到的数字,下面是表单的内容。

<form id="take_photo" name="take_photo" method="post" action="cgi-bin/take_photo.cgi">
  <td width="15%"><input type="radio" name="mode" id="mode_1" value="1" />1</td>
  <td width="15%"><input type="radio" name="mode" id="mode_2" value="3" />3</td>
  <td width="15%"><input type="radio" name="mode" id="mode_3" value="5" />5</td>
  <td width="15%"><input type="radio" name="mode" id="mode_4" value="7" />7</td>
  <td width="15%"><input type="radio" name="mode" id="mode_5" value="9" />9</td>
  <td width="25%"><input type="submit" name="take_photo" id="take_photo" value="图像抓拍" /></td>
</form>
<form id="take_photo" name="take_photo" method="post" action="cgi-bin/take_photo.cgi">
  <td width="15%"><input type="radio" name="mode" id="mode_1" value="1" />1</td>
  <td width="15%"><input type="radio" name="mode" id="mode_2" value="3" />3</td>
  <td width="15%"><input type="radio" name="mode" id="mode_3" value="5" />5</td>
  <td width="15%"><input type="radio" name="mode" id="mode_4" value="7" />7</td>
  <td width="15%"><input type="radio" name="mode" id="mode_5" value="9" />9</td>
  <td width="25%"><input type="submit" name="take_photo" id="take_photo" value="图像抓拍" /></td>
</form>

结构体message在struct.h头文件中有定义

//struct.h

union text {
    unsigned char led;
    unsigned char buzzer;
    unsigned char camera;
    unsigned char fan;
    unsigned char relay;
    unsigned char uart;
    char phone[24];
    struct control_parameter parameter;
};

struct message {
    long type;
    long msg_type;
    union text text;
};

那么谁会接收msgbuf这个消息呢?我们可以在项目源码中找到答案。

看一下,项目源码里面pthread_client_request.c中的一个代码片段

//pthread_client_request.c

    switch (msgbuf.msg_type) {

        ........

        case CAMERA:
            printf("received camera request\n");
            pthread_mutex_lock (&mutex_camera);
            dev_camera_mask = msgbuf.text.camera;
            pthread_cond_signal (&cond_camera);
            pthread_mutex_unlock (&mutex_camera);
            break;
        
        ........
        
        default:    
#if DEBUG
            printf("pthread client request default break\n");
#endif
            break;
    
    }

也就是说会 有专门的线程用来处理摄像头有关的消息

我们还记得之前take_photo.c有这样一段代码

struct message msgbuf;
    msgbuf.type = 1L;
    msgbuf.msg_type = 2L;

1L、2L这样的魔数是什么意思呢

答案在项目源码的global_variable.h头文件中

//global_variable.h

/*message queue type */
#define REQUEST 1L        //msgbuf.type=1L

/*message queue msg_type*/
#define BUZZER 1L
#define CAMERA 2L       //msgbuf.msg_type = 2L
#define FAN 3L
#define LED 4L
#define RELAY 5L
#define SET_PARAMETER 6L
#define SMS 7L
//#define SQLITE 7L
#define UART 8L
#define MODIFY_PHONE_NUM 9L
#define BEEP 10L

linux下一切都是文件,打开摄像头也不例外

//pthread_camera.c

    if ((dev_camera_fd = open (DEV_CAMERA, O_RDWR | O_NOCTTY | O_NDELAY)) < 0)
    {
        printf ("Cann't open file /tmp/webcam\n");
        exit (-1);
    }

其中DEV_CAMERA这个常量的定义在头文件global_variable.h中也能找到

//global_variable.h

#define DEV_GPRS            "/dev/ttyUSB1"
#define DEV_GPRS_2          "/dev/ttyUSB2"
#define DEV_ZIGBEE          "/dev/ttyUSB0"
#define DEV_LED             "/dev/led"
#define DEV_BUZZER          "/dev/buzzer"
#define DEV_CAMERA          "/tmp/webcam"
#define SQLITE_OPEN         "./smartfarm.db"

这个常量就是/tmp/webcam,这应该指的就是摄像头设备

(四)照片展示——show_photo.cgi

在“历史照片”这一功能中,调用的CGI程序是show_photo.cgi

我们同样还是忽略掉CGI源码中的fprintf部分。

#define DIRNAME     "../pice"
#define PHOTO_NUM_MAX   100 // 最多100张,0-99

int cgiMain()
{
    .........

    if((dir = opendir(DIRNAME)) == NULL)    //打开图片存放的目录
    {
        perror("fail to opendir");
        exit(1);
    }

    while((dirp = readdir(dir)) != NULL)  //读文件夹里文件的名字
    {
        if(dirp->d_name[0] > '9' || dirp->d_name[0] < '0')   //名字的第一个字母
            continue;
        sprintf(photoname[syncmsg1.photo_num++], "%s", dirp->d_name);
        if(syncmsg1.photo_num >= PHOTO_NUM_MAX)         //判断是否超过最大值
        {
            syncmsg1.photo_num = PHOTO_NUM_MAX - 1;
            break;
        }
    }

    .........

}

我们从中可以知道,历史照片都是保存在/www/pice目录下的,而且最多只保存100张。

四、调试过程中无法解决的问题

难题1:WIFI模块无法使用

原本这个系统是通过WIFI来访问并进行控制的,但是我们的WIFI模块出现了问题,现在只能用网线直连的方式控制系统。

我们将手机热点配置为my_accent,密码设为012345678,在rootfs/etc中添加配置文件wpa-spk-tkip.conf

# WPA-PSK/TKIP

ctrl_interface=/var/run/wpa_supplicant

network={
    ssid="my_accent"
    key_mgmt=WPA-PSK
    proto=WPA
    pairwise=CCMP
    group=CCMP
    psk="012345678"
}

更新文件系统以后重新烧写镜像,配置IP和网关后,执行命令wpa_supplicant -B -i wlan0 -c /etc/wpa-psk-tkip.conf

安防监视系统架构 安防监控系统方案设计_c/c++_19

我们可以看到,wlan网卡依然没有连接到手机热点上。我们暂时没有解决这个问题。

难题2:视频模块无法使用

我们无论直接使用含有项目内容的文件系统,还是自己重新编译mjpeg-streamer都出现了同样的问题。

我们启动开发板后,对mjpeg-streamer进行测试。

运行mjpg_streamer -i "/mjpg/input_uvc.so -y -d /dev/video0" -o "/mjpg/output_http.so -w 192.168.9.111:8080"

安防监视系统架构 安防监控系统方案设计_c/c++_20

系统显示由于设备被占用了,初始化失败。我们没有找到解决的方法。

这些问题目前依然没有找到解决方案

五、心得体会

我们小组成员之前都没有嵌入式开发的经验,这次课程设计——“远程安防监控系统”对我们来说是一次挑战吧。虽然已经有现成的源码和成熟的文档,我们真正操作的时候还是遇到了许多问题,才真正体会到什么是“纸上得来终觉浅”。

我们在这次课程设计中收获很多,我们惊讶地发现,之前的linux操作经验对我们进行嵌入式开发有很大的帮助,串口通讯之后,文件的操作与查看,网络状态的查看,进程的查看与kill,都如出一辙,不得不感叹linux的伟大,在linux知识上的投资收益真棒。

C语言的可移植性还是不错的,很多源码进行较差编译的时候,只需要把原来的编译器"CC=gcc"改为"CC=arm-none-linux-gnueabi-gcc"就可以,有时甚至不用改变C语言代码就能正常make。

嵌入式开发还是很有意思的,我们通过这次课程设计慢慢摸到了点门道。

六、参考资料

  1. Win10怎么禁用驱动程序强制签名
  2. /dev/tty /dev/ttyS0 /dev/tty0区别
  3. 《远程安防监控系统项目文档》华清远见教育集团研发中心