mirror of
				https://xff.cz/git/u-boot/
				synced 2025-10-30 18:05:48 +01:00 
			
		
		
		
	Something was wrong in the merge process into the mainline. Some added patches access driver structure fields and functions that have been modified by previous patches. The patch renames: - dev_get_platdata to dev_get_plat - dev_get_uclass_platdata to dev_get_uclass_plat - ofdata_to_platdata to of_to_plat - plat_data_alloc_size to plat_auto - priv_auto_alloc_size to priv_auto - video_uc_platdata to video_uc_plat Signed-off-by: Dario Binacchi <dariobin@libero.it>
		
			
				
	
	
		
			469 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			469 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * EHRPWM PWM driver
 | |
|  *
 | |
|  * Copyright (C) 2020 Dario Binacchi <dariobin@libero.it>
 | |
|  *
 | |
|  * Based on Linux kernel drivers/pwm/pwm-tiehrpwm.c
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <clk.h>
 | |
| #include <div64.h>
 | |
| #include <dm.h>
 | |
| #include <dm/device_compat.h>
 | |
| #include <pwm.h>
 | |
| #include <asm/io.h>
 | |
| 
 | |
| #define NSEC_PER_SEC			        1000000000L
 | |
| 
 | |
| /* Time base module registers */
 | |
| #define TI_EHRPWM_TBCTL				0x00
 | |
| #define TI_EHRPWM_TBPRD				0x0A
 | |
| 
 | |
| #define TI_EHRPWM_TBCTL_PRDLD_MASK		BIT(3)
 | |
| #define TI_EHRPWM_TBCTL_PRDLD_SHDW		0
 | |
| #define TI_EHRPWM_TBCTL_PRDLD_IMDT		BIT(3)
 | |
| #define TI_EHRPWM_TBCTL_CLKDIV_MASK		GENMASK(12, 7)
 | |
| #define TI_EHRPWM_TBCTL_CTRMODE_MASK		GENMASK(1, 0)
 | |
| #define TI_EHRPWM_TBCTL_CTRMODE_UP		0
 | |
| #define TI_EHRPWM_TBCTL_CTRMODE_DOWN		BIT(0)
 | |
| #define TI_EHRPWM_TBCTL_CTRMODE_UPDOWN		BIT(1)
 | |
| #define TI_EHRPWM_TBCTL_CTRMODE_FREEZE		GENMASK(1, 0)
 | |
| 
 | |
| #define TI_EHRPWM_TBCTL_HSPCLKDIV_SHIFT		7
 | |
| #define TI_EHRPWM_TBCTL_CLKDIV_SHIFT		10
 | |
| 
 | |
| #define TI_EHRPWM_CLKDIV_MAX			7
 | |
| #define TI_EHRPWM_HSPCLKDIV_MAX			7
 | |
| #define TI_EHRPWM_PERIOD_MAX			0xFFFF
 | |
| 
 | |
| /* Counter compare module registers */
 | |
| #define TI_EHRPWM_CMPA				0x12
 | |
| #define TI_EHRPWM_CMPB				0x14
 | |
| 
 | |
| /* Action qualifier module registers */
 | |
| #define TI_EHRPWM_AQCTLA			0x16
 | |
| #define TI_EHRPWM_AQCTLB			0x18
 | |
| #define TI_EHRPWM_AQSFRC			0x1A
 | |
| #define TI_EHRPWM_AQCSFRC			0x1C
 | |
| 
 | |
| #define TI_EHRPWM_AQCTL_CBU_MASK		GENMASK(9, 8)
 | |
| #define TI_EHRPWM_AQCTL_CBU_FRCLOW		BIT(8)
 | |
| #define TI_EHRPWM_AQCTL_CBU_FRCHIGH		BIT(9)
 | |
| #define TI_EHRPWM_AQCTL_CBU_FRCTOGGLE		GENMASK(9, 8)
 | |
| #define TI_EHRPWM_AQCTL_CAU_MASK		GENMASK(5, 4)
 | |
| #define TI_EHRPWM_AQCTL_CAU_FRCLOW		BIT(4)
 | |
| #define TI_EHRPWM_AQCTL_CAU_FRCHIGH		BIT(5)
 | |
| #define TI_EHRPWM_AQCTL_CAU_FRCTOGGLE		GENMASK(5, 4)
 | |
| #define TI_EHRPWM_AQCTL_PRD_MASK		GENMASK(3, 2)
 | |
| #define TI_EHRPWM_AQCTL_PRD_FRCLOW		BIT(2)
 | |
| #define TI_EHRPWM_AQCTL_PRD_FRCHIGH		BIT(3)
 | |
| #define TI_EHRPWM_AQCTL_PRD_FRCTOGGLE		GENMASK(3, 2)
 | |
| #define TI_EHRPWM_AQCTL_ZRO_MASK		GENMASK(1, 0)
 | |
| #define TI_EHRPWM_AQCTL_ZRO_FRCLOW		BIT(0)
 | |
| #define TI_EHRPWM_AQCTL_ZRO_FRCHIGH		BIT(1)
 | |
| #define TI_EHRPWM_AQCTL_ZRO_FRCTOGGLE		GENMASK(1, 0)
 | |
| 
 | |
| #define TI_EHRPWM_AQCTL_CHANA_POLNORMAL		(TI_EHRPWM_AQCTL_CAU_FRCLOW | \
 | |
| 						 TI_EHRPWM_AQCTL_PRD_FRCHIGH | \
 | |
| 						 TI_EHRPWM_AQCTL_ZRO_FRCHIGH)
 | |
| #define TI_EHRPWM_AQCTL_CHANA_POLINVERSED	(TI_EHRPWM_AQCTL_CAU_FRCHIGH | \
 | |
| 						 TI_EHRPWM_AQCTL_PRD_FRCLOW | \
 | |
| 						 TI_EHRPWM_AQCTL_ZRO_FRCLOW)
 | |
| #define TI_EHRPWM_AQCTL_CHANB_POLNORMAL		(TI_EHRPWM_AQCTL_CBU_FRCLOW | \
 | |
| 						 TI_EHRPWM_AQCTL_PRD_FRCHIGH | \
 | |
| 						 TI_EHRPWM_AQCTL_ZRO_FRCHIGH)
 | |
| #define TI_EHRPWM_AQCTL_CHANB_POLINVERSED	(TI_EHRPWM_AQCTL_CBU_FRCHIGH | \
 | |
| 						 TI_EHRPWM_AQCTL_PRD_FRCLOW | \
 | |
| 						 TI_EHRPWM_AQCTL_ZRO_FRCLOW)
 | |
| 
 | |
| #define TI_EHRPWM_AQSFRC_RLDCSF_MASK		GENMASK(7, 6)
 | |
| #define TI_EHRPWM_AQSFRC_RLDCSF_ZRO		0
 | |
| #define TI_EHRPWM_AQSFRC_RLDCSF_PRD		BIT(6)
 | |
| #define TI_EHRPWM_AQSFRC_RLDCSF_ZROPRD		BIT(7)
 | |
| #define TI_EHRPWM_AQSFRC_RLDCSF_IMDT		GENMASK(7, 6)
 | |
| 
 | |
| #define TI_EHRPWM_AQCSFRC_CSFB_MASK		GENMASK(3, 2)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFB_FRCDIS		0
 | |
| #define TI_EHRPWM_AQCSFRC_CSFB_FRCLOW		BIT(2)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFB_FRCHIGH		BIT(3)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFB_DISSWFRC		GENMASK(3, 2)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFA_MASK		GENMASK(1, 0)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFA_FRCDIS		0
 | |
| #define TI_EHRPWM_AQCSFRC_CSFA_FRCLOW		BIT(0)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFA_FRCHIGH		BIT(1)
 | |
| #define TI_EHRPWM_AQCSFRC_CSFA_DISSWFRC		GENMASK(1, 0)
 | |
| 
 | |
| #define TI_EHRPWM_NUM_CHANNELS                  2
 | |
| 
 | |
| struct ti_ehrpwm_priv {
 | |
| 	fdt_addr_t regs;
 | |
| 	u32 clk_rate;
 | |
| 	struct clk tbclk;
 | |
| 	unsigned long period_cycles[TI_EHRPWM_NUM_CHANNELS];
 | |
| 	bool polarity_reversed[TI_EHRPWM_NUM_CHANNELS];
 | |
| };
 | |
| 
 | |
| static void ti_ehrpwm_modify(u16 val, u16 mask, fdt_addr_t reg)
 | |
| {
 | |
| 	unsigned short v;
 | |
| 
 | |
| 	v = readw(reg);
 | |
| 	v &= ~mask;
 | |
| 	v |= val & mask;
 | |
| 	writew(v, reg);
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_set_invert(struct udevice *dev, uint channel,
 | |
| 				bool polarity)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 
 | |
| 	if (channel >= TI_EHRPWM_NUM_CHANNELS)
 | |
| 		return -ENOSPC;
 | |
| 
 | |
| 	/* Configuration of polarity in hardware delayed, do at enable */
 | |
| 	priv->polarity_reversed[channel] = polarity;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * set_prescale_div -	Set up the prescaler divider function
 | |
|  * @rqst_prescaler:	prescaler value min
 | |
|  * @prescale_div:	prescaler value set
 | |
|  * @tb_clk_div:		Time Base Control prescaler bits
 | |
|  */
 | |
| static int set_prescale_div(unsigned long rqst_prescaler, u16 *prescale_div,
 | |
| 			    u16 *tb_clk_div)
 | |
| {
 | |
| 	unsigned int clkdiv, hspclkdiv;
 | |
| 
 | |
| 	for (clkdiv = 0; clkdiv <= TI_EHRPWM_CLKDIV_MAX; clkdiv++) {
 | |
| 		for (hspclkdiv = 0; hspclkdiv <= TI_EHRPWM_HSPCLKDIV_MAX;
 | |
| 		     hspclkdiv++) {
 | |
| 			/*
 | |
| 			 * calculations for prescaler value :
 | |
| 			 * prescale_div = HSPCLKDIVIDER * CLKDIVIDER.
 | |
| 			 * HSPCLKDIVIDER =  2 ** hspclkdiv
 | |
| 			 * CLKDIVIDER = (1),            if clkdiv == 0 *OR*
 | |
| 			 *              (2 * clkdiv),   if clkdiv != 0
 | |
| 			 *
 | |
| 			 * Configure prescale_div value such that period
 | |
| 			 * register value is less than 65535.
 | |
| 			 */
 | |
| 
 | |
| 			*prescale_div = (1 << clkdiv) *
 | |
| 				(hspclkdiv ? (hspclkdiv * 2) : 1);
 | |
| 			if (*prescale_div > rqst_prescaler) {
 | |
| 				*tb_clk_div =
 | |
| 				    (clkdiv << TI_EHRPWM_TBCTL_CLKDIV_SHIFT) |
 | |
| 				    (hspclkdiv <<
 | |
| 				     TI_EHRPWM_TBCTL_HSPCLKDIV_SHIFT);
 | |
| 				return 0;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static void ti_ehrpwm_configure_polarity(struct udevice *dev, uint channel)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 	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 (channel == 1) {
 | |
| 		aqctl_reg = TI_EHRPWM_AQCTLB;
 | |
| 		aqctl_mask = TI_EHRPWM_AQCTL_CBU_MASK;
 | |
| 
 | |
| 		if (priv->polarity_reversed[channel])
 | |
| 			aqctl_val = TI_EHRPWM_AQCTL_CHANB_POLINVERSED;
 | |
| 		else
 | |
| 			aqctl_val = TI_EHRPWM_AQCTL_CHANB_POLNORMAL;
 | |
| 	} else {
 | |
| 		aqctl_reg = TI_EHRPWM_AQCTLA;
 | |
| 		aqctl_mask = TI_EHRPWM_AQCTL_CAU_MASK;
 | |
| 
 | |
| 		if (priv->polarity_reversed[channel])
 | |
| 			aqctl_val = TI_EHRPWM_AQCTL_CHANA_POLINVERSED;
 | |
| 		else
 | |
| 			aqctl_val = TI_EHRPWM_AQCTL_CHANA_POLNORMAL;
 | |
| 	}
 | |
| 
 | |
| 	aqctl_mask |= TI_EHRPWM_AQCTL_PRD_MASK | TI_EHRPWM_AQCTL_ZRO_MASK;
 | |
| 	ti_ehrpwm_modify(aqctl_val, aqctl_mask, priv->regs + aqctl_reg);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * period_ns = 10^9 * (ps_divval * period_cycles) / PWM_CLK_RATE
 | |
|  * duty_ns   = 10^9 * (ps_divval * duty_cycles) / PWM_CLK_RATE
 | |
|  */
 | |
| static int ti_ehrpwm_set_config(struct udevice *dev, uint channel,
 | |
| 				uint period_ns, uint duty_ns)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 	u32 period_cycles, duty_cycles;
 | |
| 	u16 ps_divval, tb_divval;
 | |
| 	unsigned int i, cmp_reg;
 | |
| 	unsigned long long c;
 | |
| 
 | |
| 	if (channel >= TI_EHRPWM_NUM_CHANNELS)
 | |
| 		return -ENOSPC;
 | |
| 
 | |
| 	if (period_ns > NSEC_PER_SEC)
 | |
| 		return -ERANGE;
 | |
| 
 | |
| 	c = priv->clk_rate;
 | |
| 	c = c * period_ns;
 | |
| 	do_div(c, NSEC_PER_SEC);
 | |
| 	period_cycles = (unsigned long)c;
 | |
| 
 | |
| 	if (period_cycles < 1) {
 | |
| 		period_cycles = 1;
 | |
| 		duty_cycles = 1;
 | |
| 	} else {
 | |
| 		c = priv->clk_rate;
 | |
| 		c = c * duty_ns;
 | |
| 		do_div(c, NSEC_PER_SEC);
 | |
| 		duty_cycles = (unsigned long)c;
 | |
| 	}
 | |
| 
 | |
| 	dev_dbg(dev, "channel=%d, period_ns=%d, duty_ns=%d\n",
 | |
| 		channel, period_ns, duty_ns);
 | |
| 
 | |
| 	/*
 | |
| 	 * Period values should be same for multiple PWM channels as IP uses
 | |
| 	 * same period register for multiple channels.
 | |
| 	 */
 | |
| 	for (i = 0; i < TI_EHRPWM_NUM_CHANNELS; i++) {
 | |
| 		if (priv->period_cycles[i] &&
 | |
| 		    priv->period_cycles[i] != period_cycles) {
 | |
| 			/*
 | |
| 			 * Allow channel to reconfigure period if no other
 | |
| 			 * channels being configured.
 | |
| 			 */
 | |
| 			if (i == channel)
 | |
| 				continue;
 | |
| 
 | |
| 			dev_err(dev, "period value conflicts with channel %u\n",
 | |
| 				i);
 | |
| 			return -EINVAL;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	priv->period_cycles[channel] = period_cycles;
 | |
| 
 | |
| 	/* Configure clock prescaler to support Low frequency PWM wave */
 | |
| 	if (set_prescale_div(period_cycles / TI_EHRPWM_PERIOD_MAX, &ps_divval,
 | |
| 			     &tb_divval)) {
 | |
| 		dev_err(dev, "unsupported values\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	/* Update clock prescaler values */
 | |
| 	ti_ehrpwm_modify(tb_divval, TI_EHRPWM_TBCTL_CLKDIV_MASK,
 | |
| 			 priv->regs + TI_EHRPWM_TBCTL);
 | |
| 
 | |
| 	/* 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 */
 | |
| 	ti_ehrpwm_modify(TI_EHRPWM_TBCTL_PRDLD_SHDW, TI_EHRPWM_TBCTL_PRDLD_MASK,
 | |
| 			 priv->regs + TI_EHRPWM_TBCTL);
 | |
| 
 | |
| 	writew(period_cycles, priv->regs + TI_EHRPWM_TBPRD);
 | |
| 
 | |
| 	/* Configure ehrpwm counter for up-count mode */
 | |
| 	ti_ehrpwm_modify(TI_EHRPWM_TBCTL_CTRMODE_UP,
 | |
| 			 TI_EHRPWM_TBCTL_CTRMODE_MASK,
 | |
| 			 priv->regs + TI_EHRPWM_TBCTL);
 | |
| 
 | |
| 	if (channel == 1)
 | |
| 		/* Channel 1 configured with compare B register */
 | |
| 		cmp_reg = TI_EHRPWM_CMPB;
 | |
| 	else
 | |
| 		/* Channel 0 configured with compare A register */
 | |
| 		cmp_reg = TI_EHRPWM_CMPA;
 | |
| 
 | |
| 	writew(duty_cycles, priv->regs + cmp_reg);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_disable(struct udevice *dev, uint channel)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 	u16 aqcsfrc_val, aqcsfrc_mask;
 | |
| 	int err;
 | |
| 
 | |
| 	if (channel >= TI_EHRPWM_NUM_CHANNELS)
 | |
| 		return -ENOSPC;
 | |
| 
 | |
| 	/* Action Qualifier puts PWM output low forcefully */
 | |
| 	if (channel) {
 | |
| 		aqcsfrc_val = TI_EHRPWM_AQCSFRC_CSFB_FRCLOW;
 | |
| 		aqcsfrc_mask = TI_EHRPWM_AQCSFRC_CSFB_MASK;
 | |
| 	} else {
 | |
| 		aqcsfrc_val = TI_EHRPWM_AQCSFRC_CSFA_FRCLOW;
 | |
| 		aqcsfrc_mask = TI_EHRPWM_AQCSFRC_CSFA_MASK;
 | |
| 	}
 | |
| 
 | |
| 	/* Update shadow register first before modifying active register */
 | |
| 	ti_ehrpwm_modify(TI_EHRPWM_AQSFRC_RLDCSF_ZRO,
 | |
| 			 TI_EHRPWM_AQSFRC_RLDCSF_MASK,
 | |
| 			 priv->regs + TI_EHRPWM_AQSFRC);
 | |
| 
 | |
| 	ti_ehrpwm_modify(aqcsfrc_val, aqcsfrc_mask,
 | |
| 			 priv->regs + TI_EHRPWM_AQCSFRC);
 | |
| 
 | |
| 	/*
 | |
| 	 * Changes to immediate action on Action Qualifier. This puts
 | |
| 	 * Action Qualifier control on PWM output from next TBCLK
 | |
| 	 */
 | |
| 	ti_ehrpwm_modify(TI_EHRPWM_AQSFRC_RLDCSF_IMDT,
 | |
| 			 TI_EHRPWM_AQSFRC_RLDCSF_MASK,
 | |
| 			 priv->regs + TI_EHRPWM_AQSFRC);
 | |
| 
 | |
| 	ti_ehrpwm_modify(aqcsfrc_val, aqcsfrc_mask,
 | |
| 			 priv->regs + TI_EHRPWM_AQCSFRC);
 | |
| 
 | |
| 	/* Disabling TBCLK on PWM disable */
 | |
| 	err = clk_disable(&priv->tbclk);
 | |
| 	if (err) {
 | |
| 		dev_err(dev, "failed to disable tbclk\n");
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_enable(struct udevice *dev, uint channel)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 	u16 aqcsfrc_val, aqcsfrc_mask;
 | |
| 	int err;
 | |
| 
 | |
| 	if (channel >= TI_EHRPWM_NUM_CHANNELS)
 | |
| 		return -ENOSPC;
 | |
| 
 | |
| 	/* Disabling Action Qualifier on PWM output */
 | |
| 	if (channel) {
 | |
| 		aqcsfrc_val = TI_EHRPWM_AQCSFRC_CSFB_FRCDIS;
 | |
| 		aqcsfrc_mask = TI_EHRPWM_AQCSFRC_CSFB_MASK;
 | |
| 	} else {
 | |
| 		aqcsfrc_val = TI_EHRPWM_AQCSFRC_CSFA_FRCDIS;
 | |
| 		aqcsfrc_mask = TI_EHRPWM_AQCSFRC_CSFA_MASK;
 | |
| 	}
 | |
| 
 | |
| 	/* Changes to shadow mode */
 | |
| 	ti_ehrpwm_modify(TI_EHRPWM_AQSFRC_RLDCSF_ZRO,
 | |
| 			 TI_EHRPWM_AQSFRC_RLDCSF_MASK,
 | |
| 			 priv->regs + TI_EHRPWM_AQSFRC);
 | |
| 
 | |
| 	ti_ehrpwm_modify(aqcsfrc_val, aqcsfrc_mask,
 | |
| 			 priv->regs + TI_EHRPWM_AQCSFRC);
 | |
| 
 | |
| 	/* Channels polarity can be configured from action qualifier module */
 | |
| 	ti_ehrpwm_configure_polarity(dev, channel);
 | |
| 
 | |
| 	err = clk_enable(&priv->tbclk);
 | |
| 	if (err) {
 | |
| 		dev_err(dev, "failed to enable tbclk\n");
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_set_enable(struct udevice *dev, uint channel, bool enable)
 | |
| {
 | |
| 	if (enable)
 | |
| 		return ti_ehrpwm_enable(dev, channel);
 | |
| 
 | |
| 	return ti_ehrpwm_disable(dev, channel);
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_of_to_plat(struct udevice *dev)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 
 | |
| 	priv->regs = dev_read_addr(dev);
 | |
| 	if (priv->regs == FDT_ADDR_T_NONE) {
 | |
| 		dev_err(dev, "invalid address\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	dev_dbg(dev, "regs=0x%08lx\n", priv->regs);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_remove(struct udevice *dev)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 
 | |
| 	clk_release_all(&priv->tbclk, 1);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int ti_ehrpwm_probe(struct udevice *dev)
 | |
| {
 | |
| 	struct ti_ehrpwm_priv *priv = dev_get_priv(dev);
 | |
| 	struct clk clk;
 | |
| 	int err;
 | |
| 
 | |
| 	err = clk_get_by_name(dev, "fck", &clk);
 | |
| 	if (err) {
 | |
| 		dev_err(dev, "failed to get clock\n");
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	priv->clk_rate = clk_get_rate(&clk);
 | |
| 	if (IS_ERR_VALUE(priv->clk_rate) || !priv->clk_rate) {
 | |
| 		dev_err(dev, "failed to get clock rate\n");
 | |
| 		if (IS_ERR_VALUE(priv->clk_rate))
 | |
| 			return priv->clk_rate;
 | |
| 
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	/* Acquire tbclk for Time Base EHRPWM submodule */
 | |
| 	err = clk_get_by_name(dev, "tbclk", &priv->tbclk);
 | |
| 	if (err) {
 | |
| 		dev_err(dev, "failed to get tbclk clock\n");
 | |
| 		return err;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static const struct pwm_ops ti_ehrpwm_ops = {
 | |
| 	.set_config = ti_ehrpwm_set_config,
 | |
| 	.set_enable = ti_ehrpwm_set_enable,
 | |
| 	.set_invert = ti_ehrpwm_set_invert,
 | |
| };
 | |
| 
 | |
| static const struct udevice_id ti_ehrpwm_ids[] = {
 | |
| 	{.compatible = "ti,am3352-ehrpwm"},
 | |
| 	{.compatible = "ti,am33xx-ehrpwm"},
 | |
| 	{}
 | |
| };
 | |
| 
 | |
| U_BOOT_DRIVER(ti_ehrpwm) = {
 | |
| 	.name = "ti_ehrpwm",
 | |
| 	.id = UCLASS_PWM,
 | |
| 	.of_match = ti_ehrpwm_ids,
 | |
| 	.ops = &ti_ehrpwm_ops,
 | |
| 	.of_to_plat = ti_ehrpwm_of_to_plat,
 | |
| 	.probe = ti_ehrpwm_probe,
 | |
| 	.remove = ti_ehrpwm_remove,
 | |
| 	.priv_auto = sizeof(struct ti_ehrpwm_priv),
 | |
| };
 |