/* * Copyright (c) 2022, Joep Buruma * * SPDX-License-Identifier: Apache-2.0 */ #define DT_DRV_COMPAT raspberrypi_pico_pwm #include #include #include #include #include #include LOG_MODULE_REGISTER(pwm_rpi_pico, CONFIG_PWM_LOG_LEVEL); /* pico-sdk includes */ #include #include #define PWM_RPI_PICO_COUNTER_TOP_MAX UINT16_MAX #define PWM_RPI_NUM_CHANNELS (16U) struct pwm_rpi_slice_config { uint8_t integral; uint8_t frac; bool phase_correct; }; struct pwm_rpi_config { /* * pwm_controller is the start address of the pwm peripheral. */ pwm_hw_t *pwm_controller; struct pwm_rpi_slice_config slice_configs[NUM_PWM_SLICES]; const struct pinctrl_dev_config *pcfg; const struct reset_dt_spec reset; const struct device *clk_dev; const clock_control_subsys_t clk_id; }; static inline uint32_t pwm_rpi_channel_to_slice(uint32_t channel) { return channel / 2; } static inline uint32_t pwm_rpi_channel_to_pico_channel(uint32_t channel) { return channel % 2; } static int pwm_rpi_get_cycles_per_sec(const struct device *dev, uint32_t ch, uint64_t *cycles) { const struct pwm_rpi_config *cfg = dev->config; int slice = pwm_rpi_channel_to_slice(ch); uint32_t pclk; int ret; if (ch >= PWM_RPI_NUM_CHANNELS) { return -EINVAL; } const struct pwm_rpi_slice_config *slice_config = &cfg->slice_configs[slice]; ret = clock_control_get_rate(cfg->clk_dev, cfg->clk_id, &pclk); if (ret < 0 || pclk == 0) { return -EINVAL; } if (slice_config->integral == 0) { *cycles = pclk; } else { /* No need to check for divide by 0 since the minimum value of * pwm_rpi_get_clkdiv is 1 */ *cycles = (uint64_t)pclk * 16 / ((uint64_t)slice_config->integral * 16 + slice_config->frac); } return 0; } /* The pico_sdk only allows setting the polarity of both channels at once. * This is a convenience function to make setting the polarity of a single * channel easier. */ static void pwm_rpi_set_channel_polarity(const struct device *dev, int slice, int pico_channel, bool inverted) { const struct pwm_rpi_config *cfg = dev->config; bool pwm_polarity_a = (cfg->pwm_controller->slice[slice].csr & PWM_CH0_CSR_A_INV_BITS) > 0; bool pwm_polarity_b = (cfg->pwm_controller->slice[slice].csr & PWM_CH0_CSR_B_INV_BITS) > 0; if (pico_channel == PWM_CHAN_A) { pwm_polarity_a = inverted; } else if (pico_channel == PWM_CHAN_B) { pwm_polarity_b = inverted; } pwm_set_output_polarity(slice, pwm_polarity_a, pwm_polarity_b); } static int pwm_rpi_set_cycles(const struct device *dev, uint32_t ch, uint32_t period_cycles, uint32_t pulse_cycles, pwm_flags_t flags) { const struct pwm_rpi_config *cfg = dev->config; int slice = pwm_rpi_channel_to_slice(ch); /* this is the channel within a pwm slice */ int pico_channel = pwm_rpi_channel_to_pico_channel(ch); int div_int; int div_frac; if (ch >= PWM_RPI_NUM_CHANNELS) { return -EINVAL; } div_int = cfg->slice_configs[slice].integral; div_frac = cfg->slice_configs[slice].frac; if (div_int == 0) { div_int = 1; div_frac = 0; while ((period_cycles / div_int - 1) > PWM_RPI_PICO_COUNTER_TOP_MAX) { div_int *= 2; } if (div_int > (UINT8_MAX + 1)) { return -EINVAL; } period_cycles /= div_int; pulse_cycles /= div_int; } if (period_cycles - 1 > PWM_RPI_PICO_COUNTER_TOP_MAX || pulse_cycles > PWM_RPI_PICO_COUNTER_TOP_MAX) { return -EINVAL; } pwm_rpi_set_channel_polarity(dev, slice, pico_channel, (flags & PWM_POLARITY_MASK) == PWM_POLARITY_INVERTED); pwm_set_wrap(slice, period_cycles - 1); pwm_set_chan_level(slice, pico_channel, pulse_cycles); pwm_set_clkdiv_int_frac(slice, div_int, div_frac); return 0; }; static DEVICE_API(pwm, pwm_rpi_driver_api) = { .get_cycles_per_sec = pwm_rpi_get_cycles_per_sec, .set_cycles = pwm_rpi_set_cycles, }; static int pwm_rpi_init(const struct device *dev) { const struct pwm_rpi_config *cfg = dev->config; pwm_config slice_cfg; size_t slice_idx; int err; err = pinctrl_apply_state(cfg->pcfg, PINCTRL_STATE_DEFAULT); if (err) { LOG_ERR("Failed to configure pins for PWM. err=%d", err); return err; } err = clock_control_on(cfg->clk_dev, cfg->clk_id); if (err < 0) { return err; } err = reset_line_toggle_dt(&cfg->reset); if (err < 0) { return err; } for (slice_idx = 0; slice_idx < NUM_PWM_SLICES; slice_idx++) { slice_cfg = pwm_get_default_config(); pwm_config_set_clkdiv_mode(&slice_cfg, PWM_DIV_FREE_RUNNING); pwm_init(slice_idx, &slice_cfg, false); if (cfg->slice_configs[slice_idx].integral == 0) { pwm_set_clkdiv_int_frac(slice_idx, 1, 0); } else { pwm_set_clkdiv_int_frac(slice_idx, cfg->slice_configs[slice_idx].integral, cfg->slice_configs[slice_idx].frac); } pwm_set_enabled(slice_idx, true); } return 0; } #define PWM_INST_RPI_SLICE_DIVIDER(idx, n) \ { \ .integral = DT_INST_PROP_OR(idx, UTIL_CAT(divider_int_, n), 0), \ .frac = DT_INST_PROP_OR(idx, UTIL_CAT(divider_frac_, n), 0), \ } #define PWM_RPI_INIT(idx) \ \ PINCTRL_DT_INST_DEFINE(idx); \ static const struct pwm_rpi_config pwm_rpi_config_##idx = { \ .pwm_controller = (pwm_hw_t *)DT_INST_REG_ADDR(idx), \ .pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(idx), \ .slice_configs = { \ PWM_INST_RPI_SLICE_DIVIDER(idx, 0), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 1), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 2), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 3), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 4), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 5), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 6), \ PWM_INST_RPI_SLICE_DIVIDER(idx, 7), \ }, \ .reset = RESET_DT_SPEC_INST_GET(idx), \ .clk_dev = DEVICE_DT_GET(DT_INST_CLOCKS_CTLR(idx)), \ .clk_id = (clock_control_subsys_t)DT_INST_PHA_BY_IDX(idx, clocks, 0, clk_id), \ }; \ \ DEVICE_DT_INST_DEFINE(idx, pwm_rpi_init, NULL, NULL, &pwm_rpi_config_##idx, POST_KERNEL, \ CONFIG_PWM_INIT_PRIORITY, &pwm_rpi_driver_api); DT_INST_FOREACH_STATUS_OKAY(PWM_RPI_INIT);