STM32F0 I2C 驱动光感模块 GY30(BH1750FVI)

最近因为工程需要,需要使用光感模块BH1750FVI,使用STM32F030F4。本来打算移植店主给的51代码,但移植后发现无法调通,正好STM32F0系列的I2C还未调通过,于是打算用这个模块来练手。
STM32 F0跟F3系列的自带I2C库跟F1最大的不通就是多了个 I2C_Timing,其他的区别不大。
至于这个 I2C_Timing 怎么得出的,我们需要去ST官网下载相关的一个XLS文件,查找方式建议先通过搜索 STM32F0 I2C来找到文档,文档中有工具的编号,我们就可以通过编号来直接下载了(发布文章时是 STSW-STM32126)。
用法可以参考官方社区文档:

http://www.stmcu.org/module/forum/thread-571931-1-1.html

好,下面进入正题。(本文章完全理解需要一些I2C基础)

I2C配置

代码:

void I2C_BH175()
{
    GPIO_InitTypeDef GPIO_InitStructure;
    I2C_InitTypeDef I2C_InitStructure;

    RCC_I2CCLKConfig(RCC_I2C1CLK_SYSCLK);
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE);  //使能GPIOA的时钟
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1,ENABLE);  //使能I2C1的时钟

    GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_4);//配置PA9成第4功能引脚 SCL
    GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_4);//配置PA10成第4功能引脚  SDA

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//这个我觉得影响不大,只要是推挽模式这个可以换成NOPULL
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    I2C_InitStructure.I2C_Mode = I2C_Mode_SMBusHost;//这个地方网上很多同志换成了I2C_Mode_I2C,应该也是没问题的。这里写成SMBusHost是因为他作为主机
    I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;
    I2C_InitStructure.I2C_AnalogFilter = I2C_AnalogFilter_Enable;//模拟信号滤波
    I2C_InitStructure.I2C_DigitalFilter = 0x01;//数字信号滤波
    I2C_InitStructure.I2C_OwnAddress1 = 0x00;//主机地址(就是本机地址,没做过从机的I2C所以不太清楚)
    I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;//ACK
    I2C_InitStructure.I2C_Timing = 0x0090174F;//这个需要用那个软件来计算
    I2C_Init(I2C1, &I2C_InitStructure);
    I2C_Cmd(I2C1, ENABLE);
}

I2C 发送与接收

这里只用了 I2C_TransferHandlingI2C_SendDataI2C_GenerateSTOPI2C_ReceiveData 这三个函数。
第一个函数的作用可以认为是传输中报头的设定。
比如:

I2C_TransferHandling(I2C1,SlaveAddress,1,I2C_SoftEnd_Mode,I2C_Generate_Start_Write)

首先,这里面的 1 代表需要接收的字节数。
I2C_SoftEnd_Mode 代表我们需要手动发送STOP信号,即 I2C_GenerateSTOP。还有Auto模式可以自动发送STOP,在读取了设定的数字后发送STOP结束通信。
I2C_Generate_Start_Write 顾名思义,生成SlaveAddress后面那一位 R/W。并且理所当然,还有I2C_Generate_Start_Read

好,说的差不多了,上代码。

void Single_Write_BH1750(uint8_t REG_Data)
{
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY) != RESET);//检测是否总线忙
    I2C_TransferHandling(I2C1,SlaveAddress,1,I2C_SoftEnd_Mode,I2C_Generate_Start_Write);
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_TXIS) == RESET);// 检测发送
    I2C_SendData(I2C1,REG_Data);
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_TC) == RESET);
    I2C_GenerateSTOP(I2C1,ENABLE);
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_STOPF) == RESET);//是否停止
}
uint16_t Read_BH1750()
{
    uint16_t Recev = 0x00;//光强为16位。先发送高8后低8
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY) != RESET);//IF BUSY
    I2C_TransferHandling(I2C1,SlaveAddress+1,2,I2C_AutoEnd_Mode,I2C_Generate_Start_Read);//这里设置后就不需要 I2C_GenerateSTOP 了
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_RXNE) == RESET);
    Recev |= I2C_ReceiveData(I2C1);Recev<<=8;
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_RXNE) == RESET);
    Recev |= I2C_ReceiveData(I2C1);
    while(I2C_GetFlagStatus(I2C1, I2C_FLAG_STOPF) == RESET);
    return Recev;
}

其实回过头来看很简单。有的时候我们就是卡在某一个位置,可能是我们的马虎,也可能是其他一些奇奇怪怪的原因。但只要不放弃,我们一定能得出答案,得出想要的结果。

你可能感兴趣的:(STM32)