/* * Copyright (c) 2014-2015 Wind River Systems, Inc. * Copyright (c) 2018 Synopsys Inc, Inc. * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include /* * note: This implementation assumes Timer0 is present. Be sure * to build the ARC CPU with Timer0. */ #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 512 #define COUNTER_MAX 0xffffffff #define TIMER_STOPPED 0x0 #define CYC_PER_TICK (CONFIG_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)) static struct k_spinlock lock; static u32_t last_load; static u32_t cycle_count; /** * * @brief Get contents of Timer0 count register * * @return Current Timer0 count */ static ALWAYS_INLINE u32_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 * * @return N/A */ static ALWAYS_INLINE void timer0_count_register_set(u32_t value) { z_arc_v2_aux_reg_write(_ARC_V2_TMR0_COUNT, value); } /** * * @brief Get contents of Timer0 control register * * @return N/A */ static ALWAYS_INLINE u32_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 * * @return N/A */ static ALWAYS_INLINE void timer0_control_register_set(u32_t value) { z_arc_v2_aux_reg_write(_ARC_V2_TMR0_CONTROL, value); } /** * * @brief Get contents of Timer0 limit register * * @return N/A */ static ALWAYS_INLINE u32_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 * * @return N/A */ static ALWAYS_INLINE void timer0_limit_register_set(u32_t count) { z_arc_v2_aux_reg_write(_ARC_V2_TMR0_LIMIT, count); } static u32_t elapsed(void) { u32_t val, ov, ctrl; do { val = timer0_count_register_get(); ctrl = timer0_control_register_get(); } while (timer0_count_register_get() < val); ov = (ctrl & _ARC_V2_TMR_CTRL_IP) ? last_load : 0; return val + ov; } /** * * @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. * * @return N/A */ static void timer_int_handler(void *unused) { ARG_UNUSED(unused); u32_t dticks; /* clear the interrupt by writing 0 to IP bit of the control register */ timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | _ARC_V2_TMR_CTRL_IE); cycle_count += last_load; dticks = last_load / CYC_PER_TICK; z_clock_announce(TICKLESS ? dticks : 1); } /** * * @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 */ int z_clock_driver_init(struct device *device) { ARG_UNUSED(device); /* ensure that the timer will not generate interrupts */ timer0_control_register_set(0); last_load = CYC_PER_TICK; IRQ_CONNECT(IRQ_TIMER0, CONFIG_ARCV2_TIMER_IRQ_PRIORITY, timer_int_handler, NULL, 0); timer0_limit_register_set(last_load - 1); #ifdef CONFIG_BOOT_TIME_MEASUREMENT cycle_count = timer0_count_register_get(); #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; } void z_clock_set_timeout(s32_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 (IS_ENABLED(CONFIG_TICKLESS_IDLE) && idle && ticks == K_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) u32_t delay; ticks = MIN(MAX_TICKS, MAX(ticks - 1, 0)); /* Desired delay in the future */ delay = (ticks == 0) ? MIN_DELAY : ticks * CYC_PER_TICK; k_spinlock_key_t key = k_spin_lock(&lock); delay += elapsed(); /* Round delay up to next tick boundary */ delay = ((delay + CYC_PER_TICK - 1) / CYC_PER_TICK) * CYC_PER_TICK; if (last_load != delay) { if (timer0_control_register_get() & _ARC_V2_TMR_CTRL_IP) { delay -= last_load; } timer0_limit_register_set(delay - 1); last_load = delay; timer0_control_register_set(_ARC_V2_TMR_CTRL_NH | _ARC_V2_TMR_CTRL_IE); } k_spin_unlock(&lock, key); #endif } u32_t z_clock_elapsed(void) { if (!TICKLESS) { return 0; } k_spinlock_key_t key = k_spin_lock(&lock); u32_t cyc = elapsed(); k_spin_unlock(&lock, key); return cyc / CYC_PER_TICK; } u32_t z_timer_cycle_get_32(void) { k_spinlock_key_t key = k_spin_lock(&lock); u32_t ret = elapsed() + cycle_count; k_spin_unlock(&lock, key); return ret; } /** * * @brief Stop announcing ticks into the kernel * * This routine disables timer interrupt generation and delivery. * Note that the timer's counting cannot be stopped by software. * * @return N/A */ void sys_clock_disable(void) { unsigned int key; /* interrupt lock level */ u32_t control; /* timer control register value */ key = irq_lock(); /* disable interrupt generation */ control = timer0_control_register_get(); timer0_control_register_set(control & ~_ARC_V2_TMR_CTRL_IE); irq_unlock(key); /* disable interrupt in the interrupt controller */ irq_disable(IRQ_TIMER0); }