2 * Copyright (C) 2013 Pengutronix
5 * This program is free software; you can redistribute it and/or modify it under
6 * the terms of the GNU General Public License version 2 as published by the
7 * Free Software Foundation.
10 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
12 #include <linux/kernel.h>
13 #include <linux/clocksource.h>
14 #include <linux/clockchips.h>
15 #include <linux/irq.h>
16 #include <linux/interrupt.h>
18 #include <linux/of_address.h>
19 #include <linux/of_irq.h>
20 #include <linux/clk.h>
22 #define TIMERn_CTRL 0x00
23 #define TIMERn_CTRL_PRESC(val) (((val) & 0xf) << 24)
24 #define TIMERn_CTRL_PRESC_1024 TIMERn_CTRL_PRESC(10)
25 #define TIMERn_CTRL_CLKSEL(val) (((val) & 0x3) << 16)
26 #define TIMERn_CTRL_CLKSEL_PRESCHFPERCLK TIMERn_CTRL_CLKSEL(0)
27 #define TIMERn_CTRL_OSMEN 0x00000010
28 #define TIMERn_CTRL_MODE(val) (((val) & 0x3) << 0)
29 #define TIMERn_CTRL_MODE_UP TIMERn_CTRL_MODE(0)
30 #define TIMERn_CTRL_MODE_DOWN TIMERn_CTRL_MODE(1)
32 #define TIMERn_CMD 0x04
33 #define TIMERn_CMD_START 0x00000001
34 #define TIMERn_CMD_STOP 0x00000002
36 #define TIMERn_IEN 0x0c
37 #define TIMERn_IF 0x10
38 #define TIMERn_IFS 0x14
39 #define TIMERn_IFC 0x18
40 #define TIMERn_IRQ_UF 0x00000002
42 #define TIMERn_TOP 0x1c
43 #define TIMERn_CNT 0x24
45 struct efm32_clock_event_ddata {
46 struct clock_event_device evtdev;
48 unsigned periodic_top;
51 static void efm32_clock_event_set_mode(enum clock_event_mode mode,
52 struct clock_event_device *evtdev)
54 struct efm32_clock_event_ddata *ddata =
55 container_of(evtdev, struct efm32_clock_event_ddata, evtdev);
58 case CLOCK_EVT_MODE_PERIODIC:
59 writel_relaxed(TIMERn_CMD_STOP, ddata->base + TIMERn_CMD);
60 writel_relaxed(ddata->periodic_top, ddata->base + TIMERn_TOP);
61 writel_relaxed(TIMERn_CTRL_PRESC_1024 |
62 TIMERn_CTRL_CLKSEL_PRESCHFPERCLK |
63 TIMERn_CTRL_MODE_DOWN,
64 ddata->base + TIMERn_CTRL);
65 writel_relaxed(TIMERn_CMD_START, ddata->base + TIMERn_CMD);
68 case CLOCK_EVT_MODE_ONESHOT:
69 writel_relaxed(TIMERn_CMD_STOP, ddata->base + TIMERn_CMD);
70 writel_relaxed(TIMERn_CTRL_PRESC_1024 |
71 TIMERn_CTRL_CLKSEL_PRESCHFPERCLK |
73 TIMERn_CTRL_MODE_DOWN,
74 ddata->base + TIMERn_CTRL);
77 case CLOCK_EVT_MODE_UNUSED:
78 case CLOCK_EVT_MODE_SHUTDOWN:
79 writel_relaxed(TIMERn_CMD_STOP, ddata->base + TIMERn_CMD);
82 case CLOCK_EVT_MODE_RESUME:
87 static int efm32_clock_event_set_next_event(unsigned long evt,
88 struct clock_event_device *evtdev)
90 struct efm32_clock_event_ddata *ddata =
91 container_of(evtdev, struct efm32_clock_event_ddata, evtdev);
93 writel_relaxed(TIMERn_CMD_STOP, ddata->base + TIMERn_CMD);
94 writel_relaxed(evt, ddata->base + TIMERn_CNT);
95 writel_relaxed(TIMERn_CMD_START, ddata->base + TIMERn_CMD);
100 static irqreturn_t efm32_clock_event_handler(int irq, void *dev_id)
102 struct efm32_clock_event_ddata *ddata = dev_id;
104 writel_relaxed(TIMERn_IRQ_UF, ddata->base + TIMERn_IFC);
106 ddata->evtdev.event_handler(&ddata->evtdev);
111 static struct efm32_clock_event_ddata clock_event_ddata = {
113 .name = "efm32 clockevent",
114 .features = CLOCK_EVT_FEAT_ONESHOT | CLOCK_EVT_MODE_PERIODIC,
115 .set_mode = efm32_clock_event_set_mode,
116 .set_next_event = efm32_clock_event_set_next_event,
121 static struct irqaction efm32_clock_event_irq = {
122 .name = "efm32 clockevent",
124 .handler = efm32_clock_event_handler,
125 .dev_id = &clock_event_ddata,
128 static int __init efm32_clocksource_init(struct device_node *np)
135 clk = of_clk_get(np, 0);
138 pr_err("failed to get clock for clocksource (%d)\n", ret);
142 ret = clk_prepare_enable(clk);
144 pr_err("failed to enable timer clock for clocksource (%d)\n",
148 rate = clk_get_rate(clk);
150 base = of_iomap(np, 0);
152 ret = -EADDRNOTAVAIL;
153 pr_err("failed to map registers for clocksource\n");
157 writel_relaxed(TIMERn_CTRL_PRESC_1024 |
158 TIMERn_CTRL_CLKSEL_PRESCHFPERCLK |
159 TIMERn_CTRL_MODE_UP, base + TIMERn_CTRL);
160 writel_relaxed(TIMERn_CMD_START, base + TIMERn_CMD);
162 ret = clocksource_mmio_init(base + TIMERn_CNT, "efm32 timer",
163 DIV_ROUND_CLOSEST(rate, 1024), 200, 16,
164 clocksource_mmio_readl_up);
166 pr_err("failed to init clocksource (%d)\n", ret);
167 goto err_clocksource_init;
172 err_clocksource_init:
177 clk_disable_unprepare(clk);
186 static int __init efm32_clockevent_init(struct device_node *np)
194 clk = of_clk_get(np, 0);
197 pr_err("failed to get clock for clockevent (%d)\n", ret);
201 ret = clk_prepare_enable(clk);
203 pr_err("failed to enable timer clock for clockevent (%d)\n",
207 rate = clk_get_rate(clk);
209 base = of_iomap(np, 0);
211 ret = -EADDRNOTAVAIL;
212 pr_err("failed to map registers for clockevent\n");
216 irq = irq_of_parse_and_map(np, 0);
219 pr_err("failed to get irq for clockevent\n");
223 writel_relaxed(TIMERn_IRQ_UF, base + TIMERn_IEN);
225 clock_event_ddata.base = base;
226 clock_event_ddata.periodic_top = DIV_ROUND_CLOSEST(rate, 1024 * HZ);
228 setup_irq(irq, &efm32_clock_event_irq);
230 clockevents_config_and_register(&clock_event_ddata.evtdev,
231 DIV_ROUND_CLOSEST(rate, 1024),
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 void __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);
275 CLOCKSOURCE_OF_DECLARE(efm32compat, "efm32,timer", efm32_timer_init);
276 CLOCKSOURCE_OF_DECLARE(efm32, "energymicro,efm32-timer", efm32_timer_init);