久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2284|回復: 0
打印 上一主題 下一主題
收起左側

51單片機多任務os源程序

[復制鏈接]
跳轉到指定樓層
樓主
ID:20672 發表于 2019-11-22 20:53 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
調用系統服務使任務切換時不保存寄存器變量
中斷服務執行任務切換時,保存寄存器變量
支持7個任務,支持7級中斷嵌套
基于優先級調度,0號任務具有最高優先級
包含K_FLG K_SEM K_MSG K_TMO這幾個系統服務,每種類型的系統服務最多提供16個

單片機源程序如下:
  1. #define _IN_LQ51_C_
  2. #include "lq51.h"
  3. #include<reg52.h>
  4. /******************************************
  5. /                          定義全局變量
  6. /******************************************/
  7. unsigned char data lqTaskStack[lqMaxID+1];                        /*任務堆棧指針*/
  8. unsigned char data lqSPtemp;                                                /*記錄ID號=當前任務ID+1的堆棧底部*/
  9. unsigned char data lqTaskTimer[lqMaxID];                        /*任務定時器*/
  10. unsigned char data lqTaskState[lqMaxID];                        /*任務狀態表*/
  11. unsigned char data lqRdyTbl;                                                /*就緒表*/
  12. unsigned char data lqSwitchType;                                        /*任務切換類型,如果通過中斷切換任務則在相應位置1,否則置0*/
  13. unsigned char data lqIntNum;                                                /*進入中斷服務子程序后系統把中斷號傳給這個變量*/
  14. unsigned char data lqCrt;                                                        /*當前正在運行的任務*/

  15. #if        LQ_FLG_EN                                /*該事件的最高位表示該標志位是否置位,其他位表示等待該標志量的任務,當標志位置位時,所有等待這個標志的任務都被激活*/
  16. unsigned char data lqFlgData[lqFlgMax]={
  17. 0,0
  18. };
  19. #endif

  20. #if        LQ_SEM_EN                                                        /*信號量數據結構*/
  21.                                                                                 /*一個信號量數據結構包含兩個字節,第一個字節為信號量值*/
  22. unsigned char data lqSemData[lqSemMax*2]={                        /*第二個字節為等待這個信號量的任務*/
  23.         0,0,
  24.         0,0};
  25. #endif                                                                                               

  26. #if        LQ_MSG_EN                                                        /*消息郵箱數據結構*/
  27.                                                                                 /*一個消息郵箱數據結構包含兩個字節,第一個字節為消息郵箱的消息*/
  28. unsigned char data lqMsgData[lqMsgMax*2]={                /*第二個字節的最高位表示當前郵箱是否有消息0表示沒有1表示有,其他為表示等待這個消息的任務*/
  29.         0,0,
  30.         0,0};       
  31. #endif               


  32. const unsigned char code lqMap[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};

  33. /*****************************************
  34. /                        開始函數
  35. /這個函數必須在0號任務的最開始的部分調用
  36. /        void lqStart(void);
  37. /*****************************************/
  38. void lqStart(void)
  39. {
  40.         char i;
  41.         for(i=0;i<lqMaxID;++i){
  42.                 lqTaskTimer[i] = 0;
  43.                 lqTaskState[i] = 0;
  44.         }
  45.         lqRdyTbl=0xFF;
  46.         lqSwitchType=0x00;
  47.         EA = 1;
  48.         lqCrt=0;
  49. }

  50. #if LQ_DELAY_EN
  51. /***************************************************
  52. /                                任務延遲一段時間
  53. /        void lqDelay(unsigned char tmo)
  54. /        tmo:延遲時間,如果tmo=0,那么當前任務休眠,永遠不會被激活
  55. /***************************************************/
  56. void lqDelay(unsigned char tmo)
  57. {
  58.         EA=0;
  59.         lqRdyTbl &= ~lqMap[lqCrt];
  60.         lqTaskState[lqCrt] |= K_TMO;
  61.         lqTaskTimer[lqCrt] = tmo;
  62.         EA=1;
  63.         lqSche();
  64. }
  65. #endif

  66. #if        LQ_FLG_EN
  67. /**************************************************
  68. /                        等待標志
  69. /        char lqWaitFlg(unsigned char index,unsigned char tmo)
  70. /        index        :標志量索引
  71. /        tmo                :超時等待時限,如果為0則無限等待
  72. /        返回值:
  73. /        標志量有效返回 K_FLG
  74. /        在規定的時件內標志量未發生返回 K_TMO
  75. /*************************************************/
  76. unsigned char lqWaitFlg(unsigned char index,unsigned char tmo)
  77. {
  78.         EA=0;
  79.         lqTaskState[lqCrt] &= 0xF0;
  80.         lqTaskState[lqCrt]|= index;                /*將當前索引值保存到任務狀態的低四位,因為函數是不可重入的*/
  81.         if(lqFlgData[index] & 0x80){        /*標志量置位*/
  82.                 lqFlgData[index] = 0;
  83.                 EA=1;
  84.                 return        K_FLG;
  85.         }
  86.         lqFlgData[index] |= lqMap[lqCrt];
  87.         lqRdyTbl &= ~lqMap[lqCrt];
  88.         lqTaskState[lqCrt] |= K_FLG;
  89.         if(tmo){
  90.                 lqTaskState[lqCrt] |= K_TMO;
  91.                 lqTaskTimer[lqCrt] = tmo;
  92.         }
  93.         EA=1;
  94.         lqSche();
  95.         EA=0;
  96.         index = 0x0F & lqTaskState[lqCrt];
  97.         if(lqTaskState[lqCrt] & K_FLG){                /*等待超時返回*/
  98.                 lqTaskState[lqCrt] &= ~K_FLG;
  99.                 lqTaskState[lqCrt] |= K_TMO;        /*超時標記*/
  100.                 lqFlgData[index] &= ~lqMap[lqCrt];
  101.                 EA=1;
  102.                 return K_TMO;
  103.         }
  104.         EA=1;
  105.         return K_FLG;
  106. }

  107. /************************************************
  108. /                中斷服務子程序發送標志
  109. /        char lqSendFlgISR(unsigned char index)
  110. /        index        :標志量索引
  111. /        返回值:
  112. /        1 -- 有更高優先級的任務就緒,需要執行任務切換
  113. /        0 -- 不需要執行任務切換
  114. /************************************************/
  115. char lqSendFlgISR(unsigned char index)
  116. {
  117.         char i,j;
  118.         EA=0;
  119.         if(lqFlgData[index] & 0x7F){                /*有任務在等待這個標志事件*/
  120.                 i=lqMaxID;
  121.                 for(j=0;j<lqMaxID;++j){
  122.                         if(lqFlgData[index] & lqMap[j]){        /*任務等待這個標志事件*/
  123.                                 lqTaskState[j] &= ~K_FLG;
  124.                                 lqTaskState[j] &= ~K_TMO;
  125.                                 lqTaskTimer[j] = 0;
  126.                                 lqRdyTbl |= lqMap[j];
  127.                                 if(j<i){
  128.                                         i=j;
  129.                                 }
  130.                         }
  131.                 }
  132.                 lqFlgData[index] = 0;
  133.                 if(i<lqCrt){
  134.                         EA=1;
  135.                         return 1;
  136.                 }
  137.                 EA=1;
  138.                 return 0;
  139.         }
  140.         lqFlgData[index] = 0x80;
  141.         EA=1;
  142.         return 0;
  143. }
  144. /********************************************
  145. /                任務發送標志事件
  146. /        void lqSendFlg(unsigned char index)
  147. /********************************************/
  148. void lqSendFlg(unsigned char index)
  149. {
  150.         if(lqSendFlgISR(index)){
  151.                 lqSche();
  152.         }
  153. }
  154. #endif

  155. #if        LQ_SEM_EN
  156. /************************************************
  157. /                等待一個信號量
  158. /        char lqWaitSem(unsigned char index,unsigned char tmo)
  159. /        index        :信號量索引
  160. /        tmo                :超時等待時限,如果為0則無限等待
  161. /        返回值:
  162. /        信號量有效返回 K_SEM
  163. /        在規定的時件內信號量未發生返回 K_TMO
  164. /************************************************/
  165. unsigned char lqWaitSem(unsigned char index,unsigned char tmo)
  166. {
  167.         EA=0;
  168.         lqTaskState[lqCrt] &= 0xF0;
  169.         lqTaskState[lqCrt]|= index;                /*因為函數不可重,所以存儲信號量索引*/
  170.         index *= 2;
  171.         if(lqSemData[index]){
  172.                 --lqSemData[index];
  173.                 EA=1;
  174.                 return K_SEM;
  175.         }
  176.         lqRdyTbl &= ~lqMap[lqCrt];
  177.         lqSemData[index+1] |= lqMap[lqCrt];
  178.         lqTaskState[lqCrt] |= K_SEM;
  179.         if(tmo){
  180.                 lqTaskState[lqCrt] |= K_TMO;
  181.                 lqTaskTimer[lqCrt] = tmo;
  182.         }
  183.         EA=1;
  184.         lqSche();
  185.         EA=0;
  186.         index = 0x0F & lqTaskState[lqCrt];
  187.         if(lqTaskState[lqCrt] & K_SEM){
  188.                 lqTaskState[lqCrt] &= ~K_SEM;
  189.                 lqTaskState[lqCrt] |= K_TMO;
  190.                 lqSemData[index*2+1] &= ~lqMap[lqCrt];
  191.                 EA=1;
  192.                 return K_TMO;
  193.         }
  194.         EA=1;
  195.         return K_SEM;
  196. }

  197. /*********************************************
  198. /                        中斷子程序發送信號量
  199. /        char lqSendSemISR(unsigned char index)
  200. /        index        :信號量索引
  201. /        返回值:
  202. /        1 -- 有更高優先級的任務就緒,需要執行任務切換
  203. /        0 -- 不需要執行任務切換
  204. /*********************************************/
  205. char lqSendSemISR(unsigned char index)
  206. {
  207.         char j;
  208.         EA=0;
  209.         index*=2;
  210.         if(lqSemData[index+1]){
  211.                 for(j=0;j<lqMaxID;++j){
  212.                         if(lqSemData[index+1] & lqMap[j]){
  213.                                 break;
  214.                         }
  215.                 }
  216.                 lqSemData[index+1] &= ~lqMap[j];
  217.                 lqTaskTimer[j] = 0;
  218.                 lqTaskState[j] &= ~K_SEM;
  219.                 lqTaskState[j] &= ~K_TMO;
  220.                 lqRdyTbl |= lqMap[j];
  221.                 if(j<lqCrt){
  222.                         EA=1;
  223.                         return 1;
  224.                 }
  225.                 EA=1;
  226.                 return 0;
  227.         }
  228.         ++lqSemData[index];
  229.         EA=1;
  230.         return 0;
  231. }

  232. /************************************************
  233. /                        任務發送信號量
  234. /        void lqSendSem(unsinged char index)
  235. /***********************************************/
  236. void lqSendSem(unsigned char index)
  237. {
  238.         if(lqSendSemISR(index)){
  239.                 lqSche();
  240.         }
  241. }
  242. #endif

  243. #if        LQ_MSG_EN
  244. /*********************************************
  245. /                        等待消息郵箱
  246. /        unsigned char lqWaitMsg(unsigned char index,unsigned char tmo)
  247. /        index        :郵箱索引
  248. /        tmo                :超時等待時限,如果為0則無限等待
  249. /        郵箱中的值為 0x00~0xFF
  250. /        lqMsgData[2*index+1]的最高為0表示當前沒郵件,
  251. /        否則表示當前有郵件
  252. /        返回值        :
  253. /        如果在規定的時間內有消息送到這個郵箱,則返回這個消息
  254. /        如果在規定的時間內沒有消息送到這個郵箱,則返回 MSG_TMO
  255. /        MSG_TMO在lq51.h中有定義,如果 MSG_TMO 所代表的值是消息的一部分,
  256. /        那么在這個函數返回后建議執行char lqIsTaskTmo(),檢查當前任務
  257. /        是否從超時返回
  258. /********************************************/
  259. unsigned char lqWaitMsg(unsigned char index,unsigned char tmo)
  260. {
  261.         EA=0;
  262.         lqTaskState[lqCrt] &= 0xF0;
  263.         lqTaskState[lqCrt] |= index;
  264.         index *= 2;
  265.         if(lqMsgData[index+1] & 0x80){                /*當前郵箱有郵件*/
  266.                 lqMsgData[index+1] &= 0x7F;
  267.                 EA=1;
  268.                 return lqMsgData[index];
  269.         }
  270.         lqRdyTbl &= ~lqMap[lqCrt];
  271.         lqMsgData[index+1] |= lqMap[lqCrt];
  272.         lqTaskState[lqCrt] |= K_MSG;
  273.         if(tmo){
  274.                 lqTaskState[lqCrt] |= K_TMO;
  275.                 lqTaskTimer[lqCrt] = tmo;
  276.         }
  277.         EA=1;
  278.         lqSche();
  279.         EA=0;
  280.         index = lqTaskState[lqCrt] & 0x0F;
  281.         index *= 2;
  282.         if(lqTaskState[lqCrt] & K_MSG){
  283.                 lqTaskState[lqCrt] &= ~K_MSG;
  284.                 lqTaskState[lqCrt] |= K_TMO;
  285.                 lqMsgData[index] = MSG_TMO;
  286.                 lqMsgData[index+1] &= ~lqMap[lqCrt];
  287.         }
  288.         lqMsgData[index+1] &= 0x7F;
  289.         EA=1;
  290.         return lqMsgData[index];
  291. }
  292. /*******************************************
  293. /        char lqIsMsgEmpty(unsigned char index)
  294. /        index        :郵箱索引
  295. /        返回值:
  296. /        0 -- 郵箱中有消息
  297. /        1 -- 郵箱中沒有消息
  298. /******************************************/
  299. #if        LQ_CHK_MSG_EN
  300. char lqIsMsgEmpty(unsigned char index)
  301. {
  302.         EA=0;
  303.         if(lqMsgData[index*2+1] & 0x80){
  304.                 EA=1;
  305.                 return 0;                                /*郵箱不為空,郵箱中有消息*/
  306.         }
  307.         EA=1;
  308.         return 1;                                        /*郵箱為空,郵箱中無消息*/
  309. }
  310. #endif
  311. /*******************************************
  312. /                中斷子程序發送郵件
  313. /        char lqSendMsgISR(unsigned char index,unsigned char Msg)
  314. /        index        :郵箱索引
  315. /        Msg                :被發送的消息
  316. /        如果郵箱中已經存在消息,原來的消息不會被覆蓋,函數之間返回
  317. /        返回值:
  318. /        1 -- 有更高優先級的任務就緒,需要執行任務切換
  319. /        0 -- 不需要執行任務切換
  320. /*******************************************/
  321. char lqSendMsgISR(unsigned char index,unsigned char Msg)
  322. {
  323.         EA=0;
  324.         index *= 2;
  325.         if(lqMsgData[index+1] & 0x80){
  326.                 EA=1;
  327.                 return 0;                                        /*郵箱不為空*/
  328.         }
  329.         lqMsgData[index] = Msg;
  330.         ++index;
  331.         lqMsgData[index] |= 0x80;
  332.         if(lqMsgData[index] & 0x7F){        /*有任務在等待這個郵件*/
  333.                 for(Msg=0;Msg<lqMaxID;++Msg){
  334.                         if(lqMsgData[index] & lqMap[Msg]){
  335.                                 break;
  336.                         }
  337.                 }
  338.                 lqMsgData[index] &= ~lqMap[Msg];
  339.                 lqTaskState[Msg] &= ~K_MSG;
  340.                 lqRdyTbl |= lqMap[Msg];
  341.                 if(Msg < lqCrt){
  342.                         EA=1;
  343.                         return 1;
  344.                 }
  345.         }
  346.         EA = 1;
  347.         return 0;
  348. }
  349. /********************************************
  350. /                任務發送郵件
  351. /        void lqSendMsg(unsigned char index,unsigned char Msg)
  352. /        index        :郵箱索引
  353. /        Msg                :被發送的消息
  354. /*******************************************/
  355. void lqSendMsg(unsigned char index,unsigned char Msg)
  356. {
  357.         if(lqSendMsgISR(index,Msg)){
  358.                 lqSche();
  359.         }
  360. }
  361. #endif

  362. #if        LQ_TASK_TMO_CHK_EN
  363. /***********************************************
  364. /                        任務超時返回檢查
  365. /        char lqIsTaskTmo()
  366. /        如果當前任務是超時返回,那么任務狀態(lqTaskState)的B_TMO位置位,否則該標志位被清零
  367. /        一般情況下只針對郵箱事件返回后做超時檢查
  368. /        返回值:
  369. /        1  --  當前任務是超時返回
  370. /        0  --  當前任務不是超時返回
  371. /************************************************/
  372. char lqIsTaskTmo()
  373. {
  374.         if(lqTaskState[lqCrt] & K_TMO){
  375.                 return 1;
  376.         }
  377.         return 0;
  378. }
  379. #endif
復制代碼

所有資料51hei提供下載:
lq51-master.zip (216.94 KB, 下載次數: 18)


評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 精品中文字幕一区二区 | 亚洲一区二区三区在线 | 欧美aaa级| www国产成人免费观看视频,深夜成人网 | 久久国产精品99久久久久 | 啪啪精品 | 中文字幕日韩欧美一区二区三区 | 欧美操操操 | www在线视频 | 久久久久久91香蕉国产 | 在线一级片| 精品久久久久久久久久久久久久 | 国产一区不卡在线观看 | 亚洲视频区 | 欧美日韩在线综合 | 国产精品一区二区视频 | 日韩国产免费观看 | 亚洲一区二区精品视频在线观看 | 欧美高清视频在线观看 | 国产91色在线 | 亚洲 | 免费成人国产 | 欧美成人精品欧美一级 | 国产精品久久久久久吹潮 | 国产精品久久久久久久久 | 天天色官网 | 97精品国产 | 亚洲高清免费视频 | 日韩不卡视频在线 | 91亚洲国产成人久久精品网站 | 亚洲成人av一区二区 | 亚洲欧美日韩在线 | 麻豆精品国产免费 | 91精品国产色综合久久不卡98 | 国产精品一区二区av | 在线观看www | 91精品国产色综合久久 | 国产精品99久久久久久久vr | 最新中文字幕在线 | 国产精品一区二区三区久久 | 在线视频一区二区 | 亚洲欧洲色视频 |