当前位置: 首页 > java >正文

仿正点原子驱动BMP280气压传感器实例

文章目录

前言

一、寄存器头文件定义

二、设备树文件中添加节点

 三、驱动文件编写

四、编写驱动测试文件并编译测试

总结


前言

本文驱动开发仿照正点原子的iic驱动实现,同时附上bmp280的数据手册,可访问下面的链接:

BMP280_Bosch(博世)_BMP280中文资料_PDF手册_价格-立创商城下载数据手册。


一、寄存器头文件定义

        在查看数据手册时,可发现bmp280器件对各个寄存器,如数据,校准,采样,滤波等等寄存器进行定义,如下代码所示:

#ifndef BMP280_H
#define BMP280_H#define BMP280_ADDR    	0x76	/* BMP280器件地址*//* BMP280寄存器 */
#define BMP280_ID_REG          0xD0  /* 芯片ID寄存器,值是0x58 */
#define BMP280_RESET_REG       0xE0  /* 复位寄存器 */
#define BMP280_STATUS_REG      0xF3  /* 状态寄存器 */
#define BMP280_CTRL_MEAS_REG   0xF4  /* 测量控制寄存器 */
#define BMP280_CONFIG_REG      0xF5  /* 配置寄存器 *//* 数据寄存器 */
#define BMP280_PRESSURE_MSB    0xF7  /* 压力数据高字节 */
#define BMP280_PRESSURE_LSB    0xF8  /* 压力数据中字节 */
#define BMP280_PRESSURE_XLSB   0xF9  /* 压力数据低字节 */
#define BMP280_TEMP_MSB        0xFA  /* 温度数据高字节 */
#define BMP280_TEMP_LSB        0xFB  /* 温度数据中字节 */
#define BMP280_TEMP_XLSB       0xFC  /* 温度数据低字节 *//* 校准参数寄存器 - 存储补偿计算需要的出厂校准数据 */
#define BMP280_DIG_T1_LSB_REG  0x88
#define BMP280_DIG_T1_MSB_REG  0x89
#define BMP280_DIG_T2_LSB_REG  0x8A
#define BMP280_DIG_T2_MSB_REG  0x8B
#define BMP280_DIG_T3_LSB_REG  0x8C
#define BMP280_DIG_T3_MSB_REG  0x8D
#define BMP280_DIG_P1_LSB_REG  0x8E
#define BMP280_DIG_P1_MSB_REG  0x8F
#define BMP280_DIG_P2_LSB_REG  0x90
#define BMP280_DIG_P2_MSB_REG  0x91
#define BMP280_DIG_P3_LSB_REG  0x92
#define BMP280_DIG_P3_MSB_REG  0x93
#define BMP280_DIG_P4_LSB_REG  0x94
#define BMP280_DIG_P4_MSB_REG  0x95
#define BMP280_DIG_P5_LSB_REG  0x96
#define BMP280_DIG_P5_MSB_REG  0x97
#define BMP280_DIG_P6_LSB_REG  0x98
#define BMP280_DIG_P6_MSB_REG  0x99
#define BMP280_DIG_P7_LSB_REG  0x9A
#define BMP280_DIG_P7_MSB_REG  0x9B
#define BMP280_DIG_P8_LSB_REG  0x9C
#define BMP280_DIG_P8_MSB_REG  0x9D
#define BMP280_DIG_P9_LSB_REG  0x9E
#define BMP280_DIG_P9_MSB_REG  0x9F/* 复位值 */
#define BMP280_RESET_VALUE     0xB6/* 工作模式 */
#define BMP280_SLEEP_MODE      0x00
#define BMP280_FORCED_MODE     0x01
#define BMP280_NORMAL_MODE     0x03/* 过采样设置 */
#define BMP280_OSRS_T_SKIP     0x00 /* 温度测量跳过 */
#define BMP280_OSRS_T_X1       0x20 /* 温度测量1倍过采样 */
#define BMP280_OSRS_T_X2       0x40
#define BMP280_OSRS_T_X4       0x60
#define BMP280_OSRS_T_X8       0x80
#define BMP280_OSRS_T_X16      0xA0#define BMP280_OSRS_P_SKIP     0x00 /* 压力测量跳过 */
#define BMP280_OSRS_P_X1       0x04 /* 压力测量1倍过采样 */
#define BMP280_OSRS_P_X2       0x08
#define BMP280_OSRS_P_X4       0x0C
#define BMP280_OSRS_P_X8       0x10
#define BMP280_OSRS_P_X16      0x14/* 滤波器设置 */
#define BMP280_FILTER_OFF      0x00
#define BMP280_FILTER_COEF_2   0x04
#define BMP280_FILTER_COEF_4   0x08
#define BMP280_FILTER_COEF_8   0x0C
#define BMP280_FILTER_COEF_16  0x10/* 待机时间设置 */
#define BMP280_STANDBY_0_5_MS  0x00
#define BMP280_STANDBY_62_5_MS 0x20
#define BMP280_STANDBY_125_MS  0x40
#define BMP280_STANDBY_250_MS  0x60
#define BMP280_STANDBY_500_MS  0x80
#define BMP280_STANDBY_1000_MS 0xA0
#define BMP280_STANDBY_2000_MS 0xC0
#define BMP280_STANDBY_4000_MS 0xE0/* 结构体定义 */
struct bmp280_calib_param {uint16_t dig_T1;int16_t  dig_T2;int16_t  dig_T3;uint16_t dig_P1;int16_t  dig_P2;int16_t  dig_P3;int16_t  dig_P4;int16_t  dig_P5;int16_t  dig_P6;int16_t  dig_P7;int16_t  dig_P8;int16_t  dig_P9;
};#endif 

二、设备树文件中添加节点

  在设备树文件中的i2c1节点中添加子节点,如下:,其中bmp280的器件地址即reg定义为0x76,使用的引脚定义是SCL使用:MX6UL_PAD_UART4_TX_DATA__I2C1_SCL  SDA使用:MX6UL_PAD_UART4_RX_DATA__I2C1_SDA及完成设备树的节点定义,加载到内核:

 查看节点:

 三、驱动文件编写

(1)首先就是字符设备的结构体定义:添加了温度和气压数据。

struct bmp280_dev {dev_t devid;               // 设备号struct cdev cdev;          // 字符设备结构体struct class *class;       // 类struct device *device;     // 设备struct device_node *nd;    // 设备树节点int major;                 // 主设备号void *private_data;        // I2C客户端指针int temperature;           // 温度值(单位:0.01℃)int pressure;              // 气压值(单位:Pa)struct bmp280_calib_param calib_param; // 校准参数int32_t t_fine;            // 温度补偿中间值
};

 (2)接着完成对读寄存器函数的编写:,使用i2c_msg来加载数据,发送的数据是bmp280的器件地址以及加载要写入的数据,并调用i2c_transfer来完成数据发送

 (3)写寄存器:发送的数据位器件地址0x76+寄存器地址+数据

 (4)读取寄存器的校准参数:从0x88开始两个字节的读,

(5)计算实际的温度和气压值,参考数据手册给出的计算公式和实现例程代码,有兴趣的话可以仔细看看手册:

 仿照实现的计算函数:
 

static int32_t bmp280_compensate_temperature(struct bmp280_dev *dev, int32_t adc_T)
{int32_t var1, var2, temperature;var1 = ((((adc_T >> 3) - ((int32_t)dev->calib_param.dig_T1 << 1))) * ((int32_t)dev->calib_param.dig_T2)) >> 11;var2 = (((((adc_T >> 4) - ((int32_t)dev->calib_param.dig_T1)) * ((adc_T >> 4) - ((int32_t)dev->calib_param.dig_T1))) >> 12) * ((int32_t)dev->calib_param.dig_T3)) >> 14;dev->t_fine = var1 + var2;temperature = (dev->t_fine * 5 + 128) >> 8;return temperature;
}/** @description  : 计算实际气压值,避免使用64位除法* @param - dev  : bmp280设备* @param - adc_P: 原始气压ADC值* @return       : 实际气压值(Pa)*/
static uint32_t bmp280_compensate_pressure(struct bmp280_dev *dev, int32_t adc_P)
{int32_t var1, var2;uint32_t p;var1 = (((int32_t)dev->t_fine) >> 1) - (int32_t)64000;var2 = (((var1 >> 2) * (var1 >> 2)) >> 11) * ((int32_t)dev->calib_param.dig_P6);var2 = var2 + ((var1 * ((int32_t)dev->calib_param.dig_P5)) << 1);var2 = (var2 >> 2) + (((int32_t)dev->calib_param.dig_P4) << 16);var1 = (((dev->calib_param.dig_P3 * (((var1 >> 2) * (var1 >> 2)) >> 13)) >> 3) + ((((int32_t)dev->calib_param.dig_P2) * var1) >> 1)) >> 18;var1 = ((((32768 + var1)) * ((int32_t)dev->calib_param.dig_P1)) >> 15);if (var1 == 0)return 0; // 避免除零错误p = (((uint32_t)(((int32_t)1048576) - adc_P) - (var2 >> 12))) * 3125;if (p < 0x80000000)p = (p << 1) / ((uint32_t)var1);elsep = (p / (uint32_t)var1) * 2;var1 = (((int32_t)dev->calib_param.dig_P9) * ((int32_t)(((p >> 3) * (p >> 3)) >> 13))) >> 12;var2 = (((int32_t)(p >> 2)) * ((int32_t)dev->calib_param.dig_P8)) >> 13;p = (uint32_t)((int32_t)p + ((var1 + var2 + dev->calib_param.dig_P7) >> 4));return p;
}

(6) 实现读取数据函数定义:从数据寄存器0xf7读到0xfc

(7)剩下就是完成对bmp280初始化定义,设备文件的操作函数定义,如下所示即为,设备初始化代码:,主要完成对设备的唤醒,校准以及采样配置:

static int bmp280_init_sensor(struct bmp280_dev *dev)
{uint8_t chip_id;/* 1. 读取芯片ID,确认设备存在 */chip_id = bmp280_read_reg(dev, BMP280_ID_REG);printk("BMP280 Chip ID: 0x%02X\n", chip_id);if (chip_id != 0x58) {//0x76地址存储的是0x58printk("BMP280 Chip ID not matching, expected 0x58, got 0x%02X\n", chip_id);return -ENODEV;}/* 2. 软复位 */bmp280_write_reg(dev, BMP280_RESET_REG, BMP280_RESET_VALUE);mdelay(10); // 等待复位完成/* 3. 读取校准参数 */bmp280_read_calibration_data(dev);/* 4. 配置传感器 * - 设置过采样: 温度16x,压力16x* - 设置工作模式: 正常模式*/bmp280_write_reg(dev, BMP280_CONFIG_REG, BMP280_FILTER_COEF_16 | BMP280_STANDBY_500_MS);bmp280_write_reg(dev, BMP280_CTRL_MEAS_REG, BMP280_OSRS_T_X16 | BMP280_OSRS_P_X16 | BMP280_NORMAL_MODE);mdelay(100); // 等待配置生效return 0;
}

 如上即为主要关键驱动代码的定义。

四、编写驱动测试文件并编译测试

类似正点原子的驱动测试文件:

打开设备并读取到数据时,间隔2s打印一次数据

 编译加载驱动文件和驱动测试文件:

测试结果:

 

能够正确获取数据并进行打印,完成驱动!!!


总结

通过对IIC驱动的学习测试以及完成对具体iic设备的驱动来获取数据,对嵌入式linux的驱动了解更加深刻了。

http://www.xdnf.cn/news/5762.html

相关文章:

  • 3335. 字符串转换后的长度 I
  • Babylon.js学习之路《四、Babylon.js 中的相机(Camera)与视角控制》
  • MySQL基本查询
  • git经验
  • Electron-Vue3、Electron-React、Electron-Angular打造舆情监控系统项目
  • SimScape物理建模实例1--单质量-弹簧-阻尼系统
  • maxtext开源程序是一个简单、高性能和可扩展的 Jax LLM!
  • rsync
  • 2024年北理工Python123第六章测验题整理
  • 2094. 找出 3 位偶数
  • 稠密连接网络(DensoNet)
  • OFCMS代码审计-freemaker注入sql注入xxexss文件上传
  • Qt元对象系统总结
  • .NET10 - 尝试一下Open Api的一些新特性
  • 吴恩达机器学习笔记:监督学习
  • 【Python】Python常用控制结构详解:条件判断、遍历与循环控制
  • 015枚举之滑动窗口——算法备赛
  • 纽约时报发稿刊登案例:海外新闻媒体宣传如何赢得美国决策者
  • 每日一题——樱桃分级优化问题:最小化标准差的动态规划与DFS解决方案
  • 物理:海市蜃楼是宇宙背景辐射吗?
  • 【速写】use_cache参数与decode再探讨
  • 计算机网络笔记(二十四)——4.6互联网的路由选择协议
  • 基于STM32、HAL库的BMP390L气压传感器 驱动程序设计
  • Costmap代价地图
  • IOT藍牙探測 C2 架構:社會工程/節點分離防追尋
  • 2.1 微积分基本想法
  • ABP-Book Store Application中文讲解 - Part 2: The Book List Page - TBD
  • 案例-流量统计
  • 网格图之bfs
  • 水平翻转 垂直翻转 颜色处理