forked from torvalds/linux
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge tag 'timers-core-2023-04-28' of git://git.kernel.org/pub/scm/li…
…nux/kernel/git/tip/tip Pull more timer updates from Thomas Gleixner: "Timekeeping and clocksource/event driver updates the second batch: - A trivial documentation fix in the timekeeping core - A really boring set of small fixes, enhancements and cleanups in the drivers code. No new clocksource/clockevent drivers for a change" * tag 'timers-core-2023-04-28' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: timekeeping: Fix references to nonexistent ktime_get_fast_ns() dt-bindings: timer: rockchip: Add rk3588 compatible dt-bindings: timer: rockchip: Drop superfluous rk3288 compatible clocksource/drivers/ti: Use of_property_read_bool() for boolean properties clocksource/drivers/timer-ti-dm: Fix finding alwon timer clocksource/drivers/davinci: Fix memory leak in davinci_timer_register when init fails clocksource/drivers/stm32-lp: Drop of_match_ptr for ID table clocksource/drivers/timer-ti-dm: Convert to platform remove callback returning void clocksource/drivers/timer-tegra186: Convert to platform remove callback returning void clocksource/drivers/timer-ti-dm: Improve error message in .remove clocksource/drivers/timer-stm32-lp: Mark driver as non-removable clocksource/drivers/sh_mtu2: Mark driver as non-removable clocksource/drivers/timer-ti-dm: Use of_address_to_resource() clocksource/drivers/timer-imx-gpt: Remove non-DT function clocksource/drivers/timer-mediatek: Split out CPUXGPT timers clocksource/drivers/exynos_mct: Explicitly return 0 for shared timer
- Loading branch information
Showing
15 changed files
with
223 additions
and
207 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,140 @@ | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
/* | ||
* MediaTek SoCs CPUX General Purpose Timer handling | ||
* | ||
* Based on timer-mediatek.c: | ||
* Copyright (C) 2014 Matthias Brugger <[email protected]> | ||
* | ||
* Copyright (C) 2022 Collabora Ltd. | ||
* AngeloGioacchino Del Regno <[email protected]> | ||
*/ | ||
|
||
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||
|
||
#include <linux/clockchips.h> | ||
#include <linux/clocksource.h> | ||
#include <linux/interrupt.h> | ||
#include <linux/irqreturn.h> | ||
#include <linux/sched_clock.h> | ||
#include <linux/slab.h> | ||
#include "timer-of.h" | ||
|
||
#define TIMER_SYNC_TICKS 3 | ||
|
||
/* cpux mcusys wrapper */ | ||
#define CPUX_CON_REG 0x0 | ||
#define CPUX_IDX_REG 0x4 | ||
|
||
/* cpux */ | ||
#define CPUX_IDX_GLOBAL_CTRL 0x0 | ||
#define CPUX_ENABLE BIT(0) | ||
#define CPUX_CLK_DIV_MASK GENMASK(10, 8) | ||
#define CPUX_CLK_DIV1 BIT(8) | ||
#define CPUX_CLK_DIV2 BIT(9) | ||
#define CPUX_CLK_DIV4 BIT(10) | ||
#define CPUX_IDX_GLOBAL_IRQ 0x30 | ||
|
||
static u32 mtk_cpux_readl(u32 reg_idx, struct timer_of *to) | ||
{ | ||
writel(reg_idx, timer_of_base(to) + CPUX_IDX_REG); | ||
return readl(timer_of_base(to) + CPUX_CON_REG); | ||
} | ||
|
||
static void mtk_cpux_writel(u32 val, u32 reg_idx, struct timer_of *to) | ||
{ | ||
writel(reg_idx, timer_of_base(to) + CPUX_IDX_REG); | ||
writel(val, timer_of_base(to) + CPUX_CON_REG); | ||
} | ||
|
||
static void mtk_cpux_set_irq(struct timer_of *to, bool enable) | ||
{ | ||
const unsigned long *irq_mask = cpumask_bits(cpu_possible_mask); | ||
u32 val; | ||
|
||
val = mtk_cpux_readl(CPUX_IDX_GLOBAL_IRQ, to); | ||
|
||
if (enable) | ||
val |= *irq_mask; | ||
else | ||
val &= ~(*irq_mask); | ||
|
||
mtk_cpux_writel(val, CPUX_IDX_GLOBAL_IRQ, to); | ||
} | ||
|
||
static int mtk_cpux_clkevt_shutdown(struct clock_event_device *clkevt) | ||
{ | ||
/* Clear any irq */ | ||
mtk_cpux_set_irq(to_timer_of(clkevt), false); | ||
|
||
/* | ||
* Disabling CPUXGPT timer will crash the platform, especially | ||
* if Trusted Firmware is using it (usually, for sleep states), | ||
* so we only mask the IRQ and call it a day. | ||
*/ | ||
return 0; | ||
} | ||
|
||
static int mtk_cpux_clkevt_resume(struct clock_event_device *clkevt) | ||
{ | ||
mtk_cpux_set_irq(to_timer_of(clkevt), true); | ||
return 0; | ||
} | ||
|
||
static struct timer_of to = { | ||
/* | ||
* There are per-cpu interrupts for the CPUX General Purpose Timer | ||
* but since this timer feeds the AArch64 System Timer we can rely | ||
* on the CPU timer PPIs as well, so we don't declare TIMER_OF_IRQ. | ||
*/ | ||
.flags = TIMER_OF_BASE | TIMER_OF_CLOCK, | ||
|
||
.clkevt = { | ||
.name = "mtk-cpuxgpt", | ||
.cpumask = cpu_possible_mask, | ||
.rating = 10, | ||
.set_state_shutdown = mtk_cpux_clkevt_shutdown, | ||
.tick_resume = mtk_cpux_clkevt_resume, | ||
}, | ||
}; | ||
|
||
static int __init mtk_cpux_init(struct device_node *node) | ||
{ | ||
u32 freq, val; | ||
int ret; | ||
|
||
/* If this fails, bad things are about to happen... */ | ||
ret = timer_of_init(node, &to); | ||
if (ret) { | ||
WARN(1, "Cannot start CPUX timers.\n"); | ||
return ret; | ||
} | ||
|
||
/* | ||
* Check if we're given a clock with the right frequency for this | ||
* timer, otherwise warn but keep going with the setup anyway, as | ||
* that makes it possible to still boot the kernel, even though | ||
* it may not work correctly (random lockups, etc). | ||
* The reason behind this is that having an early UART may not be | ||
* possible for everyone and this gives a chance to retrieve kmsg | ||
* for eventual debugging even on consumer devices. | ||
*/ | ||
freq = timer_of_rate(&to); | ||
if (freq > 13000000) | ||
WARN(1, "Requested unsupported timer frequency %u\n", freq); | ||
|
||
/* Clock input is 26MHz, set DIV2 to achieve 13MHz clock */ | ||
val = mtk_cpux_readl(CPUX_IDX_GLOBAL_CTRL, &to); | ||
val &= ~CPUX_CLK_DIV_MASK; | ||
val |= CPUX_CLK_DIV2; | ||
mtk_cpux_writel(val, CPUX_IDX_GLOBAL_CTRL, &to); | ||
|
||
/* Enable all CPUXGPT timers */ | ||
val = mtk_cpux_readl(CPUX_IDX_GLOBAL_CTRL, &to); | ||
mtk_cpux_writel(val | CPUX_ENABLE, CPUX_IDX_GLOBAL_CTRL, &to); | ||
|
||
clockevents_config_and_register(&to.clkevt, timer_of_rate(&to), | ||
TIMER_SYNC_TICKS, 0xffffffff); | ||
|
||
return 0; | ||
} | ||
TIMER_OF_DECLARE(mtk_mt6795, "mediatek,mt6795-systimer", mtk_cpux_init); |
Oops, something went wrong.