xref: /openbmc/linux/drivers/counter/i8254.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
1  // SPDX-License-Identifier: GPL-2.0
2  /*
3   * Intel 8254 Programmable Interval Timer
4   * Copyright (C) William Breathitt Gray
5   */
6  #include <linux/bitfield.h>
7  #include <linux/bits.h>
8  #include <linux/counter.h>
9  #include <linux/device.h>
10  #include <linux/err.h>
11  #include <linux/export.h>
12  #include <linux/i8254.h>
13  #include <linux/limits.h>
14  #include <linux/module.h>
15  #include <linux/mutex.h>
16  #include <linux/regmap.h>
17  
18  #include <asm/unaligned.h>
19  
20  #define I8254_COUNTER_REG(_counter) (_counter)
21  #define I8254_CONTROL_REG 0x3
22  
23  #define I8254_SC GENMASK(7, 6)
24  #define I8254_RW GENMASK(5, 4)
25  #define I8254_M GENMASK(3, 1)
26  #define I8254_CONTROL(_sc, _rw, _m) \
27  	(u8_encode_bits(_sc, I8254_SC) | u8_encode_bits(_rw, I8254_RW) | \
28  	 u8_encode_bits(_m, I8254_M))
29  
30  #define I8254_RW_TWO_BYTE 0x3
31  #define I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT 0
32  #define I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT 1
33  #define I8254_MODE_RATE_GENERATOR 2
34  #define I8254_MODE_SQUARE_WAVE_MODE 3
35  #define I8254_MODE_SOFTWARE_TRIGGERED_STROBE 4
36  #define I8254_MODE_HARDWARE_TRIGGERED_STROBE 5
37  
38  #define I8254_COUNTER_LATCH(_counter) I8254_CONTROL(_counter, 0x0, 0x0)
39  #define I8254_PROGRAM_COUNTER(_counter, _mode) I8254_CONTROL(_counter, I8254_RW_TWO_BYTE, _mode)
40  
41  #define I8254_NUM_COUNTERS 3
42  
43  /**
44   * struct i8254 - I8254 device private data structure
45   * @lock:	synchronization lock to prevent I/O race conditions
46   * @preset:	array of Counter Register states
47   * @out_mode:	array of mode configuration states
48   * @map:	Regmap for the device
49   */
50  struct i8254 {
51  	struct mutex lock;
52  	u16 preset[I8254_NUM_COUNTERS];
53  	u8 out_mode[I8254_NUM_COUNTERS];
54  	struct regmap *map;
55  };
56  
i8254_count_read(struct counter_device * const counter,struct counter_count * const count,u64 * const val)57  static int i8254_count_read(struct counter_device *const counter, struct counter_count *const count,
58  			    u64 *const val)
59  {
60  	struct i8254 *const priv = counter_priv(counter);
61  	int ret;
62  	u8 value[2];
63  
64  	mutex_lock(&priv->lock);
65  
66  	ret = regmap_write(priv->map, I8254_CONTROL_REG, I8254_COUNTER_LATCH(count->id));
67  	if (ret) {
68  		mutex_unlock(&priv->lock);
69  		return ret;
70  	}
71  	ret = regmap_noinc_read(priv->map, I8254_COUNTER_REG(count->id), value, sizeof(value));
72  	if (ret) {
73  		mutex_unlock(&priv->lock);
74  		return ret;
75  	}
76  
77  	mutex_unlock(&priv->lock);
78  
79  	*val = get_unaligned_le16(value);
80  
81  	return ret;
82  }
83  
i8254_function_read(struct counter_device * const counter,struct counter_count * const count,enum counter_function * const function)84  static int i8254_function_read(struct counter_device *const counter,
85  			       struct counter_count *const count,
86  			       enum counter_function *const function)
87  {
88  	*function = COUNTER_FUNCTION_DECREASE;
89  	return 0;
90  }
91  
92  #define I8254_SYNAPSES_PER_COUNT 2
93  #define I8254_SIGNAL_ID_CLK 0
94  #define I8254_SIGNAL_ID_GATE 1
95  
i8254_action_read(struct counter_device * const counter,struct counter_count * const count,struct counter_synapse * const synapse,enum counter_synapse_action * const action)96  static int i8254_action_read(struct counter_device *const counter,
97  			     struct counter_count *const count,
98  			     struct counter_synapse *const synapse,
99  			     enum counter_synapse_action *const action)
100  {
101  	struct i8254 *const priv = counter_priv(counter);
102  
103  	switch (synapse->signal->id % I8254_SYNAPSES_PER_COUNT) {
104  	case I8254_SIGNAL_ID_CLK:
105  		*action = COUNTER_SYNAPSE_ACTION_FALLING_EDGE;
106  		return 0;
107  	case I8254_SIGNAL_ID_GATE:
108  		switch (priv->out_mode[count->id]) {
109  		case I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT:
110  		case I8254_MODE_RATE_GENERATOR:
111  		case I8254_MODE_SQUARE_WAVE_MODE:
112  		case I8254_MODE_HARDWARE_TRIGGERED_STROBE:
113  			*action = COUNTER_SYNAPSE_ACTION_RISING_EDGE;
114  			return 0;
115  		default:
116  			*action = COUNTER_SYNAPSE_ACTION_NONE;
117  			return 0;
118  		}
119  	default:
120  		/* should never reach this path */
121  		return -EINVAL;
122  	}
123  }
124  
i8254_count_ceiling_read(struct counter_device * const counter,struct counter_count * const count,u64 * const ceiling)125  static int i8254_count_ceiling_read(struct counter_device *const counter,
126  				    struct counter_count *const count, u64 *const ceiling)
127  {
128  	struct i8254 *const priv = counter_priv(counter);
129  
130  	mutex_lock(&priv->lock);
131  
132  	switch (priv->out_mode[count->id]) {
133  	case I8254_MODE_RATE_GENERATOR:
134  		/* Rate Generator decrements 0 by one and the counter "wraps around" */
135  		*ceiling = (priv->preset[count->id] == 0) ? U16_MAX : priv->preset[count->id];
136  		break;
137  	case I8254_MODE_SQUARE_WAVE_MODE:
138  		if (priv->preset[count->id] % 2)
139  			*ceiling = priv->preset[count->id] - 1;
140  		else if (priv->preset[count->id] == 0)
141  			/* Square Wave Mode decrements 0 by two and the counter "wraps around" */
142  			*ceiling = U16_MAX - 1;
143  		else
144  			*ceiling = priv->preset[count->id];
145  		break;
146  	default:
147  		*ceiling = U16_MAX;
148  		break;
149  	}
150  
151  	mutex_unlock(&priv->lock);
152  
153  	return 0;
154  }
155  
i8254_count_mode_read(struct counter_device * const counter,struct counter_count * const count,enum counter_count_mode * const count_mode)156  static int i8254_count_mode_read(struct counter_device *const counter,
157  				 struct counter_count *const count,
158  				 enum counter_count_mode *const count_mode)
159  {
160  	const struct i8254 *const priv = counter_priv(counter);
161  
162  	switch (priv->out_mode[count->id]) {
163  	case I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT:
164  		*count_mode = COUNTER_COUNT_MODE_INTERRUPT_ON_TERMINAL_COUNT;
165  		return 0;
166  	case I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT:
167  		*count_mode = COUNTER_COUNT_MODE_HARDWARE_RETRIGGERABLE_ONESHOT;
168  		return 0;
169  	case I8254_MODE_RATE_GENERATOR:
170  		*count_mode = COUNTER_COUNT_MODE_RATE_GENERATOR;
171  		return 0;
172  	case I8254_MODE_SQUARE_WAVE_MODE:
173  		*count_mode = COUNTER_COUNT_MODE_SQUARE_WAVE_MODE;
174  		return 0;
175  	case I8254_MODE_SOFTWARE_TRIGGERED_STROBE:
176  		*count_mode = COUNTER_COUNT_MODE_SOFTWARE_TRIGGERED_STROBE;
177  		return 0;
178  	case I8254_MODE_HARDWARE_TRIGGERED_STROBE:
179  		*count_mode = COUNTER_COUNT_MODE_HARDWARE_TRIGGERED_STROBE;
180  		return 0;
181  	default:
182  		/* should never reach this path */
183  		return -EINVAL;
184  	}
185  }
186  
i8254_count_mode_write(struct counter_device * const counter,struct counter_count * const count,const enum counter_count_mode count_mode)187  static int i8254_count_mode_write(struct counter_device *const counter,
188  				  struct counter_count *const count,
189  				  const enum counter_count_mode count_mode)
190  {
191  	struct i8254 *const priv = counter_priv(counter);
192  	u8 out_mode;
193  	int ret;
194  
195  	switch (count_mode) {
196  	case COUNTER_COUNT_MODE_INTERRUPT_ON_TERMINAL_COUNT:
197  		out_mode = I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT;
198  		break;
199  	case COUNTER_COUNT_MODE_HARDWARE_RETRIGGERABLE_ONESHOT:
200  		out_mode = I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT;
201  		break;
202  	case COUNTER_COUNT_MODE_RATE_GENERATOR:
203  		out_mode = I8254_MODE_RATE_GENERATOR;
204  		break;
205  	case COUNTER_COUNT_MODE_SQUARE_WAVE_MODE:
206  		out_mode = I8254_MODE_SQUARE_WAVE_MODE;
207  		break;
208  	case COUNTER_COUNT_MODE_SOFTWARE_TRIGGERED_STROBE:
209  		out_mode = I8254_MODE_SOFTWARE_TRIGGERED_STROBE;
210  		break;
211  	case COUNTER_COUNT_MODE_HARDWARE_TRIGGERED_STROBE:
212  		out_mode = I8254_MODE_HARDWARE_TRIGGERED_STROBE;
213  		break;
214  	default:
215  		/* should never reach this path */
216  		return -EINVAL;
217  	}
218  
219  	mutex_lock(&priv->lock);
220  
221  	/* Counter Register is cleared when the counter is programmed */
222  	priv->preset[count->id] = 0;
223  	priv->out_mode[count->id] = out_mode;
224  	ret = regmap_write(priv->map, I8254_CONTROL_REG,
225  			   I8254_PROGRAM_COUNTER(count->id, out_mode));
226  
227  	mutex_unlock(&priv->lock);
228  
229  	return ret;
230  }
231  
i8254_count_floor_read(struct counter_device * const counter,struct counter_count * const count,u64 * const floor)232  static int i8254_count_floor_read(struct counter_device *const counter,
233  				  struct counter_count *const count, u64 *const floor)
234  {
235  	struct i8254 *const priv = counter_priv(counter);
236  
237  	mutex_lock(&priv->lock);
238  
239  	switch (priv->out_mode[count->id]) {
240  	case I8254_MODE_RATE_GENERATOR:
241  		/* counter is always reloaded after 1, but 0 is a possible reload value */
242  		*floor = (priv->preset[count->id] == 0) ? 0 : 1;
243  		break;
244  	case I8254_MODE_SQUARE_WAVE_MODE:
245  		/* counter is always reloaded after 2 for even preset values */
246  		*floor = (priv->preset[count->id] % 2 || priv->preset[count->id] == 0) ? 0 : 2;
247  		break;
248  	default:
249  		*floor = 0;
250  		break;
251  	}
252  
253  	mutex_unlock(&priv->lock);
254  
255  	return 0;
256  }
257  
i8254_count_preset_read(struct counter_device * const counter,struct counter_count * const count,u64 * const preset)258  static int i8254_count_preset_read(struct counter_device *const counter,
259  				   struct counter_count *const count, u64 *const preset)
260  {
261  	const struct i8254 *const priv = counter_priv(counter);
262  
263  	*preset = priv->preset[count->id];
264  
265  	return 0;
266  }
267  
i8254_count_preset_write(struct counter_device * const counter,struct counter_count * const count,const u64 preset)268  static int i8254_count_preset_write(struct counter_device *const counter,
269  				    struct counter_count *const count, const u64 preset)
270  {
271  	struct i8254 *const priv = counter_priv(counter);
272  	int ret;
273  	u8 value[2];
274  
275  	if (preset > U16_MAX)
276  		return -ERANGE;
277  
278  	mutex_lock(&priv->lock);
279  
280  	if (priv->out_mode[count->id] == I8254_MODE_RATE_GENERATOR ||
281  	    priv->out_mode[count->id] == I8254_MODE_SQUARE_WAVE_MODE) {
282  		if (preset == 1) {
283  			mutex_unlock(&priv->lock);
284  			return -EINVAL;
285  		}
286  	}
287  
288  	priv->preset[count->id] = preset;
289  
290  	put_unaligned_le16(preset, value);
291  	ret = regmap_noinc_write(priv->map, I8254_COUNTER_REG(count->id), value, 2);
292  
293  	mutex_unlock(&priv->lock);
294  
295  	return ret;
296  }
297  
i8254_init_hw(struct regmap * const map)298  static int i8254_init_hw(struct regmap *const map)
299  {
300  	unsigned long i;
301  	int ret;
302  
303  	for (i = 0; i < I8254_NUM_COUNTERS; i++) {
304  		/* Initialize each counter to Mode 0 */
305  		ret = regmap_write(map, I8254_CONTROL_REG,
306  				   I8254_PROGRAM_COUNTER(i, I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT));
307  		if (ret)
308  			return ret;
309  	}
310  
311  	return 0;
312  }
313  
314  static const struct counter_ops i8254_ops = {
315  	.count_read = i8254_count_read,
316  	.function_read = i8254_function_read,
317  	.action_read = i8254_action_read,
318  };
319  
320  #define I8254_SIGNAL(_id, _name) {		\
321  	.id = (_id),				\
322  	.name = (_name),			\
323  }
324  
325  static struct counter_signal i8254_signals[] = {
326  	I8254_SIGNAL(0, "CLK 0"), I8254_SIGNAL(1, "GATE 0"),
327  	I8254_SIGNAL(2, "CLK 1"), I8254_SIGNAL(3, "GATE 1"),
328  	I8254_SIGNAL(4, "CLK 2"), I8254_SIGNAL(5, "GATE 2"),
329  };
330  
331  static const enum counter_synapse_action i8254_clk_actions[] = {
332  	COUNTER_SYNAPSE_ACTION_FALLING_EDGE,
333  };
334  static const enum counter_synapse_action i8254_gate_actions[] = {
335  	COUNTER_SYNAPSE_ACTION_NONE,
336  	COUNTER_SYNAPSE_ACTION_RISING_EDGE,
337  };
338  
339  #define I8254_SYNAPSES_BASE(_id) ((_id) * I8254_SYNAPSES_PER_COUNT)
340  #define I8254_SYNAPSE_CLK(_id) {					\
341  	.actions_list	= i8254_clk_actions,				\
342  	.num_actions	= ARRAY_SIZE(i8254_clk_actions),		\
343  	.signal		= &i8254_signals[I8254_SYNAPSES_BASE(_id) + 0],	\
344  }
345  #define I8254_SYNAPSE_GATE(_id) {					\
346  	.actions_list	= i8254_gate_actions,				\
347  	.num_actions	= ARRAY_SIZE(i8254_gate_actions),		\
348  	.signal		= &i8254_signals[I8254_SYNAPSES_BASE(_id) + 1],	\
349  }
350  
351  static struct counter_synapse i8254_synapses[] = {
352  	I8254_SYNAPSE_CLK(0), I8254_SYNAPSE_GATE(0),
353  	I8254_SYNAPSE_CLK(1), I8254_SYNAPSE_GATE(1),
354  	I8254_SYNAPSE_CLK(2), I8254_SYNAPSE_GATE(2),
355  };
356  
357  static const enum counter_function i8254_functions_list[] = {
358  	COUNTER_FUNCTION_DECREASE,
359  };
360  
361  static const enum counter_count_mode i8254_count_modes[] = {
362  	COUNTER_COUNT_MODE_INTERRUPT_ON_TERMINAL_COUNT,
363  	COUNTER_COUNT_MODE_HARDWARE_RETRIGGERABLE_ONESHOT,
364  	COUNTER_COUNT_MODE_RATE_GENERATOR,
365  	COUNTER_COUNT_MODE_SQUARE_WAVE_MODE,
366  	COUNTER_COUNT_MODE_SOFTWARE_TRIGGERED_STROBE,
367  	COUNTER_COUNT_MODE_HARDWARE_TRIGGERED_STROBE,
368  };
369  
370  static DEFINE_COUNTER_AVAILABLE(i8254_count_modes_available, i8254_count_modes);
371  
372  static struct counter_comp i8254_count_ext[] = {
373  	COUNTER_COMP_CEILING(i8254_count_ceiling_read, NULL),
374  	COUNTER_COMP_COUNT_MODE(i8254_count_mode_read, i8254_count_mode_write,
375  				i8254_count_modes_available),
376  	COUNTER_COMP_FLOOR(i8254_count_floor_read, NULL),
377  	COUNTER_COMP_PRESET(i8254_count_preset_read, i8254_count_preset_write),
378  };
379  
380  #define I8254_COUNT(_id, _name) {				\
381  	.id = (_id),						\
382  	.name = (_name),					\
383  	.functions_list = i8254_functions_list,			\
384  	.num_functions = ARRAY_SIZE(i8254_functions_list),	\
385  	.synapses = &i8254_synapses[I8254_SYNAPSES_BASE(_id)],	\
386  	.num_synapses =	I8254_SYNAPSES_PER_COUNT,		\
387  	.ext = i8254_count_ext,					\
388  	.num_ext = ARRAY_SIZE(i8254_count_ext)			\
389  }
390  
391  static struct counter_count i8254_counts[I8254_NUM_COUNTERS] = {
392  	I8254_COUNT(0, "Counter 0"), I8254_COUNT(1, "Counter 1"), I8254_COUNT(2, "Counter 2"),
393  };
394  
395  /**
396   * devm_i8254_regmap_register - Register an i8254 Counter device
397   * @dev: device that is registering this i8254 Counter device
398   * @config: configuration for i8254_regmap_config
399   *
400   * Registers an Intel 8254 Programmable Interval Timer Counter device. Returns 0 on success and
401   * negative error number on failure.
402   */
devm_i8254_regmap_register(struct device * const dev,const struct i8254_regmap_config * const config)403  int devm_i8254_regmap_register(struct device *const dev,
404  			       const struct i8254_regmap_config *const config)
405  {
406  	struct counter_device *counter;
407  	struct i8254 *priv;
408  	int err;
409  
410  	if (!config->parent)
411  		return -EINVAL;
412  
413  	if (!config->map)
414  		return -EINVAL;
415  
416  	counter = devm_counter_alloc(dev, sizeof(*priv));
417  	if (!counter)
418  		return -ENOMEM;
419  	priv = counter_priv(counter);
420  	priv->map = config->map;
421  
422  	counter->name = dev_name(config->parent);
423  	counter->parent = config->parent;
424  	counter->ops = &i8254_ops;
425  	counter->counts = i8254_counts;
426  	counter->num_counts = ARRAY_SIZE(i8254_counts);
427  	counter->signals = i8254_signals;
428  	counter->num_signals = ARRAY_SIZE(i8254_signals);
429  
430  	mutex_init(&priv->lock);
431  
432  	err = i8254_init_hw(priv->map);
433  	if (err)
434  		return err;
435  
436  	err = devm_counter_add(dev, counter);
437  	if (err < 0)
438  		return dev_err_probe(dev, err, "Failed to add counter\n");
439  
440  	return 0;
441  }
442  EXPORT_SYMBOL_NS_GPL(devm_i8254_regmap_register, I8254);
443  
444  MODULE_AUTHOR("William Breathitt Gray");
445  MODULE_DESCRIPTION("Intel 8254 Programmable Interval Timer");
446  MODULE_LICENSE("GPL");
447  MODULE_IMPORT_NS(COUNTER);
448