BMP180气压传感器在天空星GD32F407开发板上的I2C驱动移植与海拔测量实战
BMP180气压传感器在天空星GD32F407开发板上的I2C驱动移植与海拔测量实战最近在做一个户外气象站的项目需要测量气压和温度来计算海拔高度正好用上了BMP180这款传感器。很多刚开始接触嵌入式开发的朋友一看到传感器数据手册里复杂的换算公式就头疼。其实只要把I2C通信调通剩下的就是“照葫芦画瓢”。今天我就以天空星GD32F407开发板为例手把手带你完成BMP180的驱动移植从硬件连接到软件实现最后还能算出海拔高度。1. 认识BMP180不只是个气压计BMP180是博世公司推出的一款高精度数字气压传感器它体积小、功耗低非常适合用在移动设备和物联网项目中。咱们简单了解一下它的特点能测什么环境温度和大气压力通信方式I2C接口接线简单测量范围气压300-1100 hPa相当于海拔-500米到9000米温度0-65℃精度温度±1℃气压±1 hPa工作电压1.8-3.6V可以直接用3.3V供电注意大气压力会随着海拔升高而降低这就是我们能用气压来计算海拔的原理。海平面标准大气压是101325帕斯卡约1013 hPa。BMP180最特别的一点是每个传感器出厂时都单独校准过校准参数存储在内部的EEPROM里。所以咱们读取数据时不能直接用原始值得用这些校准参数换算一下才能得到真实值。2. 硬件连接三根线搞定BMP180模块一般就三个引脚接线特别简单模块引脚功能连接GD32F407引脚VCC电源3.3V3.3VGND地GNDSCLI2C时钟线PA1可自定义SDAI2C数据线PA2可自定义在天空星开发板上我习惯用PA1和PA2作为I2C引脚当然你也可以换成其他GPIO。接线时注意如果模块上没有上拉电阻需要在SCL和SDA线上各加一个4.7kΩ的上拉电阻到3.3V。3. I2C通信基础模拟时序更灵活虽然GD32F407有硬件I2C外设但很多朋友反映硬件I2C调试起来比较麻烦。这里我用GPIO模拟I2C时序虽然速度慢点但稳定可靠移植也方便。3.1 GPIO初始化首先得把用到的引脚配置好。在bsp_bmp180.c文件中我们这样初始化void BMP180_GPIO_Init(void) { /* 使能GPIOA时钟 */ rcu_periph_clock_enable(RCU_GPIOA); /* 配置SCLPA1为开漏输出上拉 */ gpio_mode_set(GPIOA, GPIO_MODE_OUTPUT, GPIO_PUPD_PULLUP, GPIO_PIN_1); gpio_output_options_set(GPIOA, GPIO_OTYPE_OD, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* 配置SDAPA2为开漏输出上拉 */ gpio_mode_set(GPIOA, GPIO_MODE_OUTPUT, GPIO_PUPD_PULLUP, GPIO_PIN_2); gpio_output_options_set(GPIOA, GPIO_OTYPE_OD, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* 初始状态SCL和SDA都拉高 */ gpio_bit_set(GPIOA, GPIO_PIN_1); gpio_bit_set(GPIOA, GPIO_PIN_2); }这里有几个关键点开漏输出ODI2C总线需要“线与”功能开漏输出才能实现上拉电阻内部上拉电阻阻值较大约40kΩ长距离通信时最好外接4.7kΩ上拉初始高电平I2C总线空闲时SCL和SDA都应该为高电平3.2 模拟I2C基本时序I2C通信有几个基本时序起始、停止、发送数据、接收数据、应答。咱们一个个来实现。起始信号SCL为高时SDA从高变低void IIC_Start(void) { SDA_OUT(); // SDA设置为输出 SDA(1); // SDA高 SCL(1); // SCL高 delay_us(5); // 保持一段时间 SDA(0); // SDA拉低产生起始条件 delay_us(5); SCL(0); // SCL拉低准备传输数据 }停止信号SCL为高时SDA从低变高void IIC_Stop(void) { SDA_OUT(); SCL(0); // 确保SCL为低 SDA(0); // SDA拉低 SCL(1); // SCL拉高 delay_us(5); SDA(1); // SDA拉高产生停止条件 delay_us(5); }发送一个字节从高位到低位每个bit在SCL低电平时变化高电平时保持稳定void Send_Byte(uint8_t dat) { int i 0; SDA_OUT(); SCL(0); // 拉低时钟开始数据传输 for(i 0; i 8; i) { // 发送最高位 SDA((dat 0x80) 7); delay_us(1); SCL(1); // 拉高时钟从机读取数据 delay_us(5); SCL(0); // 拉低时钟准备下一位 delay_us(5); dat 1; // 左移准备发送下一位 } }接收一个字节主机控制SCL在SCL高电平时读取SDA状态unsigned char Read_Byte(void) { unsigned char i, receive 0; SDA_IN(); // SDA设置为输入模式 for(i 0; i 8; i) { SCL(0); delay_us(5); SCL(1); // 拉高时钟 delay_us(5); receive 1; // 左移一位 if(SDA_GET()) // 读取SDA状态 { receive | 1; // 如果为高最低位置1 } delay_us(5); } SCL(0); return receive; }等待应答发送完一个字节后从机会在第9个时钟周期拉低SDA作为应答unsigned char I2C_WaitAck(void) { char ack 0; unsigned char ack_flag 10; SCL(0); SDA(1); // 释放SDA SDA_IN(); // SDA设为输入 delay_us(5); SCL(1); // 第9个时钟 delay_us(5); // 等待SDA被拉低应答 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; }4. BMP180驱动实现一步步读取数据4.1 读取校准参数每个BMP180都有11个校准参数存储在EEPROM的特定地址。这些参数用于后续的温度和气压计算。typedef struct _BMP180_STRUCT{ short AC1; short AC2; short AC3; uint16_t AC4; uint16_t AC5; uint16_t AC6; short B1; short B2; short MB; short MC; short MD; }_BMP180_PARAM_; _BMP180_PARAM_ param {0}; void BMP180_Get_param(void) { param.AC1 BMP180_Read16(0xAA, 2); param.AC2 BMP180_Read16(0xAC, 2); param.AC3 BMP180_Read16(0xAE, 2); param.AC4 BMP180_Read16(0xB0, 2); param.AC5 BMP180_Read16(0xB2, 2); param.AC6 BMP180_Read16(0xB4, 2); param.B1 BMP180_Read16(0xB6, 2); param.B2 BMP180_Read16(0xB8, 2); param.MB BMP180_Read16(0xBA, 2); param.MC BMP180_Read16(0xBC, 2); param.MD BMP180_Read16(0xBE, 2); }这里用到的BMP180_Read16函数是我们封装的读取函数可以读取16位数据。注意校准参数有正有负short类型而AC4、AC5、AC6是无符号的uint16_t类型。4.2 读取温度数据读取温度分三步发送温度转换命令等待转换完成4-5ms读取原始温度值并计算float BMP180_Get_Temperature(void) { long UT 0; // 原始温度值 long X1 0, X2 0; // 1. 启动温度转换 BMP180_Write_Cmd(0xF4, 0x2E); delay_ms(6); // 等待转换完成手册要求至少4.5ms // 2. 读取原始温度值16位 UT BMP180_Read16(0xF6, 2); // 3. 使用校准参数计算真实温度 X1 ((long)UT - param.AC6) * param.AC5 / 32768.0; X2 ((long)param.MC * 2048.0) / (X1 param.MD); B5 X1 X2; // B5是中间变量气压计算也会用到 // 返回温度值单位℃ return ((B5 8) / 16.0) * 0.1f; }提示温度计算中乘以0.1是因为BMP180的温度分辨率是0.1℃。比如计算结果是250实际温度就是25.0℃。4.3 读取气压数据气压读取比温度复杂一些因为BMP180有四种工作模式对应不同的精度和转换时间。这里我用的是标准模式oss0。float BMP180_Get_Pressure(void) { long UP 0; // 原始气压值 uint8_t oss 0; // 工作模式0标准模式 long X1 0, X2 0; // 先读取温度因为气压计算需要B5值 BMP180_Get_Temperature(); // 启动气压转换标准模式 BMP180_Write_Cmd(0xF4, (0x34 (oss 6))); delay_ms(10); // 标准模式需要4.5ms这里给10ms确保完成 // 读取原始气压值19位分三个字节 UP BMP180_Read16(0xF6, 3); // 以下是气压换算公式来自数据手册 int32_t B6 B5 - 4000; X1 (B6 * B6 12) * param.B2 11; X2 param.AC2 * B6 11; int32_t X3 X1 X2; int32_t B3 (((param.AC1 2) X3) 2) 2; X1 param.AC3 * B6 13; X2 (B6 * B6 12) * param.B1 16; X3 (X1 X2 2) 2; uint32_t B4 param.AC4 * (uint32_t)(X3 32768) 15; uint32_t B7 ((uint32_t)UP - B3) * 50000; int32_t p; if(B7 0x80000000) { p (B7 1) / B4; } else { p B7 / B4 1; } X1 (p 8) * (p 8); X1 (X1 * 3038) 16; X2 (-7375 * p) 16; p p ((X1 X2 3791) 4); return p; // 单位Pa }这些换算公式看起来复杂其实是BMP180数据手册里给出的标准算法。每个传感器因为生产工艺的微小差异都需要用自己独特的校准参数来修正测量值。4.4 计算海拔高度有了气压值我们就可以计算海拔高度了。这里用到了气压高度公式float BMP180_Get_Altitude(float p) { float altitude 0; // 国际标准大气压模型公式 altitude 44330 * (1 - pow((p) / 101325.0f, 1.0f / 5.255f)); return altitude; // 单位米 }这个公式基于一个假设海平面标准大气压是101325 Pa。实际应用中如果你知道当地的海平面气压可以用实际值替换101325这样计算出的海拔会更准确。5. 完整工程集成与测试5.1 头文件配置在bsp_bmp180.h中我们定义了引脚和函数接口#ifndef _BSP_BMP180_H_ #define _BSP_BMP180_H_ #include gd32f4xx.h // 引脚定义 - 根据实际接线修改 #define RCU_SDA RCU_GPIOA #define PORT_SDA GPIOA #define GPIO_SDA GPIO_PIN_2 #define RCU_SCL RCU_GPIOA #define PORT_SCL GPIOA #define GPIO_SCL GPIO_PIN_1 // 引脚操作宏 #define SDA_OUT() gpio_mode_set(PORT_SDA, GPIO_MODE_OUTPUT, GPIO_PUPD_PULLUP, GPIO_SDA) #define SDA_IN() gpio_mode_set(PORT_SDA, GPIO_MODE_INPUT, GPIO_PUPD_PULLUP, GPIO_SDA) #define SDA_GET() gpio_input_bit_get(PORT_SDA, GPIO_SDA) #define SDA(x) gpio_bit_write(PORT_SDA, GPIO_SDA, (x ? SET : RESET)) #define SCL(x) gpio_bit_write(PORT_SCL, GPIO_SCL, (x ? SET : RESET)) // 函数声明 void BMP180_GPIO_Init(void); float BMP180_Get_Temperature(void); float BMP180_Get_Pressure(void); void BMP180_Write_Cmd(uint8_t regaddr, uint8_t cmd); void BMP180_Get_param(void); float BMP180_Get_Altitude(float p); #endif5.2 主函数调用在主函数中我们这样使用BMP180#include board.h #include bsp_bmp180.h int main(void) { // 系统初始化 board_init(); bsp_uart_init(); // 初始化串口用于打印数据 // BMP180初始化 BMP180_GPIO_Init(); BMP180_Get_param(); // 读取校准参数只需要一次 printf(BMP180初始化完成开始测量...\r\n); while(1) { // 读取并打印温度 float temperature BMP180_Get_Temperature(); printf(温度: %.2f ℃\r\n, temperature); // 读取并打印气压 float pressure BMP180_Get_Pressure(); printf(气压: %.2f Pa\r\n, pressure); // 计算并打印海拔 float altitude BMP180_Get_Altitude(pressure); printf(海拔: %.2f 米\r\n, altitude); printf(-------------------\r\n); delay_ms(1000); // 每秒测量一次 } }5.3 常见问题排查在实际调试中你可能会遇到这些问题读取不到数据或数据全为0检查电源BMP180需要3.3V供电检查接线SCL、SDA是否接反检查上拉电阻如果没有外部上拉尝试启用内部上拉数据明显错误确保已经正确读取了11个校准参数检查I2C地址BMP180的写地址是0xEE读地址是0xEF检查延时转换需要时间温度至少4.5ms气压根据模式不同需要4.5-25.5ms海拔计算不准确海拔计算依赖气压值气压测量有±1 hPa的误差公式基于标准大气模型实际天气变化会影响结果可以用已知海拔的地点进行校准I2C通信不稳定降低通信速度增加延时检查总线是否有其他设备地址冲突长距离传输时确保有足够强的上拉移植完成后上电运行你应该能在串口助手中看到每秒更新的温度、气压和海拔数据。第一次看到自己测量的海拔高度时还是挺有成就感的。这个驱动我在好几个项目里都用过稳定性不错你可以根据自己的需求调整测量频率和工作模式。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2408911.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!