// SPDX-License-Identifier: GPL-2.0 /* * Intel 8254 Programmable Interval Timer * Copyright (C) William Breathitt Gray */ #include #include #include #include #include #include #include #include #include #include #include #include #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");