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