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#ifdef TX_ENABLE_FIQ_SUPPORT 26ENABLE_INTS = 0xC0 @ IRQ & FIQ Interrupts enabled mask 27#else 28ENABLE_INTS = 0x80 @ IRQ Interrupts enabled mask 29#endif 30@ 31@ 32 .global _tx_thread_execute_ptr 33 .global _tx_thread_current_ptr 34 .global _tx_timer_time_slice 35 .global _tx_execution_thread_enter 36@ 37@ 38@/* Define the 16-bit Thumb mode veneer for _tx_thread_schedule for 39@ applications calling this function from to 16-bit Thumb mode. */ 40@ 41 .text 42 .align 2 43 .global $_tx_thread_schedule 44 .type $_tx_thread_schedule,function 45$_tx_thread_schedule: 46 .thumb 47 BX pc @ Switch to 32-bit mode 48 NOP @ 49 .arm 50 STMFD sp!, {lr} @ Save return address 51 BL _tx_thread_schedule @ Call _tx_thread_schedule function 52 LDMFD sp!, {lr} @ Recover saved return address 53 BX lr @ Return to 16-bit caller 54@ 55@ 56 .text 57 .align 2 58@/**************************************************************************/ 59@/* */ 60@/* FUNCTION RELEASE */ 61@/* */ 62@/* _tx_thread_schedule ARM9/GNU */ 63@/* 6.2.1 */ 64@/* AUTHOR */ 65@/* */ 66@/* William E. Lamie, Microsoft Corporation */ 67@/* */ 68@/* DESCRIPTION */ 69@/* */ 70@/* This function waits for a thread control block pointer to appear in */ 71@/* the _tx_thread_execute_ptr variable. Once a thread pointer appears */ 72@/* in the variable, the corresponding thread is resumed. */ 73@/* */ 74@/* INPUT */ 75@/* */ 76@/* None */ 77@/* */ 78@/* OUTPUT */ 79@/* */ 80@/* None */ 81@/* */ 82@/* CALLS */ 83@/* */ 84@/* None */ 85@/* */ 86@/* CALLED BY */ 87@/* */ 88@/* _tx_initialize_kernel_enter ThreadX entry function */ 89@/* _tx_thread_system_return Return to system from thread */ 90@/* _tx_thread_context_restore Restore thread's context */ 91@/* */ 92@/* RELEASE HISTORY */ 93@/* */ 94@/* DATE NAME DESCRIPTION */ 95@/* */ 96@/* 09-30-2020 William E. Lamie Initial Version 6.1 */ 97@/* 03-08-2023 Cindy Deng Modified comment(s), added */ 98@/* #include tx_user.h, */ 99@/* resulting in version 6.2.1 */ 100@/* */ 101@/**************************************************************************/ 102@VOID _tx_thread_schedule(VOID) 103@{ 104 .global _tx_thread_schedule 105 .type _tx_thread_schedule,function 106_tx_thread_schedule: 107@ 108@ /* Enable interrupts. */ 109@ 110 MRS r2, CPSR @ Pickup CPSR 111 BIC r0, r2, #ENABLE_INTS @ Clear the disable bit(s) 112 MSR CPSR_cxsf, r0 @ Enable interrupts 113@ 114@ /* Wait for a thread to execute. */ 115@ do 116@ { 117 LDR r1, =_tx_thread_execute_ptr @ Address of thread execute ptr 118@ 119__tx_thread_schedule_loop: 120@ 121 LDR r0, [r1] @ Pickup next thread to execute 122 CMP r0, #0 @ Is it NULL? 123 BEQ __tx_thread_schedule_loop @ If so, keep looking for a thread 124@ 125@ } 126@ while(_tx_thread_execute_ptr == TX_NULL); 127@ 128@ /* Yes! We have a thread to execute. Lockout interrupts and 129@ transfer control to it. */ 130@ 131 MSR CPSR_cxsf, r2 @ Disable interrupts 132@ 133@ /* Setup the current thread pointer. */ 134@ _tx_thread_current_ptr = _tx_thread_execute_ptr; 135@ 136 LDR r1, =_tx_thread_current_ptr @ Pickup address of current thread 137 STR r0, [r1] @ Setup current thread pointer 138@ 139@ /* Increment the run count for this thread. */ 140@ _tx_thread_current_ptr -> tx_thread_run_count++; 141@ 142 LDR r2, [r0, #4] @ Pickup run counter 143 LDR r3, [r0, #24] @ Pickup time-slice for this thread 144 ADD r2, r2, #1 @ Increment thread run-counter 145 STR r2, [r0, #4] @ Store the new run counter 146@ 147@ /* Setup time-slice, if present. */ 148@ _tx_timer_time_slice = _tx_thread_current_ptr -> tx_thread_time_slice; 149@ 150 LDR r2, =_tx_timer_time_slice @ Pickup address of time-slice 151 @ variable 152 LDR sp, [r0, #8] @ Switch stack pointers 153 STR r3, [r2] @ Setup time-slice 154@ 155@ /* Switch to the thread's stack. */ 156@ sp = _tx_thread_execute_ptr -> tx_thread_stack_ptr; 157@ 158#ifdef TX_ENABLE_EXECUTION_CHANGE_NOTIFY 159@ 160@ /* Call the thread entry function to indicate the thread is executing. */ 161@ 162 BL _tx_execution_thread_enter @ Call the thread execution enter function 163#endif 164@ 165@ /* Determine if an interrupt frame or a synchronous task suspension frame 166@ is present. */ 167@ 168 LDMIA sp!, {r0, r1} @ Pickup the stack type and saved CPSR 169 CMP r0, #0 @ Check for synchronous context switch 170 MSRNE SPSR_cxsf, r1 @ Setup SPSR for return 171 LDMNEIA sp!, {r0-r12, lr, pc}^ @ Return to point of thread interrupt 172 LDMIA sp!, {r4-r11, lr} @ Return to thread synchronously 173 MSR CPSR_cxsf, r1 @ Recover CPSR 174#ifdef __THUMB_INTERWORK 175 BX lr @ Return to caller 176#else 177 MOV pc, lr @ Return to caller 178#endif 179@ 180@} 181@ 182 183