單片機(jī)的ROM與RAM存貯空間有限,一般沒有多線程可用,給復(fù)雜的單片機(jī)項(xiàng)目帶來困擾。
經(jīng)過多年的單片機(jī)項(xiàng)目實(shí)踐,借鑒windows消息機(jī)制的思想,編寫了單片機(jī)多任務(wù)事件驅(qū)動C代碼,應(yīng)用于單片機(jī)項(xiàng)目,無論復(fù)雜的項(xiàng)目,還是簡單的項(xiàng)目,都可以達(dá)到優(yōu)化代碼架構(gòu)的目的。
經(jīng)過幾輪的精簡、優(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í)間到,會生成一個(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ì)列長度 #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)注
6067文章
44983瀏覽量
650272 -
ROM
+關(guān)注
關(guān)注
4文章
578瀏覽量
87332 -
WINDOWS
+關(guān)注
關(guān)注
4文章
3613瀏覽量
91382 -
定時(shí)器
+關(guān)注
關(guān)注
23文章
3298瀏覽量
118847 -
源碼
+關(guān)注
關(guān)注
8文章
671瀏覽量
30317
原文標(biāo)題:一份單片機(jī)多任務(wù)事件驅(qū)動C源碼
文章出處:【微信號:c-stm32,微信公眾號:STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
單片機(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)的課件說明

評論