一种无OS的MCU实用软件框架

来源 | 码云-魔罗技术

介绍

一种无OS的MCU实用软件框架,包括任务轮询管理,命令管理器、低功耗管理、环形缓冲区等实用模块。系统中广泛利用自定义段技术减少各个模块间的耦合关系,大大提供程序的可维护性。

主要功能

  • 支持模块自动化管理,并提供不同优先等级初始化声明接口。
  • 支持任务轮询管理,通过简单的宏声明即可实现,不需要复杂的声明调用。
  • 支持低功耗管理,休眠与唤醒通知。
  • 支持命令行解析,命令注册与执行。
  • blink设备支持,统一管理LED、震动马达、蜂鸣器

使用说明

完整的代码可以参考工程文件,系统开发平台如下:

MCU:STM32F401RET6

IDE:IAR 7.4或者Keil MDK 4.72A

任务初始化及任务轮询管理(module)

使用此模块前需要系统提供滴答定时器,用于驱动任务轮询作业。(参考platform.c)

//定时器中断(提供系统滴答)
void SysTick_Handler(void)
{
    systick_increase(SYS_TICK_INTERVAL); //增加系统节拍
}

注册初始化入口及任务(参考自key_task.c)

static void key_init(void){    /*do something*/}

static void key_scan(void){    /*do something*/}

module_init('key', key_init);              //注册按键初始化接口driver_register('key', key_scan, 20);      //注册按键任务(20ms轮询1次)

命令管理器(cli)

适用于在线调试、参数配置等(参考使用cli_task.c),用户可以通过串口输出命令行控制设备行为、查询设备状态等功能。

命令格式

cli支持的命令行格式如下:

<cmd name> < param1> < param2> < paramn> < \r\n > <cmd name> ,< param1>, < param2>, < paramn>, < \r\n >

每行命令包含一个命令名称+命令参数(可选),命令名称及参数可以通过空格或者','进行分隔。

系统默认命令

cli系统自带了2条默认命令,分别是'?'与'help'命令,输入他们可以列出当前系统包含的命令列表,如下所示:

?         - alias for 'help'
help      - list all command.
pm        - Low power control command
reset     - reset system
sysinfo   - show system infomation.

适配命令管理器

完整的例子可以参考cli_task.c.

static cli_obj_t cli;                               /*命令管理器对象 */

/*  * @brief       命令行任务初始化 * @return      none */ static void cli_task_init(void){    cli_port_t p = {tty.write, tty.read};           /*读写接口 */

    cli_init(&cli, &p);                             /*初始化命令行对象 */

    cli_enable(&cli);

    cli_exec_cmd(&cli,'sysinfo');                   /*显示系统信息*/}

/*  * @brief       命令行任务处理 * @return      none */ static void cli_task_process(void){    cli_process(&cli);}

module_init('cli', cli_task_init);                  task_register('cli', cli_task_process, 10);          /*注册命令行任务*/

命令注册

以复位命令为例(参考cmd_devinfo.c):

#include 'cli.h'
//...
/* 
 * @brief       复位命令
 */ 
int do_cmd_reset(struct cli_obj *o, int argc, char *argv[])
{
    NVIC_SystemReset();
    return 0;
}cmd_register('reset',do_cmd_reset, 'reset system');

低功耗管理器(pm)

控制间歇运行,降低系统功耗。其基本的工作原理是通过轮询系统中各个模块是否可以允许系统进入低功耗。实际上这是一种判决机制,所有模块都具有有票否决权,即只要有一个模块不允许休眠,那么系统就不会进入休眠状态。pm模块在休眠前会统计出各个模块会返回最小允许休眠时长,并以最小休眠时长为单位进行休眠。

如何适配

使用前需要通过pm_init进行初始化适配,并提供当前系统允许的最大休眠时间,进入休眠的函数接口,基本的接口定义如下:

/*低功耗适配器 ---------------------------------------------------------*/typedef struct {    /**     * @brief    系统最大休眠时长(ms)     */      unsigned int max_sleep_time;    /**     * @brief     进入休眠状态     * @param[in] time - 期待休眠时长(ms)     * @retval    实际休眠时长     * @note      休眠之后需要考虑两件事情,1个是需要定时起来给喂看门狗,否则会在休眠     *            期间发送重启.另外一件事情是需要补偿休眠时间给系统滴答时钟,否则会     *            造成时间不准。     */         unsigned int (*goto_sleep)(unsigned int time);}pm_adapter_t;void pm_init(const pm_adapter_t *adt);

void pm_enable(void);

void pm_disable(void);

void pm_process(void);

完成的使用例子可以参考platform-lowpower.c,默认情况下是禁用低功耗功能的,读者可以去除工程中原来不带低功耗版本的platform.c,并加入platform-lowpower.c文件进行编译即可使用。

注册低功耗设备

以按键扫描为例,正常情况下,如果按键没有按下,那么系统休眠可以进入休眠状态,对按键功能是没有影响的。如果按键按下时,那么系统需要定时唤醒并轮询按键任务。

所以在一个低功耗系统下,为了不影响按键实时性需要处理好两个事情:

  1. 系统休眠状态下,如果有按键按下,那系统系统应立即唤醒,以便处理接下来的扫描工作。
  2. 如果按键按下时,系统可以进入休眠,但需要定时唤醒起来轮询按键任务。

对于第一种情况,将按键配置为边沿中断唤醒即可,以STM32F4为例(参考key_task.c),它支持外部中断唤醒功能。

/* 
 * @brief       按键 io初始化
 *              PC0 -> key;
 * @return      none
 */ 
static void key_io_init(void)
{
    /* Enable GPIOA clock */
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);

gpio_conf(GPIOC, GPIO_Mode_IN, GPIO_PuPd_UP, GPIO_Pin_0);
    
    //低功耗模式下,为了能够检测到按键,配置为中断唤醒
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);
    SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOC, EXTI_PinSource0);
    exti_conf(EXTI_Line0, EXTI_Trigger_Falling, ENABLE);
    nvic_conf(EXTI0_IRQn, 0x0F, 0x0F);
    
    key_create(&key, readkey, key_event);            /*创建按键*/
}

对于第二种情况,可以通过pm_dev_register来处理,当系统请求休眠时,如果此时按键按下,则返回下次唤醒时间即可,如下面的例子所示。

//参考key_task.c#include 'pm.h'                                     /* * @brief    休眠通知 */static unsigned int  key_sleep_notify(void){    return key_busy(&key) || readkey() ? 20 : 0;    /* 非空闲时20ms要唤醒1次*/} pm_dev_register('key', NULL, key_sleep_notify, NULL);

blink模块

具有闪烁特性(led, motor, buzzer)的设备(led, motor, buzzer)管理

使用步骤:

  • 需要系统提供滴答时钟,blick.c中是通过get_tick()接口获取,依赖module模块
  • 需要在任务中定时进行轮询

或者通过'module'模块的任务注册来实现

task_register('blink', blink_dev_process, 50);  //50ms轮询1次

LED驱动

blink_dev_t led;                             //定义led设备

/* *@brief     红色LED控制(GPIOA.8) *@param[in] on - 亮灭控制 */static void led_ctrl(int on){    if (on)        GPIOA->ODR |= (1 << 8);    else         GPIOA->ODR &= ~(1 << 8);}

/* *@brief     led初始化程序 */void led_init(void){    led_io_init(void);                  //led io初始化    blink_dev_create(&led, led_ctrl);   //创建led设备

    blink_dev_ctrl(&led, 50, 100, 0);   //快闪(50ms亮, 100ms灭)}

按键管理模块

类似blink模块,使用之前有两个注意事项:

  • 需要系统提供滴答时钟,key.c中是通过get_tick()接口获取,依赖module模块
  • 需要在任务中定时进行轮询
key_t key;                             //定义按键管理器

/*
 *@brief     按键事件
 *@param[in] type     - 按键类型(KEY_PRESS, KEY_LONG_DOWN, KEY_LONG_UP)  
 *@param[in] duration - 长按持续时间
 */
void key_event(int type, unsigned int duration)
{
 if (type == KEY_PRESS) {                //短按
   
 } else if (type == KEY_LONG_DOWN) {     //长按
  
 }
}

//读取键值(假设按键输出口为STM32 MCU PA8)
int read_key(void)
{
 return GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8) == Bit_RESET;
}

/*
 *@brief     按键初始化
 */
void key_init(void)
{
    //打开GPIO 时钟
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
 //配置成输入模式
    gpio_conf(GPIOA, GPIO_Mode_IN, GPIO_PuPd_NOPULL, GPIO_Pin_8); 
    //创建1个按键
    key_create(&key, read_key, key_event);  
}

克隆链接:

git clone https://gitee.com/moluo-tech/CodeBrick.git
(0)

相关推荐

  • 单片机------ADC------39---摇杆

    / ADC摇杆 ROCKER_Ser.h #ifndef Rocker_Ser #define Rocker_Ser #define Rocker_KEY_NO 0 #define Rocker_KE ...

  • 一个学妹写的按键检测函数把我秀翻了!

    摘要:今年实验室来了三个学妹,其中一个学妹以前是物联网专业的,进了实验室老师二话没说:先把STM32单片机过一遍,有啥问题就找小师弟.还好单片机小师弟会玩一点点,玩的也不好,所以一起学习吧! 上来第一 ...

  • 手写cli

    惠善一的博客:http://huishanyi.club/ 通过命令行工具,初始化团队项目,并生成团队规范代码,一键创建项目,一键生成代码,一键生成功能模块··· 解放双手,从 cli 开始, JSe ...

  • 【新提醒】HC32L072开发板之 KEY控制LED

    开发板上LED与KEY的原理图如图1所示,可以用SW1和SW2分别控制DS1和DS2.其程序如下: 图1  LED与KEY原理图 实现控制功能的程序如下: /********************* ...

  • 原来搞单片机也可以面向对象

    摘要:在看别人单片机程序时,你也许是奔溃的,因为全局变量满天飞,不知道哪个在哪用了,哪个表示什么,而且编写极其不规范.自己写单片机程序时,也许你也是奔溃的.总感觉重新开启一个项目,之前的写过相似的代码 ...

  • stm32按键单击,双击,长按

    owlcity4发表于 2020-7-22 17:17|显示全部楼层|阅读模式 给arm初学者一个比较实用的多功能按键,实现单击,双击,长按.声明一下:这是我修改的例程.如果写到TIM3时间中断里不太 ...

  • 低耦合高内聚的MCU实用软件框架

    不知道大家有没有听过软件设计中的低耦合,高内聚的两个原则. 具体是什么意思呢? 在一个项目中:每个模块之间相联系越紧密,则耦合性越高:这样你改动其中一个模块,其他模块也需要一起改动,换言之:牵一发而动 ...

  • 田英章《波若波罗密多心经》中写了21种'无'字,还说没有变化?

    田英章<波若波罗密多心经>中写了21种'无'字,还有网友说没有变化? 书法本来就有很多种型态,结体!写的再好也人说三道四!写的再差的人也有人鼓掌称好!没有必要追求每一个人都统一,也不讨好每 ...

  • 从哲学的意义上,人应当过一种无目的的生活

    图片来源于影片<柏林苍穹下> 既然生命无意义,人就不必设定目标.无论是具体的目标还是抽象的目标,世俗的目标还是虚幻的目标.从哲学的意义上,人应当过一种无目的的生活. --李银河 如果不为自 ...

  • 一种无外接电源即热水龙头的制作方法

    本实用新型涉及管道类技术领域,具体是一种无外接电源即热水龙头. 背景技术: 在家庭厨房以及浴室水龙头使用时,为了便于热水的直接使用,一般都采用水龙头加热器,市场上的这种加热器都需要外接电源,一方面耗费 ...

  • 丛文俊:今天的书法,是一种无精神的“展览体”

    丛文俊说过一句话"任何试图脱离传统的努力和标新立异,都不会长久",这句话很适用于今天鼓噪的艺术界.就像他在今天文章中说的"展览体书法于无形中丢掉很多东西",这其 ...

  • 无拉手柜门怎么开?看了这9种无拉手设计,佩服设计师的脑力!

    定制收纳柜,是现代家庭装修中不可或缺的设计,它为我们提供了杂物的存放与收纳位置,保障了我们居家空间的整洁美观性.在定制柜的时候,我们常常是把重点放在柜体的风格外观与柜子内部格局设计上,像柜门拉手这样细 ...

  • 信念# 五种无意识限制自己的方式

    ​[cp]#信念# 五种无意识限制自己的方式 1.制约 我们从父母那里得到的想法.态度来限制了自己,关于金钱和成功,甚至关于人生到底是困难的还是简单的.... 2.关于对自己的定义 例如:我不够好,我 ...

  • 这5种无糖水果,糖尿病可以多吃,而且还能降血糖

    在近20年来,糖尿病患者越来越多,无糖食品也是应运而生,有很多水果也被赋予了"无糖水果"的称号,但水果界这么多品种,哪些可以称之为无糖水果呢? 1.番石榴: 番石榴含较高的维生素A ...

  • 8种无器械健身动作,简单易上手,随时都能练!

    方便在家做的无器械健身动作 除了俯卧撑.深蹲 还有什么呢? 下面就给大家分享 8种居家无器械健身动作 动作1 反向拉背 ▼ 动作2 搁腿箭步跳 ▼ 动作3 双臂撑控腿 ▼ 动作4 双臂屈伸 ▼ 动作5 ...