/** ****************************************************************************** * @file : app_tof.c * @author : IMG SW Application Team * @brief : This file provides code for the configuration * of the STMicroelectronics.X-CUBE-TOF1.3.2.0 instances. ****************************************************************************** * * @attention * * Copyright (c) 2022 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file * in the root directory of this software component. * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ #ifdef __cplusplus extern "C" { #endif /* Includes ------------------------------------------------------------------*/ #include "app_tof.h" #include "main.h" #include "sys_app.h" #include #include "usart.h" #include #if (defined(STS_P2)||defined(STS_T6)) #include "VL53L1X_api.h" #elif defined(L8) #include "53l8a1_ranging_sensor.h" #include "app_tof_pin_conf.h" #include "yunhorn_sts_sensors.h" #include "sts_lamp_bar.h" volatile uint8_t fhmos_fall=0, fhmos_human_movement=0, fhmos_occupancy=0, fhmos_sos_alarm=0; volatile uint32_t fhmos_fall_counter=0; volatile uint32_t sts_low_threshold=1500, sts_high_threshold=2800, sts_occupancy_threshold=2300; extern volatile uint8_t sts_head_level_low; volatile sts_fhmos_sensor_config_t fhmos_cfg={70,4,4,8,20,80,20,25}; volatile sts_fhmos_sensor_cmd_t fhmos_cmd={0x0}; volatile sts_fhmos_sensor_ambient_height_t fhmos_bg, fhmos_gesture, fhmos_net; volatile sts_fhmos_sensor_data_t fhmos_data={0}; extern volatile uint8_t sts_fhmos_result; volatile uint8_t sts_mask_bitmap[8]={0x0}, fhmos_gesture_bitmap[8]={0x0}; extern volatile uint16_t sts_sensor_install_height; //in mm extern volatile uint8_t sts_lamp_bar_color; extern volatile uint8_t sts_status_color; extern volatile uint8_t sts_hall1_read,sts_hall2_read; // Above hall1_read == reed_hall_result, hall2_read == emergency_button extern volatile uint8_t sts_pir_read; extern volatile uint8_t sts_fhmos_state_changed; extern volatile uint8_t sts_fhmos_bitmap_pending; #endif #include "stm32wlxx_nucleo.h" static int to_confirm = 0; static uint32_t STS_Get_Center_Range_Distance(RANGING_SENSOR_Result_t *Result); /* Private typedef -----------------------------------------------------------*/ /* * The application is to showcase the threshold detection * functionality of this device. * * When the device detects a target that match the configuration an IT is generated * and the host will start printing the measurement information on the serial connection (UART). * * Here is the default configuration: * * ITConfig.Criteria = RS_IT_IN_WINDOW; * ITConfig.LowThreshold = 200; // distance in mm * ITConfig.HighThreshold = 600; // distance in mm * * Other availables interrupt generation criteria for this device are: * * - RS_IT_DEFAULT // IT if new measurement is ready (no thresholds) * - RS_IT_IN_WINDOW // IT if distance > thresh_high * - RS_IT_OUT_OF_WINDOW // IT if distance < LowThreshold OR distance > HighThreshold * - RS_IT_BELOW_LOW // IT if distance <= LowThreshold * - RS_IT_ABOVE_HIGH // IT if distance > HighThreshold * - RS_IT_EQUAL_LOW // IT if distance == LowThreshold * - RS_IT_NOT_EQUAL_LOW // IT if distance != LowThreshold */ /* Private define ------------------------------------------------------------*/ //#define TIMING_BUDGET (30U) /* 16 ms < TimingBudget < 500 ms */ #define TIMING_BUDGET (50U) /* 16 ms < TimingBudget < 500 ms */ #define POLLING_PERIOD (250U) /* refresh rate for polling mode (milliseconds, shall be consistent with TimingBudget value) */ #ifdef L8 /* for VL53L8A1 */ #define RANGING_FREQUENCY (10U) /* Ranging frequency Hz (shall be consistent with TimingBudget value) */ #define LOW_THRESHOLD (600U) #define HIGH_THRESHOLD (2000U) // floor, eg. 2200 #define OCCUPANCY_THRESHOLD (1500) // assume high people 2000-450 = 1550 /* ceiling -------------------- zero - 00 ref. 3000 mm high * * * people high --------------- 2100 mm high people * * * * * people low ------------------- 900 mm child normal * * * * other things ----------------- 400 mm * * * floor ----------------------00000 * * */ /* Private variables ---------------------------------------------------------*/ static RANGING_SENSOR_Capabilities_t Cap; static RANGING_SENSOR_ProfileConfig_t Profile; static RANGING_SENSOR_Result_t Result; //static VL53L8CX_ResultsData L8CXResult; //static VL53L8CX_Motion_Configuration motion_config; /* Motion configuration*/ //static VL53L8CX_Configuration Dev; static void MX_53L8A1_ThresholdDetection_Init(void); static void MX_53L8A1_ThresholdDetection_Process(void); static void print_result(RANGING_SENSOR_Result_t *Result); //static void sts_generate_fall_gesture_map(void); #endif static int32_t status = 0; volatile uint8_t ToF_EventDetected = 0; uint16_t sensor_id=0; /* Private function prototypes -----------------------------------------------*/ #if (defined(STS_P2)||defined(STS_T6)) uint8_t sts_vl53lx_ranging(uint16_t *ranged_distance, uint8_t range_mode, uint16_t distance_threshold_mm, uint16_t inter_measurement_ms, uint16_t macro_timing,uint16_t roi_width, uint16_t sigma_mm, uint16_t signal_kcps); #endif uint8_t IsInterruptDetected(uint16_t dev); #if (defined(STS_P2)||defined(STS_T6)) void STS_TOF_VL53LX_Range_Process(uint8_t range_mode, uint16_t *range_distance); #endif void MX_TOF_Init(void) { /* USER CODE BEGIN SV */ /* USER CODE END SV */ /* USER CODE BEGIN TOF_Init_PreTreatment */ /* USER CODE END TOF_Init_PreTreatment */ /* Initialize the peripherals and the TOF components */ //APP_LOG(TS_OFF,VLEVEL_L,"\r\n###################### MX TOF Init... \r\n"); //MX_53L1A2_SimpleRanging_Init(); //STS_TOF_VL53LX_PeopleCounting_Process(); #ifdef L8 MX_53L8A1_ThresholdDetection_Init(); MX_53L8A1_ThresholdDetection_Process(); STS_FHMOS_sensor_config_init(); //sts_high_threshold = sts_cfg_nvm.sensor_install_height_in_10cm ; //sts_low_threshold = sts_cfg_nvm.sensor_install_height_in_10cm -1400; STS_TOF_L8_Reconfig(); #elif defined(STS_R1D) XWL55_WLE5_53L0X_Init(); #endif /* USER CODE BEGIN TOF_Init_PostTreatment */ /* USER CODE END TOF_Init_PostTreatment */ } /* * LM background task */ void STS_LMZ_Ambient_Height_Scan_Process(void) { #if 1 uint8_t i=0; uint32_t range_distance =0; for (i=0; i<64; i++) { fhmos_bg.h2cm[i] = 0; fhmos_bg.maskoff[i] = 0; } for (i=0;i<8;i++) sts_mask_bitmap[i] =0x0; uint8_t rio_edge[34]={0,1,2,3,4,5,6,7,8,15,16,23,24,31,32,39,40,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63}; // printf("sts sensor install height = %4d \r\n", (int)sts_sensor_install_height); sts_high_threshold = sts_sensor_install_height; sts_low_threshold = sts_sensor_install_height-1400; //APP_LOG(TS_OFF, VLEVEL_M, "\r\nReconfig ----> Threshold High=%d mm, Low=%d mm \r\n", sts_high_threshold, sts_low_threshold); STS_TOF_L8_Reconfig(); status = VL53L8A1_RANGING_SENSOR_Start(VL53L8A1_DEV_CENTER, RS_MODE_ASYNC_CONTINUOUS); if (status != BSP_ERROR_NONE) { printf("VL53L8A1_RANGING_SENSOR_Start failed\n"); while (1); } if (ToF_EventDetected != 0) { ToF_EventDetected = 0; status = VL53L8A1_RANGING_SENSOR_GetDistance(VL53L8A1_DEV_CENTER, &Result); if (status != BSP_ERROR_NONE) { APP_LOG(TS_OFF, VLEVEL_M, "\r\n x \r\n"); } } //STS_TOF_L8_Process(); APP_LOG(TS_OFF, VLEVEL_L, "\r\n ----------------------" "\r\n------Gesture Mask off above Threshold %d cm--\r\n", fhmos_cfg.th_gesture_mask_off_height_cm); for (uint8_t i = 0; i < 64; i++) { /* Print distance and status */ if (i%8==0) APP_LOG(TS_OFF, VLEVEL_L, "\r\n[cm]|"); if ((Result.ZoneResult[i].NumberOfTargets > 0)) { range_distance = (uint32_t)Result.ZoneResult[i].Distance[0]; fhmos_bg.h2cm[i] = abs(sts_sensor_install_height - range_distance)/20; // in 2 cm if (2*fhmos_bg.h2cm[i] < fhmos_cfg.th_gesture_mask_off_height_cm) { fhmos_bg.maskoff[i] = 0; } else { fhmos_bg.maskoff[i] = 1; } //sts_mask_bitmap[(uint8_t)(i/8)] |= (fhmos_bg.maskoff[i])<<(7-i%8); // sts_mask_bitmap[(uint8_t)(i/8)] |= (fhmos_bg.maskoff[i])<<(7-i%8); // 2025-JAN-03 update // if (i%8==0) APP_LOG(TS_OFF, VLEVEL_M, "\r\n"); APP_LOG(TS_OFF, VLEVEL_L, "|%3d ", fhmos_bg.h2cm[i]*2); } else { fhmos_bg.h2cm[i] = 0; APP_LOG(TS_OFF, VLEVEL_L, "|%3d ", fhmos_bg.h2cm[i]); } } APP_LOG(TS_OFF, VLEVEL_L, "\r\n Thresholds: Headlevel=%4d cm, Gesture Mask off=%d cm, Min body height=%d cm\r\n", (uint8_t)fhmos_cfg.th_head_level_height_cm, (uint8_t)fhmos_cfg.th_gesture_mask_off_height_cm, (uint8_t)(fhmos_cfg.th_fall_body_min_height_cm)); APP_LOG(TS_OFF, VLEVEL_L, "\r\n\n ------- Mask off matrix before remove edge \r\n"); for (i=0; i<64; i++) { sts_mask_bitmap[(uint8_t)(i/8)] |= (fhmos_bg.maskoff[i])<<(7-i%8); // 2025-JAN-03 update if (i%8==0) APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); APP_LOG(TS_OFF, VLEVEL_L, "|%d ", (uint8_t)fhmos_bg.maskoff[i]); } for (i=0; i<34; i++) { fhmos_bg.maskoff[rio_edge[i]] = 1; } APP_LOG(TS_OFF, VLEVEL_L, "\r\n\n ------- After Remove Edge \r\n"); for (i=0; i<64; i++) { sts_mask_bitmap[(uint8_t)(i/8)] |= (fhmos_bg.maskoff[i])<<(7-i%8); // 2025-JAN-03 update if (i%8==0) APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); APP_LOG(TS_OFF, VLEVEL_L, "|%d ",(uint8_t)fhmos_bg.maskoff[i]); } APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); APP_LOG(TS_OFF, VLEVEL_M, "\r\nmask bitmap \r\n"); for (i=0; i<8; i++) APP_LOG(TS_OFF, VLEVEL_L, "%02X\r\n",sts_mask_bitmap[i]); #endif } void sts_generate_fall_gesture_map(void) { uint8_t i=0,head_xy=0, h4=0x0, l4=0x0; uint32_t range_distance =0, sts_fall_head_position=8000; // assume max range 8000 mm, actual 4500mm for (i=0; i<64; i++) { fhmos_gesture.h2cm[i] = 0; fhmos_gesture.maskoff[i] = 0; } fhmos_gesture.head_level =2000; // 1000 mm fhmos_gesture.head_xy = 28; //center of FOV for (i=0;i<8;i++) fhmos_gesture_bitmap[i]=0x0; for (uint8_t i = 0; i < 64; i++) { if ((Result.ZoneResult[i].NumberOfTargets > 0) && (fhmos_bg.maskoff[i]==0)) { range_distance = (uint32_t)Result.ZoneResult[i].Distance[0]; if (range_distance < sts_fall_head_position) { sts_fall_head_position = range_distance; // simply find out the head level head_xy = i; // head x, y coordination in 8x8 matrix } fhmos_gesture.h2cm[i] = (uint8_t)abs(sts_sensor_install_height - range_distance)/20; //fhmos_gesture.maskoff[i] = ((2*fhmos_gesture.h2cm[i])< fhmos_cfg.th_gesture_mask_off_height_cm)? 0:1; fhmos_gesture.maskoff[i] = (fhmos_gesture.h2cm[i]==0)? 0:1; fhmos_gesture_bitmap[(uint8_t)(i/8)] |= (fhmos_gesture.maskoff[i])<<(7 - i%8); // 2025-JAN-03 UPDATE // debug // if (i%8==0) printf("\r\n"); // printf("|%4ld %4d ", range_distance, fhmos_gesture.h2cm[i]); } else { fhmos_gesture.h2cm[i] = 0; } if (i%8==0) APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); APP_LOG(TS_OFF, VLEVEL_L, "|%4d ", fhmos_gesture.h2cm[i]); } for (i = 0; i < 64; i++) { if (i%8==0) APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); APP_LOG(TS_OFF, VLEVEL_L, "|%d ", fhmos_gesture.maskoff[i]); } for (i=0; i<8; i++) APP_LOG(TS_OFF, VLEVEL_L, "%02x ",fhmos_gesture_bitmap[i]); APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); for (i = 0; i < 32; i++) { if (i%4 == 0) APP_LOG(TS_OFF, VLEVEL_L, "\r\n"); //h4 = MIN(fhmos_gesture.h2cm[2*i+0],sts_gesture_mask_cap_height_2cm)/8; //l4 = MIN(fhmos_gesture.h2cm[2*i+1],sts_gesture_mask_cap_height_2cm)/8; h4 = fhmos_gesture.h2cm[2*i+0]; l4 = fhmos_gesture.h2cm[2*i+1]; fhmos_gesture.cube[i] = ((h4&0x0f)<<4)|(l4&0x0f); APP_LOG(TS_OFF, VLEVEL_L, " [%2d_%2d]", (fhmos_gesture.cube[i]>>4)&0x0f, (fhmos_gesture.cube[i]&0x0f)); } fhmos_gesture.head_level = 0xff&((sts_sensor_install_height - sts_fall_head_position)/10); // head level from floor in CM fhmos_gesture.head_xy = head_xy&0xff; APP_LOG(TS_OFF, VLEVEL_L, "\r\n Head level =%4d cm, Head_xy=%d X:Y = %2d : %2d \r\n", fhmos_gesture.head_level, head_xy, (head_xy%8), (head_xy/8)); APP_LOG(TS_OFF, VLEVEL_L, "\r\n Thresholds: Headlevel=%4d cm, Gesture Mask off=%d cm, Min body height=%d cm\r\n", (uint8_t)fhmos_cfg.th_head_level_height_cm, (uint8_t)fhmos_cfg.th_gesture_mask_off_height_cm, (uint8_t)(fhmos_cfg.th_fall_body_min_height_cm)); #if 0 for (i=0; i<64; i++) { if (i%8==0) printf("\r\n"); printf("|%d ", (uint8_t)fhmos_gesture.maskoff[i]); } for (i=0; i<8; i++) printf("%02X\r\n",fhmos_gesture_bitmap[i]); #endif sts_fhmos_bitmap_pending = 1; APP_LOG(TS_OFF, VLEVEL_L, "\r\n Fall Gesture bitmap Generated\r\n"); } uint16_t MX_TOF_Ranging_Process(void) { #if (defined(STS_P2)||defined(STS_T6)) uint16_t range_distance=0; uint8_t range_mode = 2; //STS_TOF_LONG_RANGE; STS_TOF_VL53LX_Range_Process(range_mode, &range_distance); APP_LOG(TS_OFF, VLEVEL_M, "\n VL53L1 Range distance =%u mm \n\r", (uint16_t)range_distance); return (uint16_t) range_distance; #elif defined(L8) uint32_t range_distance=0; //uint8_t center_roi[4] = {27,28,35,36}; if (ToF_EventDetected != 0) { ToF_EventDetected = 0; status = VL53L8A1_RANGING_SENSOR_GetDistance(VL53L8A1_DEV_CENTER, &Result); if (status == BSP_ERROR_NONE) { range_distance= STS_Get_Center_Range_Distance(&Result); } else { APP_LOG(TS_OFF, VLEVEL_M, "\r\n status=%d, x \r\n", status); return 0; } } return (uint16_t) range_distance; #endif } void MX_TOF_Process(void) { /* USER CODE BEGIN TOF_Process_PreTreatment */ /* USER CODE END TOF_Process_PreTreatment */ //APP_LOG(TS_OFF,VLEVEL_L,"\r\n# MX TOF Process... #\r\n"); //STS_TOF_VL53LX_Range_Process(); //STS_TOF_VL53LX_PeopleCounting_Process(); //sts_tof_vl53lx_peoplecount_subprocess(); #ifdef L8 MX_53L8A1_ThresholdDetection_Process(); //STS_TOF_L8_Process(); #endif /* USER CODE BEGIN TOF_Process_PostTreatment */ /* USER CODE END TOF_Process_PostTreatment */ } #ifdef L8 void STS_TOF_L8_Init(void) { MX_53L8A1_ThresholdDetection_Init(); MX_53L8A1_ThresholdDetection_Process(); } void STS_TOF_L8_Process(void) { //while (1) { /* interrupt mode */ if (ToF_EventDetected != 0) { ToF_EventDetected = 0; #if 1 if (STS_Status_Door_Close == sts_hall1_read) { status = VL53L8A1_RANGING_SENSOR_GetDistance(VL53L8A1_DEV_CENTER, &Result); if (status == BSP_ERROR_NONE) { print_result(&Result); } else { APP_LOG(TS_OFF, VLEVEL_M, "\r\n x \r\n"); } } #endif } } } void STS_TOF_L8_Reconfig(void) { MX_53L8A1_ThresholdDetection_ConfigIT(sts_low_threshold, sts_high_threshold); } /* VL53L8A1 */ static void MX_53L8A1_ThresholdDetection_Init(void) { /* Initialize Virtual COM Port */ //BSP_COM_Init(COM1); MX_USART2_UART_Init(); /* Sensor reset */ HAL_GPIO_WritePin(VL53L8A1_PWR_EN_C_PORT, VL53L8A1_PWR_EN_C_PIN, GPIO_PIN_RESET); HAL_Delay(2); HAL_GPIO_WritePin(VL53L8A1_PWR_EN_C_PORT, VL53L8A1_PWR_EN_C_PIN, GPIO_PIN_SET); HAL_Delay(2); HAL_GPIO_WritePin(VL53L8A1_LPn_C_PORT, VL53L8A1_LPn_C_PIN, GPIO_PIN_RESET); HAL_Delay(2); HAL_GPIO_WritePin(VL53L8A1_LPn_C_PORT, VL53L8A1_LPn_C_PIN, GPIO_PIN_SET); HAL_Delay(2); status = VL53L8A1_RANGING_SENSOR_Init(VL53L8A1_DEV_CENTER); if (status != BSP_ERROR_NONE) { printf("VL53L8A1_RANGING_SENSOR_Init failed\n"); //while (1); } } void MX_53L8A1_ThresholdDetection_ConfigIT(uint32_t low_threshold, uint32_t high_threshold) { RANGING_SENSOR_ITConfig_t ITConfig; status = VL53L8A1_RANGING_SENSOR_Stop(VL53L8A1_DEV_CENTER); /* threshold parameters */ ITConfig.Criteria = RS_IT_IN_WINDOW; ITConfig.LowThreshold = low_threshold; /* mm */ ITConfig.HighThreshold = high_threshold; /* mm */ VL53L8A1_RANGING_SENSOR_ConfigIT(VL53L8A1_DEV_CENTER, &ITConfig); status = VL53L8A1_RANGING_SENSOR_Start(VL53L8A1_DEV_CENTER, RS_MODE_ASYNC_CONTINUOUS); if (status != BSP_ERROR_NONE) { printf("VL53L8A1_RANGING_SENSOR_Start failed\n"); while (1); } } static void MX_53L8A1_ThresholdDetection_Process(void) { uint32_t Id; RANGING_SENSOR_ITConfig_t ITConfig; VL53L8A1_RANGING_SENSOR_ReadID(VL53L8A1_DEV_CENTER, &Id); VL53L8A1_RANGING_SENSOR_GetCapabilities(VL53L8A1_DEV_CENTER, &Cap); Profile.RangingProfile = RS_PROFILE_8x8_AUTONOMOUS; Profile.TimingBudget = TIMING_BUDGET; Profile.Frequency = RANGING_FREQUENCY; /* Ranging frequency Hz (shall be consistent with TimingBudget value) */ Profile.EnableAmbient = 0; /* Enable: 1, Disable: 0 */ Profile.EnableSignal = 0; /* Enable: 1, Disable: 0 */ /* set the profile if different from default one */ VL53L8A1_RANGING_SENSOR_ConfigProfile(VL53L8A1_DEV_CENTER, &Profile); /* threshold parameters */ ITConfig.Criteria = RS_IT_IN_WINDOW; //ITConfig.LowThreshold = LOW_THRESHOLD; /* mm */ //ITConfig.HighThreshold = HIGH_THRESHOLD; /* mm */ sts_high_threshold = sts_sensor_install_height; sts_low_threshold = sts_high_threshold - 1400; APP_LOG(TS_OFF, VLEVEL_M, "\r\n Threshold High=%4d, Low=%4d \r\n", sts_high_threshold, sts_low_threshold); ITConfig.LowThreshold = sts_low_threshold; ITConfig.HighThreshold = sts_high_threshold; VL53L8A1_RANGING_SENSOR_ConfigIT(VL53L8A1_DEV_CENTER, &ITConfig); status = VL53L8A1_RANGING_SENSOR_Start(VL53L8A1_DEV_CENTER, RS_MODE_ASYNC_CONTINUOUS); if (status != BSP_ERROR_NONE) { printf("VL53L8A1_RANGING_SENSOR_Start failed\n"); while (1); } #if 0 printf("\033[2H\033[2J"); printf("53L8A1 Threshold Detection demo application\n\r"); printf("-------------------------------------------\n\r"); printf("please put a target between %d and %d millimeters from the sensor\n\r", LOW_THRESHOLD, HIGH_THRESHOLD); #endif #if 0 while (1) { /* interrupt mode */ if (ToF_EventDetected != 0) { ToF_EventDetected = 0; status = VL53L8A1_RANGING_SENSOR_GetDistance(VL53L8A1_DEV_CENTER, &Result); if (status == BSP_ERROR_NONE) { print_result(&Result); } } } #endif } uint32_t STS_Get_Center_Range_Distance(RANGING_SENSOR_Result_t *Result) { uint32_t center_range_distance=0, sumd=0; uint8_t center_roi[4] = {27,28,35,36}, valid_cnt=0; for (uint8_t i=0; i<4; i++) { //APP_LOG(TS_OFF, VLEVEL_M, "\r\n ROI#=%d Target=%d, Distance=%d \r\n", center_roi[i], Result->ZoneResult[center_roi[i]].NumberOfTargets, Result->ZoneResult[center_roi[i]].Distance[0]); if (Result->ZoneResult[center_roi[i]].NumberOfTargets > 0) { sumd += (uint32_t)(Result->ZoneResult[center_roi[i]].Distance[0]); valid_cnt ++; } //APP_LOG(TS_OFF, VLEVEL_M, "\r\n CNT=%d, SUM Distance=%d \r\n", valid_cnt, sumd); } if (valid_cnt > 2) center_range_distance = sumd/valid_cnt; //APP_LOG(TS_OFF, VLEVEL_M, "\r\n Center Range Average=%d mm\r\n", center_range_distance); return center_range_distance; } static void print_result(RANGING_SENSOR_Result_t *Result) { int8_t i; uint16_t head_distance_from_ceiling=8000; //printf("%c[2H", 27); /* clear screen */ uint16_t factor1_floor_level_from_ceiling = sts_sensor_install_height; // 50mm min body height uint16_t factor2_head_level_from_floor = (10*fhmos_cfg.th_head_level_height_cm); //(sts_sensor_install_height - 10*fhmos_cfg.th_head_level_height_cm); uint16_t head_height_level_from_floor=0; sts_head_level_low = 0; /* //APP_LOG(TS_OFF, VLEVEL_M, "\r\n Eliminate edge \r\n"); uint8_t rio_edge[36]={0,1,2,3,4,5,6,7,8,15,16,23,24,31,32,39,40,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63}; for (i=0; i<36; i++) fhmos_bg.maskoff[rio_edge[i]] = 1; */ for (i=0; i<64; i++) { if (0 == fhmos_bg.maskoff[i]) // only within the non-mask-off blocks { // Result.ZoneResult[i].NumberOfTargets > 0) if ((Result->ZoneResult[i].Distance[0] < head_distance_from_ceiling)&& (Result->ZoneResult[i].NumberOfTargets > 0)) head_distance_from_ceiling = Result->ZoneResult[i].Distance[0]; // find out the min_distance or the highest position level } } head_height_level_from_floor = MIN(1400, abs(sts_sensor_install_height - head_distance_from_ceiling)); APP_LOG(TS_OFF, VLEVEL_L, "\r\n Updown distance=%d cm, Floor Up head =%d cm\r\n", head_distance_from_ceiling/10, head_height_level_from_floor/10); /* state tree */ if ((head_distance_from_ceiling <= factor1_floor_level_from_ceiling) && ((head_height_level_from_floor) <= (10*fhmos_cfg.th_head_level_height_cm)) && (head_height_level_from_floor >= (10*fhmos_cfg.th_fall_body_min_height_cm ))) { to_confirm ++; if (to_confirm >=3 ) { sts_head_level_low = 1; // to_confirm = 0; } } else if ((head_distance_from_ceiling < (factor2_head_level_from_floor - 150))) // TODO XXX 50mm gap to avoid flapping back and forth { sts_head_level_low = 0; sts_fhmos_bitmap_pending = 0; to_confirm = 0; } OnSensorL8AStateChanged(); } #endif #ifdef L8 /* Represent Color Value Definition * Fall Blue 0 The Room is not occupied by Human * Green 1 Human in room in normal level (head above 700 mmH from floor level) * Yellow 2 Human in low level position for short time potential (Head below 700mmH from floor level * Red 3 Human in low level position with time longer than threshold, e.g. 4 mins.(Head below 700 mm from Floor level * Human * Movement Blue 0 The Room is not occupied by Human * Green 1 The room is occupied by human, with movement * Yellow 2 The room is occupied by human, with motionless for short period of time (???) * Red 3 The room is occupied by human, with motionless for a period of time longer than threshold period, e.g. 5min. * * Occupancy Blue 0 The room is not occupied by Human * Green 1 The room is occupied by human, with movement * Red 2 The room is occupied by human, for a short period of time (???) * * # height is configurable from 10cm to 90 cm, with interval of 1cm * # the period of time is configurable, from 1 min to 30 mins, with 15 sec as interval * * sts_height_threshold (10cm --90 cm, 1cm interval) (10cm=10, 90cm=90, "P11T1XXX" * sts_fall_duration_threshold_potential (1 min to 30 min. 15 sec interval, 1800 sec. 1800=15*120, 60=15*4) "P11T2XXX" * sts_fall_duration_threshold_confirm (1 min to 30 min. 15 sec interval, 1800 sec. 1800=15*120, 60=15*4) "P11T3XXX" * * sts_motionless_threshold_short (1 min to 30 min. 15 sec interval, 1800 sec. 1800=15*120, 60=15*4) "P11T4XXX" * sts_motionless_threshold_long (1 min to 30 min. 15 sec interval, 1800 sec. 1800=15*120, 60=15*4) "P11T5XXX" * * sts_occupancy_threshold_overstay (1 min to 30 min. 15 sec interval, 1800 sec. 1800=15*120, 60=15*4) "P11T6XXX" * */ void STS_FHMOS_sensor_read(sts_fhmos_sensor_data_t *sts_data) { //uint8_t fhmos_fall=0, fhmos_human_movement=0, fhmos_occupancy=0, fhmos_sos_alarm=0; sts_data->state_occupancy = fhmos_data.state_occupancy; sts_data->state_fall = fhmos_data.state_fall; sts_data->state_human_movement = fhmos_data.state_human_movement; sts_data->state_sos_alarm = fhmos_data.state_sos_alarm; sts_data->time_stamp_fall_confirmed = fhmos_data.time_stamp_fall_confirmed; sts_data->time_stamp_motionless_confirmed = fhmos_data.time_stamp_motionless_confirmed; sts_data->time_stamp_overstay_confirmed = fhmos_data.time_stamp_overstay_confirmed; sts_data->time_stamp_fall_released = fhmos_data.time_stamp_fall_released; sts_data->time_stamp_overstay_released = fhmos_data.time_stamp_overstay_released; sts_data->time_stamp_motionless_released= fhmos_data.time_stamp_motionless_released; sts_data->state_fall_released = fhmos_data.state_fall_released; sts_data->state_occupancy_released = fhmos_data.state_occupancy_released; sts_data->state_human_movement_released = fhmos_data.state_human_movement_released; sts_data->lamp_bar_color = sts_lamp_bar_color; sts_data->state_hall_1 = sts_hall1_read; sts_data->state_hall_2 = sts_hall2_read; sts_data->state_PIR = sts_pir_read;; //sts_data->state_PIR = fhmos_data.state_PIR; } #endif #if defined(STS_P2)||defined(STS_T6)||defined(L8) uint8_t IsInterruptDetected(uint16_t dev) { // To be filled with customer HW. This function should // return 1 when an interrupt is raised by the ToF on GPIO1 pin (pin7) if (ToF_EventDetected ) { APP_LOG(TS_OFF, VLEVEL_M,"############### TOF EVENT DETECTED \r\n"); ToF_EventDetected =0; return 1; } else { return 0; } } #endif #if (defined(STS_P2)||defined(STS_T6)) void STS_TOF_VL53LX_Range_Process(uint8_t range_mode, uint16_t *range_distance) { //uint8_t vl53lx_model = STS_TOF_VL53L1X; //uint8_t range_mode = STS_TOF_SHORT_RANGE; uint16_t i_distance_measured = 0; uint16_t i_distance_threshold_mm = 800; uint16_t i_inter_measurement_ms=100, i_macro_timing=33; uint16_t i_roi_width=16, i_sigma_mm=30, i_signal_kcps=2000; switch (range_mode) { case STS_TOF_SHORT_RANGE: // STS ---002 for short distance /* Example for robust and short distance measurements. Max distance reduced * but very low number of false-positives */ //status |= VL53L1X_ULP_SetSigmaThreshold(dev, 30); //status |= VL53L1X_ULP_SetSignalThreshold(dev, 2000); i_sigma_mm = 30; //increase this for longer distance, reduce for shorter distance i_signal_kcps = 2000; // 1000- 6000 kcps i_distance_threshold_mm = 900; i_inter_measurement_ms = 100; // 100 - 1000 ms i_macro_timing = 100; //i_roi_width = 16; break; case STS_TOF_LONG_RANGE: // STS --- 003 for long range /* Relax some limits. Be careful, it can create false-positives !*/ //status |= VL53L1X_ULP_SetSigmaThreshold(dev, 60); //status |= VL53L1X_ULP_SetSignalThreshold(dev, 1200); i_sigma_mm = 85; // increase this for longer distance, reduce for short distance i_signal_kcps = 1000; // 1000- 6000 kcps i_distance_threshold_mm = 4000; //4000; i_inter_measurement_ms = 200; // 100 - 1000 ms i_macro_timing = 30; // 1 - 100 ms i_roi_width = 8; break; case STS_TOF_LOW_POWER_RANGE: // STS---001 for ultra low power /* Reduce the macro timing to minimum. This is equivalent as reducing the integration time */ //status = VL53L1X_ULP_SetMacroTiming(dev, 1); i_distance_threshold_mm = 4000; i_inter_measurement_ms = 100; // 100 - 1000 ms i_macro_timing = 1; /* Reduce at maximum the SPADS */ //status = VL53L1X_ULP_SetROI(dev, 4); i_roi_width = 4; break; default: break; } sts_vl53lx_ranging(&i_distance_measured, range_mode, i_distance_threshold_mm, i_inter_measurement_ms,i_macro_timing, i_roi_width, i_sigma_mm, i_signal_kcps); APP_LOG(TS_OFF, VLEVEL_M, "I_DISTANCE_MEASURED = %d \n", i_distance_measured); *range_distance = i_distance_measured; } void BSP_PB_Callback(Button_TypeDef Button) { //PushButtonDetected = 1; } uint8_t sts_vl53lx_ranging(uint16_t *ranged_distance, uint8_t range_mode, uint16_t distance_threshold_mm, uint16_t inter_measurement_ms, uint16_t macro_timing, uint16_t roi_width, uint16_t sigma_mm, uint16_t signal_kcps) { uint8_t status=0, dev=0x52; uint16_t estimated_distance_mm=0; status = VL53L1X_SensorInit(dev); status += VL53L1X_SetDistanceMode(dev, 2); /* 1=short, 2=long, DISTANCE_MODE */ status += VL53L1X_SetTimingBudgetInMs(dev, 100); /* TIMING_BUDGET, in ms possible values [15, 20, 50, 100, 200, 500] */ status += VL53L1X_SetInterMeasurementInMs(dev, 50); status += VL53L1X_SetROI(dev, 8, 8); /* minimum ROI 4,4 */ if (status != 0) { APP_LOG(TS_OFF, VLEVEL_L,"Initialization or configuration of the device\n"); return (-1); } status = VL53L1X_StartRanging(dev); /* This function has to be called to enable the ranging */ if (status != 0) { APP_LOG(TS_OFF, VLEVEL_L,"Start Range failed\n"); return (-1); } status = VL53L1X_GetDistance(dev, &estimated_distance_mm); APP_LOG(TS_OFF,VLEVEL_M,"Target detected! Distance =%d mm \r\n", estimated_distance_mm ); *ranged_distance = estimated_distance_mm; status = VL53L1X_StopRanging(dev); APP_LOG(TS_OFF,VLEVEL_M,"End of VL53L1X Ranging Process\n"); return status; #if 0 /*********************************/ /* VL53L1X ranging variables */ /*********************************/ uint8_t status, loop; uint8_t dev; //uint16_t sensor_id=0; uint8_t measurement_status; uint16_t estimated_distance_mm, r_signal_kcps, r_sigma_mm, r_ambient_kcps; /*********************************/ /* Customer platform */ /*********************************/ /* Default VL53L1X Ultra Low Power I2C address */ dev = 0x52; /* (Optional) Change I2C address */ // status = VL53L1X_ULP_SetI2CAddress(dev, 0x52); // dev = 0x20; /*********************************/ /* Power on sensor and init */ /*********************************/ APP_LOG(TS_OFF,VLEVEL_L,"Range Mode =%d \r\n",range_mode); /* (Optional) Check if there is a VL53L1X sensor connected */ status = VL53L1X_GetSensorId(dev, &sensor_id); APP_LOG(TS_OFF,VLEVEL_L,"VL53L1X address =%X\r\n",sensor_id ); if(status || (sensor_id != 0xEACC)) { APP_LOG(TS_OFF,VLEVEL_L,"VL53L1X not detected at requested address\n"); return status; } /* (Mandatory) Init VL53L1X sensor */ status = VL53L1X_SensorInit(dev); if(status) { APP_LOG(TS_OFF,VLEVEL_L,"VL53L1X ultra low power Loading failed\n"); //HAL_Delay(100); return status; } APP_LOG(TS_OFF,VLEVEL_L,"VL53L1X ultra low power ready ! \r\n"); /*********************************/ /* Sensor configuration */ /*********************************/ /* (Optional) Program sensor to raise an interrupt ONLY below 300mm */ //status = VL53L1X_SetInterruptConfiguration(dev, distance_threshold_mm, 1); //i_distance_threshold_mm status = VL53L1X_SetDistanceThreshold(dev, distance_threshold_mm,distance_threshold_mm, 0,1); //i_distance_threshold_mm /* (Optional) Program a 10Hz ranging frequency */ status = VL53L1X_SetInterMeasurementInMs(dev, inter_measurement_ms); // range_interval_ms /* Increase the macro timing. This is equivalent as increasing the integration time */ //status = VL53L1X_SetMacroTiming(dev, macro_timing); // micro_timing_ms status = VL53L1X_SetTimingBudgetInMs(dev, macro_timing); /* Enable all the SPADS */ status = VL53L1X_SetROI(dev, roi_width, roi_width); // SPADS { 1 -- 16 } if(range_mode != STS_TOF_LOW_POWER_RANGE) { /* Example for robust and short distance measurements. Max distance reduced * but very low number of false-positives */ status |= VL53L1X_SetSigmaThreshold(dev, sigma_mm); status |= VL53L1X_SetSignalThreshold(dev, signal_kcps); } /*********************************/ /* Ranging loop */ /*********************************/ status = VL53L1X_StartRanging(dev); if(status) { APP_LOG(TS_OFF,VLEVEL_L,"VL53L1X_ULP_StartRanging failed with status %u\n", status); return status; } APP_LOG(TS_OFF,VLEVEL_L,"Ranging started. Put your hand close to the sensor to generate an interrupt...\n"); loop = 0; while(loop < 20) { /* Use this external function to detect when a hardware interrupt is generated on PIN 7 (GPIO1). It means that a new measurement is ready. */ if(IsInterruptDetected(dev)) { /* (Mandatory) Clear HW interrupt to restart measurements */ VL53L1X_ClearInterrupt(dev); /* Dump debug data */ #if 0 status = VL53L1X_DumpDebugData(dev, &measurement_status, &estimated_distance_mm, &r_sigma_mm, &r_signal_kcps, &r_ambient_kcps); *ranged_distance = estimated_distance_mm; APP_LOG(TS_OFF,VLEVEL_L,"Target detected! Interrupt raised by sensor, Distance =%d mm \r\n", estimated_distance_mm ); #endif status = VL53L1X_GetDistance(dev, &estimated_distance_mm); APP_LOG(TS_OFF,VLEVEL_L,"Target detected! Interrupt raised by sensor, Distance =%d mm \r\n", estimated_distance_mm ); loop++; } } status = VL53L1X_StopRanging(dev); APP_LOG(TS_OFF,VLEVEL_L,"End of VL53L1X ultra low power demo\n"); return status; #endif } #endif #ifdef __cplusplus } #endif