硬件——STM32 , 软件框架

单片机应用程序的框架大概有三种:

1,简单的前后台顺序执行程序.

2,时间片轮询法.

3,应用操作系统.

下面我们主要来讲解时间片轮询法:

在这里我们先介绍一下定时器的复用功能。就是使用1个定时器,可以是任意的定时器,这里不做特殊说明,下面假设有3个任务,那么我们应该做如下工作:

1. 初始化定时器,这里假设定时器的定时中断为1ms(当然你可以改成10ms,这个和操作系统一样,中断过于频繁效率就低,中断太长,实时性差)。

2. 定义一个数值:

#define TASK_NUM   (3)                       //  这里定义的任务数为3,表示有三个任务会使用此定时器定时

uint16 TaskCount[TASK_NUM] ;           //  这里为三个任务定义三个变量来存放定时值,用于存放每个任务需要的时间

uint8  TaskMark[TASK_NUM];               //  同样对应三个标志位,为0表示时间没到,为1表示定时时间到

3. 在定时器中断服务函数中添加:

/**************************************************************************************

* FunctionName : TimerInterrupt()

* Description : 定时中断服务函数

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void TimerInterrupt(void)                   //定时器溢出之后就会进入到这里,可以是1ms,也可以是10ms

{

uint8 i;

for (i=0; i<TASKS_NUM; i++)        //分别去查看每个任务,这里TASKS_NUM是表示任务的个数,这里是3个

{

if (TaskCount[i])                         //TaskCount[i],这个是代表第i个任务当前延迟的时间,如果没有到延迟时间到0,就继续延迟

{

TaskCount[i]--;                     //减少一个刻度

if (TaskCount[i] == 0)           //如果TaskCount[i]为0,说明我们指定的时间到了

{

TaskMark[i] = 0x01;       //TaskMark[i],表示第i个任务的延迟已经完成,可以执行了

}

}

}

}

4. 在我们的应用程序中,在需要的应用  [延迟]  的地方添加如下代码:

TaskCount[0] = 20;        // 延时20ms,比如按键程序,需要我们每20ms检测一次,并不需要实时查看

TaskMark[0]  = 0x00;     // 启动此任务的定时器

到此我们只需要在任务中判断TaskMark[0] 是否为0x01即可。其他任务添加相同,至此一个定时器的复用问题就实现了。在等待一个定时的到来的同时我们可以循环判断标志位,同时也可以去执行其他函数。  [即:在一个函数运行的间隙中我们可以运行其他的函数]

那么如果我们在一个函数延时的时候去执行其他函数,充分利用CPU时间,是不是和操作系统有些类似了呢?但是操作系统的任务管理和切换是非常复杂的。下面我们就将利用此方法架构一个新的应用程序。

时间片轮询法的架构:

1.设计一个结构体:这个是定义的一个  模子, 用于扣蛋糕用的,  实际上是定义任务包括几个要素,我们把任务的要素都列出来,

这样就可以使每个任务都有规定格式的要素了

typedef struct _TASK_COMPONENTS      //任务结构体,任务的几个要素  [这个是很重要的]

{

uint8 Run;                                               // 程序运行标记:0-不运行,1-运行  ,这里是程序会不会运行

uint8 Timer;                                            // 计时器, 这个是用于在定时器中减减的

uint8 ItvTime;                                          // 任务运行间隔时间, 这个是用于更新参数的

void (*TaskHook)(void);                          // 要运行的任务函数

} TASK_COMPONENTS;                           // 任务定义

2. 任务运行标志出来,此函数就相当于中断服务函数,需要在定时器的中断服务函数中调用此函数,这里独立出来,便于移植和理解。

/**************************************************************************************

* FunctionName   : TaskRemarks()          //放到定时器中

* Description    : 任务标志处理

* EntryParameter : None

* ReturnValue    : None

**************************************************************************************/

void TaskRemarks(void)                            //要放到  timer中断中

{

uint8 i;

for (i=0; i<TASKS_MAX; i++)                 // 逐个任务时间处理

{

if (TaskComps[i].Timer)                     // 时间不为0 , 可是我有一个问题, 什么时候用 . 什么时候用 ->

{

TaskComps[i].Timer--;                   // 减去一个节拍

if (TaskComps[i].Timer == 0)         // 时间减完了

{

TaskComps[i].Timer = TaskComps[i].ItvTime;       // 恢复计时器值,从新下一次

TaskComps[i].Run = 1;            // 任务可以运行, 任务的延迟已经过了, 可以继续运行了

}

}

}

}

3. 任务处理:实现任务管理操作, 用于循环判断哪个任务需要运行了

/**************************************************************************************

* FunctionName   : TaskProcess()

* Description    : 任务处理

* EntryParameter : None

* ReturnValue    : None

**************************************************************************************/

void TaskProcess(void)

{

uint8 i;

for (i=0; i<TASKS_MAX; i++)           // 逐个任务时间处理

{

if (TaskComps[i].Run)                 // 这个标志不为0, 标志为0, 标志置1是在timer时间中断中, 是在时间查询那里的

{

TaskComps[i].TaskHook();      // 运行任务, 这个是一个函数, 在运行完这个函数后, 我们下一步操作就是清除标志位

TaskComps[i].Run = 0;           // 标志清0

}

}

}

假设我们有三个任务:时钟显示,按键扫描,和工作状态显示

1. 定义一个上面定义的那种结构体变量:

/**************************************************************************************

* Variable definition

**************************************************************************************/

static TASK_COMPONENTS TaskComps[] =

{   //0:程序刚开始不运行;  60:用于减减的计数;  60:函数重复周期,用于重装减减;  TaskDisplayClock:是一个函数名,用于重复运行的刷屏;

{0, 60, 60, TaskDisplayClock},           // 显示时钟

{0, 20, 20, TaskKeySan},                   // 按键扫描

{0, 30, 30, TaskDispStatus},              // 显示工作状态

// 这里添加你的任务。。。。

};

在定义变量时,我们已经初始化了值,这些值的初始化,非常重要,跟具体的执行时间优先级等都有关系,这个需要自己掌握。

上面的结构体初始化表示,我们有三个任务:

①每1s执行一下时钟显示,因为我们的时钟最小单位是1s,所以在秒变化后才显示一次就够了。

②由于按键在按下时电平会抖动,而我们知道一般按键的抖动大概是20ms,那么我们在顺序执行的函数中一般是延迟20ms等待电平稳定后才开始采集电平,而这里我们每20ms扫描一次,效果是非常不错的,即达到了消抖的目的,也不会漏掉按键输入。

③为了能够显示按键后的其他提示和工作界面,我们这里设计每30ms显示一次,如果你觉得反应慢了,你可以让这些值小一点。

TaskDisplayClock, TaskKeySan,  TaskDispStatus 后面的名称是对应的函数名,你必须在应用程序中编写这函数来分别完成三个任务。

2. 任务列表:(也叫作任务清单)

typedef enum _TASK_LIST

{

TAST_DISP_CLOCK,               // 显示时钟

TAST_KEY_SAN,                     // 按键扫描

TASK_DISP_WS,                     // 工作状态显示

// 这里添加你的任务。。。。

TASKS_MAX                                           // 总的可供分配的定时任务数目

} TASK_LIST;

好好看看,我们这里定义这个任务清单的目的其实就是参数TASKS_MAX的值,其他值是没有具体的意义的,只是为了清晰的表面任务的关系而已。

原文地址:https://www.cnblogs.com/chulin/p/8184610.html

时间: 2024-10-10 19:46:05

硬件——STM32 , 软件框架的相关文章

[Alljoyn] 1、物联网开源软件框架alljoyn研究(一)——初步了解

What is AllJoyn?[是一个合作的开源软件框架目的在于连接万物] An Open Source API Framework For the Internet of EverythingA way devices and applications publish APIs over a network in a standard wayWhy APIs?– Because this is what software developers understand and work with

安装 SQL Server 2008 R2 的硬件和软件要求(转)

以下各部分列出了安装和运行 SQL Server 2008 R2 的最低硬件和软件要求.有关 SharePoint 集成模式下的 Analysis Services 的要求的详细信息,请参阅硬件和软件要求 (PowerPivot for SharePoint). 对于 SQL Server 2008 R2 的 32 位和 64 位版本,适用以下要求: SQL Server 2008 R2 Datacenter 有评估版可供使用,试用期为 180 天.有关详细信息,请参阅 SQL Server:

TI BLE协议栈软件框架分析

看源代码的时候,一般都是从整个代码的入口处开始,TI  BLE 协议栈源码也不例外.它的入口main()函数就是整个程序的入口,由系统上电时自动调用. 它主要做了以下几件事情: (一)底层硬件初始化配置 (二)创建任务并初始化任务配置 (三)检测并执行有效的任务事件 Main() 函数源码如下: 一:底层硬件初始化设置 75行,设置系统时钟,使能内存缓冲功能. 78行,关中断,刚启动时,系统运行不稳定,一般会首先关中断. 81行,硬件相关的I/O 口配置. 84行,初始化mcu 内部的flash

第二天:SLAM智能小车DIY乐趣-小车控制stm32软件基础

SLAM智能小车DIY乐趣-小车控制stm32软件基础 ####写在前面#### 前面介绍了小车控制stm32硬件基础,本文就来介绍配套的小车控制stm32软件基础.关于stm32开发相关的基础知识这里就不多说了,有需要的小伙伴可以查阅相关资料进行学习,这里重点从小车控制项目入手,直接进行跟实际需求相关的开发.本文主要内容: 1) 电机控制 2) 编码器数据读取 3) 串口数据收发 4) 电机速度PID控制 5) 周期性控制 6) 小车控制软件整体框图 ####正文#### 1.电机控制 电机控

基于分层思想的驱动程序软件框架

基于分层思想的驱动程序软件框架 目的 让驱动程序有很好的移植性. 实现的方法 将硬件相关的操作放入一个结构体中,类似于file_operation的结构体中,之后和将其操作的实现,一起封装在一个函数中,且需要设计提供一个函数让外部函数能得到硬件相关的结构体.(分层思想) 让向内核注册驱动的入口和出口中函数,只需要将所有驱动通用的代码放入其中即可 ① 确定主设备号,也可以让内核分配 ② 定义自己的 file_operations 结构体 ③ 实现对应的 drv_open/drv_read/drv_

SQL Server 205 的硬件和软件要求(官方全面)

SQL Server 2005 安装要求 本主题介绍了安装 SQL Server 205 的硬件和软件要求,以及查看安装文档的说明. 硬件和软件要求(32 位和 64 位) 访问 SQL Server 2005 联机丛书以获取安装信息 硬件和软件要求(32 位和 64 位) 以下部分列出了运行 Microsoft SQL Server 2005 的最低硬件和软件要求. 注意: 在 32 位平台和在 64 位平台上运行 SQL Server 2005 的要求是不同的. 硬件和软件要求(32 位和

【VMCloud云平台】SCCM(六)客户端硬件、软件资产管理

继上一篇云平台完成SCCM部署篇之后,SCCM篇正式开始,今天将开始介绍SCCM主要功能--客户端硬件.软件资产管理(紫色为完成实施,红色为实施中): 1. 点击管理: 2. 点击客户端设置: 3. 右键客户端默认设置,开启硬件清单: 4. 设置需要收集的类(当然也可以自已导入mof模板进行额外的硬件收集): 5. 点击确定后,点击软件清单,开启收集: 6. 设置收集的类型为.exe: 7. 确认设置后点击确定(设置计划): 8. 回到客户端符合性页,点击其中一台设备,右键启动-资源浏览器: 9

安装 SQL Server 2012 的硬件和软件要求(官方全面)

以下各节列出了安装和运行 SQL Server 2012 的最低硬件和软件要求. Hardware and Software Requirements (PowerPivot for SharePoint and Reporting Services in SharePoint Mode).' data-guid="ac62f277a77cf0fe4624123fb93d4313">有关 SharePoint 集成模式下 Analysis Services 的要求的详细信息,请参

Mindjet MindManager 思维导图软件-使用思维导图跟踪调用流程,绘制软件框架

思维导图.据说是每一个产品经理必备的软件.假设你阅读大型源码.使用思维导图跟踪调用流程,绘制软件框架将会很方便. 特点:没什么好说的.用过的都说好. 软件截图: 下载:http://www.mindmanager.cc/ MindManager新手新手教程 MindManager是一款创造.管理和交流思想的思维导图软件,其直观清晰的可视化界面和强大的功能能够高速捕捉.组织和共享思维.想法.资源和项目进程等等.MindManager新手新手教程专为新手用户设计,包括创建思维导图基本入门操作,让用户