Files
Smart-granary-code/components/bh1750/bh1750.c
Wang Beihong ffdb7065e3 功能:集成SU-03T语音模块,完善UI代码文档
- 在CMakeLists.txt中添加SU-03T语音模块依赖。
- 在main.cpp中实现SU-03T接收回调函数,处理接收消息。
- 完善各UI源文件文档,包括动作、屏幕和字体,明确模块作用与数据流向。
- 更新主应用逻辑,初始化并启动SU-03T接收器。
- 修改过程中确保兼容性,保留原有接口。
2026-04-22 01:06:10 +08:00

186 lines
5.7 KiB
C

/*
* 文件: components/bh1750/bh1750.c
* 角色: BH1750 光照传感器驱动封装
* 说明:
* - 本文件用于实现当前模块的核心功能或接口定义。
* - 修改前请先确认该模块与其它任务/外设之间的数据流关系。
* - 涉及协议与硬件时,优先保持现有接口兼容,避免联调回归。
*/
/*
* SPDX-FileCopyrightText: 2015-2025 Espressif Systems (Shanghai) CO LTD
*
* SPDX-License-Identifier: Apache-2.0
*/
#include "bh1750.h"
#include "driver/i2c_master.h"
#include "freertos/FreeRTOS.h"
#include "freertos/projdefs.h" // for pdMS_TO_TICKS
#define BH_1750_MEASUREMENT_ACCURACY 1.2 /*!< BH1750 传感器的典型测量精度因子 */
#define BH1750_POWER_DOWN 0x00 /*!< 设置为掉电模式的命令 */
#define BH1750_POWER_ON 0x01 /*!< 设置为上电模式的命令 */
#define I2C_CLK_SPEED 400000 /*!< I2C 通信时钟频率 (400kHz) */
/**
* @brief BH1750 设备私有结构体
*/
typedef struct {
i2c_master_dev_handle_t i2c_handle; /*!< I2C 主设备句柄 */
} bh1750_dev_t;
/**
* @brief 向 BH1750 写入一个字节的辅助函数
*/
/* 函数: bh1750_write_byte
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
static esp_err_t bh1750_write_byte(const bh1750_dev_t *const sens, const uint8_t byte)
{
return i2c_master_transmit(sens->i2c_handle, &byte, 1, pdMS_TO_TICKS(1000));
}
/**
* @brief 创建并注册 BH1750 设备
*
* @param i2c_bus I2C 总线句柄
* @param dev_addr 传感器 I2C 地址 (一般为 0x23 或 0x5C)
* @param handle_ret [out] 返回创建好的设备句柄
* @return esp_err_t 成功返回 ESP_OK
*/
/* 函数: bh1750_create
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_create(i2c_master_bus_handle_t i2c_bus, const uint8_t dev_addr, bh1750_handle_t *handle_ret)
{
esp_err_t ret = ESP_OK;
bh1750_dev_t *sensor = (bh1750_dev_t *) calloc(1, sizeof(bh1750_dev_t));
if (!sensor) {
return ESP_ERR_NO_MEM;
}
// 配置并添加新的 I2C 设备到总线
const i2c_device_config_t i2c_dev_cfg = {
.device_address = dev_addr,
.scl_speed_hz = I2C_CLK_SPEED,
};
ret = i2c_master_bus_add_device(i2c_bus, &i2c_dev_cfg, &sensor->i2c_handle);
if (ret != ESP_OK) {
free(sensor);
return ret;
}
assert(sensor->i2c_handle);
*handle_ret = sensor;
return ret;
}
/**
* @brief 删除 BH1750 设备并释放资源
*/
/* 函数: bh1750_delete
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_delete(bh1750_handle_t sensor)
{
bh1750_dev_t *sens = (bh1750_dev_t *) sensor;
if (sens->i2c_handle) {
i2c_master_bus_rm_device(sens->i2c_handle);
}
free(sens);
return ESP_OK;
}
/**
* @brief 进入掉电模式(低功耗)
*/
/* 函数: bh1750_power_down
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_power_down(bh1750_handle_t sensor)
{
bh1750_dev_t *sens = (bh1750_dev_t *) sensor;
return bh1750_write_byte(sens, BH1750_POWER_DOWN);
}
/**
* @brief 唤醒并进入上电模式
*/
/* 函数: bh1750_power_on
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_power_on(bh1750_handle_t sensor)
{
bh1750_dev_t *sens = (bh1750_dev_t *) sensor;
return bh1750_write_byte(sens, BH1750_POWER_ON);
}
/**
* @brief 设置测量时间倍率 (MTreg)
* 用于改变传感器的测量灵敏度
*/
/* 函数: bh1750_set_measure_time
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_set_measure_time(bh1750_handle_t sensor, const uint8_t measure_time)
{
bh1750_dev_t *sens = (bh1750_dev_t *) sensor;
uint32_t i = 0;
uint8_t buf[2] = {0x40, 0x60}; // MTreg 常量部分
buf[0] |= measure_time >> 5;
buf[1] |= measure_time & 0x1F;
for (i = 0; i < 2; i++) {
esp_err_t ret = bh1750_write_byte(sens, buf[i]);
if (ESP_OK != ret) {
return ret;
}
}
return ESP_OK;
}
/**
* @brief 设置测量模式(连续测量或单词测量,以及分辨率选择)
*/
/* 函数: bh1750_set_measure_mode
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_set_measure_mode(bh1750_handle_t sensor, const bh1750_measure_mode_t cmd_measure)
{
bh1750_dev_t *sens = (bh1750_dev_t *) sensor;
return bh1750_write_byte(sens, (uint8_t)cmd_measure);
}
/**
* @brief 获取测量结果
*
* @param sensor 设备句柄
* @param data [out] 返回转换后的光照强度值 (单位: Lux)
* @return esp_err_t
*/
/* 函数: bh1750_get_data
* 作用: 执行模块内与函数名对应的业务逻辑。
* 重点: 关注输入合法性、返回码与并发安全。
*/
esp_err_t bh1750_get_data(bh1750_handle_t sensor, float *const data)
{
bh1750_dev_t *sens = (bh1750_dev_t *) sensor;
uint8_t read_buffer[2];
// 从 I2C 读取 2 字节原始数据
esp_err_t ret = i2c_master_receive(sens->i2c_handle, read_buffer, sizeof(read_buffer), pdMS_TO_TICKS(1000));
if (ESP_OK != ret) {
return ret;
}
// 将原始数据转换为 Lux (公式: (高8位 << 8 | 低8位) / 1.2)
*data = (( read_buffer[0] << 8 | read_buffer[1] ) / BH_1750_MEASUREMENT_ACCURACY);
return ESP_OK;
}