Received: by 2002:a05:6359:c8b:b0:c7:702f:21d4 with SMTP id go11csp3042194rwb; Sat, 8 Oct 2022 20:27:52 -0700 (PDT) X-Google-Smtp-Source: AMsMyM5X7qLBJ/9Yw56MHhLEhuPD1NTlmacYG3usyQVb2VXDzvCsAy+2xFJSi6boHi/JZiByAOiJ X-Received: by 2002:a05:6402:1393:b0:457:ea9e:ba20 with SMTP id b19-20020a056402139300b00457ea9eba20mr11579423edv.109.1665286072322; Sat, 08 Oct 2022 20:27:52 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1665286072; cv=none; d=google.com; s=arc-20160816; b=xkCF4VomC7+9gKPwpExc9X9pl502AmVDTJuCLKT/kvkz9F2uh5N/TFAPUHuEwbw5+I NtVPXK2CSQQN19rBjIaGVLQL4A4OiJd+uPPbmXayH8QEEVJnbPosshvtMC9iJQh+IkMn SQ2U2fguRXDAZYBRJWSuOStJwdcnLbVJO4UqXrYRUesA/2rtdXS44EDzORLfZuPqklmf DI+lzfEqN0UEHgbx4Of6Fjm6qEZJdc8iaR09PLLlBYaK5sS4l5ePR5lM7GgtqGOVaBzd Cmaf03ULdKuWw9H7RMI/jt5Q0tJlITpcsFJ3SAZRx6LP5HTaJkMrgtJ6vsc3b5Q3O7ch Gaaw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :message-id:date:subject:cc:to:from; bh=CtTCmME9h6fye7mQAfPmK0XBLd4wSphz2qkaf8+ubkg=; b=NWc+HSQuZPCg3hodkDUjOiyoM7Hch93jRrBAnYsxldDLAeVZHGMY/EoOjDdAMvPUuw 46jSoGZoKHrhSsR4kZjIFDgStUtiop4n51v5FPcm7fWGn9wlsLK6muY0vT+IsY3ercU1 2GeAuNwK+M5wADezY79a0f67SAcbWrWJJCcKO6LKPj0XpC1HOnKBU0s9l5wtRLxWtfrS 5JsejcOUzNWQC7zqLoUHQoxqMUW64qFqNnIZpxzaJ5PBQzGYVlsWPNpH78ZHaTQYBFTN AbX2+Jgsc2YljSZNKKsBujIYZ2+6vOaAWbXk6wmvOi8uuSx7G61G4u7TZfOEjuKzeM7U 7Hag== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id dr14-20020a170907720e00b007877b1c7f27si5912297ejc.829.2022.10.08.20.27.27; Sat, 08 Oct 2022 20:27:52 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S229553AbiJIDZX (ORCPT + 99 others); Sat, 8 Oct 2022 23:25:23 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:51736 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229538AbiJIDZT (ORCPT ); Sat, 8 Oct 2022 23:25:19 -0400 Received: from out28-122.mail.aliyun.com (out28-122.mail.aliyun.com [115.124.28.122]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 0DA7C248CB for ; Sat, 8 Oct 2022 20:25:15 -0700 (PDT) X-Alimail-AntiSpam: AC=CONTINUE;BC=0.07436893|-1;CH=green;DM=|CONTINUE|false|;DS=CONTINUE|ham_alarm|0.118326-0.0259815-0.855693;FP=0|0|0|0|0|-1|-1|-1;HT=ay29a033018047202;MF=victor@allwinnertech.com;NM=1;PH=DS;RN=9;RT=9;SR=0;TI=SMTPD_---.PWwt9xN_1665285911; Received: from SunxiBot.allwinnertech.com(mailfrom:victor@allwinnertech.com fp:SMTPD_---.PWwt9xN_1665285911) by smtp.aliyun-inc.com; Sun, 09 Oct 2022 11:25:13 +0800 From: Victor Hassan To: daniel.lezcano@linaro.org, tglx@linutronix.de, wens@csie.org, jernej.skrabec@gmail.com, samuel@sholland.org, maxime@cerno.tech Cc: linux-kernel@vger.kernel.org, linux-arm-kernel@lists.infradead.org, linux-sunxi@lists.linux.dev Subject: [RESEND] clocksource: sun4i: Fix the bug that tick_resume stucks Date: Sun, 9 Oct 2022 11:25:07 +0800 Message-Id: <20221009032507.39145-1-victor@allwinnertech.com> X-Mailer: git-send-email 2.29.0 MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Spam-Status: No, score=-1.9 required=5.0 tests=BAYES_00,RCVD_IN_DNSWL_NONE, RCVD_IN_MSPIKE_H2,SPF_HELO_NONE,SPF_PASS,UNPARSEABLE_RELAY autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Currently syscore_resume() will stuck on tick_resume(). Fix this by changing `.tick_resume` from sun4i_clkevt_shutdown() to a new function sun4i_tick_resume(). Signed-off-by: Victor Hassan --- drivers/clocksource/timer-sun4i.c | 131 ++++++++++++++++++++++-------- 1 file changed, 96 insertions(+), 35 deletions(-) diff --git a/drivers/clocksource/timer-sun4i.c b/drivers/clocksource/timer-sun4i.c index 94dc6e42e983..574398c35a22 100644 --- a/drivers/clocksource/timer-sun4i.c +++ b/drivers/clocksource/timer-sun4i.c @@ -38,6 +38,19 @@ #define TIMER_SYNC_TICKS 3 +/* Registers which needs to be saved and restored before and after sleeping */ +static u32 sun4i_timer_regs_offset[] = { + TIMER_IRQ_EN_REG, + TIMER_CTL_REG(0), + TIMER_INTVAL_REG(0), + TIMER_CNTVAL_REG(0), + TIMER_CTL_REG(1), + TIMER_INTVAL_REG(1), + TIMER_CNTVAL_REG(1), +}; + +static void __iomem *sun4i_timer_sched_base __read_mostly; + /* * When we disable a timer, we need to wait at least for 2 cycles of * the timer source clock. We will use for that the clocksource timer @@ -79,10 +92,41 @@ static void sun4i_clkevt_time_start(void __iomem *base, u8 timer, base + TIMER_CTL_REG(timer)); } +static inline void sun4i_timer_save_regs(struct timer_of *to) +{ + void __iomem *base = timer_of_base(to); + int i; + u32 *regs_backup = (u32 *)to->private_data; + + for (i = 0; i < ARRAY_SIZE(sun4i_timer_regs_offset); i++) + regs_backup[i] = readl(base + sun4i_timer_regs_offset[i]); +} + +static inline void sun4i_timer_restore_regs(struct timer_of *to) +{ + void __iomem *base = timer_of_base(to); + int i; + u32 *regs_backup = (u32 *)to->private_data; + + for (i = 0; i < ARRAY_SIZE(sun4i_timer_regs_offset); i++) + writel(regs_backup[i], base + sun4i_timer_regs_offset[i]); +} + static int sun4i_clkevt_shutdown(struct clock_event_device *evt) { struct timer_of *to = to_timer_of(evt); + sun4i_timer_save_regs(to); + sun4i_clkevt_time_stop(timer_of_base(to), 0); + + return 0; +} + +static int sun4i_tick_resume(struct clock_event_device *evt) +{ + struct timer_of *to = to_timer_of(evt); + + sun4i_timer_restore_regs(to); sun4i_clkevt_time_stop(timer_of_base(to), 0); return 0; @@ -137,45 +181,54 @@ static irqreturn_t sun4i_timer_interrupt(int irq, void *dev_id) return IRQ_HANDLED; } -static struct timer_of to = { - .flags = TIMER_OF_IRQ | TIMER_OF_CLOCK | TIMER_OF_BASE, - - .clkevt = { - .name = "sun4i_tick", - .rating = 350, - .features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT, - .set_state_shutdown = sun4i_clkevt_shutdown, - .set_state_periodic = sun4i_clkevt_set_periodic, - .set_state_oneshot = sun4i_clkevt_set_oneshot, - .tick_resume = sun4i_clkevt_shutdown, - .set_next_event = sun4i_clkevt_next_event, - .cpumask = cpu_possible_mask, - }, - - .of_irq = { - .handler = sun4i_timer_interrupt, - .flags = IRQF_TIMER | IRQF_IRQPOLL, - }, -}; +static void __init sun4i_clockevent_init(struct timer_of *to) +{ + to->clkevt.name = "sun4i_tick"; + to->clkevt.features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; + to->clkevt.set_state_shutdown = sun4i_clkevt_shutdown; + to->clkevt.set_state_periodic = sun4i_clkevt_set_periodic; + to->clkevt.set_state_oneshot = sun4i_clkevt_set_oneshot; + to->clkevt.tick_resume = sun4i_tick_resume; + to->clkevt.set_next_event = sun4i_clkevt_next_event; + to->clkevt.cpumask = cpu_possible_mask; + to->of_irq.flags = IRQF_TIMER | IRQF_IRQPOLL; + + sun4i_timer_sched_base = timer_of_base(to) + TIMER_CNTVAL_REG(1); +} static u64 notrace sun4i_timer_sched_read(void) { - return ~readl(timer_of_base(&to) + TIMER_CNTVAL_REG(1)); + return (u64)~readl(sun4i_timer_sched_base); } static int __init sun4i_timer_init(struct device_node *node) { + struct timer_of *to; int ret; u32 val; - ret = timer_of_init(node, &to); + to = kzalloc(sizeof(*to), GFP_KERNEL); + if (!to) + return -ENOMEM; + + to->flags = TIMER_OF_IRQ | TIMER_OF_CLOCK | TIMER_OF_BASE; + to->of_irq.handler = sun4i_timer_interrupt; + ret = timer_of_init(node, to); if (ret) - return ret; + goto err; - writel(~0, timer_of_base(&to) + TIMER_INTVAL_REG(1)); + sun4i_clockevent_init(to); + + to->private_data = kcalloc(ARRAY_SIZE(sun4i_timer_regs_offset), sizeof(u32), GFP_KERNEL); + if (!to->private_data) { + ret = -ENOMEM; + goto err1; + } + + writel(~0, timer_of_base(to) + TIMER_INTVAL_REG(1)); writel(TIMER_CTL_ENABLE | TIMER_CTL_RELOAD | TIMER_CTL_CLK_SRC(TIMER_CTL_CLK_SRC_OSC24M), - timer_of_base(&to) + TIMER_CTL_REG(1)); + timer_of_base(to) + TIMER_CTL_REG(1)); /* * sched_clock_register does not have priorities, and on sun6i and @@ -186,32 +239,40 @@ static int __init sun4i_timer_init(struct device_node *node) of_machine_is_compatible("allwinner,sun5i-a10s") || of_machine_is_compatible("allwinner,suniv-f1c100s")) sched_clock_register(sun4i_timer_sched_read, 32, - timer_of_rate(&to)); + timer_of_rate(to)); - ret = clocksource_mmio_init(timer_of_base(&to) + TIMER_CNTVAL_REG(1), - node->name, timer_of_rate(&to), 350, 32, + ret = clocksource_mmio_init(timer_of_base(to) + TIMER_CNTVAL_REG(1), + node->name, timer_of_rate(to), 350, 32, clocksource_mmio_readl_down); if (ret) { pr_err("Failed to register clocksource\n"); - return ret; + goto err2; } writel(TIMER_CTL_CLK_SRC(TIMER_CTL_CLK_SRC_OSC24M), - timer_of_base(&to) + TIMER_CTL_REG(0)); + timer_of_base(to) + TIMER_CTL_REG(0)); /* Make sure timer is stopped before playing with interrupts */ - sun4i_clkevt_time_stop(timer_of_base(&to), 0); + sun4i_clkevt_time_stop(timer_of_base(to), 0); /* clear timer0 interrupt */ - sun4i_timer_clear_interrupt(timer_of_base(&to)); + sun4i_timer_clear_interrupt(timer_of_base(to)); - clockevents_config_and_register(&to.clkevt, timer_of_rate(&to), + clockevents_config_and_register(&to->clkevt, timer_of_rate(to), TIMER_SYNC_TICKS, 0xffffffff); /* Enable timer0 interrupt */ - val = readl(timer_of_base(&to) + TIMER_IRQ_EN_REG); - writel(val | TIMER_IRQ_EN(0), timer_of_base(&to) + TIMER_IRQ_EN_REG); + val = readl(timer_of_base(to) + TIMER_IRQ_EN_REG); + writel(val | TIMER_IRQ_EN(0), timer_of_base(to) + TIMER_IRQ_EN_REG); + + return ret; +err2: + kfree(to->private_data); +err1: + timer_of_cleanup(to); +err: + kfree(to); return ret; } TIMER_OF_DECLARE(sun4i, "allwinner,sun4i-a10-timer", -- 2.29.0