|  | /* | 
|  | * Copyright (c) 2014-2015 Wind River Systems, Inc. | 
|  | * Copyright (c) 2018 Synopsys Inc, Inc. | 
|  | * | 
|  | * SPDX-License-Identifier: Apache-2.0 | 
|  | */ | 
|  | #include <zephyr/init.h> | 
|  | #include <zephyr/drivers/timer/system_timer.h> | 
|  | #include <zephyr/sys_clock.h> | 
|  | #include <zephyr/spinlock.h> | 
|  | #include <zephyr/arch/arc/v2/aux_regs.h> | 
|  | #include <zephyr/irq.h> | 
|  | /* | 
|  | * note: This implementation assumes Timer0 is present. Be sure | 
|  | * to build the ARC CPU with Timer0. | 
|  | * | 
|  | * If secureshield is present and secure firmware is configured, | 
|  | * use secure Timer 0 | 
|  | */ | 
|  |  | 
|  | #ifdef CONFIG_ARC_SECURE_FIRMWARE | 
|  |  | 
|  | #undef _ARC_V2_TMR0_COUNT | 
|  | #undef _ARC_V2_TMR0_CONTROL | 
|  | #undef _ARC_V2_TMR0_LIMIT | 
|  |  | 
|  | #define _ARC_V2_TMR0_COUNT _ARC_V2_S_TMR0_COUNT | 
|  | #define _ARC_V2_TMR0_CONTROL _ARC_V2_S_TMR0_CONTROL | 
|  | #define _ARC_V2_TMR0_LIMIT _ARC_V2_S_TMR0_LIMIT | 
|  | #define IRQ_TIMER0 DT_IRQN(DT_NODELABEL(sectimer0)) | 
|  |  | 
|  | #else | 
|  | #define IRQ_TIMER0 DT_IRQN(DT_NODELABEL(timer0)) | 
|  | #endif | 
|  |  | 
|  | #define _ARC_V2_TMR_CTRL_IE 0x1 /* interrupt enable */ | 
|  | #define _ARC_V2_TMR_CTRL_NH 0x2 /* count only while not halted */ | 
|  | #define _ARC_V2_TMR_CTRL_W  0x4 /* watchdog mode enable */ | 
|  | #define _ARC_V2_TMR_CTRL_IP 0x8 /* interrupt pending flag */ | 
|  |  | 
|  | /* Minimum cycles in the future to try to program. */ | 
|  | #define MIN_DELAY 1024 | 
|  | /* arc timer has 32 bit, here use 31 bit to avoid the possible | 
|  | * overflow,e.g, 0xffffffff + any value will cause overflow | 
|  | */ | 
|  | #define COUNTER_MAX 0x7fffffff | 
|  | #define TIMER_STOPPED 0x0 | 
|  | #define CYC_PER_TICK (sys_clock_hw_cycles_per_sec()	\ | 
|  | / CONFIG_SYS_CLOCK_TICKS_PER_SEC) | 
|  |  | 
|  | #define MAX_TICKS ((COUNTER_MAX / CYC_PER_TICK) - 1) | 
|  | #define MAX_CYCLES (MAX_TICKS * CYC_PER_TICK) | 
|  |  | 
|  | #define TICKLESS (IS_ENABLED(CONFIG_TICKLESS_KERNEL)) | 
|  |  | 
|  | #define SMP_TIMER_DRIVER (CONFIG_SMP && CONFIG_MP_MAX_NUM_CPUS > 1) | 
|  |  | 
|  | #if defined(CONFIG_TEST) | 
|  | const int32_t z_sys_timer_irq_for_test = IRQ_TIMER0; | 
|  | #endif | 
|  | static struct k_spinlock lock; | 
|  |  | 
|  |  | 
|  | #if SMP_TIMER_DRIVER | 
|  | volatile static uint64_t last_time; | 
|  | volatile static uint64_t start_time; | 
|  |  | 
|  | #else | 
|  | static uint32_t last_load; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * This local variable holds the amount of timer cycles elapsed | 
|  | * and it is updated in timer_int_handler and sys_clock_set_timeout(). | 
|  | * | 
|  | * Note: | 
|  | *  At an arbitrary point in time the "current" value of the | 
|  | *  HW timer is calculated as: | 
|  | * | 
|  | * t = cycle_counter + elapsed(); | 
|  | */ | 
|  | static uint32_t cycle_count; | 
|  |  | 
|  | /* | 
|  | * This local variable holds the amount of elapsed HW cycles | 
|  | * that have been announced to the kernel. | 
|  | */ | 
|  | static uint32_t announced_cycles; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * This local variable holds the amount of elapsed HW cycles due to | 
|  | * timer wraps ('overflows') and is used in the calculation | 
|  | * in elapsed() function, as well as in the updates to cycle_count. | 
|  | * | 
|  | * Note: | 
|  | * Each time cycle_count is updated with the value from overflow_cycles, | 
|  | * the overflow_cycles must be reset to zero. | 
|  | */ | 
|  | static volatile uint32_t overflow_cycles; | 
|  | #endif | 
|  |  | 
|  | /** | 
|  | * @brief Get contents of Timer0 count register | 
|  | * | 
|  | * @return Current Timer0 count | 
|  | */ | 
|  | static ALWAYS_INLINE uint32_t timer0_count_register_get(void) | 
|  | { | 
|  | return z_arc_v2_aux_reg_read(_ARC_V2_TMR0_COUNT); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @brief Set Timer0 count register to the specified value | 
|  | */ | 
|  | static ALWAYS_INLINE void timer0_count_register_set(uint32_t value) | 
|  | { | 
|  | z_arc_v2_aux_reg_write(_ARC_V2_TMR0_COUNT, value); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @brief Get contents of Timer0 control register | 
|  | * | 
|  | * @return Contents of Timer0 control register. | 
|  | */ | 
|  | static ALWAYS_INLINE uint32_t timer0_control_register_get(void) | 
|  | { | 
|  | return z_arc_v2_aux_reg_read(_ARC_V2_TMR0_CONTROL); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @brief Set Timer0 control register to the specified value | 
|  | */ | 
|  | static ALWAYS_INLINE void timer0_control_register_set(uint32_t value) | 
|  | { | 
|  | z_arc_v2_aux_reg_write(_ARC_V2_TMR0_CONTROL, value); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @brief Get contents of Timer0 limit register | 
|  | * | 
|  | * @return Contents of Timer0 limit register. | 
|  | */ | 
|  | static ALWAYS_INLINE uint32_t timer0_limit_register_get(void) | 
|  | { | 
|  | return z_arc_v2_aux_reg_read(_ARC_V2_TMR0_LIMIT); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @brief Set Timer0 limit register to the specified value | 
|  | */ | 
|  | static ALWAYS_INLINE void timer0_limit_register_set(uint32_t count) | 
|  | { | 
|  | z_arc_v2_aux_reg_write(_ARC_V2_TMR0_LIMIT, count); | 
|  | } | 
|  |  | 
|  | #if !SMP_TIMER_DRIVER | 
|  | /* This internal function calculates the amount of HW cycles that have | 
|  | * elapsed since the last time the absolute HW cycles counter has been | 
|  | * updated. 'cycle_count' may be updated either by the ISR, or | 
|  | * in sys_clock_set_timeout(). | 
|  | * | 
|  | * Additionally, the function updates the 'overflow_cycles' counter, that | 
|  | * holds the amount of elapsed HW cycles due to (possibly) multiple | 
|  | * timer wraps (overflows). | 
|  | * | 
|  | * Prerequisites: | 
|  | * - reprogramming of LIMIT must be clearing the COUNT | 
|  | * - ISR must be clearing the 'overflow_cycles' counter. | 
|  | * - no more than one counter-wrap has occurred between | 
|  | *     - the timer reset or the last time the function was called | 
|  | *     - and until the current call of the function is completed. | 
|  | * - the function is invoked with interrupts disabled. | 
|  | */ | 
|  | static uint32_t elapsed(void) | 
|  | { | 
|  | uint32_t val, ctrl; | 
|  |  | 
|  | do { | 
|  | val =  timer0_count_register_get(); | 
|  | ctrl = timer0_control_register_get(); | 
|  | } while (timer0_count_register_get() < val); | 
|  |  | 
|  | if (ctrl & _ARC_V2_TMR_CTRL_IP) { | 
|  | overflow_cycles += last_load; | 
|  | /* clear the IP bit of the control register */ | 
|  | timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | | 
|  | _ARC_V2_TMR_CTRL_IE); | 
|  | /* use sw triggered irq to remember the timer irq request | 
|  | * which may be cleared by the above operation. when elapsed () | 
|  | * is called in timer_int_handler, no need to do this. | 
|  | */ | 
|  | if (!z_arc_v2_irq_unit_is_in_isr() || | 
|  | z_arc_v2_aux_reg_read(_ARC_V2_ICAUSE) != IRQ_TIMER0) { | 
|  | z_arc_v2_aux_reg_write(_ARC_V2_AUX_IRQ_HINT, | 
|  | IRQ_TIMER0); | 
|  | } | 
|  | } | 
|  |  | 
|  | return val + overflow_cycles; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /** | 
|  | * @brief System clock periodic tick handler | 
|  | * | 
|  | * This routine handles the system clock tick interrupt. It always | 
|  | * announces one tick when TICKLESS is not enabled, or multiple ticks | 
|  | * when TICKLESS is enabled. | 
|  | */ | 
|  | static void timer_int_handler(const void *unused) | 
|  | { | 
|  | ARG_UNUSED(unused); | 
|  | uint32_t dticks; | 
|  |  | 
|  | #if defined(CONFIG_SMP) && CONFIG_MP_MAX_NUM_CPUS > 1 | 
|  | uint64_t curr_time; | 
|  | k_spinlock_key_t key; | 
|  |  | 
|  | /* clear the IP bit of the control register */ | 
|  | timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | | 
|  | _ARC_V2_TMR_CTRL_IE); | 
|  | key = k_spin_lock(&lock); | 
|  | /* gfrc is the wall clock */ | 
|  | curr_time = z_arc_connect_gfrc_read(); | 
|  |  | 
|  | dticks = (curr_time - last_time) / CYC_PER_TICK; | 
|  | /* last_time should be aligned to ticks */ | 
|  | last_time += dticks * CYC_PER_TICK; | 
|  |  | 
|  | k_spin_unlock(&lock, key); | 
|  |  | 
|  | sys_clock_announce(dticks); | 
|  | #else | 
|  | /* timer_int_handler may be triggered by timer irq or | 
|  | * software helper irq | 
|  | */ | 
|  |  | 
|  | /* irq with higher priority may call sys_clock_set_timeout | 
|  | * so need a lock here | 
|  | */ | 
|  | uint32_t key; | 
|  |  | 
|  | key = arch_irq_lock(); | 
|  |  | 
|  | elapsed(); | 
|  | cycle_count += overflow_cycles; | 
|  | overflow_cycles = 0; | 
|  |  | 
|  | arch_irq_unlock(key); | 
|  |  | 
|  | dticks = (cycle_count - announced_cycles) / CYC_PER_TICK; | 
|  | announced_cycles += dticks * CYC_PER_TICK; | 
|  | sys_clock_announce(TICKLESS ? dticks : 1); | 
|  | #endif | 
|  |  | 
|  | } | 
|  |  | 
|  | void sys_clock_set_timeout(int32_t ticks, bool idle) | 
|  | { | 
|  | /* If the kernel allows us to miss tick announcements in idle, | 
|  | * then shut off the counter. (Note: we can assume if idle==true | 
|  | * that interrupts are already disabled) | 
|  | */ | 
|  | #if SMP_TIMER_DRIVER | 
|  | /* as 64-bits GFRC is used as wall clock, it's ok to ignore idle | 
|  | * systick will not be missed. | 
|  | * However for single core using 32-bits arc timer, idle cannot | 
|  | * be ignored, as 32-bits timer will overflow in a not-long time. | 
|  | */ | 
|  | if (IS_ENABLED(CONFIG_TICKLESS_KERNEL) && ticks == K_TICKS_FOREVER) { | 
|  | timer0_control_register_set(0); | 
|  | timer0_count_register_set(0); | 
|  | timer0_limit_register_set(0); | 
|  | return; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_TICKLESS_KERNEL) | 
|  | uint32_t delay; | 
|  | uint32_t key; | 
|  |  | 
|  | ticks = MIN(MAX_TICKS, ticks); | 
|  |  | 
|  | /* Desired delay in the future | 
|  | * use MIN_DEALY here can trigger the timer | 
|  | * irq more soon, no need to go to CYC_PER_TICK | 
|  | * later. | 
|  | */ | 
|  | delay = MAX(ticks * CYC_PER_TICK, MIN_DELAY); | 
|  |  | 
|  | key = arch_irq_lock(); | 
|  |  | 
|  | timer0_limit_register_set(delay - 1); | 
|  | timer0_count_register_set(0); | 
|  | timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | | 
|  | _ARC_V2_TMR_CTRL_IE); | 
|  |  | 
|  | arch_irq_unlock(key); | 
|  | #endif | 
|  | #else | 
|  | if (IS_ENABLED(CONFIG_TICKLESS_KERNEL) && idle && ticks == K_TICKS_FOREVER) { | 
|  | timer0_control_register_set(0); | 
|  | timer0_count_register_set(0); | 
|  | timer0_limit_register_set(0); | 
|  | last_load = TIMER_STOPPED; | 
|  | return; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_TICKLESS_KERNEL) | 
|  | uint32_t delay; | 
|  | uint32_t unannounced; | 
|  |  | 
|  | ticks = MIN(MAX_TICKS, (uint32_t)(MAX((int32_t)(ticks - 1), 0))); | 
|  |  | 
|  | k_spinlock_key_t key = k_spin_lock(&lock); | 
|  |  | 
|  |  | 
|  | cycle_count += elapsed(); | 
|  | /* clear counter early to avoid cycle loss as few as possible, | 
|  | * between cycle_count and clearing 0, few cycles are possible | 
|  | * to loss | 
|  | */ | 
|  | timer0_count_register_set(0); | 
|  | overflow_cycles = 0U; | 
|  |  | 
|  |  | 
|  | /* normal case */ | 
|  | unannounced = cycle_count - announced_cycles; | 
|  |  | 
|  | if ((int32_t)unannounced < 0) { | 
|  | /* We haven't announced for more than half the 32-bit | 
|  | * wrap duration, because new timeouts keep being set | 
|  | * before the existing one fires. Force an announce | 
|  | * to avoid loss of a wrap event, making sure the | 
|  | * delay is at least the minimum delay possible. | 
|  | */ | 
|  | last_load = MIN_DELAY; | 
|  | } else { | 
|  | /* Desired delay in the future */ | 
|  | delay = ticks * CYC_PER_TICK; | 
|  |  | 
|  | /* Round delay up to next tick boundary */ | 
|  | delay += unannounced; | 
|  | delay = DIV_ROUND_UP(delay, CYC_PER_TICK) * CYC_PER_TICK; | 
|  |  | 
|  | delay -= unannounced; | 
|  | delay = MAX(delay, MIN_DELAY); | 
|  |  | 
|  | last_load = MIN(delay, MAX_CYCLES); | 
|  | } | 
|  |  | 
|  | timer0_limit_register_set(last_load - 1); | 
|  | timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | _ARC_V2_TMR_CTRL_IE); | 
|  |  | 
|  | k_spin_unlock(&lock, key); | 
|  | #endif | 
|  | #endif | 
|  | } | 
|  |  | 
|  | uint32_t sys_clock_elapsed(void) | 
|  | { | 
|  | if (!TICKLESS) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | uint32_t cyc; | 
|  | k_spinlock_key_t key = k_spin_lock(&lock); | 
|  |  | 
|  | #if SMP_TIMER_DRIVER | 
|  | cyc = (z_arc_connect_gfrc_read() - last_time); | 
|  | #else | 
|  | cyc =  elapsed() + cycle_count - announced_cycles; | 
|  | #endif | 
|  |  | 
|  | k_spin_unlock(&lock, key); | 
|  |  | 
|  | return cyc / CYC_PER_TICK; | 
|  | } | 
|  |  | 
|  | uint32_t sys_clock_cycle_get_32(void) | 
|  | { | 
|  | #if SMP_TIMER_DRIVER | 
|  | return z_arc_connect_gfrc_read() - start_time; | 
|  | #else | 
|  | k_spinlock_key_t key = k_spin_lock(&lock); | 
|  | uint32_t ret = elapsed() + cycle_count; | 
|  |  | 
|  | k_spin_unlock(&lock, key); | 
|  | return ret; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | #if SMP_TIMER_DRIVER | 
|  | void smp_timer_init(void) | 
|  | { | 
|  | /* set the initial status of timer0 of each slave core | 
|  | */ | 
|  | timer0_control_register_set(0); | 
|  | timer0_count_register_set(0); | 
|  | timer0_limit_register_set(0); | 
|  |  | 
|  | z_irq_priority_set(IRQ_TIMER0, CONFIG_ARCV2_TIMER_IRQ_PRIORITY, 0); | 
|  | irq_enable(IRQ_TIMER0); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /** | 
|  | * | 
|  | * @brief Initialize and enable the system clock | 
|  | * | 
|  | * This routine is used to program the ARCv2 timer to deliver interrupts at the | 
|  | * rate specified via the CYC_PER_TICK. | 
|  | * | 
|  | * @return 0 | 
|  | */ | 
|  | static int sys_clock_driver_init(void) | 
|  | { | 
|  |  | 
|  | /* ensure that the timer will not generate interrupts */ | 
|  | timer0_control_register_set(0); | 
|  |  | 
|  | #if SMP_TIMER_DRIVER | 
|  | IRQ_CONNECT(IRQ_TIMER0, CONFIG_ARCV2_TIMER_IRQ_PRIORITY, | 
|  | timer_int_handler, NULL, 0); | 
|  |  | 
|  | timer0_limit_register_set(CYC_PER_TICK - 1); | 
|  | last_time = z_arc_connect_gfrc_read(); | 
|  | start_time = last_time; | 
|  | #else | 
|  | last_load = CYC_PER_TICK; | 
|  | overflow_cycles = 0; | 
|  | announced_cycles = 0; | 
|  |  | 
|  | IRQ_CONNECT(IRQ_TIMER0, CONFIG_ARCV2_TIMER_IRQ_PRIORITY, | 
|  | timer_int_handler, NULL, 0); | 
|  |  | 
|  | timer0_limit_register_set(last_load - 1); | 
|  | #endif | 
|  | timer0_count_register_set(0); | 
|  | timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | _ARC_V2_TMR_CTRL_IE); | 
|  |  | 
|  | /* everything has been configured: safe to enable the interrupt */ | 
|  |  | 
|  | irq_enable(IRQ_TIMER0); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | SYS_INIT(sys_clock_driver_init, PRE_KERNEL_2, | 
|  | CONFIG_SYSTEM_CLOCK_INIT_PRIORITY); |