1@/*************************************************************************** 2@ * Copyright (c) 2024 Microsoft Corporation 3@ * 4@ * This program and the accompanying materials are made available under the 5@ * terms of the MIT License which is available at 6@ * https://opensource.org/licenses/MIT. 7@ * 8@ * SPDX-License-Identifier: MIT 9@ **************************************************************************/ 10@ 11@ 12@/**************************************************************************/ 13@/**************************************************************************/ 14@/** */ 15@/** ThreadX Component */ 16@/** */ 17@/** Thread */ 18@/** */ 19@/**************************************************************************/ 20@/**************************************************************************/ 21#ifdef TX_INCLUDE_USER_DEFINE_FILE 22#include "tx_user.h" 23#endif 24 25 .arm 26 27SVC_MODE = 0x13 @ SVC mode 28#ifdef TX_ENABLE_FIQ_SUPPORT 29CPSR_MASK = 0xDF @ Mask initial CPSR, IRQ & FIQ interrupts enabled 30#else 31CPSR_MASK = 0x9F @ Mask initial CPSR, IRQ interrupts enabled 32#endif 33@ 34@ 35@/* Define the 16-bit Thumb mode veneer for _tx_thread_stack_build for 36@ applications calling this function from to 16-bit Thumb mode. */ 37@ 38 .text 39 .align 2 40 .thumb 41 .global $_tx_thread_stack_build 42 .type $_tx_thread_stack_build,function 43$_tx_thread_stack_build: 44 BX pc @ Switch to 32-bit mode 45 NOP @ 46 .arm 47 STMFD sp!, {lr} @ Save return address 48 BL _tx_thread_stack_build @ Call _tx_thread_stack_build function 49 LDMFD sp!, {lr} @ Recover saved return address 50 BX lr @ Return to 16-bit caller 51@ 52@ 53 .text 54 .align 2 55@/**************************************************************************/ 56@/* */ 57@/* FUNCTION RELEASE */ 58@/* */ 59@/* _tx_thread_stack_build ARM9/GNU */ 60@/* 6.2.1 */ 61@/* AUTHOR */ 62@/* */ 63@/* William E. Lamie, Microsoft Corporation */ 64@/* */ 65@/* DESCRIPTION */ 66@/* */ 67@/* This function builds a stack frame on the supplied thread's stack. */ 68@/* The stack frame results in a fake interrupt return to the supplied */ 69@/* function pointer. */ 70@/* */ 71@/* INPUT */ 72@/* */ 73@/* thread_ptr Pointer to thread control blk */ 74@/* function_ptr Pointer to return function */ 75@/* */ 76@/* OUTPUT */ 77@/* */ 78@/* None */ 79@/* */ 80@/* CALLS */ 81@/* */ 82@/* None */ 83@/* */ 84@/* CALLED BY */ 85@/* */ 86@/* _tx_thread_create Create thread service */ 87@/* */ 88@/* RELEASE HISTORY */ 89@/* */ 90@/* DATE NAME DESCRIPTION */ 91@/* */ 92@/* 09-30-2020 William E. Lamie Initial Version 6.1 */ 93@/* 03-08-2023 Cindy Deng Modified comment(s), added */ 94@/* #include tx_user.h, */ 95@/* resulting in version 6.2.1 */ 96@/* */ 97@/**************************************************************************/ 98@VOID _tx_thread_stack_build(TX_THREAD *thread_ptr, VOID (*function_ptr)(VOID)) 99@{ 100 .global _tx_thread_stack_build 101 .type _tx_thread_stack_build,function 102_tx_thread_stack_build: 103@ 104@ 105@ /* Build a fake interrupt frame. The form of the fake interrupt stack 106@ on the ARM9 should look like the following after it is built: 107@ 108@ Stack Top: 1 Interrupt stack frame type 109@ CPSR Initial value for CPSR 110@ a1 (r0) Initial value for a1 111@ a2 (r1) Initial value for a2 112@ a3 (r2) Initial value for a3 113@ a4 (r3) Initial value for a4 114@ v1 (r4) Initial value for v1 115@ v2 (r5) Initial value for v2 116@ v3 (r6) Initial value for v3 117@ v4 (r7) Initial value for v4 118@ v5 (r8) Initial value for v5 119@ sb (r9) Initial value for sb 120@ sl (r10) Initial value for sl 121@ fp (r11) Initial value for fp 122@ ip (r12) Initial value for ip 123@ lr (r14) Initial value for lr 124@ pc (r15) Initial value for pc 125@ 0 For stack backtracing 126@ 127@ Stack Bottom: (higher memory address) */ 128@ 129 LDR r2, [r0, #16] @ Pickup end of stack area 130 BIC r2, r2, #7 @ Ensure 8-byte alignment 131 SUB r2, r2, #76 @ Allocate space for the stack frame 132@ 133@ /* Actually build the stack frame. */ 134@ 135 MOV r3, #1 @ Build interrupt stack type 136 STR r3, [r2, #0] @ Store stack type 137 MOV r3, #0 @ Build initial register value 138 STR r3, [r2, #8] @ Store initial r0 139 STR r3, [r2, #12] @ Store initial r1 140 STR r3, [r2, #16] @ Store initial r2 141 STR r3, [r2, #20] @ Store initial r3 142 STR r3, [r2, #24] @ Store initial r4 143 STR r3, [r2, #28] @ Store initial r5 144 STR r3, [r2, #32] @ Store initial r6 145 STR r3, [r2, #36] @ Store initial r7 146 STR r3, [r2, #40] @ Store initial r8 147 STR r3, [r2, #44] @ Store initial r9 148 LDR r3, [r0, #12] @ Pickup stack starting address 149 STR r3, [r2, #48] @ Store initial r10 (sl) 150 LDR r3,=_tx_thread_schedule @ Pickup address of _tx_thread_schedule for GDB backtrace 151 STR r3, [r2, #60] @ Store initial r14 (lr) 152 MOV r3, #0 @ Build initial register value 153 STR r3, [r2, #52] @ Store initial r11 154 STR r3, [r2, #56] @ Store initial r12 155 STR r1, [r2, #64] @ Store initial pc 156 STR r3, [r2, #68] @ 0 for back-trace 157 MRS r1, CPSR @ Pickup CPSR 158 BIC r1, r1, #CPSR_MASK @ Mask mode bits of CPSR 159 ORR r3, r1, #SVC_MODE @ Build CPSR, SVC mode, interrupts enabled 160 STR r3, [r2, #4] @ Store initial CPSR 161@ 162@ /* Setup stack pointer. */ 163@ thread_ptr -> tx_thread_stack_ptr = r2; 164@ 165 STR r2, [r0, #8] @ Save stack pointer in thread's 166 @ control block 167#ifdef __THUMB_INTERWORK 168 BX lr @ Return to caller 169#else 170 MOV pc, lr @ Return to caller 171#endif 172@} 173 174 175