基于小熊派光強傳感器BH1750實踐(multi_timer+狀態機工程應用)

      網友投稿 1439 2022-05-25

      本實踐案例基于小熊派開發板:

      實踐光強傳感器的開發,我們需要帶上一個擴展模塊:E53_SC1,如下圖所示,最終連接的效果:

      再來看看這個拓展板以及主板上對應的硬件接口,后面我們才能夠去配置相應的硬件管腳,達到驅動使用的目的:

      轉接板E53_SC1在主板上的電路原理圖:

      BH1750光強傳感器簡介

      BH1750是一種用于兩線式串行總線接口的數字型光強度傳感器集成電路。這種集成電路可以根據收集的光線強度數據來調整液晶或者鍵盤背景的的亮度,利用它的高分辨率可以探測較大范圍的光強度變化。(1lx-65535lx)

      要控制這個傳感器,當然要了解傳感器支持的協議,以及一些指令,這是一個基于I2C接口的傳感器。

      I2C是(Inter-Integrated Circuit)的英文縮寫,是Philips公司開發的一個通信協議,只有兩根線(SDA/SCL)是用來通信的。

      BH1750支持的命令:

      BH1750從機地址計算:

      根據文檔提示,我們了解到光強傳感器的從機地址是0100011

      當主機向從機發送寫命令時為:

      0100011(從機地址:7位) 0(寫數據位:1位) ===> 0x46

      當主機向從機發送讀命令時為:

      0100011(從機地址:7位) 1(寫數據位:1位) ===> 0x47

      工程實踐:stm32cubMx

      1、芯片選型,這里選擇stm32l431rctx

      2、配置rcc時鐘以及串行調試接口

      3、配置調試LED

      4、配置串口調試

      方便根據調試信息查看程序執行流程(默認即可)。

      5、配置I2C1(PC6/PC7位于I2C1)

      默認即可。

      6、生成Keil5基礎工程

      基于小熊派光強傳感器BH1750實踐(multi_timer+狀態機工程應用)

      實際開發建議硬件外設分模塊,這樣看起來不要把所有的生成全部都擠到main.c里面去了,這點讓我非常討厭,所以生成工程時候習慣點擊設置以下這一項:

      接下來點擊生成代碼:

      工程實踐:編寫代碼

      1、將multi_timer添加到keil5工程

      2、創建一個Package目錄,將multi_timer的程序文件添加進來

      3、編寫代碼

      由于篇幅限制,只看我自己代碼添加的位置:

      usart.h

      添加重定向打印

      /* USER CODE BEGIN 1 */

      int fputc(int ch, FILE *file)

      {

      return HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, 1000);

      }

      /* USER CODE END 1 */

      stm32l4xx_it.c

      添加multi_timer計數

      /**

      * @brief This function handles System tick timer.

      */

      void SysTick_Handler(void)

      {

      /* USER CODE BEGIN SysTick_IRQn 0 */

      timer_ticks();

      /* USER CODE END SysTick_IRQn 0 */

      HAL_IncTick();

      /* USER CODE BEGIN SysTick_IRQn 1 */

      /* USER CODE END SysTick_IRQn 1 */

      }

      main.h

      添加相應的頭文件

      /* Private includes ----------------------------------------------------------*/

      /* USER CODE BEGIN Includes */

      #include

      #include

      #include

      #include "multi_timer.h"

      /* USER CODE END Includes */

      main.c

      在程序中宏定義相應的值

      /* Private define ------------------------------------------------------------*/

      /* USER CODE BEGIN PD */

      /*接收光強超時值*/

      #define TIMEOUT 200

      /*光強值檢測閾值*/

      #define LIGHT_SENSOR_THREHOLD 30

      /*mode:模式選擇*/

      //連續高分辨率模式精度 1 lux

      #define LUX_1_MODE ? 0x10

      //連續高分辨率模式2精度 0.5 lux

      #define LUX_0_5_MODE 0x11

      //低分辨率模式

      #define LUX_LOW_MODE 0x13

      #define WRITE_ADDRESS 0x46 //0100 011 ?0

      #define READ_ADDRESS ?0x47 //0100 011 ?1

      /* USER CODE END PD */

      光強采集結構體

      typedef struct light_sensor

      {

      /*光強值*/

      int Lux ;

      /*multi_timer定時器句柄*/

      Timer timer1 ;

      /*定時器計數值*/

      uint16_t Timer_Count ;

      /*是否采集完成標志*/

      uint8_t ? Conver_completed ;

      /*定時回調*/

      void (*timeout_cb)(void);

      } light_sensor_TypeDef;

      light_sensor_TypeDef lsensor ;

      定時回調函數以及結構體初始化

      /*定時器回調函數*/

      void timer1_callback(void)

      {

      ++lsensor.Timer_Count;

      }

      /*初始化參數*/

      void Init_BH750(void)

      {

      lsensor.Lux = 0 ;

      lsensor.Timer_Count = 0 ;

      lsensor.Conver_completed = 0 ;

      lsensor.timeout_cb = timer1_callback ;

      }

      這里的讀光強沒有用小熊派例程里直接延時然后讀取的方法,小熊派的讀光強函數是這么寫的:

      而我是這么寫的:

      /*讀取光強*/

      void ReadBH1750(uint8_t mode)

      {

      float lux = 0;

      uint8_t ReadData[2] = {0};

      static uint8_t Sensor_Status = 0 ;

      /*讀取光強流程*/

      switch(Sensor_Status)

      {

      /*1、發送檢測光強模式的指令*/

      case 0:

      if(HAL_OK != HAL_I2C_Master_Transmit(&hi2c1, WRITE_ADDRESS, (uint8_t *)&mode, 1, 0xff))

      return ?;

      /*切換為讀地址狀態*/

      Sensor_Status = 1 ;

      lsensor.Timer_Count = 0 ;

      lsensor.Conver_completed = 0 ;

      break ;

      case 1:

      /*

      2、發送命令后延時200ms等待讀取

      定時200ms,判斷是否已經到了

      這里相當于取代了延時等待,不占用CPU

      */

      if(TIMEOUT == lsensor.Timer_Count)

      {

      lsensor.Timer_Count = 0 ;

      //3、開始讀取光強,發送讀光強指令

      if(HAL_OK == HAL_I2C_Master_Receive(&hi2c1, READ_ADDRESS, ReadData, 2, 0xff))

      {

      lux=(float)((ReadData[0]<<8)|ReadData[1]);

      lux=(double)lux/1.2;

      lsensor.Lux = (int)lux ;

      /*4、轉換完成*/

      lsensor.Conver_completed = 1 ;

      Sensor_Status = 0 ; /*切換為寫地址狀態*/

      }

      }

      break ;

      default:

      break;

      }

      }

      這里利用了switch case+multi_timer產生一個200ms的定時計數,通過狀態1流程判斷計數器是否到達設定值,從而達到延時的效果,這段程序在主程序的while循環中幾乎不會占用CPU,非常高效!

      主程序邏輯:

      /**

      * @brief ?The application entry point.

      * @retval int

      */

      int main(void)

      {

      /* USER CODE BEGIN 1 */

      /* USER CODE END 1 */

      /* MCU Configuration--------------------------------------------------------*/

      /* Reset of all peripherals, Initializes the Flash interface and the Systick. */

      HAL_Init();

      /* USER CODE BEGIN Init */

      /* USER CODE END Init */

      /* Configure the system clock */

      SystemClock_Config();

      /* USER CODE BEGIN SysInit */

      /* USER CODE END SysInit */

      /* Initialize all configured peripherals */

      MX_GPIO_Init();

      MX_I2C1_Init();

      MX_USART1_UART_Init();

      /* USER CODE BEGIN 2 */

      /*串口初始化后加這個延時,防止后面的printf打印亂碼*/

      HAL_Delay(200);

      printf("光強讀取測試實驗\n");

      Init_BH750();

      timer_init(&lsensor.timer1, lsensor.timeout_cb, 1, 1);

      timer_start(&lsensor.timer1);

      /* USER CODE END 2 */

      /* Infinite loop */

      /* USER CODE BEGIN WHILE */

      while (1)

      {

      /* USER CODE END WHILE */

      /* USER CODE BEGIN 3 */

      //采用連續高分辨率模式精度 1 lux

      ReadBH1750(LUX_1_MODE);

      if(1 == lsensor.Conver_completed) /*如果轉換完了才會執行*/

      {

      printf("當前光強值:%d\n", lsensor.Lux);

      //當光強值大于設定的光強值閾值,則關燈,否則開燈

      if(lsensor.Lux > LIGHT_SENSOR_THREHOLD)

      {

      HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_RESET);

      }

      else

      {

      HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_SET);

      }

      }

      timer_loop();

      }

      /* USER CODE END 3 */

      }

      運行結果:

      狀態機思想前變萬化,學會靈活應用則一定能夠寫出非常高效,簡單易維護的程序。

      實際例程下載:

      鏈接:https://pan.baidu.com/s/1GA29ua5yVfvGFfRCcS52LA

      提取碼:afe5

      復制這段內容后打開百度網盤手機App,操作更方便哦

      若覺得本次分享的文章對您有幫助,關注"嵌入式云IOT技術圈"并轉發分享,也是對我的支持。

      軟件開發 小熊派 IoT

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      上一篇:DevRun開發者沙龍 | 華為云AI開發者沙龍武漢專場
      下一篇:基于小熊派光強傳感器BH1750狀態機驅動項目再度升級(帶上位機曲線顯示)
      相關文章
      中文字幕在线亚洲精品| 亚洲日韩一区二区三区| 亚洲精品国产suv一区88| 亚洲自偷自拍另类图片二区| 亚洲av永久无码精品网站| 国产亚洲精品拍拍拍拍拍| 亚洲日韩中文字幕日韩在线| 亚洲Aⅴ无码一区二区二三区软件| 亚洲av无码专区亚洲av不卡| 亚洲精品无码中文久久字幕| 亚洲精品宾馆在线精品酒店| 亚洲AV无码专区国产乱码不卡| 亚洲欧美日韩综合久久久久| 亚洲丁香婷婷综合久久| WWW国产亚洲精品久久麻豆| 相泽南亚洲一区二区在线播放| 亚洲AV无码AV日韩AV网站| 亚洲欧美成人综合久久久| 亚洲狠狠色丁香婷婷综合| 极品色天使在线婷婷天堂亚洲| 色天使色婷婷在线影院亚洲| jizzjizz亚洲| 亚洲日本一区二区一本一道| 最新精品亚洲成a人在线观看| 亚洲欧洲无码AV电影在线观看 | 国产AV无码专区亚洲AVJULIA| 黑人精品videos亚洲人| 亚洲av无码一区二区三区网站| 亚洲高清国产AV拍精品青青草原| 亚洲AV人无码激艳猛片| 337p日本欧洲亚洲大胆精品555588| 亚洲日本视频在线观看| 亚洲人成在久久综合网站| 亚洲精品美女久久7777777| 亚洲欧美在线x视频| 精品亚洲一区二区三区在线观看| 亚洲人成色77777| 亚洲综合一区二区国产精品| 亚洲成a人片在线观看中文app| 亚洲自偷自偷在线成人网站传媒| MM1313亚洲精品无码久久|