一种减少各个模块间耦合关系的方案

介绍

一种无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);
}

开源地址:

https://gitee.com/moluo-tech/CodeBrick

(0)

相关推荐

  • 手写cli

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

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

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

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

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

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

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

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

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

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

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

  • 体质间互联关系与体质泡浴(附每种偏颇体质的泡浴搭配方案)

    质 九种体质间的互联关系 当多种偏颇体质兼顾时,如何抓住主要体质? 中医有一句话:血为气之母,气为血之帅.气乃人之生存根本,气存则生,气断则死.其中的"气"指的是什么呢? 我们知道 ...

  • PNAS:大脑区域间耦合的增加和减少会相应增加和减少人类大脑中的振荡活动

    大脑中振荡活动的起源目前仍有争议,但许多假说都认为它们反映了大脑区域之间的相互作用.本文中,我们通过控制两个人类大脑区域之间的耦合强度来检验这种可能性,这两个区域是腹侧前运动皮层(ventral pr ...

  • 与儿女间的关系,能达到这种境界的,均是高手!

    人到了老年,退出了工作的舞台,家庭就成了生活的重心.这时候,如何与儿女们处好关系,就显得尤为重要.在当下这个时代,看看周围那些跟儿女们关系处的好的,其实无非就是做到了这两个字:放手!而且能达到这种境界 ...

  • 关于 血压间的关系,认真看,很有益处!

    版权声明  图文综合来自网络 版权归原作者所有,如有侵权请联系删除 「老同学」M先生综合编辑 关于血压间的关系 认真看,很有益处! (放到你圈子里,朋友们会感激您)  ◆  ◆  ◆ 健康年轻人的血压 ...

  • 逆天了,由报表间内在关系发现财报造假的惊心秘密!

    财务报表中主表就一张,即资产负债表,利润表.现金流量表都是资产负债表的附表. 为什么说资产负债表是财务报告中唯一的主表呢? 因为第一,要是没有利润表,可以通过对资产负债表中的净资产期末数与期初数进行比 ...

  • ​5种方式缓解腰间盘突

    得了腰间盘突出,有时候确实让很多朋友痛苦万分,甚至有时候产生了绝望的念头.对于这种日常生活中比较常见的病痛之一,要保持乐观的心态去面对,对于腰痛问题也可以了解一下太一斋,今天就给大家分享5个方法来缓解 ...

  • 家庭关系是孩子情绪的隐形控制器:这四种不健康的家庭三角关系,伤孩子最深

    作者:夏小雨 原创首发:萨提亚育儿(ID:satiryuer) 01 在网上,看到一位网友的故事,因为从小父母不和,经常吵架,他渐渐的变得胆小怕事,不善言谈,成绩也不好. 他说:我觉得很绝望,我看不到 ...

  • 既然Power Pivot表间建关系后可直接调数据,还要Lookupvalue这个函数干嘛?

    小勤:在Power Pivot里对两个表建立了关系之后,就可以直接调用相关表的数据了,那还要Related或Lookupvalue这个函数干嘛呢? 大海:这个问题好难回答.不过,举个实际工作中的例子吧 ...

  • 评分和票房,大概是好基友间的关系?

    本文首发于电影界 本文字数:1819 阅读时间:6分钟 正在上映的<无问西东>,上演了一场豆瓣评分逆袭奇迹,从上映后的6.5分到6.2分再到7.2分.7.3分.7.4分. 围绕分数变动的同 ...