191d59bdfSNeil Armstrong // SPDX-License-Identifier: GPL-2.0
2adab4ec3SDaniel Lezcano /*
3adab4ec3SDaniel Lezcano * Amlogic Meson6 SoCs timer handling.
4adab4ec3SDaniel Lezcano *
5adab4ec3SDaniel Lezcano * Copyright (C) 2014 Carlo Caione <[email protected]>
6adab4ec3SDaniel Lezcano *
7adab4ec3SDaniel Lezcano * Based on code from Amlogic, Inc
8adab4ec3SDaniel Lezcano */
9adab4ec3SDaniel Lezcano
10adab4ec3SDaniel Lezcano #include <linux/bitfield.h>
11adab4ec3SDaniel Lezcano #include <linux/bitops.h>
12adab4ec3SDaniel Lezcano #include <linux/clk.h>
13adab4ec3SDaniel Lezcano #include <linux/clockchips.h>
14adab4ec3SDaniel Lezcano #include <linux/interrupt.h>
15adab4ec3SDaniel Lezcano #include <linux/irq.h>
16adab4ec3SDaniel Lezcano #include <linux/irqreturn.h>
17adab4ec3SDaniel Lezcano #include <linux/sched_clock.h>
18adab4ec3SDaniel Lezcano #include <linux/of.h>
19adab4ec3SDaniel Lezcano #include <linux/of_address.h>
20adab4ec3SDaniel Lezcano #include <linux/of_irq.h>
21adab4ec3SDaniel Lezcano
22adab4ec3SDaniel Lezcano #ifdef CONFIG_ARM
23adab4ec3SDaniel Lezcano #include <linux/delay.h>
24adab4ec3SDaniel Lezcano #endif
25adab4ec3SDaniel Lezcano
26adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX 0x00
27adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERD_EN BIT(19)
28adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERC_EN BIT(18)
29adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERB_EN BIT(17)
30adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERA_EN BIT(16)
31adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERD_MODE BIT(15)
32adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERC_MODE BIT(14)
33adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERB_MODE BIT(13)
34adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERA_MODE BIT(12)
35adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_MASK GENMASK(10, 8)
36adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_SYSTEM_CLOCK 0x0
37adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_1US 0x1
38adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_10US 0x2
39adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_100US 0x3
40adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_1MS 0x4
41adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERD_INPUT_CLOCK_MASK GENMASK(7, 6)
42adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERC_INPUT_CLOCK_MASK GENMASK(5, 4)
43adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERB_INPUT_CLOCK_MASK GENMASK(3, 2)
44adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERA_INPUT_CLOCK_MASK GENMASK(1, 0)
45adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERABCD_INPUT_CLOCK_1US 0x0
46adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERABCD_INPUT_CLOCK_10US 0x1
47adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERABCD_INPUT_CLOCK_100US 0x0
48adab4ec3SDaniel Lezcano #define MESON_ISA_TIMER_MUX_TIMERABCD_INPUT_CLOCK_1MS 0x3
49adab4ec3SDaniel Lezcano
50adab4ec3SDaniel Lezcano #define MESON_ISA_TIMERA 0x04
51adab4ec3SDaniel Lezcano #define MESON_ISA_TIMERB 0x08
52adab4ec3SDaniel Lezcano #define MESON_ISA_TIMERC 0x0c
53adab4ec3SDaniel Lezcano #define MESON_ISA_TIMERD 0x10
54adab4ec3SDaniel Lezcano #define MESON_ISA_TIMERE 0x14
55adab4ec3SDaniel Lezcano
56adab4ec3SDaniel Lezcano static void __iomem *timer_base;
57adab4ec3SDaniel Lezcano
58adab4ec3SDaniel Lezcano #ifdef CONFIG_ARM
meson6_read_current_timer(void)59adab4ec3SDaniel Lezcano static unsigned long meson6_read_current_timer(void)
60adab4ec3SDaniel Lezcano {
61adab4ec3SDaniel Lezcano return readl_relaxed(timer_base + MESON_ISA_TIMERE);
62adab4ec3SDaniel Lezcano }
63adab4ec3SDaniel Lezcano
64adab4ec3SDaniel Lezcano static struct delay_timer meson6_delay_timer = {
65adab4ec3SDaniel Lezcano .read_current_timer = meson6_read_current_timer,
66adab4ec3SDaniel Lezcano .freq = 1000 * 1000,
67adab4ec3SDaniel Lezcano };
68adab4ec3SDaniel Lezcano #endif
69adab4ec3SDaniel Lezcano
meson6_timer_sched_read(void)70adab4ec3SDaniel Lezcano static u64 notrace meson6_timer_sched_read(void)
71adab4ec3SDaniel Lezcano {
72adab4ec3SDaniel Lezcano return (u64)readl(timer_base + MESON_ISA_TIMERE);
73adab4ec3SDaniel Lezcano }
74adab4ec3SDaniel Lezcano
meson6_clkevt_time_stop(void)75adab4ec3SDaniel Lezcano static void meson6_clkevt_time_stop(void)
76adab4ec3SDaniel Lezcano {
77adab4ec3SDaniel Lezcano u32 val = readl(timer_base + MESON_ISA_TIMER_MUX);
78adab4ec3SDaniel Lezcano
79adab4ec3SDaniel Lezcano writel(val & ~MESON_ISA_TIMER_MUX_TIMERA_EN,
80adab4ec3SDaniel Lezcano timer_base + MESON_ISA_TIMER_MUX);
81adab4ec3SDaniel Lezcano }
82adab4ec3SDaniel Lezcano
meson6_clkevt_time_setup(unsigned long delay)83adab4ec3SDaniel Lezcano static void meson6_clkevt_time_setup(unsigned long delay)
84adab4ec3SDaniel Lezcano {
85adab4ec3SDaniel Lezcano writel(delay, timer_base + MESON_ISA_TIMERA);
86adab4ec3SDaniel Lezcano }
87adab4ec3SDaniel Lezcano
meson6_clkevt_time_start(bool periodic)88adab4ec3SDaniel Lezcano static void meson6_clkevt_time_start(bool periodic)
89adab4ec3SDaniel Lezcano {
90adab4ec3SDaniel Lezcano u32 val = readl(timer_base + MESON_ISA_TIMER_MUX);
91adab4ec3SDaniel Lezcano
92adab4ec3SDaniel Lezcano if (periodic)
93adab4ec3SDaniel Lezcano val |= MESON_ISA_TIMER_MUX_TIMERA_MODE;
94adab4ec3SDaniel Lezcano else
95adab4ec3SDaniel Lezcano val &= ~MESON_ISA_TIMER_MUX_TIMERA_MODE;
96adab4ec3SDaniel Lezcano
97adab4ec3SDaniel Lezcano writel(val | MESON_ISA_TIMER_MUX_TIMERA_EN,
98adab4ec3SDaniel Lezcano timer_base + MESON_ISA_TIMER_MUX);
99adab4ec3SDaniel Lezcano }
100adab4ec3SDaniel Lezcano
meson6_shutdown(struct clock_event_device * evt)101adab4ec3SDaniel Lezcano static int meson6_shutdown(struct clock_event_device *evt)
102adab4ec3SDaniel Lezcano {
103adab4ec3SDaniel Lezcano meson6_clkevt_time_stop();
104adab4ec3SDaniel Lezcano return 0;
105adab4ec3SDaniel Lezcano }
106adab4ec3SDaniel Lezcano
meson6_set_oneshot(struct clock_event_device * evt)107adab4ec3SDaniel Lezcano static int meson6_set_oneshot(struct clock_event_device *evt)
108adab4ec3SDaniel Lezcano {
109adab4ec3SDaniel Lezcano meson6_clkevt_time_stop();
110adab4ec3SDaniel Lezcano meson6_clkevt_time_start(false);
111adab4ec3SDaniel Lezcano return 0;
112adab4ec3SDaniel Lezcano }
113adab4ec3SDaniel Lezcano
meson6_set_periodic(struct clock_event_device * evt)114adab4ec3SDaniel Lezcano static int meson6_set_periodic(struct clock_event_device *evt)
115adab4ec3SDaniel Lezcano {
116adab4ec3SDaniel Lezcano meson6_clkevt_time_stop();
117adab4ec3SDaniel Lezcano meson6_clkevt_time_setup(USEC_PER_SEC / HZ - 1);
118adab4ec3SDaniel Lezcano meson6_clkevt_time_start(true);
119adab4ec3SDaniel Lezcano return 0;
120adab4ec3SDaniel Lezcano }
121adab4ec3SDaniel Lezcano
meson6_clkevt_next_event(unsigned long evt,struct clock_event_device * unused)122adab4ec3SDaniel Lezcano static int meson6_clkevt_next_event(unsigned long evt,
123adab4ec3SDaniel Lezcano struct clock_event_device *unused)
124adab4ec3SDaniel Lezcano {
125adab4ec3SDaniel Lezcano meson6_clkevt_time_stop();
126adab4ec3SDaniel Lezcano meson6_clkevt_time_setup(evt);
127adab4ec3SDaniel Lezcano meson6_clkevt_time_start(false);
128adab4ec3SDaniel Lezcano
129adab4ec3SDaniel Lezcano return 0;
130adab4ec3SDaniel Lezcano }
131adab4ec3SDaniel Lezcano
132adab4ec3SDaniel Lezcano static struct clock_event_device meson6_clockevent = {
133adab4ec3SDaniel Lezcano .name = "meson6_tick",
134adab4ec3SDaniel Lezcano .rating = 400,
135adab4ec3SDaniel Lezcano .features = CLOCK_EVT_FEAT_PERIODIC |
136adab4ec3SDaniel Lezcano CLOCK_EVT_FEAT_ONESHOT,
137adab4ec3SDaniel Lezcano .set_state_shutdown = meson6_shutdown,
138adab4ec3SDaniel Lezcano .set_state_periodic = meson6_set_periodic,
139adab4ec3SDaniel Lezcano .set_state_oneshot = meson6_set_oneshot,
140adab4ec3SDaniel Lezcano .tick_resume = meson6_shutdown,
141adab4ec3SDaniel Lezcano .set_next_event = meson6_clkevt_next_event,
142adab4ec3SDaniel Lezcano };
143adab4ec3SDaniel Lezcano
meson6_timer_interrupt(int irq,void * dev_id)144adab4ec3SDaniel Lezcano static irqreturn_t meson6_timer_interrupt(int irq, void *dev_id)
145adab4ec3SDaniel Lezcano {
146adab4ec3SDaniel Lezcano struct clock_event_device *evt = (struct clock_event_device *)dev_id;
147adab4ec3SDaniel Lezcano
148adab4ec3SDaniel Lezcano evt->event_handler(evt);
149adab4ec3SDaniel Lezcano
150adab4ec3SDaniel Lezcano return IRQ_HANDLED;
151adab4ec3SDaniel Lezcano }
152adab4ec3SDaniel Lezcano
meson6_timer_init(struct device_node * node)153adab4ec3SDaniel Lezcano static int __init meson6_timer_init(struct device_node *node)
154adab4ec3SDaniel Lezcano {
155adab4ec3SDaniel Lezcano u32 val;
156adab4ec3SDaniel Lezcano int ret, irq;
157adab4ec3SDaniel Lezcano
158adab4ec3SDaniel Lezcano timer_base = of_io_request_and_map(node, 0, "meson6-timer");
159adab4ec3SDaniel Lezcano if (IS_ERR(timer_base)) {
160adab4ec3SDaniel Lezcano pr_err("Can't map registers\n");
161adab4ec3SDaniel Lezcano return -ENXIO;
162adab4ec3SDaniel Lezcano }
163adab4ec3SDaniel Lezcano
164adab4ec3SDaniel Lezcano irq = irq_of_parse_and_map(node, 0);
165adab4ec3SDaniel Lezcano if (irq <= 0) {
166adab4ec3SDaniel Lezcano pr_err("Can't parse IRQ\n");
167adab4ec3SDaniel Lezcano return -EINVAL;
168adab4ec3SDaniel Lezcano }
169adab4ec3SDaniel Lezcano
170adab4ec3SDaniel Lezcano /* Set 1us for timer E */
171adab4ec3SDaniel Lezcano val = readl(timer_base + MESON_ISA_TIMER_MUX);
172adab4ec3SDaniel Lezcano val &= ~MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_MASK;
173adab4ec3SDaniel Lezcano val |= FIELD_PREP(MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_MASK,
174adab4ec3SDaniel Lezcano MESON_ISA_TIMER_MUX_TIMERE_INPUT_CLOCK_1US);
175adab4ec3SDaniel Lezcano writel(val, timer_base + MESON_ISA_TIMER_MUX);
176adab4ec3SDaniel Lezcano
177adab4ec3SDaniel Lezcano sched_clock_register(meson6_timer_sched_read, 32, USEC_PER_SEC);
178adab4ec3SDaniel Lezcano clocksource_mmio_init(timer_base + MESON_ISA_TIMERE, node->name,
179adab4ec3SDaniel Lezcano 1000 * 1000, 300, 32, clocksource_mmio_readl_up);
180adab4ec3SDaniel Lezcano
181adab4ec3SDaniel Lezcano /* Timer A base 1us */
182adab4ec3SDaniel Lezcano val &= ~MESON_ISA_TIMER_MUX_TIMERA_INPUT_CLOCK_MASK;
183adab4ec3SDaniel Lezcano val |= FIELD_PREP(MESON_ISA_TIMER_MUX_TIMERA_INPUT_CLOCK_MASK,
184adab4ec3SDaniel Lezcano MESON_ISA_TIMER_MUX_TIMERABCD_INPUT_CLOCK_1US);
185adab4ec3SDaniel Lezcano writel(val, timer_base + MESON_ISA_TIMER_MUX);
186adab4ec3SDaniel Lezcano
187adab4ec3SDaniel Lezcano /* Stop the timer A */
188adab4ec3SDaniel Lezcano meson6_clkevt_time_stop();
189adab4ec3SDaniel Lezcano
190*cc2550b4Safzal mohammed ret = request_irq(irq, meson6_timer_interrupt,
191*cc2550b4Safzal mohammed IRQF_TIMER | IRQF_IRQPOLL, "meson6_timer",
192*cc2550b4Safzal mohammed &meson6_clockevent);
193adab4ec3SDaniel Lezcano if (ret) {
194adab4ec3SDaniel Lezcano pr_warn("failed to setup irq %d\n", irq);
195adab4ec3SDaniel Lezcano return ret;
196adab4ec3SDaniel Lezcano }
197adab4ec3SDaniel Lezcano
198adab4ec3SDaniel Lezcano meson6_clockevent.cpumask = cpu_possible_mask;
199adab4ec3SDaniel Lezcano meson6_clockevent.irq = irq;
200adab4ec3SDaniel Lezcano
201adab4ec3SDaniel Lezcano clockevents_config_and_register(&meson6_clockevent, USEC_PER_SEC,
202adab4ec3SDaniel Lezcano 1, 0xfffe);
203adab4ec3SDaniel Lezcano
204adab4ec3SDaniel Lezcano #ifdef CONFIG_ARM
205adab4ec3SDaniel Lezcano /* Also use MESON_ISA_TIMERE for delays */
206adab4ec3SDaniel Lezcano register_current_timer_delay(&meson6_delay_timer);
207adab4ec3SDaniel Lezcano #endif
208adab4ec3SDaniel Lezcano
209adab4ec3SDaniel Lezcano return 0;
210adab4ec3SDaniel Lezcano }
211adab4ec3SDaniel Lezcano TIMER_OF_DECLARE(meson6, "amlogic,meson6-timer",
212adab4ec3SDaniel Lezcano meson6_timer_init);
213