女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

如何設定PendSV優先級?

技術讓夢想更偉大 ? 來源:技術讓夢想更偉大 ? 作者:技術讓夢想更偉大 ? 2022-12-05 11:38 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

先了解下如何使用PendSV異常。(為何要使用PendSV而不是其他的異常,請參考《cortex-M3權威指南》)

1,如何設定PendSV優先級?

37b73aaa-7432-11ed-8abf-dac502259ad0.png
NVIC_SYSPRI14EQU0xE000ED22
NVIC_PENDSV_PRIEQU0xFF
LDRR0,=NVIC_SYSPRI14LDRR1,=NVIC_PENDSV_PRI
STRBR1,[R0]

2,如何觸發PendSV異常?

37c38d64-7432-11ed-8abf-dac502259ad0.png

往ICSR第28位寫1,即可將PendSV異常掛起。若是當前沒有高優先級中斷產生,那么程序將會進入PendSV handler

NVIC_INT_CTRLEQU0xE000ED04
NVIC_PENDSVSETEQU0x10000000

LDRR0,=NVIC_INT_CTRL
LDRR1,=NVIC_PENDSVSET
STRR1,[R0]

3,編寫PendSV異常handler

這里用PendSV_Handler來觸發LED點亮,以此證明PendSV異常觸發的設置是正確的。

#include"stm32f10x_conf.h"

#defineLED0*((volatileunsignedlong*)(0x422101a0))//PA8

unsignedcharflag=0;
voidLEDInit(void)
{
RCC->APB2ENR|=1<<2;
GPIOA->CRH&=0XFFFFFFF0;
GPIOA->CRH|=0X00000003;
GPIOA->ODR|=1<<8;
}

__asmvoidSetPendSVPro(void)
{
NVIC_SYSPRI14EQU0xE000ED22
NVIC_PENDSV_PRIEQU0xFF
LDRR1,=NVIC_PENDSV_PRI
LDRR0,=NVIC_SYSPRI14
STRBR1,[R0]
BXLR
}

__asmvoidTriggerPendSV(void)
{
NVIC_INT_CTRLEQU0xE000ED04
NVIC_PENDSVSETEQU0x10000000
LDRR0,=NVIC_INT_CTRL
LDRR1,=NVIC_PENDSVSET
STRR1,[R0]
BXLR
}

intmain(void)
{
SetPendSVPro();
LEDInit();
TriggerPendSV();
while(1);
}

voidPendSV_Handler(void)
{
LED0=0;
}

上述代碼可以正常點亮LED,說明PendSV異常是正常觸發了。

OK,是時候挑戰任務切換了。

如何實現任務切換?三個步驟:

步驟一:在進入中斷前先設置PSP。

curr_task=0;

設置任務0為當前任務:

__set_PSP((PSP_array[curr_task]+16*4));

設置PSP指向task0堆棧的棧頂位置:

__set_CONTROL(0x3);

設置為用戶級,并使用PSP堆棧:

__ISB();

指令同步隔離。

步驟二:將當前寄存器的內容保存到當前任務堆棧中。進入ISR時,cortex-m3會自動保存八個寄存器到PSP中,剩下的幾個需要我們手動保存。

步驟三:在Handler中將下一個任務的堆棧中的內容加載到寄存器中,并將PSP指向下一個任務的堆棧。這樣就完成了任務切換。

要在PendSV 的ISR中完成這兩個步驟,我們先需了解下在進入PendSV ISR時,cortex-M3做了什么?

1,入棧。會有8個寄存器自動入棧。入棧內容及順序如下:

37caed20-7432-11ed-8abf-dac502259ad0.pngimg

在步驟一中,我們已經設置了PSP,那這8個寄存器就會自動入棧到PSP所指地址處。

2,取向量。找到PendSV ISR的入口地址,這樣就能跳到ISR了。,

3,更新寄存器內容

做完這三步后,程序就進入ISR了。

進入ISR前,我們已經完成了步驟一,cortex-M3已經幫我們完成了步驟二的一部分,剩下的需要我們手動完成。

在ISR中添加代碼如下:

MRSR0,PSP

保存PSP到R0。為什么是PSP而不是MSP。因為在OS啟動的時候,我們已經把SP設置為PSP了。這樣使得用戶程序使用任務堆棧,OS使用主堆棧,不會互相干擾。不會因為用戶程序導致OS崩潰。

STMDBR0!,{R4-R11}

保存R4-R11到PSP中。C語言表達是*(--R0)={R4-R11},R0中值先自減1,然后將R4-R11的值保存到該值所指向的地址中,即PSP中。

STMDB Rd!,{寄存器列表} 連續存儲多個字到Rd中的地址值所指地址處。每次存儲前,Rd先自減一次。

若是ISR是從從task0進來,那么此時task0的堆棧中已經保存了該任務的寄存器參數。保存完成后,當前任務堆棧中的內容如下(假設是task0)

37d21abe-7432-11ed-8abf-dac502259ad0.png

左邊表格是預期值,右邊是keil調試的實際值。可以看出,是一致的。在任務初始化時(步驟一),我們將PSP指向任務0的棧頂0x20000080。在進入PendSV之前,cortex-M3自動入棧八個值,此時PSP指向了0x20000060。然后我們再保存R4-R11到0x20000040~0x2000005C。

這樣很容易看明白,如果需要下次再切換到task0,只需恢復R4~R11,再將PSP指向0x20000060即可。

所以切換到另一個任務的代碼:

LDRR1,=__cpp(&curr_task)
LDRR3,=__cpp(&PSP_array)
LDRR4,=__cpp(&next_task)
LDRR4,[R4]

獲取下一個任務的編號:

STRR4,[R1]
Curr_task=next_task
LDRR0,[R3,R4,LSL#2]

獲得任務堆棧地址,若是task0,那么R0=0x20000040( R0=R3+R4*4)

LDMIAR0!,{R4-R11}

恢復堆棧中的值到R4~R11。R4=*(R0++)。執行完后,R0中值變為0x20000060

LDMIA Rd! {寄存器列表} 先將Rd中值所指地址處的值送出寄存器中,Rd再自增1.*

MSRPSP,R0
PSP=R0。
BXLR

中斷返回。

完整代碼

#include"stm32f10x.h"
#include"stm32f10x_usart.h"
#include"stm32f10x_gpio.h"
#include"stm32f10x_rcc.h"
#include"stdio.h"
#include"misc.h"

#defineHW32_REG(ADDRESS)(*((volatileunsignedlong*)(ADDRESS)))
#defineLED0*((volatileunsignedlong*)(0x422101a0))//PA8
voidUSART1_Init(void);
voidtask0(void);
unsignedcharflag=1;

uint32_tcurr_task=0;//當前執行任務
uint32_tnext_task=1;//下一個任務
uint32_ttask0_stack[17];
uint32_ttask1_stack[17];
uint32_tPSP_array[4];

u8task0_handle=1;
u8task1_handle=1;

voidtask0(void)
{
while(1)
{
if(task0_handle==1)
{
printf("task0
");
task0_handle=0;
task1_handle=1;
}
}
}

voidtask1(void)
{
while(1)
{
if(task1_handle==1)
{
printf("task1
");
task1_handle=0;
task0_handle=1;
}
}
}

voidLEDInit(void)
{
RCC->APB2ENR|=1<<2;
GPIOA->CRH&=0XFFFFFFF0;
GPIOA->CRH|=0X00000003;
GPIOA->ODR|=1<<8;
}

__asmvoidSetPendSVPro(void)
{
NVIC_SYSPRI14EQU0xE000ED22
NVIC_PENDSV_PRIEQU0xFF

LDRR1,=NVIC_PENDSV_PRI
LDRR0,=NVIC_SYSPRI14
STRBR1,[R0]
BXLR
}

__asmvoidTriggerPendSV(void)
{
NVIC_INT_CTRLEQU0xE000ED04
NVIC_PENDSVSETEQU0x10000000

LDRR0,=NVIC_INT_CTRL
LDRR1,=NVIC_PENDSVSET
STRR1,[R0]
BXLR
}

intmain(void)
{
USART1_Init();

SetPendSVPro();
LEDInit();

printf("OStest
");

PSP_array[0]=((unsignedint)task0_stack)+(sizeoftask0_stack)-16*4;
//PSP_array中存儲的為task0_stack數組的尾地址-16*4,即task0_stack[1023-16]地址
HW32_REG((PSP_array[0]+(14<<2)))=(unsignedlong)task0;/*PC*/
//task0的PC存儲在task0_stack[1023-16]地址+14<<2中,即task0_stack[1022]中
HW32_REG((PSP_array[0]+(15<<2)))=0x01000000;/*xPSR*/

PSP_array[1]=((unsignedint)task1_stack)+(sizeoftask1_stack)-16*4;
HW32_REG((PSP_array[1]+(14<<2)))=(unsignedlong)task1;/*PC*/
HW32_REG((PSP_array[1]+(15<<2)))=0x01000000;/*xPSR*/

/*任務0先執行*/
curr_task=0;

/*設置PSP指向任務0堆棧的棧頂*/
__set_PSP((PSP_array[curr_task]+16*4));

SysTick_Config(9000000);
SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);//72/8=9MHZ
/*使用堆棧指針,非特權級狀態*/
__set_CONTROL(0x3);

/*改變CONTROL后執行ISB(architecturalrecommendation)*/
__ISB();

/*啟動任務0*/
task0();
//LED0=0;
while(1);
}

__asmvoidPendSV_Handler(void)
{
//保存當前任務的寄存器內容
MRSR0,PSP//得到PSPR0=PSP
//xPSR,PC,LR,R12,R0-R3已自動保存
STMDBR0!,{R4-R11}//保存R4-R11共8個寄存器得到當前任務堆棧

//加載下一個任務的內容
LDRR1,=__cpp(&curr_task)
LDRR3,=__cpp(&PSP_array)
LDRR4,=__cpp(&next_task)
LDRR4,[R4]//得到下一個任務的ID
STRR4,[R1]//設置curr_task=next_task
LDRR0,[R3,R4,LSL#2]//從PSP_array中獲取PSP的值
LDMIAR0!,{R4-R11}//將任務堆棧中的數值加載到R4-R11中
//ADDSR0,R0,#0x20
MSRPSP,R0//設置PSP指向此任務
//ORRLR,LR,#0x04
BXLR//返回
//xPSR,PC,LR,R12,R0-R3會自動的恢復
ALIGN4
}

voidSysTick_Handler(void)
{
flag=~flag;
LED0=flag;
if(curr_task==0)
next_task=1;
else
next_task=0;
TriggerPendSV();
}

voidUSART1_Init(void)
{
GPIO_InitTypeDefGPIO_InitStructure;
USART_InitTypeDefUSART_InitStructure;

/*configUSART1clock*/
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA,ENABLE);

/*USART1GPIOconfig*/
/*ConfigureUSART1Tx(PA.09)asalternatefunctionpush-pull*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);
/*ConfigureUSART1Rx(PA.10)asinputfloating*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA,&GPIO_InitStructure);

/*USART1modeconfig*/
USART_InitStructure.USART_BaudRate=9600;
USART_InitStructure.USART_WordLength=USART_WordLength_8b;
USART_InitStructure.USART_StopBits=USART_StopBits_1;
USART_InitStructure.USART_Parity=USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode=USART_Mode_Rx|USART_Mode_Tx;
USART_Init(USART1,&USART_InitStructure);
USART_Cmd(USART1,ENABLE);
}

intfputc(intch,FILE*f)
{
USART_SendData(USART1,(unsignedchar)ch);
while(!(USART1->SR&USART_FLAG_TXE));

return(ch);
}

測試后結果如圖:

37e621e4-7432-11ed-8abf-dac502259ad0.png

可以看出,兩個任務可以切換了。

上述代碼參考《cortex-M3權威指南》和《安富萊_STM32-V5開發板_μCOS-III教程》得來。

審核編輯 :李倩


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • led
    led
    +關注

    關注

    242

    文章

    23840

    瀏覽量

    673956
  • 程序
    +關注

    關注

    117

    文章

    3826

    瀏覽量

    82965

原文標題:例說OS前的任務切換(附代碼)

文章出處:【微信號:技術讓夢想更偉大,微信公眾號:技術讓夢想更偉大】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    深度剖析 RT-Thread 線程調度流程

    rt_system_scheduler_start:調度系統第一個線程rt_hw_context_switch_to:初始化上下文切換環境,觸發PendSV異常first_thread:系統中優先級最高
    的頭像 發表于 06-25 18:24 ?703次閱讀
    深度剖析 RT-Thread 線程調度流程

    ADL5308可以通過軟件和硬件配置的參數,配置的優先級是什么?

    你好,麻煩問一下ADL5308可以通過軟件和硬件配置的參數,配置的優先級是什么?有沒有更詳細的寄存器配置手冊,截距配置的步進是多少?
    發表于 06-10 06:39

    CyU3PDebugPrint的最高優先級和最低優先級是什么?

    [i]CyU3PDebugPrint的最高優先級和最低優先級是什么?
    發表于 05-13 08:22

    TSN幀搶占測試:為數據傳輸打造“綠色通道”

    導語: 在復雜多變的汽車、工業控制等領域中,數據傳輸的時效性和確定性至關重要。在TAS或CQF等調度整形機制中,將不同優先級業務映射到不同隊列中,使高優先級業務在發送時不會受到低優先級業務的干擾
    的頭像 發表于 04-23 15:31 ?229次閱讀
    TSN幀搶占測試:為數據傳輸打造“綠色通道”

    優先編碼器:高效數據選擇與編碼的解決方案

    ,在這一領域發揮著重要作用。 優先編碼器,顧名思義,能夠按照設定優先級對多個輸入信號進行選擇和編碼。它通常具有多個輸入端和一個或多個輸出端,輸入端接收來自不同源的數據信號,而輸出端則輸出被選中的數據的編碼
    的頭像 發表于 03-25 08:33 ?495次閱讀

    配電柜—斷電危機?配電柜故障排查優先級指南

    在排查配電柜故障過程中,合理安排排查優先級至關重要。下面聊一下如何科學合理安排配電柜故障排查優先級順序。
    的頭像 發表于 03-06 18:55 ?414次閱讀
    配電柜—斷電危機?配電柜故障排查<b class='flag-5'>優先級</b>指南

    技術干貨驛站 ▏解鎖C語言高效編程秘訣:深入解析運算符與優先級

    在C語言的學習過程中,運算符的使用是不可忽視的重要環節。本文將繼續深入探討C語言中的運算符,重點介紹位運算符、賦值運算符及其優先級。掌握這些運算符的用法,將幫助你在編程過程中實現更加高效的代碼操作
    的頭像 發表于 10-13 08:09 ?632次閱讀
    技術干貨驛站 ▏解鎖C語言高效編程秘訣:深入解析運算符與<b class='flag-5'>優先級</b>

    瑞芯微RK3399開發板Android7.1修改網絡優先級方法,觸覺智能SBC3968開發板演示

    本文介紹Android7.1修改網絡優先級方法,基于觸覺智能SBC3968主板,搭載瑞芯微RK3399芯片,底板具備千兆網口、USB、HDMI、TYPE-C、音頻等接口。可適用于工業主機、物聯網設備、醫療健康設備、廣告機、互動自助終端、教學實驗平臺、顯示控制、車載安防和人臉識別等多個領域。
    的頭像 發表于 10-09 11:37 ?622次閱讀
    瑞芯微RK3399開發板Android7.1修改網絡<b class='flag-5'>優先級</b>方法,觸覺智能SBC3968開發板演示

    PCM9211的默認模式下,ADC和RXIN2( 光纖輸入)是自動識別的嗎,并且光纖具有輸入優先級

    咨詢一下,PCM9211的默認模式下,ADC和RXIN2( 光纖輸入)是自動識別的嗎,并且光纖具有輸入優先級
    發表于 09-29 06:44

    使用CH32V103C8TC設置中斷優先級分組時,編譯報錯的原因?

    使用CH32V103C8TC設置中斷優先級分組時,設置為NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);編譯報錯 而使用NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1); 卻不會報錯 這是什么原因呢??
    發表于 09-27 06:03

    CH32V103 使用中斷優先級分組0時報錯

    使用CH32V103C8TC設置中斷優先級分組時,設置為NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);而使用NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);卻不會報錯
    發表于 09-23 13:52

    FS312B系列芯片概述和優勢

    FS312BL 支持 PD 和 QC,最高設定 20V。 FS312BH 支持 PD 和 QC,最高設定 48V。 FS312BLE/FS312BHE,支持模擬 emarker。 PD 協議的優先級高于 QC
    的頭像 發表于 09-11 16:51 ?734次閱讀
    FS312B系列芯片概述和優勢

    車載以太網交換機入門基本功(4)—優先級設計與VLAN測試

    VLAN通過報文Tag、交換機端口屬性、交換機端口過濾功能得以實現,并通過優先級和隊列實現轉發的有效調度。為了驗證交換機芯片的功能實現,需要參考OPEN聯盟的TC11交換機芯片測試規范,包括通用測試、地址解析、VLAN等9個方面,此外,實時更進規范修改方案并與讀者分享。
    的頭像 發表于 09-06 14:45 ?919次閱讀
    車載以太網交換機入門基本功(4)—<b class='flag-5'>優先級</b>設計與VLAN測試

    在汽車區域模塊中使用理想二極管的優先級電源多路復用器

    電子發燒友網站提供《在汽車區域模塊中使用理想二極管的優先級電源多路復用器.pdf》資料免費下載
    發表于 09-05 11:08 ?0次下載
    在汽車區域模塊中使用理想二極管的<b class='flag-5'>優先級</b>電源多路復用器

    freertos中斷優先級在哪設置

    FreeRTOS是一個流行的實時操作系統,它廣泛應用于嵌入式系統開發。在FreeRTOS中,中斷優先級是一個重要的概念,因為它決定了中斷處理的順序和響應時間。 1. 理解中斷優先級 在討論如何設置
    的頭像 發表于 09-02 14:17 ?1571次閱讀