177 lines
4.4 KiB
C
177 lines
4.4 KiB
C
/*
|
|
* Copyright (c) 2018 Intel Corporation
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
#include <drivers/timer/system_timer.h>
|
|
#include <sys_clock.h>
|
|
#include <spinlock.h>
|
|
#include <irq.h>
|
|
|
|
#define HPET_REG32(off) (*(volatile u32_t *)(long) \
|
|
(DT_INST_0_INTEL_HPET_BASE_ADDRESS + (off)))
|
|
|
|
#define CLK_PERIOD_REG HPET_REG32(0x04) /* High dword of caps reg */
|
|
#define GENERAL_CONF_REG HPET_REG32(0x10)
|
|
#define MAIN_COUNTER_REG HPET_REG32(0xf0)
|
|
#define TIMER0_CONF_REG HPET_REG32(0x100)
|
|
#define TIMER0_COMPARATOR_REG HPET_REG32(0x108)
|
|
|
|
/* GENERAL_CONF_REG bits */
|
|
#define GCONF_ENABLE BIT(0)
|
|
#define GCONF_LR BIT(1) /* legacy interrupt routing, disables PIT */
|
|
|
|
/* TIMERn_CONF_REG bits */
|
|
#define TCONF_INT_ENABLE BIT(2)
|
|
#define TCONF_PERIODIC BIT(3)
|
|
#define TCONF_VAL_SET BIT(6)
|
|
#define TCONF_MODE32 BIT(8)
|
|
|
|
#define MIN_DELAY 1000
|
|
|
|
static struct k_spinlock lock;
|
|
static unsigned int max_ticks;
|
|
static unsigned int cyc_per_tick;
|
|
static unsigned int last_count;
|
|
|
|
static void hpet_isr(void *arg)
|
|
{
|
|
ARG_UNUSED(arg);
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
u32_t now = MAIN_COUNTER_REG;
|
|
|
|
if (IS_ENABLED(CONFIG_SMP) &&
|
|
IS_ENABLED(CONFIG_QEMU_TARGET)) {
|
|
/* Qemu in SMP mode has observed the clock going
|
|
* "backwards" relative to interrupts already received
|
|
* on the other CPU, despite the HPET being
|
|
* theoretically a global device.
|
|
*/
|
|
s32_t diff = (s32_t)(now - last_count);
|
|
|
|
if (last_count && diff < 0) {
|
|
now = last_count;
|
|
}
|
|
}
|
|
u32_t dticks = (now - last_count) / cyc_per_tick;
|
|
|
|
last_count += dticks * cyc_per_tick;
|
|
|
|
if (!IS_ENABLED(CONFIG_TICKLESS_KERNEL) ||
|
|
IS_ENABLED(CONFIG_QEMU_TICKLESS_WORKAROUND)) {
|
|
u32_t next = last_count + cyc_per_tick;
|
|
|
|
if ((s32_t)(next - now) < MIN_DELAY) {
|
|
next += cyc_per_tick;
|
|
}
|
|
TIMER0_COMPARATOR_REG = next;
|
|
}
|
|
|
|
k_spin_unlock(&lock, key);
|
|
z_clock_announce(IS_ENABLED(CONFIG_TICKLESS_KERNEL) ? dticks : 1);
|
|
}
|
|
|
|
static void set_timer0_irq(unsigned int irq)
|
|
{
|
|
/* 5-bit IRQ field starting at bit 9 */
|
|
u32_t val = (TIMER0_CONF_REG & ~(0x1f << 9)) | ((irq & 0x1f) << 9);
|
|
|
|
TIMER0_CONF_REG = val;
|
|
}
|
|
|
|
int z_clock_driver_init(struct device *device)
|
|
{
|
|
extern int z_clock_hw_cycles_per_sec;
|
|
u32_t hz;
|
|
|
|
IRQ_CONNECT(DT_INST_0_INTEL_HPET_IRQ_0,
|
|
DT_INST_0_INTEL_HPET_IRQ_0_PRIORITY,
|
|
hpet_isr, 0, 0);
|
|
set_timer0_irq(DT_INST_0_INTEL_HPET_IRQ_0);
|
|
irq_enable(DT_INST_0_INTEL_HPET_IRQ_0);
|
|
|
|
/* CLK_PERIOD_REG is in femtoseconds (1e-15 sec) */
|
|
hz = (u32_t)(1000000000000000ull / CLK_PERIOD_REG);
|
|
z_clock_hw_cycles_per_sec = hz;
|
|
cyc_per_tick = hz / CONFIG_SYS_CLOCK_TICKS_PER_SEC;
|
|
|
|
/* Note: we set the legacy routing bit, because otherwise
|
|
* nothing in Zephyr disables the PIT which then fires
|
|
* interrupts into the same IRQ. But that means we're then
|
|
* forced to use IRQ2 contra the way the kconfig IRQ selection
|
|
* is supposed to work. Should fix this.
|
|
*/
|
|
GENERAL_CONF_REG |= GCONF_LR | GCONF_ENABLE;
|
|
TIMER0_CONF_REG &= ~TCONF_PERIODIC;
|
|
TIMER0_CONF_REG |= TCONF_MODE32;
|
|
|
|
max_ticks = (0x7fffffff - cyc_per_tick) / cyc_per_tick;
|
|
last_count = MAIN_COUNTER_REG;
|
|
|
|
TIMER0_CONF_REG |= TCONF_INT_ENABLE;
|
|
TIMER0_COMPARATOR_REG = MAIN_COUNTER_REG + cyc_per_tick;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void smp_timer_init(void)
|
|
{
|
|
/* Noop, the HPET is a single system-wide device and it's
|
|
* configured to deliver interrupts to every CPU, so there's
|
|
* nothing to do at initialization on auxiliary CPUs.
|
|
*/
|
|
}
|
|
|
|
void z_clock_set_timeout(s32_t ticks, bool idle)
|
|
{
|
|
ARG_UNUSED(idle);
|
|
|
|
#if defined(CONFIG_TICKLESS_KERNEL) && !defined(CONFIG_QEMU_TICKLESS_WORKAROUND)
|
|
if (ticks == K_FOREVER && idle) {
|
|
GENERAL_CONF_REG &= ~GCONF_ENABLE;
|
|
return;
|
|
}
|
|
|
|
ticks = ticks == K_FOREVER ? max_ticks : ticks;
|
|
ticks = MAX(MIN(ticks - 1, (s32_t)max_ticks), 0);
|
|
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
u32_t now = MAIN_COUNTER_REG, cyc;
|
|
|
|
/* Round up to next tick boundary */
|
|
cyc = ticks * cyc_per_tick + (now - last_count) + (cyc_per_tick - 1);
|
|
cyc = (cyc / cyc_per_tick) * cyc_per_tick;
|
|
cyc += last_count;
|
|
|
|
if ((cyc - now) < MIN_DELAY) {
|
|
cyc += cyc_per_tick;
|
|
}
|
|
|
|
TIMER0_COMPARATOR_REG = cyc;
|
|
k_spin_unlock(&lock, key);
|
|
#endif
|
|
}
|
|
|
|
u32_t z_clock_elapsed(void)
|
|
{
|
|
if (!IS_ENABLED(CONFIG_TICKLESS_KERNEL)) {
|
|
return 0;
|
|
}
|
|
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
u32_t ret = (MAIN_COUNTER_REG - last_count) / cyc_per_tick;
|
|
|
|
k_spin_unlock(&lock, key);
|
|
return ret;
|
|
}
|
|
|
|
u32_t z_timer_cycle_get_32(void)
|
|
{
|
|
return MAIN_COUNTER_REG;
|
|
}
|
|
|
|
void z_clock_idle_exit(void)
|
|
{
|
|
GENERAL_CONF_REG |= GCONF_ENABLE;
|
|
}
|