www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當(dāng)前位置:首頁(yè) > 通信技術(shù) > 雷達(dá)通信電子戰(zhàn)
[導(dǎo)讀]關(guān)注、星標(biāo)公眾號(hào),直達(dá)精彩內(nèi)容ID:技術(shù)讓夢(mèng)想更偉大整理:李肖遙回顧任務(wù)的創(chuàng)建刪除在FreeRTOS基礎(chǔ)系列《FreeRTOS系列第10篇---FreeRTOS任務(wù)創(chuàng)建和刪除》中介紹了任務(wù)創(chuàng)建API函數(shù)xTaskCreate(),我們這里先回顧一下這個(gè)函數(shù)的聲明:BaseType...


關(guān)注、星標(biāo)公眾號(hào),直達(dá)精彩內(nèi)容

ID:技術(shù)讓夢(mèng)想更偉大

整理:李肖遙


回顧任務(wù)的創(chuàng)建刪除

在FreeRTOS基礎(chǔ)系列《FreeRTOS系列第10篇---FreeRTOS任務(wù)創(chuàng)建和刪除》中介紹了任務(wù)創(chuàng)建API函數(shù)xTaskCreate(),我們這里先回顧一下這個(gè)函數(shù)的聲明:

BaseType_t?xTaskCreate(
????TaskFunction_tp?vTaskCode,
????const?char?*?constpcName,
????unsigned?short?usStackDepth,
????void?*pvParameters,
????UBaseType_t?uxPriority,
????TaskHandle_t?*pvCreatedTask
);
這個(gè)API函數(shù)的作用是創(chuàng)建新的任務(wù)并將它加入到任務(wù)就緒列表,函數(shù)參數(shù)含義為:

  • 「pvTaskCode」:函數(shù)指針,指向任務(wù)函數(shù)的入口。任務(wù)永遠(yuǎn)不會(huì)返回(位于死循環(huán)內(nèi))。該參數(shù)類型TaskFunction_t定義在文件projdefs.h中,定義為:typedef void(*TaskFunction_t)( void * ),即參數(shù)為空指針類型并返回空類型。

  • 「pcName」:任務(wù)描述。主要用于調(diào)試。字符串的最大長(zhǎng)度(包括字符串結(jié)束字符)由宏configMAX_TASK_NAME_LEN指定,該宏位于FreeRTOSConfig.h文件中。

  • 「usStackDepth」:指定任務(wù)堆棧大小,能夠支持的堆棧變量數(shù)量(堆棧深度),而不是字節(jié)數(shù)。比如,在16位寬度的堆棧下,usStackDepth定義為100,則實(shí)際使用200字節(jié)堆棧存儲(chǔ)空間。堆棧的寬度乘以深度必須不超過(guò)size_t類型所能表示的最大值。比如,size_t為16位,則可以表示堆棧的最大值是65535字節(jié)。這是因?yàn)槎褩T谏暾?qǐng)時(shí)是以字節(jié)為單位的,申請(qǐng)的字節(jié)數(shù)就是堆棧寬度乘以深度,如果這個(gè)乘積超出size_t所表示的范圍,就會(huì)溢出,分配的堆??臻g也不是我們想要的。

  • 「pvParameters」:指針,當(dāng)任務(wù)創(chuàng)建時(shí),作為一個(gè)參數(shù)傳遞給任務(wù)。

  • 「uxPriority」:任務(wù)的優(yōu)先級(jí)。具有MPU支持的系統(tǒng),可以通過(guò)置位優(yōu)先級(jí)參數(shù)的portPRIVILEGE_BIT位,隨意的在特權(quán)(系統(tǒng))模式下創(chuàng)建任務(wù)。比如,創(chuàng)建一個(gè)優(yōu)先級(jí)為2的特權(quán)任務(wù),參數(shù)uxPriority可以設(shè)置為 ( 2 | portPRIVILEGE_BIT )

  • 「pvCreatedTask」:用于回傳一個(gè)句柄(ID),創(chuàng)建任務(wù)后可以使用這個(gè)句柄引用任務(wù)。

雖然xTaskCreate()看上去很像函數(shù),但其實(shí)是一個(gè)宏,真正被調(diào)用的函數(shù)是xTaskGenericCreate()xTaskCreate()宏定義如下所示:

#define?xTaskCreate(?pvTaskCode,?pcName,?usStackDepth,pvParameters,?uxPriority,?pxCreatedTask?)????\
??????xTaskGenericCreate(?(?pvTaskCode?),(?pcName?),?(?usStackDepth?),?(?pvParameters?),?(?uxPriority?),?(?pxCreatedTask),?(?NULL?),?(?NULL?),?(?NULL?)?)
可以看到,xTaskCreatexTaskGenericCreate少了三個(gè)參數(shù),在宏定義中,這三個(gè)參數(shù)被設(shè)置為NULL。

這三個(gè)參數(shù)用于使用靜態(tài)變量的方法分配堆棧、任務(wù)TCB空間以及設(shè)置MPU相關(guān)的參數(shù)。

一般情況下,這三個(gè)參數(shù)是不使用的,所以任務(wù)創(chuàng)建宏xTaskCreate定義的時(shí)候,將這三個(gè)參數(shù)對(duì)用戶隱藏了。

接下來(lái)的章節(jié)中,為了方便,我們還是稱xTaskCreate()為函數(shù),雖然它是一個(gè)宏定義。

上面我們提到了任務(wù)TCB(任務(wù)控制塊),這是一個(gè)需要重點(diǎn)介紹的關(guān)鍵點(diǎn)。

它用于存儲(chǔ)任務(wù)的狀態(tài)信息,包括任務(wù)運(yùn)行時(shí)的環(huán)境。每個(gè)任務(wù)都有自己的任務(wù)TCB。

任務(wù)TCB是一個(gè)相對(duì)比較大的數(shù)據(jù)結(jié)構(gòu),這也是情理之中的,因?yàn)榕c任務(wù)相關(guān)的代碼占到整個(gè)FreeRTOS代碼量的一半左右,這些代碼大都與任務(wù)TCB相關(guān)。

「我們先來(lái)介紹一下任務(wù)TCB數(shù)據(jù)結(jié)構(gòu)的定義」

typedef?struct?tskTaskControlBlock
{
????volatile?StackType_t????*pxTopOfStack;?/*當(dāng)前堆棧的棧頂,必須位于結(jié)構(gòu)體的第一項(xiàng)*/
?
????#if?(?portUSING_MPU_WRAPPERS?==?1?)
????????xMPU_SETTINGS???xMPUSettings;??????/*MPU設(shè)置,必須位于結(jié)構(gòu)體的第二項(xiàng)*/
????#endif
?
????ListItem_t??????????xStateListItem;?/*任務(wù)的狀態(tài)列表項(xiàng),以引用的方式表示任務(wù)的狀態(tài)*/
????ListItem_t??????????xEventListItem;????/*事件列表項(xiàng),用于將任務(wù)以引用的方式掛接到事件列表*/
????UBaseType_t?????????uxPriority;????????/*保存任務(wù)優(yōu)先級(jí),0表示最低優(yōu)先級(jí)*/
????StackType_t?????????*pxStack;???????????/*指向堆棧的起始位置*/
????char???????????????pcTaskName[?configMAX_TASK_NAME_LEN?];/*任務(wù)名字*/
?
????#if?(?portSTACK_GROWTH?>?0?)
????????StackType_t?????*pxEndOfStack;?????/*指向堆棧的尾部*/
????#endif
?
????#if?(?portCRITICAL_NESTING_IN_TCB?==?1?)
????????UBaseType_t?????uxCriticalNesting;?/*保存臨界區(qū)嵌套深度*/
????#endif
?
????#if?(?configUSE_TRACE_FACILITY?==?1?)
????????UBaseType_t?????uxTCBNumber;???????/*保存一個(gè)數(shù)值,每個(gè)任務(wù)都有唯一的值*/
????????UBaseType_t?????uxTaskNumber;??????/*存儲(chǔ)一個(gè)特定數(shù)值*/
????#endif
?
????#if?(?configUSE_MUTEXES?==?1?)
????????UBaseType_t?????uxBasePriority;????/*保存任務(wù)的基礎(chǔ)優(yōu)先級(jí)*/
????????UBaseType_t?????uxMutexesHeld;
????#endif
?
????#if?(?configUSE_APPLICATION_TASK_TAG?==?1?)
????????TaskHookFunction_t?pxTaskTag;
????#endif
?
????#if(?configNUM_THREAD_LOCAL_STORAGE_POINTERS?>?0?)
????????void?*pvThreadLocalStoragePointers[configNUM_THREAD_LOCAL_STORAGE_POINTERS?];
????#endif
?
????#if(?configGENERATE_RUN_TIME_STATS?==?1?)
????????uint32_t????????ulRunTimeCounter;??/*記錄任務(wù)在運(yùn)行狀態(tài)下執(zhí)行的總時(shí)間*/
????#endif
?
????#if?(?configUSE_NEWLIB_REENTRANT?==?1?)
????????/*?為任務(wù)分配一個(gè)Newlibreent結(jié)構(gòu)體變量。Newlib是一個(gè)C庫(kù)函數(shù),并非FreeRTOS維護(hù),F(xiàn)reeRTOS也不對(duì)使用結(jié)果負(fù)責(zé)。如果用戶使用Newlib,必須熟知Newlib的細(xì)節(jié)*/
????????struct?_reent?xNewLib_reent;
????#endif
?
????#if(?configUSE_TASK_NOTIFICATIONS?==?1?)
????????volatile?uint32_t?ulNotifiedValue;?/*與任務(wù)通知相關(guān)*/
????????volatile?uint8_t?ucNotifyState;
????#endif
?
????#if(?configSUPPORT_STATIC_ALLOCATION?==?1?)
????????uint8_t?ucStaticAllocationFlags;?/*?如果堆棧由靜態(tài)數(shù)組分配,則設(shè)置為pdTRUE,如果堆棧是動(dòng)態(tài)分配的,則設(shè)置為pdFALSE*/
????#endif
?
????#if(?INCLUDE_xTaskAbortDelay?==?1?)
????????uint8_t?ucDelayAborted;
????#endif
?
}?tskTCB;
?
typedef?tskTCB?TCB_t;
「下面我們?cè)敿?xì)的介紹這個(gè)數(shù)據(jù)結(jié)構(gòu)的主要成員:」

指針pxTopOfStack必須位于結(jié)構(gòu)體的第一項(xiàng),指向當(dāng)前堆棧的棧頂,對(duì)于向下增長(zhǎng)的堆棧,pxTopOfStack總是指向最后一個(gè)入棧的項(xiàng)目。

如果使用MPU,xMPUSettings必須位于結(jié)構(gòu)體的第二項(xiàng),用于MPU設(shè)置。

接下來(lái)是狀態(tài)列表項(xiàng)xStateListItem和事件列表項(xiàng)xEventListItem,我們?cè)谏弦徽陆榻B列表和列表項(xiàng)的文章中提到過(guò):列表被FreeRTOS調(diào)度器使用,用于跟蹤任務(wù),處于就緒、掛起、延時(shí)的任務(wù),都會(huì)被掛接到各自的列表中。

調(diào)度器就是通過(guò)把任務(wù)TCB中的狀態(tài)列表項(xiàng)xStateListItem和事件列表項(xiàng)xEventListItem掛接到不同的列表中來(lái)實(shí)現(xiàn)上述過(guò)程的。

在task.c中,定義了一些靜態(tài)列表變量,其中有就緒、阻塞、掛起列表,例如當(dāng)某個(gè)任務(wù)處于就緒態(tài)時(shí),調(diào)度器就將這個(gè)任務(wù)TCB的xStateListItem列表項(xiàng)掛接到就緒列表。

事件列表項(xiàng)也與之類似,當(dāng)隊(duì)列滿的情況下,任務(wù)因入隊(duì)操作而阻塞時(shí),就會(huì)將事件列表項(xiàng)掛接到隊(duì)列的等待入隊(duì)列表上。

uxPriority用于保存任務(wù)的優(yōu)先級(jí),0為最低優(yōu)先級(jí)。任務(wù)創(chuàng)建時(shí),指定的任務(wù)優(yōu)先級(jí)就被保存到該變量中。

指針pxStack指向堆棧的起始位置,任務(wù)創(chuàng)建時(shí)會(huì)分配指定數(shù)目的任務(wù)堆棧,申請(qǐng)堆棧內(nèi)存函數(shù)返回的指針就被賦給該變量。

很多剛接觸FreeRTOS的人會(huì)分不清指針pxTopOfStackpxStack的區(qū)別,「這里簡(jiǎn)單說(shuō)一下:」

pxTopOfStack指向當(dāng)前堆棧棧頂,隨著進(jìn)棧出棧,pxTopOfStack指向的位置是會(huì)變化的;

pxStack指向當(dāng)前堆棧的起始位置,一經(jīng)分配后,堆棧起始位置就固定了,不會(huì)被改變了。

「那么為什么需要pxStack變量呢?」

這是因?yàn)殡S著任務(wù)的運(yùn)行,堆??赡軙?huì)溢出,在堆棧向下增長(zhǎng)的系統(tǒng)中,這個(gè)變量可用于檢查堆棧是否溢出;

如果在堆棧向上增長(zhǎng)的系統(tǒng)中,要想確定堆棧是否溢出,還需要另外一個(gè)變量pxEndOfStack來(lái)輔助診斷是否堆棧溢出,后面會(huì)講到這個(gè)變量。

字符數(shù)組pcTaskName用于保存任務(wù)的描述或名字,在任務(wù)創(chuàng)建時(shí),由參數(shù)指定。

名字的長(zhǎng)度由宏configMAX_TASK_NAME_LEN(位于FreeRTOSConfig.h中)指定,包含字符串結(jié)束標(biāo)志。

如果堆棧向上生長(zhǎng)(portSTACK_GROWTH > 0),指針pxEndOfStack指向堆棧尾部,用于檢驗(yàn)堆棧是否溢出。

變量uxCriticalNesting用于保存臨界區(qū)嵌套深度,初始值為0。

接下來(lái)兩個(gè)變量用于可視化追蹤,僅當(dāng)宏configUSE_TRACE_FACILITY(位于FreeRTOSConfig.h中)為1時(shí)有效。

變量uxTCBNumber存儲(chǔ)一個(gè)數(shù)值,在創(chuàng)建任務(wù)時(shí)由內(nèi)核自動(dòng)分配數(shù)值(通常每創(chuàng)建一個(gè)任務(wù),值增加1),每個(gè)任務(wù)的uxTCBNumber值都不同,主要用于調(diào)試。

變量uxTaskNumber用于存儲(chǔ)一個(gè)特定值,與變量uxTCBNumber不同,uxTaskNumber的數(shù)值不是由內(nèi)核分配的,而是通過(guò)API函數(shù)vTaskSetTaskNumber()來(lái)設(shè)置的,數(shù)值由函數(shù)參數(shù)指定。

如果使用互斥量(configUSE_MUTEXES == 1),任務(wù)優(yōu)先級(jí)被臨時(shí)提高時(shí),變量uxBasePriority用來(lái)保存任務(wù)原來(lái)的優(yōu)先級(jí)。

變量ucStaticAllocationFlags也需要說(shuō)明一下,我們前面說(shuō)過(guò)任務(wù)創(chuàng)建API函數(shù)xTaskCreate()只能使用動(dòng)態(tài)內(nèi)存分配的方式創(chuàng)建任務(wù)堆棧和任務(wù)TCB,如果要使用靜態(tài)變量實(shí)現(xiàn)任務(wù)堆棧和任務(wù)TCB就需要使用函數(shù)xTaskGenericCreate()來(lái)實(shí)現(xiàn)。

如果任務(wù)堆?;蛉蝿?wù)TCB由靜態(tài)數(shù)組和靜態(tài)變量實(shí)現(xiàn),則將該變量設(shè)置為pdTRUE(任務(wù)堆??臻g由靜態(tài)數(shù)組變量實(shí)現(xiàn)時(shí)為0x01,任務(wù)TCB由靜態(tài)變量實(shí)現(xiàn)時(shí)為0x02,任務(wù)堆棧和任務(wù)TCB都由靜態(tài)變量實(shí)現(xiàn)時(shí)為0x03),如果堆棧是動(dòng)態(tài)分配的,則將該變量設(shè)置為pdFALSE。

到這里任務(wù)TCB的數(shù)據(jù)結(jié)構(gòu)就講完了,下面我們用一個(gè)例子「來(lái)講述任務(wù)創(chuàng)建的過(guò)程」,為方便起見(jiàn),假設(shè)被創(chuàng)建的任務(wù)叫“任務(wù)A”,任務(wù)函數(shù)為vTask_A():

TaskHandle_t xHandle;
xTaskCreate(vTask_A,”Task?A”,120,NULL,1,
本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過(guò)流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問(wèn)題卻十分常見(jiàn),不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問(wèn)題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問(wèn)題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來(lái)解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開(kāi)關(guān)電源具有效率高的特性,而且開(kāi)關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開(kāi)關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉