Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753298AbcL3B24 (ORCPT ); Thu, 29 Dec 2016 20:28:56 -0500 Received: from mail-wj0-f194.google.com ([209.85.210.194]:36287 "EHLO mail-wj0-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753096AbcL3B2z (ORCPT ); Thu, 29 Dec 2016 20:28:55 -0500 MIME-Version: 1.0 In-Reply-To: <1483029949-6925-1-git-send-email-fweisbec@gmail.com> References: <1483029949-6925-1-git-send-email-fweisbec@gmail.com> From: Wanpeng Li Date: Fri, 30 Dec 2016 09:28:53 +0800 Message-ID: Subject: Re: [PATCH] x86/apic: Implement set_state_oneshot_stopped() callback To: Frederic Weisbecker Cc: Thomas Gleixner , LKML , Borislav Petkov , Wanpeng Li , Ingo Molnar , Viresh Kumar Content-Type: text/plain; charset=UTF-8 Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 3303 Lines: 68 2016-12-30 0:45 GMT+08:00 Frederic Weisbecker : > When clock_event_device::set_state_oneshot_stopped() is not implemented, > hrtimer_cancel() can't stop the clock when there is no more timer in > the queue. So the ghost of the freshly cancelled hrtimer haunts us back > later with an extra interrupt: > > -0 [002] d..2 2248.557659: hrtimer_cancel: hrtimer=ffff88021fa92d80 > -0 [002] d.h1 2249.303659: local_timer_entry: vector=239 > > So let's implement this missing callback for the lapic clock. This > consist in calling its set_state_shutdown() callback. There don't seem > to be a lighter way to stop the clock. Simply writing 0 to APIC_TMICT > won't be enough to stop the clock and avoid the extra interrupt, as > opposed to what is specified in the specs. We must also mask the > timer interrupt in the device. > > Cc: Thomas Gleixner > Cc: Borislav Petkov > Cc: Ingo Molnar > Cc: Wanpeng Li > Cc: Viresh Kumar > Signed-off-by: Frederic Weisbecker > --- Reviewed-by: Wanpeng Li > arch/x86/kernel/apic/apic.c | 25 +++++++++++++------------ > 1 file changed, 13 insertions(+), 12 deletions(-) > > diff --git a/arch/x86/kernel/apic/apic.c b/arch/x86/kernel/apic/apic.c > index 5b7e43e..a8e90db 100644 > --- a/arch/x86/kernel/apic/apic.c > +++ b/arch/x86/kernel/apic/apic.c > @@ -529,18 +529,19 @@ static void lapic_timer_broadcast(const struct cpumask *mask) > * The local apic timer can be used for any function which is CPU local. > */ > static struct clock_event_device lapic_clockevent = { > - .name = "lapic", > - .features = CLOCK_EVT_FEAT_PERIODIC | > - CLOCK_EVT_FEAT_ONESHOT | CLOCK_EVT_FEAT_C3STOP > - | CLOCK_EVT_FEAT_DUMMY, > - .shift = 32, > - .set_state_shutdown = lapic_timer_shutdown, > - .set_state_periodic = lapic_timer_set_periodic, > - .set_state_oneshot = lapic_timer_set_oneshot, > - .set_next_event = lapic_next_event, > - .broadcast = lapic_timer_broadcast, > - .rating = 100, > - .irq = -1, > + .name = "lapic", > + .features = CLOCK_EVT_FEAT_PERIODIC | > + CLOCK_EVT_FEAT_ONESHOT | CLOCK_EVT_FEAT_C3STOP > + | CLOCK_EVT_FEAT_DUMMY, > + .shift = 32, > + .set_state_shutdown = lapic_timer_shutdown, > + .set_state_periodic = lapic_timer_set_periodic, > + .set_state_oneshot = lapic_timer_set_oneshot, > + .set_state_oneshot_stopped = lapic_timer_shutdown, > + .set_next_event = lapic_next_event, > + .broadcast = lapic_timer_broadcast, > + .rating = 100, > + .irq = -1, > }; > static DEFINE_PER_CPU(struct clock_event_device, lapic_events); > > -- > 2.7.4 >