416 lines
12 KiB
C
416 lines
12 KiB
C
/*
|
|
* Copyright (c) 2018, Cue Health Inc
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
#include <nrfx_pwm.h>
|
|
#include <zephyr/drivers/pwm.h>
|
|
#include <zephyr/pm/device.h>
|
|
#include <zephyr/drivers/pinctrl.h>
|
|
#include <soc.h>
|
|
#include <hal/nrf_gpio.h>
|
|
#include <stdbool.h>
|
|
|
|
#define LOG_LEVEL CONFIG_PWM_LOG_LEVEL
|
|
#include <zephyr/logging/log.h>
|
|
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_set_cycles(const struct device *dev, uint32_t channel,
|
|
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;
|
|
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_set_cycles call. Relying only on state returned by
|
|
* nrfx_pwm_is_stopped may cause race condition if the
|
|
* pwm_nrfx_set_cycles 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 channel,
|
|
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 = {
|
|
.set_cycles = pwm_nrfx_set_cycles,
|
|
.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
|