|  | /* | 
|  | * Copyright (c) 2024-2025 Renesas Electronics Corporation | 
|  | * | 
|  | * SPDX-License-Identifier: Apache-2.0 | 
|  | */ | 
|  |  | 
|  | #include <zephyr/kernel.h> | 
|  | #include <zephyr/device.h> | 
|  | #include <zephyr/irq.h> | 
|  | #include <zephyr/drivers/pwm.h> | 
|  | #include <zephyr/drivers/pinctrl.h> | 
|  | #include <zephyr/dt-bindings/pwm/renesas_rz_pwm.h> | 
|  | #include "r_gpt.h" | 
|  | #include "r_gpt_cfg.h" | 
|  | #include <zephyr/logging/log.h> | 
|  |  | 
|  | LOG_MODULE_REGISTER(pwm_renesas_rz_gpt, CONFIG_PWM_LOG_LEVEL); | 
|  |  | 
|  | #define DT_DRV_COMPAT renesas_rz_gpt_pwm | 
|  |  | 
|  | #define GPT_PRV_GTIO_HIGH_COMPARE_MATCH_LOW_CYCLE_END 0x6U | 
|  | #define GPT_PRV_GTIO_LOW_COMPARE_MATCH_HIGH_CYCLE_END 0x9U | 
|  | #define GPT_PRV_GTIOR_INITIAL_LEVEL_BIT               4 | 
|  |  | 
|  | #define CAPTURE_BOTH_MODE_FIRST_EVENT_IS_CAPTURE_PULSE   1 | 
|  | #define CAPTURE_BOTH_MODE_SECOND_EVENT_IS_CAPTURE_PERIOD 2 | 
|  |  | 
|  | #define RZ_GPT_GTIOR_OAE_Msk    (0x100UL) | 
|  | #define RZ_GPT_GTIOR_OADFLT_Pos (6UL) | 
|  | #define RZ_GPT_GTIOR_GTIOA_Pos  (0UL) | 
|  | #define RZ_GPT_GTIOR_GTIOB_Pos  (16UL) | 
|  | #define RZ_GPT_GTIOR_NFAEN_Pos  (13UL) | 
|  | #define RZ_GPT_GTIOR_NFBEN_Pos  (29UL) | 
|  |  | 
|  | struct pwm_rz_gpt_capture_data { | 
|  | pwm_capture_callback_handler_t callback; | 
|  | void *user_data; | 
|  | uint64_t period; | 
|  | uint64_t pulse; | 
|  | uint16_t capture_type_flag; | 
|  | uint32_t capture_both_event_count; | 
|  | bool is_busy; | 
|  | uint32_t overflows; | 
|  | bool continuous; | 
|  | uint32_t capture_channel; | 
|  | }; | 
|  |  | 
|  | struct pwm_rz_gpt_data { | 
|  | timer_cfg_t *fsp_cfg; | 
|  | gpt_instance_ctrl_t *fsp_ctrl; | 
|  | #ifdef CONFIG_PWM_CAPTURE | 
|  | struct pwm_rz_gpt_capture_data capture; | 
|  | #endif /* CONFIG_PWM_CAPTURE */ | 
|  | }; | 
|  |  | 
|  | struct pwm_rz_gpt_config { | 
|  | const struct pinctrl_dev_config *pincfg; | 
|  | const timer_api_t *fsp_api; | 
|  | }; | 
|  |  | 
|  | static uint32_t pwm_rz_gpt_gtior_calculate(gpt_pin_level_t const stop_level) | 
|  | { | 
|  | /* The stop level is used as both the initial level and the stop level */ | 
|  | uint32_t gtior = RZ_GPT_GTIOR_OAE_Msk | ((uint32_t)stop_level << RZ_GPT_GTIOR_OADFLT_Pos) | | 
|  | ((uint32_t)stop_level << GPT_PRV_GTIOR_INITIAL_LEVEL_BIT); | 
|  |  | 
|  | uint32_t gtion = GPT_PRV_GTIO_LOW_COMPARE_MATCH_HIGH_CYCLE_END; | 
|  |  | 
|  | /* Calculate the gtior value for PWM mode only */ | 
|  | gtior |= gtion; | 
|  |  | 
|  | return gtior; | 
|  | } | 
|  |  | 
|  | static int pwm_rz_gpt_apply_gtior_config(gpt_instance_ctrl_t *const p_ctrl, | 
|  | timer_cfg_t const *const p_cfg) | 
|  | { | 
|  | gpt_extended_cfg_t *p_extend = (gpt_extended_cfg_t *)p_cfg->p_extend; | 
|  | uint32_t gtior = 0; | 
|  |  | 
|  | #if GPT_CFG_OUTPUT_SUPPORT_ENABLE | 
|  | /* Calculate GTIOR */ | 
|  | if (p_extend->gtioca.output_enabled) { | 
|  | uint32_t gtioca_gtior = pwm_rz_gpt_gtior_calculate(p_extend->gtioca.stop_level); | 
|  |  | 
|  | gtior |= gtioca_gtior << RZ_GPT_GTIOR_GTIOA_Pos; | 
|  | } | 
|  |  | 
|  | if (p_extend->gtiocb.output_enabled) { | 
|  | uint32_t gtiocb_gtior = pwm_rz_gpt_gtior_calculate(p_extend->gtiocb.stop_level); | 
|  |  | 
|  | gtior |= gtiocb_gtior << RZ_GPT_GTIOR_GTIOB_Pos; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /* Configure the noise filter for the GTIOC pins */ | 
|  | gtior |= (uint32_t)(p_extend->capture_filter_gtioca << RZ_GPT_GTIOR_NFAEN_Pos); | 
|  | gtior |= (uint32_t)(p_extend->capture_filter_gtiocb << RZ_GPT_GTIOR_NFBEN_Pos); | 
|  |  | 
|  | /* Set the I/O control register */ | 
|  | p_ctrl->p_reg->GTIOR = gtior; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int pwm_rz_gpt_set_cycles(const struct device *dev, uint32_t channel, uint32_t period_cycles, | 
|  | uint32_t pulse_cycles, pwm_flags_t flags) | 
|  | { | 
|  | const struct pwm_rz_gpt_config *cfg = dev->config; | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | gpt_extended_cfg_t *fsp_cfg_extend = (gpt_extended_cfg_t *)data->fsp_cfg->p_extend; | 
|  | uint32_t pulse; | 
|  | fsp_err_t err; | 
|  | uint32_t pin; | 
|  |  | 
|  | /* GTIOCA and GTIOCB setting */ | 
|  | if (channel == RZ_PWM_GPT_IO_A) { | 
|  | pin = GPT_IO_PIN_GTIOCA; | 
|  | fsp_cfg_extend->gtioca.output_enabled = true; | 
|  | } else if (channel == RZ_PWM_GPT_IO_B) { | 
|  | pin = GPT_IO_PIN_GTIOCB; | 
|  | fsp_cfg_extend->gtiocb.output_enabled = true; | 
|  | } else { | 
|  | LOG_ERR("Valid only for RZ_PWM_GPT_IO_A and RZ_PWM_GPT_IO_B pins"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if ((data->fsp_ctrl->variant == TIMER_VARIANT_16_BIT && period_cycles > UINT16_MAX) || | 
|  | (data->fsp_ctrl->variant == TIMER_VARIANT_32_BIT && period_cycles > UINT32_MAX)) { | 
|  | LOG_ERR("Out of range period cycles are not valid"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | pulse = (flags & PWM_POLARITY_INVERTED) ? period_cycles - pulse_cycles : pulse_cycles; | 
|  |  | 
|  | /* Apply gtio output setting */ | 
|  | pwm_rz_gpt_apply_gtior_config(data->fsp_ctrl, data->fsp_cfg); | 
|  |  | 
|  | /* Stop timer */ | 
|  | err = cfg->fsp_api->stop(data->fsp_ctrl); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Update period cycles, reflected at an overflow */ | 
|  | err = cfg->fsp_api->periodSet(data->fsp_ctrl, period_cycles); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Update pulse cycles, reflected at an overflow */ | 
|  | err = cfg->fsp_api->dutyCycleSet(data->fsp_ctrl, pulse, pin); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Start timer */ | 
|  | err = cfg->fsp_api->start(data->fsp_ctrl); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | }; | 
|  |  | 
|  | static int pwm_rz_gpt_get_cycles_per_sec(const struct device *dev, uint32_t channel, | 
|  | uint64_t *cycles) | 
|  | { | 
|  | const struct pwm_rz_gpt_config *cfg = dev->config; | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | timer_info_t info; | 
|  | fsp_err_t err; | 
|  |  | 
|  | if (!(channel == RZ_PWM_GPT_IO_A || channel == RZ_PWM_GPT_IO_B)) { | 
|  | LOG_ERR("Valid only for RZ_PWM_GPT_IO_A and RZ_PWM_GPT_IO_B pins"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = cfg->fsp_api->infoGet(data->fsp_ctrl, &info); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  | *cycles = (uint64_t)info.clock_frequency; | 
|  |  | 
|  | return 0; | 
|  | }; | 
|  |  | 
|  | #ifdef CONFIG_PWM_CAPTURE | 
|  | static int pwm_rz_gpt_configure_capture(const struct device *dev, uint32_t channel, | 
|  | pwm_flags_t flags, pwm_capture_callback_handler_t cb, | 
|  | void *user_data) | 
|  | { | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | gpt_extended_cfg_t *fsp_cfg_extend = (gpt_extended_cfg_t *)data->fsp_cfg->p_extend; | 
|  |  | 
|  | if (!(flags & PWM_CAPTURE_TYPE_MASK)) { | 
|  | LOG_ERR("No PWWM capture type specified"); | 
|  | return -EINVAL; | 
|  | } | 
|  | data->capture.capture_type_flag = flags & PWM_CAPTURE_TYPE_MASK; | 
|  | data->capture.capture_channel = channel; | 
|  | if (data->capture.is_busy) { | 
|  | LOG_ERR("Capture already active on this pin"); | 
|  | return -EBUSY; | 
|  | } | 
|  | if (channel == RZ_PWM_GPT_IO_A) { | 
|  | if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_BOTH) { | 
|  | data->capture.capture_both_event_count = 0; | 
|  | if (flags & PWM_POLARITY_INVERTED) { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } else { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } | 
|  | fsp_cfg_extend->capture_a_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } else if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_PERIOD) { | 
|  | if (flags & PWM_POLARITY_INVERTED) { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_a_source = fsp_cfg_extend->start_source; | 
|  | } else { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_a_source = fsp_cfg_extend->start_source; | 
|  | } | 
|  | } else { | 
|  | if (flags & PWM_POLARITY_INVERTED) { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_a_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } else { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_RISING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_a_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_LOW | | 
|  | GPT_SOURCE_GTIOCA_FALLING_WHILE_GTIOCB_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } | 
|  | } | 
|  | } else if (channel == RZ_PWM_GPT_IO_B) { | 
|  | if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_BOTH) { | 
|  | data->capture.capture_both_event_count = 0; | 
|  | if (flags & PWM_POLARITY_INVERTED) { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } else { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } | 
|  | fsp_cfg_extend->capture_b_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } else if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_PERIOD) { | 
|  | if (flags & PWM_POLARITY_INVERTED) { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_b_source = fsp_cfg_extend->start_source; | 
|  | } else { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_b_source = fsp_cfg_extend->start_source; | 
|  | } | 
|  | } else { | 
|  | if (flags & PWM_POLARITY_INVERTED) { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_b_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } else { | 
|  | fsp_cfg_extend->start_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_RISING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->capture_b_source = | 
|  | (gpt_source_t)(GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_LOW | | 
|  | GPT_SOURCE_GTIOCB_FALLING_WHILE_GTIOCA_HIGH | | 
|  | GPT_SOURCE_NONE); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | data->capture.callback = cb; | 
|  | data->capture.user_data = user_data; | 
|  | data->capture.continuous = flags & PWM_CAPTURE_MODE_CONTINUOUS; | 
|  | if (data->capture.continuous) { | 
|  | if (channel == RZ_PWM_GPT_IO_A) { | 
|  | fsp_cfg_extend->stop_source = fsp_cfg_extend->capture_a_source; | 
|  | } else if (channel == RZ_PWM_GPT_IO_B) { | 
|  | fsp_cfg_extend->stop_source = fsp_cfg_extend->capture_b_source; | 
|  | } | 
|  | fsp_cfg_extend->clear_source = fsp_cfg_extend->start_source; | 
|  | } | 
|  |  | 
|  | else { | 
|  | fsp_cfg_extend->stop_source = (gpt_source_t)(GPT_SOURCE_NONE); | 
|  | fsp_cfg_extend->clear_source = (gpt_source_t)(GPT_SOURCE_NONE); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int pwm_rz_gpt_enable_capture(const struct device *dev, uint32_t channel) | 
|  | { | 
|  | const struct pwm_rz_gpt_config *cfg = dev->config; | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | gpt_extended_cfg_t *fsp_cfg_extend = (gpt_extended_cfg_t *)data->fsp_cfg->p_extend; | 
|  | fsp_err_t err; | 
|  |  | 
|  | data->capture.capture_channel = channel; | 
|  |  | 
|  | if (data->capture.is_busy) { | 
|  | LOG_ERR("Capture already active on this pin"); | 
|  | return -EBUSY; | 
|  | } | 
|  |  | 
|  | if (!data->capture.callback) { | 
|  | LOG_ERR("PWM capture not configured"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | data->capture.is_busy = true; | 
|  |  | 
|  | /* Enable capture source */ | 
|  | err = cfg->fsp_api->enable(data->fsp_ctrl); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Enable interruption */ | 
|  | irq_enable(data->fsp_cfg->cycle_end_irq); | 
|  | if (channel == RZ_PWM_GPT_IO_A) { | 
|  | irq_enable(fsp_cfg_extend->capture_a_irq); | 
|  | } else if (channel == RZ_PWM_GPT_IO_B) { | 
|  | irq_enable(fsp_cfg_extend->capture_b_irq); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int pwm_rz_gpt_disable_capture(const struct device *dev, uint32_t channel) | 
|  | { | 
|  | const struct pwm_rz_gpt_config *cfg = dev->config; | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | fsp_err_t err; | 
|  | gpt_extended_cfg_t *fsp_cfg_extend = (gpt_extended_cfg_t *)data->fsp_cfg->p_extend; | 
|  |  | 
|  | data->capture.capture_channel = channel; | 
|  | data->capture.is_busy = false; | 
|  |  | 
|  | /* Disable interruption */ | 
|  | irq_disable(data->fsp_cfg->cycle_end_irq); | 
|  | if (channel == RZ_PWM_GPT_IO_A) { | 
|  | irq_disable(fsp_cfg_extend->capture_a_irq); | 
|  | } else if (channel == RZ_PWM_GPT_IO_B) { | 
|  | irq_disable(fsp_cfg_extend->capture_b_irq); | 
|  | } | 
|  |  | 
|  | /* Disable capture source */ | 
|  | err = cfg->fsp_api->disable(data->fsp_ctrl); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Stop timer */ | 
|  | err = cfg->fsp_api->stop(data->fsp_ctrl); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Clear timer */ | 
|  | err = cfg->fsp_api->reset(data->fsp_ctrl); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void fsp_callback(timer_callback_args_t *p_args) | 
|  | { | 
|  | const struct device *dev = p_args->p_context; | 
|  | const struct pwm_rz_gpt_config *cfg = dev->config; | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | timer_info_t info; | 
|  |  | 
|  | (void)cfg->fsp_api->infoGet(data->fsp_ctrl, &info); | 
|  |  | 
|  | uint64_t period = info.period_counts; | 
|  |  | 
|  | /* The maximum period is one more than the maximum 16,32-bit number, but will be reflected | 
|  | * as 0 | 
|  | */ | 
|  | if (period == 0U) { | 
|  | if (data->fsp_ctrl->variant == TIMER_VARIANT_16_BIT) { | 
|  | period = UINT16_MAX + 1U; | 
|  | } else { | 
|  | period = UINT32_MAX + 1U; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Capture event */ | 
|  | if (p_args->event == TIMER_EVENT_CAPTURE_A) { | 
|  | if (p_args->capture != 0U) { | 
|  | bool check_disable_capture = false; | 
|  |  | 
|  | if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_BOTH) { | 
|  | data->capture.capture_both_event_count++; | 
|  | if (data->capture.capture_both_event_count == | 
|  | CAPTURE_BOTH_MODE_FIRST_EVENT_IS_CAPTURE_PULSE) { | 
|  | data->capture.pulse = (data->capture.overflows * period) + | 
|  | p_args->capture; | 
|  | } | 
|  | if (data->capture.capture_both_event_count == | 
|  | CAPTURE_BOTH_MODE_SECOND_EVENT_IS_CAPTURE_PERIOD) { | 
|  | data->capture.capture_both_event_count = 0; | 
|  | data->capture.period = (data->capture.overflows * period) + | 
|  | p_args->capture; | 
|  | data->capture.callback( | 
|  | dev, GPT_IO_PIN_GTIOCA, data->capture.period, | 
|  | data->capture.pulse, 0, data->capture.user_data); | 
|  |  | 
|  | check_disable_capture = true; | 
|  | } | 
|  | } else if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_PULSE) { | 
|  | data->capture.pulse = | 
|  | (data->capture.overflows * period) + p_args->capture; | 
|  | data->capture.callback(dev, GPT_IO_PIN_GTIOCA, 0, | 
|  | data->capture.pulse, 0, | 
|  | data->capture.user_data); | 
|  |  | 
|  | check_disable_capture = true; | 
|  | } else { | 
|  | data->capture.period = | 
|  | (data->capture.overflows * period) + p_args->capture; | 
|  | data->capture.callback(dev, GPT_IO_PIN_GTIOCA, data->capture.period, | 
|  | 0, 0, data->capture.user_data); | 
|  |  | 
|  | check_disable_capture = true; | 
|  | } | 
|  | if (check_disable_capture) { | 
|  | data->capture.overflows = 0U; | 
|  | /* Disable capture in single mode */ | 
|  | if (data->capture.continuous == false) { | 
|  | pwm_rz_gpt_disable_capture(dev, GPT_IO_PIN_GTIOCA); | 
|  | } | 
|  | } | 
|  | } | 
|  | } else if (p_args->event == TIMER_EVENT_CAPTURE_B) { | 
|  | if (p_args->capture != 0U) { | 
|  | bool check_disable_capture = false; | 
|  |  | 
|  | if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_BOTH) { | 
|  | data->capture.capture_both_event_count++; | 
|  | if (data->capture.capture_both_event_count == | 
|  | CAPTURE_BOTH_MODE_FIRST_EVENT_IS_CAPTURE_PULSE) { | 
|  | data->capture.pulse = (data->capture.overflows * period) + | 
|  | p_args->capture; | 
|  | } | 
|  | if (data->capture.capture_both_event_count == | 
|  | CAPTURE_BOTH_MODE_SECOND_EVENT_IS_CAPTURE_PERIOD) { | 
|  | data->capture.capture_both_event_count = 0; | 
|  | data->capture.period = (data->capture.overflows * period) + | 
|  | p_args->capture; | 
|  | data->capture.callback( | 
|  | dev, GPT_IO_PIN_GTIOCB, data->capture.period, | 
|  | data->capture.pulse, 0, data->capture.user_data); | 
|  |  | 
|  | check_disable_capture = true; | 
|  | } | 
|  | } else if (data->capture.capture_type_flag == PWM_CAPTURE_TYPE_PULSE) { | 
|  | data->capture.pulse = | 
|  | (data->capture.overflows * period) + p_args->capture; | 
|  | data->capture.callback(dev, GPT_IO_PIN_GTIOCB, 0, | 
|  | data->capture.pulse, 0, | 
|  | data->capture.user_data); | 
|  |  | 
|  | check_disable_capture = true; | 
|  | } else { | 
|  | data->capture.period = | 
|  | (data->capture.overflows * period) + p_args->capture; | 
|  | data->capture.callback(dev, GPT_IO_PIN_GTIOCB, data->capture.period, | 
|  | 0, 0, data->capture.user_data); | 
|  |  | 
|  | check_disable_capture = true; | 
|  | } | 
|  | if (check_disable_capture) { | 
|  | data->capture.overflows = 0U; | 
|  | /* Disable capture in single mode */ | 
|  | if (data->capture.continuous == false) { | 
|  | pwm_rz_gpt_disable_capture(dev, GPT_IO_PIN_GTIOCB); | 
|  | } | 
|  | } | 
|  | } | 
|  | } else if (p_args->event == TIMER_EVENT_CYCLE_END) { | 
|  | data->capture.overflows++; | 
|  | } else { | 
|  | if (data->capture.capture_channel == RZ_PWM_GPT_IO_A) { | 
|  | data->capture.callback(dev, GPT_IO_PIN_GTIOCA, 0, 0, -ECANCELED, | 
|  | data->capture.user_data); | 
|  | } else if (data->capture.capture_channel == RZ_PWM_GPT_IO_B) { | 
|  | data->capture.callback(dev, GPT_IO_PIN_GTIOCB, 0, 0, -ECANCELED, | 
|  | data->capture.user_data); | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif /* CONFIG_PWM_CAPTURE */ | 
|  |  | 
|  | static DEVICE_API(pwm, pwm_rz_gpt_driver_api) = { | 
|  | .get_cycles_per_sec = pwm_rz_gpt_get_cycles_per_sec, | 
|  | .set_cycles = pwm_rz_gpt_set_cycles, | 
|  | #ifdef CONFIG_PWM_CAPTURE | 
|  | .configure_capture = pwm_rz_gpt_configure_capture, | 
|  | .enable_capture = pwm_rz_gpt_enable_capture, | 
|  | .disable_capture = pwm_rz_gpt_disable_capture, | 
|  | #endif /* CONFIG_PWM_CAPTURE */ | 
|  | }; | 
|  |  | 
|  | static int pwm_rz_gpt_init(const struct device *dev) | 
|  | { | 
|  | const struct pwm_rz_gpt_config *cfg = dev->config; | 
|  | struct pwm_rz_gpt_data *data = dev->data; | 
|  | gpt_extended_cfg_t *fsp_cfg_extend = (gpt_extended_cfg_t *)data->fsp_cfg->p_extend; | 
|  | int err; | 
|  |  | 
|  | err = pinctrl_apply_state(cfg->pincfg, PINCTRL_STATE_DEFAULT); | 
|  | if (err) { | 
|  | LOG_ERR("Failed to configure pins for PWM (%d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_PWM_CAPTURE | 
|  | data->fsp_cfg->p_callback = fsp_callback; | 
|  | data->fsp_cfg->p_context = dev; | 
|  | #endif /* CONFIG_PWM_CAPTURE */ | 
|  |  | 
|  | err = cfg->fsp_api->open(data->fsp_ctrl, data->fsp_cfg); | 
|  | if (err != FSP_SUCCESS) { | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | irq_disable(data->fsp_cfg->cycle_end_irq); | 
|  | irq_disable(fsp_cfg_extend->capture_a_irq); | 
|  | irq_disable(fsp_cfg_extend->capture_b_irq); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #define GPT(idx) DT_INST_PARENT(idx) | 
|  |  | 
|  | #ifdef CONFIG_PWM_CAPTURE | 
|  | extern void gpt_capture_compare_a_isr(void); | 
|  | extern void gpt_capture_compare_b_isr(void); | 
|  | extern void gpt_counter_overflow_isr(void); | 
|  |  | 
|  | static void pwm_rz_gpt_ccmpa_isr(const struct device *dev) | 
|  | { | 
|  | ARG_UNUSED(dev); | 
|  | gpt_capture_compare_a_isr(); | 
|  | } | 
|  |  | 
|  | static void pwm_rz_gpt_ccmpb_isr(const struct device *dev) | 
|  | { | 
|  | ARG_UNUSED(dev); | 
|  | gpt_capture_compare_b_isr(); | 
|  | } | 
|  |  | 
|  | static void pwm_rz_gpt_ovf_isr(const struct device *dev) | 
|  | { | 
|  | ARG_UNUSED(dev); | 
|  | gpt_counter_overflow_isr(); | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_CPU_CORTEX_M | 
|  | #define GPT_GET_IRQ_FLAGS(idx, irq_name) 0 | 
|  | #else /* Cortex-A/R */ | 
|  | #define GPT_GET_IRQ_FLAGS(idx, irq_name) DT_IRQ_BY_NAME(GPT(idx), irq_name, flags) | 
|  | #endif | 
|  |  | 
|  | #define PWM_RZ_IRQ_CONFIG_INIT(inst)                                                               \ | 
|  | do {                                                                                       \ | 
|  | IRQ_CONNECT(DT_IRQ_BY_NAME(GPT(inst), ccmpa, irq),                                 \ | 
|  | DT_IRQ_BY_NAME(GPT(inst), ccmpa, priority), pwm_rz_gpt_ccmpa_isr,      \ | 
|  | DEVICE_DT_INST_GET(inst), GPT_GET_IRQ_FLAGS(inst, ccmpa));             \ | 
|  | IRQ_CONNECT(DT_IRQ_BY_NAME(GPT(inst), ccmpb, irq),                                 \ | 
|  | DT_IRQ_BY_NAME(GPT(inst), ccmpb, priority), pwm_rz_gpt_ccmpb_isr,      \ | 
|  | DEVICE_DT_INST_GET(inst), GPT_GET_IRQ_FLAGS(inst, ccmpb));             \ | 
|  | IRQ_CONNECT(DT_IRQ_BY_NAME(GPT(inst), ovf, irq),                                   \ | 
|  | DT_IRQ_BY_NAME(GPT(inst), ovf, priority), pwm_rz_gpt_ovf_isr,          \ | 
|  | DEVICE_DT_INST_GET(inst), GPT_GET_IRQ_FLAGS(inst, ovf));               \ | 
|  | } while (0) | 
|  | #endif /* CONFIG_PWM_CAPTURE */ | 
|  |  | 
|  | #define PWM_RZ_INIT(inst)                                                                          \ | 
|  | PINCTRL_DT_INST_DEFINE(inst);                                                              \ | 
|  | static gpt_instance_ctrl_t g_timer##inst##_ctrl;                                           \ | 
|  | static gpt_extended_cfg_t g_timer##inst##_extend = {                                       \ | 
|  | .gtioca =                                                                          \ | 
|  | {                                                                          \ | 
|  | .output_enabled = false,                                           \ | 
|  | .stop_level = GPT_PIN_LEVEL_LOW,                                   \ | 
|  | },                                                                         \ | 
|  | .gtiocb =                                                                          \ | 
|  | {                                                                          \ | 
|  | .output_enabled = false,                                           \ | 
|  | .stop_level = GPT_PIN_LEVEL_LOW,                                   \ | 
|  | },                                                                         \ | 
|  | .start_source = (gpt_source_t)(GPT_SOURCE_NONE),                                   \ | 
|  | .stop_source = (gpt_source_t)(GPT_SOURCE_NONE),                                    \ | 
|  | .clear_source = (gpt_source_t)(GPT_SOURCE_NONE),                                   \ | 
|  | .count_up_source = (gpt_source_t)(GPT_SOURCE_NONE),                                \ | 
|  | .count_down_source = (gpt_source_t)(GPT_SOURCE_NONE),                              \ | 
|  | .capture_a_source = (gpt_source_t)(GPT_SOURCE_NONE),                               \ | 
|  | .capture_b_source = (gpt_source_t)(GPT_SOURCE_NONE),                               \ | 
|  | .capture_a_ipl = DT_IRQ_BY_NAME(GPT(inst), ccmpa, priority),                       \ | 
|  | .capture_b_ipl = DT_IRQ_BY_NAME(GPT(inst), ccmpb, priority),                       \ | 
|  | .capture_a_irq = DT_IRQ_BY_NAME(GPT(inst), ccmpa, irq),                            \ | 
|  | .capture_b_irq = DT_IRQ_BY_NAME(GPT(inst), ccmpb, irq),                            \ | 
|  | .capture_filter_gtioca = GPT_CAPTURE_FILTER_NONE,                                  \ | 
|  | .capture_filter_gtiocb = GPT_CAPTURE_FILTER_NONE,                                  \ | 
|  | .p_pwm_cfg = NULL,                                                                 \ | 
|  | };                                                                                         \ | 
|  | static timer_cfg_t g_timer##inst##_cfg = {                                                 \ | 
|  | .mode = TIMER_MODE_PWM,                                                            \ | 
|  | .channel = DT_PROP(GPT(inst), channel),                                            \ | 
|  | .source_div = DT_ENUM_IDX(GPT(inst), prescaler),                                   \ | 
|  | .cycle_end_ipl = DT_IRQ_BY_NAME(GPT(inst), ovf, priority),                         \ | 
|  | .cycle_end_irq = DT_IRQ_BY_NAME(GPT(inst), ovf, irq),                              \ | 
|  | .p_extend = &g_timer##inst##_extend,                                               \ | 
|  | };                                                                                         \ | 
|  | static const struct pwm_rz_gpt_config pwm_rz_gpt_config_##inst = {                         \ | 
|  | .pincfg = PINCTRL_DT_INST_DEV_CONFIG_GET(inst),                                    \ | 
|  | .fsp_api = &g_timer_on_gpt,                                                        \ | 
|  | };                                                                                         \ | 
|  | static struct pwm_rz_gpt_data pwm_rz_gpt_data_##inst = {                                   \ | 
|  | .fsp_cfg = &g_timer##inst##_cfg, .fsp_ctrl = &g_timer##inst##_ctrl};               \ | 
|  | static int pwm_rz_gpt_init_##inst(const struct device *dev)                                \ | 
|  | {                                                                                          \ | 
|  | IF_ENABLED(CONFIG_PWM_CAPTURE,                                 \ | 
|  | (PWM_RZ_IRQ_CONFIG_INIT(inst);))                   \ | 
|  | int err = pwm_rz_gpt_init(dev);                                                    \ | 
|  | if (err != 0) {                                                                    \ | 
|  | return err;                                                                \ | 
|  | }                                                                                  \ | 
|  | return 0;                                                                          \ | 
|  | }                                                                                          \ | 
|  | DEVICE_DT_INST_DEFINE(inst, pwm_rz_gpt_init_##inst, NULL, &pwm_rz_gpt_data_##inst,         \ | 
|  | &pwm_rz_gpt_config_##inst, POST_KERNEL, CONFIG_PWM_INIT_PRIORITY,    \ | 
|  | &pwm_rz_gpt_driver_api); | 
|  |  | 
|  | DT_INST_FOREACH_STATUS_OKAY(PWM_RZ_INIT); |