From 7a0488541f68d1a9eb712a4eb5c6c36da150bc4a Mon Sep 17 00:00:00 2001 From: guowenxue <guowenxue@gmail.com> Date: Fri, 23 Apr 2021 22:43:13 +0800 Subject: [PATCH] update iotd and test infared ok --- iotd/hal/gpio.c | 118 +++++++++++++++++++++++++++++++------- iotd/etc/iotd.conf | 7 +- iotd/main.c | 15 +++- iotd/hal/gpio.h | 6 + iotd/lylib/util_time.h | 9 +++ 5 files changed, 122 insertions(+), 33 deletions(-) diff --git a/iotd/etc/iotd.conf b/iotd/etc/iotd.conf index b9cb8c0..2491c31 100644 --- a/iotd/etc/iotd.conf +++ b/iotd/etc/iotd.conf @@ -22,8 +22,7 @@ [hardware] # LED或继电器等GPIO输出控制, 格式: {名称:BCM编码:控制电平} -gpio_outpin={light_indoor:6:1},{light_livroomL:13:1},{light_livroomR:19:1},{light_hallway:26:1} -#gpio_outpin={led_red:6:1},{led_green:13:1},{led_blue:19:1} +gpio_outpin={light_indoor:6:1},{light_hallway:26:1} # 红外探测到人后,继电器控制灯亮的时长 light_intval=15 @@ -65,13 +64,13 @@ [subsciber] -subTopic="$Sys/Studio/iotd/Downlink" +subTopic="$Sys/Studio/Downlink/iotd" subQos=0 [publisher] -pubTopic="$Sys/Studio/iotd/Uplink" +pubTopic="$Sys/Studio/Uplink/" pubQos=0 # Publisher上报传感器数据的周期,单位是秒 diff --git a/iotd/hal/gpio.c b/iotd/hal/gpio.c index 72dd269..73340a6 100644 --- a/iotd/hal/gpio.c +++ b/iotd/hal/gpio.c @@ -12,6 +12,7 @@ ********************************************************************************/ #include "lylib/logger.h" +#include "lylib/util_proc.h" #include "gpio.h" #define RPI_GPIONAME "gpiochip0" @@ -158,55 +159,126 @@ return ; } -#define MS_NANOSEC (1*1000*1000) + +void *light_on_worker(void *arg) +{ + int i; + int found = 0; + char *name = (char *)arg; + gpio_info_t *gpiout; + + if( !name ) + { + log_err("Invalid input arugment\n"); + return NULL; + } + + if( s_gpio->light_intval <= 0 ) + { + log_err("Invalid light_intval value[%d] in configure file\n", s_gpio->light_intval); + return NULL; + } + + for(i=0; i<s_gpio->outcnt; i++) + { + if( strstr(s_gpio->output[i].name, name) ) + { + gpiout = &(s_gpio->output[i]); + found = 1; + } + } + + if( !found ) + { + log_err("Light '%s' not found\n", name); + return NULL; + } + + log_nrml("Trun on %s for [%d] seconds\n", gpiout->name, s_gpio->light_intval); + + gpio_out(gpiout->name, "on"); + sleep(s_gpio->light_intval); + gpio_out(gpiout->name, "off"); + + return NULL; +} /* Return value: 0(LOW): Nobody detected, !0: indoor or hallway detected */ -int infrared_detect(int interval) +int infrared_detect(void) { int i; int rv = 0; int res = 0; - struct gpiod_line_event event; - int event_type; - struct timespec ts = { 0, interval*MS_NANOSEC }; /* second and nanoseconds, */ + + struct gpiod_line_event ev; + struct gpiod_line_bulk bulk, event_bulk; + struct gpiod_line *line; + + int num_lines = 0; + + + gpiod_line_bulk_init(&bulk); for(i=0; i<s_gpio->incnt; i++) { if( strstr(s_gpio->input[i].name, "infrared")) { - /* This function will block, must call it to setup */ - rv = gpiod_line_event_wait(s_gpio->input[i].line, &ts); - if( rv < 0 ) - { - //log_err("infrared gpiod line wait[%s] event failure!\n", s_gpio->input[i].name); - return 0; - } - else if( rv == 0 ) - { - //log_dbg("infrared gpiod line[%s] wait event timeout\n", s_gpio->input[i].name); - continue; - } + gpiod_line_bulk_add(&bulk, s_gpio->input[i].line); + num_lines ++; + } + } - /* Must read to clear the event */ - rv = gpiod_line_event_read(s_gpio->input[i].line, &event); + if( !num_lines ) + { + log_err("No infrared detect gpiod lines found\n"); + return 0; + } - event_type = s_gpio->input[i].active_level? GPIOD_LINE_EVENT_RISING_EDGE : GPIOD_LINE_EVENT_FALLING_EDGE; + log_dbg("infrared start detect event now...\n"); + rv = gpiod_line_event_wait_bulk(&bulk, NULL, &event_bulk); + if( rv <= 0 ) + { + log_err("infrared gpiod line wait[%s] event failure!\n", s_gpio->input[i].name); + return 0; + } - if ( event.event_type == event_type ) + for (i=0; i<num_lines; i++) + { + line = gpiod_line_bulk_get_line(&event_bulk, 0); + + /* must call this function to clear the event */ + rv = gpiod_line_event_read(line, &ev); + if( rv < 0) + { + log_err("gpiod_line_event_read get failure\n"); + break; + } + + for(i=0; i<s_gpio->incnt; i++) + { + if(line == s_gpio->input[i].line) { + if( s_gpio->input[i].active_level != gpiod_line_get_value(line) ) + { + log_dbg("infrared '%s' detect get wrong power level\n", s_gpio->input[i].name); + continue; + } + if( strstr(s_gpio->input[i].name, "indoor") ) { + log_dbg("indoor infrared gpiod wait get event.\n", s_gpio->input[i].name); res |= FLAG_INFRARED_INDOOR; } else if( strstr(s_gpio->input[i].name, "hallway") ) { + log_dbg("hallway infrared gpiod wait get event.\n", s_gpio->input[i].name); res |= FLAG_INFRARED_HALLWAY; } } - } } - return res; + return res; } + diff --git a/iotd/hal/gpio.h b/iotd/hal/gpio.h index 2623e72..804916a 100644 --- a/iotd/hal/gpio.h +++ b/iotd/hal/gpio.h @@ -45,10 +45,14 @@ /* turn which light on/off */ extern void gpio_out(char *name, char *cmd); + +/*thread work body to turn light $name on for some seconds */ +void *light_on_worker(void *arg); + /* Return value: 0(LOW): Nobody detected, !0: Which infrared detect incoming */ #define FLAG_INFRARED_INDOOR (1<<0) #define FLAG_INFRARED_HALLWAY (1<<1) -extern int infrared_detect(int interval); +extern int infrared_detect(void); #endif /* ----- #ifndef _GPIO_H_ ----- */ diff --git a/iotd/lylib/util_time.h b/iotd/lylib/util_time.h index 05a2f9a..a48d8ae 100644 --- a/iotd/lylib/util_time.h +++ b/iotd/lylib/util_time.h @@ -25,6 +25,15 @@ #include <sys/stat.h> #include <sys/time.h> + +#ifndef offsetof +#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER) +#endif + +#define container_of(ptr, type, member) ({ \ + const typeof( ((type *)0)->member ) *__mptr = (ptr); \ + (type *)( (char *)__mptr - offsetof(type,member) );}) + typedef struct date_time_s { int year; diff --git a/iotd/main.c b/iotd/main.c index ed29b06..53cdf9c 100644 --- a/iotd/main.c +++ b/iotd/main.c @@ -194,26 +194,31 @@ if( lux > hal_ctx->lux_threshold ) { - log_nrml("Lux[%.3f] > Treshold[%.3f], don't need light on.\n", lux, hal_ctx->lux_threshold); - sleep(30); + log_nrml("Lux[%.3f] > Treshold[%.3f], don't need light on and sleep now.\n", lux, hal_ctx->lux_threshold); + if( hal_ctx->gpio.light_intval > 0) + sleep( hal_ctx->gpio.light_intval ); + else + sleep(30); + continue; } } - rv = infrared_detect(250); - + rv = infrared_detect(); if( rv & FLAG_INFRARED_INDOOR ) { log_nrml("Someone incoming detected by indoor infrared\n"); + thread_start(NULL, light_on_worker, "indoor" ); } if( rv & FLAG_INFRARED_HALLWAY ) { log_nrml("Someone incoming detected by hallway infrared\n"); + thread_start(NULL, light_on_worker, "hallway" ); } } - msleep(500); + msleep(100); } } -- Gitblit v1.9.1