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;/** ThreadX Component */ 15;/** */ 16;/** Timer */ 17;/** */ 18;/**************************************************************************/ 19;/**************************************************************************/ 20#ifdef TX_INCLUDE_USER_DEFINE_FILE 21#include "tx_user.h" 22#endif 23 24;/**************************************************************************/ 25;/* */ 26;/* FUNCTION RELEASE */ 27;/* */ 28;/* _tx_timer_interrupt ARCv2_EM/MetaWare */ 29;/* 6.2.1 */ 30;/* AUTHOR */ 31;/* */ 32;/* William E. Lamie, Microsoft Corporation */ 33;/* */ 34;/* DESCRIPTION */ 35;/* */ 36;/* This function processes the hardware timer interrupt. This */ 37;/* processing includes incrementing the system clock and checking for */ 38;/* time slice and/or timer expiration. If either is found, the */ 39;/* interrupt context save/restore functions are called along with the */ 40;/* expiration functions. */ 41;/* */ 42;/* INPUT */ 43;/* */ 44;/* None */ 45;/* */ 46;/* OUTPUT */ 47;/* */ 48;/* None */ 49;/* */ 50;/* CALLS */ 51;/* */ 52;/* _tx_timer_expiration_process Process timer expiration */ 53;/* _tx_thread_time_slice Time slice interrupted thread */ 54;/* _tx_thread_context_save Save interrupt context */ 55;/* _tx_thread_context_restore Restore interrupt context */ 56;/* */ 57;/* CALLED BY */ 58;/* */ 59;/* interrupt vector */ 60;/* */ 61;/* RELEASE HISTORY */ 62;/* */ 63;/* DATE NAME DESCRIPTION */ 64;/* */ 65;/* 09-30-2020 William E. Lamie Initial Version 6.1 */ 66;/* 12-31-2020 Scott Larson Modified comment(s), remove */ 67;/* unneeded load of */ 68;/* _tx_thread_preempt_disable, */ 69;/* resulting in version 6.1.3 */ 70;/* 10-15-2021 Andres Mlinar Modified comment(s), and */ 71;/* fixed possible race */ 72;/* condition on preemption */ 73;/* resulting in version 6.1.9 */ 74;/* 03-08-2023 Cindy Deng Modified comment(s), added */ 75;/* #include tx_user.h, */ 76;/* resulting in version 6.2.1 */ 77;/* */ 78;/**************************************************************************/ 79;VOID _tx_timer_interrupt(VOID) 80;{ 81 .global _tx_timer_interrupt 82 .type _tx_timer_interrupt, @function 83_tx_timer_interrupt: 84; 85; /* Upon entry to this routine, it is assumed the interrupt stack frame has 86; already been allocated and registers r0, r1, and r2 have already been saved 87; at offsets 0, 4, and 8 respectively. */ 88; 89; /* Increment the system clock. */ 90; _tx_timer_system_clock++; 91; 92 clri ; Lockout interrupts 93 ld r0, [gp,_tx_timer_system_clock@sda] ; Pickup current system clock 94 ld r2, [gp, _tx_timer_time_slice@sda] ; Pickup current time-slice 95 add r0, r0, 1 ; Increment the system clock 96 st r0, [gp,_tx_timer_system_clock@sda] ; Store system clock back in memory 97 98; /* Test for time-slice expiration. */ 99; if (_tx_timer_time_slice) 100; { 101; 102 mov r1, 0 ; Clear expiration flag 103 breq r2, 0, __tx_timer_no_time_slice ; If zero, no time-slice is active 104; 105; /* Decrement the time_slice. */ 106; _tx_timer_time_slice--; 107; 108 sub r2, r2, 1 ; Decrement time-slice 109 st r2, [gp, _tx_timer_time_slice@sda] ; Store new time-slice value 110; 111; /* Check for expiration. */ 112; if (__tx_timer_time_slice == 0) 113; 114 brne r2, 0, __tx_timer_no_time_slice ; If non-zero, skip over expiration 115; 116; /* Set the time-slice expired flag. */ 117; _tx_timer_expired_time_slice = TX_TRUE; 118; 119 mov r1, 1 ; Set register flag 120 st r1, [gp, _tx_timer_expired_time_slice@sda] ; Set the time-slice expired flag 121 122; 123; } 124; 125__tx_timer_no_time_slice: 126; 127; /* Test for timer expiration. */ 128; if (*_tx_timer_current_ptr) 129; { 130; 131 ld r0, [gp, _tx_timer_current_ptr@sda] ; Pickup current timer pointer 132 ld r2, [r0, 0] ; Pickup examine actual list entry 133 breq r2, 0, __tx_timer_no_timer ; 134 ; If NULL, no timer has expired, just move to the next entry 135; 136; /* Set expiration flag. */ 137; _tx_timer_expired = TX_TRUE; 138; 139 mov r1, 1 ; Build expiration value 140 b.d __tx_timer_done ; Skip moving the timer pointer 141 st r1, [gp, _tx_timer_expired@sda] ; Set the expired value 142; 143; } 144; else 145; { 146__tx_timer_no_timer: 147; 148; /* No timer expired, increment the timer pointer. */ 149; _tx_timer_current_ptr++; 150; 151 ld r2, [gp, _tx_timer_list_end@sda] ; Pickup end of list 152 add r0, r0, 4 ; Move to next timer entry 153; 154; /* Check for wrap-around. */ 155; if (_tx_timer_current_ptr == _tx_timer_list_end) 156; 157 st r0, [gp, _tx_timer_current_ptr@sda] ; Store the current timer 158 brne r0, r2, __tx_timer_skip_wrap ; If not equal, don't wrap the list 159; 160; /* Wrap to beginning of list. */ 161; _tx_timer_current_ptr = _tx_timer_list_start; 162; 163 ld r2, [gp, _tx_timer_list_start@sda] ; Pickup start of timer list 164 st r2, [gp, _tx_timer_current_ptr@sda] ; Set current timer to the start 165; 166__tx_timer_skip_wrap: 167; 168; } 169; 170__tx_timer_done: 171; 172; 173; /* See if anything has expired. */ 174; if ((_tx_timer_expired_time_slice) || (_tx_timer_expired)) 175; { 176; 177 ld r0, [gp, _tx_thread_current_ptr@sda] 178 ld r2, [gp, _tx_thread_execute_ptr@sda] 179 brne r0, r2, __tx_something_expired 180; 181 breq r1, 0, __tx_timer_nothing_expired ; If 0, nothing has expired 182; 183__tx_something_expired: 184; 185 ld r0, [sp, 0] ; Recover r0 186 ld r1, [sp, 4] ; Recover r1 187 ld r2, [sp, 8] ; Recover r2 188 st blink, [sp, 16] ; Save blink 189 bl _tx_thread_context_save ; Save interrupted context 190; 191; /* Did a timer expire? */ 192; if (_tx_timer_expired) 193; { 194; 195 ld r2, [gp, _tx_timer_expired@sda] ; Pickup timer expired flag 196 breq r2, 0, __tx_timer_dont_activate ; If not set, skip expiration processing 197; 198; /* Process the timer expiration. */ 199; /* _tx_timer_expiration_process(); */ 200 bl.d _tx_timer_expiration_process ; Call the timer expiration handling routine 201 sub sp, sp, 16 ; ..allocating some space on the stack 202 add sp, sp, 16 ; Recover the stack space 203; 204; } 205__tx_timer_dont_activate: 206; 207; /* Did time slice expire? */ 208; if (_tx_timer_expired_time_slice) 209; { 210; 211 ld r2, [gp, _tx_timer_expired_time_slice@sda] ; Pickup expired time-slice flag 212 breq r2, 0, __tx_timer_not_ts_expiration ; If not set, skip time-slice 213; 214; /* Time slice interrupted thread. */ 215; /* _tx_thread_time_slice(); */ 216 217 bl.d _tx_thread_time_slice ; Call time-slice processing 218 sub sp, sp, 16 ; ..allocating some stack space 219 add sp, sp, 16 ; Recover stack space 220; 221; } 222; 223__tx_timer_not_ts_expiration: 224; 225 st 0, [gp, _tx_timer_expired_time_slice@sda] 226 b _tx_thread_context_restore ; Go restore interrupt context.. 227 ; ..clearing time-slice expired flag 228 ; Note that we don't return from 229 ; this function. 230; 231; } 232; 233__tx_timer_nothing_expired: 234; 235 ld r0, [sp, 0] ; Recover r0 236 ld r1, [sp, 4] ; Recover r1 237 ld r2, [sp, 8] ; Recover r2 238 add sp, sp, 160 ; Recover interrupt stack frame 239 rtie ; Return to point of interrupt 240; 241;} 242 .end 243 244