日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > > 大橙子瘋嵌入式


前言

在計算機科學(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,否則會影響代碼閱讀,當然,不排除封裝成一個特殊的功能外



本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
關(guān)閉