| // 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); |