1 Microsoft's Azure RTOS ThreadX for ARM9 2 3 Thumb & 32-bit Mode 4 5 Using the IAR Tools 6 71. Building the ThreadX run-time Library 8 9Building the ThreadX library is easy. First, open the Azure RTOS workspace 10azure_rtos.eww. Next, make the TX project the "active project" in the 11IAR Embedded Workbench and select the "Make" button. You should observe 12assembly and compilation of a series of ThreadX source files. This 13results in the ThreadX run-time library file tx.a, which is needed by 14the application. 15 16 172. Demonstration System 18 19The ThreadX demonstration is designed to execute under the IAR 20Windows-based ARM9 simulator. 21 22Building the demonstration is easy; simply make the sample_threadx.ewp project 23the "active project" in the IAR Embedded Workbench and select the 24"Make" button. 25 26You should observe the compilation of sample_threadx.c (which is the demonstration 27application) and linking with tx.a. The resulting file sample_threadx.out is a 28binary file that can be downloaded and executed on IAR's ARM9 simulator. 29 30A SPECIAL NOTE: The IAR ARM simulator does simulate interrupts. In order 31for the ThreadX demonstration to run properly, a periodic IRQ interrupt must 32be setup in the IAR debugging environment. We recommend setting an IRQ 33interrupt to execute every 9999 cycles. 34 35 363. System Initialization 37 38The entry point in ThreadX for the ARM9 using IAR tools is at label 39?cstartup. This is defined within the IAR compiler's startup code. In 40addition, this is where all static and global preset C variable 41initialization processing takes place. 42 43The ThreadX tx_initialize_low_level.s file is responsible for setting up 44various system data structures, and a periodic timer interrupt source. 45By default, the vector area is defined at the top of cstartup.s, which is 46a slightly modified from the base IAR file. 47 48The _tx_initialize_low_level function inside of tx_initialize_low_level.s 49also determines the first available address for use by the application, which 50is supplied as the sole input parameter to your application definition function, 51tx_application_define. To accomplish this, a section is created in 52tx_initialize_low_level.s called FREE_MEM, which must be located after all 53other RAM sections in memory. 54 55 564. Register Usage and Stack Frames 57 58The IAR ARM compiler assumes that registers r0-r3 (a1-a4) and r12 (ip) are 59scratch registers for each function. All other registers used by a C function 60must be preserved by the function. ThreadX takes advantage of this in 61situations where a context switch happens as a result of making a ThreadX 62service call (which is itself a C function). In such cases, the saved 63context of a thread is only the non-scratch registers. 64 65The following defines the saved context stack frames for context switches 66that occur as a result of interrupt handling or from thread-level API calls. 67All suspended threads have one of these two types of stack frames. The top 68of the suspended thread's stack is pointed to by tx_thread_stack_ptr in the 69associated thread control block TX_THREAD. 70 71 72 73 Offset Interrupted Stack Frame Non-Interrupt Stack Frame 74 75 0x00 1 0 76 0x04 CPSR CPSR 77 0x08 r0 (a1) r4 (v1) 78 0x0C r1 (a2) r5 (v2) 79 0x10 r2 (a3) r6 (v3) 80 0x14 r3 (a4) r7 (v4) 81 0x18 r4 (v1) r8 (v5) 82 0x1C r5 (v2) r9 (v6) 83 0x20 r6 (v3) r10 (v7) 84 0x24 r7 (v4) r11 (fp) 85 0x28 r8 (v5) r14 (lr) 86 0x2C r9 (v6) 87 0x30 r10 (v7) 88 0x34 r11 (fp) 89 0x38 r12 (ip) 90 0x3C r14 (lr) 91 0x40 PC 92 93 945. Conditional Compilation Switches 95 96The following are conditional compilation options for building the ThreadX library 97and application: 98 99 100 TX_ENABLE_FIQ_SUPPORT This assembler/compiler define enables 101 FIQ interrupt handling support in the 102 ThreadX assembly files. If used, 103 it should be used on all assembly 104 files and the generic C source of 105 ThreadX should be compiled with 106 TX_ENABLE_FIQ_SUPPORT defined as well. 107 108 TX_ENABLE_IRQ_NESTING This assembler define enables IRQ 109 nested support. If IRQ nested 110 interrupt support is needed, this 111 define should be applied to 112 tx_initialize_low_level.s. 113 114 TX_ENABLE_FIQ_NESTING This assembler define enables FIQ 115 nested support. If FIQ nested 116 interrupt support is needed, this 117 define should be applied to 118 tx_initialize_low_level.s. In addition, 119 IRQ nesting should also be enabled. 120 121 TX_DISABLE_ERROR_CHECKING If defined before tx_api.h is included, 122 this define causes basic ThreadX error 123 checking to be disabled. Please see 124 Chapter 2 in the "ThreadX User Guide" 125 for more details. 126 127 TX_MAX_PRIORITIES Defines the priority levels for ThreadX. 128 Legal values range from 32 through 129 1024 (inclusive) and MUST be evenly divisible 130 by 32. Increasing the number of priority levels 131 supported increases the RAM usage by 128 bytes 132 for every group of 32 priorities. However, there 133 is only a negligible effect on performance. By 134 default, this value is set to 32 priority levels. 135 136 TX_MINIMUM_STACK Defines the minimum stack size (in bytes). It is 137 used for error checking when threads are created. 138 The default value is port-specific and is found 139 in tx_port.h. 140 141 TX_TIMER_THREAD_STACK_SIZE Defines the stack size (in bytes) of the internal 142 ThreadX timer thread. This thread processes all 143 thread sleep requests as well as all service call 144 timeouts. In addition, all application timer callback 145 routines are invoked from this context. The default 146 value is port-specific and is found in tx_port.h. 147 148 TX_TIMER_THREAD_PRIORITY Defines the priority of the internal ThreadX timer 149 thread. The default value is priority 0 - the highest 150 priority in ThreadX. The default value is defined 151 in tx_port.h. 152 153 TX_TIMER_PROCESS_IN_ISR Defined, this option eliminates the internal system 154 timer thread for ThreadX. This results in improved 155 performance on timer events and smaller RAM requirements 156 because the timer stack and control block are no 157 longer needed. However, using this option moves all 158 the timer expiration processing to the timer ISR level. 159 By default, this option is not defined. 160 161 TX_REACTIVATE_INLINE Defined, this option performs reactivation of ThreadX 162 timers in-line instead of using a function call. This 163 improves performance but slightly increases code size. 164 By default, this option is not defined. 165 166 TX_DISABLE_STACK_FILLING Defined, placing the 0xEF value in each byte of each 167 thread's stack is disabled. By default, this option is 168 not defined. 169 170 TX_ENABLE_STACK_CHECKING Defined, this option enables ThreadX run-time stack checking, 171 which includes analysis of how much stack has been used and 172 examination of data pattern "fences" before and after the 173 stack area. If a stack error is detected, the registered 174 application stack error handler is called. This option does 175 result in slightly increased overhead and code size. Please 176 review the tx_thread_stack_error_notify API for more information. 177 By default, this option is not defined. 178 179 TX_DISABLE_PREEMPTION_THRESHOLD Defined, this option disables the preemption-threshold feature 180 and slightly reduces code size and improves performance. Of course, 181 the preemption-threshold capabilities are no longer available. 182 By default, this option is not defined. 183 184 TX_DISABLE_REDUNDANT_CLEARING Defined, this option removes the logic for initializing ThreadX 185 global C data structures to zero. This should only be used if 186 the compiler's initialization code sets all un-initialized 187 C global data to zero. Using this option slightly reduces 188 code size and improves performance during initialization. 189 By default, this option is not defined. 190 191 TX_DISABLE_NOTIFY_CALLBACKS Defined, this option disables the notify callbacks for various 192 ThreadX objects. Using this option slightly reduces code size 193 and improves performance. 194 195 TX_BLOCK_POOL_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 196 information on block pools. By default, this option is 197 not defined. 198 199 TX_BYTE_POOL_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 200 information on byte pools. By default, this option is 201 not defined. 202 203 TX_EVENT_FLAGS_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 204 information on event flags groups. By default, this option 205 is not defined. 206 207 TX_MUTEX_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 208 information on mutexes. By default, this option is 209 not defined. 210 211 TX_QUEUE_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 212 information on queues. By default, this option is 213 not defined. 214 215 TX_SEMAPHORE_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 216 information on semaphores. By default, this option is 217 not defined. 218 219 TX_THREAD_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 220 information on threads. By default, this option is 221 not defined. 222 223 TX_TIMER_ENABLE_PERFORMANCE_INFO Defined, this option enables the gathering of performance 224 information on timers. By default, this option is 225 not defined. 226 227 TX_ENABLE_EVENT_TRACE Defined, this option enables the internal ThreadX trace 228 feature. The trace buffer is supplied at a later time 229 via an application call to tx_trace_enable. 230 231 TX_TRACE_TIME_SOURCE This defines the time-stamp source for event tracing. 232 This define is only pertinent if the ThreadX library is 233 built with TX_ENABLE_EVENT_TRACE defined. 234 235 TX_TRACE_TIME_MASK This defines the number of valid bits in the event trace 236 time-stamp source defined previously. If the time-stamp 237 source is 16-bits, this value should be 0xFFFF. Alternatively, 238 if the time-stamp source is 32-bits, this value should be 239 0xFFFFFFFF. This define is only pertinent if the ThreadX 240 library is built with TX_ENABLE_EVENT_TRACE defined. 241 242 TX_THUMB Defined, this option enables the BX LR calling return sequence 243 in assembly files, to ensure correct operation on systems that 244 use both ARM and Thumb mode. By default, this option is 245 not defined 246 247 248 249 250 2516. Improving Performance 252 253The distribution version of ThreadX is built without any compiler 254optimizations. This makes it easy to debug because you can trace or set 255breakpoints inside of ThreadX itself. Of course, this costs some 256performance. To make it run faster, you can change the ThreadX library 257project to enable various compiler optimizations. 258 259In addition, you can eliminate the ThreadX basic API error checking by 260compiling your application code with the symbol TX_DISABLE_ERROR_CHECKING 261defined. 262 263 2647. Interrupt Handling 265 266ThreadX provides complete and high-performance interrupt handling for ARM9 267targets. There are a certain set of requirements that are defined in the 268following sub-sections: 269 270 2717.1 Vector Area 272 273The ARM9 vectors start at address zero. The demonstration system startup 274cstartup.s file contains the vectors and is loaded at address zero. 275On actual hardware platforms, this area might have to be copied to address 0. 276 277 2787.2 IRQ ISRs 279 280ThreadX fully manages standard and vectored IRQ interrupts. ThreadX also supports nested 281IRQ interrupts. The following sub-sections define the IRQ capabilities. 282 283 2847.2.1 Standard IRQ ISRs 285 286The standard ARM IRQ mechanism has a single interrupt vector at address 0x18. This IRQ 287interrupt is managed by the __tx_irq_handler code in tx_initialize_low_level. The following 288is the default IRQ handler defined in tx_initialize_low_level.s: 289 290 PUBLIC __tx_irq_handler 291 PUBLIC __tx_irq_processing_return 292__tx_irq_handler 293; 294; /* Jump to context save to save system context. */ 295 B _tx_thread_context_save 296__tx_irq_processing_return 297; 298; /* At this point execution is still in the IRQ mode. The CPSR, point of 299; interrupt, and all C scratch registers are available for use. Note 300; that IRQ interrupts are still disabled upon return from the context 301; save function. */ 302; 303; /* Application ISR dispatch call goes here! */ 304; 305; /* Jump to context restore to restore system context. */ 306 B _tx_thread_context_restore 307 308 3097.2.2 Vectored IRQ ISRs 310 311The vectored ARM IRQ mechanism has multiple interrupt vectors at addresses specified 312by the particular implementation. The following is an example IRQ handler defined in 313tx_initialize_low_level.s: 314 315 316 RSEG .text:CODE:NOROOT(2) 317 PUBLIC __tx_example_vectored_irq_handler 318__tx_example_vectored_irq_handler 319; 320; /* Jump to context save to save system context. */ 321 STMDB sp!, {r0-r3} ; Save some scratch registers 322 MRS r0, SPSR ; Pickup saved SPSR 323 SUB lr, lr, #4 ; Adjust point of interrupt 324 STMDB sp!, {r0, r10, r12, lr} ; Store other registers 325 BL _tx_thread_vectored_context_save 326; 327; /* At this point execution is still in the IRQ mode. The CPSR, point of 328; interrupt, and all C scratch registers are available for use. Note 329; that IRQ interrupts are still disabled upon return from the context 330; save function. */ 331; 332; /* Application ISR dispatch call goes here! */ 333; 334; /* Jump to context restore to restore system context. */ 335 B _tx_thread_context_restore 336 337 3387.2.3 Nested IRQ Support 339 340By default, nested IRQ interrupt support is not enabled. To enable nested 341IRQ support, the entire library should be built with TX_ENABLE_IRQ_NESTING 342defined. With this defined, two new IRQ interrupt management services are 343available, namely _tx_thread_irq_nesting_start and _tx_thread_irq_nesting_end. 344These function should be called between the IRQ context save and restore 345calls. 346 347Execution between the calls to _tx_thread_irq_nesting_start and 348_tx_thread_irq_nesting_end is enabled for IRQ nesting. This is achieved 349by switching from IRQ mode to SYS mode and enabling IRQ interrupts. 350The SYS mode stack is used during the SYS mode operation, which was 351setup in tx_initialize_low_level.s. When nested IRQ interrupts are no 352longer required, calling the _tx_thread_irq_nesting_end service disables 353nesting by disabling IRQ interrupts and switching back to IRQ mode in 354preparation for the IRQ context restore service. 355 356The following is an example of enabling IRQ nested interrupts in a standard 357IRQ handler: 358 359 RSEG .text:CODE:NOROOT(2) 360 PUBLIC __tx_irq_handler 361 RSEG .text:CODE:NOROOT(2) 362 PUBLIC __tx_irq_processing_return 363__tx_irq_handler 364; 365; /* Jump to context save to save system context. */ 366 B _tx_thread_context_save 367__tx_irq_processing_return 368; 369; /* At this point execution is still in the IRQ mode. The CPSR, point of 370; interrupt, and all C scratch registers are available for use. Note 371; that IRQ interrupts are still disabled upon return from the context 372; save function. */ 373; 374; /* Interrupt nesting is allowed after calling _tx_thread_irq_nesting_start 375; from IRQ mode with interrupts disabled. This routine switches to the 376; system mode and returns with IRQ interrupts enabled. 377; 378; NOTE: It is very important to ensure all IRQ interrupts are cleared 379; prior to enabling nested IRQ interrupts. */ 380; 381 BL _tx_thread_irq_nesting_start 382 383; /* Application ISR dispatch call goes here! */ 384; 385; /* If interrupt nesting was started earlier, the end of interrupt nesting 386; service must be called before returning to _tx_thread_context_restore. 387; This routine returns in processing in IRQ mode with interrupts disabled. */ 388; 389 BL _tx_thread_irq_nesting_end 390; 391; /* Jump to context restore to restore system context. */ 392 B _tx_thread_context_restore 393 394 3957.3 FIQ Interrupts 396 397By default, ARM9 FIQ interrupts are left alone by ThreadX. Of course, this 398means that the application is fully responsible for enabling the FIQ interrupt 399and saving/restoring any registers used in the FIQ ISR processing. To globally 400enable FIQ interrupts, the application should enable FIQ interrupts at the 401beginning of each thread or before any threads are created in tx_application_define. 402In addition, the application must ensure that no ThreadX service calls are made 403from default FIQ ISRs, which is located in tx_initialize_low_level.s. 404 405 4067.3.1 Managed FIQ Interrupts 407 408Full ThreadX management of FIQ interrupts is provided if the ThreadX sources 409are built with the TX_ENABLE_FIQ_SUPPORT defined. If the library is built 410this way, the FIQ interrupt handlers are very similar to the IRQ interrupt 411handlers defined previously. The following is default FIQ handler 412defined in tx_initialize_low_level.s: 413 414 415 RSEG .text:CODE:NOROOT(2) 416 PUBLIC __tx_fiq_handler 417 RSEG .text:CODE:NOROOT(2) 418 PUBLIC __tx_fiq_processing_return 419__tx_fiq_handler 420; 421; /* Jump to fiq context save to save system context. */ 422 B _tx_thread_fiq_context_save 423__tx_fiq_processing_return: 424; 425; /* At this point execution is still in the FIQ mode. The CPSR, point of 426; interrupt, and all C scratch registers are available for use. */ 427; 428; /* Application FIQ dispatch call goes here! */ 429; 430; /* Jump to fiq context restore to restore system context. */ 431 B _tx_thread_fiq_context_restore 432 433 434 4357.3.1.1 Nested FIQ Support 436 437By default, nested FIQ interrupt support is not enabled. To enable nested 438FIQ support, the entire library should be built with TX_ENABLE_FIQ_NESTING 439defined. With this defined, two new FIQ interrupt management services are 440available, namely _tx_thread_fiq_nesting_start and _tx_thread_fiq_nesting_end. 441These function should be called between the FIQ context save and restore 442calls. 443 444Execution between the calls to _tx_thread_fiq_nesting_start and 445_tx_thread_fiq_nesting_end is enabled for FIQ nesting. This is achieved 446by switching from FIQ mode to SYS mode and enabling FIQ interrupts. 447The SYS mode stack is used during the SYS mode operation, which was 448setup in tx_initialize_low_level.s. When nested FIQ interrupts are no 449longer required, calling the _tx_thread_fiq_nesting_end service disables 450nesting by disabling FIQ interrupts and switching back to FIQ mode in 451preparation for the FIQ context restore service. 452 453The following is an example of enabling FIQ nested interrupts in the 454typical FIQ handler: 455 456 457 RSEG .text:CODE:NOROOT(2) 458 PUBLIC __tx_fiq_handler 459 RSEG .text:CODE:NOROOT(2) 460 PUBLIC __tx_fiq_processing_return 461__tx_fiq_handler 462; 463; /* Jump to fiq context save to save system context. */ 464 B _tx_thread_fiq_context_save 465__tx_fiq_processing_return: 466; 467; /* At this point execution is still in the FIQ mode. The CPSR, point of 468; interrupt, and all C scratch registers are available for use. */ 469; 470; /* Enable nested FIQ interrupts. NOTE: Since this service returns 471; with FIQ interrupts enabled, all FIQ interrupt sources must be 472; cleared prior to calling this service. */ 473 BL _tx_thread_fiq_nesting_start 474; 475; /* Application FIQ dispatch call goes here! */ 476; 477; /* Disable nested FIQ interrupts. The mode is switched back to 478; FIQ mode and FIQ interrupts are disable upon return. */ 479 BL _tx_thread_fiq_nesting_end 480; 481; /* Jump to fiq context restore to restore system context. */ 482 B _tx_thread_fiq_context_restore 483 484 4858. ThreadX Timer Interrupt 486 487ThreadX requires a periodic interrupt source to manage all time-slicing, 488thread sleeps, timeouts, and application timers. Without such a timer 489interrupt source, these services are not functional. However, all other 490ThreadX services are operational without a periodic timer source. 491 492To add the timer interrupt processing, simply make a call to _tx_timer_interrupt 493in the IRQ processing. 494 495 4969. Thumb/ARM9 Mixed Mode 497 498By default, ThreadX is setup for running in ARM9 32-bit mode. This is 499also true for the demonstration system. It is possible to build any 500ThreadX file and/or the application in Thumb mode. The only exception 501to this is the file tx_thread_shell_entry.c. This file must always be 502built in 32-bit mode. In addition, if any Thumb code is used the entire 503ThreadX assembly source should be built with TX_THUMB defined. 504 505 50610. IAR Thread-safe Library Support 507 508Thread-safe support for the IAR tools is easily enabled by building the ThreadX library 509and the application with TX_ENABLE_IAR_LIBRARY_SUPPORT. Also, the linker control file 510should have the following line added (if not already in place): 511 512initialize by copy with packing = none { section __DLIB_PERTHREAD }; // Required in a multi-threaded application 513 514 51511. Revision History 516 517For generic code revision information, please refer to the readme_threadx_generic.txt 518file, which is included in your distribution. The following details the revision 519information associated with this specific port of ThreadX: 520 52104-02-2021 Release 6.1.6 changes: 522 tx_port.h Updated macro definition 523 52409-30-2020 Initial ThreadX version 6.1 for ARM9 using IAR's ARM tools. 525 526 527Copyright(c) 1996-2020 Microsoft Corporation 528 529 530https://azure.com/rtos 531 532