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;/** Initialize */ 18;/** */ 19;/**************************************************************************/ 20;/**************************************************************************/ 21; 22; 23;#define TX_SOURCE_CODE 24; 25; 26;/* Include necessary system files. */ 27; 28;#include "tx_api.h" 29;#include "tx_initialize.h" 30;#include "tx_thread.h" 31;#include "tx_timer.h" 32; 33; 34SVC_MODE DEFINE 0xD3 ; Disable irq,fiq SVC mode 35IRQ_MODE DEFINE 0xD2 ; Disable irq,fiq IRQ mode 36FIQ_MODE DEFINE 0xD1 ; Disable irq,fiq FIQ mode 37SYS_MODE DEFINE 0xDF ; Disable irq,fiq SYS mode 38; 39; 40 41 EXTERN _tx_thread_system_stack_ptr 42 EXTERN _tx_initialize_unused_memory 43 EXTERN _tx_thread_context_save 44; EXTERN _tx_thread_vectored_context_save 45 EXTERN _tx_thread_context_restore 46#ifdef TX_ENABLE_FIQ_SUPPORT 47 EXTERN _tx_thread_fiq_context_save 48 EXTERN _tx_thread_fiq_context_restore 49#endif 50#ifdef TX_ENABLE_IRQ_NESTING 51 EXTERN _tx_thread_irq_nesting_start 52 EXTERN _tx_thread_irq_nesting_end 53#endif 54#ifdef TX_ENABLE_FIQ_NESTING 55 EXTERN _tx_thread_fiq_nesting_start 56 EXTERN _tx_thread_fiq_nesting_end 57#endif 58 EXTERN _tx_timer_interrupt 59 EXTERN ?cstartup 60 EXTERN _tx_build_options 61 EXTERN _tx_version_id 62; 63; 64; 65;/* Define the FREE_MEM segment that will specify where free memory is 66; defined. This must also be located in at the end of other RAM segments 67; in the linker control file. The value of this segment is what is passed 68; to tx_application_define. */ 69; 70 RSEG FREE_MEM:DATA 71 PUBLIC __tx_free_memory_start 72__tx_free_memory_start 73 DS32 4 74; 75; 76; 77; 78;/**************************************************************************/ 79;/* */ 80;/* FUNCTION RELEASE */ 81;/* */ 82;/* _tx_initialize_low_level ARM9/IAR */ 83;/* 6.1 */ 84;/* AUTHOR */ 85;/* */ 86;/* William E. Lamie, Microsoft Corporation */ 87;/* */ 88;/* DESCRIPTION */ 89;/* */ 90;/* This function is responsible for any low-level processor */ 91;/* initialization, including setting up interrupt vectors, setting */ 92;/* up a periodic timer interrupt source, saving the system stack */ 93;/* pointer for use in ISR processing later, and finding the first */ 94;/* available RAM memory address for tx_application_define. */ 95;/* */ 96;/* INPUT */ 97;/* */ 98;/* None */ 99;/* */ 100;/* OUTPUT */ 101;/* */ 102;/* None */ 103;/* */ 104;/* CALLS */ 105;/* */ 106;/* None */ 107;/* */ 108;/* CALLED BY */ 109;/* */ 110;/* _tx_initialize_kernel_enter ThreadX entry function */ 111;/* */ 112;/* RELEASE HISTORY */ 113;/* */ 114;/* DATE NAME DESCRIPTION */ 115;/* */ 116;/* 09-30-2020 William E. Lamie Initial Version 6.1 */ 117;/* */ 118;/**************************************************************************/ 119;VOID _tx_initialize_low_level(VOID) 120;{ 121 RSEG .text:CODE:NOROOT(2) 122 CODE32 123 PUBLIC _tx_initialize_low_level 124_tx_initialize_low_level 125; 126; /****** NOTE ****** The IAR 4.11a and above releases call main in SYS mode. */ 127; 128; /* Remember the stack pointer, link register, and switch to SVC mode. */ 129; 130 MOV r0, sp ; Remember the SP 131 MOV r1, lr ; Remember the LR 132 MOV r3, #SVC_MODE ; Build SVC mode CPSR 133 MSR CPSR_cxsf, r3 ; Switch to SVC mode 134 MOV sp, r0 ; Inherit the stack pointer setup by cstartup 135 MOV lr, r1 ; Inherit the link register 136; 137; /* Pickup the start of free memory. */ 138; 139 LDR r0, =__tx_free_memory_start ; Get end of non-initialized RAM area 140; 141; /* Save the system stack pointer. */ 142; _tx_thread_system_stack_ptr = (VOID_PTR) (sp); 143; 144; /* Save the first available memory address. */ 145; _tx_initialize_unused_memory = (VOID_PTR) FREE_MEM; 146; 147 LDR r2, =_tx_initialize_unused_memory ; Pickup unused memory ptr address 148 STR r0, [r2, #0] ; Save first free memory address 149; 150; /* Setup Timer for periodic interrupts. */ 151; 152; /* Done, return to caller. */ 153; 154#ifdef TX_THUMB 155 BX lr ; Return to caller 156#else 157 MOV pc, lr ; Return to caller 158#endif 159;} 160; 161;/* Define shells for each of the interrupt vectors. */ 162; 163 RSEG .text:CODE:NOROOT(2) 164 PUBLIC __tx_undefined 165__tx_undefined 166 B __tx_undefined ; Undefined handler 167; 168 RSEG .text:CODE:NOROOT(2) 169 PUBLIC __tx_swi_interrupt 170__tx_swi_interrupt 171 B __tx_swi_interrupt ; Software interrupt handler 172; 173 RSEG .text:CODE:NOROOT(2) 174 PUBLIC __tx_prefetch_handler 175__tx_prefetch_handler 176 B __tx_prefetch_handler ; Prefetch exception handler 177; 178 RSEG .text:CODE:NOROOT(2) 179 PUBLIC __tx_abort_handler 180__tx_abort_handler 181 B __tx_abort_handler ; Abort exception handler 182; 183 RSEG .text:CODE:NOROOT(2) 184 PUBLIC __tx_reserved_handler 185__tx_reserved_handler 186 B __tx_reserved_handler ; Reserved exception handler 187; 188 RSEG .text:CODE:NOROOT(2) 189 PUBLIC __tx_irq_handler 190 RSEG .text:CODE:NOROOT(2) 191 PUBLIC __tx_irq_processing_return 192__tx_irq_handler 193; 194; /* Jump to context save to save system context. */ 195 B _tx_thread_context_save 196__tx_irq_processing_return 197; 198; /* At this point execution is still in the IRQ mode. The CPSR, point of 199; interrupt, and all C scratch registers are available for use. In 200; addition, IRQ interrupts may be re-enabled - with certain restrictions - 201; if nested IRQ interrupts are desired. Interrupts may be re-enabled over 202; small code sequences where lr is saved before enabling interrupts and 203; restored after interrupts are again disabled. */ 204; 205; /* Interrupt nesting is allowed after calling _tx_thread_irq_nesting_start 206; from IRQ mode with interrupts disabled. This routine switches to the 207; system mode and returns with IRQ interrupts enabled. 208; 209; NOTE: It is very important to ensure all IRQ interrupts are cleared 210; prior to enabling nested IRQ interrupts. */ 211#ifdef TX_ENABLE_IRQ_NESTING 212 BL _tx_thread_irq_nesting_start 213#endif 214; 215; /* For debug purpose, execute the timer interrupt processing here. In 216; a real system, some kind of status indication would have to be checked 217; before the timer interrupt handler could be called. */ 218; 219 BL _tx_timer_interrupt ; Timer interrupt handler 220; 221; /* Application IRQ handlers can be called here! */ 222; 223; /* If interrupt nesting was started earlier, the end of interrupt nesting 224; service must be called before returning to _tx_thread_context_restore. 225; This routine returns in processing in IRQ mode with interrupts disabled. */ 226#ifdef TX_ENABLE_IRQ_NESTING 227 BL _tx_thread_irq_nesting_end 228#endif 229; 230; /* Jump to context restore to restore system context. */ 231 B _tx_thread_context_restore 232; 233; 234; /* This is an example of a vectored IRQ handler. */ 235; 236; RSEG .text:CODE:NOROOT(2) 237; PUBLIC __tx_example_vectored_irq_handler 238;__tx_example_vectored_irq_handler 239; 240; /* Jump to context save to save system context. */ 241; STMDB sp!, {r0-r3} ; Save some scratch registers 242; MRS r0, SPSR ; Pickup saved SPSR 243; SUB lr, lr, #4 ; Adjust point of interrupt 244; STMDB sp!, {r0, r10, r12, lr} ; Store other registers 245; BL _tx_thread_vectored_context_save 246; 247; /* At this point execution is still in the IRQ mode. The CPSR, point of 248; interrupt, and all C scratch registers are available for use. In 249; addition, IRQ interrupts may be re-enabled - with certain restrictions - 250; if nested IRQ interrupts are desired. Interrupts may be re-enabled over 251; small code sequences where lr is saved before enabling interrupts and 252; restored after interrupts are again disabled. */ 253; 254; /* Interrupt nesting is allowed after calling _tx_thread_irq_nesting_start 255; from IRQ mode with interrupts disabled. This routine switches to the 256; system mode and returns with IRQ interrupts enabled. 257; 258; NOTE: It is very important to ensure all IRQ interrupts are cleared 259; prior to enabling nested IRQ interrupts. */ 260;#ifdef TX_ENABLE_IRQ_NESTING 261; BL _tx_thread_irq_nesting_start 262;#endif 263; 264; /* Application IRQ handler is called here! */ 265; 266; /* If interrupt nesting was started earlier, the end of interrupt nesting 267; service must be called before returning to _tx_thread_context_restore. 268; This routine returns in processing in IRQ mode with interrupts disabled. */ 269;#ifdef TX_ENABLE_IRQ_NESTING 270; BL _tx_thread_irq_nesting_end 271;#endif 272; 273; /* Jump to context restore to restore system context. */ 274; B _tx_thread_context_restore 275; 276; 277#ifdef TX_ENABLE_FIQ_SUPPORT 278 RSEG .text:CODE:NOROOT(2) 279 PUBLIC __tx_fiq_handler 280 RSEG .text:CODE:NOROOT(2) 281 PUBLIC __tx_fiq_processing_return 282__tx_fiq_handler 283; 284; /* Jump to fiq context save to save system context. */ 285 B _tx_thread_fiq_context_save 286__tx_fiq_processing_return 287; 288; /* At this point execution is still in the FIQ mode. The CPSR, point of 289; interrupt, and all C scratch registers are available for use. */ 290; 291; /* Interrupt nesting is allowed after calling _tx_thread_fiq_nesting_start 292; from FIQ mode with interrupts disabled. This routine switches to the 293; system mode and returns with FIQ interrupts enabled. 294; 295; NOTE: It is very important to ensure all FIQ interrupts are cleared 296; prior to enabling nested FIQ interrupts. */ 297#ifdef TX_ENABLE_FIQ_NESTING 298 BL _tx_thread_fiq_nesting_start 299#endif 300; 301; /* Application FIQ handlers can be called here! */ 302; 303; /* If interrupt nesting was started earlier, the end of interrupt nesting 304; service must be called before returning to _tx_thread_fiq_context_restore. */ 305#ifdef TX_ENABLE_FIQ_NESTING 306 BL _tx_thread_fiq_nesting_end 307#endif 308; 309; /* Jump to fiq context restore to restore system context. */ 310 B _tx_thread_fiq_context_restore 311; 312; 313#else 314 RSEG .text:CODE:NOROOT(2) 315 PUBLIC __tx_fiq_handler 316__tx_fiq_handler 317 B __tx_fiq_handler ; FIQ interrupt handler 318#endif 319; 320; 321BUILD_OPTIONS 322 DC32 _tx_build_options ; Reference to ensure it comes in 323VERSION_ID 324 DC32 _tx_version_id ; Reference to ensure it comes in 325 END 326 327