采用ROHM原裝BH1750FVI芯片供電電源:3-5V,光照度范圍:0-65535lx傳感器內置16bitAD轉換器,直接數字輸出,省略復雜的計算,省略標定,不區分環境光源接近于視覺靈敏度的分光特性,可對廣泛的亮度進行1勒克斯的高精度測定。標準NXPICC通信協議,模塊內部包含通信電平轉換,可以與5V單片機io直接連接。
一、模塊來源
模塊實物展示:
資料下載鏈接:
https://pan.baidu.com/s/13bVmmj0eM22mT8pBusjIyQ?pwd=8889
資料提取碼:8889
工作電壓:3-5V
工作電流:200uA
探測范圍:1~65536 lx
模塊尺寸:32.6mm×15.2mm×11.6mm
輸出方式: IIC
管腳數量:5 Pin
以上信息見廠家資料文件
三、移植過程
我們的目標是將例程移植至CW32F030C8T6開發板上【能夠測量光照強度】。首先要獲取資料,查看數據手冊應如何實現讀取數據,再移植至我們的工程。
3.1查看資料
測量步驟:
模塊上電后,進入掉電模式,需要通過IIC發送Power On命令啟動。
模塊啟動之后通過IIC發送測量命令進行測量。
測量命令分有單次測量和連續測量,測量完畢之后又進入掉電模式。
各個命令的對應的值見下表。
我們使用到的有:
Power On(0x01):啟動模塊,讓其等待測量命令。
Continuously H-Resolution Mode(0X10):以1LX分辨率開始測量。測量時間一般為120ms(手冊推薦使用該命令)
One Time H-Resolution Mode(0X20):以1lx分辨率開始測量,測量時間通常為120ms。操作完成后,系統自動設置為”掉電”模式。
發送時序:
起始信號 -> 發送器件地址+寫 -> 等待模塊應答 -> 發送命令 -> 等待模塊應答 -> 停止信號。
讀取時序:
起始信號 -> 發送器件地址+讀 -> 等待模塊應答 -> 接收數據高8位 -> 主機發送應答 -> 接收數據低8位 -> 主機發送非應答 -> 停止信號。
讀取完成之后,將數據高低位整合再除以1.2即可得到光照強度數據。
3.2引腳選擇
模塊接線圖
3.3移植至工程
移植步驟中的導入.c和.h文件與【CW32模塊使用】DHT11溫濕度傳感器相同,只是將.c和.h文件更改為bsp_gy30.c與bsp_gy30.h。這里不再過多講述,移植完成后面修改相關代碼。
在文件bsp_gy30.c中,編寫如下代碼。
/* * Change Logs: * Date Author Notes * 2024-06-20 LCKFB-LP first version */ #include "bsp_gy30.h" #include "stdio.h" unsigned char BUF[8]; //接收數據緩存區 /****************************************************************** * 函 數 名 稱:GY30_GPIO_Init * 函 數 說 明:MLX90614的引腳初始化 * 函 數 形 參:無 * 函 數 返 回:無 * 作 者:LC * 備 注:無 ******************************************************************/ void GY30_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct; // GPIO初始化結構體 RCC_GY30_ENABLE(); // 使能GPIO時鐘 GPIO_InitStruct.Pins = GPIO_SDA|GPIO_SCL; // GPIO引腳 GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; // 開漏輸出 GPIO_InitStruct.Speed = GPIO_SPEED_HIGH; // 輸出速度高 GPIO_Init(PORT_GY30, &GPIO_InitStruct); // 初始化 } /****************************************************************** * 函 數 名 稱:IIC_Start * 函 數 說 明:IIC起始時序 * 函 數 形 參:無 * 函 數 返 回:無 * 作 者:LC * 備 注:無 ******************************************************************/ void IIC_Start(void) { SDA_OUT(); SDA(1); delay_us(5); SCL(1); delay_us(5); SDA(0); delay_us(5); SCL(0); delay_us(5); } /****************************************************************** * 函 數 名 稱:IIC_Stop * 函 數 說 明:IIC停止信號 * 函 數 形 參:無 * 函 數 返 回:無 * 作 者:LC * 備 注:無 ******************************************************************/ void IIC_Stop(void) { SDA_OUT(); SCL(0); SDA(0); SCL(1); delay_us(5); SDA(1); delay_us(5); } /****************************************************************** * 函 數 名 稱:IIC_Send_Ack * 函 數 說 明:主機發送應答或者非應答信號 * 函 數 形 參:0發送應答 1發送非應答 * 函 數 返 回:無 * 作 者:LC * 備 注:無 ******************************************************************/ void IIC_Send_Ack(unsigned char ack) { SDA_OUT(); SCL(0); SDA(0); delay_us(5); if(!ack) SDA(0); else SDA(1); SCL(1); delay_us(5); SCL(0); SDA(1); } /****************************************************************** * 函 數 名 稱:I2C_WaitAck * 函 數 說 明:等待從機應答 * 函 數 形 參:無 * 函 數 返 回:0有應答 1超時無應答 * 作 者:LC * 備 注:無 ******************************************************************/ unsigned char I2C_WaitAck(void) { char ack = 0; unsigned char ack_flag = 10; SCL(0); SDA(1); SDA_IN(); delay_us(5); SCL(1); delay_us(5); while( (SDA_GET()==1) && ( ack_flag ) ) { ack_flag--; delay_us(5); } if( ack_flag <= 0 ) { IIC_Stop(); return 1; } else { SCL(0); SDA_OUT(); } return ack; } /****************************************************************** * 函 數 名 稱:Send_Byte * 函 數 說 明:寫入一個字節 * 函 數 形 參:dat要寫人的數據 * 函 數 返 回:無 * 作 者:LC * 備 注:無 ******************************************************************/ void Send_Byte(uint8_t dat) { int i = 0; SDA_OUT(); SCL(0);//拉低時鐘開始數據傳輸 for( i = 0; i < 8; i++ ) { SDA( dat & 0x80 ); delay_us(1); SCL(1); delay_us(5); SCL(0); delay_us(5); dat<=1; } } /****************************************************************** * 函 數 名 稱:Read_Byte * 函 數 說 明:IIC讀時序 * 函 數 形 參:無 * 函 數 返 回:讀到的數據 * 作 者:LC * 備 注:無 ******************************************************************/ unsigned char Read_Byte(void) { unsigned char i,receive=0; SDA_IN();//SDA設置為輸入 for(i=0;i8;i++ ) { SCL(0); delay_us(5); SCL(1); delay_us(5); receive<=1; if( SDA_GET() ) { receive|=1; } delay_us(5); } SCL(0); return receive; } /****************************************************************** * 函 數 名 稱:Single_Write * 函 數 說 明:向BH1750寫入命令 * 函 數 形 參:REG_Address=寫入的命令 * 函 數 返 回:0寫入成功 1=器件地址錯誤(識別不到模塊) 2=命令錯誤 * 作 者:LC * 備 注:無 ******************************************************************/ char Single_Write_BH1750(uint8_t REG_Address) { IIC_Start(); //起始信號 Send_Byte(SlaveAddress); //發送設備地址+寫信號 if( I2C_WaitAck() != 0 )return 1; Send_Byte(REG_Address); //內部寄存器地址 if( I2C_WaitAck() != 0 )return 2; IIC_Stop(); //發送停止信號 return 0; } /****************************************************************** * 函 數 名 稱:Multiple_read_BH1750 * 函 數 說 明:讀取BH1750內部數據 * 函 數 形 參:無 * 函 數 返 回:光照度 單位:lx * 作 者:LC * 備 注:無 ******************************************************************/ float Multiple_read_BH1750(void) { uint16_t dis_data=0; uint8_t dat_buff[2]; IIC_Start(); //起始信號 Send_Byte(SlaveAddress+1); //發送設備地址+讀信號 I2C_WaitAck(); dat_buff[0] = Read_Byte(); //讀取高8位 IIC_Send_Ack(0); //回應ACK dat_buff[1] = Read_Byte(); //讀取低8位 IIC_Send_Ack(1); //回應NOACK IIC_Stop(); //停止信號 //合成數據,即光照數據 dis_data=( dat_buff[0] < 8 ) + dat_buff[1]; return ((float)dis_data/1.2f); } /****************************************************************** * 函 數 名 稱:GY30_Init * 函 數 說 明:初始化BH1750 * 函 數 形 參:無 * 函 數 返 回:無 * 作 者:LC * 備 注:無 ******************************************************************/ void GY30_Init(void) { GY30_GPIO_Init(); Single_Write_BH1750(0x01);//上電 }
在文件bsp_gy30.h中,編寫如下代碼。
/* * Change Logs: * Date Author Notes * 2024-06-20 LCKFB-LP first version */ #ifndef _BSP_GY30_H_ #define _BSP_GY30_H_ #include "board.h" //端口移植 #define RCC_GY30_ENABLE() __RCC_GPIOB_CLK_ENABLE() #define PORT_GY30 CW_GPIOB #define GPIO_SDA GPIO_PIN_8 #define GPIO_SCL GPIO_PIN_9 //設置SDA輸出模式 #define SDA_OUT() { GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.Pins = GPIO_SDA; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; GPIO_InitStruct.Speed = GPIO_SPEED_HIGH; GPIO_Init(PORT_GY30, &GPIO_InitStruct); } //設置SDA輸入模式 #define SDA_IN() { GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.Pins = GPIO_SDA; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Speed = GPIO_SPEED_HIGH; GPIO_Init(PORT_GY30, &GPIO_InitStruct); } //獲取SDA引腳的電平變化 #define SDA_GET() GPIO_ReadPin(PORT_GY30, GPIO_SDA) //SDA與SCL輸出 #define SDA(x) GPIO_WritePin(PORT_GY30, GPIO_SDA, (x?GPIO_Pin_SET:GPIO_Pin_RESET) ) #define SCL(x) GPIO_WritePin(PORT_GY30, GPIO_SCL, (x?GPIO_Pin_SET:GPIO_Pin_RESET) ) #define SlaveAddress 0x46 //定義器件在IIC總線中的從地址,根據ALT ADDRESS地址引腳不同修改 //ALT ADDRESS引腳接地時地址為0x46,接電源時地址為0xB8 float Multiple_read_BH1750(void); char Single_Write_BH1750(uint8_t REG_Address); void GY30_Init(void); #endif
四、移植驗證
在自己工程中的main主函數中,編寫如下。
/* * Change Logs: * Date Author Notes * 2024-06-20 LCKFB-LP first version */ #include "board.h" #include "stdio.h" #include "bsp_uart.h" #include "bsp_gy30.h" int32_t main(void) { board_init(); // 開發板初始化 uart1_init(115200); // 串口1波特率115200 GY30_Init(); printf("startrn"); while(1) { Single_Write_BH1750(0x10); // 連續高分辨率模式測量 delay_ms(180); //測量一般需要120ms float temp = Multiple_read_BH1750(); //讀出數據 printf("光照強度 = %.2f lxrn",temp); delay_ms(500); } }
移植現象:輸出當前光照強度。
模塊移植成功案例代碼:
鏈接:https://pan.baidu.com/s/1wmC7OwwJk2LB0tdP0Zx0GA?pwd=LCKF
提取碼:LCKF
審核編輯 黃宇
-
傳感器
+關注
關注
2552文章
51246瀏覽量
754855 -
CW32
+關注
關注
1文章
210瀏覽量
690
發布評論請先 登錄
相關推薦
評論