2019-05-20 09:19:02 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2012-07-25 16:58:19 +05:30
/*
* EHRPWM PWM driver
*
2020-07-08 19:59:24 +02:00
* Copyright ( C ) 2012 Texas Instruments , Inc . - https : //www.ti.com/
2012-07-25 16:58:19 +05:30
*/
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/pwm.h>
# include <linux/io.h>
# include <linux/err.h>
# include <linux/clk.h>
# include <linux/pm_runtime.h>
2012-11-27 14:18:13 +05:30
# include <linux/of_device.h>
2012-07-25 16:58:19 +05:30
/* EHRPWM registers and bits definitions */
/* Time base module registers */
# define TBCTL 0x00
# define TBPRD 0x0A
# define TBCTL_PRDLD_MASK BIT(3)
# define TBCTL_PRDLD_SHDW 0
# define TBCTL_PRDLD_IMDT BIT(3)
# define TBCTL_CLKDIV_MASK (BIT(12) | BIT(11) | BIT(10) | BIT(9) | \
BIT ( 8 ) | BIT ( 7 ) )
# define TBCTL_CTRMODE_MASK (BIT(1) | BIT(0))
# define TBCTL_CTRMODE_UP 0
# define TBCTL_CTRMODE_DOWN BIT(0)
# define TBCTL_CTRMODE_UPDOWN BIT(1)
# define TBCTL_CTRMODE_FREEZE (BIT(1) | BIT(0))
# define TBCTL_HSPCLKDIV_SHIFT 7
# define TBCTL_CLKDIV_SHIFT 10
# define CLKDIV_MAX 7
# define HSPCLKDIV_MAX 7
# define PERIOD_MAX 0xFFFF
/* compare module registers */
# define CMPA 0x12
# define CMPB 0x14
/* Action qualifier module registers */
# define AQCTLA 0x16
# define AQCTLB 0x18
# define AQSFRC 0x1A
# define AQCSFRC 0x1C
# define AQCTL_CBU_MASK (BIT(9) | BIT(8))
# define AQCTL_CBU_FRCLOW BIT(8)
# define AQCTL_CBU_FRCHIGH BIT(9)
# define AQCTL_CBU_FRCTOGGLE (BIT(9) | BIT(8))
# define AQCTL_CAU_MASK (BIT(5) | BIT(4))
# define AQCTL_CAU_FRCLOW BIT(4)
# define AQCTL_CAU_FRCHIGH BIT(5)
# define AQCTL_CAU_FRCTOGGLE (BIT(5) | BIT(4))
# define AQCTL_PRD_MASK (BIT(3) | BIT(2))
# define AQCTL_PRD_FRCLOW BIT(2)
# define AQCTL_PRD_FRCHIGH BIT(3)
# define AQCTL_PRD_FRCTOGGLE (BIT(3) | BIT(2))
# define AQCTL_ZRO_MASK (BIT(1) | BIT(0))
# define AQCTL_ZRO_FRCLOW BIT(0)
# define AQCTL_ZRO_FRCHIGH BIT(1)
# define AQCTL_ZRO_FRCTOGGLE (BIT(1) | BIT(0))
2012-09-06 10:40:03 +05:30
# 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 )
2012-07-25 16:58:19 +05:30
# define AQSFRC_RLDCSF_MASK (BIT(7) | BIT(6))
# define AQSFRC_RLDCSF_ZRO 0
# define AQSFRC_RLDCSF_PRD BIT(6)
# define AQSFRC_RLDCSF_ZROPRD BIT(7)
# define AQSFRC_RLDCSF_IMDT (BIT(7) | BIT(6))
# define AQCSFRC_CSFB_MASK (BIT(3) | BIT(2))
# define AQCSFRC_CSFB_FRCDIS 0
# define AQCSFRC_CSFB_FRCLOW BIT(2)
# define AQCSFRC_CSFB_FRCHIGH BIT(3)
# define AQCSFRC_CSFB_DISSWFRC (BIT(3) | BIT(2))
# define AQCSFRC_CSFA_MASK (BIT(1) | BIT(0))
# define AQCSFRC_CSFA_FRCDIS 0
# define AQCSFRC_CSFA_FRCLOW BIT(0)
# define AQCSFRC_CSFA_FRCHIGH BIT(1)
# define AQCSFRC_CSFA_DISSWFRC (BIT(1) | BIT(0))
# define NUM_PWM_CHANNEL 2 /* EHRPWM channels */
2013-01-17 14:50:02 +05:30
struct ehrpwm_context {
u16 tbctl ;
u16 tbprd ;
u16 cmpa ;
u16 cmpb ;
u16 aqctla ;
u16 aqctlb ;
u16 aqsfrc ;
u16 aqcsfrc ;
} ;
2012-07-25 16:58:19 +05:30
struct ehrpwm_pwm_chip {
2017-08-21 08:42:25 +02:00
struct pwm_chip chip ;
unsigned long clk_rate ;
void __iomem * mmio_base ;
2012-09-06 10:44:25 +05:30
unsigned long period_cycles [ NUM_PWM_CHANNEL ] ;
2012-09-06 10:40:03 +05:30
enum pwm_polarity polarity [ NUM_PWM_CHANNEL ] ;
2017-08-21 08:42:25 +02:00
struct clk * tbclk ;
2013-01-17 14:50:02 +05:30
struct ehrpwm_context ctx ;
2012-07-25 16:58:19 +05:30
} ;
static inline struct ehrpwm_pwm_chip * to_ehrpwm_pwm_chip ( struct pwm_chip * chip )
{
return container_of ( chip , struct ehrpwm_pwm_chip , chip ) ;
}
2017-08-21 08:42:25 +02:00
static inline u16 ehrpwm_read ( void __iomem * base , unsigned int offset )
2013-01-17 14:50:02 +05:30
{
return readw ( base + offset ) ;
}
2017-08-21 08:42:25 +02:00
static inline void ehrpwm_write ( void __iomem * base , unsigned int offset ,
u16 value )
2012-07-25 16:58:19 +05:30
{
2017-08-21 08:42:25 +02:00
writew ( value , base + offset ) ;
2012-07-25 16:58:19 +05:30
}
2017-08-21 08:42:25 +02:00
static void ehrpwm_modify ( void __iomem * base , unsigned int offset , u16 mask ,
u16 value )
2012-07-25 16:58:19 +05:30
{
2017-08-21 08:42:25 +02:00
unsigned short val ;
2012-07-25 16:58:19 +05:30
2017-08-21 08:42:25 +02:00
val = readw ( base + offset ) ;
val & = ~ mask ;
val | = value & mask ;
writew ( val , base + offset ) ;
2012-07-25 16:58:19 +05:30
}
/**
* 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
*/
2017-08-21 08:42:25 +02:00
static int set_prescale_div ( unsigned long rqst_prescaler , u16 * prescale_div ,
u16 * tb_clk_div )
2012-07-25 16:58:19 +05:30
{
unsigned int clkdiv , hspclkdiv ;
for ( clkdiv = 0 ; clkdiv < = CLKDIV_MAX ; clkdiv + + ) {
for ( hspclkdiv = 0 ; hspclkdiv < = 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 < < TBCTL_CLKDIV_SHIFT ) |
( hspclkdiv < < TBCTL_HSPCLKDIV_SHIFT ) ;
return 0 ;
}
}
}
2017-08-21 08:42:25 +02:00
2012-07-25 16:58:19 +05:30
return 1 ;
}
2012-09-06 10:40:03 +05:30
static void configure_polarity ( struct ehrpwm_pwm_chip * pc , int chan )
2012-07-25 16:58:19 +05:30
{
2017-08-21 08:42:25 +02:00
u16 aqctl_val , aqctl_mask ;
unsigned int aqctl_reg ;
2012-07-25 16:58:19 +05:30
/*
2012-09-06 10:40:03 +05:30
* 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
2012-07-25 16:58:19 +05:30
*/
if ( chan = = 1 ) {
aqctl_reg = AQCTLB ;
aqctl_mask = AQCTL_CBU_MASK ;
2012-09-06 10:40:03 +05:30
if ( pc - > polarity [ chan ] = = PWM_POLARITY_INVERSED )
aqctl_val = AQCTL_CHANB_POLINVERSED ;
else
aqctl_val = AQCTL_CHANB_POLNORMAL ;
2012-07-25 16:58:19 +05:30
} else {
aqctl_reg = AQCTLA ;
aqctl_mask = AQCTL_CAU_MASK ;
2012-09-06 10:40:03 +05:30
if ( pc - > polarity [ chan ] = = PWM_POLARITY_INVERSED )
aqctl_val = AQCTL_CHANA_POLINVERSED ;
else
aqctl_val = AQCTL_CHANA_POLNORMAL ;
2012-07-25 16:58:19 +05:30
}
aqctl_mask | = AQCTL_PRD_MASK | AQCTL_ZRO_MASK ;
2012-09-06 10:40:03 +05:30
ehrpwm_modify ( pc - > mmio_base , aqctl_reg , aqctl_mask , aqctl_val ) ;
2012-07-25 16:58:19 +05:30
}
/*
* 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 ,
2017-08-21 08:42:25 +02:00
int duty_ns , int period_ns )
2012-07-25 16:58:19 +05:30
{
struct ehrpwm_pwm_chip * pc = to_ehrpwm_pwm_chip ( chip ) ;
2017-08-21 08:42:25 +02:00
u32 period_cycles , duty_cycles ;
u16 ps_divval , tb_divval ;
unsigned int i , cmp_reg ;
2012-07-25 16:58:19 +05:30
unsigned long long c ;
2012-09-02 22:13:40 +02:00
if ( period_ns > NSEC_PER_SEC )
2012-07-25 16:58:19 +05:30
return - ERANGE ;
c = pc - > 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 = pc - > clk_rate ;
c = c * duty_ns ;
do_div ( c , NSEC_PER_SEC ) ;
duty_cycles = ( unsigned long ) c ;
}
2012-09-06 10:44:25 +05:30
/*
* Period values should be same for multiple PWM channels as IP uses
* same period register for multiple channels .
*/
for ( i = 0 ; i < NUM_PWM_CHANNEL ; i + + ) {
if ( pc - > period_cycles [ i ] & &
( pc - > period_cycles [ i ] ! = period_cycles ) ) {
/*
* Allow channel to reconfigure period if no other
* channels being configured .
*/
if ( i = = pwm - > hwpwm )
continue ;
2017-08-21 08:42:25 +02:00
dev_err ( chip - > dev ,
" period value conflicts with channel %u \n " ,
i ) ;
2012-09-06 10:44:25 +05:30
return - EINVAL ;
}
}
pc - > period_cycles [ pwm - > hwpwm ] = period_cycles ;
2012-07-25 16:58:19 +05:30
/* Configure clock prescaler to support Low frequency PWM wave */
if ( set_prescale_div ( period_cycles / PERIOD_MAX , & ps_divval ,
2017-08-21 08:42:25 +02:00
& tb_divval ) ) {
2012-07-25 16:58:19 +05:30
dev_err ( chip - > dev , " Unsupported values \n " ) ;
return - EINVAL ;
}
pm_runtime_get_sync ( chip - > dev ) ;
/* 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 ) ;
ehrpwm_write ( pc - > mmio_base , TBPRD , period_cycles ) ;
/* Configure ehrpwm counter for up-count mode */
ehrpwm_modify ( pc - > mmio_base , TBCTL , TBCTL_CTRMODE_MASK ,
2017-08-21 08:42:25 +02:00
TBCTL_CTRMODE_UP ) ;
2012-07-25 16:58:19 +05:30
2012-09-06 10:40:03 +05:30
if ( pwm - > hwpwm = = 1 )
/* Channel 1 configured with compare B register */
cmp_reg = CMPB ;
else
/* Channel 0 configured with compare A register */
cmp_reg = CMPA ;
ehrpwm_write ( pc - > mmio_base , cmp_reg , duty_cycles ) ;
2012-07-25 16:58:19 +05:30
pm_runtime_put_sync ( chip - > dev ) ;
2017-08-21 08:42:25 +02:00
2012-07-25 16:58:19 +05:30
return 0 ;
}
2012-09-06 10:40:03 +05:30
static int ehrpwm_pwm_set_polarity ( struct pwm_chip * chip ,
2017-08-21 08:42:25 +02:00
struct pwm_device * pwm ,
enum pwm_polarity polarity )
2012-09-06 10:40:03 +05:30
{
struct ehrpwm_pwm_chip * pc = to_ehrpwm_pwm_chip ( chip ) ;
/* Configuration of polarity in hardware delayed, do at enable */
pc - > polarity [ pwm - > hwpwm ] = polarity ;
2017-08-21 08:42:25 +02:00
2012-09-06 10:40:03 +05:30
return 0 ;
}
2012-07-25 16:58:19 +05:30
static int ehrpwm_pwm_enable ( struct pwm_chip * chip , struct pwm_device * pwm )
{
struct ehrpwm_pwm_chip * pc = to_ehrpwm_pwm_chip ( chip ) ;
2017-08-21 08:42:25 +02:00
u16 aqcsfrc_val , aqcsfrc_mask ;
2013-01-10 18:35:26 +05:30
int ret ;
2012-07-25 16:58:19 +05:30
/* Leave clock enabled on enabling PWM */
pm_runtime_get_sync ( chip - > dev ) ;
/* Disabling Action Qualifier on PWM output */
if ( pwm - > hwpwm ) {
aqcsfrc_val = AQCSFRC_CSFB_FRCDIS ;
aqcsfrc_mask = AQCSFRC_CSFB_MASK ;
} else {
aqcsfrc_val = AQCSFRC_CSFA_FRCDIS ;
aqcsfrc_mask = AQCSFRC_CSFA_MASK ;
}
/* Changes to shadow mode */
ehrpwm_modify ( pc - > mmio_base , AQSFRC , AQSFRC_RLDCSF_MASK ,
2017-08-21 08:42:25 +02:00
AQSFRC_RLDCSF_ZRO ) ;
2012-07-25 16:58:19 +05:30
ehrpwm_modify ( pc - > mmio_base , AQCSFRC , aqcsfrc_mask , aqcsfrc_val ) ;
2012-09-06 10:40:03 +05:30
/* Channels polarity can be configured from action qualifier module */
configure_polarity ( pc , pwm - > hwpwm ) ;
2018-06-11 11:39:55 +05:30
/* Enable TBCLK */
2013-06-26 14:38:04 +02:00
ret = clk_enable ( pc - > tbclk ) ;
2013-01-10 18:35:26 +05:30
if ( ret ) {
2017-08-21 08:42:25 +02:00
dev_err ( chip - > dev , " Failed to enable TBCLK for %s: %d \n " ,
dev_name ( pc - > chip . dev ) , ret ) ;
2013-01-10 18:35:26 +05:30
return ret ;
}
2012-11-27 14:18:12 +05:30
2012-07-25 16:58:19 +05:30
return 0 ;
}
static void ehrpwm_pwm_disable ( struct pwm_chip * chip , struct pwm_device * pwm )
{
struct ehrpwm_pwm_chip * pc = to_ehrpwm_pwm_chip ( chip ) ;
2017-08-21 08:42:25 +02:00
u16 aqcsfrc_val , aqcsfrc_mask ;
2012-07-25 16:58:19 +05:30
/* Action Qualifier puts PWM output low forcefully */
if ( pwm - > hwpwm ) {
aqcsfrc_val = AQCSFRC_CSFB_FRCLOW ;
aqcsfrc_mask = AQCSFRC_CSFB_MASK ;
} else {
aqcsfrc_val = AQCSFRC_CSFA_FRCLOW ;
aqcsfrc_mask = AQCSFRC_CSFA_MASK ;
}
2018-06-11 11:39:56 +05:30
/* Update shadow register first before modifying active register */
2019-03-12 14:38:46 +05:30
ehrpwm_modify ( pc - > mmio_base , AQSFRC , AQSFRC_RLDCSF_MASK ,
AQSFRC_RLDCSF_ZRO ) ;
2018-06-11 11:39:56 +05:30
ehrpwm_modify ( pc - > mmio_base , AQCSFRC , aqcsfrc_mask , aqcsfrc_val ) ;
2012-07-25 16:58:19 +05:30
/*
* Changes to immediate action on Action Qualifier . This puts
* Action Qualifier control on PWM output from next TBCLK
*/
ehrpwm_modify ( pc - > mmio_base , AQSFRC , AQSFRC_RLDCSF_MASK ,
2017-08-21 08:42:25 +02:00
AQSFRC_RLDCSF_IMDT ) ;
2012-07-25 16:58:19 +05:30
ehrpwm_modify ( pc - > mmio_base , AQCSFRC , aqcsfrc_mask , aqcsfrc_val ) ;
2012-11-27 14:18:12 +05:30
/* Disabling TBCLK on PWM disable */
2013-06-26 14:38:04 +02:00
clk_disable ( pc - > tbclk ) ;
2012-11-27 14:18:12 +05:30
2012-07-25 16:58:19 +05:30
/* Disable clock on PWM disable */
pm_runtime_put_sync ( chip - > dev ) ;
}
static void ehrpwm_pwm_free ( struct pwm_chip * chip , struct pwm_device * pwm )
{
2012-09-06 10:44:25 +05:30
struct ehrpwm_pwm_chip * pc = to_ehrpwm_pwm_chip ( chip ) ;
2015-07-01 10:21:47 +02:00
if ( pwm_is_enabled ( pwm ) ) {
2012-07-25 16:58:19 +05:30
dev_warn ( chip - > dev , " Removing PWM device without disabling \n " ) ;
pm_runtime_put_sync ( chip - > dev ) ;
}
2012-09-06 10:44:25 +05:30
/* set period value to zero on free */
pc - > period_cycles [ pwm - > hwpwm ] = 0 ;
2012-07-25 16:58:19 +05:30
}
static const struct pwm_ops ehrpwm_pwm_ops = {
2017-08-21 08:42:25 +02:00
. free = ehrpwm_pwm_free ,
. config = ehrpwm_pwm_config ,
. set_polarity = ehrpwm_pwm_set_polarity ,
. enable = ehrpwm_pwm_enable ,
. disable = ehrpwm_pwm_disable ,
. owner = THIS_MODULE ,
2012-07-25 16:58:19 +05:30
} ;
2012-11-27 14:18:13 +05:30
static const struct of_device_id ehrpwm_of_match [ ] = {
2017-08-21 08:42:25 +02:00
{ . compatible = " ti,am3352-ehrpwm " } ,
{ . compatible = " ti,am33xx-ehrpwm " } ,
2012-11-27 14:18:13 +05:30
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , ehrpwm_of_match ) ;
2012-11-19 13:23:14 -05:00
static int ehrpwm_pwm_probe ( struct platform_device * pdev )
2012-07-25 16:58:19 +05:30
{
2016-05-03 10:56:52 -05:00
struct device_node * np = pdev - > dev . of_node ;
2017-08-21 08:42:25 +02:00
struct ehrpwm_pwm_chip * pc ;
2012-07-25 16:58:19 +05:30
struct clk * clk ;
2017-08-21 08:42:25 +02:00
int ret ;
2012-07-25 16:58:19 +05:30
pc = devm_kzalloc ( & pdev - > dev , sizeof ( * pc ) , GFP_KERNEL ) ;
2014-04-23 18:41:48 +09:00
if ( ! pc )
2012-07-25 16:58:19 +05:30
return - ENOMEM ;
clk = devm_clk_get ( & pdev - > dev , " fck " ) ;
2016-05-03 10:56:52 -05:00
if ( IS_ERR ( clk ) ) {
if ( of_device_is_compatible ( np , " ti,am33xx-ecap " ) ) {
dev_warn ( & pdev - > dev , " Binding is obsolete. \n " ) ;
clk = devm_clk_get ( pdev - > dev . parent , " fck " ) ;
}
}
2020-10-30 22:12:54 +02:00
if ( IS_ERR ( clk ) )
return dev_err_probe ( & pdev - > dev , PTR_ERR ( clk ) , " Failed to get fck \n " ) ;
2012-07-25 16:58:19 +05:30
pc - > clk_rate = clk_get_rate ( clk ) ;
if ( ! pc - > clk_rate ) {
dev_err ( & pdev - > dev , " failed to get clock rate \n " ) ;
return - EINVAL ;
}
pc - > chip . dev = & pdev - > dev ;
pc - > chip . ops = & ehrpwm_pwm_ops ;
pc - > chip . npwm = NUM_PWM_CHANNEL ;
2019-12-29 08:05:56 +00:00
pc - > mmio_base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2013-01-21 11:09:16 +01:00
if ( IS_ERR ( pc - > mmio_base ) )
return PTR_ERR ( pc - > mmio_base ) ;
2012-07-25 16:58:19 +05:30
2012-11-27 14:18:12 +05:30
/* Acquire tbclk for Time Base EHRPWM submodule */
pc - > tbclk = devm_clk_get ( & pdev - > dev , " tbclk " ) ;
2020-10-30 22:12:54 +02:00
if ( IS_ERR ( pc - > tbclk ) )
return dev_err_probe ( & pdev - > dev , PTR_ERR ( pc - > tbclk ) , " Failed to get tbclk \n " ) ;
2012-11-27 14:18:12 +05:30
2013-06-26 14:38:04 +02:00
ret = clk_prepare ( pc - > tbclk ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " clk_prepare() failed: %d \n " , ret ) ;
return ret ;
}
2012-07-25 16:58:19 +05:30
ret = pwmchip_add ( & pc - > chip ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " pwmchip_add() failed: %d \n " , ret ) ;
2017-07-20 12:48:17 +02:00
goto err_clk_unprepare ;
2012-07-25 16:58:19 +05:30
}
2017-08-21 08:42:56 +02:00
platform_set_drvdata ( pdev , pc ) ;
2012-07-25 16:58:19 +05:30
pm_runtime_enable ( & pdev - > dev ) ;
2012-11-27 14:18:13 +05:30
2012-07-25 16:58:19 +05:30
return 0 ;
2017-07-20 12:48:17 +02:00
err_clk_unprepare :
clk_unprepare ( pc - > tbclk ) ;
return ret ;
2012-07-25 16:58:19 +05:30
}
2012-11-19 13:26:09 -05:00
static int ehrpwm_pwm_remove ( struct platform_device * pdev )
2012-07-25 16:58:19 +05:30
{
struct ehrpwm_pwm_chip * pc = platform_get_drvdata ( pdev ) ;
2013-06-26 14:38:04 +02:00
clk_unprepare ( pc - > tbclk ) ;
2012-07-25 16:58:19 +05:30
pm_runtime_disable ( & pdev - > dev ) ;
2017-08-21 08:42:25 +02:00
2012-07-25 16:58:19 +05:30
return pwmchip_remove ( & pc - > chip ) ;
}
2014-04-02 13:56:21 +02:00
# ifdef CONFIG_PM_SLEEP
2013-03-26 22:55:57 +08:00
static void ehrpwm_pwm_save_context ( struct ehrpwm_pwm_chip * pc )
2013-01-17 14:50:02 +05:30
{
pm_runtime_get_sync ( pc - > chip . dev ) ;
2017-08-21 08:42:25 +02:00
2013-01-17 14:50:02 +05:30
pc - > ctx . tbctl = ehrpwm_read ( pc - > mmio_base , TBCTL ) ;
pc - > ctx . tbprd = ehrpwm_read ( pc - > mmio_base , TBPRD ) ;
pc - > ctx . cmpa = ehrpwm_read ( pc - > mmio_base , CMPA ) ;
pc - > ctx . cmpb = ehrpwm_read ( pc - > mmio_base , CMPB ) ;
pc - > ctx . aqctla = ehrpwm_read ( pc - > mmio_base , AQCTLA ) ;
pc - > ctx . aqctlb = ehrpwm_read ( pc - > mmio_base , AQCTLB ) ;
pc - > ctx . aqsfrc = ehrpwm_read ( pc - > mmio_base , AQSFRC ) ;
pc - > ctx . aqcsfrc = ehrpwm_read ( pc - > mmio_base , AQCSFRC ) ;
2017-08-21 08:42:25 +02:00
2013-01-17 14:50:02 +05:30
pm_runtime_put_sync ( pc - > chip . dev ) ;
}
2013-03-26 22:55:57 +08:00
static void ehrpwm_pwm_restore_context ( struct ehrpwm_pwm_chip * pc )
2013-01-17 14:50:02 +05:30
{
ehrpwm_write ( pc - > mmio_base , TBPRD , pc - > ctx . tbprd ) ;
ehrpwm_write ( pc - > mmio_base , CMPA , pc - > ctx . cmpa ) ;
ehrpwm_write ( pc - > mmio_base , CMPB , pc - > ctx . cmpb ) ;
ehrpwm_write ( pc - > mmio_base , AQCTLA , pc - > ctx . aqctla ) ;
ehrpwm_write ( pc - > mmio_base , AQCTLB , pc - > ctx . aqctlb ) ;
ehrpwm_write ( pc - > mmio_base , AQSFRC , pc - > ctx . aqsfrc ) ;
ehrpwm_write ( pc - > mmio_base , AQCSFRC , pc - > ctx . aqcsfrc ) ;
ehrpwm_write ( pc - > mmio_base , TBCTL , pc - > ctx . tbctl ) ;
}
static int ehrpwm_pwm_suspend ( struct device * dev )
{
struct ehrpwm_pwm_chip * pc = dev_get_drvdata ( dev ) ;
2017-08-21 08:42:25 +02:00
unsigned int i ;
2013-01-17 14:50:02 +05:30
ehrpwm_pwm_save_context ( pc ) ;
2017-08-21 08:42:25 +02:00
2013-01-17 14:50:02 +05:30
for ( i = 0 ; i < pc - > chip . npwm ; i + + ) {
struct pwm_device * pwm = & pc - > chip . pwms [ i ] ;
2015-07-01 10:21:47 +02:00
if ( ! pwm_is_enabled ( pwm ) )
2013-01-17 14:50:02 +05:30
continue ;
/* Disable explicitly if PWM is running */
pm_runtime_put_sync ( dev ) ;
}
2017-08-21 08:42:25 +02:00
2013-01-17 14:50:02 +05:30
return 0 ;
}
static int ehrpwm_pwm_resume ( struct device * dev )
{
struct ehrpwm_pwm_chip * pc = dev_get_drvdata ( dev ) ;
2017-08-21 08:42:25 +02:00
unsigned int i ;
2013-01-17 14:50:02 +05:30
for ( i = 0 ; i < pc - > chip . npwm ; i + + ) {
struct pwm_device * pwm = & pc - > chip . pwms [ i ] ;
2015-07-01 10:21:47 +02:00
if ( ! pwm_is_enabled ( pwm ) )
2013-01-17 14:50:02 +05:30
continue ;
/* Enable explicitly if PWM was running */
pm_runtime_get_sync ( dev ) ;
}
2017-08-21 08:42:25 +02:00
2013-01-17 14:50:02 +05:30
ehrpwm_pwm_restore_context ( pc ) ;
2017-08-21 08:42:25 +02:00
2013-01-17 14:50:02 +05:30
return 0 ;
}
2013-03-11 11:14:41 +09:00
# endif
2013-01-17 14:50:02 +05:30
static SIMPLE_DEV_PM_OPS ( ehrpwm_pwm_pm_ops , ehrpwm_pwm_suspend ,
2017-08-21 08:42:25 +02:00
ehrpwm_pwm_resume ) ;
2013-01-17 14:50:02 +05:30
2012-07-25 16:58:19 +05:30
static struct platform_driver ehrpwm_pwm_driver = {
. driver = {
2017-08-21 08:42:25 +02:00
. name = " ehrpwm " ,
2012-11-27 14:18:13 +05:30
. of_match_table = ehrpwm_of_match ,
2017-08-21 08:42:25 +02:00
. pm = & ehrpwm_pwm_pm_ops ,
2012-07-25 16:58:19 +05:30
} ,
. probe = ehrpwm_pwm_probe ,
2012-11-19 13:21:28 -05:00
. remove = ehrpwm_pwm_remove ,
2012-07-25 16:58:19 +05:30
} ;
module_platform_driver ( ehrpwm_pwm_driver ) ;
MODULE_DESCRIPTION ( " EHRPWM PWM driver " ) ;
MODULE_AUTHOR ( " Texas Instruments " ) ;
MODULE_LICENSE ( " GPL " ) ;