Received: by 2002:a05:6359:c8b:b0:c7:702f:21d4 with SMTP id go11csp279695rwb; Wed, 28 Sep 2022 02:34:43 -0700 (PDT) X-Google-Smtp-Source: AMsMyM4ocsaUOxSW6sbKkQlABRtrKnbjmr7xTL2am8+AKqeeze9EZR7K75CqbmBYcYaKPj8KwgR7 X-Received: by 2002:a05:6a00:15c9:b0:541:1767:4ce2 with SMTP id o9-20020a056a0015c900b0054117674ce2mr33096092pfu.30.1664357682949; Wed, 28 Sep 2022 02:34:42 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1664357682; cv=none; d=google.com; s=arc-20160816; b=y6HfntmsAxnI5TpDW5Znc5oq01l2Pr5/kvmP4sD4VQDnpau+k1qDue52dfc/mMQCeL CFgKfxLJRSSYWpqeZThy+ek22Tonxsf2le9UpwwAC88GmcZraD3ksaVh17MkoGSOq53V s80Ese0SsRgg2kC6UhNk8YypkLWeW26utdLt/j2C9SkSdAA2LINV2c3KEm2EMH0JyyDg aagJ7ch87dIJQtzsAQNW6A3JzJtvEhwxP/94vdWzI2cbnSy0L6fGXZn9VbomEX3/j61a tQUTBZmmxlg+tt7Tj61X8p8UNc9CE/tlCNNZrIZaVrjCmnjfgyeQIGg1p9D/Rb9MdHaB 1T1g== 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=ditGqFlio8Krt96SDGahZShh8JySlnDTLOvaKcmJcvSPQATTfDdKW8+P2EHEb/aQOs KpTxeH1OciVs+WuLYtuCHt1NejZEmW8gpKJFWpn7ryqaKBE4ur7PcYkqhDji4+kKY1a9 oTujsZWOcOkwhAJzCzPPa7CO7ccGZwEn3b2hALEvByidr4/z2WZ/HNgq05+/uHne9VkZ AAZKb8ehRGBisf5jvsM+dJ9X/vwniOD0ePgLGBg7kWprDHNnLqqBM+VeomBXIs0s9edl 1kQ9rrXs4osolh57FvZ9D8KJM4WrpzptCPa4Fb8QUXXF5FnRT2vl6xbWdrPXEhGPk3cY 5v+w== 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 t64-20020a638143000000b00439286e51a2si4754240pgd.155.2022.09.28.02.34.31; Wed, 28 Sep 2022 02:34:42 -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 S232660AbiI1IuB (ORCPT + 99 others); Wed, 28 Sep 2022 04:50:01 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:46010 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S233716AbiI1ItX (ORCPT ); Wed, 28 Sep 2022 04:49:23 -0400 Received: from out28-3.mail.aliyun.com (out28-3.mail.aliyun.com [115.124.28.3]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 599521BEB9 for ; Wed, 28 Sep 2022 01:49:07 -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=ay29a033018047187;MF=victor@allwinnertech.com;NM=1;PH=DS;RN=8;RT=8;SR=0;TI=SMTPD_---.PQPBNkb_1664354942; Received: from SunxiBot.allwinnertech.com(mailfrom:victor@allwinnertech.com fp:SMTPD_---.PQPBNkb_1664354942) by smtp.aliyun-inc.com; Wed, 28 Sep 2022 16:49:03 +0800 From: Victor Hassan To: daniel.lezcano@linaro.org, tglx@linutronix.de, wens@csie.org, jernej.skrabec@gmail.com, samuel@sholland.org 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: Wed, 28 Sep 2022 16:48:57 +0800 Message-Id: <20220928084857.49185-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