【嵌入式Linux项目】基于Linux的全志H616开发板智能家居项目(语音控制、人脸识别、安卓APP和PC端QT客户端远程操控)有视频功能展示

目录

一、功能需求

二、开发环境

1、硬件:

2、软件:

3、引脚分配:

三、关键点

1、设计模式之工厂模式

2、线程

3、wiringPi库下的相关硬件操作函数调用

4、语音模块的串口通信

5、摄像头的实时监控和拍照功能

6、人脸识别

7、qt程序跨平台运行(编译成安卓APP)

四、编译和运行

五、视频功能展示


一、功能需求

  • 火焰传感器检测到火焰时,蜂鸣器报警,直到无火焰时停止报警
  • 语音控制多个LED灯的开和关(二楼灯、餐厅灯、客厅灯、卫生间灯)
  • 语音开启摄像头,并在对应IP地址的网页上实时监控画面
  • 语音控制摄像头拍摄照片,存在当前文件夹下(通过filezilla将照片传到PC上查看)
  • 语言开启人脸识别功能,将拍摄照片与本人照片对比,识别成功蜂鸣器滴一声,失败滴四声
  • 通过socket网络,实现开发板跑服务端,安卓手机跑客户端APP或PC上位机跑客户端,实现手机或上位机远程发送指令完成以上功能,并实时将温湿度传感器所测数据在安卓APP或PC上位机的QT界面上显示

二、开发环境

1、硬件:

Orangepi Zero2 全志H616开发板,语音模块SU-03T,摄像头模组OV9726,蜂鸣器,火焰传感器,4个LED等,4路继电器组,6v电源,若干杜邦线。

2、软件:

MobaXterm、VS Code、FileZilla

3、引脚分配:

在MobaXterm命令控制终端输入gpio readall可以查看开发板上的所有引脚。语音模块、蜂鸣器、火焰传感器和4路继电器组的引脚接线在下图框出。

由于加了温湿度传感器之后板子上引脚不够,所以在第二个视频演示时会把语言识别模块替换下来。温湿度传感器对应IO为,VCC--5V, GND--GND, DAT--3wPi。

三、关键点

1、设计模式之工厂模式

工厂模式提供了一种将对象的实例化过程封装在工厂类中的方式。本文通过使用工厂模式,将对象的创建与使用代码分离,提供一种统一的接口来创建不同类型的对象。在工厂模式中,我们在创建对象时不会对客户端暴露创建逻辑,并且是通过使用一个共同的接口来指向新创建的对象(链表)。

    //1.指令工厂初始化
    pCommandHead = addVoiceContrlToInputCommandLink(pCommandHead);//串口
    pCommandHead = addSocketContrlToInputCommandLink(pCommandHead);
    
    //2.设备控制工厂初始化
    //四个LED+火灾+蜂鸣器+摄像头
    pdeviceHead = addBathroomLightToDeviceLink(pdeviceHead);
    pdeviceHead = addUpstairLightToDeviceLink(pdeviceHead);
    pdeviceHead = addRestaurantLightToDeviceLink(pdeviceHead);
    pdeviceHead = addLivingroomLightToDeviceLink(pdeviceHead);
    pdeviceHead = addFireToDeviceLink(pdeviceHead);
    pdeviceHead = addBeepToDeviceLink(pdeviceHead);
    pdeviceHead = addCameraToDeviceLink(pdeviceHead);

2、线程

主函数中创建了三个线程:语音线程、火焰检测线程、网络线程。

语音线程(voiceThread):完成串口的配置和初始化,在while循环里每隔0.3s检查串口是否有语音命令词到来,有则执行对应操作。

火焰检查线程(socketThread):while循环里每隔0.5s检测是否有火焰,有则蜂鸣器发出警报,知道无火焰。

网络线程(socketThread):接收客户端指令,实现上位机远程发送指令完成 [ 功能要求 ] 所述功能;给客户端发送温湿度传感器所测数据,实时显示在QT界面上。

//1、接收客户端的指令控制灯和摄像头 执行指令功能与语音控制复用
//2、向客户端发送温湿度数据 
void *socket_thread(){
    struct sockaddr_in c_addr;
    memset(&c_addr,0,sizeof(struct sockaddr_in));
    int clen = sizeof(struct sockaddr_in);

    socketHandler = findCommandByName("socketServer", pCommandHead);
    if(socketHandler == NULL){
            printf("find socketHandler error\n");
            pthread_exit(NULL);
    }
    printf("%s init success\n", socketHandler->commandName);
    socketHandler->Init(socketHandler, NULL, NULL);

    while(1){
        c_fd = accept(socketHandler->sfd, (struct sockaddr *)&c_addr, &clen);
        piThreadCreate(read_thread);
        pthread_tempAndHumi_create();
    }
}

 在网络线程中,首先对套接字进行初始化配置,包括IPV4因特网协议、TCP协议的配置(socket),绑定IP地址和端口号(bind),通过套接字标识符监听对应端口(listen)并等待客户端接入(accept)。

客户端接入后,即创建读数据线程,在while循环中进程阻塞在read函数,直到客户端发出指令,指令的执行函数与语音模块的语音指令执行函数复用(voiceContrlFunc函数在第4小点展示)。例如:在QT界面按下 [开客厅灯] 按钮,客户端通过网络发出字符串"OLL" ,在服务端read读到指令放入socketHandler->command,调用函数voiceContrlFunc(socketHandler->command)执行指令。

同时客户端在接入之后会调用函数pthread_tempAndHumi_create()创建发数据的线程,即实时发送温湿度的数据,QT中配合信号槽接收数据并在QT界面上显示,代码如下。

void *read_thread(){
    while(1){
        int n_read = 0;
        memset(socketHandler->command, '\0', sizeof(socketHandler->command));
        n_read = read(c_fd, socketHandler->command, sizeof(socketHandler->command));//n_read是读到字节数
        voiceContrlFunc(socketHandler->command);
        if(n_read == -1){
		    perror("read");
	    }
        else if(n_read>0){
		    printf("\nget: %d, %s\n",n_read, socketHandler->command);
	    }
        else{			
		    printf("client quit\n");
            break;
        }
    }
}

 QT程序,控制开关客厅灯的函数:

void Widget::on_livingRoomLight_clicked()
{
    if(livingRoomLightFlag == 1){
        //客户端向服务端发送消息
        if(tcpSocket->state() == QAbstractSocket::ConnectedState){
            ui->livingRoomLight->setText("关客厅灯");
            tcpSocket->write("OLL");
            ui->textBrowser->append("> 客厅灯已打开\n");
            livingRoomLightFlag = 0;
        }
        else{
            ui->textBrowser->append("请先与服务端连接!");
        }
    }
    else{
        if(tcpSocket->state() == QAbstractSocket::ConnectedState){
            ui->livingRoomLight->setText("开客厅灯");
            tcpSocket->write("CLL");
            ui->textBrowser->append("> 客厅灯已关闭\n");
            livingRoomLightFlag = 1;
        }
        else{
            ui->textBrowser->append("请先与服务端连接!");
        }
    }
}

 QT程序中的信号槽连接和TCP网络读函数:

connect(tcpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessages()));

void Widget::receiveMessages()
{
    QByteArray tmpByteArray = tcpSocket->readAll();
    char* tempHumi;
    tempHumi = tmpByteArray.data();
    ui->tempLabel->setText(QString::number(tempHumi[0]));
    ui->humiLabel->setText(QString::number(tempHumi[2]));
}

 

                                                                   PC端QT界面

                                                                         APP界面

3、wiringPi库下的相关硬件操作函数调用

包括wiringPi库的初始化,蜂鸣器、火焰传感器、继电器组的输入输出引脚配置和高低电平设置。wiringP库下的串口配置和初始化。

4、语音模块的串口通信

本文用的是全志H616芯片中的串口设备/dev/ttyS5,波特率115200,实现与语音模块的串口通信。语音模块收到我们发出的命令词,将命令词转化为16进制数据通过串口发送到开发板,在程序中完成对串口数据接收、存储、判断是哪种命令、执行对应命令的操作,包括开关二楼灯、餐厅灯、客厅灯、卫生间灯、开启摄像头、拍一张照片和人脸识别。

语音模块SU-03T需要烧入对应命令的SDK,本文配置的SDK是在智能公元/AI产品零代码平台上完成,免费的。

int get_voice_type(char *cmd)
{
	if(!strcmp("OLL", cmd))    return OLL;
	if(!strcmp("ORL", cmd))    return ORL;
	if(!strcmp("OUL", cmd))    return OUL;
    if(!strcmp("OBL", cmd))    return OBL;
    if(!strcmp("CLL", cmd))    return CLL;
    if(!strcmp("CRL", cmd))    return CRL;
    if(!strcmp("CUL", cmd))    return CUL;
    if(!strcmp("CBL", cmd))    return CBL;
    if(!strcmp("OC" , cmd))    return OC ;
    if(!strcmp("TAP", cmd))    return TAP;
    if(!strcmp("OFR", cmd))    return OFR;
    perror("voice recognition failure");
}

void voiceContrlFunc(char *cmd){
	switch(get_voice_type(cmd)){
		case OLL://OLL ASCII对应的16进制4f 4c 4c
			printf("open livingroom light\n");
			struct Devices *tmpOpenLivingroomLight = findDeviceByName("livingroomLight", pdeviceHead);
			tmpOpenLivingroomLight->open(tmpOpenLivingroomLight->pinNum);
            break;
		case ORL://ORL 4f 52 4c
			printf("open restaurant light\n");
            struct Devices *tmpOpenRestaurantLight = findDeviceByName("restaurantLight", pdeviceHead);
			tmpOpenRestaurantLight->open(tmpOpenRestaurantLight->pinNum);
			break;
		case OUL://OUL 4f 55 4c
			printf("open upstair light\n");
            struct Devices *tmpOpenUpstairLight = findDeviceByName("upstairLight", pdeviceHead);
			tmpOpenUpstairLight->open(tmpOpenUpstairLight->pinNum);
			break;
		case OBL://OBL 4f 42 4c
			printf("open bathroom light\n");
            struct Devices *tmpOpenBathroomLight = findDeviceByName("bathroomLight", pdeviceHead);
			tmpOpenBathroomLight->open(tmpOpenBathroomLight->pinNum);
			break;
		case CLL://CLL 43 4c 4c
			printf("close livingroom light\n");
            struct Devices *tmpCloseLivingroomLight = findDeviceByName("livingroomLight", pdeviceHead);
			tmpCloseLivingroomLight->close(tmpCloseLivingroomLight->pinNum);
			break;
		case CRL://CRL 43 52 4c
			printf("close restaurant light\n");
            struct Devices *tmpCloseRestaurantLight = findDeviceByName("restaurantLight", pdeviceHead);
			tmpCloseRestaurantLight->close(tmpCloseRestaurantLight->pinNum);
			break;
		case CUL://CUL 43 55 4c
			printf("close upstair light\n");
            struct Devices *tmpCloseUpstairLight = findDeviceByName("upstairLight", pdeviceHead);
			tmpCloseUpstairLight->close(tmpCloseUpstairLight->pinNum);
			break;
		case CBL://CBL 43 42 4c
			printf("close bathroom light\n");
            struct Devices *tmpCloseBathroomLight = findDeviceByName("bathroomLight", pdeviceHead);
			tmpCloseBathroomLight->close(tmpCloseBathroomLight->pinNum);
			break;
		case OC://OC 4f 43
			printf("open camera\n");
            printf("   -------------------------------------------------------------------\n");
            printf("   --\033[1;32m 已开启摄像头,请到指定网页观看画面 https//192.168.43.206:8081 \033[0m--\n");//黄色字体
            printf("   -------------------------------------------------------------------\n");
            printf("\n");
			break;
		case TAP://TAP 54 41 50
			printf("take a picture\n");
            struct Devices *tmpTakeAPictureCamera = findDeviceByName("camera", pdeviceHead);
			tmpTakeAPictureCamera->takeAPicture();
			printf("   --------------------------------------\n");
            printf("   --\033[1;32m 已拍照,请在当前文件夹下查看照片 \033[0m--\n");//黄色字体
            printf("   --------------------------------------\n");
            printf("\n");
			break;
		case OFR://OFR 4f 46 52
			printf("open face recognition\n");
            struct Devices *tmpFaceRecCamera = findDeviceByName("camera", pdeviceHead);
			tmpFaceRecCamera->faceRecognition();
			break;
	}
}

5、摄像头的实时监控和拍照功能

参考这篇文章:(1031条消息) 香橙派Orange Pi Zero 2开发板配置USB摄像头的方法_阿龙还在写代码的博客-CSDN博客

6、人脸识别

通过命令词【人脸识别】让摄像头拍摄一张照片并与本地的本人照片做对比,成功则蜂鸣器滴一声,失败则滴四声。人脸识别的处理程序调用的是翔云OCR人脸识别 (netocr.com)的人脸识别API,该接口地址https://netocr.com/api/faceliu.do是https协议的地址,https协议是在http和tcp之间多添加了一层,进行身份验证和数据加密。

若要访问https协议的地址,则需要用到Libcurl这个跨平台的网络协议库,配合OpenSSL库,就可以访问https协议的接口。(编译OpenSSL支持Libcurl的https访问,如果直接编译Libcurl,只能访问http不能访问https,需要OpenSSL库才能访问https)

 camera.c

#include "contrlDevices.h"
#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>

extern struct Devices *pdeviceHead;//extern表面变量或函数是定义在其他文件中,声明 为全局变量在该源文件使用
struct Devices* findDeviceByName(char *name, struct Devices *phead);

#define true 1
#define false 0
struct Devices camera;
char buf[1024] = {'\0'};//全局

void cameraTakeAPicture(){
    system("(fswebcam -d /dev/video0 --no-banner -r 1280x720 -S 5 ./image.jpg) > tmpFile");//照片存放在当前目录下
}

size_t readData(void *ptr, size_t size, size_t nmemb, void *stream){
    strncpy(buf, ptr, 1024);
}


char *getPicBase64FromFile(char *filePath){
    char *bufPic = NULL;
    char cmd[128] = {'\0'};
    sprintf(cmd, "base64 %s > tmpFile", filePath);
    system(cmd);//图片的base64流数据存入tmpFile文件中

    int fd = open("./tmpFile", O_RDWR);
    int filelen = lseek(fd, 0, SEEK_END);
    lseek(fd, 0, SEEK_SET);//重新让文件的光标回到初始位置
    bufPic = (char *)malloc(filelen + 2);//+1也可以 多加点没毛病
    memset(bufPic, '\0', filelen + 2);
    read(fd, bufPic, filelen);
    close(fd);
    system("rm -f tmpFile");
    return bufPic;
}

void cameraFaceRecognition(){
    camera.takeAPicture();
    CURL *curl;
    CURLcode res;
    char *postString;
    char *img1;
    char *img2;
    char *key = "DYRrmZz2rTwYGywyWdhKzR";
    char *secret = "56bc8e083a9b4d9fbf590413ddcb3a61";
    int  typeId = 21;
    char *format = "xml";
    char *bufPic1 = getPicBase64FromFile("./image.jpg");
    char *bufPic2 = getPicBase64FromFile("./zyl.jpg");

    int len = strlen(key) + strlen(secret) + strlen(bufPic1) + strlen(bufPic2) + 128;
    postString = (char *)malloc(len);
    memset(postString, '\0', len);//sizeof(postString)替换成len,因为postString是指针
    sprintf(postString, "&img1=%s&img2=%s&key=%s&secret=%s&typeId=%d&format=%s", bufPic1, bufPic2, key, secret, 21, 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);

        struct Devices *beepHandler = findDeviceByName("beep", pdeviceHead);
        if(strstr(buf, "是") != NULL){ 
            beepHandler->open(beepHandler->pinNum); usleep(300000);
            beepHandler->close(beepHandler->pinNum);
            printf("\n");
            printf("   -----------------------------------\n");
            printf("   --\033[1;32m 人脸识别成功: the same person \033[0m--\n");//绿色字体
            printf("   -----------------------------------\n");
            printf("\n");
        }
        else{
            int i = 4;
            while(i--){
                beepHandler->open(beepHandler->pinNum); usleep(200000);
                beepHandler->close(beepHandler->pinNum); usleep(100000);
            }
            printf("\n");
            printf("   ------------------------------------\n");
            printf("   --\033[1;31m 人脸识别失败: different person \033[0m--\n");//红色字体
            printf("   ------------------------------------\n");
            printf("\n");
        }
        curl_easy_cleanup(curl);
    }
}

//实例化对象
struct Devices camera = {
    .deviceName = "camera",

    .takeAPicture = cameraTakeAPicture,
    .faceRecognition = cameraFaceRecognition

};

struct Devices* addCameraToDeviceLink(struct Devices *phead){
    if(phead == NULL){
        return &camera;
    }
    else{//头插
        camera.next = phead;
        phead = &camera;
        return phead;
    }
}

7、qt程序跨平台运行(编译成安卓APP)

搭建环境所需的安装包:

四、编译和运行

编译时需要用到一些库文件和该库文件里的头文件,用到温湿度传感器时加上tempAndHumi.c

gcc bathroomLight.c livingroomLight.c restaurantLight.c upstairLight.c socketContrl.c voiceContrl.c fireDetection.c beep.c camera.c usartContrl.c main.c -I ../httpHandler/curl-7.71.1/_install/include/ -L ../httpHandler/curl-7.71.1/_install/lib/ -lcurl -lwiringPi -lpthread

运行:

sudo ./a.out

五、视频功能展示

           智能家居功能展示

        PC端QT界面功能展示

            安卓app功能展示

猜你喜欢

转载自blog.csdn.net/qq_43460230/article/details/131945903