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

當(dāng)前位置:首頁 > 嵌入式 > 嵌入式大雜燴
[導(dǎo)讀]之前分享了很多關(guān)于freeRTOS的知識(shí),那么我們?cè)趺丛趯?shí)戰(zhàn)中去寫代碼呢?本篇文章重在對(duì)基于freeRTOS的架構(gòu)代碼的解析。



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

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

作者:李肖遙


之前分享了很多關(guān)于freeRTOS的知識(shí),那么我們?cè)趺丛趯?shí)戰(zhàn)中去寫代碼呢?本篇文章重在對(duì)基于freeRTOS的架構(gòu)代碼的解析。整個(gè)功能如下圖:


為什么要用freeRTOS

在實(shí)際項(xiàng)目中,如果程序等待一個(gè)超時(shí)事件,傳統(tǒng)的無RTOS情況下,就只能在原地等待而不能執(zhí)行其它任務(wù),如果使用RTOS,則可以很方便的將當(dāng)前任務(wù)阻塞在該事件下,然后自動(dòng)去執(zhí)行別的任務(wù),這樣可以高效的利用CPU了。

一般使用情況

我們?cè)陂_發(fā)的時(shí)候,我總是在main函數(shù)看到以下的代碼,這讓我感覺不是很爽

int?main()
{
??xTaskCreate(?vTask1,?"Task?1",?1000,?NULL,?1,?NULL?);
??xTaskCreate(?vTask2,?"Task?2",?1000,?NULL,?1,?NULL?);
??xTaskCreate(?vTask3,?"Task?3",?1000,?NULL,?2,?NULL?);

??vTaskStartScheduler();

??while(1);
}

然后在每個(gè)task中,一般代碼會(huì)這樣寫

void?vTask1(?void?*pvParameters?)
{
??volatile?unsigned?long?ul;
??for(?;;?)
??{
????xQueueSend(?USART1_MSGQ,?"task?1?!\n",portMAX_DELAY);
????for(?ul?=?0;?ul???}
}

而任務(wù)之間的通信也是比較繁瑣,總體來說,代碼不易維護(hù),增減一個(gè)任務(wù)的話要改的東西太多了。為此我特意設(shè)計(jì)一個(gè)框架,可以很方便的增減任務(wù),同時(shí)任務(wù)之間通過事件隊(duì)列來通信。

demo

任務(wù)創(chuàng)建函數(shù)的封裝

我們首先定義兩個(gè)任務(wù),把所有任務(wù)信息封裝在taskRecord里,并且申明如下:

#define?TASK_NUM?2
//所有任務(wù)的信息
static?TaskRecord?taskRecord[TASK_NUM];

那么TaskRecord怎么安排呢,我們把所有的任務(wù)信息都放在結(jié)構(gòu)體里。其中包括任務(wù)ID,任務(wù)任務(wù)函數(shù)taskFucn,任務(wù)名字,棧的大小stackDep,還有優(yōu)先級(jí)prio,任務(wù)句柄taskHandle,任務(wù)隊(duì)列queue。

typedef?struct
{
?int16_t?Id;
?TaskFunction_t?taskFucn;
?const?char?*??name;
?configSTACK_DEPTH_TYPE?stackDep;
?void?*??parameters;
?UBaseType_t?prio;
?TaskHandle_t?taskHandle;
?QueueHandle_t??queue;
}?TaskRecord;

把任務(wù)中的一些參數(shù)封裝起來,放在結(jié)構(gòu)體TaskInitPara,其中包括了任務(wù)函數(shù)taskFucn,任務(wù)名字,棧的大小stackDep,還有優(yōu)先級(jí)prio。

typedef?struct
{
??TaskFunction_t?taskFucn;
??const?char?*??name;
??const?configSTACK_DEPTH_TYPE?stackDep;
??UBaseType_t?prio;
}?TaskInitPara;

我們做好了這些之后,就需要把結(jié)構(gòu)體中的參數(shù)放到創(chuàng)建任務(wù)函數(shù)中,那么這個(gè)函數(shù)createTasks代碼如下:

void?createTasks(TaskRecord*?taskRecord,?const?TaskInitPara*?taskIniPara,?int?num){
?int?i;
?for(i=0;i??taskRecord[i].Id?=?i;
??taskRecord[i].taskFucn?=?taskIniPara[i].taskFucn;
??taskRecord[i].name?=?taskIniPara[i].name;
??taskRecord[i].stackDep?=?taskIniPara[i].stackDep;
??taskRecord[i].parameters?=?&taskRecord[i];
??taskRecord[i].prio?=?taskIniPara[i].prio;
??
??xTaskCreate(?taskRecord[i].taskFucn,
????taskRecord[i].name,
????taskRecord[i].stackDep,
????taskRecord[i].parameters,
????taskRecord[i].prio,
????&taskRecord[i].taskHandle?);

??taskRecord[i].queue?=?xQueueCreate(?100,?sizeof(?Event?)?);
?}
}

其中num為任務(wù)數(shù)量,先把任務(wù)信息放到初始化的taskRecord中,再把其中的信息創(chuàng)建任務(wù)。那么任務(wù)創(chuàng)建函數(shù)就做好了。

main函數(shù)

接著,在我們的main函數(shù)中,就不需要那么繁瑣的一個(gè)一個(gè)的創(chuàng)建任務(wù)了,按照這個(gè)封裝的main函數(shù)如下:

int?main(?void?)
{
?createTasks(taskRecord,taskInitPara,TASK_NUM);
?/*?Start?the?tasks?and?timer?running.?*/
?vTaskStartScheduler();
}

任務(wù)間通信

首先我們想想,兩個(gè)任務(wù)之間通信需要知道什么,task1想往task2的發(fā)送一些數(shù)據(jù),那么需要知道task2的ID吧,需要把數(shù)據(jù)打包吧,task2需要知道是誰發(fā)的,那么task1本身的ID也需要知道吧。

按照這幾個(gè)明確的東西,我們首先把任務(wù)事件ID枚舉如下

typedef?enum?{
?eventID_1,
?eventID_2,
?eventID_3
}Event_ID;

然后把事件ID,發(fā)送者ID,以及要傳輸?shù)慕Y(jié)構(gòu)或者數(shù)據(jù)打包封裝在結(jié)構(gòu)體Event中,代碼如下:

typedef?struct{
?Event_ID?ID;
?int16_t?src;?//發(fā)送者ID
?void*?pData;?//傳結(jié)構(gòu)、數(shù)據(jù)
}Event;

接著,我們需要構(gòu)造一個(gè)事件,把這些信息都放在這個(gè)事件中,代碼如下:

void?makeEvent(Event*?pEvent,int16_t?myId,Event_ID?evtId,?const?void*?pData){
?pEvent->ID?=?evtId;
?pEvent->src?=?myId;
?pEvent->pData?=?(void*)?pData;
}

現(xiàn)在我們假設(shè)task2要往task1發(fā)送一系列數(shù)據(jù),那么task任務(wù)中,我們需要做的事如下,獲取task1中隊(duì)列,看是否為空。

?QueueHandle_t?task1Queue;
?int16_t?myId?=?pMyTaskRecord->Id;
?task1Queue?=?getTaskQueue(getTaskId("task1"));

構(gòu)造事件

?Event?event;
?int*?ptemp;?//這里自定義一些數(shù)據(jù)
?makeEvent(&event,myId,eventID_1,(void*)ptemp);

然后把事件發(fā)送出去:

xQueueSendToBack(?task1Queue,?&event,?0);

對(duì)于task1來說,看隊(duì)列中是否為空,如果有任務(wù)事件來,從隊(duì)列中獲取事件

?TaskRecord*?pMyTaskRecord?=?(TaskRecord*)pPara;
?QueueHandle_t*?evntQueue=pMyTaskRecord->queue;

當(dāng)隊(duì)列中確實(shí)有事件時(shí),接收事件

BaseType_t?status?=?xQueueReceive(?*evntQueue,?&event,?portMAX_DELAY?);
if(?status?==?pdPASS?)
{
??task1HandleEvent(event);
}
else
{
??printf(?"Task1?could?not?receive?from?the?queue.\r\n"?);}

然后我們?cè)?code style="font-size: 14px;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;background-color: rgba(27, 31, 35, 0.05);font-family: "Operator Mono", Consolas, Monaco, Menlo, monospace;word-break: break-all;color: rgb(92, 157, 255);">task1HandleEvent處理接收到的事件,代碼如下:

void?task1HandleEvent(Event?event){
?xil_printf(?"Task1?is?processing?event...\r\n"?);
?int*?p;
?switch(event.ID){
?case?eventID_1:
??p=?(int*)?event.pData;
??xil_printf("ID=%d?From:?%d?data=%d\r\n",event.ID,?event.src,p[7]);
??free(event.pData);
??break;
?case?eventID_2:
??break;
?default:
??break;
?}
}

上面代碼表示根據(jù)事件ID來判斷接收的是哪個(gè)事件,再把事件ID,數(shù)據(jù)等等打印出來。

外部中斷通信

如果不是任務(wù)間的通信,而是有外部中斷觸發(fā),需要與某個(gè)任務(wù)進(jìn)行信息交互,怎么辦?例如有一個(gè)以太網(wǎng)任務(wù),當(dāng)外部網(wǎng)絡(luò)需要發(fā)送一個(gè)數(shù)據(jù)包到這個(gè)網(wǎng)絡(luò)任務(wù)的時(shí)候,那么就需要進(jìn)行外部通信了。同樣我們這樣做,在以太網(wǎng)接收函數(shù)中,構(gòu)造事件

?Event?event;
?int*?ptemp;?//這里自定義一些數(shù)據(jù)
?makeEvent(&event,myId,IntrID_1,(void*)ptemp);//可以再自定一些事件ID如IntrID_1

然后再發(fā)送到這個(gè)事件到這個(gè)任務(wù)中,如下

測(cè)試

如上,我們構(gòu)造一個(gè)事件,發(fā)送一些數(shù)據(jù)如下

?Event?event;
?int*?ptemp?=?malloc(sizeof(int)*10);
?memset(ptemp,0x77,sizeof(int)*10);
?makeEvent(&event,myId,eventID_1,(void*)ptemp);

我們看到結(jié)果如下

task1接到來自任務(wù)ID為0,事件1的數(shù)據(jù)。這里每個(gè)任務(wù)的等待時(shí)間也是可以設(shè)置的,設(shè)置方法如下:

/*?設(shè)置最大等待時(shí)間500ms?*/
const?TickType_t?xMaxBlockTime?=?pdMS_TO_TICKS(500);?
BaseType_t?status?=?xQueueReceive(?*evntQueue,?&event,?xMaxBlockTime?);

如果等待時(shí)間為portMAX_DELAY或者0的話,說明某個(gè)任務(wù)一直處于激活狀態(tài),比如task2,當(dāng)?shù)却龝r(shí)間為portMAX_DELAY時(shí)候,則測(cè)試結(jié)果如下:

所以每個(gè)任務(wù)設(shè)置的時(shí)間,優(yōu)先級(jí),棧大小都是很重要的,具體的就需要在項(xiàng)目中調(diào)試了。

最后總結(jié)

本篇是屬于代碼實(shí)戰(zhàn)篇,對(duì)于freeRTOS的具體講解需要大家自己去領(lǐng)會(huì),我這里是寫了一個(gè)架構(gòu),幫助大家在項(xiàng)目中去更好的搭好架子,當(dāng)我們有很多任務(wù)的時(shí)候,任務(wù)間又有很多交互通信的時(shí)候,就更需要理解這種架構(gòu)了。



猜你喜歡

C語言、嵌入式位操作精華技巧大匯總

認(rèn)識(shí)認(rèn)識(shí)#pragma、#error指令

C語言、嵌入式中幾個(gè)非常實(shí)用的宏技巧

    
            

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(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ì)抑制與過流保護(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)電源易損壞的問題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗(yà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)問題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

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

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

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開關(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)閉