freertos任务切换xPortPendSVHandler

本文深入剖析FreeRTOS V8.2.3版本中的任务切换过程,详细解读xPortPendSVHandler函数如何通过汇编指令实现上下文切换,包括保存当前任务状态、更新任务控制块、恢复新任务上下文等关键步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

FreeRTOS版本:FreeRTOS V8.2.3

void xPortPendSVHandler( void )
{
	/* This is a naked function. */

        /*__asm__(汇编语句模板: 输出部分: 输入部分: 破坏描述部分)
        /*共四个部分:汇编语句模板,输出部分,输入部分,破坏描述部分,各部分使用":"格开,
        /*汇编语句模板必不可少,其他三部分可选,如果使用了后面的部分,而前面部分为空,
        /*也需要用":"格开,相应部分内容为空. __asm volatile("...." : : "i"());破坏描述部分没    
        /*用,i表示立即数,官方文档网址
     /*https://gcc.gnu.org/onlinedocs/gcc/Extended-Asm.html
        */
	 __asm volatile
	(
	"	mrs r0, psp				\n"	//将psp值放到r0,此时sp得值为msp
	"	isb					\n"
	"						\n"
	"	ldr	r3, pxCurrentTCBConst		\n" 	/* Get the location of the current TCB. */
								//将pxCurrentTCBConst标签(指针变量pxCurrentTCB的内存地址为&pxCurrentTCB或者到map看)放到r3,此时r3=&pxCurrentTCB
	"	ldr	r2, [r3]			\n"	//将r3寄存器值作为指针取内容存到r2,此时r2保存的为任务控制块首地址
	"						\n"
	"	stmdb r0!, {r4-r11}			\n" 	/* Save the remaining registers. */
								//r0寄存器地址减去4在将r11-r4依次入栈r0(此时r0为程序栈psp地址)-4地址后
	"	str r0, [r2]				\n" 	/* Save the new top of stack into the first member of the TCB. */
								//将r0(psp)值放到r2内容(pxCurrentTCBConst)所指的地址
	"						\n"
	"	stmdb sp!, {r3, r14}			\n"	//sp寄存器地址减去4在将r14,r3依次入栈sp-4地址后
	"	mov r0, %0				\n"     //获取中断优先级到r0,%0对应后面的configMAX_SYSCALL_INTERRUPT_PRIORITY
	"	msr basepri, r0				\n"     //屏蔽低于优先级的中断
	"	bl vTaskSwitchContext			\n"     //跳到任务切换c函数,找到下一个任务控制块
	"	mov r0, #0				\n"
	"	msr basepri, r0				\n"     //开中断
	"       dmia sp!, {r3, r14}			\n"	//从sp中取出r3,r14
	"						\n"	/* Restore the context, including the critical nesting count. */
	"	ldr r1, [r3]				\n"	//将r3(存放的是任务控制块指针变量的地址&pxCurrentTCB)寄存器内容作为地址取内容放到r1(此时r1为新的任务控制块地址)	
        "	ldr r0, [r1]				\n" 	/* The first item in pxCurrentTCB is the task top of stack. */
								//将r1寄存器内容作为地址取内容放到r0(此时r0为新任务的pxTopOfStack)
	"	ldmia r0!, {r4-r11}			\n" 	/* Pop the registers. */
                                                                //将新的任务堆栈顶指针出堆栈到r4-r11
	"	msr psp, r0				\n"     //将新的任务堆栈顶指针放到psp
	"	isb					\n"
	"	bx r14					\n"     //之后硬件会自动把PC指针出堆栈(因为此时psp为新任务的堆栈顶指针所以出堆栈也是新任务的寄存器)
	"						\n"
	"	.align 2				\n"
	"pxCurrentTCBConst: .word pxCurrentTCB	\n"
	::"i"(configMAX_SYSCALL_INTERRUPT_PRIORITY)
	);
}

              图1 两个任务切换时堆栈存储分布情况

               图2 由任务2(task2)切换到任务1(idle)寄存器及内存变化

其中memory1列的是栈顶指针开始往上增长对应图1就是压栈方向存储情况,结合图1可以看出xPSR、PC指针的值,切换的过程是先将任务2的堆栈保存,找到要切换的任务1的任务控制块,将堆栈指针更新到psp中,然后出栈时,硬件会自动出栈R0-R3、R12、LR、PC、xPSR寄存器(此时出栈寄存器psp内容已经是任务1的内容了)任务切换完成后出栈PC指针就更新为切换后的任务函数值。

<think>好的,我现在需要帮助用户查找FreeRTOS中关于任务切换与调度的相关代码实现。首先,我需要回忆一下FreeRTOS任务切换机制。根据用户提供的引用,任务切换主要发生在系统滴答中断(xPortSysTickHandler)和主动调用调度器的时候,比如portYIELD_FROM_ISR。 接下来,我应该先确认FreeRTOS任务调度方式,用户提到的引用3中提到了抢占式调度和时间片调度。抢占式调度涉及到优先级高的任务抢占当前任务,而时间片调度可能是在同优先级任务之间轮转。不过根据引用3,FreeRTOS的同级任务时间片只有一个,这可能意味着每个任务运行一个时间片后切换。 然后,我需要找到任务切换的具体代码位置。通常,FreeRTOS任务切换在port.c文件中,特别是与硬件相关的部分,比如ARM Cortex-M的port.c。例如,vPortYield函数和xPortPendSVHandler(PendSV中断处理)可能是关键点。 用户提供的引用1提到xPortSysTickHandler会触发任务切换,这可能是在系统时钟中断中调用任务调度。引用2中的portYIELD_FROM_ISR用于在中断中触发调度,这通常会设置一个上下文切换的标志,比如xYieldPending。 现在需要结合具体的代码示例。例如,系统滴答中断处理函数会调用xTaskIncrementTick,如果返回pdTRUE,则触发任务切换。此外,PendSV中断处理函数实际执行上下文切换,这里会保存当前任务的上下文并恢复下一个任务的。 用户可能希望看到代码片段,比如在系统滴答中断中的处理,以及PendSV处理程序中的汇编代码。同时,主动调度的宏定义,如taskYIELD(),会触发SVC或PendSV中断。 需要注意的是,不同硬件平台的实现可能不同,比如ARM Cortex-M使用PendSV,而其他架构可能有不同的中断处理方式。因此,在示例代码中应指出具体架构相关的部分,并建议用户参考对应端口的代码。 最后,生成相关问题时要围绕任务切换机制、调度策略、代码位置、API函数和性能优化等方面,确保问题覆盖用户可能感兴趣的扩展内容。</think>### FreeRTOS任务切换与调度代码实现解析 根据FreeRTOS的调度策略(抢占式/时间片)和系统架构特性,任务切换的实现涉及以下关键代码模块: #### 1. 系统滴答定时器中断中的任务切换 在`xPortSysTickHandler()`中实现时间片轮转: ```c // ports/ARM_CM4F/port.c void xPortSysTickHandler(void) { if (xTaskGetSchedulerState() != taskSCHEDULER_NOT_STARTED) { if (xTaskIncrementTick() != pdFALSE) { // 时间片计数+1 portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT; // 触发PendSV中断[^1] } } } ``` #### 2. 上下文切换核心实现 通过PendSV中断执行上下文保存/恢复: ```assembly ; ports/ARM_CM4F/portasm.s xPortPendSVHandler: mrs r0, psp ; 获取当前任务栈指针 ldr r3, =pxCurrentTCB ; 加载任务控制块地址 ldr r2, [r3] ; 获取当前TCB指针 stmdb r0!, {r4-r11} ; 保存寄存器组到任务栈 str r0, [r2] ; 更新TCB栈顶指针 stmdb sp!, {r3, r14} ; 保存R3和LR到系统栈 bl vTaskSwitchContext ; 选择新任务[^4] ldmia sp!, {r3, r14} ; 恢复寄存器 ldr r1, [r3] ; 获取新任务TCB指针 ldr r0, [r1] ; 加载新任务栈顶指针 ldmia r0!, {r4-r11} ; 恢复新任务寄存器组 msr psp, r0 ; 更新进程栈指针 bx lr ; 返回新任务上下文 ``` #### 3. 主动触发任务切换 通过`taskYIELD()`宏实现即时调度: ```c // include/task.h #define taskYIELD() portYIELD() // ports/ARM_CM4F/portmacro.h #define portYIELD() \ { \ portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT; \ __dsb( portSY_FULL_READ_WRITE ); \ __isb( portSY_FULL_READ_WRITE ); \ } ``` #### 4. 任务上下文切换请求标志 在中断中触发调度的安全方式: ```c // 示例中断服务程序 void USART1_IRQHandler(void) { BaseType_t xHigherPriorityTaskWoken = pdFALSE; // 中断处理逻辑... portYIELD_FROM_ISR(xHigherPriorityTaskWoken); // 安全触发调度[^2] } ``` ### 关键代码位置指引 1. 调度策略实现:`tasks.c` 中的`vTaskSwitchContext()` 2. 架构相关切换代码:`port.c`和`portasm.s` 3. 时间片管理:`xTaskIncrementTick()`函数 4. 优先级调度:`taskSELECT_HIGHEST_PRIORITY_TASK()`宏
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值