diff options
author | Stanley Chu <stanley.chu@mediatek.com> | 2018-07-05 19:11:27 -0400 |
---|---|---|
committer | Daniel Lezcano <daniel.lezcano@linaro.org> | 2018-07-26 05:26:33 -0400 |
commit | a0858f937960d24e05cedf43d5cbcb95cf088434 (patch) | |
tree | 5e69a39b9dec892a2da61b8ac25d7f9509d3c6f5 /drivers/clocksource/timer-mediatek.c | |
parent | 56d52d3f56192049221105efa6ee76251a949c0e (diff) |
clocksource/drivers/timer-mediatek: Convert the driver to timer-of
Convert the driver to use the timer_of helpers.
This allows to remove custom proprietary structure,
factors out and simplifies the code.
Signed-off-by: Stanley Chu <stanley.chu@mediatek.com>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Diffstat (limited to 'drivers/clocksource/timer-mediatek.c')
-rw-r--r-- | drivers/clocksource/timer-mediatek.c | 205 |
1 files changed, 80 insertions, 125 deletions
diff --git a/drivers/clocksource/timer-mediatek.c b/drivers/clocksource/timer-mediatek.c index e3657d29e765..e57c4d7414a4 100644 --- a/drivers/clocksource/timer-mediatek.c +++ b/drivers/clocksource/timer-mediatek.c | |||
@@ -18,16 +18,13 @@ | |||
18 | 18 | ||
19 | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | 19 | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt |
20 | 20 | ||
21 | #include <linux/clk.h> | ||
22 | #include <linux/clockchips.h> | 21 | #include <linux/clockchips.h> |
22 | #include <linux/clocksource.h> | ||
23 | #include <linux/interrupt.h> | 23 | #include <linux/interrupt.h> |
24 | #include <linux/irq.h> | ||
25 | #include <linux/irqreturn.h> | 24 | #include <linux/irqreturn.h> |
26 | #include <linux/of.h> | ||
27 | #include <linux/of_address.h> | ||
28 | #include <linux/of_irq.h> | ||
29 | #include <linux/sched_clock.h> | 25 | #include <linux/sched_clock.h> |
30 | #include <linux/slab.h> | 26 | #include <linux/slab.h> |
27 | #include "timer-of.h" | ||
31 | 28 | ||
32 | #define TIMER_CLK_EVT (1) | 29 | #define TIMER_CLK_EVT (1) |
33 | #define TIMER_CLK_SRC (2) | 30 | #define TIMER_CLK_SRC (2) |
@@ -59,12 +56,6 @@ | |||
59 | #define GPT_CNT_REG(val) (0x08 + (0x10 * (val))) | 56 | #define GPT_CNT_REG(val) (0x08 + (0x10 * (val))) |
60 | #define GPT_CMP_REG(val) (0x0C + (0x10 * (val))) | 57 | #define GPT_CMP_REG(val) (0x0C + (0x10 * (val))) |
61 | 58 | ||
62 | struct mtk_clock_event_device { | ||
63 | void __iomem *gpt_base; | ||
64 | u32 ticks_per_jiffy; | ||
65 | struct clock_event_device dev; | ||
66 | }; | ||
67 | |||
68 | static void __iomem *gpt_sched_reg __read_mostly; | 59 | static void __iomem *gpt_sched_reg __read_mostly; |
69 | 60 | ||
70 | static u64 notrace mtk_gpt_read_sched_clock(void) | 61 | static u64 notrace mtk_gpt_read_sched_clock(void) |
@@ -72,36 +63,30 @@ static u64 notrace mtk_gpt_read_sched_clock(void) | |||
72 | return readl_relaxed(gpt_sched_reg); | 63 | return readl_relaxed(gpt_sched_reg); |
73 | } | 64 | } |
74 | 65 | ||
75 | static inline struct mtk_clock_event_device *to_mtk_clk( | 66 | static void mtk_gpt_clkevt_time_stop(struct timer_of *to, u8 timer) |
76 | struct clock_event_device *c) | ||
77 | { | ||
78 | return container_of(c, struct mtk_clock_event_device, dev); | ||
79 | } | ||
80 | |||
81 | static void mtk_gpt_clkevt_time_stop(struct mtk_clock_event_device *evt, u8 timer) | ||
82 | { | 67 | { |
83 | u32 val; | 68 | u32 val; |
84 | 69 | ||
85 | val = readl(evt->gpt_base + GPT_CTRL_REG(timer)); | 70 | val = readl(timer_of_base(to) + GPT_CTRL_REG(timer)); |
86 | writel(val & ~GPT_CTRL_ENABLE, evt->gpt_base + | 71 | writel(val & ~GPT_CTRL_ENABLE, timer_of_base(to) + |
87 | GPT_CTRL_REG(timer)); | 72 | GPT_CTRL_REG(timer)); |
88 | } | 73 | } |
89 | 74 | ||
90 | static void mtk_gpt_clkevt_time_setup(struct mtk_clock_event_device *evt, | 75 | static void mtk_gpt_clkevt_time_setup(struct timer_of *to, |
91 | unsigned long delay, u8 timer) | 76 | unsigned long delay, u8 timer) |
92 | { | 77 | { |
93 | writel(delay, evt->gpt_base + GPT_CMP_REG(timer)); | 78 | writel(delay, timer_of_base(to) + GPT_CMP_REG(timer)); |
94 | } | 79 | } |
95 | 80 | ||
96 | static void mtk_gpt_clkevt_time_start(struct mtk_clock_event_device *evt, | 81 | static void mtk_gpt_clkevt_time_start(struct timer_of *to, |
97 | bool periodic, u8 timer) | 82 | bool periodic, u8 timer) |
98 | { | 83 | { |
99 | u32 val; | 84 | u32 val; |
100 | 85 | ||
101 | /* Acknowledge interrupt */ | 86 | /* Acknowledge interrupt */ |
102 | writel(GPT_IRQ_ACK(timer), evt->gpt_base + GPT_IRQ_ACK_REG); | 87 | writel(GPT_IRQ_ACK(timer), timer_of_base(to) + GPT_IRQ_ACK_REG); |
103 | 88 | ||
104 | val = readl(evt->gpt_base + GPT_CTRL_REG(timer)); | 89 | val = readl(timer_of_base(to) + GPT_CTRL_REG(timer)); |
105 | 90 | ||
106 | /* Clear 2 bit timer operation mode field */ | 91 | /* Clear 2 bit timer operation mode field */ |
107 | val &= ~GPT_CTRL_OP(0x3); | 92 | val &= ~GPT_CTRL_OP(0x3); |
@@ -112,160 +97,130 @@ static void mtk_gpt_clkevt_time_start(struct mtk_clock_event_device *evt, | |||
112 | val |= GPT_CTRL_OP(GPT_CTRL_OP_ONESHOT); | 97 | val |= GPT_CTRL_OP(GPT_CTRL_OP_ONESHOT); |
113 | 98 | ||
114 | writel(val | GPT_CTRL_ENABLE | GPT_CTRL_CLEAR, | 99 | writel(val | GPT_CTRL_ENABLE | GPT_CTRL_CLEAR, |
115 | evt->gpt_base + GPT_CTRL_REG(timer)); | 100 | timer_of_base(to) + GPT_CTRL_REG(timer)); |
116 | } | 101 | } |
117 | 102 | ||
118 | static int mtk_gpt_clkevt_shutdown(struct clock_event_device *clk) | 103 | static int mtk_gpt_clkevt_shutdown(struct clock_event_device *clk) |
119 | { | 104 | { |
120 | mtk_gpt_clkevt_time_stop(to_mtk_clk(clk), TIMER_CLK_EVT); | 105 | mtk_gpt_clkevt_time_stop(to_timer_of(clk), TIMER_CLK_EVT); |
106 | |||
121 | return 0; | 107 | return 0; |
122 | } | 108 | } |
123 | 109 | ||
124 | static int mtk_gpt_clkevt_set_periodic(struct clock_event_device *clk) | 110 | static int mtk_gpt_clkevt_set_periodic(struct clock_event_device *clk) |
125 | { | 111 | { |
126 | struct mtk_clock_event_device *evt = to_mtk_clk(clk); | 112 | struct timer_of *to = to_timer_of(clk); |
113 | |||
114 | mtk_gpt_clkevt_time_stop(to, TIMER_CLK_EVT); | ||
115 | mtk_gpt_clkevt_time_setup(to, to->of_clk.period, TIMER_CLK_EVT); | ||
116 | mtk_gpt_clkevt_time_start(to, true, TIMER_CLK_EVT); | ||
127 | 117 | ||
128 | mtk_gpt_clkevt_time_stop(evt, TIMER_CLK_EVT); | ||
129 | mtk_gpt_clkevt_time_setup(evt, evt->ticks_per_jiffy, TIMER_CLK_EVT); | ||
130 | mtk_gpt_clkevt_time_start(evt, true, TIMER_CLK_EVT); | ||
131 | return 0; | 118 | return 0; |
132 | } | 119 | } |
133 | 120 | ||
134 | static int mtk_gpt_clkevt_next_event(unsigned long event, | 121 | static int mtk_gpt_clkevt_next_event(unsigned long event, |
135 | struct clock_event_device *clk) | 122 | struct clock_event_device *clk) |
136 | { | 123 | { |
137 | struct mtk_clock_event_device *evt = to_mtk_clk(clk); | 124 | struct timer_of *to = to_timer_of(clk); |
138 | 125 | ||
139 | mtk_gpt_clkevt_time_stop(evt, TIMER_CLK_EVT); | 126 | mtk_gpt_clkevt_time_stop(to, TIMER_CLK_EVT); |
140 | mtk_gpt_clkevt_time_setup(evt, event, TIMER_CLK_EVT); | 127 | mtk_gpt_clkevt_time_setup(to, event, TIMER_CLK_EVT); |
141 | mtk_gpt_clkevt_time_start(evt, false, TIMER_CLK_EVT); | 128 | mtk_gpt_clkevt_time_start(to, false, TIMER_CLK_EVT); |
142 | 129 | ||
143 | return 0; | 130 | return 0; |
144 | } | 131 | } |
145 | 132 | ||
146 | static irqreturn_t mtk_gpt_interrupt(int irq, void *dev_id) | 133 | static irqreturn_t mtk_gpt_interrupt(int irq, void *dev_id) |
147 | { | 134 | { |
148 | struct mtk_clock_event_device *evt = dev_id; | 135 | struct clock_event_device *clkevt = (struct clock_event_device *)dev_id; |
136 | struct timer_of *to = to_timer_of(clkevt); | ||
149 | 137 | ||
150 | /* Acknowledge timer0 irq */ | 138 | /* Acknowledge timer0 irq */ |
151 | writel(GPT_IRQ_ACK(TIMER_CLK_EVT), evt->gpt_base + GPT_IRQ_ACK_REG); | 139 | writel(GPT_IRQ_ACK(TIMER_CLK_EVT), timer_of_base(to) + GPT_IRQ_ACK_REG); |
152 | evt->dev.event_handler(&evt->dev); | 140 | clkevt->event_handler(clkevt); |
153 | 141 | ||
154 | return IRQ_HANDLED; | 142 | return IRQ_HANDLED; |
155 | } | 143 | } |
156 | 144 | ||
157 | static void | 145 | static void |
158 | __init mtk_gpt_setup(struct mtk_clock_event_device *evt, u8 timer, u8 option) | 146 | __init mtk_gpt_setup(struct timer_of *to, u8 timer, u8 option) |
159 | { | 147 | { |
160 | writel(GPT_CTRL_CLEAR | GPT_CTRL_DISABLE, | 148 | writel(GPT_CTRL_CLEAR | GPT_CTRL_DISABLE, |
161 | evt->gpt_base + GPT_CTRL_REG(timer)); | 149 | timer_of_base(to) + GPT_CTRL_REG(timer)); |
162 | 150 | ||
163 | writel(GPT_CLK_SRC(GPT_CLK_SRC_SYS13M) | GPT_CLK_DIV1, | 151 | writel(GPT_CLK_SRC(GPT_CLK_SRC_SYS13M) | GPT_CLK_DIV1, |
164 | evt->gpt_base + GPT_CLK_REG(timer)); | 152 | timer_of_base(to) + GPT_CLK_REG(timer)); |
165 | 153 | ||
166 | writel(0x0, evt->gpt_base + GPT_CMP_REG(timer)); | 154 | writel(0x0, timer_of_base(to) + GPT_CMP_REG(timer)); |
167 | 155 | ||
168 | writel(GPT_CTRL_OP(option) | GPT_CTRL_ENABLE, | 156 | writel(GPT_CTRL_OP(option) | GPT_CTRL_ENABLE, |
169 | evt->gpt_base + GPT_CTRL_REG(timer)); | 157 | timer_of_base(to) + GPT_CTRL_REG(timer)); |
170 | } | 158 | } |
171 | 159 | ||
172 | static void mtk_gpt_enable_irq(struct mtk_clock_event_device *evt, u8 timer) | 160 | static void mtk_gpt_enable_irq(struct timer_of *to, u8 timer) |
173 | { | 161 | { |
174 | u32 val; | 162 | u32 val; |
175 | 163 | ||
176 | /* Disable all interrupts */ | 164 | /* Disable all interrupts */ |
177 | writel(0x0, evt->gpt_base + GPT_IRQ_EN_REG); | 165 | writel(0x0, timer_of_base(to) + GPT_IRQ_EN_REG); |
178 | 166 | ||
179 | /* Acknowledge all spurious pending interrupts */ | 167 | /* Acknowledge all spurious pending interrupts */ |
180 | writel(0x3f, evt->gpt_base + GPT_IRQ_ACK_REG); | 168 | writel(0x3f, timer_of_base(to) + GPT_IRQ_ACK_REG); |
181 | 169 | ||
182 | val = readl(evt->gpt_base + GPT_IRQ_EN_REG); | 170 | val = readl(timer_of_base(to) + GPT_IRQ_EN_REG); |
183 | writel(val | GPT_IRQ_ENABLE(timer), | 171 | writel(val | GPT_IRQ_ENABLE(timer), |
184 | evt->gpt_base + GPT_IRQ_EN_REG); | 172 | timer_of_base(to) + GPT_IRQ_EN_REG); |
185 | } | 173 | } |
186 | 174 | ||
175 | static struct timer_of to = { | ||
176 | .flags = TIMER_OF_IRQ | TIMER_OF_BASE | TIMER_OF_CLOCK, | ||
177 | |||
178 | .clkevt = { | ||
179 | .name = "mtk-clkevt", | ||
180 | .rating = 300, | ||
181 | .cpumask = cpu_possible_mask, | ||
182 | }, | ||
183 | |||
184 | .of_irq = { | ||
185 | .flags = IRQF_TIMER | IRQF_IRQPOLL, | ||
186 | }, | ||
187 | }; | ||
188 | |||
187 | static int __init mtk_gpt_init(struct device_node *node) | 189 | static int __init mtk_gpt_init(struct device_node *node) |
188 | { | 190 | { |
189 | struct mtk_clock_event_device *evt; | 191 | int ret; |
190 | struct resource res; | 192 | |
191 | unsigned long rate = 0; | 193 | to.clkevt.features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; |
192 | struct clk *clk; | 194 | to.clkevt.set_state_shutdown = mtk_gpt_clkevt_shutdown; |
193 | 195 | to.clkevt.set_state_periodic = mtk_gpt_clkevt_set_periodic; | |
194 | evt = kzalloc(sizeof(*evt), GFP_KERNEL); | 196 | to.clkevt.set_state_oneshot = mtk_gpt_clkevt_shutdown; |
195 | if (!evt) | 197 | to.clkevt.tick_resume = mtk_gpt_clkevt_shutdown; |
196 | return -ENOMEM; | 198 | to.clkevt.set_next_event = mtk_gpt_clkevt_next_event; |
197 | 199 | to.of_irq.handler = mtk_gpt_interrupt; | |
198 | evt->dev.name = "mtk_tick"; | 200 | |
199 | evt->dev.rating = 300; | 201 | ret = timer_of_init(node, &to); |
200 | evt->dev.features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; | 202 | if (ret) |
201 | evt->dev.set_state_shutdown = mtk_gpt_clkevt_shutdown; | 203 | goto err; |
202 | evt->dev.set_state_periodic = mtk_gpt_clkevt_set_periodic; | ||
203 | evt->dev.set_state_oneshot = mtk_gpt_clkevt_shutdown; | ||
204 | evt->dev.tick_resume = mtk_gpt_clkevt_shutdown; | ||
205 | evt->dev.set_next_event = mtk_gpt_clkevt_next_event; | ||
206 | evt->dev.cpumask = cpu_possible_mask; | ||
207 | |||
208 | evt->gpt_base = of_io_request_and_map(node, 0, "mtk-timer-gpt"); | ||
209 | if (IS_ERR(evt->gpt_base)) { | ||
210 | pr_err("Can't get resource\n"); | ||
211 | goto err_kzalloc; | ||
212 | } | ||
213 | |||
214 | evt->dev.irq = irq_of_parse_and_map(node, 0); | ||
215 | if (evt->dev.irq <= 0) { | ||
216 | pr_err("Can't parse IRQ\n"); | ||
217 | goto err_mem; | ||
218 | } | ||
219 | |||
220 | clk = of_clk_get(node, 0); | ||
221 | if (IS_ERR(clk)) { | ||
222 | pr_err("Can't get timer clock\n"); | ||
223 | goto err_irq; | ||
224 | } | ||
225 | |||
226 | if (clk_prepare_enable(clk)) { | ||
227 | pr_err("Can't prepare clock\n"); | ||
228 | goto err_clk_put; | ||
229 | } | ||
230 | rate = clk_get_rate(clk); | ||
231 | |||
232 | if (request_irq(evt->dev.irq, mtk_gpt_interrupt, | ||
233 | IRQF_TIMER | IRQF_IRQPOLL, "mtk_timer", evt)) { | ||
234 | pr_err("failed to setup irq %d\n", evt->dev.irq); | ||
235 | goto err_clk_disable; | ||
236 | } | ||
237 | |||
238 | evt->ticks_per_jiffy = DIV_ROUND_UP(rate, HZ); | ||
239 | 204 | ||
240 | /* Configure clock source */ | 205 | /* Configure clock source */ |
241 | mtk_gpt_setup(evt, TIMER_CLK_SRC, GPT_CTRL_OP_FREERUN); | 206 | mtk_gpt_setup(&to, TIMER_CLK_SRC, GPT_CTRL_OP_FREERUN); |
242 | clocksource_mmio_init(evt->gpt_base + GPT_CNT_REG(TIMER_CLK_SRC), | 207 | clocksource_mmio_init(timer_of_base(&to) + GPT_CNT_REG(TIMER_CLK_SRC), |
243 | node->name, rate, 300, 32, clocksource_mmio_readl_up); | 208 | node->name, timer_of_rate(&to), 300, 32, |
244 | gpt_sched_reg = evt->gpt_base + GPT_CNT_REG(TIMER_CLK_SRC); | 209 | clocksource_mmio_readl_up); |
245 | sched_clock_register(mtk_gpt_read_sched_clock, 32, rate); | 210 | gpt_sched_reg = timer_of_base(&to) + GPT_CNT_REG(TIMER_CLK_SRC); |
211 | sched_clock_register(mtk_gpt_read_sched_clock, 32, timer_of_rate(&to)); | ||
246 | 212 | ||
247 | /* Configure clock event */ | 213 | /* Configure clock event */ |
248 | mtk_gpt_setup(evt, TIMER_CLK_EVT, GPT_CTRL_OP_REPEAT); | 214 | mtk_gpt_setup(&to, TIMER_CLK_EVT, GPT_CTRL_OP_REPEAT); |
249 | clockevents_config_and_register(&evt->dev, rate, TIMER_SYNC_TICKS, | 215 | clockevents_config_and_register(&to.clkevt, timer_of_rate(&to), |
250 | 0xffffffff); | 216 | TIMER_SYNC_TICKS, 0xffffffff); |
251 | 217 | ||
252 | mtk_gpt_enable_irq(evt, TIMER_CLK_EVT); | 218 | mtk_gpt_enable_irq(&to, TIMER_CLK_EVT); |
253 | 219 | ||
254 | return 0; | 220 | return 0; |
255 | 221 | ||
256 | err_clk_disable: | 222 | err: |
257 | clk_disable_unprepare(clk); | 223 | timer_of_cleanup(&to); |
258 | err_clk_put: | 224 | return ret; |
259 | clk_put(clk); | ||
260 | err_irq: | ||
261 | irq_dispose_mapping(evt->dev.irq); | ||
262 | err_mem: | ||
263 | iounmap(evt->gpt_base); | ||
264 | of_address_to_resource(node, 0, &res); | ||
265 | release_mem_region(res.start, resource_size(&res)); | ||
266 | err_kzalloc: | ||
267 | kfree(evt); | ||
268 | |||
269 | return -EINVAL; | ||
270 | } | 225 | } |
271 | TIMER_OF_DECLARE(mtk_mt6577, "mediatek,mt6577-timer", mtk_gpt_init); | 226 | TIMER_OF_DECLARE(mtk_mt6577, "mediatek,mt6577-timer", mtk_gpt_init); |