單片機(jī)的ROM與RAM存貯空間有限,一般沒有多線程可用,給復(fù)雜的單片機(jī)項(xiàng)目帶來困擾。
經(jīng)過多年的單片機(jī)項(xiàng)目實(shí)踐,借鑒windows消息機(jī)制的思想,編寫了單片機(jī)多任務(wù)事件驅(qū)動(dòng)C代碼,應(yīng)用于單片機(jī)項(xiàng)目,無論復(fù)雜的項(xiàng)目,還是簡(jiǎn)單的項(xiàng)目,都可以達(dá)到優(yōu)化代碼架構(gòu)的目的。
經(jīng)過幾輪的精簡(jiǎn)、優(yōu)化,現(xiàn)在分享給大家。
代碼分為3個(gè)模塊:任務(wù)列表、事件列表、定時(shí)器列表。
任務(wù)列表創(chuàng)建一個(gè)全局列表管理任務(wù),通過調(diào)用taskCreat()創(chuàng)建事件處理任務(wù),創(chuàng)建成功返回任務(wù)ID,任務(wù)列表、事件列表與定時(shí)器列表通過任務(wù)ID關(guān)聯(lián)。
事件列表創(chuàng)建一個(gè)全局循環(huán)列表管理事件,調(diào)用taskEventIssue()生成一個(gè)事件,放到事件循環(huán)列表,taskEventLoop()函數(shù)放到主線程循環(huán)調(diào)用,當(dāng)事件循環(huán)列表中有事件時(shí),根據(jù)任務(wù)ID分發(fā)到具體的事件處理任務(wù)。
定時(shí)器列表創(chuàng)建一個(gè)全局列表管理定時(shí)器,taskTimer()建立一個(gè)定時(shí)器,放到定時(shí)器列表執(zhí)行,當(dāng)定時(shí)時(shí)間到,會(huì)生成一個(gè)定時(shí)器事件,放到事件列表,分發(fā)到具體的事件處理任務(wù)。
//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 // 最多任務(wù)數(shù)量 #define TASK_EVENT_MAX 100 // 任務(wù)隊(duì)列長(zhǎng)度 #define TASK_TIMER_MAX 100 // 定時(shí)器最大數(shù)量 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(); //定時(shí)、休眠 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);
}
}
// 定時(shí)、休眠
uint32_t g_taskTicks=0;
uint32_t getTaskTicks()
{
return g_taskTicks;
}
void taskTicksInc() // 1ms時(shí)間基準(zhǔn)
{
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);
}
-
單片機(jī)
+關(guān)注
關(guān)注
6072文章
45267瀏覽量
661178 -
ROM
+關(guān)注
關(guān)注
4文章
578瀏覽量
88325 -
WINDOWS
+關(guān)注
關(guān)注
4文章
3623瀏覽量
92900 -
定時(shí)器
+關(guān)注
關(guān)注
23文章
3350瀏覽量
121270 -
源碼
+關(guān)注
關(guān)注
8文章
678瀏覽量
30915
原文標(biāo)題:一份單片機(jī)多任務(wù)事件驅(qū)動(dòng)C源碼
文章出處:【微信號(hào):c-stm32,微信公眾號(hào):STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
單片機(jī)小型操作系統(tǒng)多任務(wù)并行的原理
51單片機(jī)多任務(wù)機(jī)制的實(shí)現(xiàn)策略研究
單片機(jī)系統(tǒng)多任務(wù)實(shí)現(xiàn)方法
MSP430單片機(jī)實(shí)時(shí)多任務(wù)操作系統(tǒng)c源代碼
51單片機(jī)多任務(wù)操作系統(tǒng)的原理與實(shí)現(xiàn)
基于實(shí)時(shí)操作系統(tǒng)的單片機(jī)多任務(wù)應(yīng)用程序設(shè)計(jì)
如何使用51單片機(jī)實(shí)現(xiàn)多任務(wù)機(jī)制的策略詳細(xì)資料說明
單片機(jī)多任務(wù)處理方案
使用單片機(jī)實(shí)現(xiàn)道路交通燈多任務(wù)控制系統(tǒng)的課件說明

一份單片機(jī)多任務(wù)事件驅(qū)動(dòng)C源碼
評(píng)論