// SPDX-License-Identifier: GPL-2.0 /* * Renesas RZ/G2L MTU3a Counter driver * * Copyright (C) 2022 Renesas Electronics Corporation */ #include <linux/clk.h> #include <linux/counter.h> #include <linux/mfd/rz-mtu3.h> #include <linux/module.h> #include <linux/platform_device.h> #include <linux/pm_runtime.h> #include <linux/types.h> /* * Register descriptions * TSR: Timer Status Register * TMDR1: Timer Mode Register 1 * TMDR3: Timer Mode Register 3 * TIOR: Timer I/O Control Register * TCR: Timer Control Register * TCNT: Timer Counter * TGRA: Timer general register A * TCNTLW: Timer Longword Counter * TGRALW: Timer longword general register A */ #define RZ_MTU3_TSR_TCFD BIT(7) /* Count Direction Flag */ #define RZ_MTU3_TMDR1_PH_CNT_MODE_1 (4) /* Phase counting mode 1 */ #define RZ_MTU3_TMDR1_PH_CNT_MODE_2 (5) /* Phase counting mode 2 */ #define RZ_MTU3_TMDR1_PH_CNT_MODE_3 (6) /* Phase counting mode 3 */ #define RZ_MTU3_TMDR1_PH_CNT_MODE_4 (7) /* Phase counting mode 4 */ #define RZ_MTU3_TMDR1_PH_CNT_MODE_5 (9) /* Phase counting mode 5 */ #define RZ_MTU3_TMDR1_PH_CNT_MODE_MASK (0xf) /* * LWA: MTU1/MTU2 Combination Longword Access Control * 0: 16-bit, 1: 32-bit */ #define RZ_MTU3_TMDR3_LWA (0) /* * PHCKSEL: External Input Phase Clock Select * 0: MTCLKA and MTCLKB, 1: MTCLKC and MTCLKD */ #define RZ_MTU3_TMDR3_PHCKSEL (1) #define RZ_MTU3_16_BIT_MTU1_CH (0) #define RZ_MTU3_16_BIT_MTU2_CH (1) #define RZ_MTU3_32_BIT_CH (2) #define RZ_MTU3_TIOR_NO_OUTPUT (0) /* Output prohibited */ #define RZ_MTU3_TIOR_IC_BOTH (10) /* Input capture at both edges */ #define SIGNAL_A_ID (0) #define SIGNAL_B_ID (1) #define SIGNAL_C_ID (2) #define SIGNAL_D_ID (3) #define RZ_MTU3_MAX_HW_CNTR_CHANNELS (2) #define RZ_MTU3_MAX_LOGICAL_CNTR_CHANNELS (3) /** * struct rz_mtu3_cnt - MTU3 counter private data * * @clk: MTU3 module clock * @lock: Lock to prevent concurrent access for ceiling and count * @ch: HW channels for the counters * @count_is_enabled: Enabled state of Counter value channel * @mtu_16bit_max: Cache for 16-bit counters * @mtu_32bit_max: Cache for 32-bit counters */ struct rz_mtu3_cnt { struct clk *clk; struct mutex lock; struct rz_mtu3_channel *ch; bool count_is_enabled[RZ_MTU3_MAX_LOGICAL_CNTR_CHANNELS]; union { u16 mtu_16bit_max[RZ_MTU3_MAX_HW_CNTR_CHANNELS]; u32 mtu_32bit_max; }; }; static const enum counter_function rz_mtu3_count_functions[] = { COUNTER_FUNCTION_QUADRATURE_X4, COUNTER_FUNCTION_PULSE_DIRECTION, COUNTER_FUNCTION_QUADRATURE_X2_B, }; static inline size_t rz_mtu3_get_hw_ch(const size_t id) { return (id == RZ_MTU3_32_BIT_CH) ? 0 : id; } static inline struct rz_mtu3_channel *rz_mtu3_get_ch(struct counter_device *counter, int id) { struct rz_mtu3_cnt *const priv = counter_priv(counter); const size_t ch_id = rz_mtu3_get_hw_ch(id); return &priv->ch[ch_id]; } static bool rz_mtu3_is_counter_invalid(struct counter_device *counter, int id) { struct rz_mtu3_cnt *const priv = counter_priv(counter); unsigned long tmdr; pm_runtime_get_sync(priv->ch->dev); tmdr = rz_mtu3_shared_reg_read(priv->ch, RZ_MTU3_TMDR3); pm_runtime_put(priv->ch->dev); if (id == RZ_MTU3_32_BIT_CH && test_bit(RZ_MTU3_TMDR3_LWA, &tmdr)) return false; if (id != RZ_MTU3_32_BIT_CH && !test_bit(RZ_MTU3_TMDR3_LWA, &tmdr)) return false; return true; } static int rz_mtu3_lock_if_counter_is_valid(struct counter_device *counter, struct rz_mtu3_channel *const ch, struct rz_mtu3_cnt *const priv, int id) { mutex_lock(&priv->lock); if (ch->is_busy && !priv->count_is_enabled[id]) { mutex_unlock(&priv->lock); return -EINVAL; } if (rz_mtu3_is_counter_invalid(counter, id)) { mutex_unlock(&priv->lock); return -EBUSY; } return 0; } static int rz_mtu3_lock_if_count_is_enabled(struct rz_mtu3_channel *const ch, struct rz_mtu3_cnt *const priv, int id) { mutex_lock(&priv->lock); if (ch->is_busy && !priv->count_is_enabled[id]) { mutex_unlock(&priv->lock); return -EINVAL; } return 0; } static int rz_mtu3_count_read(struct counter_device *counter, struct counter_count *count, u64 *val) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; ret = rz_mtu3_lock_if_counter_is_valid(counter, ch, priv, count->id); if (ret) return ret; pm_runtime_get_sync(ch->dev); if (count->id == RZ_MTU3_32_BIT_CH) *val = rz_mtu3_32bit_ch_read(ch, RZ_MTU3_TCNTLW); else *val = rz_mtu3_16bit_ch_read(ch, RZ_MTU3_TCNT); pm_runtime_put(ch->dev); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_count_write(struct counter_device *counter, struct counter_count *count, const u64 val) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; ret = rz_mtu3_lock_if_counter_is_valid(counter, ch, priv, count->id); if (ret) return ret; pm_runtime_get_sync(ch->dev); if (count->id == RZ_MTU3_32_BIT_CH) rz_mtu3_32bit_ch_write(ch, RZ_MTU3_TCNTLW, val); else rz_mtu3_16bit_ch_write(ch, RZ_MTU3_TCNT, val); pm_runtime_put(ch->dev); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_count_function_read_helper(struct rz_mtu3_channel *const ch, struct rz_mtu3_cnt *const priv, enum counter_function *function) { u8 timer_mode; pm_runtime_get_sync(ch->dev); timer_mode = rz_mtu3_8bit_ch_read(ch, RZ_MTU3_TMDR1); pm_runtime_put(ch->dev); switch (timer_mode & RZ_MTU3_TMDR1_PH_CNT_MODE_MASK) { case RZ_MTU3_TMDR1_PH_CNT_MODE_1: *function = COUNTER_FUNCTION_QUADRATURE_X4; return 0; case RZ_MTU3_TMDR1_PH_CNT_MODE_2: *function = COUNTER_FUNCTION_PULSE_DIRECTION; return 0; case RZ_MTU3_TMDR1_PH_CNT_MODE_4: *function = COUNTER_FUNCTION_QUADRATURE_X2_B; return 0; default: /* * TODO: * - need to add RZ_MTU3_TMDR1_PH_CNT_MODE_3 * - need to add RZ_MTU3_TMDR1_PH_CNT_MODE_5 */ return -EINVAL; } } static int rz_mtu3_count_function_read(struct counter_device *counter, struct counter_count *count, enum counter_function *function) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; ret = rz_mtu3_lock_if_count_is_enabled(ch, priv, count->id); if (ret) return ret; ret = rz_mtu3_count_function_read_helper(ch, priv, function); mutex_unlock(&priv->lock); return ret; } static int rz_mtu3_count_function_write(struct counter_device *counter, struct counter_count *count, enum counter_function function) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); u8 timer_mode; int ret; ret = rz_mtu3_lock_if_count_is_enabled(ch, priv, count->id); if (ret) return ret; switch (function) { case COUNTER_FUNCTION_QUADRATURE_X4: timer_mode = RZ_MTU3_TMDR1_PH_CNT_MODE_1; break; case COUNTER_FUNCTION_PULSE_DIRECTION: timer_mode = RZ_MTU3_TMDR1_PH_CNT_MODE_2; break; case COUNTER_FUNCTION_QUADRATURE_X2_B: timer_mode = RZ_MTU3_TMDR1_PH_CNT_MODE_4; break; default: /* * TODO: * - need to add RZ_MTU3_TMDR1_PH_CNT_MODE_3 * - need to add RZ_MTU3_TMDR1_PH_CNT_MODE_5 */ mutex_unlock(&priv->lock); return -EINVAL; } pm_runtime_get_sync(ch->dev); rz_mtu3_8bit_ch_write(ch, RZ_MTU3_TMDR1, timer_mode); pm_runtime_put(ch->dev); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_count_direction_read(struct counter_device *counter, struct counter_count *count, enum counter_count_direction *direction) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; u8 tsr; ret = rz_mtu3_lock_if_count_is_enabled(ch, priv, count->id); if (ret) return ret; pm_runtime_get_sync(ch->dev); tsr = rz_mtu3_8bit_ch_read(ch, RZ_MTU3_TSR); pm_runtime_put(ch->dev); *direction = (tsr & RZ_MTU3_TSR_TCFD) ? COUNTER_COUNT_DIRECTION_FORWARD : COUNTER_COUNT_DIRECTION_BACKWARD; mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_count_ceiling_read(struct counter_device *counter, struct counter_count *count, u64 *ceiling) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); const size_t ch_id = rz_mtu3_get_hw_ch(count->id); int ret; ret = rz_mtu3_lock_if_counter_is_valid(counter, ch, priv, count->id); if (ret) return ret; switch (count->id) { case RZ_MTU3_16_BIT_MTU1_CH: case RZ_MTU3_16_BIT_MTU2_CH: *ceiling = priv->mtu_16bit_max[ch_id]; break; case RZ_MTU3_32_BIT_CH: *ceiling = priv->mtu_32bit_max; break; default: /* should never reach this path */ mutex_unlock(&priv->lock); return -EINVAL; } mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_count_ceiling_write(struct counter_device *counter, struct counter_count *count, u64 ceiling) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); const size_t ch_id = rz_mtu3_get_hw_ch(count->id); int ret; ret = rz_mtu3_lock_if_counter_is_valid(counter, ch, priv, count->id); if (ret) return ret; switch (count->id) { case RZ_MTU3_16_BIT_MTU1_CH: case RZ_MTU3_16_BIT_MTU2_CH: if (ceiling > U16_MAX) { mutex_unlock(&priv->lock); return -ERANGE; } priv->mtu_16bit_max[ch_id] = ceiling; break; case RZ_MTU3_32_BIT_CH: if (ceiling > U32_MAX) { mutex_unlock(&priv->lock); return -ERANGE; } priv->mtu_32bit_max = ceiling; break; default: /* should never reach this path */ mutex_unlock(&priv->lock); return -EINVAL; } pm_runtime_get_sync(ch->dev); if (count->id == RZ_MTU3_32_BIT_CH) rz_mtu3_32bit_ch_write(ch, RZ_MTU3_TGRALW, ceiling); else rz_mtu3_16bit_ch_write(ch, RZ_MTU3_TGRA, ceiling); rz_mtu3_8bit_ch_write(ch, RZ_MTU3_TCR, RZ_MTU3_TCR_CCLR_TGRA); pm_runtime_put(ch->dev); mutex_unlock(&priv->lock); return 0; } static void rz_mtu3_32bit_cnt_setting(struct counter_device *counter) { struct rz_mtu3_channel *const ch1 = rz_mtu3_get_ch(counter, 0); struct rz_mtu3_channel *const ch2 = rz_mtu3_get_ch(counter, 1); /* Phase counting mode 1 is used as default in initialization. */ rz_mtu3_8bit_ch_write(ch1, RZ_MTU3_TMDR1, RZ_MTU3_TMDR1_PH_CNT_MODE_1); rz_mtu3_8bit_ch_write(ch1, RZ_MTU3_TCR, RZ_MTU3_TCR_CCLR_TGRA); rz_mtu3_8bit_ch_write(ch1, RZ_MTU3_TIOR, RZ_MTU3_TIOR_IC_BOTH); rz_mtu3_enable(ch1); rz_mtu3_enable(ch2); } static void rz_mtu3_16bit_cnt_setting(struct counter_device *counter, int id) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, id); /* Phase counting mode 1 is used as default in initialization. */ rz_mtu3_8bit_ch_write(ch, RZ_MTU3_TMDR1, RZ_MTU3_TMDR1_PH_CNT_MODE_1); rz_mtu3_8bit_ch_write(ch, RZ_MTU3_TCR, RZ_MTU3_TCR_CCLR_TGRA); rz_mtu3_8bit_ch_write(ch, RZ_MTU3_TIOR, RZ_MTU3_TIOR_NO_OUTPUT); rz_mtu3_enable(ch); } static int rz_mtu3_initialize_counter(struct counter_device *counter, int id) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, id); struct rz_mtu3_channel *const ch1 = rz_mtu3_get_ch(counter, 0); struct rz_mtu3_channel *const ch2 = rz_mtu3_get_ch(counter, 1); switch (id) { case RZ_MTU3_16_BIT_MTU1_CH: case RZ_MTU3_16_BIT_MTU2_CH: if (!rz_mtu3_request_channel(ch)) return -EBUSY; rz_mtu3_16bit_cnt_setting(counter, id); return 0; case RZ_MTU3_32_BIT_CH: /* * 32-bit phase counting need MTU1 and MTU2 to create 32-bit * cascade counter. */ if (!rz_mtu3_request_channel(ch1)) return -EBUSY; if (!rz_mtu3_request_channel(ch2)) { rz_mtu3_release_channel(ch1); return -EBUSY; } rz_mtu3_32bit_cnt_setting(counter); return 0; default: /* should never reach this path */ return -EINVAL; } } static void rz_mtu3_terminate_counter(struct counter_device *counter, int id) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, id); struct rz_mtu3_channel *const ch1 = rz_mtu3_get_ch(counter, 0); struct rz_mtu3_channel *const ch2 = rz_mtu3_get_ch(counter, 1); if (id == RZ_MTU3_32_BIT_CH) { rz_mtu3_release_channel(ch2); rz_mtu3_release_channel(ch1); rz_mtu3_disable(ch2); rz_mtu3_disable(ch1); } else { rz_mtu3_release_channel(ch); rz_mtu3_disable(ch); } } static int rz_mtu3_count_enable_read(struct counter_device *counter, struct counter_count *count, u8 *enable) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_channel *const ch1 = rz_mtu3_get_ch(counter, 0); struct rz_mtu3_channel *const ch2 = rz_mtu3_get_ch(counter, 1); struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; ret = rz_mtu3_lock_if_count_is_enabled(ch, priv, count->id); if (ret) return ret; if (count->id == RZ_MTU3_32_BIT_CH) *enable = rz_mtu3_is_enabled(ch1) && rz_mtu3_is_enabled(ch2); else *enable = rz_mtu3_is_enabled(ch); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_count_enable_write(struct counter_device *counter, struct counter_count *count, u8 enable) { struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret = 0; if (enable) { mutex_lock(&priv->lock); pm_runtime_get_sync(ch->dev); ret = rz_mtu3_initialize_counter(counter, count->id); if (ret == 0) priv->count_is_enabled[count->id] = true; mutex_unlock(&priv->lock); } else { mutex_lock(&priv->lock); rz_mtu3_terminate_counter(counter, count->id); priv->count_is_enabled[count->id] = false; pm_runtime_put(ch->dev); mutex_unlock(&priv->lock); } return ret; } static int rz_mtu3_lock_if_ch0_is_enabled(struct rz_mtu3_cnt *const priv) { mutex_lock(&priv->lock); if (priv->ch->is_busy && !(priv->count_is_enabled[RZ_MTU3_16_BIT_MTU1_CH] || priv->count_is_enabled[RZ_MTU3_32_BIT_CH])) { mutex_unlock(&priv->lock); return -EINVAL; } return 0; } static int rz_mtu3_cascade_counts_enable_get(struct counter_device *counter, u8 *cascade_enable) { struct rz_mtu3_cnt *const priv = counter_priv(counter); unsigned long tmdr; int ret; ret = rz_mtu3_lock_if_ch0_is_enabled(priv); if (ret) return ret; pm_runtime_get_sync(priv->ch->dev); tmdr = rz_mtu3_shared_reg_read(priv->ch, RZ_MTU3_TMDR3); pm_runtime_put(priv->ch->dev); *cascade_enable = test_bit(RZ_MTU3_TMDR3_LWA, &tmdr); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_cascade_counts_enable_set(struct counter_device *counter, u8 cascade_enable) { struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; ret = rz_mtu3_lock_if_ch0_is_enabled(priv); if (ret) return ret; pm_runtime_get_sync(priv->ch->dev); rz_mtu3_shared_reg_update_bit(priv->ch, RZ_MTU3_TMDR3, RZ_MTU3_TMDR3_LWA, cascade_enable); pm_runtime_put(priv->ch->dev); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_ext_input_phase_clock_select_get(struct counter_device *counter, u32 *ext_input_phase_clock_select) { struct rz_mtu3_cnt *const priv = counter_priv(counter); unsigned long tmdr; int ret; ret = rz_mtu3_lock_if_ch0_is_enabled(priv); if (ret) return ret; pm_runtime_get_sync(priv->ch->dev); tmdr = rz_mtu3_shared_reg_read(priv->ch, RZ_MTU3_TMDR3); pm_runtime_put(priv->ch->dev); *ext_input_phase_clock_select = test_bit(RZ_MTU3_TMDR3_PHCKSEL, &tmdr); mutex_unlock(&priv->lock); return 0; } static int rz_mtu3_ext_input_phase_clock_select_set(struct counter_device *counter, u32 ext_input_phase_clock_select) { struct rz_mtu3_cnt *const priv = counter_priv(counter); int ret; ret = rz_mtu3_lock_if_ch0_is_enabled(priv); if (ret) return ret; pm_runtime_get_sync(priv->ch->dev); rz_mtu3_shared_reg_update_bit(priv->ch, RZ_MTU3_TMDR3, RZ_MTU3_TMDR3_PHCKSEL, ext_input_phase_clock_select); pm_runtime_put(priv->ch->dev); mutex_unlock(&priv->lock); return 0; } static struct counter_comp rz_mtu3_count_ext[] = { COUNTER_COMP_DIRECTION(rz_mtu3_count_direction_read), COUNTER_COMP_ENABLE(rz_mtu3_count_enable_read, rz_mtu3_count_enable_write), COUNTER_COMP_CEILING(rz_mtu3_count_ceiling_read, rz_mtu3_count_ceiling_write), }; static const enum counter_synapse_action rz_mtu3_synapse_actions[] = { COUNTER_SYNAPSE_ACTION_BOTH_EDGES, COUNTER_SYNAPSE_ACTION_RISING_EDGE, COUNTER_SYNAPSE_ACTION_NONE, }; static int rz_mtu3_action_read(struct counter_device *counter, struct counter_count *count, struct counter_synapse *synapse, enum counter_synapse_action *action) { const bool is_signal_ab = (synapse->signal->id == SIGNAL_A_ID) || (synapse->signal->id == SIGNAL_B_ID); struct rz_mtu3_channel *const ch = rz_mtu3_get_ch(counter, count->id); struct rz_mtu3_cnt *const priv = counter_priv(counter); enum counter_function function; bool mtclkc_mtclkd; unsigned long tmdr; int ret; ret = rz_mtu3_lock_if_count_is_enabled(ch, priv, count->id); if (ret) return ret; ret = rz_mtu3_count_function_read_helper(ch, priv, &function); if (ret) { mutex_unlock(&priv->lock); return ret; } /* Default action mode */ *action = COUNTER_SYNAPSE_ACTION_NONE; if (count->id != RZ_MTU3_16_BIT_MTU1_CH) { tmdr = rz_mtu3_shared_reg_read(priv->ch, RZ_MTU3_TMDR3); mtclkc_mtclkd = test_bit(RZ_MTU3_TMDR3_PHCKSEL, &tmdr); if ((mtclkc_mtclkd && is_signal_ab) || (!mtclkc_mtclkd && !is_signal_ab)) { mutex_unlock(&priv->lock); return 0; } } switch (function) { case COUNTER_FUNCTION_PULSE_DIRECTION: /* * Rising edges on signal A (signal C) updates the respective * count. The input level of signal B (signal D) determines * direction. */ if (synapse->signal->id == SIGNAL_A_ID || synapse->signal->id == SIGNAL_C_ID) *action = COUNTER_SYNAPSE_ACTION_RISING_EDGE; break; case COUNTER_FUNCTION_QUADRATURE_X2_B: /* * Any state transition on quadrature pair signal B (signal D) * updates the respective count. */ if (synapse->signal->id == SIGNAL_B_ID || synapse->signal->id == SIGNAL_D_ID) *action = COUNTER_SYNAPSE_ACTION_BOTH_EDGES; break; case COUNTER_FUNCTION_QUADRATURE_X4: /* counts up/down on both edges of A (C) and B (D) signal */ *action = COUNTER_SYNAPSE_ACTION_BOTH_EDGES; break; default: /* should never reach this path */ mutex_unlock(&priv->lock); return -EINVAL; } mutex_unlock(&priv->lock); return 0; } static const struct counter_ops rz_mtu3_cnt_ops = { .count_read = rz_mtu3_count_read, .count_write = rz_mtu3_count_write, .function_read = rz_mtu3_count_function_read, .function_write = rz_mtu3_count_function_write, .action_read = rz_mtu3_action_read, }; #define RZ_MTU3_PHASE_SIGNAL(_id, _name) { \ .id = (_id), \ .name = (_name), \ } static struct counter_signal rz_mtu3_signals[] = { RZ_MTU3_PHASE_SIGNAL(SIGNAL_A_ID, "MTU1 MTCLKA"), RZ_MTU3_PHASE_SIGNAL(SIGNAL_B_ID, "MTU1 MTCLKB"), RZ_MTU3_PHASE_SIGNAL(SIGNAL_C_ID, "MTU2 MTCLKC"), RZ_MTU3_PHASE_SIGNAL(SIGNAL_D_ID, "MTU2 MTCLKD"), }; static struct counter_synapse rz_mtu3_mtu1_count_synapses[] = { { .actions_list = rz_mtu3_synapse_actions, .num_actions = ARRAY_SIZE(rz_mtu3_synapse_actions), .signal = rz_mtu3_signals, }, { .actions_list = rz_mtu3_synapse_actions, .num_actions = ARRAY_SIZE(rz_mtu3_synapse_actions), .signal = rz_mtu3_signals + 1, } }; static struct counter_synapse rz_mtu3_mtu2_count_synapses[] = { { .actions_list = rz_mtu3_synapse_actions, .num_actions = ARRAY_SIZE(rz_mtu3_synapse_actions), .signal = rz_mtu3_signals, }, { .actions_list = rz_mtu3_synapse_actions, .num_actions = ARRAY_SIZE(rz_mtu3_synapse_actions), .signal = rz_mtu3_signals + 1, }, { .actions_list = rz_mtu3_synapse_actions, .num_actions = ARRAY_SIZE(rz_mtu3_synapse_actions), .signal = rz_mtu3_signals + 2, }, { .actions_list = rz_mtu3_synapse_actions, .num_actions = ARRAY_SIZE(rz_mtu3_synapse_actions), .signal = rz_mtu3_signals + 3, } }; static struct counter_count rz_mtu3_counts[] = { { .id = RZ_MTU3_16_BIT_MTU1_CH, .name = "Channel 1 Count", .functions_list = rz_mtu3_count_functions, .num_functions = ARRAY_SIZE(rz_mtu3_count_functions), .synapses = rz_mtu3_mtu1_count_synapses, .num_synapses = ARRAY_SIZE(rz_mtu3_mtu1_count_synapses), .ext = rz_mtu3_count_ext, .num_ext = ARRAY_SIZE(rz_mtu3_count_ext), }, { .id = RZ_MTU3_16_BIT_MTU2_CH, .name = "Channel 2 Count", .functions_list = rz_mtu3_count_functions, .num_functions = ARRAY_SIZE(rz_mtu3_count_functions), .synapses = rz_mtu3_mtu2_count_synapses, .num_synapses = ARRAY_SIZE(rz_mtu3_mtu2_count_synapses), .ext = rz_mtu3_count_ext, .num_ext = ARRAY_SIZE(rz_mtu3_count_ext), }, { .id = RZ_MTU3_32_BIT_CH, .name = "Channel 1 and 2 (cascaded) Count", .functions_list = rz_mtu3_count_functions, .num_functions = ARRAY_SIZE(rz_mtu3_count_functions), .synapses = rz_mtu3_mtu2_count_synapses, .num_synapses = ARRAY_SIZE(rz_mtu3_mtu2_count_synapses), .ext = rz_mtu3_count_ext, .num_ext = ARRAY_SIZE(rz_mtu3_count_ext), } }; static const char *const rz_mtu3_ext_input_phase_clock_select[] = { "MTCLKA-MTCLKB", "MTCLKC-MTCLKD", }; static DEFINE_COUNTER_ENUM(rz_mtu3_ext_input_phase_clock_select_enum, rz_mtu3_ext_input_phase_clock_select); static struct counter_comp rz_mtu3_device_ext[] = { COUNTER_COMP_DEVICE_BOOL("cascade_counts_enable", rz_mtu3_cascade_counts_enable_get, rz_mtu3_cascade_counts_enable_set), COUNTER_COMP_DEVICE_ENUM("external_input_phase_clock_select", rz_mtu3_ext_input_phase_clock_select_get, rz_mtu3_ext_input_phase_clock_select_set, rz_mtu3_ext_input_phase_clock_select_enum), }; static int rz_mtu3_cnt_pm_runtime_suspend(struct device *dev) { struct clk *const clk = dev_get_drvdata(dev); clk_disable_unprepare(clk); return 0; } static int rz_mtu3_cnt_pm_runtime_resume(struct device *dev) { struct clk *const clk = dev_get_drvdata(dev); clk_prepare_enable(clk); return 0; } static DEFINE_RUNTIME_DEV_PM_OPS(rz_mtu3_cnt_pm_ops, rz_mtu3_cnt_pm_runtime_suspend, rz_mtu3_cnt_pm_runtime_resume, NULL); static void rz_mtu3_cnt_pm_disable(void *data) { struct device *dev = data; pm_runtime_disable(dev); pm_runtime_set_suspended(dev); } static int rz_mtu3_cnt_probe(struct platform_device *pdev) { struct rz_mtu3 *ddata = dev_get_drvdata(pdev->dev.parent); struct device *dev = &pdev->dev; struct counter_device *counter; struct rz_mtu3_channel *ch; struct rz_mtu3_cnt *priv; unsigned int i; int ret; counter = devm_counter_alloc(dev, sizeof(*priv)); if (!counter) return -ENOMEM; priv = counter_priv(counter); priv->clk = ddata->clk; priv->mtu_32bit_max = U32_MAX; priv->ch = &ddata->channels[RZ_MTU3_CHAN_1]; ch = &priv->ch[0]; for (i = 0; i < RZ_MTU3_MAX_HW_CNTR_CHANNELS; i++) { ch->dev = dev; priv->mtu_16bit_max[i] = U16_MAX; ch++; } mutex_init(&priv->lock); platform_set_drvdata(pdev, priv->clk); clk_prepare_enable(priv->clk); pm_runtime_set_active(&pdev->dev); pm_runtime_enable(&pdev->dev); ret = devm_add_action_or_reset(&pdev->dev, rz_mtu3_cnt_pm_disable, dev); if (ret < 0) goto disable_clock; counter->name = dev_name(dev); counter->parent = dev; counter->ops = &rz_mtu3_cnt_ops; counter->counts = rz_mtu3_counts; counter->num_counts = ARRAY_SIZE(rz_mtu3_counts); counter->signals = rz_mtu3_signals; counter->num_signals = ARRAY_SIZE(rz_mtu3_signals); counter->ext = rz_mtu3_device_ext; counter->num_ext = ARRAY_SIZE(rz_mtu3_device_ext); /* Register Counter device */ ret = devm_counter_add(dev, counter); if (ret < 0) { dev_err_probe(dev, ret, "Failed to add counter\n"); goto disable_clock; } return 0; disable_clock: clk_disable_unprepare(priv->clk); return ret; } static struct platform_driver rz_mtu3_cnt_driver = { .probe = rz_mtu3_cnt_probe, .driver = { .name = "rz-mtu3-counter", .pm = pm_ptr(&rz_mtu3_cnt_pm_ops), }, }; module_platform_driver(rz_mtu3_cnt_driver); MODULE_AUTHOR("Biju Das <biju.das.jz@bp.renesas.com>"); MODULE_ALIAS("platform:rz-mtu3-counter"); MODULE_DESCRIPTION("Renesas RZ/G2L MTU3a counter driver"); MODULE_LICENSE("GPL"); MODULE_IMPORT_NS(COUNTER);