查詢式協(xié)作多任務(wù)系統(tǒng)
掃描二維碼
隨時隨地手機看文章
前言
在計算機科學(xué)領(lǐng)域,任務(wù)調(diào)度和協(xié)作是關(guān)鍵的概念。雖然傳統(tǒng)的操作系統(tǒng)提供了各種任務(wù)調(diào)度算法和機制,但有時我們需要更靈活、個性化的任務(wù)管理方式。
即使采用定時器實現(xiàn)時間片論法任務(wù)調(diào)度,但是也必須等單個完整的任務(wù)執(zhí)行完成后才能執(zhí)行下一個完整的任務(wù)。
本文將介紹使用標準庫頭文件中的setjmp和longjmp函數(shù)構(gòu)建一個簡單的查詢式協(xié)作多任務(wù)系統(tǒng),無需使用定時器進行任務(wù)切換。
setjmp和longjmp是C語言標準庫頭文件
中提供的函數(shù)。它們的功能是實現(xiàn)非局部跳轉(zhuǎn),可以在程序的不同位置之間進行跳轉(zhuǎn),類似于goto語句的擴展。這種非局部跳轉(zhuǎn)的能力為我們構(gòu)建查詢式協(xié)作多任務(wù)系統(tǒng)提供了基礎(chǔ)。
介紹
setjmp和longjmp
setjmp函數(shù)用于保存當前程序狀態(tài),創(chuàng)建一個可以供后續(xù)longjmp函數(shù)跳轉(zhuǎn)的上下文環(huán)境。在調(diào)用setjmp時,程序會記錄當前的程序計數(shù)器、寄存器和堆棧等狀態(tài)信息,并將這些信息保存在一個jmp_buf結(jié)構(gòu)中。同時,setjmp函數(shù)返回0作為普通調(diào)用的返回值,并將jmp_buf作為標識符存儲起來。
不同平臺的jmp_buf的類型定義不一樣,大概占用不到30個字節(jié),因為不同平臺的相關(guān)寄存器等不一樣,因此占用的大小也不同。
longjmp函數(shù)則實現(xiàn)了對保存的上下文環(huán)境的跳轉(zhuǎn)操作。通過傳遞之前由setjmp函數(shù)保存的jmp_buf標識符,longjmp函數(shù)會將程序的狀態(tài)還原到對應(yīng)的上下文環(huán)境,并且會返回到setjmp處繼續(xù)執(zhí)行。
協(xié)作式
在協(xié)作式多任務(wù)調(diào)度下,當前任務(wù)需要通過主動放棄時間片提供給其他任務(wù)運行,而并非是被其他任務(wù)搶占,因此這里面并沒有所謂的優(yōu)先級概念之分。
雖然協(xié)作式?jīng)]有所謂的優(yōu)先級概念之分,但是可以通過一定的方式也能實現(xiàn)一個簡單的優(yōu)先級,比如當前任務(wù)主動放棄時間片后,查詢更高優(yōu)先級的任務(wù)運行。
時間片論法任務(wù)調(diào)度只能等任務(wù)運行完成才會給下一個任務(wù)時間片運行,并不存在主動放棄時間片的功能。
實現(xiàn)思路
了解到setjmp和longjmp的功能和原理后,我們能不能通過它們來構(gòu)建一個任務(wù)調(diào)度算法和機制呢?
雖然setjmp可以記錄當前的程序計數(shù)器、寄存器和堆棧等狀態(tài)信息,但是實現(xiàn)多任務(wù)切換時堆棧里面的數(shù)據(jù)是會發(fā)生變化的。
jmp_buf只記錄堆棧指針,不記錄堆棧指針指向的數(shù)據(jù)內(nèi)容。
因此,如果要實現(xiàn)多任務(wù)切換,則需要為每個任務(wù)分配一定的堆棧預(yù)留空間,由于不使用堆,因此可以只考慮棧分配即可。
當前任務(wù)主動放棄時間片后,不斷查詢滿足條件需要執(zhí)行的其他任務(wù)。
代碼實現(xiàn)
創(chuàng)建任務(wù),使用了setjmp函數(shù)。
int cotOs_Creat(OsTask_cb pfnOsTaskEnter, size_t stack) { size_t oldsp; if (sg_OsInfo.taskNum >= COT_OS_MAX_TASK || sg_OsInfo.pfnGetTimerMs == NULL) { return -1; } COT_OS_GET_STACK(oldsp); COT_OS_SET_STACK(sg_OsInfo.stackTop); if (0 == setjmp(sg_OsInfo.tcb[sg_OsInfo.taskNum].env)) { COT_OS_SET_STACK(oldsp); sg_OsInfo.tcb[sg_OsInfo.taskNum].pfnOsTaskEnter = pfnOsTaskEnter; sg_OsInfo.taskNum++; sg_OsInfo.stackTop -= stack; } else { sg_OsInfo.tcb[sg_OsInfo.taskId].pfnOsTaskEnter(); } return 0; }
放棄時間片,使用了longjmp,這里集成了時間等待功能,即放棄時間片的時長(即使時長減至0也要等待其他任務(wù)主動放棄時間片才會運行)
void cotOs_WaitFor(uint32_t time) { uint32_t timer = sg_OsInfo.pfnGetTimerMs(); setjmp(sg_OsInfo.tcb[sg_OsInfo.taskId].env); if (!(sg_OsInfo.pfnGetTimerMs() - timer > time)) { sg_OsInfo.taskId++; if (sg_OsInfo.taskId >= sg_OsInfo.taskNum) { sg_OsInfo.taskId = 0; } longjmp(sg_OsInfo.tcb[sg_OsInfo.taskId].env, 1); } }
除了繼承時間等待功能外,還定義了一個等待條件的主動放棄放棄時間片功能,即條件不滿足時主動放棄時間片(即使條件滿足后也要等待其他任務(wù)主動放棄時間片才會運行)
#define cotOs_WaitFor_Cond(cond) do{\
extern jmp_buf *cotOs_GetTaskEnv1(void);\
setjmp((*cotOs_GetTaskEnv1()));\ if (!(cond)){\
extern void cotOs_RunNextTask(void);\
cotOs_RunNextTask();\
}\
}while (0)
代碼鏈接
目前已完成在STM32板子上的查詢式協(xié)作多任務(wù)系統(tǒng):
下載鏈接(點擊閱讀原文):
https://gitee.com/cot_package/cot_os
擴展
setjmp和longjmp除了實現(xiàn)一個多任務(wù)系統(tǒng)外,其實還可以有其他的用法,比如實現(xiàn)C++中的try catch拋出異常處理功能。
由于其特性問題,在實際代碼中,特別是應(yīng)用程序代碼上,建議少用setjmp和longjmp,否則會影響代碼閱讀,當然,不排除封裝成一個特殊的功能外





