基于ELFboard的远程监测及人脸识别

一.项目简介

基于ELFboard开发板上的硬件资源,实现一个视频监控项目,同时完成对其中出现的人脸对比识别。

开发板如图:

二. 项目实现步骤

2.1 视频监控

这一步骤中需要实现两个程序:

  1. 在连接摄像头的ARM板子上,实现一个服务器程序:它一边读取摄像头数据,一边等待客户端连接并发送数据。可以用两个线程实现,一个负责采集图像信息;一个负责等待链接,并发送数据。

  2. 在手机或电脑上,编写客户端程序,它会从ARM板上获得数据并显示出来。同样,也可以用两个线程来实现。一个负责接受数据,一个负责显示数据。这2个程序之间,并不需要实现复杂的协议。

mjpg‐streamer是一个开源软件。MJPG streamer从Linux UVC兼容的网络摄像头、文件系统或其他输入插件获取JPG,并通过HTTP、RTSP、UDP等将其作为M-JPEG流式传输到WebBrowser、VLC和其他软件。

mjpg-streamer 需要很少的CPU和内存资源就可以工作,大部分编码工作都是摄像头完成的,所以对于内存和性能都有限的嵌入式系统十分适用。

将mjpg-streamer移植并运行在ARM板上,在同一局域网内的设备输入正确的ip地址即可直接观看到视频画面。对ARM板的性能要求不高,主频200MHz的ARM芯片也能实现。

下载mjpg-streamer:

git clone https://github.com/shrkey/mjpg-streamer

后续交叉编译移植不再赘述。

启动mjpg-streamer后,输入ip地址以及端口号即可看到摄像头内容如下图:

同时后续人脸识别功能中需要能够从视屏流中提取出照片,需要修改mjpg-streamer源码,使其支持拍照功能。具体修改如下:

修改完成之后只要向有名管道/tmp/webcom写入相应的字符串就能实现拍照功能。

# cd mjpg-streamer-rc63/plugins/output_file

# vim output_file.c

//在96行 函数 void*worker_thread(void *arg) 体中加入以下代码:

charbuf[10];   //
intflags = 0;   //
intfd_com = 0; //打开管道
stop_num = 0; //拍照计数

if ( access(“/tmp/webcom”,F_OK) < 0 )    //创建有名管道用于接收拍照命令
{
    if ( mkfifo(“/tmp/webcom”,0666 ) < 0)
    {
        Printf(“ photo fifo create failed\n”);
    }
}
  fd_com = open (“/tmp/webcom”,O_RDONLY,0666);
  if (fd < 0)
{
    perror (“open the file webcom error”);
}

//在while( ok >= 0 && !pglobal->stop){ 后加入

if (flags == 0)
{
    while(1)
    {
        reade(fd_com,buf,sizeof(buf));
        if(strncmp(buf,”danger”,6) == 0)    //拍11张照片
        {
            flags = 1;   
            bzero(buf,sizeof(buf));
            break;
        }
    if(strncmp(buf,”one”,3) == 0)   //拍1张照片
    {
        flags = 2;
        bzero(buf,sizeof(buf));
        break;
    }
}
}

 
//在if (delay > 0){

   usleep(1000*delay);

}后加入

stop_num++
if(flags == 1)        //判断拍照的数量
{
    if ( stop_num > 9)
   {
        stop_num= 0;
        flsgs= 0;    
    }
}
elseif (flags == 2)
{
    stop_num= 0;
    flags= 0;
}

视频监控展示

2.2人脸检测

'haarcascade_frontalface_default.xml'是opencv中已经训练好的人脸分类器文件。它是基于Haar特征的级联分类器,可以用于检测正脸的人脸。该文件是通过大量的正负样本训练而成,可以用于人脸检测的应用中。具体调用代码如下:

#! user/bin/python
#- * -coding:UTF-8 - * -

import cv2
import numpy as np


def myfilter(img):
    # 图像转化为灰度格式
    gray=cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)

    # 导入人脸级联分类器引擎,'xml'文件包含训练好的人脸特征
          
    face_cascade=cv2.CascadeClassifier('  \
    /home/xuyang/test1/haarcascade_frontalface_default.xml')
    #为防止报错使用该文件在opencv下的绝对路径
    # 用人脸级联分类器引擎进行人脸识别,返回的faces为人脸坐标列表
    faces = face_cascade.detectMultiScale(gray)
    
    return faces

def myfaces_count(img,faces):    
    count = 0 #人脸计数初值
    # 对每张脸,操作如下
    for (x,y,w,h) in faces:
        '''画矩形圈出人脸
        输入参数依次为:图片,右上角的点坐标,矩形大小,线条颜色,宽度
        '''
        cv2.rectangle(img,(x,y),(x+w,y+h),(255,255,255),2)
        count += 1 # 累计人数
        # 把统计人数显示出来
        cv2.putText(img,'{}'.format(count),(x,y-7),3,1.2,(0,0,255),2)

    return img

#打开mjpg-streamer视频流(通过URL)
#cap = cv2.VideoCapture('http://192.168.106.128:8080/?action=stream')
#打开视频
cap = cv2.VideoCapture('video.mp4')
y = 0
#获取视频相关数据以便于保存视频
width = int(cap.get(3))
height = int(cap.get(4))
fps = cap.get(5)
fourcc = cv2.VideoWriter_fourcc(*'MJPG')
output_file  = 'output_video.mp4'
video_writer = cv2.VideoWriter(output_file,fourcc,fps,(width,height),isColor = True)
while True:
    # 读取每一帧图像
    ret, frame = cap.read()
    
    if not ret:
        break
    if y == 0:
        faces = myfilter(frame)  #人脸识别特征每10次循环做一次 不然运行速度太慢了
    y = y + 1
    if y == 10:
        y =0
    frame = myfaces_count(frame,faces)
    # 在窗口上显示当前帧的图像
    cv2.imshow("Frame", frame)
    video_writer.write(frame) #保存视频
    # 按下 'q' 退出循环
    if cv2.waitKey(1) & 0xFF == ord('q'):
        break
 
# 关闭所有窗口及释放对象
cap.release()
video_writer.release()
cv2.destroyAllWindows()

以下展示一段开发板处理的视频效果:

人脸识别视频

可以看到,大部分时候该模型都能够准确识别到人脸的位置。

2.3 人脸识别

鉴于开发板运行人脸检测模型已经有一定的运算压力,同时为了丰富项目内容,人脸识别部分我们通过传送照片在云端完成。

本文通过libcurl库调用云端API实现人脸识别。需要libcurl库支持https协议。要让libcurl库支持https协议实现人脸识别,就需要安装移植openssl这个库。此篇人脸识别介绍主要目的是判断两张人脸图片的相似程度或者接近程度。安装移植libcurl库和openssl库不多赘述。

首先是注册一个OCR云识别平台账号如图:

查询对应平台的API和接口地址:

下面是调用人脸识别API的代码:

#include <stdio.h>
#include <curl/curl.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <time.h> 

typedef unsigned int bool;
#define true 1            
#define false 0
 
 
char buf[1024]={'\0'};


size_t readData(void *ptr, size_t size, size_t nmemb, void *stream)
{
    strncpy(buf,ptr,1024);
}
 
char* getPic(char *pic)
{
    char cmd[128]={'\0'};
    memset(cmd,'\0',128);
    sprintf(cmd,"base64 %s > tmpFile",pic);
    system(cmd);
    int fd = open("./tmpFile",O_RDWR);
    int filelen = lseek(fd,0,SEEK_END);
    lseek(fd,0,SEEK_SET);
    char *base64Buf = (char*)malloc(filelen + 8);
    memset(base64Buf,'\0',filelen + 8);
    read(fd,base64Buf,filelen+8);
    close(fd);
    system("rm -f tmpFile");
    return base64Buf;
}
 
 
 
bool postUrl()//POST请求
{
    char buf1[1024] = {0},buf2[1024] = {0};
    unsigned long long counter = 0;
 
    static char *folder = "/tmp";
    time_t t;
    struct tm *now;
    t = time(NULL);
    now = localtime(&t);
  
    system("echo one > /tmp/webcom"); //向有名管道webcom写入字符串实现拍照
    strftime(buf1, sizeof(buf1), "%%s/%Y_%m_%d_%H_%M_%S_picture_%%09llu.jpg", now);
    snprintf(buf2, sizeof(buf2), buf1,"/tmp", counter);
    sleep(1);
    CURL *curl;
    CURLcode res;
    char *postString = NULL;
    char *base64Buf1 = getPic(buf2);
    char *base64Buf2 = getPic("./me5.jpg");
 
    char *key = "xxxxxxxxxxxxxxxxxxxxxxx";
    char *secret = "xxxxxxxxxxxxxxxxxxxxxxx";
    int typeld = 21;
    char *format = "xml";
    int len = strlen(key)+strlen(secret)+strlen(format)+ \
    strlen(base64Buf1)+strlen(base64Buf2)+128;
    printf("%d",len);
    postString = (char *)malloc(len);
                
    sprintf(postString,"img1=%s&img2=%s&key=%s&secret=%s&typeId=%d&format=%s",\
    base64Buf1,base64Buf2,key,secret,typeld,format);
    curl = curl_easy_init();
    if (curl)
    {
        curl_easy_setopt(curl, CURLOPT_COOKIEFILE, "/tmp/cookie.txt"); // 指定cookie文件
        curl_easy_setopt(curl, CURLOPT_POSTFIELDS, postString);    
        // 指定post内容
        curl_easy_setopt(curl, CURLOPT_URL, "https://netocr.com/api/faceliu.do"); 
        //指定url
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, readData);
        res = curl_easy_perform(curl);//执行
        //printf("res = %d\n",res);
        if(strstr(buf,"是")!=NULL)
        {
            printf("is same people\n");
        }
        else
        {
            printf("not the same people\n");
        }
            curl_easy_cleanup(curl);
    }
    return true;
}

int main(void)
{
    postUrl();
}

本项目的逻辑是首先视频监控,通过检测是否有人脸来决定是否拍照进行人脸识别。上述视频中检测到了人脸并在合适的时机拍照,然后通过调用云端API进行人脸识别对比并返回结果,上述视频的对比结果如下:

左图为提前准备好的相关人脸的照片,右图为拍照得到的照片,下面为对比结果判定为是同一个人。同时开发板返回结果也正确:

  • 21
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值