diff options
Diffstat (limited to 'drivers/pwm/pwm-tiehrpwm.c')
| -rw-r--r-- | drivers/pwm/pwm-tiehrpwm.c | 154 | 
1 files changed, 61 insertions, 93 deletions
| diff --git a/drivers/pwm/pwm-tiehrpwm.c b/drivers/pwm/pwm-tiehrpwm.c index 0125e73b98df..7a86cb090f76 100644 --- a/drivers/pwm/pwm-tiehrpwm.c +++ b/drivers/pwm/pwm-tiehrpwm.c @@ -36,7 +36,7 @@  #define CLKDIV_MAX		7  #define HSPCLKDIV_MAX		7 -#define PERIOD_MAX		0xFFFF +#define PERIOD_MAX		0x10000  /* compare module registers */  #define CMPA			0x12 @@ -65,14 +65,10 @@  #define AQCTL_ZRO_FRCHIGH	BIT(1)  #define AQCTL_ZRO_FRCTOGGLE	(BIT(1) | BIT(0)) -#define AQCTL_CHANA_POLNORMAL	(AQCTL_CAU_FRCLOW | AQCTL_PRD_FRCHIGH | \ -				AQCTL_ZRO_FRCHIGH) -#define AQCTL_CHANA_POLINVERSED	(AQCTL_CAU_FRCHIGH | AQCTL_PRD_FRCLOW | \ -				AQCTL_ZRO_FRCLOW) -#define AQCTL_CHANB_POLNORMAL	(AQCTL_CBU_FRCLOW | AQCTL_PRD_FRCHIGH | \ -				AQCTL_ZRO_FRCHIGH) -#define AQCTL_CHANB_POLINVERSED	(AQCTL_CBU_FRCHIGH | AQCTL_PRD_FRCLOW | \ -				AQCTL_ZRO_FRCLOW) +#define AQCTL_CHANA_POLNORMAL	(AQCTL_CAU_FRCLOW | AQCTL_ZRO_FRCHIGH) +#define AQCTL_CHANA_POLINVERSED	(AQCTL_CAU_FRCHIGH | AQCTL_ZRO_FRCLOW) +#define AQCTL_CHANB_POLNORMAL	(AQCTL_CBU_FRCLOW | AQCTL_ZRO_FRCHIGH) +#define AQCTL_CHANB_POLINVERSED	(AQCTL_CBU_FRCHIGH | AQCTL_ZRO_FRCLOW)  #define AQSFRC_RLDCSF_MASK	(BIT(7) | BIT(6))  #define AQSFRC_RLDCSF_ZRO	0 @@ -108,7 +104,6 @@ struct ehrpwm_pwm_chip {  	unsigned long clk_rate;  	void __iomem *mmio_base;  	unsigned long period_cycles[NUM_PWM_CHANNEL]; -	enum pwm_polarity polarity[NUM_PWM_CHANNEL];  	struct clk *tbclk;  	struct ehrpwm_context ctx;  }; @@ -166,7 +161,7 @@ static int set_prescale_div(unsigned long rqst_prescaler, u16 *prescale_div,  			*prescale_div = (1 << clkdiv) *  					(hspclkdiv ? (hspclkdiv * 2) : 1); -			if (*prescale_div > rqst_prescaler) { +			if (*prescale_div >= rqst_prescaler) {  				*tb_clk_div = (clkdiv << TBCTL_CLKDIV_SHIFT) |  					(hspclkdiv << TBCTL_HSPCLKDIV_SHIFT);  				return 0; @@ -177,51 +172,20 @@ static int set_prescale_div(unsigned long rqst_prescaler, u16 *prescale_div,  	return 1;  } -static void configure_polarity(struct ehrpwm_pwm_chip *pc, int chan) -{ -	u16 aqctl_val, aqctl_mask; -	unsigned int aqctl_reg; - -	/* -	 * Configure PWM output to HIGH/LOW level on counter -	 * reaches compare register value and LOW/HIGH level -	 * on counter value reaches period register value and -	 * zero value on counter -	 */ -	if (chan == 1) { -		aqctl_reg = AQCTLB; -		aqctl_mask = AQCTL_CBU_MASK; - -		if (pc->polarity[chan] == PWM_POLARITY_INVERSED) -			aqctl_val = AQCTL_CHANB_POLINVERSED; -		else -			aqctl_val = AQCTL_CHANB_POLNORMAL; -	} else { -		aqctl_reg = AQCTLA; -		aqctl_mask = AQCTL_CAU_MASK; - -		if (pc->polarity[chan] == PWM_POLARITY_INVERSED) -			aqctl_val = AQCTL_CHANA_POLINVERSED; -		else -			aqctl_val = AQCTL_CHANA_POLNORMAL; -	} - -	aqctl_mask |= AQCTL_PRD_MASK | AQCTL_ZRO_MASK; -	ehrpwm_modify(pc->mmio_base, aqctl_reg, aqctl_mask, aqctl_val); -} -  /*   * period_ns = 10^9 * (ps_divval * period_cycles) / PWM_CLK_RATE   * duty_ns   = 10^9 * (ps_divval * duty_cycles) / PWM_CLK_RATE   */  static int ehrpwm_pwm_config(struct pwm_chip *chip, struct pwm_device *pwm, -			     u64 duty_ns, u64 period_ns) +			     u64 duty_ns, u64 period_ns, enum pwm_polarity polarity)  {  	struct ehrpwm_pwm_chip *pc = to_ehrpwm_pwm_chip(chip);  	u32 period_cycles, duty_cycles;  	u16 ps_divval, tb_divval;  	unsigned int i, cmp_reg;  	unsigned long long c; +	u16 aqctl_val, aqctl_mask; +	unsigned int aqctl_reg;  	if (period_ns > NSEC_PER_SEC)  		return -ERANGE; @@ -231,15 +195,10 @@ static int ehrpwm_pwm_config(struct pwm_chip *chip, struct pwm_device *pwm,  	do_div(c, NSEC_PER_SEC);  	period_cycles = (unsigned long)c; -	if (period_cycles < 1) { -		period_cycles = 1; -		duty_cycles = 1; -	} else { -		c = pc->clk_rate; -		c = c * duty_ns; -		do_div(c, NSEC_PER_SEC); -		duty_cycles = (unsigned long)c; -	} +	c = pc->clk_rate; +	c = c * duty_ns; +	do_div(c, NSEC_PER_SEC); +	duty_cycles = (unsigned long)c;  	/*  	 * Period values should be same for multiple PWM channels as IP uses @@ -265,52 +224,73 @@ static int ehrpwm_pwm_config(struct pwm_chip *chip, struct pwm_device *pwm,  	pc->period_cycles[pwm->hwpwm] = period_cycles;  	/* Configure clock prescaler to support Low frequency PWM wave */ -	if (set_prescale_div(period_cycles/PERIOD_MAX, &ps_divval, +	if (set_prescale_div(DIV_ROUND_UP(period_cycles, PERIOD_MAX), &ps_divval,  			     &tb_divval)) {  		dev_err(pwmchip_parent(chip), "Unsupported values\n");  		return -EINVAL;  	} -	pm_runtime_get_sync(pwmchip_parent(chip)); - -	/* Update clock prescaler values */ -	ehrpwm_modify(pc->mmio_base, TBCTL, TBCTL_CLKDIV_MASK, tb_divval); -  	/* Update period & duty cycle with presacler division */  	period_cycles = period_cycles / ps_divval;  	duty_cycles = duty_cycles / ps_divval; -	/* Configure shadow loading on Period register */ -	ehrpwm_modify(pc->mmio_base, TBCTL, TBCTL_PRDLD_MASK, TBCTL_PRDLD_SHDW); +	if (period_cycles < 1) +		period_cycles = 1; -	ehrpwm_write(pc->mmio_base, TBPRD, period_cycles); +	pm_runtime_get_sync(pwmchip_parent(chip)); -	/* Configure ehrpwm counter for up-count mode */ -	ehrpwm_modify(pc->mmio_base, TBCTL, TBCTL_CTRMODE_MASK, -		      TBCTL_CTRMODE_UP); +	/* Update clock prescaler values */ +	ehrpwm_modify(pc->mmio_base, TBCTL, TBCTL_CLKDIV_MASK, tb_divval); -	if (pwm->hwpwm == 1) +	if (pwm->hwpwm == 1) {  		/* Channel 1 configured with compare B register */  		cmp_reg = CMPB; -	else + +		aqctl_reg = AQCTLB; +		aqctl_mask = AQCTL_CBU_MASK; + +		if (polarity == PWM_POLARITY_INVERSED) +			aqctl_val = AQCTL_CHANB_POLINVERSED; +		else +			aqctl_val = AQCTL_CHANB_POLNORMAL; + +		/* if duty_cycle is big, don't toggle on CBU */ +		if (duty_cycles > period_cycles) +			aqctl_val &= ~AQCTL_CBU_MASK; + +	} else {  		/* Channel 0 configured with compare A register */  		cmp_reg = CMPA; -	ehrpwm_write(pc->mmio_base, cmp_reg, duty_cycles); +		aqctl_reg = AQCTLA; +		aqctl_mask = AQCTL_CAU_MASK; -	pm_runtime_put_sync(pwmchip_parent(chip)); +		if (polarity == PWM_POLARITY_INVERSED) +			aqctl_val = AQCTL_CHANA_POLINVERSED; +		else +			aqctl_val = AQCTL_CHANA_POLNORMAL; -	return 0; -} +		/* if duty_cycle is big, don't toggle on CAU */ +		if (duty_cycles > period_cycles) +			aqctl_val &= ~AQCTL_CAU_MASK; +	} -static int ehrpwm_pwm_set_polarity(struct pwm_chip *chip, -				   struct pwm_device *pwm, -				   enum pwm_polarity polarity) -{ -	struct ehrpwm_pwm_chip *pc = to_ehrpwm_pwm_chip(chip); +	aqctl_mask |= AQCTL_PRD_MASK | AQCTL_ZRO_MASK; +	ehrpwm_modify(pc->mmio_base, aqctl_reg, aqctl_mask, aqctl_val); + +	/* Configure shadow loading on Period register */ +	ehrpwm_modify(pc->mmio_base, TBCTL, TBCTL_PRDLD_MASK, TBCTL_PRDLD_SHDW); + +	ehrpwm_write(pc->mmio_base, TBPRD, period_cycles - 1); -	/* Configuration of polarity in hardware delayed, do at enable */ -	pc->polarity[pwm->hwpwm] = polarity; +	/* Configure ehrpwm counter for up-count mode */ +	ehrpwm_modify(pc->mmio_base, TBCTL, TBCTL_CTRMODE_MASK, +		      TBCTL_CTRMODE_UP); + +	if (!(duty_cycles > period_cycles)) +		ehrpwm_write(pc->mmio_base, cmp_reg, duty_cycles); + +	pm_runtime_put_sync(pwmchip_parent(chip));  	return 0;  } @@ -339,9 +319,6 @@ static int ehrpwm_pwm_enable(struct pwm_chip *chip, struct pwm_device *pwm)  	ehrpwm_modify(pc->mmio_base, AQCSFRC, aqcsfrc_mask, aqcsfrc_val); -	/* Channels polarity can be configured from action qualifier module */ -	configure_polarity(pc, pwm->hwpwm); -  	/* Enable TBCLK */  	ret = clk_enable(pc->tbclk);  	if (ret) { @@ -391,12 +368,7 @@ static void ehrpwm_pwm_free(struct pwm_chip *chip, struct pwm_device *pwm)  {  	struct ehrpwm_pwm_chip *pc = to_ehrpwm_pwm_chip(chip); -	if (pwm_is_enabled(pwm)) { -		dev_warn(pwmchip_parent(chip), "Removing PWM device without disabling\n"); -		pm_runtime_put_sync(pwmchip_parent(chip)); -	} - -	/* set period value to zero on free */ +	/* Don't let a pwm without consumer block requests to the other channel */  	pc->period_cycles[pwm->hwpwm] = 0;  } @@ -411,10 +383,6 @@ static int ehrpwm_pwm_apply(struct pwm_chip *chip, struct pwm_device *pwm,  			ehrpwm_pwm_disable(chip, pwm);  			enabled = false;  		} - -		err = ehrpwm_pwm_set_polarity(chip, pwm, state->polarity); -		if (err) -			return err;  	}  	if (!state->enabled) { @@ -423,7 +391,7 @@ static int ehrpwm_pwm_apply(struct pwm_chip *chip, struct pwm_device *pwm,  		return 0;  	} -	err = ehrpwm_pwm_config(chip, pwm, state->duty_cycle, state->period); +	err = ehrpwm_pwm_config(chip, pwm, state->duty_cycle, state->period, state->polarity);  	if (err)  		return err; | 
