1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (C) 2013 Pengutronix
4 * Uwe Kleine-Koenig <u.kleine-koenig@pengutronix.de>
7 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
9 #include <linux/kernel.h>
10 #include <linux/clocksource.h>
11 #include <linux/clockchips.h>
12 #include <linux/irq.h>
13 #include <linux/interrupt.h>
15 #include <linux/of_address.h>
16 #include <linux/of_irq.h>
17 #include <linux/clk.h>
19 #define TIMERn_CTRL 0x00
20 #define TIMERn_CTRL_PRESC(val) (((val) & 0xf) << 24)
21 #define TIMERn_CTRL_PRESC_1024 TIMERn_CTRL_PRESC(10)
22 #define TIMERn_CTRL_CLKSEL(val) (((val) & 0x3) << 16)
23 #define TIMERn_CTRL_CLKSEL_PRESCHFPERCLK TIMERn_CTRL_CLKSEL(0)
24 #define TIMERn_CTRL_OSMEN 0x00000010
25 #define TIMERn_CTRL_MODE(val) (((val) & 0x3) << 0)
26 #define TIMERn_CTRL_MODE_UP TIMERn_CTRL_MODE(0)
27 #define TIMERn_CTRL_MODE_DOWN TIMERn_CTRL_MODE(1)
29 #define TIMERn_CMD 0x04
30 #define TIMERn_CMD_START 0x00000001
31 #define TIMERn_CMD_STOP 0x00000002
33 #define TIMERn_IEN 0x0c
34 #define TIMERn_IF 0x10
35 #define TIMERn_IFS 0x14
36 #define TIMERn_IFC 0x18
37 #define TIMERn_IRQ_UF 0x00000002
39 #define TIMERn_TOP 0x1c
40 #define TIMERn_CNT 0x24
42 struct efm32_clock_event_ddata
{
43 struct clock_event_device evtdev
;
45 unsigned periodic_top
;
48 static int efm32_clock_event_shutdown(struct clock_event_device
*evtdev
)
50 struct efm32_clock_event_ddata
*ddata
=
51 container_of(evtdev
, struct efm32_clock_event_ddata
, evtdev
);
53 writel_relaxed(TIMERn_CMD_STOP
, ddata
->base
+ TIMERn_CMD
);
57 static int efm32_clock_event_set_oneshot(struct clock_event_device
*evtdev
)
59 struct efm32_clock_event_ddata
*ddata
=
60 container_of(evtdev
, struct efm32_clock_event_ddata
, evtdev
);
62 writel_relaxed(TIMERn_CMD_STOP
, ddata
->base
+ TIMERn_CMD
);
63 writel_relaxed(TIMERn_CTRL_PRESC_1024
|
64 TIMERn_CTRL_CLKSEL_PRESCHFPERCLK
|
66 TIMERn_CTRL_MODE_DOWN
,
67 ddata
->base
+ TIMERn_CTRL
);
71 static int efm32_clock_event_set_periodic(struct clock_event_device
*evtdev
)
73 struct efm32_clock_event_ddata
*ddata
=
74 container_of(evtdev
, struct efm32_clock_event_ddata
, evtdev
);
76 writel_relaxed(TIMERn_CMD_STOP
, ddata
->base
+ TIMERn_CMD
);
77 writel_relaxed(ddata
->periodic_top
, ddata
->base
+ TIMERn_TOP
);
78 writel_relaxed(TIMERn_CTRL_PRESC_1024
|
79 TIMERn_CTRL_CLKSEL_PRESCHFPERCLK
|
80 TIMERn_CTRL_MODE_DOWN
,
81 ddata
->base
+ TIMERn_CTRL
);
82 writel_relaxed(TIMERn_CMD_START
, ddata
->base
+ TIMERn_CMD
);
86 static int efm32_clock_event_set_next_event(unsigned long evt
,
87 struct clock_event_device
*evtdev
)
89 struct efm32_clock_event_ddata
*ddata
=
90 container_of(evtdev
, struct efm32_clock_event_ddata
, evtdev
);
92 writel_relaxed(TIMERn_CMD_STOP
, ddata
->base
+ TIMERn_CMD
);
93 writel_relaxed(evt
, ddata
->base
+ TIMERn_CNT
);
94 writel_relaxed(TIMERn_CMD_START
, ddata
->base
+ TIMERn_CMD
);
99 static irqreturn_t
efm32_clock_event_handler(int irq
, void *dev_id
)
101 struct efm32_clock_event_ddata
*ddata
= dev_id
;
103 writel_relaxed(TIMERn_IRQ_UF
, ddata
->base
+ TIMERn_IFC
);
105 ddata
->evtdev
.event_handler(&ddata
->evtdev
);
110 static struct efm32_clock_event_ddata clock_event_ddata
= {
112 .name
= "efm32 clockevent",
113 .features
= CLOCK_EVT_FEAT_ONESHOT
| CLOCK_EVT_FEAT_PERIODIC
,
114 .set_state_shutdown
= efm32_clock_event_shutdown
,
115 .set_state_periodic
= efm32_clock_event_set_periodic
,
116 .set_state_oneshot
= efm32_clock_event_set_oneshot
,
117 .set_next_event
= efm32_clock_event_set_next_event
,
122 static int __init
efm32_clocksource_init(struct device_node
*np
)
129 clk
= of_clk_get(np
, 0);
132 pr_err("failed to get clock for clocksource (%d)\n", ret
);
136 ret
= clk_prepare_enable(clk
);
138 pr_err("failed to enable timer clock for clocksource (%d)\n",
142 rate
= clk_get_rate(clk
);
144 base
= of_iomap(np
, 0);
146 ret
= -EADDRNOTAVAIL
;
147 pr_err("failed to map registers for clocksource\n");
151 writel_relaxed(TIMERn_CTRL_PRESC_1024
|
152 TIMERn_CTRL_CLKSEL_PRESCHFPERCLK
|
153 TIMERn_CTRL_MODE_UP
, base
+ TIMERn_CTRL
);
154 writel_relaxed(TIMERn_CMD_START
, base
+ TIMERn_CMD
);
156 ret
= clocksource_mmio_init(base
+ TIMERn_CNT
, "efm32 timer",
157 DIV_ROUND_CLOSEST(rate
, 1024), 200, 16,
158 clocksource_mmio_readl_up
);
160 pr_err("failed to init clocksource (%d)\n", ret
);
161 goto err_clocksource_init
;
166 err_clocksource_init
:
171 clk_disable_unprepare(clk
);
180 static int __init
efm32_clockevent_init(struct device_node
*np
)
188 clk
= of_clk_get(np
, 0);
191 pr_err("failed to get clock for clockevent (%d)\n", ret
);
195 ret
= clk_prepare_enable(clk
);
197 pr_err("failed to enable timer clock for clockevent (%d)\n",
201 rate
= clk_get_rate(clk
);
203 base
= of_iomap(np
, 0);
205 ret
= -EADDRNOTAVAIL
;
206 pr_err("failed to map registers for clockevent\n");
210 irq
= irq_of_parse_and_map(np
, 0);
213 pr_err("failed to get irq for clockevent\n");
217 writel_relaxed(TIMERn_IRQ_UF
, base
+ TIMERn_IEN
);
219 clock_event_ddata
.base
= base
;
220 clock_event_ddata
.periodic_top
= DIV_ROUND_CLOSEST(rate
, 1024 * HZ
);
222 clockevents_config_and_register(&clock_event_ddata
.evtdev
,
223 DIV_ROUND_CLOSEST(rate
, 1024),
226 ret
= request_irq(irq
, efm32_clock_event_handler
, IRQF_TIMER
,
227 "efm32 clockevent", &clock_event_ddata
);
229 pr_err("Failed setup irq\n");
241 clk_disable_unprepare(clk
);
251 * This function asserts that we have exactly one clocksource and one
252 * clock_event_device in the end.
254 static int __init
efm32_timer_init(struct device_node
*np
)
256 static int has_clocksource
, has_clockevent
;
259 if (!has_clocksource
) {
260 ret
= efm32_clocksource_init(np
);
267 if (!has_clockevent
) {
268 ret
= efm32_clockevent_init(np
);
277 TIMER_OF_DECLARE(efm32compat
, "efm32,timer", efm32_timer_init
);
278 TIMER_OF_DECLARE(efm32
, "energymicro,efm32-timer", efm32_timer_init
);