结合时间触发+消息+protothread思想+支持优先级的非抢占调度器

2019-12-13 18:20发布

本帖最后由 summarize 于 2013-5-19 14:47 编辑

废话少说,先上stm8s103 IAR库工程代码压缩包。
Schedule-IAR-STM8S103.rar (416.35 KB, 下载次数: 538) 2013-5-19 14:15 上传 点击文件名下载附件

工程是在stm8s103f3单片机上调度通过,已经用消息实现了 UART1_TX模块的共享,即UART1_RX接收到的数据+0x11后再通过UART_TX模块发送回去,同时ADC1 通道3的转换结果也通过UART1_TX模块发送出去.见下图

1.ADC1转换结果每1秒上传一次到PC。测试式给ADC1通道3供的是5V电,所以结果是0x03ff.即1023.

共享时接收到的数据.png (61.3 KB, 下载次数: 0) 下载附件 2013-5-19 14:16 上传

3.支持非抢占式优先级调度,优先级顺序就是创建任务时的顺序,由高到底。其实现思想是,每一个任务运行结束后,都重新回到第一个创建的任务处按顺序查找某个任务是否满足运行条件,所以先创建的任务会先被“发现”其满足运行条件并运行之,核心代码如下

a.任务控制块数据结构
  1. struct SchTcb
  2. {
  3. #if SCH_CFG_Q_EN > 0u
  4.   void          *pData;       //消息指针
  5.   SCH_UINT8 Size;         //消息大小
  6. #endif

  7.   SCH_DLY_TYPE        TimeCounter;  //定时计数器,时基为 "SCH_SYS_TICKS_MS"
  8.   void          (*pTask)();   //任务指针
  9.   struct SchTcb *pNextTCB;    //下一个任务控制块指针
  10. };
复制代码b.调度核心
  1. void SCHTaskSchedStart(void)
  2. {
  3. SCHED_SART:
  4.        
  5.   pCurTCB = pFirstTCB;                        //指向第一个创建的任务,之后按创建时的顺序执行下去

  6.   while (1)                                 
  7.   {
  8.     SCHTimeTick();                            //如果任务Tick满足条件,则将其置于可执行状态

  9.     if (SCH_TASK_RUN == pCurTCB->TimeCounter) //任务处于可执行状态
  10.     {
  11.       pCurTCB->TimeCounter = SCH_TASK_PEND;   //设置为挂起状态,保证任务只执行一次

  12.       pCurTCB->pTask();                       //执行当前任务控制块指向的任务

  13.       goto SCHED_SART;                        //每执行完一个任务,都重新查找一次可执行最高优先级任务
  14.     }

  15.     pCurTCB = pCurTCB->pNextTCB;              //指向下一个任务控制块,查找下个任务是否可执行
  16.   }
  17. }
复制代码“schedule.c”和"schedule.h"已经设置为只读属性,无特殊情况不建议修改,"sch_cfg.h"则为开放给用户的接口,可定义数据类型、调度器节拍和配置是否使用消息。

本人水平有限,欢迎大家测试、指正不足。

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
101条回答
乡村男孩
1楼-- · 2019-12-24 15:34
Mark 留名~!有时间慢慢品味
wazhiyi
2楼-- · 2019-12-24 16:14
顶一下。。。。。。。。。。。。
aming2046
3楼-- · 2019-12-24 19:45
大哥我按照你的方法把他们定义的展开了,但是展开后这个函数我怎么看不懂啊
void vAdcResultSend(void)
{
  static uint8_t s_u8AdcBuffer[ADC_BUF_SIZE];
  static uint8_t i;
  static uint8_t *pDataBuffer, u8DataSize;

  //SCHTaskBegin();
  
  static SCH_UINT8 SchLc=0;
  switch(SchLc)
  {
   case 0://跳转开始,中间可插入延时,调用子函数;(适用主/子函数)

  while (1)
  {   
    //SCHCurTaskDly(1000 / SCH_SYS_TICKS_MS);   //delay 1000ms
   
    {
      SchLc=SCH_CURR_LINE;
      pCurTCB->TimeCounter=Ticks;
   
    }return ;
  case SCH_CURR_LINE:

    //SCHTaskQpend();//任务等待消息
   
    {
      SchLc=SCH_CURR_LINE;
      pCurTCB->TimeCounter=SCH_TASK_PEND;
      pCurTCB->pData=(void *)0;
    }
    return;
  case SCH_CURR_LINE:
   


    pDataBuffer = (uint8_t *)AdcResultSendTcb.pData;
    u8DataSize  = AdcResultSendTcb.Size;

    if (u8DataSize > ADC_BUF_SIZE)
    {
      u8DataSize = ADC_BUF_SIZE;
      //出错处理
    }

    for (i = 0; i < u8DataSize; i++)
    {
      s_u8AdcBuffer[i] = *pDataBuffer; //copy to s_u8AdcBuffer
      pDataBuffer++;
    }

    //使用如下类似方法便可与其它任务共享UART_TX资源,即任务要使用UART_TX前先检查其队列是否可用,不可用则等待
    for (u8DataSize = 0; u8DataSize < 255; u8DataSize++)  //借用u8DataSize
    {
      if (SCHTaskGetQFree(&UartTxTcb) == SCH_Q_FREE)      //检查UART_TX发送任务队列是否可用
      {
        SCHTaskQpost(&UartTxTcb,
                     &s_u8AdcBuffer[0],
                     i);
        break;
      }
      else
      {
        SCHCurTaskDly(1 / SCH_SYS_TICKS_MS);  //delay 1ms
      }
    }

    if (u8DataSize >= 100)
    {
      //出错处理
    }   
  };
};
  SchLc=0;
  
  //SCHTaskEnd();
}
aming2046
4楼-- · 2019-12-25 00:53
case 0:后面的while(1){}大括号里面怎么没有switch而是直接出现了两个 case SCH_CURR_LINE:
wanyou132
5楼-- · 2019-12-25 06:12
 精彩回答 2  元偷偷看……
summarize
6楼-- · 2019-12-25 09:18
aming2046 发表于 2013-12-17 20:57
case 0:后面的while(1){}大括号里面怎么没有switch而是直接出现了两个 case SCH_CURR_LINE: ...

将"while(1){}"看作一般的语句即可,后面的"case SCH_CURR_LINE: ..." 与 最前的"case 0:"是并列关系.即:
switch(*)
{
case 0: ...

case SCH_CURR_LINE: ...

case SCH_CURR_LINE: ...
}

一周热门 更多>