Blame view

drivers/clocksource/timer-cadence-ttc.c 14.5 KB
9c92ab619   Thomas Gleixner   treewide: Replace...
1
  // SPDX-License-Identifier: GPL-2.0-only
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
2
  /*
9e09dc5f7   Michal Simek   arm: zynq: Do not...
3
   * This file contains driver for the Cadence Triple Timer Counter Rev 06
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
4
   *
e932900a3   Michal Simek   arm: zynq: Use st...
5
   *  Copyright (C) 2011-2013 Xilinx
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
6
7
   *
   * based on arch/mips/kernel/time.c timer driver
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
8
   */
e932900a3   Michal Simek   arm: zynq: Use st...
9
  #include <linux/clk.h>
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
10
  #include <linux/interrupt.h>
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
11
  #include <linux/clockchips.h>
459fa246d   Stephen Rothwell   clocksource: Expl...
12
  #include <linux/clocksource.h>
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
13
14
15
  #include <linux/of_address.h>
  #include <linux/of_irq.h>
  #include <linux/slab.h>
3d77b30ef   Soren Brinkmann   clocksource: cade...
16
  #include <linux/sched_clock.h>
f5ac896b6   Rajan Vaja   clocksource/drive...
17
18
  #include <linux/module.h>
  #include <linux/of_platform.h>
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
19

b85a3ef4a   John Linn   ARM: Xilinx: Addi...
20
  /*
4e2bec0c3   Michal Simek   clocksource: cade...
21
   * This driver configures the 2 16/32-bit count-up timers as follows:
e932900a3   Michal Simek   arm: zynq: Use st...
22
23
24
25
26
27
28
29
30
31
32
33
34
35
   *
   * T1: Timer 1, clocksource for generic timekeeping
   * T2: Timer 2, clockevent source for hrtimers
   * T3: Timer 3, <unused>
   *
   * The input frequency to the timer module for emulation is 2.5MHz which is
   * common to all the timer channels (T1, T2, and T3). With a pre-scaler of 32,
   * the timers are clocked at 78.125KHz (12.8 us resolution).
  
   * The input frequency to the timer module in silicon is configurable and
   * obtained from device tree. The pre-scaler of 32 is used.
   */
  
  /*
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
36
37
38
   * Timer Register Offset Definitions of Timer 1, Increment base address by 4
   * and use same offsets for Timer 2
   */
9e09dc5f7   Michal Simek   arm: zynq: Do not...
39
40
41
42
43
44
  #define TTC_CLK_CNTRL_OFFSET		0x00 /* Clock Control Reg, RW */
  #define TTC_CNT_CNTRL_OFFSET		0x0C /* Counter Control Reg, RW */
  #define TTC_COUNT_VAL_OFFSET		0x18 /* Counter Value Reg, RO */
  #define TTC_INTR_VAL_OFFSET		0x24 /* Interval Count Reg, RW */
  #define TTC_ISR_OFFSET		0x54 /* Interrupt Status Reg, RO */
  #define TTC_IER_OFFSET		0x60 /* Interrupt Enable Reg, RW */
f184c5caa   Soren Brinkmann   arm: zynq: timer:...
45

9e09dc5f7   Michal Simek   arm: zynq: Do not...
46
  #define TTC_CNT_CNTRL_DISABLE_MASK	0x1
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
47

30e1e2859   Soren Brinkmann   arm: zynq: Migrat...
48
  #define TTC_CLK_CNTRL_CSRC_MASK		(1 << 5)	/* clock source */
b3e90722f   Soren Brinkmann   clocksource/caden...
49
50
  #define TTC_CLK_CNTRL_PSV_MASK		0x1e
  #define TTC_CLK_CNTRL_PSV_SHIFT		1
30e1e2859   Soren Brinkmann   arm: zynq: Migrat...
51

03377e585   Soren Brinkmann   arm: zynq: timer:...
52
53
  /*
   * Setup the timers to use pre-scaling, using a fixed value for now that will
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
54
55
56
57
58
59
   * work across most input frequency, but it may need to be more dynamic
   */
  #define PRESCALE_EXPONENT	11	/* 2 ^ PRESCALE_EXPONENT = PRESCALE */
  #define PRESCALE		2048	/* The exponent must match this */
  #define CLK_CNTRL_PRESCALE	((PRESCALE_EXPONENT - 1) << 1)
  #define CLK_CNTRL_PRESCALE_EN	1
e932900a3   Michal Simek   arm: zynq: Use st...
60
  #define CNT_CNTRL_RESET		(1 << 4)
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
61

b3e90722f   Soren Brinkmann   clocksource/caden...
62
  #define MAX_F_ERR 50
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
63
  /**
9e09dc5f7   Michal Simek   arm: zynq: Do not...
64
   * struct ttc_timer - This definition defines local timer structure
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
65
66
   *
   * @base_addr:	Base address of timer
c1dcc927d   Soren Brinkmann   clocksource: cade...
67
   * @freq:	Timer input clock frequency
e932900a3   Michal Simek   arm: zynq: Use st...
68
69
70
   * @clk:	Associated clock source
   * @clk_rate_change_nb	Notifier block for clock rate changes
   */
9e09dc5f7   Michal Simek   arm: zynq: Do not...
71
  struct ttc_timer {
e932900a3   Michal Simek   arm: zynq: Use st...
72
  	void __iomem *base_addr;
c1dcc927d   Soren Brinkmann   clocksource: cade...
73
  	unsigned long freq;
e932900a3   Michal Simek   arm: zynq: Use st...
74
75
  	struct clk *clk;
  	struct notifier_block clk_rate_change_nb;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
76
  };
9e09dc5f7   Michal Simek   arm: zynq: Do not...
77
78
  #define to_ttc_timer(x) \
  		container_of(x, struct ttc_timer, clk_rate_change_nb)
e932900a3   Michal Simek   arm: zynq: Use st...
79

9e09dc5f7   Michal Simek   arm: zynq: Do not...
80
  struct ttc_timer_clocksource {
b3e90722f   Soren Brinkmann   clocksource/caden...
81
82
  	u32			scale_clk_ctrl_reg_old;
  	u32			scale_clk_ctrl_reg_new;
9e09dc5f7   Michal Simek   arm: zynq: Do not...
83
  	struct ttc_timer	ttc;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
84
  	struct clocksource	cs;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
85
  };
9e09dc5f7   Michal Simek   arm: zynq: Do not...
86
87
  #define to_ttc_timer_clksrc(x) \
  		container_of(x, struct ttc_timer_clocksource, cs)
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
88

9e09dc5f7   Michal Simek   arm: zynq: Do not...
89
90
  struct ttc_timer_clockevent {
  	struct ttc_timer		ttc;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
91
  	struct clock_event_device	ce;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
92
  };
9e09dc5f7   Michal Simek   arm: zynq: Do not...
93
94
  #define to_ttc_timer_clkevent(x) \
  		container_of(x, struct ttc_timer_clockevent, ce)
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
95

3d77b30ef   Soren Brinkmann   clocksource: cade...
96
  static void __iomem *ttc_sched_clock_val_reg;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
97
  /**
9e09dc5f7   Michal Simek   arm: zynq: Do not...
98
   * ttc_set_interval - Set the timer interval value
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
99
100
101
102
   *
   * @timer:	Pointer to the timer instance
   * @cycles:	Timer interval ticks
   **/
9e09dc5f7   Michal Simek   arm: zynq: Do not...
103
  static void ttc_set_interval(struct ttc_timer *timer,
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
104
105
106
107
108
  					unsigned long cycles)
  {
  	u32 ctrl_reg;
  
  	/* Disable the counter, set the counter value  and re-enable counter */
87ab4361b   Michal Simek   clocksource: cade...
109
  	ctrl_reg = readl_relaxed(timer->base_addr + TTC_CNT_CNTRL_OFFSET);
9e09dc5f7   Michal Simek   arm: zynq: Do not...
110
  	ctrl_reg |= TTC_CNT_CNTRL_DISABLE_MASK;
87ab4361b   Michal Simek   clocksource: cade...
111
  	writel_relaxed(ctrl_reg, timer->base_addr + TTC_CNT_CNTRL_OFFSET);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
112

87ab4361b   Michal Simek   clocksource: cade...
113
  	writel_relaxed(cycles, timer->base_addr + TTC_INTR_VAL_OFFSET);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
114

03377e585   Soren Brinkmann   arm: zynq: timer:...
115
116
117
118
  	/*
  	 * Reset the counter (0x10) so that it starts from 0, one-shot
  	 * mode makes this needed for timing to be right.
  	 */
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
119
  	ctrl_reg |= CNT_CNTRL_RESET;
9e09dc5f7   Michal Simek   arm: zynq: Do not...
120
  	ctrl_reg &= ~TTC_CNT_CNTRL_DISABLE_MASK;
87ab4361b   Michal Simek   clocksource: cade...
121
  	writel_relaxed(ctrl_reg, timer->base_addr + TTC_CNT_CNTRL_OFFSET);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
122
123
124
  }
  
  /**
9e09dc5f7   Michal Simek   arm: zynq: Do not...
125
   * ttc_clock_event_interrupt - Clock event timer interrupt handler
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
126
127
   *
   * @irq:	IRQ number of the Timer
9e09dc5f7   Michal Simek   arm: zynq: Do not...
128
   * @dev_id:	void pointer to the ttc_timer instance
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
129
130
131
   *
   * returns: Always IRQ_HANDLED - success
   **/
9e09dc5f7   Michal Simek   arm: zynq: Do not...
132
  static irqreturn_t ttc_clock_event_interrupt(int irq, void *dev_id)
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
133
  {
9e09dc5f7   Michal Simek   arm: zynq: Do not...
134
135
  	struct ttc_timer_clockevent *ttce = dev_id;
  	struct ttc_timer *timer = &ttce->ttc;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
136
137
  
  	/* Acknowledge the interrupt and call event handler */
87ab4361b   Michal Simek   clocksource: cade...
138
  	readl_relaxed(timer->base_addr + TTC_ISR_OFFSET);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
139

9e09dc5f7   Michal Simek   arm: zynq: Do not...
140
  	ttce->ce.event_handler(&ttce->ce);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
141
142
143
  
  	return IRQ_HANDLED;
  }
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
144
  /**
9e09dc5f7   Michal Simek   arm: zynq: Do not...
145
   * __ttc_clocksource_read - Reads the timer counter register
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
146
147
148
   *
   * returns: Current timer counter register value
   **/
a5a1d1c29   Thomas Gleixner   clocksource: Use ...
149
  static u64 __ttc_clocksource_read(struct clocksource *cs)
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
150
  {
9e09dc5f7   Michal Simek   arm: zynq: Do not...
151
  	struct ttc_timer *timer = &to_ttc_timer_clksrc(cs)->ttc;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
152

a5a1d1c29   Thomas Gleixner   clocksource: Use ...
153
  	return (u64)readl_relaxed(timer->base_addr +
9e09dc5f7   Michal Simek   arm: zynq: Do not...
154
  				TTC_COUNT_VAL_OFFSET);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
155
  }
dfded0090   Stephen Boyd   clocksource: cade...
156
  static u64 notrace ttc_sched_clock_read(void)
3d77b30ef   Soren Brinkmann   clocksource: cade...
157
  {
87ab4361b   Michal Simek   clocksource: cade...
158
  	return readl_relaxed(ttc_sched_clock_val_reg);
3d77b30ef   Soren Brinkmann   clocksource: cade...
159
  }
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
160
  /**
9e09dc5f7   Michal Simek   arm: zynq: Do not...
161
   * ttc_set_next_event - Sets the time interval for next event
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
162
163
164
165
166
167
   *
   * @cycles:	Timer interval ticks
   * @evt:	Address of clock event instance
   *
   * returns: Always 0 - success
   **/
9e09dc5f7   Michal Simek   arm: zynq: Do not...
168
  static int ttc_set_next_event(unsigned long cycles,
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
169
170
  					struct clock_event_device *evt)
  {
9e09dc5f7   Michal Simek   arm: zynq: Do not...
171
172
  	struct ttc_timer_clockevent *ttce = to_ttc_timer_clkevent(evt);
  	struct ttc_timer *timer = &ttce->ttc;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
173

9e09dc5f7   Michal Simek   arm: zynq: Do not...
174
  	ttc_set_interval(timer, cycles);
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
175
176
177
178
  	return 0;
  }
  
  /**
5c0a4bbef   Viresh Kumar   clockevents/drive...
179
   * ttc_set_{shutdown|oneshot|periodic} - Sets the state of timer
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
180
   *
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
181
182
   * @evt:	Address of clock event instance
   **/
5c0a4bbef   Viresh Kumar   clockevents/drive...
183
  static int ttc_shutdown(struct clock_event_device *evt)
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
184
  {
9e09dc5f7   Michal Simek   arm: zynq: Do not...
185
186
  	struct ttc_timer_clockevent *ttce = to_ttc_timer_clkevent(evt);
  	struct ttc_timer *timer = &ttce->ttc;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
187
  	u32 ctrl_reg;
5c0a4bbef   Viresh Kumar   clockevents/drive...
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
  	ctrl_reg = readl_relaxed(timer->base_addr + TTC_CNT_CNTRL_OFFSET);
  	ctrl_reg |= TTC_CNT_CNTRL_DISABLE_MASK;
  	writel_relaxed(ctrl_reg, timer->base_addr + TTC_CNT_CNTRL_OFFSET);
  	return 0;
  }
  
  static int ttc_set_periodic(struct clock_event_device *evt)
  {
  	struct ttc_timer_clockevent *ttce = to_ttc_timer_clkevent(evt);
  	struct ttc_timer *timer = &ttce->ttc;
  
  	ttc_set_interval(timer,
  			 DIV_ROUND_CLOSEST(ttce->ttc.freq, PRESCALE * HZ));
  	return 0;
  }
  
  static int ttc_resume(struct clock_event_device *evt)
  {
  	struct ttc_timer_clockevent *ttce = to_ttc_timer_clkevent(evt);
  	struct ttc_timer *timer = &ttce->ttc;
  	u32 ctrl_reg;
  
  	ctrl_reg = readl_relaxed(timer->base_addr + TTC_CNT_CNTRL_OFFSET);
  	ctrl_reg &= ~TTC_CNT_CNTRL_DISABLE_MASK;
  	writel_relaxed(ctrl_reg, timer->base_addr + TTC_CNT_CNTRL_OFFSET);
  	return 0;
b85a3ef4a   John Linn   ARM: Xilinx: Addi...
214
  }
9e09dc5f7   Michal Simek   arm: zynq: Do not...
215
  static int ttc_rate_change_clocksource_cb(struct notifier_block *nb,
e932900a3   Michal Simek   arm: zynq: Use st...
216
217
218
  		unsigned long event, void *data)
  {
  	struct clk_notifier_data *ndata = data;
9e09dc5f7   Michal Simek   arm: zynq: Do not...
219
220
221
  	struct ttc_timer *ttc = to_ttc_timer(nb);
  	struct ttc_timer_clocksource *ttccs = container_of(ttc,
  			struct ttc_timer_clocksource, ttc);
e932900a3   Michal Simek   arm: zynq: Use st...
222
223
  
  	switch (event) {
b3e90722f   Soren Brinkmann   clocksource/caden...
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
  	case PRE_RATE_CHANGE:
  	{
  		u32 psv;
  		unsigned long factor, rate_low, rate_high;
  
  		if (ndata->new_rate > ndata->old_rate) {
  			factor = DIV_ROUND_CLOSEST(ndata->new_rate,
  					ndata->old_rate);
  			rate_low = ndata->old_rate;
  			rate_high = ndata->new_rate;
  		} else {
  			factor = DIV_ROUND_CLOSEST(ndata->old_rate,
  					ndata->new_rate);
  			rate_low = ndata->new_rate;
  			rate_high = ndata->old_rate;
  		}
  
  		if (!is_power_of_2(factor))
  				return NOTIFY_BAD;
  
  		if (abs(rate_high - (factor * rate_low)) > MAX_F_ERR)
  			return NOTIFY_BAD;
  
  		factor = __ilog2_u32(factor);
e932900a3   Michal Simek   arm: zynq: Use st...
248
  		/*
b3e90722f   Soren Brinkmann   clocksource/caden...
249
250
  		 * store timer clock ctrl register so we can restore it in case
  		 * of an abort.
e932900a3   Michal Simek   arm: zynq: Use st...
251
  		 */
b3e90722f   Soren Brinkmann   clocksource/caden...
252
  		ttccs->scale_clk_ctrl_reg_old =
87ab4361b   Michal Simek   clocksource: cade...
253
254
  			readl_relaxed(ttccs->ttc.base_addr +
  			TTC_CLK_CNTRL_OFFSET);
b3e90722f   Soren Brinkmann   clocksource/caden...
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
  
  		psv = (ttccs->scale_clk_ctrl_reg_old &
  				TTC_CLK_CNTRL_PSV_MASK) >>
  				TTC_CLK_CNTRL_PSV_SHIFT;
  		if (ndata->new_rate < ndata->old_rate)
  			psv -= factor;
  		else
  			psv += factor;
  
  		/* prescaler within legal range? */
  		if (psv & ~(TTC_CLK_CNTRL_PSV_MASK >> TTC_CLK_CNTRL_PSV_SHIFT))
  			return NOTIFY_BAD;
  
  		ttccs->scale_clk_ctrl_reg_new = ttccs->scale_clk_ctrl_reg_old &
  			~TTC_CLK_CNTRL_PSV_MASK;
  		ttccs->scale_clk_ctrl_reg_new |= psv << TTC_CLK_CNTRL_PSV_SHIFT;
  
  
  		/* scale down: adjust divider in post-change notification */
  		if (ndata->new_rate < ndata->old_rate)
  			return NOTIFY_DONE;
  
  		/* scale up: adjust divider now - before frequency change */
87ab4361b   Michal Simek   clocksource: cade...
278
279
  		writel_relaxed(ttccs->scale_clk_ctrl_reg_new,
  			       ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET);
b3e90722f   Soren Brinkmann   clocksource/caden...
280
281
282
283
284
285
286
287
  		break;
  	}
  	case POST_RATE_CHANGE:
  		/* scale up: pre-change notification did the adjustment */
  		if (ndata->new_rate > ndata->old_rate)
  			return NOTIFY_OK;
  
  		/* scale down: adjust divider now - after frequency change */
87ab4361b   Michal Simek   clocksource: cade...
288
289
  		writel_relaxed(ttccs->scale_clk_ctrl_reg_new,
  			       ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET);
b3e90722f   Soren Brinkmann   clocksource/caden...
290
  		break;
e932900a3   Michal Simek   arm: zynq: Use st...
291
  	case ABORT_RATE_CHANGE:
b3e90722f   Soren Brinkmann   clocksource/caden...
292
293
294
295
296
  		/* we have to undo the adjustment in case we scale up */
  		if (ndata->new_rate < ndata->old_rate)
  			return NOTIFY_OK;
  
  		/* restore original register value */
87ab4361b   Michal Simek   clocksource: cade...
297
298
  		writel_relaxed(ttccs->scale_clk_ctrl_reg_old,
  			       ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET);
df561f668   Gustavo A. R. Silva   treewide: Use fal...
299
  		fallthrough;
e932900a3   Michal Simek   arm: zynq: Use st...
300
301
302
  	default:
  		return NOTIFY_DONE;
  	}
b3e90722f   Soren Brinkmann   clocksource/caden...
303
304
  
  	return NOTIFY_DONE;
e932900a3   Michal Simek   arm: zynq: Use st...
305
  }
70504f311   Daniel Lezcano   clocksource/drive...
306
  static int __init ttc_setup_clocksource(struct clk *clk, void __iomem *base,
4e2bec0c3   Michal Simek   clocksource: cade...
307
  					 u32 timer_width)
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
308
  {
9e09dc5f7   Michal Simek   arm: zynq: Do not...
309
  	struct ttc_timer_clocksource *ttccs;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
310
  	int err;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
311
312
  
  	ttccs = kzalloc(sizeof(*ttccs), GFP_KERNEL);
70504f311   Daniel Lezcano   clocksource/drive...
313
314
  	if (!ttccs)
  		return -ENOMEM;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
315

9e09dc5f7   Michal Simek   arm: zynq: Do not...
316
  	ttccs->ttc.clk = clk;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
317

9e09dc5f7   Michal Simek   arm: zynq: Do not...
318
  	err = clk_prepare_enable(ttccs->ttc.clk);
70504f311   Daniel Lezcano   clocksource/drive...
319
  	if (err) {
c5263bb8b   Michal Simek   arm: zynq: Move t...
320
  		kfree(ttccs);
70504f311   Daniel Lezcano   clocksource/drive...
321
  		return err;
c5263bb8b   Michal Simek   arm: zynq: Move t...
322
  	}
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
323

c1dcc927d   Soren Brinkmann   clocksource: cade...
324
  	ttccs->ttc.freq = clk_get_rate(ttccs->ttc.clk);
9e09dc5f7   Michal Simek   arm: zynq: Do not...
325
326
327
  	ttccs->ttc.clk_rate_change_nb.notifier_call =
  		ttc_rate_change_clocksource_cb;
  	ttccs->ttc.clk_rate_change_nb.next = NULL;
70504f311   Daniel Lezcano   clocksource/drive...
328
329
330
331
  
  	err = clk_notifier_register(ttccs->ttc.clk,
  				    &ttccs->ttc.clk_rate_change_nb);
  	if (err)
e932900a3   Michal Simek   arm: zynq: Use st...
332
333
  		pr_warn("Unable to register clock notifier.
  ");
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
334

9e09dc5f7   Michal Simek   arm: zynq: Do not...
335
336
  	ttccs->ttc.base_addr = base;
  	ttccs->cs.name = "ttc_clocksource";
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
337
  	ttccs->cs.rating = 200;
9e09dc5f7   Michal Simek   arm: zynq: Do not...
338
  	ttccs->cs.read = __ttc_clocksource_read;
4e2bec0c3   Michal Simek   clocksource: cade...
339
  	ttccs->cs.mask = CLOCKSOURCE_MASK(timer_width);
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
340
  	ttccs->cs.flags = CLOCK_SOURCE_IS_CONTINUOUS;
e932900a3   Michal Simek   arm: zynq: Use st...
341
342
343
344
345
  	/*
  	 * Setup the clock source counter to be an incrementing counter
  	 * with no interrupt and it rolls over at 0xFFFF. Pre-scale
  	 * it by 32 also. Let it start running now.
  	 */
87ab4361b   Michal Simek   clocksource: cade...
346
347
  	writel_relaxed(0x0,  ttccs->ttc.base_addr + TTC_IER_OFFSET);
  	writel_relaxed(CLK_CNTRL_PRESCALE | CLK_CNTRL_PRESCALE_EN,
9e09dc5f7   Michal Simek   arm: zynq: Do not...
348
  		     ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET);
87ab4361b   Michal Simek   clocksource: cade...
349
  	writel_relaxed(CNT_CNTRL_RESET,
9e09dc5f7   Michal Simek   arm: zynq: Do not...
350
  		     ttccs->ttc.base_addr + TTC_CNT_CNTRL_OFFSET);
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
351

c1dcc927d   Soren Brinkmann   clocksource: cade...
352
  	err = clocksource_register_hz(&ttccs->cs, ttccs->ttc.freq / PRESCALE);
70504f311   Daniel Lezcano   clocksource/drive...
353
  	if (err) {
c5263bb8b   Michal Simek   arm: zynq: Move t...
354
  		kfree(ttccs);
70504f311   Daniel Lezcano   clocksource/drive...
355
  		return err;
c5263bb8b   Michal Simek   arm: zynq: Move t...
356
  	}
3d77b30ef   Soren Brinkmann   clocksource: cade...
357
358
  
  	ttc_sched_clock_val_reg = base + TTC_COUNT_VAL_OFFSET;
4e2bec0c3   Michal Simek   clocksource: cade...
359
360
  	sched_clock_register(ttc_sched_clock_read, timer_width,
  			     ttccs->ttc.freq / PRESCALE);
70504f311   Daniel Lezcano   clocksource/drive...
361
362
  
  	return 0;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
363
  }
9e09dc5f7   Michal Simek   arm: zynq: Do not...
364
  static int ttc_rate_change_clockevent_cb(struct notifier_block *nb,
e932900a3   Michal Simek   arm: zynq: Use st...
365
366
367
  		unsigned long event, void *data)
  {
  	struct clk_notifier_data *ndata = data;
9e09dc5f7   Michal Simek   arm: zynq: Do not...
368
369
370
  	struct ttc_timer *ttc = to_ttc_timer(nb);
  	struct ttc_timer_clockevent *ttcce = container_of(ttc,
  			struct ttc_timer_clockevent, ttc);
e932900a3   Michal Simek   arm: zynq: Use st...
371
372
373
  
  	switch (event) {
  	case POST_RATE_CHANGE:
c1dcc927d   Soren Brinkmann   clocksource: cade...
374
375
  		/* update cached frequency */
  		ttc->freq = ndata->new_rate;
5f0ba3b46   Soren Brinkmann   clocksource/caden...
376
  		clockevents_update_freq(&ttcce->ce, ndata->new_rate / PRESCALE);
df561f668   Gustavo A. R. Silva   treewide: Use fal...
377
  		fallthrough;
e932900a3   Michal Simek   arm: zynq: Use st...
378
379
380
381
382
383
  	case PRE_RATE_CHANGE:
  	case ABORT_RATE_CHANGE:
  	default:
  		return NOTIFY_DONE;
  	}
  }
70504f311   Daniel Lezcano   clocksource/drive...
384
385
  static int __init ttc_setup_clockevent(struct clk *clk,
  				       void __iomem *base, u32 irq)
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
386
  {
9e09dc5f7   Michal Simek   arm: zynq: Do not...
387
  	struct ttc_timer_clockevent *ttcce;
e932900a3   Michal Simek   arm: zynq: Use st...
388
  	int err;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
389
390
  
  	ttcce = kzalloc(sizeof(*ttcce), GFP_KERNEL);
70504f311   Daniel Lezcano   clocksource/drive...
391
392
  	if (!ttcce)
  		return -ENOMEM;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
393

9e09dc5f7   Michal Simek   arm: zynq: Do not...
394
  	ttcce->ttc.clk = clk;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
395

9e09dc5f7   Michal Simek   arm: zynq: Do not...
396
  	err = clk_prepare_enable(ttcce->ttc.clk);
93bf92079   Yu Kuai   clocksource/drive...
397
398
  	if (err)
  		goto out_kfree;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
399

9e09dc5f7   Michal Simek   arm: zynq: Do not...
400
401
402
  	ttcce->ttc.clk_rate_change_nb.notifier_call =
  		ttc_rate_change_clockevent_cb;
  	ttcce->ttc.clk_rate_change_nb.next = NULL;
70504f311   Daniel Lezcano   clocksource/drive...
403
404
405
406
  
  	err = clk_notifier_register(ttcce->ttc.clk,
  				    &ttcce->ttc.clk_rate_change_nb);
  	if (err) {
e932900a3   Michal Simek   arm: zynq: Use st...
407
408
  		pr_warn("Unable to register clock notifier.
  ");
93bf92079   Yu Kuai   clocksource/drive...
409
  		goto out_kfree;
70504f311   Daniel Lezcano   clocksource/drive...
410
  	}
c1dcc927d   Soren Brinkmann   clocksource: cade...
411
  	ttcce->ttc.freq = clk_get_rate(ttcce->ttc.clk);
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
412

9e09dc5f7   Michal Simek   arm: zynq: Do not...
413
414
  	ttcce->ttc.base_addr = base;
  	ttcce->ce.name = "ttc_clockevent";
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
415
  	ttcce->ce.features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT;
9e09dc5f7   Michal Simek   arm: zynq: Do not...
416
  	ttcce->ce.set_next_event = ttc_set_next_event;
5c0a4bbef   Viresh Kumar   clockevents/drive...
417
418
419
420
  	ttcce->ce.set_state_shutdown = ttc_shutdown;
  	ttcce->ce.set_state_periodic = ttc_set_periodic;
  	ttcce->ce.set_state_oneshot = ttc_shutdown;
  	ttcce->ce.tick_resume = ttc_resume;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
421
422
  	ttcce->ce.rating = 200;
  	ttcce->ce.irq = irq;
87e4ee759   Soren Brinkmann   arm: zynq: timer:...
423
  	ttcce->ce.cpumask = cpu_possible_mask;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
424

e932900a3   Michal Simek   arm: zynq: Use st...
425
426
427
428
429
  	/*
  	 * Setup the clock event timer to be an interval timer which
  	 * is prescaled by 32 using the interval interrupt. Leave it
  	 * disabled for now.
  	 */
87ab4361b   Michal Simek   clocksource: cade...
430
431
  	writel_relaxed(0x23, ttcce->ttc.base_addr + TTC_CNT_CNTRL_OFFSET);
  	writel_relaxed(CLK_CNTRL_PRESCALE | CLK_CNTRL_PRESCALE_EN,
9e09dc5f7   Michal Simek   arm: zynq: Do not...
432
  		     ttcce->ttc.base_addr + TTC_CLK_CNTRL_OFFSET);
87ab4361b   Michal Simek   clocksource: cade...
433
  	writel_relaxed(0x1,  ttcce->ttc.base_addr + TTC_IER_OFFSET);
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
434

9e09dc5f7   Michal Simek   arm: zynq: Do not...
435
  	err = request_irq(irq, ttc_clock_event_interrupt,
38c30a842   Michael Opdenacker   clocksource: misc...
436
  			  IRQF_TIMER, ttcce->ce.name, ttcce);
93bf92079   Yu Kuai   clocksource/drive...
437
438
  	if (err)
  		goto out_kfree;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
439
440
  
  	clockevents_config_and_register(&ttcce->ce,
c1dcc927d   Soren Brinkmann   clocksource: cade...
441
  			ttcce->ttc.freq / PRESCALE, 1, 0xfffe);
70504f311   Daniel Lezcano   clocksource/drive...
442
443
  
  	return 0;
93bf92079   Yu Kuai   clocksource/drive...
444
445
446
447
  
  out_kfree:
  	kfree(ttcce);
  	return err;
91dc985c5   Josh Cartwright   ARM: zynq: add cl...
448
  }
f5ac896b6   Rajan Vaja   clocksource/drive...
449
  static int __init ttc_timer_probe(struct platform_device *pdev)
e932900a3   Michal Simek   arm: zynq: Use st...
450
451
452
  {
  	unsigned int irq;
  	void __iomem *timer_baseaddr;
30e1e2859   Soren Brinkmann   arm: zynq: Migrat...
453
  	struct clk *clk_cs, *clk_ce;
c5263bb8b   Michal Simek   arm: zynq: Move t...
454
  	static int initialized;
70504f311   Daniel Lezcano   clocksource/drive...
455
  	int clksel, ret;
4e2bec0c3   Michal Simek   clocksource: cade...
456
  	u32 timer_width = 16;
f5ac896b6   Rajan Vaja   clocksource/drive...
457
  	struct device_node *timer = pdev->dev.of_node;
c5263bb8b   Michal Simek   arm: zynq: Move t...
458
459
  
  	if (initialized)
70504f311   Daniel Lezcano   clocksource/drive...
460
  		return 0;
c5263bb8b   Michal Simek   arm: zynq: Move t...
461
462
  
  	initialized = 1;
e932900a3   Michal Simek   arm: zynq: Use st...
463
464
465
466
467
468
469
470
471
472
  
  	/*
  	 * Get the 1st Triple Timer Counter (TTC) block from the device tree
  	 * and use it. Note that the event timer uses the interrupt and it's the
  	 * 2nd TTC hence the irq_of_parse_and_map(,1)
  	 */
  	timer_baseaddr = of_iomap(timer, 0);
  	if (!timer_baseaddr) {
  		pr_err("ERROR: invalid timer base address
  ");
70504f311   Daniel Lezcano   clocksource/drive...
473
  		return -ENXIO;
e932900a3   Michal Simek   arm: zynq: Use st...
474
475
476
477
478
479
  	}
  
  	irq = irq_of_parse_and_map(timer, 1);
  	if (irq <= 0) {
  		pr_err("ERROR: invalid interrupt number
  ");
70504f311   Daniel Lezcano   clocksource/drive...
480
  		return -EINVAL;
e932900a3   Michal Simek   arm: zynq: Use st...
481
  	}
4e2bec0c3   Michal Simek   clocksource: cade...
482
  	of_property_read_u32(timer, "timer-width", &timer_width);
87ab4361b   Michal Simek   clocksource: cade...
483
  	clksel = readl_relaxed(timer_baseaddr + TTC_CLK_CNTRL_OFFSET);
30e1e2859   Soren Brinkmann   arm: zynq: Migrat...
484
485
486
487
488
  	clksel = !!(clksel & TTC_CLK_CNTRL_CSRC_MASK);
  	clk_cs = of_clk_get(timer, clksel);
  	if (IS_ERR(clk_cs)) {
  		pr_err("ERROR: timer input clock not found
  ");
70504f311   Daniel Lezcano   clocksource/drive...
489
  		return PTR_ERR(clk_cs);
30e1e2859   Soren Brinkmann   arm: zynq: Migrat...
490
  	}
87ab4361b   Michal Simek   clocksource: cade...
491
  	clksel = readl_relaxed(timer_baseaddr + 4 + TTC_CLK_CNTRL_OFFSET);
30e1e2859   Soren Brinkmann   arm: zynq: Migrat...
492
493
494
  	clksel = !!(clksel & TTC_CLK_CNTRL_CSRC_MASK);
  	clk_ce = of_clk_get(timer, clksel);
  	if (IS_ERR(clk_ce)) {
e932900a3   Michal Simek   arm: zynq: Use st...
495
496
  		pr_err("ERROR: timer input clock not found
  ");
34c720a91   Christophe Jaillet   clocksource/drive...
497
  		return PTR_ERR(clk_ce);
e932900a3   Michal Simek   arm: zynq: Use st...
498
  	}
70504f311   Daniel Lezcano   clocksource/drive...
499
500
501
502
503
504
505
  	ret = ttc_setup_clocksource(clk_cs, timer_baseaddr, timer_width);
  	if (ret)
  		return ret;
  
  	ret = ttc_setup_clockevent(clk_ce, timer_baseaddr + 4, irq);
  	if (ret)
  		return ret;
e932900a3   Michal Simek   arm: zynq: Use st...
506

2a4849d26   Rob Herring   clocksource: Conv...
507
508
  	pr_info("%pOFn #0 at %p, irq=%d
  ", timer, timer_baseaddr, irq);
70504f311   Daniel Lezcano   clocksource/drive...
509
510
  
  	return 0;
e932900a3   Michal Simek   arm: zynq: Use st...
511
  }
f5ac896b6   Rajan Vaja   clocksource/drive...
512
513
514
515
516
517
518
519
520
521
522
523
524
525
  static const struct of_device_id ttc_timer_of_match[] = {
  	{.compatible = "cdns,ttc"},
  	{},
  };
  
  MODULE_DEVICE_TABLE(of, ttc_timer_of_match);
  
  static struct platform_driver ttc_timer_driver = {
  	.driver = {
  		.name	= "cdns_ttc_timer",
  		.of_match_table = ttc_timer_of_match,
  	},
  };
  builtin_platform_driver_probe(ttc_timer_driver, ttc_timer_probe);