查看原文
其他

LiteOS裸机驱动移植04 | E53_IA1智慧农业扩展板驱动及使用

mculover666 小熊派开源社区 2021-02-01

(小熊派IoT开发套件全部教程   ↑点击启阅↑)


▷01 | 物联网一站式开发工具 IoT-Studio

02 | Hello World

▷03 | 任务管理

04 | 信号量(任务同步)

▷05 | 互斥锁(共享资源保护)

06 | 内存管理

01 | 以LED为例说明裸机驱动移植方法

02 | LCD驱动移植及使用

03 | E53_SC1智慧路灯扩展板驱动及使用

1.E53_IA1扩展板及其驱动

关于E53标准接口

E53接口标准的E取自扩展(Expansion)的英文首字母,板子的尺寸为5×3cm,故采用E53作为前缀来命名尺寸为 5×3cm 类型的案例扩展板,任何一款满足标准设计的开发板均可直接适配E53扩展板。

E53扩展板是根据不同的应用场景来设计的,以最大的程度在扩展板上还原真实应用场景,不同案例的扩展板根据不同的应用场景来命名后缀。例如:E53_SC1,SC是智慧城市(Smart City)的缩写,SC1表示的是智慧城市中的智慧路灯,再比如SC2则表示的是智慧城市中的智慧井盖。

E53扩展接口在电气特性上,包含了常用的物联网感知层传感器通信接口,比如5V、3.3V、GND、SPI、UART、IIC、ADC、DAC等等,可以适配各种传感器,还留有4个普通GPIO,如图:

E53接口电气特性

E53_IA1智慧农业扩展板

E53_IA1扩展板采用了E53标准接口,包含了一个补光灯,一个BH1750光照强度传感器,一个小的贴片电机,一个温湿度传感器SHT30,一个其中补光灯和贴片电机使用普通GPIO控制,BH1750和SHT30使用IIC接口通信。

E53_IA1扩展板

如果你对该扩展板板载的BH1750光照强度和温湿度传感器驱动不熟悉,请先阅读嵌入式基础教程:E53扩展板实验 —— 使用硬件I2C读写环境光强度传感器BH1750E53扩展板实验 —— 使用硬件I2C读写温湿度传感器SHT30

2. 移植E53_IA1驱动到LiteOS

如果你对移植裸机驱动到LiteOS的方法还不了解,请先阅读这篇文章:LiteOS裸机驱动移植教程01 | 以LED为例说明裸机驱动移植方法

复制裸机驱动文件到LiteOS工程

E53_IA1扩展板上的 BH1750 光照强度传感器和SHT30温湿度传感器使用的是 IIC 通信接口,所以除了复制 STM32CubeMX 生成的i2c.hi2c.h文件,还需要在此基础上复制包含了 BH1750 传感器驱动和SHT30传感器驱动的 E53_IA1 扩展板驱动文件。

在复制文件的时候,按照上一篇文章中所说的,复制i2c.h到Inc 文件夹,复制i2c.c到 Src 文件夹,再复制自己编写的驱动文件E53_IA1.cE53_IA1.h到 Hardware文件夹。

IoT-Studio中提供的默认工程已经复制好了这些文件,无需再次添加,如图:

E53_IA1扩展板驱动

添加驱动文件路径

因为 LiteOS 的整个项目工程使用 make 构建,所以复制驱动文件之后,需要添加驱动文件的路径到 makefile 中,加入编译。

project.mk文件指明了工程中所有文件的路径:

project.mk文件

在该文件中:

  • C文件路径

    • HARDWARE_SRC:对应Hardware文件夹下的Src文件夹

    • USER_SRC:对应Src文件夹

  • 头文件路径

    • HARDWARE_INC:对应Hardware文件夹下的Inc文件夹

    • USER_INC:对应Inc文件夹

如下,E53_IA1 驱动的底层 I2C 接口代码i2c.c路径添加到USER_SRC中:

I2C驱动代码

E53_IA1 驱动的底层 I2C 接口代码i2c.h路径添加到USER_INC中:

I2C驱动头文件路径

基于 I2C 驱动的 E53_IA1 驱动文件E53_IA1.c添加到HARDWARE_SRC中(默认未添加,需要手动添加):

E53_IA1驱动文件

基于 I2C 驱动的 E53_IA1 驱动文件E53_IA1添加到 HARDWARE_INC 中(默认未添加,需要手动添加):

E53_IA1驱动头文件路径

因为SC1和IA1的驱动中都包含BH1750的驱动,所以添加的时候需要注意去掉E53_SC1 的驱动文件E53_SC1.c和E53_SC1.h,否则会引起冲突。

至此,复制文件到LiteOS工程中,并将新复制的文件路径添加到makefile中,加入工程编译,就完成了驱动的移植。

3. E53_IA1裸机驱动的使用

初始化E53_IA1扩展板

在第一篇文章中详细的讲述了在LiteOS中初始化设备的两种方式:

  • 在系统启动调度之前初始化:设备在系统中随时可被任意任务使用

  • 在任务中初始化:设备一般只在该任务中被使用

本文中移植的 E53_IA1 扩展板驱动,不需要多个任务去操作,只需要传感器数据采集任务操作即可,所以初始化放在数据采集任务中

操作E53_IA1扩展板

接下来首先创建一个文件夹(如果已有,不用再次创建),用于存放本系列教程实验的代码:

新建文件夹
文件夹名称

在该文件夹中创建一个文件:

创建Demo文件

编写代码:

#include <osal.h>
#include "lcd.h"
#include "E53_IA1.h"

/* 存放E53_IA1扩展板传感器数据,可在E53_IA1.h中查看定义 */
E53_IA1_Data_TypeDef E53_IA1_Data;

/* 用于数据采集和数据处理任务间同步的信号量 */
osal_semp_t sync_semp;

/* 数据采集任务-低优先级 */
static int data_collect_task_entry()
{
    /* 初始化扩展板 */
    Init_E53_IA1();

    while (1)
    {
        /* 读取扩展板板载数据,存到数据结构体E53_IA1_Data中 */
        E53_IA1_Read_Data();

        /* 数据读取完毕,释放信号量,唤醒数据处理任务 */
        osal_semp_post(sync_semp);

        /* 任务睡眠2s */
        osal_task_sleep(2*1000);
    }
}

/* 数据处理任务-高优先级 */
static int data_deal_task_entry()
{
    /* lux- 当次数据,old-lux-上次数据 */
    int lux = 0, old_lux = 0;
    int temperature = 0, old_temperature = 0;;
    int humidity;

    /* LCD清屏,防止干扰显示 */
    LCD_Clear(WHITE);

    while (1)
    {
        /* 等待信号量,未等到说明数据还未采集,阻塞等待 */
        osal_semp_pend(sync_semp, cn_osal_timeout_forever);

        /* 信号量等待,被唤醒,开始处理数据 */
        //处理光照强度
        old_lux = lux;
        lux = (int)E53_IA1_Data.Lux;
        printf("BH1750 Value is %d\r\n", lux);
        LCD_ShowString(101002001616"BH1750 Value is:");
        LCD_ShowNum(140100, lux, 516);

        /* 光照阈值为1000,自动点亮或者熄灭路灯 */
        if(old_lux < 1000 && lux > 1000)
        {
            HAL_GPIO_WritePin(IA1_Light_GPIO_Port, IA1_Light_Pin, GPIO_PIN_RESET);
            printf("Light OFF!\r\n");
        }
        else if(old_lux > 1000 && lux < 1000)
        {
            HAL_GPIO_WritePin(IA1_Light_GPIO_Port, IA1_Light_Pin, GPIO_PIN_SET);
            printf("Light ON!\r\n");
        }

        //处理湿度数据
        humidity = E53_IA1_Data.Humidity;
        printf("Humidity is %d\r\n", humidity);
        LCD_ShowString(101202001616"Humidity: ");
        LCD_ShowNum(140120, humidity, 516);

        //处理温度数据
        old_temperature = temperature;
        temperature = E53_IA1_Data.Temperature;
        printf("Temperature is %d\r\n", temperature);
        LCD_ShowString(101402001616"Temperature: ");
        LCD_ShowNum(140140, temperature, 516);

        /* 温度阈值为30,自动开启或者关闭电机 */
        if(old_temperature < 30 && temperature >= 30)
        {
            HAL_GPIO_WritePin(IA1_Motor_GPIO_Port, IA1_Motor_Pin, GPIO_PIN_SET);
            printf("Motor ON!\r\n");
        }
        else if(old_temperature >= 30 && temperature < 30)
        {
            HAL_GPIO_WritePin(IA1_Motor_GPIO_Port, IA1_Motor_Pin, GPIO_PIN_RESET);
            printf("Motor OFF!\r\n");
        }
    }
}

/* 标准demo启动函数,函数名不要修改,否则会影响下一步实验 */
int standard_app_demo_main()
{
    /* 创建信号量 */
    osal_semp_create(&sync_semp, 10);

    /* 数据处理任务的优先级应高于数据采集任务 */
    osal_task_create("data_collect",data_collect_task_entry,NULL,0x400,NULL,3);
    osal_task_create("data_deal",data_deal_task_entry,NULL,0x400,NULL,2);
    return 0;
}

然后按照之前的方法,在 user_demo.mk 中将e51_ia1_driver_demo.c文件添加到makefile中,加入编译:

user_demo.mk文件

最后在.sdkconfig中配置开启宏定义:

.sdkconfig文件

编译,烧录,即可看到实验现象。

LCD屏幕上显示当前传感器采集的亮度值,温度值,湿度值,并且每2s更新一次。

当亮度值低于1000时,E53_IA1扩展板的补光灯自动点亮:

补光灯自动点亮

当亮度值高于1000时,E53_IA1扩展板的补光灯自动熄灭:

补光灯自动熄灭

在调节温度的时候,可以用手按着SHT30温湿度传感器,天干物燥,务必提前触摸一下金属物体,释放静电,防止静电破坏传感器!

当温度上升到30°的时候,电机自动启动,当温度降低到30°以下时,电机自动关闭。

另外,打开IoT-Studio自带的串口终端,可以查看到串口输出的工作信息:

linkmain:V1.2.1 AT 10:40:09 ON Dec  5 2019 

BH1750 Value is 237
Humidity is 19
Temperature is 28
WELCOME TO IOT_LINK SHELL

LiteOS:/>BH1750 Value is 1074
Light OFF!
Humidity is 19
Temperature is 29
BH1750 Value is 14086
Humidity is 20
Temperature is 30
Motor ON!
BH1750 Value is 284
Light ON!
Humidity is 21
Temperature is 30
BH1750 Value is 303
Humidity is 21
Temperature is 29
Motor OFF!
……


推荐阅读


《物联网创意大赛1.0》万元大奖等你来拿!!!

《物联网创意大赛1.0》一阶段:创意表达

《物联网创意大赛1.0》二阶段:动手实践

《物联网创意大赛1.0》三阶段:风采展示


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

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