1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * MediaTek SoCs CPUX General Purpose Timer handling
5 * Based on timer-mediatek.c:
6 * Copyright (C) 2014 Matthias Brugger <matthias.bgg@gmail.com>
8 * Copyright (C) 2022 Collabora Ltd.
9 * AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>
12 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
14 #include <linux/clockchips.h>
15 #include <linux/clocksource.h>
16 #include <linux/interrupt.h>
17 #include <linux/irqreturn.h>
18 #include <linux/sched_clock.h>
19 #include <linux/slab.h>
22 #define TIMER_SYNC_TICKS 3
24 /* cpux mcusys wrapper */
25 #define CPUX_CON_REG 0x0
26 #define CPUX_IDX_REG 0x4
29 #define CPUX_IDX_GLOBAL_CTRL 0x0
30 #define CPUX_ENABLE BIT(0)
31 #define CPUX_CLK_DIV_MASK GENMASK(10, 8)
32 #define CPUX_CLK_DIV1 BIT(8)
33 #define CPUX_CLK_DIV2 BIT(9)
34 #define CPUX_CLK_DIV4 BIT(10)
35 #define CPUX_IDX_GLOBAL_IRQ 0x30
37 static u32
mtk_cpux_readl(u32 reg_idx
, struct timer_of
*to
)
39 writel(reg_idx
, timer_of_base(to
) + CPUX_IDX_REG
);
40 return readl(timer_of_base(to
) + CPUX_CON_REG
);
43 static void mtk_cpux_writel(u32 val
, u32 reg_idx
, struct timer_of
*to
)
45 writel(reg_idx
, timer_of_base(to
) + CPUX_IDX_REG
);
46 writel(val
, timer_of_base(to
) + CPUX_CON_REG
);
49 static void mtk_cpux_set_irq(struct timer_of
*to
, bool enable
)
51 const unsigned long *irq_mask
= cpumask_bits(cpu_possible_mask
);
54 val
= mtk_cpux_readl(CPUX_IDX_GLOBAL_IRQ
, to
);
61 mtk_cpux_writel(val
, CPUX_IDX_GLOBAL_IRQ
, to
);
64 static int mtk_cpux_clkevt_shutdown(struct clock_event_device
*clkevt
)
67 mtk_cpux_set_irq(to_timer_of(clkevt
), false);
70 * Disabling CPUXGPT timer will crash the platform, especially
71 * if Trusted Firmware is using it (usually, for sleep states),
72 * so we only mask the IRQ and call it a day.
77 static int mtk_cpux_clkevt_resume(struct clock_event_device
*clkevt
)
79 mtk_cpux_set_irq(to_timer_of(clkevt
), true);
83 static struct timer_of to
= {
85 * There are per-cpu interrupts for the CPUX General Purpose Timer
86 * but since this timer feeds the AArch64 System Timer we can rely
87 * on the CPU timer PPIs as well, so we don't declare TIMER_OF_IRQ.
89 .flags
= TIMER_OF_BASE
| TIMER_OF_CLOCK
,
92 .name
= "mtk-cpuxgpt",
93 .cpumask
= cpu_possible_mask
,
95 .set_state_shutdown
= mtk_cpux_clkevt_shutdown
,
96 .tick_resume
= mtk_cpux_clkevt_resume
,
100 static int __init
mtk_cpux_init(struct device_node
*node
)
105 /* If this fails, bad things are about to happen... */
106 ret
= timer_of_init(node
, &to
);
108 WARN(1, "Cannot start CPUX timers.\n");
113 * Check if we're given a clock with the right frequency for this
114 * timer, otherwise warn but keep going with the setup anyway, as
115 * that makes it possible to still boot the kernel, even though
116 * it may not work correctly (random lockups, etc).
117 * The reason behind this is that having an early UART may not be
118 * possible for everyone and this gives a chance to retrieve kmsg
119 * for eventual debugging even on consumer devices.
121 freq
= timer_of_rate(&to
);
123 WARN(1, "Requested unsupported timer frequency %u\n", freq
);
125 /* Clock input is 26MHz, set DIV2 to achieve 13MHz clock */
126 val
= mtk_cpux_readl(CPUX_IDX_GLOBAL_CTRL
, &to
);
127 val
&= ~CPUX_CLK_DIV_MASK
;
128 val
|= CPUX_CLK_DIV2
;
129 mtk_cpux_writel(val
, CPUX_IDX_GLOBAL_CTRL
, &to
);
131 /* Enable all CPUXGPT timers */
132 val
= mtk_cpux_readl(CPUX_IDX_GLOBAL_CTRL
, &to
);
133 mtk_cpux_writel(val
| CPUX_ENABLE
, CPUX_IDX_GLOBAL_CTRL
, &to
);
135 clockevents_config_and_register(&to
.clkevt
, timer_of_rate(&to
),
136 TIMER_SYNC_TICKS
, 0xffffffff);
140 TIMER_OF_DECLARE(mtk_mt6795
, "mediatek,mt6795-systimer", mtk_cpux_init
);