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),这对大多数没有操作系统的单片机软件来说。 我们也不例外,请看下面的程序框架:
根据电路图和传感器的数据手册,进行如下的配置
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); } }以为加速度传感器的程序为例,当然气压器的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官网还提供了很多示例,根据实际项目需要可以添加需要的传感器功能。