timer/trace: Improve timer tracing
Timers are added to the timer wheel off by one. This is required in case a timer is queued directly before incrementing jiffies to prevent early timer expiry. When reading a timer trace and relying only on the expiry time of the timer in the timer_start trace point and on the now in the timer_expiry_entry trace point, it seems that the timer fires late. With the current timer_expiry_entry trace point information only now=jiffies is printed but not the value of base->clk. This makes it impossible to draw a conclusion to the index of base->clk and makes it impossible to examine timer problems without additional trace points. Therefore add the base->clk value to the timer_expire_entry trace point, to be able to calculate the index the timer base is located at during collecting expired timers. Signed-off-by: Anna-Maria Gleixner <anna-maria@linutronix.de> Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Cc: fweisbec@gmail.com Cc: peterz@infradead.org Cc: Steven Rostedt <rostedt@goodmis.org> Link: https://lkml.kernel.org/r/20190321120921.16463-5-anna-maria@linutronix.de
This commit is contained in:
parent
6849cbb0f9
commit
f28d3d5346
|
@ -89,24 +89,27 @@ TRACE_EVENT(timer_start,
|
||||||
*/
|
*/
|
||||||
TRACE_EVENT(timer_expire_entry,
|
TRACE_EVENT(timer_expire_entry,
|
||||||
|
|
||||||
TP_PROTO(struct timer_list *timer),
|
TP_PROTO(struct timer_list *timer, unsigned long baseclk),
|
||||||
|
|
||||||
TP_ARGS(timer),
|
TP_ARGS(timer, baseclk),
|
||||||
|
|
||||||
TP_STRUCT__entry(
|
TP_STRUCT__entry(
|
||||||
__field( void *, timer )
|
__field( void *, timer )
|
||||||
__field( unsigned long, now )
|
__field( unsigned long, now )
|
||||||
__field( void *, function)
|
__field( void *, function)
|
||||||
|
__field( unsigned long, baseclk )
|
||||||
),
|
),
|
||||||
|
|
||||||
TP_fast_assign(
|
TP_fast_assign(
|
||||||
__entry->timer = timer;
|
__entry->timer = timer;
|
||||||
__entry->now = jiffies;
|
__entry->now = jiffies;
|
||||||
__entry->function = timer->function;
|
__entry->function = timer->function;
|
||||||
|
__entry->baseclk = baseclk;
|
||||||
),
|
),
|
||||||
|
|
||||||
TP_printk("timer=%p function=%ps now=%lu",
|
TP_printk("timer=%p function=%ps now=%lu baseclk=%lu",
|
||||||
__entry->timer, __entry->function, __entry->now)
|
__entry->timer, __entry->function, __entry->now,
|
||||||
|
__entry->baseclk)
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -1293,7 +1293,9 @@ int del_timer_sync(struct timer_list *timer)
|
||||||
EXPORT_SYMBOL(del_timer_sync);
|
EXPORT_SYMBOL(del_timer_sync);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static void call_timer_fn(struct timer_list *timer, void (*fn)(struct timer_list *))
|
static void call_timer_fn(struct timer_list *timer,
|
||||||
|
void (*fn)(struct timer_list *),
|
||||||
|
unsigned long baseclk)
|
||||||
{
|
{
|
||||||
int count = preempt_count();
|
int count = preempt_count();
|
||||||
|
|
||||||
|
@ -1316,7 +1318,7 @@ static void call_timer_fn(struct timer_list *timer, void (*fn)(struct timer_list
|
||||||
*/
|
*/
|
||||||
lock_map_acquire(&lockdep_map);
|
lock_map_acquire(&lockdep_map);
|
||||||
|
|
||||||
trace_timer_expire_entry(timer);
|
trace_timer_expire_entry(timer, baseclk);
|
||||||
fn(timer);
|
fn(timer);
|
||||||
trace_timer_expire_exit(timer);
|
trace_timer_expire_exit(timer);
|
||||||
|
|
||||||
|
@ -1337,6 +1339,13 @@ static void call_timer_fn(struct timer_list *timer, void (*fn)(struct timer_list
|
||||||
|
|
||||||
static void expire_timers(struct timer_base *base, struct hlist_head *head)
|
static void expire_timers(struct timer_base *base, struct hlist_head *head)
|
||||||
{
|
{
|
||||||
|
/*
|
||||||
|
* This value is required only for tracing. base->clk was
|
||||||
|
* incremented directly before expire_timers was called. But expiry
|
||||||
|
* is related to the old base->clk value.
|
||||||
|
*/
|
||||||
|
unsigned long baseclk = base->clk - 1;
|
||||||
|
|
||||||
while (!hlist_empty(head)) {
|
while (!hlist_empty(head)) {
|
||||||
struct timer_list *timer;
|
struct timer_list *timer;
|
||||||
void (*fn)(struct timer_list *);
|
void (*fn)(struct timer_list *);
|
||||||
|
@ -1350,11 +1359,11 @@ static void expire_timers(struct timer_base *base, struct hlist_head *head)
|
||||||
|
|
||||||
if (timer->flags & TIMER_IRQSAFE) {
|
if (timer->flags & TIMER_IRQSAFE) {
|
||||||
raw_spin_unlock(&base->lock);
|
raw_spin_unlock(&base->lock);
|
||||||
call_timer_fn(timer, fn);
|
call_timer_fn(timer, fn, baseclk);
|
||||||
raw_spin_lock(&base->lock);
|
raw_spin_lock(&base->lock);
|
||||||
} else {
|
} else {
|
||||||
raw_spin_unlock_irq(&base->lock);
|
raw_spin_unlock_irq(&base->lock);
|
||||||
call_timer_fn(timer, fn);
|
call_timer_fn(timer, fn, baseclk);
|
||||||
raw_spin_lock_irq(&base->lock);
|
raw_spin_lock_irq(&base->lock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue