STM32模拟IIC与IIC四种实现数字光强采集模块GY30(标准库与HAL库)

2023-10-14 02:30

本文主要是介绍STM32模拟IIC与IIC四种实现数字光强采集模块GY30(标准库与HAL库),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

代码实现是的IIC通信,数据采集后在串口显示,方便大家实现二次开发

原件选择

GY-30 数字光强度介绍

BH1750芯片参数

引脚说明

 BH1750指令集

接线表设计

通过四种方式实现GY-30数据采集

1.标准库模拟IIC实现GY-30采集并串口1显示

 2.标准库IIC实现GY-30采集并串口1显示

3.HAL库模拟IIC实现GY-30采集并串口1显示

4.HAL库IIC实现GY-30采集并串口1显示

源码下载链接


代码实现是的IIC通信,数据采集后在串口显示,方便大家实现二次开发

原件选择

1.STM32F103

2.GY-30 数字光强度 光照传感器 BH1750FVI 

GY-30 数字光强度介绍

GY30简介
GY-30是一款内置ROHM-BH1750FLV芯片的数字光照强度模块,用于光照强度采集。
BH1750FVl是一种用于RC总线接口的数字环境光传感器LC。该芯片最适合于获取移动电话液晶显示器和按键背光功率的环境光数据。在高分辨率(1~65535 l×)范围内检测是可行的。

特性
IIC总线接口
光强数字转换器
16位分辩率(1~65535)
直接数字输出,省略复杂的计算,省略标定
不区分环境光源,接近于视觉灵敏度的分光特性
应用
手机、数码相机、车载导航,PDA、LCD显示等。


GY-30模块是一款基于IIC通信的16bit的数字型传感器。模块主要是以BH1750数字型光强感应芯片为核心及一些外围驱动电路。模块整体电路如图:

其中C1、C2 为电源滤波电容,R1、R3 为 I2C 上拉电阻,ADDR 是 I2C 通讯时设备地址的选择,即接电源或接地时,读操作、写操作的指令有所区别,如下图。一般为接地,即写操作指令为0x46,读操作指令为0x47。

BH1750芯片参数

BH1750是一款内部集成了光电转换、ADC转换、IIC信号转换等电路的芯片,省去了复杂信号处理电路,即能保持良好的稳定型又节省空间。BH1750内部简要框图如图:

该芯片内部电路主要分为4部分:其中

  1. 光敏二极管,导通电流随着光强的变化而变化;
  2. I/V转换电路:主要是将电流信号转换为电压信号;
  3. ADC转换电路:将电压信号转换为数字信号,分辨率为16bit;
  4. IIC逻辑电路:主要是将光强数据打包成I标准的IIC通信信号;

引脚说明

 BH1750指令集

#define BHAddWrite     0x46      //写地址
#define BHAddRead      0x47      //读地址
#define BHPowDown      0x00      //关闭模块
#define BHPowOn        0x01      //打开模块等待指令 
#define BHReset        0x07      //重置数据寄存器仅在BHpowOn模式下有效
#define BHModeH1       0x10      //高分辨率 模式1 单位 11X 测量时间 120ms 
#define BHModeH2       0x11      //高分辨率 模式2 单位 11X 测量时间 120ms 
#define BHModeL        0x13      //低分辨率 单位4lx 测量时间16ms  
#define BHSigModeH     0x20      //一次测量高分辨率模式1,然后转到powerdown模式
#define BHSigModeH2    0x21      //同上类似
#define BHSigModeL     0x23      //同上类似  

接线表设计

序号GY-30单片机STM32
1VCC3.3V/5V
2SCLPB6
3SDAPB7
4GNDGND
5-PA9(USART1_ TX)
6-PA10(USART1_RX)

通过四种方式实现GY-30数据采集

1.标准库模拟IIC实现GY-30采集并串口1显示

核心代码gy30.c

//发送起始信号
void IIC_Start(void)
{IIC_Sdaout_Mode();//输出模式IIC_SCL=1;IIC_SDAout=1;delay_us(2);IIC_SDAout=0;IIC_SCL=0;//方便后续数据收发
}
//停止信号
void IIC_Stop(void)
{IIC_Sdaout_Mode();//输出模式IIC_SCL=0;IIC_SDAout=0;delay_us(2);IIC_SCL=1;delay_us(2);IIC_SDAout=1;
}
/***********发送应答信号**************************
**
**形参:u8 ack -- 0应答,1非应答
**
***************************************************/
void IIC_SendAck(u8 ack)
{IIC_Sdaout_Mode();//输出模式	IIC_SCL=0;//告诉从机,主机开始发送数据IIC_SDAout=ack&0x01;delay_us(2);IIC_SCL=1;//告诉从机,主机数据发送完成//方便下一次数据收发delay_us(2);IIC_SCL=0;
}
//获取应答信号
u8 IIC_Wait_Ack(void)
{u8 cnt=0;IIC_SdaIn_Mode();//配置为输入模式IIC_SDAout=1;	IIC_SCL=0;//告诉从机,主机需要获取数据delay_us(2);IIC_SCL=1;//从机数据发送完成,主机开始读取数据while(IIC_SDAin){cnt++;delay_us(1);if(cnt>=100)return 1;}delay_us(2);IIC_SCL=0;//方便下一次数据收发return 0;
}
//发送一个字节数据
void IIC_Send_Byte(u8 data)
{u8 i=0;IIC_Sdaout_Mode();//输出模式for(i=0;i<8;i++){IIC_SCL=0;//告诉从机,主机开始发送数据if(data&0x80)IIC_SDAout=1;else IIC_SDAout=0;delay_us(2);IIC_SCL=1;//主机数据发送完成data<<=1;delay_us(2);}IIC_SCL=0;//方便下一次数据收发
}
//读取一个字节数据
u8 IIC_Read_Byte(void)
{u8 i=0;u8 data=0;IIC_SdaIn_Mode();//配置为输入模式for(i=0;i<8;i++){IIC_SCL=0;//告诉从机,主机需要获取数据delay_us(2);IIC_SCL=1;//开始读取数据data<<=1;//默认收到0if(IIC_SDAin)data|=0x01;delay_us(2);}IIC_SCL=0;return data;
}void bh_data_send(u8 command)  
{  do{  IIC_Start();                      //iic开始IIC_Send_Byte(BHAddWrite);       //写地址  }while(IIC_Wait_Ack());           //等待响应 IIC_Send_Byte(command);          //发送命令  IIC_Wait_Ack();                   //等待响应 IIC_Stop();                       //iic停止  
}   u16 bh_data_read(void)  
{  u16 buf;  IIC_Start();                       //iic开始  IIC_Send_Byte(BHAddRead);         //发送读地址IIC_Wait_Ack();                     //等待响应  buf=IIC_Read_Byte();              //读取数据  IIC_SendAck(0);buf=buf<<8;                        //读取并保存高八位数据buf+=0x00ff&IIC_Read_Byte();      //读取并保存第八位数据IIC_SendAck(1);IIC_Stop();                        //发送停止信号return buf;   
}  void BH1750init(void)
{IIC_Init();//GPIO初始化bh_data_send(BHPowOn);    //发送启动信号bh_data_send(BHReset);    //清除寄存器  bh_data_send(BHModeH1);   //设置为模式1delay_ms(180);            //最高延时180ms 
}

实现

 2.标准库IIC实现GY-30采集并串口1显示

gy30.c核心代码

void I2C_GY30_Config(void)
{GPIO_InitTypeDef    GPIO_InitStuctrue;I2C_InitTypeDef     I2C_InitStuctrue;//开启GPIO外设时钟GY30_I2C_GPIO_APBxClkCmd(GY30_I2C_SCL_GPIO_CLK|GY30_I2C_SDA_GPIO_CLK,ENABLE);//开启IIC外设时钟GY30_I2C_APBxClkCmd(GY30_I2C_CLK,ENABLE);//SCL引脚-复用开漏输出GPIO_InitStuctrue.GPIO_Mode=GPIO_Mode_AF_OD;GPIO_InitStuctrue.GPIO_Pin=GY30_I2C_SCL_GPIO_PIN;GPIO_InitStuctrue.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GY30_I2C_SCL_GPIO_PORT,&GPIO_InitStuctrue);//SDA引脚-复用开漏输出GPIO_InitStuctrue.GPIO_Mode = GPIO_Mode_AF_OD;GPIO_InitStuctrue.GPIO_Pin = GY30_I2C_SDA_GPIO_PIN;GPIO_InitStuctrue.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GY30_I2C_SDA_GPIO_PORT,&GPIO_InitStuctrue);//IIC结构体成员配置I2C_InitStuctrue.I2C_Ack=I2C_Ack_Enable;I2C_InitStuctrue.I2C_AcknowledgedAddress=I2C_AcknowledgedAddress_7bit;I2C_InitStuctrue.I2C_ClockSpeed=GY30_I2C_BAUDRATE;I2C_InitStuctrue.I2C_DutyCycle=I2C_DutyCycle_2;I2C_InitStuctrue.I2C_Mode=I2C_Mode_I2C;I2C_InitStuctrue.I2C_OwnAddress1=STM32_I2C_OWN_ADDR;I2C_Init(GY30_I2C,&I2C_InitStuctrue);I2C_Cmd(GY30_I2C,ENABLE);}//向EEPROM写入一个字节
void  GY30_Byte_Write(uint8_t addr)
{//发送起始信号I2C_GenerateSTART(GY30_I2C,ENABLE);//检测EV5事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_MODE_SELECT)==ERROR);//发送设备写地址I2C_Send7bitAddress(GY30_I2C,GY30_I2C_Address,I2C_Direction_Transmitter);//检测EV6事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)==ERROR);//发送要操作设备内部的地址I2C_SendData(GY30_I2C,addr);while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTED )==ERROR);
//  I2C_SendData(EEPROM_I2C,data);
//	//检测EV8_2事件
//	while( I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTED )==ERROR);//发送停止信号I2C_GenerateSTOP(GY30_I2C,ENABLE);}//向EEPROM写入多个字节
uint32_t  GY30_Page_Write(uint8_t addr,uint8_t *data,uint16_t Num_ByteToWrite)
{I2CTimeout = I2CT_LONG_TIMEOUT;//判断IIC总线是否忙碌while(I2C_GetFlagStatus(GY30_I2C, I2C_FLAG_BUSY))   {if((I2CTimeout--) == 0) return 1;} //重新赋值I2CTimeout = I2CT_FLAG_TIMEOUT;//发送起始信号I2C_GenerateSTART(GY30_I2C,ENABLE);//检测EV5事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_MODE_SELECT)==ERROR){if((I2CTimeout--) == 0) return 2;} I2CTimeout = I2CT_FLAG_TIMEOUT;//发送设备写地址I2C_Send7bitAddress(GY30_I2C,GY30_I2C_Address,I2C_Direction_Transmitter);//检测EV6事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)==ERROR){if((I2CTimeout--) == 0) return 3;} I2CTimeout = I2CT_FLAG_TIMEOUT;//发送要操作设备内部的地址I2C_SendData(GY30_I2C,addr);//检测EV8事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING )==ERROR){if((I2CTimeout--) == 0) return 4;} while(Num_ByteToWrite){I2C_SendData(GY30_I2C,*data);I2CTimeout = I2CT_FLAG_TIMEOUT;while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING )==ERROR){if((I2CTimeout--) == 0) return  5;} Num_ByteToWrite--;data++;}I2CTimeout = I2CT_FLAG_TIMEOUT;//检测EV8_2事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTED )==ERROR){if((I2CTimeout--) == 0) return 6;} //发送停止信号I2C_GenerateSTOP(GY30_I2C,ENABLE);return 1;
}//向EEPROM读取多个字节
uint32_t GY30_Read(uint8_t *data,uint8_t addr,uint16_t Num_ByteToRead)
{I2CTimeout = I2CT_LONG_TIMEOUT;//判断IIC总线是否忙碌while(I2C_GetFlagStatus(GY30_I2C, I2C_FLAG_BUSY))   {if((I2CTimeout--) == 0) return 1;} I2CTimeout = I2CT_FLAG_TIMEOUT;//发送起始信号I2C_GenerateSTART(GY30_I2C,ENABLE);//检测EV5事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_MODE_SELECT )==ERROR){if((I2CTimeout--) == 0) return 7;} I2CTimeout = I2CT_FLAG_TIMEOUT;//发送设备写地址I2C_Send7bitAddress(GY30_I2C,GY30_I2C_Address,I2C_Direction_Transmitter);//检测EV6事件等待从机应答while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED )==ERROR){if((I2CTimeout--) == 0) return 8;}I2CTimeout = I2CT_FLAG_TIMEOUT;//发送要操作设备内部存储器的地址I2C_SendData(GY30_I2C,addr);//检测EV8事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING )==ERROR){if((I2CTimeout--) == 0) return 9;}I2CTimeout = I2CT_FLAG_TIMEOUT;//发送起始信号I2C_GenerateSTART(GY30_I2C,ENABLE);//检测EV5事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_MODE_SELECT )==ERROR){if((I2CTimeout--) == 0) return 10;}I2CTimeout = I2CT_FLAG_TIMEOUT;	 //发送设备读地址I2C_Send7bitAddress(GY30_I2C,GY30_I2C_Address,I2C_Direction_Receiver);//检测EV6事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED )==ERROR){if((I2CTimeout--) == 0) return 10;}while(Num_ByteToRead--){//是否是最后一个字节,若是则发送非应答信号if( Num_ByteToRead==0){//发送非应答信号I2C_AcknowledgeConfig(GY30_I2C,DISABLE);//发送停止信号I2C_GenerateSTOP(GY30_I2C,ENABLE);}I2CTimeout = I2CT_FLAG_TIMEOUT;	 //检测EV7事件while( I2C_CheckEvent(GY30_I2C,I2C_EVENT_MASTER_BYTE_RECEIVED )==ERROR){if((I2CTimeout--) == 0) return 10;}*data=I2C_ReceiveData(GY30_I2C);data++; }//重新开启应答信号I2C_AcknowledgeConfig(GY30_I2C,ENABLE);return 1;
}
void I2C_GY30_BufferWrite(uint8_t* pBuffer,uint8_t WriteAddr, uint16_t NumByteToWrite)
{u8 NumOfPage = 0, NumOfSingle = 0, Addr = 0, count = 0;//I2C_PageSize=8Addr = WriteAddr % I2C_PageSize;count = I2C_PageSize - Addr;NumOfPage =  NumByteToWrite / I2C_PageSize;NumOfSingle = NumByteToWrite % I2C_PageSize;/* 写入数据的地址对齐,对齐数为8 */if(Addr == 0) {/* 如果写入的数据个数小于8 */if(NumOfPage == 0) {GY30_Page_Write(WriteAddr, pBuffer, NumOfSingle);GY30_WaitForWriteEnd();}/* 如果写入的数据个数大于8 */else  {//按页写入while(NumOfPage--){GY30_Page_Write(WriteAddr, pBuffer, I2C_PageSize); GY30_WaitForWriteEnd();WriteAddr +=  I2C_PageSize;pBuffer += I2C_PageSize;}//不足一页(8个)单独写入if(NumOfSingle!=0){GY30_Page_Write(WriteAddr, pBuffer, NumOfSingle);GY30_WaitForWriteEnd();}}}/*写的数据的地址不对齐*/else {NumByteToWrite -= count;NumOfPage =  NumByteToWrite / I2C_PageSize;NumOfSingle = NumByteToWrite % I2C_PageSize;	if(count != 0){  GY30_Page_Write(WriteAddr, pBuffer, count);GY30_WaitForWriteEnd();WriteAddr += count;pBuffer += count;} while(NumOfPage--){GY30_Page_Write(WriteAddr, pBuffer, I2C_PageSize);GY30_WaitForWriteEnd();WriteAddr +=  I2C_PageSize;pBuffer += I2C_PageSize;  }if(NumOfSingle != 0){GY30_Page_Write(WriteAddr, pBuffer, NumOfSingle); GY30_WaitForWriteEnd();}} 
}

效果如下: 

3.HAL库模拟IIC实现GY-30采集并串口1显示

#include "gy30.h" uint8_t mcy=0;
uint8_t BUF[3];
/***开始信号**/
void BH1750_Start()
{HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_SET);                    //拉高数据线HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_SET);                   //拉高时钟线delay_us(5);                 HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_RESET);                    //产生下降沿delay_us(5);                HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_RESET);                    //拉低时钟线
}/*****停止信号******/
void BH1750_Stop()
{HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_RESET);                   //拉低数据线HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_SET);                      //拉高时钟线delay_us(5);                 HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_SET);                    //产生上升沿delay_us(5);                 
}
/*****初始化BH1750******/
void Init_BH1750()
{BH1750_Start();                                                 //起始信号BH1750_SendByte(SlaveAddress);                                  //发送设备地址+写信号BH1750_SendByte(0x01);                                  //内部寄存器地址BH1750_Stop();                                                  //停止信号}//连续读出BH1750内部数据
void mread(void)
{   uint8_t i;	BH1750_Start();                          //起始信号BH1750_SendByte(SlaveAddress+1);         //发送设备地址·+读信号for (i=0; i<3; i++)                      //连续读取6个地址数据到BUF{BUF[i] = BH1750_RecvByte();         if (i == 3){BH1750_SendACK(1);                //最后一个数据需要回NOACK}else{		BH1750_SendACK(0);                //回应ACK}}BH1750_Stop();                          //停止信号Delay_mms(5);}uint32_t Value_GY30(void)
{uint16_t dis_data;uint16_t Value_GY_30;Single_Write_BH1750(0x01);   // power onSingle_Write_BH1750(0x10);   // H- resolution mode   HAL_Delay(180);            //延时180ms                    mread();       //连续读出数据,存储在BUF中dis_data=BUF[0];dis_data=(dis_data<<8)+BUF[1];//字节合成数据Value_GY_30=dis_data;return Value_GY_30;
}
//系统主频72MHZ
void delay_us(uint16_t us)
{while(us--){__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();}
}
void Delay_mms(uint16_t tmp)
{uint16_t i=0;while(tmp--){i=12000;while(i--);}
}/**************************************
发送应答信号
入口参数:ack (0:ACK 1:NAK)
**************************************/
void BH1750_SendACK(int ack)
{GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.Pin = scl|sda;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); if(ack == 1)   //写应答信号HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_SET); else if(ack == 0)HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_RESET);elsereturn;HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_SET);     delay_us(5);               HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_RESET);      delay_us(5);               
}/**************************************
接收应答信号
**************************************/
int BH1750_RecvACK()
{		GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.Mode = GPIO_MODE_INPUT;  /*输入上拉*/GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;GPIO_InitStruct.Pin = sda;HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); 	HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_SET);            //拉高时钟线delay_us(5);                if(HAL_GPIO_ReadPin( GPIOB, sda ) == 1 )//读应答信号mcy = 1 ;  elsemcy = 0 ;			HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_RESET);                    //拉低时钟线delay_us(5);               GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;HAL_GPIO_Init( GPIOB, &GPIO_InitStruct );return mcy;
}/**************************************
向iic总线发送一个字节数据
**************************************/
void BH1750_SendByte(uint8_t dat)
{uint8_t i;for (i=0; i<8; i++)         //8位计数器{if( 0X80 & dat )HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_SET);elseHAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_RESET);dat <<= 1;HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_SET);               //拉高时钟线delay_us(5);             HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_RESET);                //拉低时钟线delay_us(5);            }BH1750_RecvACK();
}//我们对BH1750发送命令时,要先发送器件地址+写入位,然后发送指令
//读取数据的时候,需要先发送器件地址+读入位,然后读取两字节数据//写入指令
void Single_Write_BH1750(uint8_t REG_Address)//REG_Address是要写入的指令
{BH1750_Start();                  //起始信号BH1750_SendByte(SlaveAddress);  //发送器件地址+写信号BH1750_SendByte(REG_Address);   //写入指令,内部寄存器地址BH1750_Stop();                   //结束信号
}
/**************************************
从iic总线读取一个字节地址
**************************************/
uint8_t BH1750_RecvByte()
{uint8_t i;uint8_t dat = 0;uint8_t bit;GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.Mode = GPIO_MODE_INPUT;   /*上拉输入*/GPIO_InitStruct.Pin = sda;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init( GPIOB, &GPIO_InitStruct );HAL_GPIO_WritePin(GPIOB, sda,GPIO_PIN_SET);          //准备读取数据for (i=0; i<8; i++)         //8位计数器{dat <<= 1;HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_SET);               //拉高时钟线delay_us(5);            if( SET == HAL_GPIO_ReadPin( GPIOB, sda ) )bit = 0X01;elsebit = 0x00;  dat |= bit;             //读数据 HAL_GPIO_WritePin(GPIOB, scl,GPIO_PIN_RESET);                //拉低时钟线delay_us(5);           }GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;HAL_GPIO_Init( GPIOB, &GPIO_InitStruct );return dat;
}

 

4.HAL库IIC实现GY-30采集并串口1显示

//写命令
void GY30_WR_CMD(uint8_t cmd)
{HAL_I2C_Master_Transmit(&hi2c1, 0x46, &cmd,1, 0x100);//HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint8_t *pData, uint16_t Size, uint32_t Timeout);//HAL_I2C_Mem_Write(&hi2c1 ,0x46,0x00,I2C_MEMADD_SIZE_8BIT,&cmd,1,0x100);
}/**@brief BH1750初始化函数@param 无@return 无
*/
void BH1750_Init(void)
{GY30_WR_CMD(0x01);GY30_WR_CMD(0x10);GY30_WR_CMD(0x47);}//读数据
void GY30_READ_DATA(uint8_t data[])
{
//	uint8_t temp = 0X47;
//	
//	HAL_I2C_Mem_Write(&hi2c1 ,0x47,0x00,I2C_MEMADD_SIZE_8BIT,&temp,1,0x100);BH1750_Init();HAL_I2C_Master_Receive(&hi2c1, 0x46, data, 2, 0x100);//HAL_I2C_Mem_Read_IT(&hi2c1, 0x46, 0X00,I2C_MEMADD_SIZE_8BIT, data, 2);//HAL_I2C_Mem_Read(&hi2c1 ,0x46,0x47,I2C_MEMADD_SIZE_8BIT,data,2,0x100);
}/**@brief BH1750获取光强度@param 无@return 光强度
*/
uint32_t BH1750_ReadLightIntensity(void)
{uint32_t lux = 0;uint8_t sensorData[2] = {0};GY30_READ_DATA(sensorData);lux = (sensorData[0] << 8 | sensorData[1]);return lux;
}

四种实现效果一样,硬件IIC比模拟IIC的确要简单一些,看各人的喜好,花了一天时间特意研究整理,实现效果一定没有问题的。

创作不易

源码下载链接

(2条消息) STM32模拟IIC与IIC四种实现实现数字光强采集模块GY30(标准库与HAL库)-C文档类资源-CSDN文库

吾芯电子工作室

这篇关于STM32模拟IIC与IIC四种实现数字光强采集模块GY30(标准库与HAL库)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Nginx 配置跨域的实现及常见问题解决

《Nginx配置跨域的实现及常见问题解决》本文主要介绍了Nginx配置跨域的实现及常见问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来... 目录1. 跨域1.1 同源策略1.2 跨域资源共享(CORS)2. Nginx 配置跨域的场景2.1

Python中提取文件名扩展名的多种方法实现

《Python中提取文件名扩展名的多种方法实现》在Python编程中,经常会遇到需要从文件名中提取扩展名的场景,Python提供了多种方法来实现这一功能,不同方法适用于不同的场景和需求,包括os.pa... 目录技术背景实现步骤方法一:使用os.path.splitext方法二:使用pathlib模块方法三

CSS实现元素撑满剩余空间的五种方法

《CSS实现元素撑满剩余空间的五种方法》在日常开发中,我们经常需要让某个元素占据容器的剩余空间,本文将介绍5种不同的方法来实现这个需求,并分析各种方法的优缺点,感兴趣的朋友一起看看吧... css实现元素撑满剩余空间的5种方法 在日常开发中,我们经常需要让某个元素占据容器的剩余空间。这是一个常见的布局需求

HTML5 getUserMedia API网页录音实现指南示例小结

《HTML5getUserMediaAPI网页录音实现指南示例小结》本教程将指导你如何利用这一API,结合WebAudioAPI,实现网页录音功能,从获取音频流到处理和保存录音,整个过程将逐步... 目录1. html5 getUserMedia API简介1.1 API概念与历史1.2 功能与优势1.3

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解

一文深入详解Python的secrets模块

《一文深入详解Python的secrets模块》在构建涉及用户身份认证、权限管理、加密通信等系统时,开发者最不能忽视的一个问题就是“安全性”,Python在3.6版本中引入了专门面向安全用途的secr... 目录引言一、背景与动机:为什么需要 secrets 模块?二、secrets 模块的核心功能1. 基

PostgreSQL中MVCC 机制的实现

《PostgreSQL中MVCC机制的实现》本文主要介绍了PostgreSQL中MVCC机制的实现,通过多版本数据存储、快照隔离和事务ID管理实现高并发读写,具有一定的参考价值,感兴趣的可以了解一下... 目录一 MVCC 基本原理python1.1 MVCC 核心概念1.2 与传统锁机制对比二 Postg

SpringBoot整合Flowable实现工作流的详细流程

《SpringBoot整合Flowable实现工作流的详细流程》Flowable是一个使用Java编写的轻量级业务流程引擎,Flowable流程引擎可用于部署BPMN2.0流程定义,创建这些流程定义的... 目录1、流程引擎介绍2、创建项目3、画流程图4、开发接口4.1 Java 类梳理4.2 查看流程图4

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri