blob: f2fcc076451ae8e543b83b6dddcb3047208bcb49 [file] [log] [blame] [edit]
// Copyright 2016 The Fuchsia Authors
// Copyright (c) 2008-2014 Travis Geiselbrecht
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
/**
* @file
* @brief Kernel timer subsystem
* @defgroup timer Timers
*
* The timer subsystem allows functions to be scheduled for later
* execution. Each timer object is used to cause one function to
* be executed at a later time.
*
* Timer callback functions are called in interrupt context.
*
* @{
*/
#include "kernel/timer.h"
#include <assert.h>
#include <debug.h>
#include <inttypes.h>
#include <lib/affine/ratio.h>
#include <lib/arch/intrin.h>
#include <lib/counters.h>
#include <platform.h>
#include <stdlib.h>
#include <trace.h>
#include <zircon/compiler.h>
#include <zircon/errors.h>
#include <zircon/listnode.h>
#include <zircon/time.h>
#include <zircon/types.h>
#include <kernel/align.h>
#include <kernel/lockdep.h>
#include <kernel/mp.h>
#include <kernel/percpu.h>
#include <kernel/scheduler.h>
#include <kernel/spinlock.h>
#include <kernel/stats.h>
#include <kernel/thread.h>
#include <platform/timer.h>
#define LOCAL_TRACE 0
// Total number of timers set. Always increasing.
KCOUNTER(timer_created_counter, "timer.created")
// Number of timers merged into an existing timer because of slack.
KCOUNTER(timer_coalesced_counter, "timer.coalesced")
// Number of timers that have fired (i.e. callback was invoked).
KCOUNTER(timer_fired_counter, "timer.fired")
// Number of timers that were successfully canceled. Attempts to cancel a timer that is currently
// firing are not counted.
KCOUNTER(timer_canceled_counter, "timer.canceled")
zx_ticks_t current_ticks(void) {
return platform_current_ticks();
}
namespace {
SpinLock timer_lock __CPU_ALIGN_EXCLUSIVE;
DECLARE_SINGLETON_LOCK_WRAPPER(TimerLock, timer_lock);
affine::Ratio gTicksToTime;
uint64_t gTicksPerSecond;
} // anonymous namespace
void platform_set_ticks_to_time_ratio(const affine::Ratio& ticks_to_time) {
// ASSERT that we are not calling this function twice. Once set, this ratio
// may not change.
DEBUG_ASSERT(gTicksPerSecond == 0);
DEBUG_ASSERT(ticks_to_time.numerator() != 0);
DEBUG_ASSERT(ticks_to_time.denominator() != 0);
gTicksToTime = ticks_to_time;
gTicksPerSecond = gTicksToTime.Inverse().Scale(ZX_SEC(1));
}
const affine::Ratio& platform_get_ticks_to_time_ratio(void) { return gTicksToTime; }
zx_time_t current_time(void) { return gTicksToTime.Scale(current_ticks()); }
zx_ticks_t ticks_per_second(void) { return gTicksPerSecond; }
void TimerQueue::UpdatePlatformTimer(zx_time_t new_deadline) {
DEBUG_ASSERT(arch_ints_disabled());
if (new_deadline < next_timer_deadline_) {
LTRACEF("rescheduling timer for %" PRIi64 " nsecs\n", new_deadline);
platform_set_oneshot_timer(new_deadline);
next_timer_deadline_ = new_deadline;
}
}
void TimerQueue::Insert(Timer* timer, zx_time_t earliest_deadline, zx_time_t latest_deadline) {
DEBUG_ASSERT(arch_ints_disabled());
cpu_num_t cpu = arch_curr_cpu_num();
LTRACEF("timer %p, cpu %u, scheduled %" PRIi64 "\n", timer, cpu, timer->scheduled_time_);
// For inserting the timer we consider several cases. In general we
// want to coalesce with the current timer unless we can prove that
// either that:
// 1- there is no slack overlap with current timer OR
// 2- the next timer is a better fit.
//
// In diagrams that follow
// - Let |e| be the current (existing) timer deadline
// - Let |t| be the deadline of the timer we are inserting
// - Let |n| be the next timer deadline if any
// - Let |x| be the end of the list (not a timer)
// - Let |(| and |)| the earliest_deadline and latest_deadline.
for (Timer& entry : timer_list_) {
if (entry.scheduled_time_ > latest_deadline) {
// New timer latest is earlier than the current timer.
// Just add upfront as is, without slack.
//
// ---------t---)--e-------------------------------> time
timer->slack_ = 0ll;
timer_list_.insert(entry, timer);
return;
}
if (entry.scheduled_time_ >= timer->scheduled_time_) {
// New timer slack overlaps and is to the left (or equal). We
// coalesce with current by scheduling late.
//
// --------(----t---e-)----------------------------> time
timer->slack_ = zx_time_sub_time(entry.scheduled_time_, timer->scheduled_time_);
timer->scheduled_time_ = entry.scheduled_time_;
kcounter_add(timer_coalesced_counter, 1);
timer_list_.insert_after(timer_list_.make_iterator(entry), timer);
return;
}
if (entry.scheduled_time_ < earliest_deadline) {
// new timer earliest is later than the current timer. This case
// is handled in a future iteration.
//
// ----------------e--(---t-----------------------> time
continue;
}
// New timer is to the right of current timer and there is overlap
// with the current timer, but could the next timer (if any) be
// a better fit?
//
// -------------(--e---t-----?-------------------> time
auto iter = timer_list_.make_iterator(entry);
++iter;
if (iter != timer_list_.end()) {
const Timer& next = *iter;
if (next.scheduled_time_ <= timer->scheduled_time_) {
// The new timer is to the right of the next timer. There is no
// chance the current timer is a better fit.
//
// -------------(--e---n---t----------------------> time
continue;
}
if (next.scheduled_time_ < latest_deadline) {
// There is slack overlap with the next timer, and also with the
// current timer. Which coalescing is a better match?
//
// --------------(-e---t---n-)-----------------------> time
zx_duration_t delta_entry = zx_time_sub_time(timer->scheduled_time_, entry.scheduled_time_);
zx_duration_t delta_next = zx_time_sub_time(next.scheduled_time_, timer->scheduled_time_);
if (delta_next < delta_entry) {
// New timer is closer to the next timer, handle it in the
// next iteration.
continue;
}
}
}
// Handles the remaining cases, note that there is overlap with
// the current timer.
//
// 1- this is the last timer (next == NULL) or
// 2- there is no overlap with the next timer, or
// 3- there is overlap with both current and next but
// current is closer.
//
// So we coalesce by scheduling early.
timer->slack_ = zx_time_sub_time(entry.scheduled_time_, timer->scheduled_time_);
timer->scheduled_time_ = entry.scheduled_time_;
kcounter_add(timer_coalesced_counter, 1);
timer_list_.insert_after(timer_list_.make_iterator(entry), timer);
return;
}
// Walked off the end of the list and there was no overlap.
timer->slack_ = 0;
timer_list_.push_back(timer);
}
Timer::~Timer() {
// Ensure that we are not on any TimerQueue's list.
ZX_DEBUG_ASSERT(!InContainer());
// Ensure that we are not active on some cpu.
ZX_DEBUG_ASSERT(active_cpu_.load(ktl::memory_order_relaxed) == INVALID_CPU);
}
void Timer::Set(const Deadline& deadline, Callback callback, void* arg) {
LTRACEF("timer %p deadline.when %" PRIi64 " deadline.slack.amount %" PRIi64
" deadline.slack.mode %u callback %p arg %p\n",
this, deadline.when(), deadline.slack().amount(), deadline.slack().mode(), callback, arg);
DEBUG_ASSERT(magic_ == kMagic);
DEBUG_ASSERT(deadline.slack().mode() <= TIMER_SLACK_LATE);
DEBUG_ASSERT(deadline.slack().amount() >= 0);
if (InContainer()) {
panic("timer %p already in list\n", this);
}
const zx_time_t latest_deadline = deadline.latest();
const zx_time_t earliest_deadline = deadline.earliest();
Guard<SpinLock, IrqSave> guard{TimerLock::Get()};
cpu_num_t cpu = arch_curr_cpu_num();
cpu_num_t active_cpu = active_cpu_.load(ktl::memory_order_relaxed);
bool currently_active = (active_cpu == cpu);
if (unlikely(currently_active)) {
// The timer is active on our own cpu, we must be inside the callback.
if (cancel_.load(ktl::memory_order_relaxed)) {
return;
}
} else if (unlikely(active_cpu != INVALID_CPU)) {
panic("timer %p currently active on a different cpu %u\n", this, active_cpu);
}
// Set up the structure.
scheduled_time_ = deadline.when();
callback_ = callback;
arg_ = arg;
cancel_.store(false, ktl::memory_order_relaxed);
// We don't need to modify active_cpu_ because it is managed by timer_tick().
LTRACEF("scheduled time %" PRIi64 "\n", scheduled_time_);
TimerQueue& timer_queue = percpu::Get(cpu).timer_queue;
timer_queue.Insert(this, earliest_deadline, latest_deadline);
kcounter_add(timer_created_counter, 1);
if (!timer_queue.timer_list_.is_empty() && &timer_queue.timer_list_.front() == this) {
// We just modified the head of the timer queue.
timer_queue.UpdatePlatformTimer(deadline.when());
}
}
void TimerQueue::PreemptReset(zx_time_t deadline) {
DEBUG_ASSERT(arch_ints_disabled());
LTRACEF("preempt timer cpu %u deadline %" PRIi64 "\n", arch_curr_cpu_num(), deadline);
preempt_timer_deadline_ = deadline;
UpdatePlatformTimer(deadline);
}
bool Timer::Cancel() {
DEBUG_ASSERT(magic_ == kMagic);
Guard<SpinLock, IrqSave> guard{TimerLock::Get()};
cpu_num_t cpu = arch_curr_cpu_num();
// mark the timer as canceled
cancel_.store(true, ktl::memory_order_relaxed);
// TODO(fxbug.dev/64092): Consider whether this DeviceMemoryBarrier is required
arch::DeviceMemoryBarrier();
// see if we're trying to cancel the timer we're currently in the middle of handling
if (unlikely(active_cpu_.load(ktl::memory_order_relaxed) == cpu)) {
// zero it out
callback_ = nullptr;
arg_ = nullptr;
// we're done, so return back to the callback
return false;
}
bool callback_not_running;
// If this Timer is in a queue, remove it and adjust hardware timers if needed.
if (InContainer()) {
callback_not_running = true;
TimerQueue& timer_queue = percpu::Get(cpu).timer_queue;
// Save a copy of the old head of the queue so later we can see if we modified the head.
const Timer* oldhead = nullptr;
if (!timer_queue.timer_list_.is_empty()) {
oldhead = &timer_queue.timer_list_.front();
}
// Remove this Timer from this whatever TimerQueue it's on.
RemoveFromContainer();
kcounter_add(timer_canceled_counter, 1);
// TODO(cpu): If, after removing |timer| there is one other single Timer with
// the same scheduled_time_ and slack_ non-zero, then it is possible to return
// that timer to the ideal scheduled_time_.
// See if we've just modified the head of this TimerQueue.
//
// If Timer was on another cpu's queue, we'll just let it fire and sort itself out.
if (unlikely(oldhead == this)) {
// The Timer we're canceling was at head of this queue, so see if we should update platform
// timer.
if (!timer_queue.timer_list_.is_empty()) {
timer_queue.UpdatePlatformTimer(timer_queue.timer_list_.front().scheduled_time_);
} else if (timer_queue.next_timer_deadline_ == ZX_TIME_INFINITE) {
LTRACEF("clearing old hw timer, preempt timer not set, nothing in the queue\n");
platform_stop_timer();
}
}
} else {
callback_not_running = false;
}
guard.Release();
// wait for the timer to become un-busy in case a callback is currently active on another cpu
while (active_cpu_.load(ktl::memory_order_relaxed) != INVALID_CPU) {
arch::Yield();
}
// zero it out
callback_ = nullptr;
arg_ = nullptr;
return callback_not_running;
}
// called at interrupt time to process any pending timers
void timer_tick(zx_time_t now) {
DEBUG_ASSERT(arch_ints_disabled());
CPU_STATS_INC(timer_ints);
cpu_num_t cpu = arch_curr_cpu_num();
LTRACEF("cpu %u now %" PRIi64 ", sp %p\n", cpu, now, __GET_FRAME());
percpu::Get(cpu).timer_queue.Tick(now, cpu);
}
void TimerQueue::Tick(zx_time_t now, cpu_num_t cpu) {
// The platform timer has fired, so no deadline is set.
next_timer_deadline_ = ZX_TIME_INFINITE;
// Service the preemption timer before acquiring the timer lock.
if (now >= preempt_timer_deadline_) {
preempt_timer_deadline_ = ZX_TIME_INFINITE;
Scheduler::TimerTick(SchedTime{now});
}
Guard<SpinLock, NoIrqSave> guard{TimerLock::Get()};
for (;;) {
// See if there's an event to process.
if (timer_list_.is_empty()) {
break;
}
Timer& timer = timer_list_.front();
LTRACEF("next item on timer queue %p at %" PRIi64 " now %" PRIi64 " (%p, arg %p)\n", &timer,
timer.scheduled_time_, now, timer.callback_, timer.arg_);
if (likely(now < timer.scheduled_time_)) {
break;
}
// Process it.
LTRACEF("timer %p\n", &timer);
DEBUG_ASSERT_MSG(timer.magic_ == Timer::kMagic,
"ASSERT: timer failed magic check: timer %p, magic 0x%x\n", &timer,
(uint)timer.magic_);
timer_list_.erase(timer);
// Mark the timer busy.
timer.active_cpu_.store(cpu, ktl::memory_order_relaxed);
// Unlocking the spinlock in CallUnlocked acts as a release fence.
// Now that the timer is off of the list, release the spinlock to handle
// the callback, then re-acquire in case it is requeued.
guard.CallUnlocked([&timer, now]() {
LTRACEF("dequeued timer %p, scheduled %" PRIi64 "\n", &timer, timer.scheduled_time_);
CPU_STATS_INC(timers);
kcounter_add(timer_fired_counter, 1);
LTRACEF("timer %p firing callback %p, arg %p\n", &timer, timer.callback_, timer.arg_);
timer.callback_(&timer, now, timer.arg_);
DEBUG_ASSERT(arch_ints_disabled());
});
// Mark it not busy.
timer.active_cpu_.store(INVALID_CPU, ktl::memory_order_relaxed);
// TODO(fxbug.dev/64092): Consider whether this DeviceMemoryBarrier is required
arch::DeviceMemoryBarrier();
}
// Get the deadline of the event at the head of the queue (if any).
zx_time_t deadline = ZX_TIME_INFINITE;
if (!timer_list_.is_empty()) {
deadline = timer_list_.front().scheduled_time_;
// This has to be the case or it would have fired already.
DEBUG_ASSERT(deadline > now);
}
// We're done manipulating the timer queue.
guard.Release();
// Set the platform timer to the *soonest* of queue event and preemption timer.
if (preempt_timer_deadline_ < deadline) {
deadline = preempt_timer_deadline_;
}
UpdatePlatformTimer(deadline);
}
zx_status_t Timer::TrylockOrCancel(SpinLock* lock) {
// spin trylocking on the passed in spinlock either waiting for it
// to grab or the passed in timer to be canceled.
while (unlikely(lock->TryAcquire())) {
// we failed to grab it, check for cancel
if (cancel_.load(ktl::memory_order_relaxed)) {
// we were canceled, so bail immediately
return ZX_ERR_TIMED_OUT;
}
// tell the arch to wait
arch::Yield();
}
return ZX_OK;
}
void TimerQueue::TransitionOffCpu(TimerQueue& source) {
Guard<SpinLock, IrqSave> guard{TimerLock::Get()};
Timer* old_head = nullptr;
if (!timer_list_.is_empty()) {
old_head = &timer_list_.front();
}
// Move all timers from |source| to this TimerQueue.
Timer* timer;
while ((timer = source.timer_list_.pop_front()) != nullptr) {
// We lost the original asymmetric slack information so when we combine them
// with the other timer queue they are not coalesced again.
// TODO(cpu): figure how important this case is.
Insert(timer, timer->scheduled_time_, timer->scheduled_time_);
// Note, we do not increment the "created" counter here because we are simply moving these
// timers from one queue to another and we already counted them when they were first
// created.
}
Timer* new_head = nullptr;
if (!timer_list_.is_empty()) {
new_head = &timer_list_.front();
}
if (new_head != nullptr && new_head != old_head) {
// We just modified the head of the timer queue.
UpdatePlatformTimer(new_head->scheduled_time_);
}
// The old TimerQueue has no tasks left, so reset the deadlines.
source.preempt_timer_deadline_ = ZX_TIME_INFINITE;
source.next_timer_deadline_ = ZX_TIME_INFINITE;
}
void TimerQueue::ThawPercpu(void) {
DEBUG_ASSERT(arch_ints_disabled());
Guard<SpinLock, NoIrqSave> guard{TimerLock::Get()};
// Reset next_timer_deadline_ so that UpdatePlatformTimer will reconfigure the timer.
next_timer_deadline_ = ZX_TIME_INFINITE;
zx_time_t deadline = preempt_timer_deadline_;
if (!timer_list_.is_empty()) {
Timer& t = timer_list_.front();
if (t.scheduled_time_ < deadline) {
deadline = t.scheduled_time_;
}
}
guard.Release();
UpdatePlatformTimer(deadline);
}
void TimerQueue::PrintTimerQueues(char* buf, size_t len) {
size_t ptr = 0;
zx_time_t now = current_time();
Guard<SpinLock, IrqSave> guard{TimerLock::Get()};
for (cpu_num_t i = 0; i < percpu::processor_count(); i++) {
if (mp_is_cpu_online(i)) {
ptr += snprintf(buf + ptr, len - ptr, "cpu %u:\n", i);
if (ptr >= len) {
return;
}
zx_time_t last = now;
for (Timer& t : percpu::Get(i).timer_queue.timer_list_) {
zx_duration_t delta_now = zx_time_sub_time(t.scheduled_time_, now);
zx_duration_t delta_last = zx_time_sub_time(t.scheduled_time_, last);
ptr += snprintf(buf + ptr, len - ptr,
"\ttime %" PRIi64 " delta_now %" PRIi64 " delta_last %" PRIi64
" func %p arg %p\n",
t.scheduled_time_, delta_now, delta_last, t.callback_, t.arg_);
if (ptr >= len) {
return;
}
last = t.scheduled_time_;
}
}
}
}
#include <lib/console.h>
static int cmd_timers(int argc, const cmd_args* argv, uint32_t flags) {
const size_t timer_buffer_size = PAGE_SIZE;
// allocate a buffer to dump the timer queue into to avoid reentrancy issues with the
// timer spinlock
char* buf = static_cast<char*>(malloc(timer_buffer_size));
if (!buf) {
return ZX_ERR_NO_MEMORY;
}
TimerQueue::PrintTimerQueues(buf, timer_buffer_size);
printf("%s", buf);
free(buf);
return 0;
}
STATIC_COMMAND_START
STATIC_COMMAND_MASKED("timers", "dump the current kernel timer queues", &cmd_timers,
CMD_AVAIL_NORMAL)
STATIC_COMMAND_END(kernel)