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 EQU 0xD3 ; Disable IRQ/FIQ SVC mode 35IRQ_MODE EQU 0xD2 ; Disable IRQ/FIQ IRQ mode 36FIQ_MODE EQU 0xD1 ; Disable IRQ/FIQ FIQ mode 37SYS_MODE EQU 0xDF ; Disable IRQ/FIQ SYS mode 38HEAP_SIZE EQU 4096 ; Heap size 39FIQ_STACK_SIZE EQU 512 ; FIQ stack size 40SYS_STACK_SIZE EQU 1024 ; SYS stack size (used for nested interrupts) 41IRQ_STACK_SIZE EQU 1024 ; IRQ stack size 42; 43; 44;/* ARM9 ARMulator Timer and Interrupt controller information. This depends on 45; the ARMulator's Interrupt Controller and Timer being enabled in the default.ami. 46; In addition, the addresses must match those specified in the peripherals.ami file. 47; Please refer to section 2.10 and 4.16 of the Debug Target Guide, version 1.2. */ 48; 49IRQStatus EQU 0x0a000000 ; IRQ Status Register 50IRQRawStatus EQU 0x0a000004 ; IRQ Raw Status Register 51IRQEnable EQU 0x0a000008 ; IRQ Enable Set Register 52IRQEnableClear EQU 0x0a00000C ; IRQ Enable Clear Register 53IRQSoft EQU 0x0a000010 ; IRQ Soft 54FIQStatus EQU 0x0a000100 ; FIQ Status Register 55FIQRawStatus EQU 0x0a000104 ; FIQ Raw Status Register 56FIQEnable EQU 0x0a000108 ; FIQ Enable Set Register 57FIQEnableClear EQU 0x0a00010C ; FIQ Enable Clear Register 58 59TIMER1_BIT EQU 0x00000010 ; IRQ/FIQ Timer1 bit 60TIMER2_BIT EQU 0x00000020 ; IRQ/FIQ Timer2 bit 61 62Timer1Load EQU 0x0a800000 ; Timer1 Load Register 63Timer1Value EQU 0x0a800004 ; Timer1 Value Register 64Timer1Control EQU 0x0a800008 ; Timer1 Control Register 65Timer1Clear EQU 0x0a80000C ; Timer1 Clear Register 66 67Timer1Mode EQU 0x000000C0 ; Timer1 Control Value, Timer enable, periodic, no prescaler 68Timer1Period EQU 0x0000FFFF ; Timer1 count-down period, maximum value 69 70Timer2Load EQU 0x0a800020 ; Timer2 Load Register 71Timer2Value EQU 0x0a800024 ; Timer2 Value Register 72Timer2Control EQU 0x0a800028 ; Timer2 Control Register 73Timer2Clear EQU 0x0a80002C ; Timer2 Clear Register 74; 75; 76 IMPORT _tx_thread_system_stack_ptr 77 IMPORT _tx_initialize_unused_memory 78 IMPORT _tx_thread_context_save 79 IMPORT _tx_thread_context_restore 80 IF :DEF:TX_ENABLE_FIQ_SUPPORT 81 IMPORT _tx_thread_fiq_context_save 82 IMPORT _tx_thread_fiq_context_restore 83 ENDIF 84 IF :DEF:TX_ENABLE_IRQ_NESTING 85 IMPORT _tx_thread_irq_nesting_start 86 IMPORT _tx_thread_irq_nesting_end 87 ENDIF 88 IF :DEF:TX_ENABLE_FIQ_NESTING 89 IMPORT _tx_thread_fiq_nesting_start 90 IMPORT _tx_thread_fiq_nesting_end 91 ENDIF 92 IMPORT _tx_timer_interrupt 93 IMPORT __main 94 IMPORT _tx_version_id 95 IMPORT _tx_build_options 96 IMPORT |Image$$ZI$$Limit| 97; 98; 99 AREA Init, CODE, READONLY 100; 101;/* Define the ARM9 vector area. This should be located or copied to 0. */ 102; 103 EXPORT __vectors 104__vectors 105 LDR pc,=__main ; Reset goes to startup function 106 LDR pc,=__tx_undefined ; Undefined handler 107 LDR pc,=__tx_swi_interrupt ; Software interrupt handler 108 LDR pc,=__tx_prefetch_handler ; Prefetch exception handler 109 LDR pc,=__tx_abort_handler ; Abort exception handler 110 LDR pc,=__tx_reserved_handler ; Reserved exception handler 111 LDR pc,=__tx_irq_handler ; IRQ interrupt handler 112 LDR pc,=__tx_fiq_handler ; FIQ interrupt handler 113; 114; 115 AREA ||.text||, CODE, READONLY 116;/**************************************************************************/ 117;/* */ 118;/* FUNCTION RELEASE */ 119;/* */ 120;/* _tx_initialize_low_level ARM9/AC5 */ 121;/* 6.1 */ 122;/* AUTHOR */ 123;/* */ 124;/* William E. Lamie, Microsoft Corporation */ 125;/* */ 126;/* DESCRIPTION */ 127;/* */ 128;/* This function is responsible for any low-level processor */ 129;/* initialization, including setting up interrupt vectors, setting */ 130;/* up a periodic timer interrupt source, saving the system stack */ 131;/* pointer for use in ISR processing later, and finding the first */ 132;/* available RAM memory address for tx_application_define. */ 133;/* */ 134;/* INPUT */ 135;/* */ 136;/* None */ 137;/* */ 138;/* OUTPUT */ 139;/* */ 140;/* None */ 141;/* */ 142;/* CALLS */ 143;/* */ 144;/* None */ 145;/* */ 146;/* CALLED BY */ 147;/* */ 148;/* _tx_initialize_kernel_enter ThreadX entry function */ 149;/* */ 150;/* RELEASE HISTORY */ 151;/* */ 152;/* DATE NAME DESCRIPTION */ 153;/* */ 154;/* 09-30-2020 William E. Lamie Initial Version 6.1 */ 155;/* */ 156;/**************************************************************************/ 157;VOID _tx_initialize_low_level(VOID) 158;{ 159 EXPORT _tx_initialize_low_level 160_tx_initialize_low_level 161; 162; 163; /****** NOTE ****** We must be in SVC MODE at this point. Some monitors 164; enter this routine in USER mode and require a software interrupt to 165; change into SVC mode. */ 166; 167 LDR r1, =|Image$$ZI$$Limit| ; Get end of non-initialized RAM area 168 LDR r2, =HEAP_SIZE ; Pickup the heap size 169 ADD r1, r2, r1 ; Setup heap limit 170 ADD r1, r1, #4 ; Setup stack limit 171; 172 IF :DEF:TX_ENABLE_IRQ_NESTING 173; /* Setup the system mode stack for nested interrupt support */ 174 LDR r2, =SYS_STACK_SIZE ; Pickup stack size 175 MOV r3, #SYS_MODE ; Build SYS mode CPSR 176 MSR CPSR_cxsf, r3 ; Enter SYS mode 177 ADD r1, r1, r2 ; Calculate start of SYS stack 178 BIC r1, r1, #7 ; Ensure 8-byte alignment 179 MOV sp, r1 ; Setup SYS stack pointer 180 ENDIF 181; 182 LDR r2, =FIQ_STACK_SIZE ; Pickup stack size 183 MOV r0, #FIQ_MODE ; Build FIQ mode CPSR 184 MSR CPSR_c, r0 ; Enter FIQ mode 185 ADD r1, r1, r2 ; Calculate start of FIQ stack 186 BIC r1, r1, #7 ; Ensure 8-byte alignment 187 MOV sp, r1 ; Setup FIQ stack pointer 188 MOV sl, #0 ; Clear sl 189 MOV fp, #0 ; Clear fp 190 LDR r2, =IRQ_STACK_SIZE ; Pickup IRQ (system stack size) 191 MOV r0, #IRQ_MODE ; Build IRQ mode CPSR 192 MSR CPSR_c, r0 ; Enter IRQ mode 193 ADD r1, r1, r2 ; Calculate start of IRQ stack 194 BIC r1, r1, #7 ; Ensure 8-byte alignment 195 MOV sp, r1 ; Setup IRQ stack pointer 196 MOV r0, #SVC_MODE ; Build SVC mode CPSR 197 MSR CPSR_c, r0 ; Enter SVC mode 198 LDR r3, =_tx_thread_system_stack_ptr ; Pickup stack pointer 199 STR r1, [r3, #0] ; Save the system stack 200; 201; /* Save the system stack pointer. */ 202; _tx_thread_system_stack_ptr = (VOID_PTR) (sp); 203; 204 LDR r1, =_tx_thread_system_stack_ptr ; Pickup address of system stack ptr 205 LDR r0, [r1, #0] ; Pickup system stack 206 ADD r0, r0, #4 ; Increment to next free word 207; 208; /* Save the first available memory address. */ 209; _tx_initialize_unused_memory = (VOID_PTR) |Image$$ZI$$Limit| + HEAP + [SYS_STACK] + FIQ_STACK + IRQ_STACK; 210; 211 LDR r2, =_tx_initialize_unused_memory ; Pickup unused memory ptr address 212 STR r0, [r2, #0] ; Save first free memory address 213; 214; /* Setup Timer for periodic interrupts. */ 215; 216; /* Setup ARMulator Timer1 for periodic interrupts. */ 217; 218 LDR r0,=IRQEnable ; Build address of IRQ enable register 219 LDR r1,=TIMER1_BIT ; Build value of Timer1 IRQ enable 220 STR r1,[r0] ; Enable IRQ interrupts for Timer1 221 222 LDR r0,=Timer1Load ; Build address of Timer1 load register 223 LDR r1,=Timer1Period ; Build Timer1 periodic value 224 STR r1,[r0] ; Set Timer1 load value 225 226 LDR r0,=Timer1Control ; Build address of Timer1 control register 227 LDR r1,=Timer1Mode ; Build Timer1 control value 228 STR r1,[r0] ; Enable Timer1 229; 230; /* Done, return to caller. */ 231; 232 IF {INTER} = {TRUE} 233 BX lr ; Return to caller 234 ELSE 235 MOV pc, lr ; Return to caller 236 ENDIF 237;} 238; 239; 240;/* Define initial heap/stack routine for the ARM RealView (and ADS) startup code. This 241; routine will set the initial stack to use the ThreadX IRQ & FIQ & 242; (optionally SYS) stack areas. */ 243; 244 EXPORT __user_initial_stackheap 245__user_initial_stackheap 246 LDR r0, =|Image$$ZI$$Limit| ; Get end of non-initialized RAM area 247 LDR r2, =HEAP_SIZE ; Pickup the heap size 248 ADD r2, r2, r0 ; Setup heap limit 249 ADD r3, r2, #4 ; Setup stack limit 250 MOV r1, r3 ; Setup start of stack 251 IF :DEF:TX_ENABLE_IRQ_NESTING 252 LDR r12, =SYS_STACK_SIZE ; Pickup IRQ system stack 253 ADD r1, r1, r12 ; Setup the return system stack 254 BIC r1, r1, #7 ; Ensure 8-byte alignment 255 ENDIF 256 LDR r12, =FIQ_STACK_SIZE ; Pickup FIQ stack size 257 ADD r1, r1, r12 ; Setup the return system stack 258 BIC r1, r1, #7 ; Ensure 8-byte alignment 259 LDR r12, =IRQ_STACK_SIZE ; Pickup IRQ system stack 260 ADD r1, r1, r12 ; Setup the return system stack 261 BIC r1, r1, #7 ; Ensure 8-byte alignment 262 IF {INTER} = {TRUE} 263 BX lr ; Return to caller 264 ELSE 265 MOV pc, lr ; Return to caller 266 ENDIF 267; 268; 269;/* Define shells for each of the interrupt vectors. */ 270; 271 EXPORT __tx_undefined 272__tx_undefined 273 B __tx_undefined ; Undefined handler 274; 275 EXPORT __tx_swi_interrupt 276__tx_swi_interrupt 277 B __tx_swi_interrupt ; Software interrupt handler 278; 279 EXPORT __tx_prefetch_handler 280__tx_prefetch_handler 281 B __tx_prefetch_handler ; Prefetch exception handler 282; 283 EXPORT __tx_abort_handler 284__tx_abort_handler 285 B __tx_abort_handler ; Abort exception handler 286; 287 EXPORT __tx_reserved_handler 288__tx_reserved_handler 289 B __tx_reserved_handler ; Reserved exception handler 290; 291; 292 EXPORT __tx_irq_handler 293 EXPORT __tx_irq_processing_return 294__tx_irq_handler 295; 296; /* Jump to context save to save system context. */ 297 B _tx_thread_context_save 298__tx_irq_processing_return 299; 300; /* At this point execution is still in the IRQ mode. The CPSR, point of 301; interrupt, and all C scratch registers are available for use. In 302; addition, IRQ interrupts may be re-enabled - with certain restrictions - 303; if nested IRQ interrupts are desired. Interrupts may be re-enabled over 304; small code sequences where lr is saved before enabling interrupts and 305; restored after interrupts are again disabled. */ 306; 307; 308; /* Check for Timer1 interrupts on the ARMulator. */ 309 310 LDR r1,=IRQStatus ; Pickup address of IRQStatus register 311 LDR r2, [r1] ; Read IRQStatus 312 LDR r0,=TIMER1_BIT ; Pickup Timer1 interrupt present bit 313 AND r2, r2, r0 ; Is this a timer interrupt? 314 CMP r2, r0 ; 315 BNE _tx_not_timer_interrupt ; If 0, not a timer interrupt 316 317 LDR r1,=Timer1Clear ; Build address of Timer1 clear register 318 MOV r0,#0 ; 319 STR r0, [r1] ; Clear timer 0 interrupt 320 321 BL _tx_timer_interrupt ; Timer interrupt handler 322_tx_not_timer_interrupt 323; 324; /* Interrupt nesting is allowed after calling _tx_thread_irq_nesting_start 325; from IRQ mode with interrupts disabled. This routine switches to the 326; system mode and returns with IRQ interrupts enabled. 327; 328; NOTE: It is very important to ensure all IRQ interrupts are cleared 329; prior to enabling nested IRQ interrupts. */ 330 IF :DEF:TX_ENABLE_IRQ_NESTING 331 BL _tx_thread_irq_nesting_start 332 ENDIF 333; 334; 335; /* Application IRQ handlers can be called here! */ 336; 337; /* If interrupt nesting was started earlier, the end of interrupt nesting 338; service must be called before returning to _tx_thread_context_restore. 339; This routine returns in processing in IRQ mode with interrupts disabled. */ 340 IF :DEF:TX_ENABLE_IRQ_NESTING 341 BL _tx_thread_irq_nesting_end 342 ENDIF 343; 344; /* Jump to context restore to restore system context. */ 345 B _tx_thread_context_restore 346; 347; 348; /* This is an example of a vectored IRQ handler. */ 349; 350 EXPORT __tx_example_vectored_irq_handler 351__tx_example_vectored_irq_handler 352; 353; 354; /* Save initial context and call context save to prepare for 355; vectored ISR execution. */ 356; 357; STMDB sp!, {r0-r3} ; Save some scratch registers 358; MRS r0, SPSR ; Pickup saved SPSR 359; SUB lr, lr, #4 ; Adjust point of interrupt 360; STMDB sp!, {r0, r10, r12, lr} ; Store other scratch registers 361; BL _tx_thread_vectored_context_save ; Vectored context save 362; 363; /* At this point execution is still in the IRQ mode. The CPSR, point of 364; interrupt, and all C scratch registers are available for use. In 365; addition, IRQ interrupts may be re-enabled - with certain restrictions - 366; if nested IRQ interrupts are desired. Interrupts may be re-enabled over 367; small code sequences where lr is saved before enabling interrupts and 368; restored after interrupts are again disabled. */ 369; 370; 371; /* Interrupt nesting is allowed after calling _tx_thread_irq_nesting_start 372; from IRQ mode with interrupts disabled. This routine switches to the 373; system mode and returns with IRQ interrupts enabled. 374; 375; NOTE: It is very important to ensure all IRQ interrupts are cleared 376; prior to enabling nested IRQ interrupts. */ 377; IF :DEF:TX_ENABLE_IRQ_NESTING 378; BL _tx_thread_irq_nesting_start 379; ENDIF 380; 381; /* Application IRQ handlers can be called here! */ 382; 383; /* If interrupt nesting was started earlier, the end of interrupt nesting 384; service must be called before returning to _tx_thread_context_restore. 385; This routine returns in processing in IRQ mode with interrupts disabled. */ 386; IF :DEF:TX_ENABLE_IRQ_NESTING 387; BL _tx_thread_irq_nesting_end 388; ENDIF 389; 390; /* Jump to context restore to restore system context. */ 391; B _tx_thread_context_restore 392; 393; 394 IF :DEF:TX_ENABLE_FIQ_SUPPORT 395 EXPORT __tx_fiq_handler 396 EXPORT __tx_fiq_processing_return 397__tx_fiq_handler 398; 399; /* Jump to fiq context save to save system context. */ 400 B _tx_thread_fiq_context_save 401__tx_fiq_processing_return 402; 403; /* At this point execution is still in the FIQ mode. The CPSR, point of 404; interrupt, and all C scratch registers are available for use. */ 405; 406; /* Interrupt nesting is allowed after calling _tx_thread_fiq_nesting_start 407; from FIQ mode with interrupts disabled. This routine switches to the 408; system mode and returns with FIQ interrupts enabled. 409; 410; NOTE: It is very important to ensure all FIQ interrupts are cleared 411; prior to enabling nested FIQ interrupts. */ 412 IF :DEF:TX_ENABLE_FIQ_NESTING 413 BL _tx_thread_fiq_nesting_start 414 ENDIF 415; 416; /* Application FIQ handlers can be called here! */ 417; 418; /* If interrupt nesting was started earlier, the end of interrupt nesting 419; service must be called before returning to _tx_thread_fiq_context_restore. */ 420 IF :DEF:TX_ENABLE_FIQ_NESTING 421 BL _tx_thread_fiq_nesting_end 422 ENDIF 423; 424; /* Jump to fiq context restore to restore system context. */ 425 B _tx_thread_fiq_context_restore 426; 427; 428 ELSE 429 EXPORT __tx_fiq_handler 430__tx_fiq_handler 431 B __tx_fiq_handler ; FIQ interrupt handler 432 ENDIF 433; 434; /* Reference build options and version ID to ensure they come in. */ 435; 436 LDR r2, =_tx_build_options ; Pickup build options variable address 437 LDR r0, [r2, #0] ; Pickup build options content 438 LDR r2, =_tx_version_id ; Pickup version ID variable address 439 LDR r0, [r2, #0] ; Pickup version ID content 440; 441; 442 END 443 444