xref: /linux/drivers/counter/i8254.c (revision eb01fe7abbe2d0b38824d2a93fdb4cc3eaf2ccc1)
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 
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 
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 
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 
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 
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 
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 
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 
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 
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 
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  */
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