查看原文
其他

【RT-Thread笔记】BH1750软件包的使用

ZhengNL 嵌入式大杂烩 2021-01-31

点击上方「嵌入式大杂烩」,选择「置顶公众号」第一时间查看嵌入式笔记!

BH1750简介

BH1750是一种用于两线制串行总线接口的16位数字型光强度传感器集成电路。利用它的高分辨率可以探测较大范围的光强度变化。1lx~65535lx)。

创建工程、验证

在RT-Thread中读取BH1750数据有两种方法:一种方法是借助bh1750软件包另一种方法是直接使用i2c驱动框架读取BH1750数据。

关于直接使用i2c驱动框架读取BH1750数据的方法可以阅读Mculover666兄的这篇:

https://blog.csdn.net/Mculover666/article/details/104675712

本次实验我们借助bh1750软件包来读取BH1750传感器数据,使用RT-Thread Studio V1.1.0来创建工程。

1、添加bh1750软件包

2、打开i2c设备驱动

目前只有软件i2c驱动。保存RT-Thread Settings文件。编译报错:


那是因为我们没有打开I2C相关的宏,drv_soft_i2c.c中模拟了几个i2c,我们要确认我们使用哪一个i2c与bh1750传感器相连。


这里可以看到bh1750的示例的初始化函数中使用了i2c2,所以相应的我们需要在board.h中打开i2c2相关的宏,需要打开、修改哪些宏可以看相关注释:


这里我们使用的是小熊派开发板,bh1750与MCU通过PB6、PB7引脚相连:


虽然PB6、PB7可以配置为硬件i2功能,但是我们这里使用的是软件i2c,所以这里的PB6、PB7是当做gpio来用的。然后我们根据注释的说明把代码改为:


然后编译报错,错误提示这几个宏有问题。反反复复检查,好像没什么问题,为啥就会疯狂报错。隐约记得有些例程例程中表示引脚好像不是这么表示的,而是类似这样的:


每个引脚都有一个新的代号,而这些引脚与代号的关系可以drv_gpio.c中查看:


可以看到我们的PB6、PB7引脚的代号分别是22、23。然后尝试着把上面的i2c宏代码改为:


编译成功!然后试着读取传感器数据,也成功了。所以,这大概是RT-Thread Studio V1.1.0的一个小bug,模板工程的board.h里关于i2c的注释有问题,严重误导了我们。。

3、下载、验证


若执行sensor read命令无数据输出时,需要打开\components\drivers\sensors\sensor_cmd.c,在sensor_show_data函数后面自行增加环境光照强度打印代码:

case RT_SENSOR_CLASS_LIGHT:
        LOG_I("num:%3d, light:%4d.%d, timestamp:%5d", num, sensor_data->data.light / 10, sensor_data->data.light % 10, sensor_data->timestamp);
        break;

4、编写应用

上面能输入那些命令对bh1750进行测试的前提是官方已经给我们写好了相关应用demo,在sensor_cmd.c中,如:

(1)测试函数

(2)显示数据


除此之外还有其它几个应用相关的函数。

我们也可以模仿sensor_cmd.c里面的代码来写我们自己的应用代码:

左右滑动查看全部代码>>>

static void bh1750_thread_entry(void *parameter)
{
    rt_device_t dev = RT_NULL;
    struct rt_sensor_data data;
    rt_size_t res;

    /* 查找bh1750传感器  */
    dev = rt_device_find("li_bh1750");
    if (dev == RT_NULL)
    {
        rt_kprintf("Can't find device:li_bh1750\n");
        return;
    }

    /* 以只读模式打开bh1750 */
    if (rt_device_open(dev, RT_DEVICE_FLAG_RDONLY) != RT_EOK)
    {
        rt_kprintf("open device failed!");
        return;
    }

    while(1)
    {
        /* 从传感器读取一个数据 */
        res = rt_device_read(dev, 0, &data, 1);
        if (1 != res)
        {
            rt_kprintf("read data failed!size is %d", res);
        }
        else
        {
            rt_kprintf("light:%4d.%d lux\n", data.data.light / 10, data.data.light % 10);
        }
        rt_thread_mdelay(1000);
    }
}

int bh1750_example(void)
{
    rt_thread_t tid;    /* 线程句柄 */

    tid = rt_thread_create("bh1750_thread",
                            bh1750_thread_entry,
                            RT_NULL,
                            1024,
                            20,
                            10);
   if(tid != RT_NULL)
   {
        /* 线程创建成功,启动线程 */
        rt_thread_startup(tid);
   }

   return 0;
}

/* 导出到 msh 命令列表中 */
MSH_CMD_EXPORT(bh1750_example, bh1750 example);


运行结果:


使用RT-Thread提供给我的I/O设备管理接口rt_device_find、rt_device_open、rt_device_read、rt_device_close来编写应用。相关框图:


关于RT-Thread的I/O设备框架的介绍可阅读往期笔记:【RT-Thread笔记】IO设备模型及PIN设备

最后

以上就是本次的分享,如有错误,欢迎指出!

原创不易,期待您的在看、转发。感谢支持!


猜你喜欢:

C语言、嵌入式重点知识:回调函数

C语言、嵌入式位操作精华技巧大汇总

【Linux笔记】设备树实例分析

【Linux笔记】通俗易懂的Linux驱动基础

【Linux笔记】pc机_开发板_ubuntu互ping实验

学习STM32的一些经验分享

基于LiteOS的智慧农业案例实验分享

从单片机工程师的角度看嵌入式Linux

应届生求职的那些事

基于GUILite的简易万年历

后台回复:加群。添加ZhengN微信,加入交流群

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存