智能家居6 -- 配置 ini文件优化设备添加

2024-05-24 22:20

本文主要是介绍智能家居6 -- 配置 ini文件优化设备添加,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

不知道什么是ini的朋友可以先看这篇:一文带你入门ini格式-CSDN博客

准备

如下图:

在src 下面添加 ini.c

在inc 下面添加 ini.h

在 receive_interface.c 里面包含头文件,把之前添加的设备类注释掉

这时候就可以把相关设备的(.c   .h)文件给删掉了

如下图:

  

修改/添加 代码

gdevice.ini

[lock]
key=0x44
gpio_pin=8
gpio_mode=OUTPUT
gpio_status=HIGH
check_face_status=1
voice_set_status=1[beep]
key=0x45
gpio_pin=9
gpio_mode=OUTPUT
gpio_status=HIGH
check_face_status=0
voice_set_status=1[BR led]
key=0x42
gpio_pin=5
gpio_mode=OUTPUT
gpio_status=HIGH
check_face_status=0
voice_set_status=0[LV led]
key=0x41
gpio_pin=2
gpio_mode=OUTPUT
gpio_status=HIGH
check_face_status=0
voice_set_status=0[fan]
key=0x43
gpio_pin=7
gpio_mode=OUTPUT
gpio_status=HIGH
check_face_status=0
voice_set_status=0

修改init

我们需要把gdevice.ini 文件传送到arm 设备的 /etc/下面

sudo cp gdevice.ini /etc/

修改后的receive_interface.c

#include <pthread.h>
#include <mqueue.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
#include <stdio.h>
#include <wiringPi.h>#include "receive_interface.h"
#include "control.h"
#include "mq_queue.h"
#include "global.h"
#include "face.h"
#include "myoled.h"
//#include "lrled_gdevice.h"
#include "gdevice.h"
// #include "fan_gdevice.h"
// #include "bled_gdevice.h"
// #include "beep_gdevice.h"
// #include "lock_gdevice.h"#include "ini.h"
#include "face.h"#define MATCH(s, n) strcmp(section, s) == 0 && strcmp(name, n) == 0/*
接收模块:
对接收到消息做出相应处理
包括 oled 人脸识别 语言播报 GPIO  引脚状态配置*/static int oled_fd = -1;
static struct gdevice *pdevhead = NULL;typedef struct
{int msg_len;unsigned char *buffer;ctrl_info_t *ctrl_info;
} recv_msg_t;static int handler_gdevice(void *user, const char *section, const char *name, const char *value)
{struct gdevice *pdev = NULL;if (NULL == pdevhead){pdevhead = (struct gdevice *)malloc(sizeof(struct gdevice));memset(pdevhead, 0, sizeof(struct gdevice));pdevhead->next = NULL;strcpy(pdevhead->dev_name, section);}// printf("section = %s, name = %s, value = %s\n", section, name, value);else if (0 != strcmp(section, pdevhead->dev_name)) // 当section对不上的时候,表示到了下一个设备{// 把新节点(设备)使用头插法插入pdev = (struct gdevice *)malloc(sizeof(struct gdevice));memset(pdev, 0, sizeof(struct gdevice));strcpy(pdev->dev_name, section);pdev->next = pdevhead;pdevhead = pdev;}if (NULL != pdevhead){if (MATCH(pdevhead->dev_name, "key")){sscanf(value, "%x", &pdevhead->key); // 把value(string)的值 转为int类型 16进行格式 传递给  pdevhead->key)printf("%d  pdevhead->key = 0x%x\n", __LINE__, pdevhead->key);}else if (MATCH(pdevhead->dev_name, "gpio_pin")){pdevhead->gpio_pin = atoi(value);}else if (MATCH(pdevhead->dev_name, "gpio_mode")){if (strcmp(value, "OUTPUT") == 0){pdevhead->gpio_mode = OUTPUT;}else if (strcmp(value, "INPUT") == 0){pdevhead->gpio_mode = INPUT;}else{printf("gpio_mode error\n");}}else if (MATCH(pdevhead->dev_name, "gpio_status")){if (strcmp(value, "LOW") == 0){pdevhead->gpio_mode = LOW;}else if (strcmp(value, "HIGH") == 0){pdevhead->gpio_mode = HIGH;}else{printf("gpio_status error\n");}}else if (MATCH(pdevhead->dev_name, "check_face_status")){pdevhead->check_face_status = atoi(value);}else if (MATCH(pdevhead->dev_name, "voice_set_status")){pdevhead->voice_set_status = atoi(value);}}return 1;
}static int receive_init(void)
{// pdevhead = add_lrled_to_gdevice_list(pdevhead); // 头插法加入 客厅灯// pdevhead = add_bled_to_gdevice_list(pdevhead);  // 加入卧室灯// pdevhead = add_fan_to_gdevice_list(pdevhead);   // 加入风扇// pdevhead = add_beep_to_gdevice_list(pdevhead);  // 蜂鸣器// pdevhead = add_lock_to_gdevice_list(pdevhead);  // 开锁if (ini_parse("/etc/gdevice.ini", handler_gdevice, NULL) < 0) {printf("Can't load 'gdevice.ini'\n");return 1;}struct gdevice *pdev = NULL;pdev = pdevhead;while (pdev != NULL){// printf("inside %d",__LINE__);printf("dev_name:%s\n", pdev->dev_name);printf("key:%x\n", pdev->key);printf("gpio_pin:%d\n", pdev->gpio_pin);printf("gpio_mode:%d\n", pdev->gpio_mode);printf("gpio_status:%d\n", pdev->gpio_status);printf("check_face_status:%d\n", pdev->check_face_status);printf("voice_set_status:%d\n", pdev->voice_set_status);pdev = pdev->next;}// 设备类链表添加oled_fd = myoled_init(); // 初始化oledface_init();             // 初始化人脸识别return oled_fd;
}static void receive_final(void)
{face_final();if (-1 != oled_fd){close(oled_fd); // 关闭oled 打开的文件oled_fd = -1;   // 复位}
}//  处理设备 --  比如打开灯 和风扇等static void *handler_device(void *arg)
{pthread_detach(pthread_self()); // 和主线程(他的父线程)分离recv_msg_t *recv_msg = NULL;struct gdevice *cur_gdev = NULL;char success_or_failed[20] = "success";pthread_t tid = -1;int smoke_status = 0;double face_result = 0.0; //存放人脸匹配度int ret = -1;if (NULL != arg) // 有参数{recv_msg = (recv_msg_t *)arg; // 获取参数printf("recv_len = %d\n", recv_msg->msg_len);printf("%s|%s|%d, handler: 0x%x,0x%x,0x%x,0x%x,0x%x,0x%x\n", __FILE__, __func__, __LINE__,recv_msg->buffer[0], recv_msg->buffer[1], recv_msg->buffer[2], recv_msg->buffer[3], recv_msg->buffer[4], recv_msg->buffer[5]);}// need to do somethingif (NULL != recv_msg && NULL != recv_msg->buffer) // if 消息队列非空,并且buffer 里面接收到数据{// recv_msg->buffer[2] -->  第三位 用于存放设备类型cur_gdev = find_device_by_key(pdevhead, recv_msg->buffer[2]);printf("%s|%s|%d,find success   buffer[2] = 0x%x \n", __FILE__, __func__, __LINE__, recv_msg->buffer[2]);}if (NULL != cur_gdev) // if 能找到的这设备 --> 设备存在{printf("%s|%s|%d, cur_gdev \n", __FILE__, __func__, __LINE__);// BUFFER 的第四个参数  用于 存放开关状态 0 表示开, 1 表示关cur_gdev->gpio_status = recv_msg->buffer[3] == 0 ? LOW : HIGH; // 获取状态存入cur_gdev中//人脸识别if(1 == cur_gdev->check_face_status){face_result = face_status(); //得到人脸识别的匹配度if(face_result > 0.6){ //匹配成功ret = set_gpio_device_status(cur_gdev); // 设置电平 --> 开锁recv_msg->buffer[2] = 0x47;  //识别成功的语音播报}else{recv_msg->buffer[2] = 0x46;}}else if( 0 == cur_gdev->check_face_status){// printf("%s|%s|%d,Set  before set_gpio_device_status\n",__FILE__,__func__,__LINE__);ret = set_gpio_device_status(cur_gdev); // 将获取到的状态真正赋值给引脚// printf("%s|%s|%d, after set_gpio_device_status \n",__FILE__,__func__,__LINE__);}printf("%s|%s|%d, = %d\n", __FILE__, __func__, __LINE__,cur_gdev->voice_set_status);// 需要语言播报if (1 == cur_gdev->voice_set_status) {printf("%s|%s|%d,2\n", __FILE__, __func__, __LINE__);if (NULL != recv_msg && NULL != recv_msg->ctrl_info && NULL != recv_msg->ctrl_info->ctrl_phead){printf("%s|%s|%d,2\n", __FILE__, __func__, __LINE__);struct control *pcontrol = recv_msg->ctrl_info->ctrl_phead;while (NULL != pcontrol){if (strstr(pcontrol->control_name, "voice")) //匹配到语言播报{if (0x45 == recv_msg->buffer[2] && 0 == recv_msg->buffer[3]) // 语音播报 打开{smoke_status = 1;}pthread_create(&tid, NULL, pcontrol->set, (void *)recv_msg->buffer); // 新开线程区进行语言播报break;}pcontrol = pcontrol->next;}}}printf("%s|%s|%d,2\n", __FILE__, __func__, __LINE__);if (-1 == ret) // 设置失败{printf("%s|%s|%d,2\n", __FILE__, __func__, __LINE__);memset(success_or_failed, '\0', sizeof(success_or_failed));strncpy(success_or_failed, "failed", 6);}printf("%s|%s|%d,2\n", __FILE__, __func__, __LINE__);// 配置OLEDchar oled_msg[512];memset(oled_msg, 0, sizeof(oled_msg));char *change_status = cur_gdev->gpio_status == LOW ? "Open" : "Close";sprintf(oled_msg, "%s %s %s!\n", change_status, cur_gdev->dev_name, success_or_failed);if(smoke_status == 1){memset(oled_msg, 0, sizeof(oled_msg));sprintf(oled_msg, "A risk of fire!\n");}  myoled_show(oled_msg);//让门打开5s自动关闭if(1 == cur_gdev->check_face_status && 0 == ret && face_result >0.6){sleep(5); //开门5s后关门cur_gdev->gpio_status = HIGH; //设置高电平(低电平有效)ret = set_gpio_device_status(cur_gdev); //关门}}pthread_exit(0);
}static void *receive_get(void *arg) // 接收消息队列里面的 数据
{printf("enter receive_get\n");//  通过参数 初始化我们 定义的recv_msg_t 结构体recv_msg_t *recv_msg = NULL;unsigned char *buffer = NULL;struct mq_attr attr;pthread_t tid = -1;ssize_t read_len = -1;if (NULL != arg){recv_msg = (recv_msg_t *)malloc(sizeof(recv_msg_t));recv_msg->ctrl_info = (ctrl_info_t *)arg; // 这里实际上就获取到了mqd 和 phead(我们需要操作的struct control 链表 的头节点)recv_msg->msg_len = 0;recv_msg->buffer = NULL;}elsepthread_exit(0);if (mq_getattr(recv_msg->ctrl_info->mqd, &attr) == -1){ // 获取消息队列失败 -- 异常pthread_exit(0);}// 能获取到消息队列recv_msg->buffer = (unsigned char *)malloc(attr.mq_msgsize); // 分配内存buffer = (unsigned char *)malloc(attr.mq_msgsize);// mq_msgsize -- 每条消息的大小memset(recv_msg->buffer, 0, attr.mq_msgsize); // 初始化memset(buffer, 0, attr.mq_msgsize);           // 初始化pthread_detach(pthread_self()); // 和主线程(他的父线程)分离while (1){read_len = mq_receive(recv_msg->ctrl_info->mqd, buffer, attr.mq_msgsize, NULL);printf("%s|%s|%d, recv: 0x%x,0x%x,0x%x,0x%x,0x%x,0x%x\n", __FILE__, __func__, __LINE__, buffer[0], buffer[1], buffer[2], buffer[3], buffer[4], buffer[5]);printf("%s|%s|%d: read_len = %ld\n", __FILE__, __func__, __LINE__, read_len);if (-1 == read_len){ // 接收失败if (errno == EAGAIN){printf("queue is empty\n");}else{break;}}// 以下是接收到正常数据的情况else if (buffer[0] == 0xAA && buffer[1] == 0x55 && buffer[4] == 0x55 && buffer[5] == 0xAA){recv_msg->msg_len = read_len;memcpy(recv_msg->buffer, buffer, read_len);//  创建线程去 处理我们的接收到的信号pthread_create(&tid, NULL, handler_device, (void *)recv_msg);}}if (NULL != recv_msg)free(recv_msg);if (NULL != buffer)free(buffer);pthread_exit(0);
}struct control receive_control = {.control_name = "receive",.init = receive_init,.final = receive_final,.get = receive_get,.set = NULL, // 不需要实现 设置.next = NULL};struct control *add_receive_to_ctrl_list(struct control *phead)
{// 头插法实现 添加链表节点return add_interface_to_ctrl_list(phead, &receive_control);
};

这样要是我们需要对设备进行维护: 

比如添加某个新的设备只需要去 gdevice.ini 里面去添加即可 -->方便维护

这篇关于智能家居6 -- 配置 ini文件优化设备添加的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/999694

相关文章

MySQL深分页进行性能优化的常见方法

《MySQL深分页进行性能优化的常见方法》在Web应用中,分页查询是数据库操作中的常见需求,然而,在面对大型数据集时,深分页(deeppagination)却成为了性能优化的一个挑战,在本文中,我们将... 目录引言:深分页,真的只是“翻页慢”那么简单吗?一、背景介绍二、深分页的性能问题三、业务场景分析四、

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

nginx 负载均衡配置及如何解决重复登录问题

《nginx负载均衡配置及如何解决重复登录问题》文章详解Nginx源码安装与Docker部署,介绍四层/七层代理区别及负载均衡策略,通过ip_hash解决重复登录问题,对nginx负载均衡配置及如何... 目录一:源码安装:1.配置编译参数2.编译3.编译安装 二,四层代理和七层代理区别1.二者混合使用举例

Java JDK1.8 安装和环境配置教程详解

《JavaJDK1.8安装和环境配置教程详解》文章简要介绍了JDK1.8的安装流程,包括官网下载对应系统版本、安装时选择非系统盘路径、配置JAVA_HOME、CLASSPATH和Path环境变量,... 目录1.下载JDK2.安装JDK3.配置环境变量4.检验JDK官网下载地址:Java Downloads

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

Spring Boot spring-boot-maven-plugin 参数配置详解(最新推荐)

《SpringBootspring-boot-maven-plugin参数配置详解(最新推荐)》文章介绍了SpringBootMaven插件的5个核心目标(repackage、run、start... 目录一 spring-boot-maven-plugin 插件的5个Goals二 应用场景1 重新打包应用

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

Jenkins分布式集群配置方式

《Jenkins分布式集群配置方式》:本文主要介绍Jenkins分布式集群配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装jenkins2.配置集群总结Jenkins是一个开源项目,它提供了一个容易使用的持续集成系统,并且提供了大量的plugin满

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

SQL Server配置管理器无法打开的四种解决方法

《SQLServer配置管理器无法打开的四种解决方法》本文总结了SQLServer配置管理器无法打开的四种解决方法,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录方法一:桌面图标进入方法二:运行窗口进入检查版本号对照表php方法三:查找文件路径方法四:检查 S