blob: 1e4b1911e27e300d6faec220837e273c0f99f72b [file] [log] [blame]
/* Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
/* Timer module for Chrome EC operating system */
#include "atomic.h"
#include "console.h"
#include "hooks.h"
#include "hwtimer.h"
#include "system.h"
#include "util.h"
#include "task.h"
#include "timer.h"
#define TIMER_SYSJUMP_TAG 0x4d54 /* "TM" */
/* High word of the 64-bit timestamp counter */
static volatile uint32_t clksrc_high;
/* Bitmap of currently running timers */
static uint32_t timer_running;
/* Deadlines of all timers */
static timestamp_t timer_deadline[TASK_ID_COUNT];
static uint32_t next_deadline = 0xffffffff;
/* Hardware timer routine IRQ number */
static int timer_irq;
static void expire_timer(task_id_t tskid)
{
/* we are done with this timer */
atomic_clear(&timer_running, 1 << tskid);
/* wake up the taks waiting for this timer */
task_set_event(tskid, TASK_EVENT_TIMER, 0);
}
int timestamp_expired(timestamp_t deadline, const timestamp_t *now)
{
timestamp_t now_val;
if (!now) {
now_val = get_time();
now = &now_val;
}
return ((int64_t)(now->val - deadline.val) >= 0);
}
void process_timers(int overflow)
{
uint32_t check_timer, running_t0;
timestamp_t next;
timestamp_t now;
if (overflow)
clksrc_high++;
do {
next.val = -1ull;
now = get_time();
do {
/* read atomically the current state of timer running */
check_timer = running_t0 = timer_running;
while (check_timer) {
int tskid = __fls(check_timer);
/* timer has expired ? */
if (timer_deadline[tskid].val <= now.val)
expire_timer(tskid);
else if ((timer_deadline[tskid].le.hi ==
now.le.hi) &&
(timer_deadline[tskid].le.lo <
next.le.lo))
next.val = timer_deadline[tskid].val;
check_timer &= ~(1 << tskid);
}
/* if there is a new timer, let's retry */
} while (timer_running & ~running_t0);
if (next.le.hi == 0xffffffff) {
/* no deadline to set */
__hw_clock_event_clear();
next_deadline = 0xffffffff;
return;
}
__hw_clock_event_set(next.le.lo);
next_deadline = next.le.lo;
} while (next.val <= get_time().val);
}
#ifndef CONFIG_HW_SPECIFIC_UDELAY
void udelay(unsigned us)
{
unsigned t0 = __hw_clock_source_read();
/*
* udelay() may be called with interrupts disabled, so we can't rely on
* process_timers() updating the top 32 bits. So handle wraparound
* ourselves rather than calling get_time() and comparing with a
* deadline.
*
* This may fail for delays close to 2^32 us (~4000 sec), because the
* subtraction below can overflow. That's acceptable, because the
* watchdog timer would have tripped long before that anyway.
*/
while (__hw_clock_source_read() - t0 <= us)
;
}
#endif
int timer_arm(timestamp_t tstamp, task_id_t tskid)
{
ASSERT(tskid < TASK_ID_COUNT);
if (timer_running & (1<<tskid))
return EC_ERROR_BUSY;
timer_deadline[tskid] = tstamp;
atomic_or(&timer_running, 1<<tskid);
/* Modify the next event if needed */
if ((tstamp.le.hi < clksrc_high) ||
((tstamp.le.hi == clksrc_high) && (tstamp.le.lo <= next_deadline)))
task_trigger_irq(timer_irq);
return EC_SUCCESS;
}
void timer_cancel(task_id_t tskid)
{
ASSERT(tskid < TASK_ID_COUNT);
atomic_clear(&timer_running, 1 << tskid);
/*
* Don't need to cancel the hardware timer interrupt, instead do
* timer-related housekeeping when the next timer interrupt fires.
*/
}
/*
* For us < (2^31 - task scheduling latency)(~ 2147 sec), this function will
* sleep for at least us, and no more than 2*us. As us approaches 2^32-1, the
* probability of delay longer than 2*us (and possibly infinite delay)
* increases.
*/
void usleep(unsigned us)
{
uint32_t evt = 0;
uint32_t t0 = __hw_clock_source_read();
/* If task scheduling has not started, just delay */
if (!task_start_called()) {
udelay(us);
return;
}
ASSERT(us);
do {
evt |= task_wait_event(us);
} while (!(evt & TASK_EVENT_TIMER) &&
((__hw_clock_source_read() - t0) < us));
/* Re-queue other events which happened in the meanwhile */
if (evt)
atomic_or(task_get_event_bitmap(task_get_current()),
evt & ~TASK_EVENT_TIMER);
}
timestamp_t get_time(void)
{
timestamp_t ts;
ts.le.hi = clksrc_high;
ts.le.lo = __hw_clock_source_read();
if (ts.le.hi != clksrc_high) {
ts.le.hi = clksrc_high;
ts.le.lo = __hw_clock_source_read();
}
return ts;
}
clock_t clock(void)
{
/* __hw_clock_source_read() returns a microsecond resolution timer.*/
return (clock_t) __hw_clock_source_read() / 1000;
}
void force_time(timestamp_t ts)
{
clksrc_high = ts.le.hi;
__hw_clock_source_set(ts.le.lo);
/* some timers might be already expired : process them */
task_trigger_irq(timer_irq);
}
#ifdef CONFIG_CMD_TIMERINFO
void timer_print_info(void)
{
uint64_t t = get_time().val;
uint64_t deadline = (uint64_t)clksrc_high << 32 |
__hw_clock_event_get();
int tskid;
ccprintf("Time: 0x%016lx us, %11.6ld s\n"
"Deadline: 0x%016lx -> %11.6ld s from now\n"
"Active timers:\n",
t, t, deadline, deadline - t);
cflush();
for (tskid = 0; tskid < TASK_ID_COUNT; tskid++) {
if (timer_running & (1<<tskid)) {
ccprintf(" Tsk %2d 0x%016lx -> %11.6ld\n", tskid,
timer_deadline[tskid].val,
timer_deadline[tskid].val - t);
cflush();
}
}
}
#else
void timer_print_info(void) { }
#endif
void timer_init(void)
{
const timestamp_t *ts;
int size, version;
BUILD_ASSERT(TASK_ID_COUNT < sizeof(timer_running) * 8);
/* Restore time from before sysjump */
ts = (const timestamp_t *)system_get_jump_tag(TIMER_SYSJUMP_TAG,
&version, &size);
if (ts && version == 1 && size == sizeof(timestamp_t)) {
clksrc_high = ts->le.hi;
timer_irq = __hw_clock_source_init(ts->le.lo);
} else {
clksrc_high = 0;
timer_irq = __hw_clock_source_init(0);
}
}
/* Preserve time across a sysjump */
static void timer_sysjump(void)
{
timestamp_t ts = get_time();
system_add_jump_tag(TIMER_SYSJUMP_TAG, 1, sizeof(ts), &ts);
}
DECLARE_HOOK(HOOK_SYSJUMP, timer_sysjump, HOOK_PRIO_DEFAULT);
#ifdef CONFIG_CMD_WAITMS
static int command_wait(int argc, char **argv)
{
char *e;
int i;
if (argc < 2)
return EC_ERROR_PARAM_COUNT;
i = strtoi(argv[1], &e, 0);
if (*e)
return EC_ERROR_PARAM1;
udelay(i * 1000);
return EC_SUCCESS;
}
DECLARE_CONSOLE_COMMAND(waitms, command_wait,
"msec",
"Busy-wait for msec");
#endif
#ifdef CONFIG_CMD_FORCETIME
static int command_force_time(int argc, char **argv)
{
char *e;
timestamp_t new;
if (argc < 3)
return EC_ERROR_PARAM_COUNT;
new.le.hi = strtoi(argv[1], &e, 0);
if (*e)
return EC_ERROR_PARAM1;
new.le.lo = strtoi(argv[2], &e, 0);
if (*e)
return EC_ERROR_PARAM2;
ccprintf("Time: 0x%016lx = %.6ld s\n", new.val, new.val);
force_time(new);
return EC_SUCCESS;
}
DECLARE_CONSOLE_COMMAND(forcetime, command_force_time,
"hi lo",
"Force current time");
#endif
static int command_get_time(int argc, char **argv)
{
timestamp_t ts = get_time();
ccprintf("Time: 0x%016lx = %.6ld s\n", ts.val, ts.val);
return EC_SUCCESS;
}
DECLARE_SAFE_CONSOLE_COMMAND(gettime, command_get_time,
NULL,
"Print current time");
#ifdef CONFIG_CMD_TIMERINFO
int command_timer_info(int argc, char **argv)
{
timer_print_info();
return EC_SUCCESS;
}
DECLARE_SAFE_CONSOLE_COMMAND(timerinfo, command_timer_info,
NULL,
"Print timer info");
#endif