嵌入式学习笔记 - freeRTOS任务栈在初始化以及任务切换时的压栈出栈过程分析
一前言
ARM的堆栈采用向下增长栈,就是栈顶地址初始值最大,压栈时越来越小:
比如定义一个任务栈为Task2Stack[N],那么任务栈的在ARM内存中的排列如下图示所,数组Task2Stack代表的地址是任务栈的最低地址,当栈为空时,就是栈内不存任何内容时,栈顶指针地址SP=Task2Stack+N-1,Task2Stack 为定义的任务栈的起始地址,也就是定义的任务栈数组的地址,N为定义的任务栈的总大小(长度)。
二 任务栈初始化过程
freeRTOS的任务栈初始化使用的是pxPortInitialiseStack函数,函数实体如下:
pxTopOfStack 为栈顶指针,在进入这个函数之前freeRTOS已经将栈顶指针进行过初始化,即将栈顶指针移动到栈内存为空时的位置,函数语句如下
/* 获取栈顶地址 */
pxTopOfStack = pxNewTCB->pxStack + ( ulStackDepth - ( uint32_t ) 1 );
pxNewTCB->pxStack为任务栈数组地址, ulStackDepth为任务栈的长度,根据前言部分所述,此时栈顶pxTopOfStack正好为空栈时的栈顶位置,
知道了栈顶指针的具体位置之后,再来看上述任务栈初始化函数具体过程,这个函数的具体实现就是将xPSR,PC,LR,R0~R12 按照一定的顺序存入任务栈内部。
69~70行,将栈顶指针减1,存入xRSP,(这里的地址减1代表一个字,4个字节,下同)
71~72行,将栈顶指针再减1,存入PC
73~74行,栈顶指针减1,存入LR,
75~76行,栈顶指针减5,存入R0
最后经过函数初始化之后的的任务栈内容如下图所示,可以看出ARM的堆栈压栈过程,是栈顶指针先自减,再存放,随着栈内容越来越多,栈顶指针越来越小,所以叫向下增长栈。
三 任务切换时的压栈出栈过程分析
freeRTOS的任务栈的切换使用的是xPortPendSVHandler函数,函数实体如下:
这个函数的具体实现功能就是进行任务切换,就是将当前任务的运行环境,压入当前的任务栈进行保护,然后将下一个任务的任务栈里面的内容进行出栈,之后系统就可以切换到下一个任务的程序进行运行。
第172行,获取当前的任务栈指针到R0
第175~176行,获取任务控制块地址到R2
第178~179行,将寄存器r4~R11的值存入堆栈,这几个寄存器存放的通常是局部变量,同时栈顶指针自减,这几个寄存器中保存的通常是任务的局部变量,
第182行,将当前R3,R14的值压栈,进行函数调用,因为调用新函数,R14(LR)是肯定要用到的,而R14目前存放的是当前层的上一层的返回地址(状态),一会中断退出时要用,而关于R3目前存放的是pxcurrentTCB指针(注意是指针不是指针指向的内容,这个指针指向的内容调用函数中给与了新的地址),这个指针关系到新任务的栈顶指针地址,函数调用出来后也要用,在不确定R3是否会被调用函数使用的情况,最好保存起来,
第192行,出栈恢复R3,R14
第194行,获取pxcurrentTCB指针指向的内容到R1,这时R1存的是当前任务块的地址,注意此时R3存放的还是pxcurrentTCB这个指针,但是指针指向的内容已经变成新任务的控制块地址了,切换任务的时候对其进行了改变
195行,获取当前任务(新任务)的堆栈栈顶指针到R0,
196行,将站内内容出栈到R4~R11,
197行,新的栈顶指针赋值给PSP,
199行,R14寄存器也就是LR寄存器,异常发生时跟函数调用时LR的作用不一样,LR此时只保存相关的状态,决定是返回什么模式执行,
剩余的寄存器内容xPSR, PC, R0~R3( 存放形参),R12,以及剩余的堆栈内容硬件会在转向具体的新任务时自动操作进行出栈
四 题外话,当任务的局部变量大于R4~R11时,
任务的局部变量通常会存放于堆栈中,下图是用反汇编验证的图示