单片机的ROM与RAM存贮空间有限,一般没有多线程可用,给复杂的单片机项目带来困扰。
经过多年的单片机项目实践,借鉴windows消息机制的思想,编写了单片机多任务事件驱动C代码,应用于单片机项目,无论复杂的项目,还是简单的项目,都可以达到优化代码架构的目的。
经过几轮的精简、优化,现在分享给大家。
代码分为3个模块:任务列表、事件列表、定时器列表。
任务列表创建一个全局列表管理任务,通过调用taskCreat()创建事件处理任务,创建成功返回任务ID,任务列表、事件列表与定时器列表通过任务ID关联。
事件列表创建一个全局循环列表管理事件,调用taskEventIssue()生成一个事件,放到事件循环列表,taskEventLoop()函数放到主线程循环调用,当事件循环列表中有事件时,根据任务ID分发到具体的事件处理任务。
定时器列表创建一个全局列表管理定时器,taskTimer()建立一个定时器,放到定时器列表执行,当定时时间到,会生成一个定时器事件,放到事件列表,分发到具体的事件处理任务。
//common.h #ifndef __COMMON_H #define __COMMON_H #include "stdio.h" #include#include typedef short int16_t; typedef int int32_t; typedef long long int64_t; typedef unsigned char uint8_t; typedef unsigned short uint16_t; typedef unsigned int uint32_t; typedef unsigned long long uint64_t; typedef unsigned char bool; #define false 0 #define true 1 #endif // __COMMON_H
//task.h #ifndef _THREAD_H #define _THREAD_H #define TASK_MAX 20 // 最多任务数量 #define TASK_EVENT_MAX 100 // 任务队列长度 #define TASK_TIMER_MAX 100 // 定时器最大数量 typedef void (*CBTaskEvent)(int taskID,uint32_t eventID); typedef struct _TASK_EVENT { int taskID; uint32_t eventID; } TASK_EVENT; int taskCreat(CBTaskEvent task); void taskLoop(); void taskEventIssue(int taskID,uint32_t eventID); void taskEventLoop(); //定时、休眠 typedef struct _TASK_TIMER { bool isValid; int taskID; uint32_t eventID; uint32_t timeMs; uint32_t start; } TASK_TIMER; void taskTicksInc(); void taskTimer(int taskID,uint32_t eventID,uint32_t time_ms); void taskTimerLoop(); #endif // _THREAD_H
//task.c
#include "common.h"
#include "task.h"
CBTaskEvent g_taskList[TASK_MAX]={0};
int taskFindEmpty()
{
static int index = -1;
for(int i=0; i
",taskID);
return taskID;
}
void taskDestroy(int taskID)
{
printf("Destroy task<%d>
",taskID);
g_taskList[taskID] = NULL;
}
void taskLoop()
{
taskEventLoop();
taskTimerLoop();
}
TASK_EVENT g_taskEventList[TASK_EVENT_MAX];
int g_TKEventWrite=0;
int g_TKEventRead=0;
int tkEventGetSize()
{
return (g_TKEventWrite + TASK_EVENT_MAX - g_TKEventRead)% TASK_EVENT_MAX;
}
void taskEventIssue(int taskID,uint32_t eventID)
{
int writePos;
if(taskID >= TASK_EVENT_MAX || taskID < 0)
{
printf("taskEventIssue() error:taskID
");
return;
}
writePos = (g_TKEventWrite + 1)% TASK_EVENT_MAX;
if(writePos == g_TKEventRead)
{
printf("taskEventIssue() error:task<%d> event list is full!
",taskID);
return;
}
g_taskEventList[g_TKEventWrite].taskID=taskID;
g_taskEventList[g_TKEventWrite].eventID=eventID;
g_TKEventWrite=writePos;
//printf("add event:%x
",eventID);
}
void taskEventLoop()
{
TASK_EVENT event;
CBTaskEvent task;
int size;
size=tkEventGetSize();
while(size-- >0)
{
event=g_taskEventList[g_TKEventRead];
g_TKEventRead = (g_TKEventRead + 1)% TASK_EVENT_MAX;
task = g_taskList[event.taskID];
if(!task)
{
printf("taskEventLoop() error:task is NULL
");
continue;
}
task(event.taskID,event.eventID);
}
}
// 定时、休眠
uint32_t g_taskTicks=0;
uint32_t getTaskTicks()
{
return g_taskTicks;
}
void taskTicksInc() // 1ms时间基准
{
g_taskTicks++;
}
uint32_t taskTickDiff(uint32_t now,uint32_t last)
{
uint64_t diff;
diff = now + 0x100000000 - last;
return (diff & 0xffffffff);
}
TASK_TIMER g_taskTimerList[TASK_TIMER_MAX]={0};
int taskTimerFindEmpty()
{
for(int i=0; i %ums
",taskID,eventID,time_ms);
}
void taskTimerLoop()
{
static uint32_t start=0;
if(taskTickDiff(getTaskTicks(),start)<3)
{
return;
}
start=getTaskTicks();
for(int i=0; i=g_taskTimerList[i].timeMs)
{
taskEventIssue(g_taskTimerList[i].taskID,g_taskTimerList[i].eventID);
g_taskTimerList[i].isValid=false;
}
}
}
}
//test_task.h #ifndef _TEST_THREAD_H #define _TEST_THREAD_H void testInit(); void testLoop(); #endif //
//test_task.c
#include "common.h"
#include "task.h"
#define CTRL_EVENT1 0x01
#define CTRL_EVENT2 0x02
#define CTRL_EVENT3 0x04
void eventProcess(int taskID,uint32_t event)
{
switch(event)
{
case CTRL_EVENT1:
printf("task[%d] CTRL_EVENT1
",taskID);
//taskEventIssue(taskID,CTRL_EVENT2);
taskTimer(taskID,CTRL_EVENT2,1000);
break;
case CTRL_EVENT2:
printf("task[%d] CTRL_EVENT2
",taskID);
//taskEventIssue(taskID,CTRL_EVENT3);
taskTimer(taskID,CTRL_EVENT3,2000);
break;
case CTRL_EVENT3:
printf("task[%d] CTRL_EVENT3
",taskID);
taskTimer(taskID,CTRL_EVENT1,4000);
break;
default:
break;
}
}
void testLoop()
{
taskLoop();
}
void testInit()
{
int taskID1,taskID2;
printf("testInit()
");
taskID1 = taskCreat((CBTaskEvent)&eventProcess);
taskTimer(taskID1,CTRL_EVENT1,5000);
taskID2 = taskCreat((CBTaskEvent)&eventProcess);
taskEventIssue(taskID2,CTRL_EVENT2);
taskDestroy(taskID1);
taskDestroy(taskID2);
//taskEventIssue(taskID1,CTRL_EVENT1);
taskID1 = taskCreat((CBTaskEvent)&eventProcess);
taskEventIssue(taskID1,CTRL_EVENT1);
}
审核编辑:汤梓红
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
单片机
+关注
关注
6078文章
45629浏览量
675401 -
ROM
+关注
关注
4文章
580浏览量
89482 -
WINDOWS
+关注
关注
4文章
3711浏览量
94450 -
定时器
+关注
关注
23文章
3378浏览量
125080 -
源码
+关注
关注
8文章
689浏览量
31559
原文标题:一份单片机多任务事件驱动C源码
文章出处:【微信号:c-stm32,微信公众号:STM32嵌入式开发】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
热点推荐
单片机小型操作系统多任务并行的原理
如果在单片机的中断中修改了堆栈中的PC指针值,那么是不是中断就返不回来原来的程序地址?如果还是这个地址是我故意设定的值那么他是不是执行完这个中断就跳到我的设定的这个地址上去了?如果这个地址又是另外一个任务
发表于 10-18 10:59
51单片机多任务机制的实现策略研究
从操作系统实现多任务机制的原理入手,分析了51单片机实现多任务机制的基本条件,论述了5l单片机实现多任务控制的二种方案。
发表于 09-19 17:26
•159次下载
单片机系统多任务实现方法
考虑到成本等因素,大多数单片机系统的开发还是基于处理器直接编写,但开发者经常面临同时处理多个任务的要求,提出了一种单片机多任务程序设计的方法
发表于 11-21 16:47
•751次下载
MSP430单片机实时多任务操作系统c源代码
: 资料-【C】嵌入系统-【C0】嵌入式综合-【2】单片机编程-【参考程序】-MSP430单片机的实时多任务操作系统
发表于 09-18 10:18
•91次下载
如何使用51单片机进行多任务机制及应用
传统的单片机程序一般采用单任务机制,单任务系统具有简单直观、易于控制的优点。然而由于程序只能按顺序依次执行,缺乏灵活性,只能使用中断函数实时地处理一
发表于 04-15 18:24
•8次下载
如何使用51单片机实现多任务机制的策略详细资料说明
51单片机在微型智能控制系统中应用很广,随着人们对控制系统的要求不断提高,51单片机的功能局限越发明显。特别是51系列单片机不具有实时多任务支持功能,大大限制其在控制系统中的进
发表于 07-17 17:38
•6次下载
使用单片机实现道路交通灯多任务控制系统的课件说明
RTX-51是运行于8051单片机环境中的实时多任务操作系统(RTOS),其常用于处理复杂的多任务控制系统。8051单片机支持典型的RTX-51 Tiny和RTX-51 Full实时
发表于 04-16 17:31
•7次下载
基于M16C62单片机构成的实时多任务系统
、多任务系统的设计合应用成为单片机应用的新的发展趋势。 μC/OS-Ⅱ是一个源码公开的实时嵌入式操作系统,它的特点在于公开的源代码,很强的移
一份单片机多任务事件驱动C源码
评论