| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Intel 8254 Programmable Interval Timer
- * Copyright (C) William Breathitt Gray
- */
- #include <linux/bitfield.h>
- #include <linux/bits.h>
- #include <linux/counter.h>
- #include <linux/device.h>
- #include <linux/err.h>
- #include <linux/export.h>
- #include <linux/i8254.h>
- #include <linux/limits.h>
- #include <linux/module.h>
- #include <linux/mutex.h>
- #include <linux/regmap.h>
- #include <linux/unaligned.h>
- #define I8254_COUNTER_REG(_counter) (_counter)
- #define I8254_CONTROL_REG 0x3
- #define I8254_SC GENMASK(7, 6)
- #define I8254_RW GENMASK(5, 4)
- #define I8254_M GENMASK(3, 1)
- #define I8254_CONTROL(_sc, _rw, _m) \
- (u8_encode_bits(_sc, I8254_SC) | u8_encode_bits(_rw, I8254_RW) | \
- u8_encode_bits(_m, I8254_M))
- #define I8254_RW_TWO_BYTE 0x3
- #define I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT 0
- #define I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT 1
- #define I8254_MODE_RATE_GENERATOR 2
- #define I8254_MODE_SQUARE_WAVE_MODE 3
- #define I8254_MODE_SOFTWARE_TRIGGERED_STROBE 4
- #define I8254_MODE_HARDWARE_TRIGGERED_STROBE 5
- #define I8254_COUNTER_LATCH(_counter) I8254_CONTROL(_counter, 0x0, 0x0)
- #define I8254_PROGRAM_COUNTER(_counter, _mode) I8254_CONTROL(_counter, I8254_RW_TWO_BYTE, _mode)
- #define I8254_NUM_COUNTERS 3
- /**
- * struct i8254 - I8254 device private data structure
- * @lock: synchronization lock to prevent I/O race conditions
- * @preset: array of Counter Register states
- * @out_mode: array of mode configuration states
- * @map: Regmap for the device
- */
- struct i8254 {
- struct mutex lock;
- u16 preset[I8254_NUM_COUNTERS];
- u8 out_mode[I8254_NUM_COUNTERS];
- struct regmap *map;
- };
- static int i8254_count_read(struct counter_device *const counter, struct counter_count *const count,
- u64 *const val)
- {
- struct i8254 *const priv = counter_priv(counter);
- int ret;
- u8 value[2];
- mutex_lock(&priv->lock);
- ret = regmap_write(priv->map, I8254_CONTROL_REG, I8254_COUNTER_LATCH(count->id));
- if (ret) {
- mutex_unlock(&priv->lock);
- return ret;
- }
- ret = regmap_noinc_read(priv->map, I8254_COUNTER_REG(count->id), value, sizeof(value));
- if (ret) {
- mutex_unlock(&priv->lock);
- return ret;
- }
- mutex_unlock(&priv->lock);
- *val = get_unaligned_le16(value);
- return ret;
- }
- static int i8254_function_read(struct counter_device *const counter,
- struct counter_count *const count,
- enum counter_function *const function)
- {
- *function = COUNTER_FUNCTION_DECREASE;
- return 0;
- }
- #define I8254_SYNAPSES_PER_COUNT 2
- #define I8254_SIGNAL_ID_CLK 0
- #define I8254_SIGNAL_ID_GATE 1
- static int i8254_action_read(struct counter_device *const counter,
- struct counter_count *const count,
- struct counter_synapse *const synapse,
- enum counter_synapse_action *const action)
- {
- struct i8254 *const priv = counter_priv(counter);
- switch (synapse->signal->id % I8254_SYNAPSES_PER_COUNT) {
- case I8254_SIGNAL_ID_CLK:
- *action = COUNTER_SYNAPSE_ACTION_FALLING_EDGE;
- return 0;
- case I8254_SIGNAL_ID_GATE:
- switch (priv->out_mode[count->id]) {
- case I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT:
- case I8254_MODE_RATE_GENERATOR:
- case I8254_MODE_SQUARE_WAVE_MODE:
- case I8254_MODE_HARDWARE_TRIGGERED_STROBE:
- *action = COUNTER_SYNAPSE_ACTION_RISING_EDGE;
- return 0;
- default:
- *action = COUNTER_SYNAPSE_ACTION_NONE;
- return 0;
- }
- default:
- /* should never reach this path */
- return -EINVAL;
- }
- }
- static int i8254_count_ceiling_read(struct counter_device *const counter,
- struct counter_count *const count, u64 *const ceiling)
- {
- struct i8254 *const priv = counter_priv(counter);
- mutex_lock(&priv->lock);
- switch (priv->out_mode[count->id]) {
- case I8254_MODE_RATE_GENERATOR:
- /* Rate Generator decrements 0 by one and the counter "wraps around" */
- *ceiling = (priv->preset[count->id] == 0) ? U16_MAX : priv->preset[count->id];
- break;
- case I8254_MODE_SQUARE_WAVE_MODE:
- if (priv->preset[count->id] % 2)
- *ceiling = priv->preset[count->id] - 1;
- else if (priv->preset[count->id] == 0)
- /* Square Wave Mode decrements 0 by two and the counter "wraps around" */
- *ceiling = U16_MAX - 1;
- else
- *ceiling = priv->preset[count->id];
- break;
- default:
- *ceiling = U16_MAX;
- break;
- }
- mutex_unlock(&priv->lock);
- return 0;
- }
- static int i8254_count_mode_read(struct counter_device *const counter,
- struct counter_count *const count,
- enum counter_count_mode *const count_mode)
- {
- const struct i8254 *const priv = counter_priv(counter);
- switch (priv->out_mode[count->id]) {
- case I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT:
- *count_mode = COUNTER_COUNT_MODE_INTERRUPT_ON_TERMINAL_COUNT;
- return 0;
- case I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT:
- *count_mode = COUNTER_COUNT_MODE_HARDWARE_RETRIGGERABLE_ONESHOT;
- return 0;
- case I8254_MODE_RATE_GENERATOR:
- *count_mode = COUNTER_COUNT_MODE_RATE_GENERATOR;
- return 0;
- case I8254_MODE_SQUARE_WAVE_MODE:
- *count_mode = COUNTER_COUNT_MODE_SQUARE_WAVE_MODE;
- return 0;
- case I8254_MODE_SOFTWARE_TRIGGERED_STROBE:
- *count_mode = COUNTER_COUNT_MODE_SOFTWARE_TRIGGERED_STROBE;
- return 0;
- case I8254_MODE_HARDWARE_TRIGGERED_STROBE:
- *count_mode = COUNTER_COUNT_MODE_HARDWARE_TRIGGERED_STROBE;
- return 0;
- default:
- /* should never reach this path */
- return -EINVAL;
- }
- }
- static int i8254_count_mode_write(struct counter_device *const counter,
- struct counter_count *const count,
- const enum counter_count_mode count_mode)
- {
- struct i8254 *const priv = counter_priv(counter);
- u8 out_mode;
- int ret;
- switch (count_mode) {
- case COUNTER_COUNT_MODE_INTERRUPT_ON_TERMINAL_COUNT:
- out_mode = I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT;
- break;
- case COUNTER_COUNT_MODE_HARDWARE_RETRIGGERABLE_ONESHOT:
- out_mode = I8254_MODE_HARDWARE_RETRIGGERABLE_ONESHOT;
- break;
- case COUNTER_COUNT_MODE_RATE_GENERATOR:
- out_mode = I8254_MODE_RATE_GENERATOR;
- break;
- case COUNTER_COUNT_MODE_SQUARE_WAVE_MODE:
- out_mode = I8254_MODE_SQUARE_WAVE_MODE;
- break;
- case COUNTER_COUNT_MODE_SOFTWARE_TRIGGERED_STROBE:
- out_mode = I8254_MODE_SOFTWARE_TRIGGERED_STROBE;
- break;
- case COUNTER_COUNT_MODE_HARDWARE_TRIGGERED_STROBE:
- out_mode = I8254_MODE_HARDWARE_TRIGGERED_STROBE;
- break;
- default:
- /* should never reach this path */
- return -EINVAL;
- }
- mutex_lock(&priv->lock);
- /* Counter Register is cleared when the counter is programmed */
- priv->preset[count->id] = 0;
- priv->out_mode[count->id] = out_mode;
- ret = regmap_write(priv->map, I8254_CONTROL_REG,
- I8254_PROGRAM_COUNTER(count->id, out_mode));
- mutex_unlock(&priv->lock);
- return ret;
- }
- static int i8254_count_floor_read(struct counter_device *const counter,
- struct counter_count *const count, u64 *const floor)
- {
- struct i8254 *const priv = counter_priv(counter);
- mutex_lock(&priv->lock);
- switch (priv->out_mode[count->id]) {
- case I8254_MODE_RATE_GENERATOR:
- /* counter is always reloaded after 1, but 0 is a possible reload value */
- *floor = (priv->preset[count->id] == 0) ? 0 : 1;
- break;
- case I8254_MODE_SQUARE_WAVE_MODE:
- /* counter is always reloaded after 2 for even preset values */
- *floor = (priv->preset[count->id] % 2 || priv->preset[count->id] == 0) ? 0 : 2;
- break;
- default:
- *floor = 0;
- break;
- }
- mutex_unlock(&priv->lock);
- return 0;
- }
- static int i8254_count_preset_read(struct counter_device *const counter,
- struct counter_count *const count, u64 *const preset)
- {
- const struct i8254 *const priv = counter_priv(counter);
- *preset = priv->preset[count->id];
- return 0;
- }
- static int i8254_count_preset_write(struct counter_device *const counter,
- struct counter_count *const count, const u64 preset)
- {
- struct i8254 *const priv = counter_priv(counter);
- int ret;
- u8 value[2];
- if (preset > U16_MAX)
- return -ERANGE;
- mutex_lock(&priv->lock);
- if (priv->out_mode[count->id] == I8254_MODE_RATE_GENERATOR ||
- priv->out_mode[count->id] == I8254_MODE_SQUARE_WAVE_MODE) {
- if (preset == 1) {
- mutex_unlock(&priv->lock);
- return -EINVAL;
- }
- }
- priv->preset[count->id] = preset;
- put_unaligned_le16(preset, value);
- ret = regmap_noinc_write(priv->map, I8254_COUNTER_REG(count->id), value, 2);
- mutex_unlock(&priv->lock);
- return ret;
- }
- static int i8254_init_hw(struct regmap *const map)
- {
- unsigned long i;
- int ret;
- for (i = 0; i < I8254_NUM_COUNTERS; i++) {
- /* Initialize each counter to Mode 0 */
- ret = regmap_write(map, I8254_CONTROL_REG,
- I8254_PROGRAM_COUNTER(i, I8254_MODE_INTERRUPT_ON_TERMINAL_COUNT));
- if (ret)
- return ret;
- }
- return 0;
- }
- static const struct counter_ops i8254_ops = {
- .count_read = i8254_count_read,
- .function_read = i8254_function_read,
- .action_read = i8254_action_read,
- };
- #define I8254_SIGNAL(_id, _name) { \
- .id = (_id), \
- .name = (_name), \
- }
- static struct counter_signal i8254_signals[] = {
- I8254_SIGNAL(0, "CLK 0"), I8254_SIGNAL(1, "GATE 0"),
- I8254_SIGNAL(2, "CLK 1"), I8254_SIGNAL(3, "GATE 1"),
- I8254_SIGNAL(4, "CLK 2"), I8254_SIGNAL(5, "GATE 2"),
- };
- static const enum counter_synapse_action i8254_clk_actions[] = {
- COUNTER_SYNAPSE_ACTION_FALLING_EDGE,
- };
- static const enum counter_synapse_action i8254_gate_actions[] = {
- COUNTER_SYNAPSE_ACTION_NONE,
- COUNTER_SYNAPSE_ACTION_RISING_EDGE,
- };
- #define I8254_SYNAPSES_BASE(_id) ((_id) * I8254_SYNAPSES_PER_COUNT)
- #define I8254_SYNAPSE_CLK(_id) { \
- .actions_list = i8254_clk_actions, \
- .num_actions = ARRAY_SIZE(i8254_clk_actions), \
- .signal = &i8254_signals[I8254_SYNAPSES_BASE(_id) + 0], \
- }
- #define I8254_SYNAPSE_GATE(_id) { \
- .actions_list = i8254_gate_actions, \
- .num_actions = ARRAY_SIZE(i8254_gate_actions), \
- .signal = &i8254_signals[I8254_SYNAPSES_BASE(_id) + 1], \
- }
- static struct counter_synapse i8254_synapses[] = {
- I8254_SYNAPSE_CLK(0), I8254_SYNAPSE_GATE(0),
- I8254_SYNAPSE_CLK(1), I8254_SYNAPSE_GATE(1),
- I8254_SYNAPSE_CLK(2), I8254_SYNAPSE_GATE(2),
- };
- static const enum counter_function i8254_functions_list[] = {
- COUNTER_FUNCTION_DECREASE,
- };
- static const enum counter_count_mode i8254_count_modes[] = {
- COUNTER_COUNT_MODE_INTERRUPT_ON_TERMINAL_COUNT,
- COUNTER_COUNT_MODE_HARDWARE_RETRIGGERABLE_ONESHOT,
- COUNTER_COUNT_MODE_RATE_GENERATOR,
- COUNTER_COUNT_MODE_SQUARE_WAVE_MODE,
- COUNTER_COUNT_MODE_SOFTWARE_TRIGGERED_STROBE,
- COUNTER_COUNT_MODE_HARDWARE_TRIGGERED_STROBE,
- };
- static DEFINE_COUNTER_AVAILABLE(i8254_count_modes_available, i8254_count_modes);
- static struct counter_comp i8254_count_ext[] = {
- COUNTER_COMP_CEILING(i8254_count_ceiling_read, NULL),
- COUNTER_COMP_COUNT_MODE(i8254_count_mode_read, i8254_count_mode_write,
- i8254_count_modes_available),
- COUNTER_COMP_FLOOR(i8254_count_floor_read, NULL),
- COUNTER_COMP_PRESET(i8254_count_preset_read, i8254_count_preset_write),
- };
- #define I8254_COUNT(_id, _name) { \
- .id = (_id), \
- .name = (_name), \
- .functions_list = i8254_functions_list, \
- .num_functions = ARRAY_SIZE(i8254_functions_list), \
- .synapses = &i8254_synapses[I8254_SYNAPSES_BASE(_id)], \
- .num_synapses = I8254_SYNAPSES_PER_COUNT, \
- .ext = i8254_count_ext, \
- .num_ext = ARRAY_SIZE(i8254_count_ext) \
- }
- static struct counter_count i8254_counts[I8254_NUM_COUNTERS] = {
- I8254_COUNT(0, "Counter 0"), I8254_COUNT(1, "Counter 1"), I8254_COUNT(2, "Counter 2"),
- };
- /**
- * devm_i8254_regmap_register - Register an i8254 Counter device
- * @dev: device that is registering this i8254 Counter device
- * @config: configuration for i8254_regmap_config
- *
- * Registers an Intel 8254 Programmable Interval Timer Counter device. Returns 0 on success and
- * negative error number on failure.
- */
- int devm_i8254_regmap_register(struct device *const dev,
- const struct i8254_regmap_config *const config)
- {
- struct counter_device *counter;
- struct i8254 *priv;
- int err;
- if (!config->parent)
- return -EINVAL;
- if (!config->map)
- return -EINVAL;
- counter = devm_counter_alloc(dev, sizeof(*priv));
- if (!counter)
- return -ENOMEM;
- priv = counter_priv(counter);
- priv->map = config->map;
- counter->name = dev_name(config->parent);
- counter->parent = config->parent;
- counter->ops = &i8254_ops;
- counter->counts = i8254_counts;
- counter->num_counts = ARRAY_SIZE(i8254_counts);
- counter->signals = i8254_signals;
- counter->num_signals = ARRAY_SIZE(i8254_signals);
- mutex_init(&priv->lock);
- err = i8254_init_hw(priv->map);
- if (err)
- return err;
- err = devm_counter_add(dev, counter);
- if (err < 0)
- return dev_err_probe(dev, err, "Failed to add counter\n");
- return 0;
- }
- EXPORT_SYMBOL_NS_GPL(devm_i8254_regmap_register, I8254);
- MODULE_AUTHOR("William Breathitt Gray");
- MODULE_DESCRIPTION("Intel 8254 Programmable Interval Timer");
- MODULE_LICENSE("GPL");
- MODULE_IMPORT_NS(COUNTER);
|