• 注册 / 登录
  • 切换到窄版
  • 查看: 1885|回复: 0

    单片机多任务事件驱动C源码

    [复制链接]

    26

    主题

    26

    帖子

    124

    积分

    注册会员

    Rank: 2

    积分
    124
    发表于 2023-7-24 09:28:41 | 显示全部楼层 |阅读模式

    路线栈欢迎您!

    您需要 登录 才可以下载或查看,没有帐号?立即注册

    x
    前言

    单片机的ROM与RAM存贮空间有限,一般没有多线程可用,给复杂的单片机项目带来困扰。经过多年的单片机项目实践,借鉴windows消息机制的思想,编写了单片机多任务事件驱动C代码,应用于单片机项目,无论复杂的项目,还是简单的项目,都可以达到优化代码架构的目的。经过几轮的精简、优化,现在分享给大家。

    设计思路

    代码分为3个模块:任务列表、事件列表、定时器列表。

    任务列表创建一个全局列表管理任务,通过调用taskCreat()创建事件处理任务,创建成功返回任务ID,任务列表、事件列表与定时器列表通过任务ID关联。

    事件列表创建一个全局循环列表管理事件,调用taskEventIssue()生成一个事件,放到事件循环列表,taskEventLoop()函数放到主线程循环调用,当事件循环列表中有事件时,根据任务ID分发到具体的事件处理任务。

    定时器列表创建一个全局列表管理定时器,taskTimer()建立一个定时器,放到定时器列表执行,当定时时间到,会生成一个定时器事件,放到事件列表,分发到具体的事件处理任务。

    代码例程:

    1. //common.h
    2. #ifndef __COMMON_H
    3. #define __COMMON_H

    4. #include "stdio.h"
    5. #include <stdlib.h>
    6. #include <string.h>

    7. typedef short int16_t;
    8. typedef int int32_t;
    9. typedef long long int64_t;
    10. typedef unsigned char uint8_t;
    11. typedef unsigned short uint16_t;
    12. typedef unsigned int uint32_t;
    13. typedef unsigned long long uint64_t;
    14. typedef unsigned char bool;

    15. #define false 0
    16. #define true 1

    17. #endif // __COMMON_H
    复制代码
    1. //task.h
    2. #ifndef _THREAD_H
    3. #define _THREAD_H

    4. #define TASK_MAX 20 // 最多任务数量
    5. #define TASK_EVENT_MAX 100 // 任务队列长度
    6. #define TASK_TIMER_MAX 100 // 定时器最大数量

    7. typedef void (*CBTaskEvent)(int taskID,uint32_t eventID);
    8. typedef struct _TASK_EVENT
    9. {
    10.     int taskID;
    11.     uint32_t eventID;

    12. } TASK_EVENT;

    13. int taskCreat(CBTaskEvent task);
    14. void taskLoop();
    15. void taskEventIssue(int taskID,uint32_t eventID);
    16. void taskEventLoop();

    17. //定时、休眠

    18. typedef struct _TASK_TIMER
    19. {
    20.     bool isValid;
    21.     int taskID;
    22.     uint32_t eventID;
    23.     uint32_t timeMs;
    24.     uint32_t start;

    25. } TASK_TIMER;

    26. void taskTicksInc();
    27. void taskTimer(int taskID,uint32_t eventID,uint32_t time_ms);
    28. void taskTimerLoop();

    29. #endif // _THREAD_H
    复制代码
    1. //task.c
    2. #include "common.h"
    3. #include "task.h"

    4. CBTaskEvent g_taskList[TASK_MAX]={0};

    5. int taskFindEmpty()
    6. {
    7.     static int index = -1;

    8.     for(int i=0; i<TASK_MAX; i++)
    9.     {
    10.         index++;
    11.         index %= TASK_MAX;
    12.         if(g_taskList[index]==NULL)
    13.         {
    14.             return index;
    15.         }
    16.     }

    17.     return -1;
    18. }

    19. int taskCreat(CBTaskEvent task)
    20. {
    21.     int taskID;

    22.     taskID=taskFindEmpty();
    23.     if(taskID == -1)
    24.     {
    25.         printf("error:task list is full!\n");
    26.         return -1;
    27.     }

    28.     g_taskList[taskID] = task;

    29.     printf("creat task<%d>\n",taskID);

    30.     return taskID;
    31. }

    32. void taskDestroy(int taskID)
    33. {
    34.     printf("Destroy task<%d>\n",taskID);

    35.     g_taskList[taskID] = NULL;
    36. }

    37. void taskLoop()
    38. {
    39.     taskEventLoop();
    40.     taskTimerLoop();
    41. }

    42. TASK_EVENT g_taskEventList[TASK_EVENT_MAX];
    43. int g_TKEventWrite=0;
    44. int g_TKEventRead=0;

    45. int tkEventGetSize()
    46. {
    47.     return (g_TKEventWrite + TASK_EVENT_MAX - g_TKEventRead)% TASK_EVENT_MAX;
    48. }

    49. void taskEventIssue(int taskID,uint32_t eventID)
    50. {
    51.     int writePos;

    52.     if(taskID >= TASK_EVENT_MAX || taskID < 0)
    53.     {
    54.         printf("taskEventIssue() error:taskID\n");
    55.         return;
    56.     }

    57.     writePos = (g_TKEventWrite + 1)% TASK_EVENT_MAX;

    58.     if(writePos == g_TKEventRead)
    59.     {
    60.         printf("taskEventIssue() error:task<%d> event list is full!\n",taskID);
    61.         return;
    62.     }

    63.     g_taskEventList[g_TKEventWrite].taskID=taskID;
    64.     g_taskEventList[g_TKEventWrite].eventID=eventID;
    65.     g_TKEventWrite=writePos;

    66.     //printf("add event:%x\n",eventID);
    67. }

    68. void taskEventLoop()
    69. {
    70.     TASK_EVENT event;
    71.     CBTaskEvent task;
    72.     int size;

    73.     size=tkEventGetSize();
    74.     while(size-- >0)
    75.     {
    76.         event=g_taskEventList[g_TKEventRead];
    77.         g_TKEventRead = (g_TKEventRead + 1)% TASK_EVENT_MAX;

    78.         task = g_taskList[event.taskID];
    79.         if(!task)
    80.         {
    81.             printf("taskEventLoop() error:task is NULL\n");
    82.             continue;
    83.         }

    84.         task(event.taskID,event.eventID);
    85.     }
    86. }

    87. // 定时、休眠

    88. uint32_t g_taskTicks=0;

    89. uint32_t getTaskTicks()
    90. {
    91.     return g_taskTicks;
    92. }

    93. void taskTicksInc() // 1ms时间基准
    94. {
    95.     g_taskTicks++;
    96. }

    97. uint32_t taskTickDiff(uint32_t now,uint32_t last)
    98. {
    99.     uint64_t diff;
    100.     diff = now + 0x100000000 - last;

    101.     return (diff & 0xffffffff);
    102. }

    103. TASK_TIMER g_taskTimerList[TASK_TIMER_MAX]={0};

    104. int taskTimerFindEmpty()
    105. {
    106.     for(int i=0; i<TASK_TIMER_MAX; i++)
    107.     {
    108.         if(!g_taskTimerList[i].isValid)
    109.         {
    110.             return i;
    111.         }
    112.     }

    113.     return -1;
    114. }

    115. void taskTimer(int taskID,uint32_t eventID,uint32_t time_ms)
    116. {
    117.     int index;

    118.     index=taskTimerFindEmpty();
    119.     if(index==-1)
    120.     {
    121.         printf("taskTimer() error:timer list is full\n");
    122.         return;
    123.     }

    124.     g_taskTimerList[index].taskID=taskID;
    125.     g_taskTimerList[index].eventID=eventID;
    126.     g_taskTimerList[index].timeMs=time_ms;
    127.     g_taskTimerList[index].start=getTaskTicks();
    128.     g_taskTimerList[index].isValid=true;

    129.     printf("add timer:<%d,%x> %ums\n",taskID,eventID,time_ms);

    130. }

    131. void taskTimerLoop()
    132. {
    133.     static uint32_t start=0;
    134.     if(taskTickDiff(getTaskTicks(),start)<3)
    135.     {
    136.         return;
    137.     }

    138.     start=getTaskTicks();

    139.     for(int i=0; i<TASK_TIMER_MAX; i++)
    140.     {
    141.         if(g_taskTimerList[i].isValid)
    142.         {
    143.             if(taskTickDiff(start,g_taskTimerList[i].start)>=g_taskTimerList[i].timeMs)
    144.             {
    145.                 taskEventIssue(g_taskTimerList[i].taskID,g_taskTimerList[i].eventID);
    146.                 g_taskTimerList[i].isValid=false;
    147.             }
    148.         }
    149.     }
    150. }
    复制代码
    1. //test_task.h
    2. #ifndef _TEST_THREAD_H
    3. #define _TEST_THREAD_H

    4. void testInit();
    5. void testLoop();

    6. #endif //
    复制代码
    1. //test_task.c
    2. #include "common.h"
    3. #include "task.h"

    4. #define CTRL_EVENT1 0x01
    5. #define CTRL_EVENT2 0x02
    6. #define CTRL_EVENT3 0x04

    7. void eventProcess(int taskID,uint32_t event)
    8. {
    9.     switch(event)
    10.     {
    11.         case CTRL_EVENT1:
    12.             printf("task[%d] CTRL_EVENT1\n",taskID);
    13.             //taskEventIssue(taskID,CTRL_EVENT2);
    14.             taskTimer(taskID,CTRL_EVENT2,1000);
    15.             break;

    16.         case CTRL_EVENT2:
    17.             printf("task[%d] CTRL_EVENT2\n",taskID);
    18.             //taskEventIssue(taskID,CTRL_EVENT3);
    19.             taskTimer(taskID,CTRL_EVENT3,2000);
    20.             break;

    21.         case CTRL_EVENT3:
    22.             printf("task[%d] CTRL_EVENT3\n",taskID);
    23.             taskTimer(taskID,CTRL_EVENT1,4000);
    24.             break;

    25.         default:
    26.             break;
    27.     }
    28. }

    29. void testLoop()
    30. {
    31.     taskLoop();
    32. }

    33. void testInit()
    34. {
    35.     int taskID1,taskID2;

    36.     printf("testInit()\n");

    37.     taskID1 = taskCreat((CBTaskEvent)&eventProcess);

    38.     taskTimer(taskID1,CTRL_EVENT1,5000);

    39.     taskID2 = taskCreat((CBTaskEvent)&eventProcess);
    40.     taskEventIssue(taskID2,CTRL_EVENT2);
    41.     taskDestroy(taskID1);
    42.     taskDestroy(taskID2);
    43.     //taskEventIssue(taskID1,CTRL_EVENT1);
    44.     taskID1 = taskCreat((CBTaskEvent)&eventProcess);
    45.     taskEventIssue(taskID1,CTRL_EVENT1);
    46. }
    复制代码


    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    小黑屋|路丝栈 ( 粤ICP备2021053448号 )

    GMT+8, 2024-12-22 15:20 , Processed in 0.041844 second(s), 18 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表