[PATCH v2] pwm: rk_pwm: Make PWM driver to support all Rockchip Socs

Kever Yang kever.yang at rock-chips.com
Thu Dec 5 16:22:53 CET 2019


On 2019/12/3 下午5:49, David Wu wrote:
> This PWM driver can be used to support pwm functions
> for on all Rockchip Socs.
>
> The previous chips than RK3288 did not support polarity,
> and register layout was different from the RK3288 PWM.
>
> The RK3288 keep the current functions.
>
> RK3308 and the chips after it, which can support hardware lock,
> configure duty, period and polarity at next same period, to
> prevent the intermediate temporary state.
>
> Signed-off-by: David Wu <david.wu at rock-chips.com>

Reviewed-by: Kever Yang <kever.yang at rock-chips.com>

Thanks,
- Kever
> ---
> Change in v2: None
>   - Remove RK3399 compatible
>
>   arch/arm/include/asm/arch-rockchip/pwm.h |  17 ++-
>   drivers/pwm/rk_pwm.c                     | 138 +++++++++++++++++++----
>   2 files changed, 130 insertions(+), 25 deletions(-)
>
> diff --git a/arch/arm/include/asm/arch-rockchip/pwm.h b/arch/arm/include/asm/arch-rockchip/pwm.h
> index b5178db394..e8594055cd 100644
> --- a/arch/arm/include/asm/arch-rockchip/pwm.h
> +++ b/arch/arm/include/asm/arch-rockchip/pwm.h
> @@ -7,13 +7,15 @@
>   #ifndef _ASM_ARCH_PWM_H
>   #define _ASM_ARCH_PWM_H
>   
> -struct rk3288_pwm {
> -	u32 cnt;
> -	u32 period_hpr;
> -	u32 duty_lpr;
> -	u32 ctrl;
> +struct rockchip_pwm_regs {
> +	unsigned long duty;
> +	unsigned long period;
> +	unsigned long cntr;
> +	unsigned long ctrl;
>   };
> -check_member(rk3288_pwm, ctrl, 0xc);
> +
> +#define PWM_CTRL_TIMER_EN		(1 << 0)
> +#define PWM_CTRL_OUTPUT_EN		(1 << 3)
>   
>   #define RK_PWM_DISABLE                  (0 << 0)
>   #define RK_PWM_ENABLE                   (1 << 0)
> @@ -33,6 +35,9 @@ check_member(rk3288_pwm, ctrl, 0xc);
>   #define PWM_OUTPUT_LEFT                 (0 << 5)
>   #define PWM_OUTPUT_CENTER               (1 << 5)
>   
> +#define PWM_LOCK			(1 << 6)
> +#define PWM_UNLOCK			(0 << 6)
> +
>   #define PWM_LP_ENABLE                   (1 << 8)
>   #define PWM_LP_DISABLE                  (0 << 8)
>   
> diff --git a/drivers/pwm/rk_pwm.c b/drivers/pwm/rk_pwm.c
> index 88db294cf1..46888e9077 100644
> --- a/drivers/pwm/rk_pwm.c
> +++ b/drivers/pwm/rk_pwm.c
> @@ -15,22 +15,38 @@
>   #include <asm/arch-rockchip/pwm.h>
>   #include <power/regulator.h>
>   
> +DECLARE_GLOBAL_DATA_PTR;
> +
> +struct rockchip_pwm_data {
> +	struct rockchip_pwm_regs regs;
> +	unsigned int prescaler;
> +	bool supports_polarity;
> +	bool supports_lock;
> +	u32 enable_conf;
> +	u32 enable_conf_mask;
> +};
> +
>   struct rk_pwm_priv {
> -	struct rk3288_pwm *regs;
> +	fdt_addr_t base;
>   	ulong freq;
> -	uint enable_conf;
> +	u32 conf_polarity;
> +	const struct rockchip_pwm_data *data;
>   };
>   
>   static int rk_pwm_set_invert(struct udevice *dev, uint channel, bool polarity)
>   {
>   	struct rk_pwm_priv *priv = dev_get_priv(dev);
>   
> +	if (!priv->data->supports_polarity) {
> +		debug("%s: Do not support polarity\n", __func__);
> +		return 0;
> +	}
> +
>   	debug("%s: polarity=%u\n", __func__, polarity);
> -	priv->enable_conf &= ~(PWM_DUTY_MASK | PWM_INACTIVE_MASK);
>   	if (polarity)
> -		priv->enable_conf |= PWM_DUTY_NEGATIVE | PWM_INACTIVE_POSTIVE;
> +		priv->conf_polarity = PWM_DUTY_NEGATIVE | PWM_INACTIVE_POSTIVE;
>   	else
> -		priv->enable_conf |= PWM_DUTY_POSTIVE | PWM_INACTIVE_NEGATIVE;
> +		priv->conf_polarity = PWM_DUTY_POSTIVE | PWM_INACTIVE_NEGATIVE;
>   
>   	return 0;
>   }
> @@ -39,20 +55,44 @@ static int rk_pwm_set_config(struct udevice *dev, uint channel, uint period_ns,
>   			     uint duty_ns)
>   {
>   	struct rk_pwm_priv *priv = dev_get_priv(dev);
> -	struct rk3288_pwm *regs = priv->regs;
> +	const struct rockchip_pwm_regs *regs = &priv->data->regs;
>   	unsigned long period, duty;
> +	u32 ctrl;
>   
>   	debug("%s: period_ns=%u, duty_ns=%u\n", __func__, period_ns, duty_ns);
> -	writel(PWM_SEL_SRC_CLK | PWM_OUTPUT_LEFT | PWM_LP_DISABLE |
> -		PWM_CONTINUOUS | priv->enable_conf |
> -		RK_PWM_DISABLE,
> -		&regs->ctrl);
>   
> -	period = lldiv((uint64_t)(priv->freq / 1000) * period_ns, 1000000);
> -	duty = lldiv((uint64_t)(priv->freq / 1000) * duty_ns, 1000000);
> +	ctrl = readl(priv->base + regs->ctrl);
> +	/*
> +	 * Lock the period and duty of previous configuration, then
> +	 * change the duty and period, that would not be effective.
> +	 */
> +	if (priv->data->supports_lock) {
> +		ctrl |= PWM_LOCK;
> +		writel(ctrl, priv->base + regs->ctrl);
> +	}
> +
> +	period = lldiv((uint64_t)priv->freq * period_ns,
> +		       priv->data->prescaler * 1000000000);
> +	duty = lldiv((uint64_t)priv->freq * duty_ns,
> +		     priv->data->prescaler * 1000000000);
> +
> +	writel(period, priv->base + regs->period);
> +	writel(duty, priv->base + regs->duty);
> +
> +	if (priv->data->supports_polarity) {
> +		ctrl &= ~(PWM_DUTY_MASK | PWM_INACTIVE_MASK);
> +		ctrl |= priv->conf_polarity;
> +	}
> +
> +	/*
> +	 * Unlock and set polarity at the same time,
> +	 * the configuration of duty, period and polarity
> +	 * would be effective together at next period.
> +	 */
> +	if (priv->data->supports_lock)
> +		ctrl &= ~PWM_LOCK;
> +	writel(ctrl, priv->base + regs->ctrl);
>   
> -	writel(period, &regs->period_hpr);
> -	writel(duty, &regs->duty_lpr);
>   	debug("%s: period=%lu, duty=%lu\n", __func__, period, duty);
>   
>   	return 0;
> @@ -61,10 +101,20 @@ static int rk_pwm_set_config(struct udevice *dev, uint channel, uint period_ns,
>   static int rk_pwm_set_enable(struct udevice *dev, uint channel, bool enable)
>   {
>   	struct rk_pwm_priv *priv = dev_get_priv(dev);
> -	struct rk3288_pwm *regs = priv->regs;
> +	const struct rockchip_pwm_regs *regs = &priv->data->regs;
> +	u32 ctrl;
>   
>   	debug("%s: Enable '%s'\n", __func__, dev->name);
> -	clrsetbits_le32(&regs->ctrl, RK_PWM_ENABLE, enable ? RK_PWM_ENABLE : 0);
> +
> +	ctrl = readl(priv->base + regs->ctrl);
> +	ctrl &= ~priv->data->enable_conf_mask;
> +
> +	if (enable)
> +		ctrl |= priv->data->enable_conf;
> +	else
> +		ctrl &= ~priv->data->enable_conf;
> +
> +	writel(ctrl, priv->base + regs->ctrl);
>   
>   	return 0;
>   }
> @@ -73,7 +123,7 @@ static int rk_pwm_ofdata_to_platdata(struct udevice *dev)
>   {
>   	struct rk_pwm_priv *priv = dev_get_priv(dev);
>   
> -	priv->regs = (struct rk3288_pwm *)dev_read_addr(dev);
> +	priv->base = dev_read_addr(dev);
>   
>   	return 0;
>   }
> @@ -89,8 +139,12 @@ static int rk_pwm_probe(struct udevice *dev)
>   		debug("%s get clock fail!\n", __func__);
>   		return -EINVAL;
>   	}
> +
>   	priv->freq = clk_get_rate(&clk);
> -	priv->enable_conf = PWM_DUTY_POSTIVE | PWM_INACTIVE_POSTIVE;
> +	priv->data = (struct rockchip_pwm_data *)dev_get_driver_data(dev);
> +
> +	if (priv->data->supports_polarity)
> +		priv->conf_polarity = PWM_DUTY_POSTIVE | PWM_INACTIVE_POSTIVE;
>   
>   	return 0;
>   }
> @@ -101,8 +155,54 @@ static const struct pwm_ops rk_pwm_ops = {
>   	.set_enable	= rk_pwm_set_enable,
>   };
>   
> +static const struct rockchip_pwm_data pwm_data_v1 = {
> +	.regs = {
> +		.duty = 0x04,
> +		.period = 0x08,
> +		.cntr = 0x00,
> +		.ctrl = 0x0c,
> +	},
> +	.prescaler = 2,
> +	.supports_polarity = false,
> +	.supports_lock = false,
> +	.enable_conf = PWM_CTRL_OUTPUT_EN | PWM_CTRL_TIMER_EN,
> +	.enable_conf_mask = BIT(1) | BIT(3),
> +};
> +
> +static const struct rockchip_pwm_data pwm_data_v2 = {
> +	.regs = {
> +		.duty = 0x08,
> +		.period = 0x04,
> +		.cntr = 0x00,
> +		.ctrl = 0x0c,
> +	},
> +	.prescaler = 1,
> +	.supports_polarity = true,
> +	.supports_lock = false,
> +	.enable_conf = PWM_OUTPUT_LEFT | PWM_LP_DISABLE | RK_PWM_ENABLE |
> +		       PWM_CONTINUOUS,
> +	.enable_conf_mask = GENMASK(2, 0) | BIT(5) | BIT(8),
> +};
> +
> +static const struct rockchip_pwm_data pwm_data_v3 = {
> +	.regs = {
> +		.duty = 0x08,
> +		.period = 0x04,
> +		.cntr = 0x00,
> +		.ctrl = 0x0c,
> +	},
> +	.prescaler = 1,
> +	.supports_polarity = true,
> +	.supports_lock = true,
> +	.enable_conf = PWM_OUTPUT_LEFT | PWM_LP_DISABLE | RK_PWM_ENABLE |
> +		       PWM_CONTINUOUS,
> +	.enable_conf_mask = GENMASK(2, 0) | BIT(5) | BIT(8),
> +};
> +
>   static const struct udevice_id rk_pwm_ids[] = {
> -	{ .compatible = "rockchip,rk3288-pwm" },
> +	{ .compatible = "rockchip,rk2928-pwm", .data = (ulong)&pwm_data_v1},
> +	{ .compatible = "rockchip,rk3288-pwm", .data = (ulong)&pwm_data_v2},
> +	{ .compatible = "rockchip,rk3328-pwm", .data = (ulong)&pwm_data_v3},
>   	{ }
>   };
>   




More information about the U-Boot mailing list