/* * Copyright (c) 2018, Cue Health Inc * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #include #include #define LOG_LEVEL CONFIG_PWM_LOG_LEVEL #include LOG_MODULE_REGISTER(pwm_nrfx); #define PWM_NRFX_CH_POLARITY_MASK BIT(15) #define PWM_NRFX_CH_PULSE_CYCLES_MASK BIT_MASK(15) #define PWM_NRFX_CH_VALUE(value, inverted) \ (value | (inverted ? 0 : PWM_NRFX_CH_POLARITY_MASK)) struct pwm_nrfx_config { nrfx_pwm_t pwm; nrfx_pwm_config_t initial_config; nrf_pwm_sequence_t seq; #ifdef CONFIG_PINCTRL const struct pinctrl_dev_config *pcfg; #endif }; struct pwm_nrfx_data { uint32_t period_cycles; uint16_t current[NRF_PWM_CHANNEL_COUNT]; uint16_t countertop; uint8_t prescaler; uint8_t initially_inverted; }; static int pwm_period_check_and_set(const struct pwm_nrfx_config *config, struct pwm_nrfx_data *data, uint32_t channel, uint32_t period_cycles) { uint8_t i; uint8_t prescaler; uint32_t countertop; /* If any other channel (other than the one being configured) is set up * with a non-zero pulse cycle, the period that is currently set cannot * be changed, as this would influence the output for this channel. */ for (i = 0; i < NRF_PWM_CHANNEL_COUNT; ++i) { if (i != channel) { uint16_t channel_pulse_cycle = data->current[i] & PWM_NRFX_CH_PULSE_CYCLES_MASK; if (channel_pulse_cycle > 0) { LOG_ERR("Incompatible period."); return -EINVAL; } } } /* Try to find a prescaler that will allow setting the requested period * after prescaling as the countertop value for the PWM peripheral. */ prescaler = 0; countertop = period_cycles; do { if (countertop <= PWM_COUNTERTOP_COUNTERTOP_Msk) { data->period_cycles = period_cycles; data->prescaler = prescaler; data->countertop = (uint16_t)countertop; nrf_pwm_configure(config->pwm.p_registers, data->prescaler, config->initial_config.count_mode, data->countertop); return 0; } countertop >>= 1; ++prescaler; } while (prescaler <= PWM_PRESCALER_PRESCALER_Msk); LOG_ERR("Prescaler for period_cycles %u not found.", period_cycles); return -EINVAL; } static bool pwm_channel_is_active(uint8_t channel, const struct pwm_nrfx_data *data) { uint16_t pulse_cycle = data->current[channel] & PWM_NRFX_CH_PULSE_CYCLES_MASK; return (pulse_cycle > 0 && pulse_cycle < data->countertop); } static bool any_other_channel_is_active(uint8_t channel, const struct pwm_nrfx_data *data) { uint8_t i; for (i = 0; i < NRF_PWM_CHANNEL_COUNT; ++i) { if (i != channel && pwm_channel_is_active(i, data)) { return true; } } return false; } static bool channel_psel_get(uint32_t channel, uint32_t *psel, const struct pwm_nrfx_config *config) { *psel = nrf_pwm_pin_get(config->pwm.p_registers, channel); return (((*psel & PWM_PSEL_OUT_CONNECT_Msk) >> PWM_PSEL_OUT_CONNECT_Pos) == PWM_PSEL_OUT_CONNECT_Connected); } static int pwm_nrfx_pin_set(const struct device *dev, uint32_t pwm, uint32_t period_cycles, uint32_t pulse_cycles, pwm_flags_t flags) { /* We assume here that period_cycles will always be 16MHz * peripheral clock. Since pwm_nrfx_get_cycles_per_sec() function might * be removed, see ISSUE #6958. * TODO: Remove this comment when issue has been resolved. */ const struct pwm_nrfx_config *config = dev->config; struct pwm_nrfx_data *data = dev->data; uint8_t channel = pwm; bool inverted = (flags & PWM_POLARITY_INVERTED); bool was_stopped; if (channel >= NRF_PWM_CHANNEL_COUNT) { LOG_ERR("Invalid channel: %u.", channel); return -EINVAL; } /* Check if nrfx_pwm_stop function was called in previous * pwm_nrfx_pin_set call. Relying only on state returned by * nrfx_pwm_is_stopped may cause race condition if the pwm_nrfx_pin_set * is called multiple times in quick succession. */ was_stopped = !pwm_channel_is_active(channel, data) && !any_other_channel_is_active(channel, data); /* If this PWM is in center-aligned mode, pulse and period lengths * are effectively doubled by the up-down count, so halve them here * to compensate. */ if (config->initial_config.count_mode == NRF_PWM_MODE_UP_AND_DOWN) { period_cycles /= 2; pulse_cycles /= 2; } /* Check if period_cycles is either matching currently used, or * possible to use with our prescaler options. * Don't do anything if the period length happens to be zero. * In such case, the channel is treated as inactive. */ if (period_cycles != 0 && period_cycles != data->period_cycles) { int ret = pwm_period_check_and_set(config, data, channel, period_cycles); if (ret) { return ret; } } data->current[channel] = PWM_NRFX_CH_VALUE(pulse_cycles >> data->prescaler, inverted); LOG_DBG("channel %u, pulse %u, period %u, prescaler: %u.", channel, pulse_cycles, period_cycles, data->prescaler); /* If this channel turns out to not need to be driven by the PWM * peripheral (it is off or fully on - duty 0% or 100%), set properly * the GPIO configuration for its output pin. This will provide * the correct output state for this channel when the PWM peripheral * is disabled after all channels appear to be inactive. */ if (!pwm_channel_is_active(channel, data)) { uint32_t psel; if (channel_psel_get(channel, &psel, config)) { /* If pulse 0% and pin not inverted, set LOW. * If pulse 100% and pin inverted, set LOW. * If pulse 0% and pin inverted, set HIGH. * If pulse 100% and pin not inverted, set HIGH. */ bool pulse_0_and_not_inverted = (pulse_cycles == 0U) && !inverted; bool pulse_100_and_inverted = (pulse_cycles == period_cycles) && inverted; uint32_t value = (pulse_0_and_not_inverted || pulse_100_and_inverted) ? 0 : 1; nrf_gpio_pin_write(psel, value); } if (!any_other_channel_is_active(channel, data)) { nrfx_pwm_stop(&config->pwm, false); } } else { /* Since we are playing the sequence in a loop, the * sequence only has to be started when its not already * playing. The new channel values will be used * immediately when they are written into the seq array. */ if (was_stopped) { /* Wait until PWM will be stopped and then start the * sequence. */ while (!nrfx_pwm_is_stopped(&config->pwm)) { } nrfx_pwm_simple_playback(&config->pwm, &config->seq, 1, NRFX_PWM_FLAG_LOOP); } } return 0; } static int pwm_nrfx_get_cycles_per_sec(const struct device *dev, uint32_t pwm, uint64_t *cycles) { /* TODO: Since this function might be removed, we will always return * 16MHz from this function and handle the conversion with prescaler, * etc, in the pin set function. See issue #6958. */ *cycles = 16ul * 1000ul * 1000ul; return 0; } static const struct pwm_driver_api pwm_nrfx_drv_api_funcs = { .pin_set = pwm_nrfx_pin_set, .get_cycles_per_sec = pwm_nrfx_get_cycles_per_sec, }; static int pwm_nrfx_init(const struct device *dev) { const struct pwm_nrfx_config *config = dev->config; struct pwm_nrfx_data *data = dev->data; #ifdef CONFIG_PINCTRL int ret = pinctrl_apply_state(config->pcfg, PINCTRL_STATE_DEFAULT); if (ret < 0) { return ret; } data->initially_inverted = 0; for (size_t i = 0; i < ARRAY_SIZE(data->current); i++) { uint32_t psel; if (channel_psel_get(i, &psel, config)) { /* Mark channels as inverted according to what initial * state of their outputs has been set by pinctrl (high * idle state means that the channel is inverted). */ data->initially_inverted |= nrf_gpio_pin_out_read(psel) ? BIT(i) : 0; } } #endif for (size_t i = 0; i < ARRAY_SIZE(data->current); i++) { bool inverted = data->initially_inverted & BIT(i); data->current[i] = PWM_NRFX_CH_VALUE(0, inverted); } nrfx_err_t result = nrfx_pwm_init(&config->pwm, &config->initial_config, NULL, NULL); if (result != NRFX_SUCCESS) { LOG_ERR("Failed to initialize device: %s", dev->name); return -EBUSY; } return 0; } #ifdef CONFIG_PM_DEVICE static void pwm_nrfx_uninit(const struct device *dev) { const struct pwm_nrfx_config *config = dev->config; nrfx_pwm_uninit(&config->pwm); memset(dev->data, 0, sizeof(struct pwm_nrfx_data)); } static int pwm_nrfx_pm_action(const struct device *dev, enum pm_device_action action) { #ifdef CONFIG_PINCTRL const struct pwm_nrfx_config *config = dev->config; #endif int ret = 0; switch (action) { case PM_DEVICE_ACTION_RESUME: #ifdef CONFIG_PINCTRL ret = pinctrl_apply_state(config->pcfg, PINCTRL_STATE_DEFAULT); if (ret < 0) { return ret; } #endif ret = pwm_nrfx_init(dev); break; case PM_DEVICE_ACTION_SUSPEND: pwm_nrfx_uninit(dev); #ifdef CONFIG_PINCTRL ret = pinctrl_apply_state(config->pcfg, PINCTRL_STATE_SLEEP); if (ret < 0) { return ret; } #endif break; default: return -ENOTSUP; } return ret; } #else #define pwm_nrfx_pm_action NULL #endif /* CONFIG_PM_DEVICE */ #define PWM(dev_idx) DT_NODELABEL(pwm##dev_idx) #define PWM_PROP(dev_idx, prop) DT_PROP(PWM(dev_idx), prop) #define PWM_CH_INVERTED(dev_idx, ch_idx) \ PWM_PROP(dev_idx, ch##ch_idx##_inverted) #define PWM_OUTPUT_PIN(dev_idx, ch_idx) \ COND_CODE_1(DT_NODE_HAS_PROP(PWM(dev_idx), ch##ch_idx##_pin), \ (PWM_PROP(dev_idx, ch##ch_idx##_pin) | \ (PWM_CH_INVERTED(dev_idx, ch_idx) \ ? NRFX_PWM_PIN_INVERTED : 0)), \ (NRFX_PWM_PIN_NOT_USED)) #define PWM_NRFX_DEVICE(idx) \ NRF_DT_CHECK_PIN_ASSIGNMENTS(PWM(idx), 1, \ ch0_pin, ch1_pin, ch2_pin, ch3_pin); \ static struct pwm_nrfx_data pwm_nrfx_##idx##_data = { \ COND_CODE_1(CONFIG_PINCTRL, (), \ (.initially_inverted = \ (PWM_CH_INVERTED(idx, 0) ? BIT(0) : 0) | \ (PWM_CH_INVERTED(idx, 1) ? BIT(1) : 0) | \ (PWM_CH_INVERTED(idx, 2) ? BIT(2) : 0) | \ (PWM_CH_INVERTED(idx, 3) ? BIT(3) : 0),)) \ }; \ IF_ENABLED(CONFIG_PINCTRL, (PINCTRL_DT_DEFINE(PWM(idx)))); \ static const struct pwm_nrfx_config pwm_nrfx_##idx##config = { \ .pwm = NRFX_PWM_INSTANCE(idx), \ .initial_config = { \ COND_CODE_1(CONFIG_PINCTRL, \ (.skip_gpio_cfg = true, \ .skip_psel_cfg = true,), \ (.output_pins = { \ PWM_OUTPUT_PIN(idx, 0), \ PWM_OUTPUT_PIN(idx, 1), \ PWM_OUTPUT_PIN(idx, 2), \ PWM_OUTPUT_PIN(idx, 3), \ },)) \ .base_clock = NRF_PWM_CLK_1MHz, \ .count_mode = (PWM_PROP(idx, center_aligned) \ ? NRF_PWM_MODE_UP_AND_DOWN \ : NRF_PWM_MODE_UP), \ .top_value = 1000, \ .load_mode = NRF_PWM_LOAD_INDIVIDUAL, \ .step_mode = NRF_PWM_STEP_TRIGGERED, \ }, \ .seq.values.p_raw = pwm_nrfx_##idx##_data.current, \ .seq.length = NRF_PWM_CHANNEL_COUNT, \ IF_ENABLED(CONFIG_PINCTRL, \ (.pcfg = PINCTRL_DT_DEV_CONFIG_GET(PWM(idx)),)) \ }; \ PM_DEVICE_DT_DEFINE(PWM(idx), pwm_nrfx_pm_action); \ DEVICE_DT_DEFINE(PWM(idx), \ pwm_nrfx_init, PM_DEVICE_DT_GET(PWM(idx)), \ &pwm_nrfx_##idx##_data, \ &pwm_nrfx_##idx##config, \ POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE, \ &pwm_nrfx_drv_api_funcs) #if DT_NODE_HAS_STATUS(DT_NODELABEL(pwm0), okay) PWM_NRFX_DEVICE(0); #endif #if DT_NODE_HAS_STATUS(DT_NODELABEL(pwm1), okay) PWM_NRFX_DEVICE(1); #endif #if DT_NODE_HAS_STATUS(DT_NODELABEL(pwm2), okay) PWM_NRFX_DEVICE(2); #endif #if DT_NODE_HAS_STATUS(DT_NODELABEL(pwm3), okay) PWM_NRFX_DEVICE(3); #endif