實時事件觸發(fā)實時調度問題--使用一個子函數(shù)來觸發(fā)結果無法返回了

1、對比了兩種寫法:

第一種寫法在任務task1里直接由內嵌匯編swi來觸發(fā)實現(xiàn)任務調度,swi之后的指令還能正常工作。其它任務調度邏輯是正常的。
第二種寫法在任務task1里通過調用taskDelay(1)函數(shù),在該函數(shù)里使用內嵌匯編swi來觸發(fā)任務調度,那么taskDelay(1)函數(shù)之后的指令無法再下一次調用該任務時還原到此指令執(zhí)行,原因用戶棧不在了,使用任務棧保存的參數(shù)無法在復原了。

小結:

第一種寫法在任務task1里直接由內嵌匯編swi來觸發(fā)實現(xiàn)任務調度,swi之后的指令還能正常工作。其它任務調度邏輯是正常的。
第二種寫法在任務task1里通過調用taskDelay(1)函數(shù),在該函數(shù)里使用內嵌匯編swi來觸發(fā)任務調度,那么taskDelay(1)函數(shù)之后下一條指令printf無法執(zhí)行,分析原因用戶棧被破壞了,保存的局部參數(shù)無法在復原了。
修改了任務棧保存信息,把當前任務的R0-R15都給保存到任務棧里,還是不行。

目前關于寄存器值保存和恢復的做法:

A、每次進入中斷和svc模式,保存用戶態(tài)的所有寄存器到當前任務棧
B、每次將要運行的任務的SP值不恢復,直接跳過,其它寄存器值都恢復。這樣避免了用戶態(tài)SP值得改變,從一個任務到另外一個任務是連續(xù)的,各個任務調度是正常的。這種方式只適合上面第一種寫法,而不適合第二種寫法。
C、每次將要運行的任務的SP值也恢復,那么通過中斷產生任務調度再次運行該任務時會重啟了,不能繼續(xù)運行。

最后解決方法:

對比了兩種寫法的匯編,第二種寫法的sp被調用下一個任務sp給占用了,所以在此問題導致了。第一個任務的sp雖然恢復了,但是sp里的局部參數(shù)(改參數(shù)是返回到任務的地址)被下一個任務的sp給占用了,所以第一個任務sp在切換前應該保存一段空間,用來隔開第二個任務的sp即可。

第一種寫法代碼如下:

int TEST_TestTask1(void)
{
    printf("\n\rTask1 start...\n\r");
    printf("user mode ...      SP 0x%x\n\r",sp_value());
    printf("CPSR==> 0x%x\n\r",cpsr_value());
    while(1){
    printf("\n\rTask1 running again...\n\r");

    #if  1
        tickVal = 1;
        printf("start taskDelay %ds ..... now!\n\r",tickVal);
        __asm__(
            " swi #255 \n\t"                /* 
        );
    #endif
        printf("Task1 Delay 1s over...\n\r");

    }
    printf("Task1 is running over!\n\r"); 

    return 0;
}


對應匯編:

第一種寫法匯編
30001894 <TEST_TestTask1>:
30001894:   e92d4030    stmdb   sp!, {r4, r5, lr}
30001898:   e59f0050    ldr r0, [pc, #80]   ; 300018f0 <.text+0x18f0>
3000189c:   eb00009e    bl  30001b1c <printf>
300018a0:   ebffff72    bl  30001670 <sp_value>
300018a4:   e1a01000    mov r1, r0
300018a8:   e59f0044    ldr r0, [pc, #68]   ; 300018f4 <.text+0x18f4>
300018ac:   eb00009a    bl  30001b1c <printf>
300018b0:   ebffff67    bl  30001654 <cpsr_value>
300018b4:   e1a01000    mov r1, r0
300018b8:   e59f0038    ldr r0, [pc, #56]   ; 300018f8 <.text+0x18f8>
300018bc:   eb000096    bl  30001b1c <printf>
300018c0:   e59f5034    ldr r5, [pc, #52]   ; 300018fc <.text+0x18fc>
300018c4:   e3a04001    mov r4, #1  ; 0x1
300018c8:   e59f0030    ldr r0, [pc, #48]   ; 30001900 <.text+0x1900>
300018cc:   eb000092    bl  30001b1c <printf>
300018d0:   e1a01004    mov r1, r4
300018d4:   e59f0028    ldr r0, [pc, #40]   ; 30001904 <.text+0x1904>
300018d8:   e5854000    str r4, [r5]
300018dc:   eb00008e    bl  30001b1c <printf>
##300018e0: ef0000ff    swi 0x000000ff
##300018e4: e59f001c    ldr r0, [pc, #28]   ; 30001908 <.text+0x1908>
##300018e8: eb00008b    bl  30001b1c <printf>
300018ec:   eafffff5    b   300018c8 <TEST_TestTask1+0x34>
300018f0:   3000411c    andcc   r4, r0, ip, lsl r1
300018f4:   30004084    andcc   r4, r0, r4, lsl #1
300018f8:   30003978    andcc   r3, r0, r8, ror r9
300018fc:   30005bfc    strccd  r5, [r0], -ip
30001900:   30004130    andcc   r4, r0, r0, lsr r1
30001904:   30003c04    andcc   r3, r0, r4, lsl #24
30001908:   3000414c    andcc   r4, r0, ip, asr #2

第二種寫法代碼如下:

int TEST_TestTask1(void)
{
    printf("\n\rTask1 start...\n\r");
    printf("user mode ...      SP 0x%x\n\r",sp_value());
    printf("CPSR==> 0x%x\n\r",cpsr_value());
    while(1){
      printf("\n\rTask1 running again...\n\r");
          taskDelay(1);
      printf("Task1 Delay 1s over...\n\r");

    }
    printf("Task1 is running over!\n\r"); 
    return 0;
}


對應匯編

30001894 <TEST_TestTask1>:
30001894:   e52de004    str lr, [sp, #-4]!
30001898:   e59f003c    ldr r0, [pc, #60]   ; 300018dc <.text+0x18dc>
3000189c:   eb000097    bl  30001b00 <printf>
300018a0:   ebffff72    bl  30001670 <sp_value>
300018a4:   e1a01000    mov r1, r0
300018a8:   e59f0030    ldr r0, [pc, #48]   ; 300018e0 <.text+0x18e0>
300018ac:   eb000093    bl  30001b00 <printf>
300018b0:   ebffff67    bl  30001654 <cpsr_value>
300018b4:   e1a01000    mov r1, r0
300018b8:   e59f0024    ldr r0, [pc, #36]   ; 300018e4 <.text+0x18e4>
300018bc:   eb00008f    bl  30001b00 <printf>
300018c0:   e59f0020    ldr r0, [pc, #32]   ; 300018e8 <.text+0x18e8>
300018c4:   eb00008d    bl  30001b00 <printf>
300018c8:   e3a00001    mov r0, #1  ; 0x1
##300018cc: ebfffced    bl  30000c88 <taskDelay>
300018d0:   e59f0014    ldr r0, [pc, #20]   ; 300018ec <.text+0x18ec>
300018d4:   eb000089    bl  30001b00 <printf>
300018d8:   eafffff8    b   300018c0 <TEST_TestTask1+0x2c>

30000c88 <taskDelay>:
30000c88:   e59f201c    ldr r2, [pc, #28]   ; 30000cac <.text+0xcac>
30000c8c:   e1a03000    mov r3, r0
##30000c90: e52de004    str lr, [sp, #-4]!
30000c94:   e1a01000    mov r1, r0
30000c98:   e59f0010    ldr r0, [pc, #16]   ; 30000cb0 <.text+0xcb0>
30000c9c:   e5823000    str r3, [r2]
30000ca0:   eb000396    bl  30001b00 <printf>
##30000ca4: ef0000ff    swi 0x000000ff
##30000ca8: e49df004    ldr pc, [sp], #4
30000cac:   30005bdc    ldrccd  r5, [r0], -ip
30000cb0:   30003be4    andcc   r3, r0, r4, ror #23


最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容