1、I2C总线具有两根双向信号线,一根是数据线SDA,另一根是时钟线SCL
2、IIC总线上可以挂很多设备:多个主设备,多个从设备(外围 设备)。
3、多主机会产生总线裁决问题。当多个主机同时想占用总线时,企图启动总线传输数据,就叫做总线竞争。I2C通过总线仲裁,以决定哪台主机控制总线
在一般的项目中,一般不会涉及到IIC总线上挂载多主机多从机的情况。但挂载单个主机多个从机的情况还是有的。
在嵌入式领域:要不专注于硬件的设计,要不专注于软件的实现。当然最好两者都兼备。相对来说,软件层面的复杂度会相对高一点,在项目开发所占的任务比重也比较大,特别是一些组网的大型项目,往往还会涉及到上位机测试软件的开发等。本篇主要讲述STM32系列单片机的IIC挂载多个从机的程序实现。
如上图,将LIS2HH12加速度传感器、LPS25HB气压传感器通过IIC总线相连,与STM32Lxx 系列MCU的管脚PB6\PB7 相连。
对于一个嵌入式软件工程师来说,IIC通信的基本知识需要了解,不过最重要的还是关心程序如何实现。
ST公司目前将STM32全系列都是支持HAL库的开发。cubeMx软件可以外设的驱动程序自动生成。但是多说情况下,还是需要手工改写。要不怎么称得上嵌入式软件开发工程师呢?
一般来说,对于嵌入式领域的单片机程序开发,
程序开头都会做一些初始化,初始化完成后然后进入一个死循环while(1),这对大多数没有操作系统的单片机软件来说。
我们也不例外,请看下面的程序框架:
int main(void)
{
HAL_Init();
SystemClock_Config(); //系统时钟初始化
MX_I2C1_Init(); //外设IIC的初始化,PB6\PB7,对应的外设IIC1
/* 看门狗初始化 */
MX_IWDG_Init();
HAL_IWDG_Refresh(&hiwdg);
/*加速度传感器的配置*/
BSP_ACC_Init(&dev_ctx);
BSP_ACC_Config(&dev_ctx);
/*气压计的配置及功能实现*/
BSP_BARO_Init(&dev_baro));
BSP_BARO_Config(&dev_baro);
while(1)
{
/*加速度传感器运用中断方式实现,所以while(1)中就出添加加速度的处理程序了*/
/*气压计的处理程序*/
BSP_BARO_handle(&dev_baro);
}
}
根据电路图和传感器的数据手册,进行如下的配置
uint8_t MX_I2C1_Init(void) //加速度、气压传感器SensorIIC接口
{
hi2c1.Instance = I2C1;
hi2c1.Init.ClockSpeed = 400000; //
hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2;
hi2c1.Init.OwnAddress1 = 0x34; //这里的排至值并不影响。因为后面我们IIC的读写程序不用库函数,而是重写IIC的读写功能。
hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;
hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;
hi2c1.Init.OwnAddress2 = 0;//这里的排至值并不影响。因为后面我们IIC的读写程序不用库函数,而是重写IIC的读写功能。
hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;
hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;
if (HAL_I2C_Init(&hi2c1) != HAL_OK)
{
_Error_Handler(__FILE__, __LINE__);
}
if( HAL_I2C_GetState( &hi2c1) == HAL_I2C_STATE_READY )
{
return 0;
}
else
{
return 1;
}
}
这里我们定义了传感器对象的一个结构体,将此结构体的指针指向两个IIC的读写程序platform_write、platform_read。应为传感器挂载在IIC1上,这里将传感器结构体指针的句柄定位为IIC1的地址。
I2C_HandleTypeDef hi2c1; //传感器的I2C接口
# define SENSOR_BUS hi2c1
void BSP_ACC_Init(lis2hh12_ctx_t *dev_acc)
{
dev_acc->write_reg = platform_write;
dev_acc->read_reg = platform_read;
dev_acc->handle = &SENSOR_BUS;
BSP_ACC_IO_ITConfig(); // 使能ACC MEMS 中断
}
//使能加速度传感器的中断管脚功能。
void BSP_ACC_IO_ITConfig( void )
{
/* At the moment this feature is only implemented for LPS22HB */
GPIO_InitTypeDef GPIO_InitStructureInt1;
/* Enable INT1 GPIO clock */
__GPIOA_CLK_ENABLE();
/* Configure GPIO PINs to detect Interrupts */
GPIO_InitStructureInt1.Pin = GPIO_PIN_0;
GPIO_InitStructureInt1.Mode = GPIO_MODE_IT_RISING;
GPIO_InitStructureInt1.Speed = GPIO_SPEED_MEDIUM;
GPIO_InitStructureInt1.Pull = GPIO_NOPULL;
HAL_GPIO_Init(GPIOA, &GPIO_InitStructureInt1);
/* Enable and set EXTI Interrupt priority */
HAL_NVIC_SetPriority(EXTI0_IRQn, 4, 0x00);
}
/************************************************************************
* 函数名称:BSP_ACC_Config(lis2hh12_ctx_t *dev_acc)
* 功 能: 加速度配置
* 输入参数:无
* 返 回 值:dev_acc 结构体
* 其 他:无
************************************************************************/
void BSP_ACC_Config(lis2hh12_ctx_t *dev_acc)
{
/* Check device ID */
lis2hh12_dev_id_get(dev_acc, &whoamI);
if (whoamI != LIS2HH12_ID)
{
while(1)
{
/* manage here device not found */
}
}
/* Restore default configuration */
lis2hh12_dev_reset_set(dev_acc, PROPERTY_ENABLE);
do {
lis2hh12_dev_reset_get(dev_acc, &rst);
} while (rst);
/* Enable Block Data Update */
lis2hh12_block_data_update_set(dev_acc, PROPERTY_ENABLE);
/* Set full scale */
lis2hh12_xl_full_scale_set(dev_acc, LIS2HH12_8g); //在加速度全域范围内运行【-8g:8g】
/* Configure filtering chain */
/* Accelerometer data output- filter path / bandwidth */
lis2hh12_xl_filter_aalias_bandwidth_set(dev_acc, LIS2HH12_AUTO); // 自适应 bandwidth
lis2hh12_xl_filter_out_path_set(dev_acc, LIS2HH12_BYPASSED); // 开启内部低通滤波
lis2hh12_xl_filter_low_bandwidth_set(dev_acc, LIS2HH12_LP_ODR_DIV_9); //设置低通滤波的频率
/* Accelerometer interrrupt - filter path / bandwidth */
lis2hh12_xl_filter_int_path_set(dev_acc, LIS2HH12_HP_DISABLE); //开启 内部高通滤波
/* Set Output Data Rate */
lis2hh12_xl_data_rate_set(dev_acc, LIS2HH12_XL_ODR_50Hz); // ODR 设为100Hz
#ifdef FIFO_ACC
lis2hh12_fifo_mode_set(dev_acc,LIS2HH12_STREAM_MODE); // 设置 FIFO Mode
lis2hh12_fifo_watermark_set(dev_acc,PROPERTY_ENABLE); //开启watermark
lis2hh12_pin_int1_route_t pinValue;
pinValue.int1_drdy = 0;
pinValue.int1_fth = 1;
pinValue.int1_inact = 0;
pinValue.int1_ig1 = 0;
pinValue.int1_ig2 = 0;
pinValue.int1_ovr = 0;
lis2hh12_pin_int1_route_set(dev_acc,pinValue); //设置中断
lis2hh12_fifo_watermark_set_level(dev_acc,THRESH_MASK);
#endif
};
加速度传感器的中断管脚对应的中断服务函数:
在中断函数中实现计步功能。
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
if(GPIO_Pin == GPIO_PIN_0) //
{
ACC_DRY_Flag = SET;
lis2hh12_handle(&dev_ctx);
}
}
void BSP_ACC_Init(lps25hb_ctx_t *dev_ctx)
{
dev_ctx.write_reg = platform_write;
dev_ctx.read_reg = platform_read;
dev_ctx.handle = &hi2c1;
}
void BSP_ACC_Config(lps25hb_ctx_t *dev_ctx)
{
/* Check device ID */
whoamI = 0;
lps25hb_device_id_get(&dev_ctx, &whoamI);
if ( whoamI != LPS25HB_ID )
while(1); /*manage here device not found */
/* Restore default configuration */
lps25hb_reset_set(&dev_ctx, PROPERTY_ENABLE);
do {
lps25hb_reset_get(&dev_ctx, &rst);
} while (rst);
/* Enable Block Data Update */
lps25hb_block_data_update_set(&dev_ctx, PROPERTY_ENABLE);
/* Set Output Data Rate */
lps25hb_data_rate_set(&dev_ctx, LPS25HB_ODR_1Hz);
}
void BSP_BARO_handle(lps25hb_ctx_t *dev_ctx);
{
while(1)
{
HAL_IWDG_Refresh(&hiwdg);
/* Read output only if new value is available */
lps25hb_reg_t reg;
lps25hb_status_get(&dev_ctx, ®.status_reg);
if (reg.status_reg.p_da)
{
memset(data_raw_pressure.u8bit, 0x00, sizeof(int32_t));
lps25hb_pressure_raw_get(dev_ctx, data_raw_pressure.u8bit);
pressure_hPa = lps25hb_from_lsb_to_hpa( data_raw_pressure.i32bit);
}
if (reg.status_reg.t_da)
{
memset(data_raw_temperature.u8bit, 0x00, sizeof(int16_t));
lps25hb_temperature_raw_get(dev_ctx, data_raw_temperature.u8bit);
temperature_degC = lps25hb_from_lsb_to_degc( data_raw_temperature.i16bit);
}
HAL_Delay(500);
}
}
以为加速度传感器的程序为例,当然气压器的IIC读写程序可以类似仿写。
读写时将从机的地址要写入
#define LIS2HH12_I2C_ADD_L 0x3DU
/*
* @brief Write generic device register (platform dependent)
*
* @param handle customizable argument. In this examples is used in
* order to select the correct sensor bus handler.
* @param reg register to write
* @param bufp pointer to data to write in register reg
* @param len number of consecutive register to write
*
*/
static int32_t platform_write(void *handle, uint8_t reg, uint8_t *bufp,
uint16_t len)
{
if (handle == &hi2c1)
{
HAL_I2C_Mem_Write(handle, LIS2HH12_I2C_ADD_L, reg,
I2C_MEMADD_SIZE_8BIT, bufp, len, 1000);
}
return 0;
}
/*
* @brief Read generic device register (platform dependent)
*
* @param handle customizable argument. In this examples is used in
* order to select the correct sensor bus handler.
* @param reg register to read
* @param bufp pointer to buffer that store the data read
* @param len number of consecutive register to read
*
*/
static int32_t platform_read(void *handle, uint8_t reg, uint8_t *bufp,
uint16_t len)
{
if (handle == &hi2c1)
{
HAL_I2C_Mem_Read(handle, LIS2HH12_I2C_ADD_L, reg,
I2C_MEMADD_SIZE_8BIT, bufp, len, 1000);
}
return 0;
}
ok,将代码烧录到板子上,测试成功。debug 可以看到气压计和计步值都正常输出了。
ST的Mems扩展板的示例程序上有IIC总结挂载多个从机的Demo,但是移植程序比较困难。因为需要改写很多内容。
这个示例比较简单实用。当然ST官网还提供了很多示例,根据实际项目需要可以添加需要的传感器功能。