c - 如何为 ARM 上的每个任务定义堆栈

标签 c assembly arm

今天我遇到了一个小问题,我认为起源是关于堆栈的。

这是我的问题:

我声明了三个用户任务,如下所示:

void task1Function(void) {
    print_uart0("-usertask : First task is started...\r\n");
    while(1){
        //syscall(1);
    }
}

void task2Function(void) {
        print_uart0("-usertask : Second task is running...\r\n");
        //syscall(); /* To return in the kernel's mode */
        while(1){

        }
}

 void task3Function(void) {
            print_uart0("-usertask : Third task is running...\r\n");
            //syscall(); /* To return in the kernel's mode */
            while(1){

            }
    }

我有一个包含三个任务的数组,其结构如下:

typedef struct
{

    unsigned int *sp;
    unsigned int registers[12];
    unsigned int lr;
    unsigned int pc;
    unsigned int cpsr;
    unsigned int mode;
    unsigned int num;
    void *stack;
    int stacksize;

    int priority;
    int state;                    /* Running, Ready, Waiting, Suspended */

    /* Next and previous task in the queue */
    struct taskstruct *qnext, *qprev;
}taskstruct;

这是我的任务的初始化:

void init_task(taskstruct * task, void (*function)(void) ){
    task->sp = (unsigned int*)&function;
    task->registers[0] = 0; // r0
    task->registers[1] = 0; // r1
    task->registers[2] = 0; // r2
    task->registers[3] = 0; // r3
    task->registers[4] = 0; // r4
    task->registers[5] = 0; // r5
    task->registers[6] = 0; // r6
    task->registers[7] = 0; // r7
    task->registers[8] = 0; // r8
    task->registers[9] = 0; // r9
    task->registers[10] = 0; // r10
    task->registers[11] = 0; // r11
    task->registers[12] = 0; // r12
    task->lr = 0;
    task->pc = 0;
    task->cpsr = 0;
    task->mode = 0x10;
}

init_task(&task[0],&task1Function);
init_task(&task[1],&task2Function);
init_task(&task[2],&task3Function);

但是当我将任务[0].sp传递给我的激活函数时,它始终是最后一个声明的启动任务(即第三个):

.global activate
activate:

LDR r12, [r0]
/*STMFD sp!,{r1-r11,lr}*/
NOP

msr CPSR_c, #0x10 /* User mode with IRQ enabled and FIQ disabled*/
mov pc, r12

所以我想我的用户堆栈有问题,我必须为每个用户设置不同的堆栈,对吗?在这种情况下,有人可以告诉我如何继续吗?

问候,文森特

最佳答案

是的,你说得对。

每个任务必须有单独的堆栈,因为堆栈的状态是任务总状态的一部分。

你可以添加类似的内容

uint32_t stack[512];

到您的taskstruct,然后在切换任务时相应地设置/恢复任务指针。在这种情况下很难提供足够的细节。

关于c - 如何为 ARM 上的每个任务定义堆栈,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15090840/

相关文章:

assembly - x86堆栈指针指向哪里?

c - 为什么GAS汇编语言可以不声明直接调用C语言的全局变量?

c - 为什么这个数组的所有剩余值都初始化为零?

看不到错误

c - Getrusage 内联汇编

c - gdb 跳转某些部分的汇编代码

arm - 如何调试闪存中的引导加载程序?

Tensorflow Lite 错误 undefined reference to `tflite::DefaultErrorReporter()'

c - 关于 C 编译器 asm 输出的另一个问题

c - 在 C 中读取执行程序标准输出的更好方法