2013-12-13 10:41:49 +04:00
/*
* Driver for Atmel Pulse Width Modulation Controller
*
* Copyright ( C ) 2013 Atmel Corporation
* Bo Shen < voice . shen @ atmel . com >
*
* Licensed under GPLv2 .
*/
# include <linux/clk.h>
2015-05-25 19:11:49 +03:00
# include <linux/delay.h>
2013-12-13 10:41:49 +04:00
# include <linux/err.h>
# include <linux/io.h>
# include <linux/module.h>
2015-05-25 19:11:49 +03:00
# include <linux/mutex.h>
2013-12-13 10:41:49 +04:00
# include <linux/of.h>
# include <linux/of_device.h>
# include <linux/platform_device.h>
# include <linux/pwm.h>
# include <linux/slab.h>
/* The following is global registers for PWM controller */
# define PWM_ENA 0x04
# define PWM_DIS 0x08
# define PWM_SR 0x0C
2015-05-25 19:11:49 +03:00
# define PWM_ISR 0x1C
2013-12-13 10:41:49 +04:00
/* Bit field in SR */
# define PWM_SR_ALL_CH_ON 0x0F
/* The following register is PWM channel related registers */
# define PWM_CH_REG_OFFSET 0x200
# define PWM_CH_REG_SIZE 0x20
# define PWM_CMR 0x0
/* Bit field in CMR */
# define PWM_CMR_CPOL (1 << 9)
# define PWM_CMR_UPD_CDTY (1 << 10)
2014-03-14 18:19:08 +04:00
# define PWM_CMR_CPRE_MSK 0xF
2013-12-13 10:41:49 +04:00
/* The following registers for PWM v1 */
# define PWMV1_CDTY 0x04
# define PWMV1_CPRD 0x08
# define PWMV1_CUPD 0x10
/* The following registers for PWM v2 */
# define PWMV2_CDTY 0x04
# define PWMV2_CDTYUPD 0x08
# define PWMV2_CPRD 0x0C
# define PWMV2_CPRDUPD 0x10
2017-03-22 16:29:34 +03:00
struct atmel_pwm_registers {
u8 period ;
u8 period_upd ;
u8 duty ;
u8 duty_upd ;
} ;
2019-02-25 19:44:37 +03:00
struct atmel_pwm_config {
u32 max_period ;
u32 max_pres ;
} ;
2019-02-25 19:44:33 +03:00
struct atmel_pwm_data {
struct atmel_pwm_registers regs ;
2019-02-25 19:44:37 +03:00
struct atmel_pwm_config cfg ;
2019-02-25 19:44:33 +03:00
} ;
2013-12-13 10:41:49 +04:00
struct atmel_pwm_chip {
struct pwm_chip chip ;
struct clk * clk ;
void __iomem * base ;
2019-02-25 19:44:33 +03:00
const struct atmel_pwm_data * data ;
2013-12-13 10:41:49 +04:00
2015-05-25 19:11:49 +03:00
unsigned int updated_pwms ;
2016-07-11 13:14:34 +03:00
/* ISR is cleared when read, ensure only one thread does that */
struct mutex isr_lock ;
2013-12-13 10:41:49 +04:00
} ;
static inline struct atmel_pwm_chip * to_atmel_pwm_chip ( struct pwm_chip * chip )
{
return container_of ( chip , struct atmel_pwm_chip , chip ) ;
}
static inline u32 atmel_pwm_readl ( struct atmel_pwm_chip * chip ,
unsigned long offset )
{
return readl_relaxed ( chip - > base + offset ) ;
}
static inline void atmel_pwm_writel ( struct atmel_pwm_chip * chip ,
unsigned long offset , unsigned long val )
{
writel_relaxed ( val , chip - > base + offset ) ;
}
static inline u32 atmel_pwm_ch_readl ( struct atmel_pwm_chip * chip ,
unsigned int ch , unsigned long offset )
{
unsigned long base = PWM_CH_REG_OFFSET + ch * PWM_CH_REG_SIZE ;
return readl_relaxed ( chip - > base + base + offset ) ;
}
static inline void atmel_pwm_ch_writel ( struct atmel_pwm_chip * chip ,
unsigned int ch , unsigned long offset ,
unsigned long val )
{
unsigned long base = PWM_CH_REG_OFFSET + ch * PWM_CH_REG_SIZE ;
writel_relaxed ( val , chip - > base + base + offset ) ;
}
2017-03-22 16:29:34 +03:00
static int atmel_pwm_calculate_cprd_and_pres ( struct pwm_chip * chip ,
const struct pwm_state * state ,
unsigned long * cprd , u32 * pres )
2013-12-13 10:41:49 +04:00
{
struct atmel_pwm_chip * atmel_pwm = to_atmel_pwm_chip ( chip ) ;
2017-03-22 16:29:34 +03:00
unsigned long long cycles = state - > period ;
2013-12-13 10:41:49 +04:00
2014-09-23 17:30:21 +04:00
/* Calculate the period cycles and prescale value */
2017-03-22 16:29:34 +03:00
cycles * = clk_get_rate ( atmel_pwm - > clk ) ;
do_div ( cycles , NSEC_PER_SEC ) ;
2013-12-13 10:41:49 +04:00
2019-02-25 19:44:37 +03:00
for ( * pres = 0 ; cycles > atmel_pwm - > data - > cfg . max_period ; cycles > > = 1 )
2017-03-22 16:29:34 +03:00
( * pres ) + + ;
2014-09-23 17:30:21 +04:00
2019-02-25 19:44:37 +03:00
if ( * pres > atmel_pwm - > data - > cfg . max_pres ) {
2014-09-23 17:30:21 +04:00
dev_err ( chip - > dev , " pres exceeds the maximum value \n " ) ;
return - EINVAL ;
2013-12-13 10:41:49 +04:00
}
2017-03-22 16:29:34 +03:00
* cprd = cycles ;
2013-12-13 10:41:49 +04:00
2017-03-22 16:29:34 +03:00
return 0 ;
2013-12-13 10:41:49 +04:00
}
2017-03-22 16:29:34 +03:00
static void atmel_pwm_calculate_cdty ( const struct pwm_state * state ,
unsigned long cprd , unsigned long * cdty )
2013-12-13 10:41:49 +04:00
{
2017-03-22 16:29:34 +03:00
unsigned long long cycles = state - > duty_cycle ;
2013-12-13 10:41:49 +04:00
2017-03-22 16:29:34 +03:00
cycles * = cprd ;
do_div ( cycles , state - > period ) ;
* cdty = cprd - cycles ;
2013-12-13 10:41:49 +04:00
}
2017-03-22 16:29:34 +03:00
static void atmel_pwm_update_cdty ( struct pwm_chip * chip , struct pwm_device * pwm ,
unsigned long cdty )
2013-12-13 10:41:49 +04:00
{
struct atmel_pwm_chip * atmel_pwm = to_atmel_pwm_chip ( chip ) ;
u32 val ;
2019-02-25 19:44:33 +03:00
if ( atmel_pwm - > data - > regs . duty_upd = =
atmel_pwm - > data - > regs . period_upd ) {
2017-03-22 16:29:34 +03:00
val = atmel_pwm_ch_readl ( atmel_pwm , pwm - > hwpwm , PWM_CMR ) ;
val & = ~ PWM_CMR_UPD_CDTY ;
atmel_pwm_ch_writel ( atmel_pwm , pwm - > hwpwm , PWM_CMR , val ) ;
2013-12-13 10:41:49 +04:00
}
2017-03-22 16:29:34 +03:00
atmel_pwm_ch_writel ( atmel_pwm , pwm - > hwpwm ,
2019-02-25 19:44:33 +03:00
atmel_pwm - > data - > regs . duty_upd , cdty ) ;
2013-12-13 10:41:49 +04:00
}
2017-03-22 16:29:34 +03:00
static void atmel_pwm_set_cprd_cdty ( struct pwm_chip * chip ,
struct pwm_device * pwm ,
unsigned long cprd , unsigned long cdty )
2013-12-13 10:41:49 +04:00
{
struct atmel_pwm_chip * atmel_pwm = to_atmel_pwm_chip ( chip ) ;
2017-03-22 16:29:34 +03:00
atmel_pwm_ch_writel ( atmel_pwm , pwm - > hwpwm ,
2019-02-25 19:44:33 +03:00
atmel_pwm - > data - > regs . duty , cdty ) ;
2017-03-22 16:29:34 +03:00
atmel_pwm_ch_writel ( atmel_pwm , pwm - > hwpwm ,
2019-02-25 19:44:33 +03:00
atmel_pwm - > data - > regs . period , cprd ) ;
2013-12-13 10:41:49 +04:00
}
2017-03-22 16:29:34 +03:00
static void atmel_pwm_disable ( struct pwm_chip * chip , struct pwm_device * pwm ,
bool disable_clk )
2013-12-13 10:41:49 +04:00
{
struct atmel_pwm_chip * atmel_pwm = to_atmel_pwm_chip ( chip ) ;
2015-05-25 19:11:49 +03:00
unsigned long timeout = jiffies + 2 * HZ ;
/*
* Wait for at least a complete period to have passed before disabling a
* channel to be sure that CDTY has been updated
*/
mutex_lock ( & atmel_pwm - > isr_lock ) ;
atmel_pwm - > updated_pwms | = atmel_pwm_readl ( atmel_pwm , PWM_ISR ) ;
while ( ! ( atmel_pwm - > updated_pwms & ( 1 < < pwm - > hwpwm ) ) & &
time_before ( jiffies , timeout ) ) {
usleep_range ( 10 , 100 ) ;
atmel_pwm - > updated_pwms | = atmel_pwm_readl ( atmel_pwm , PWM_ISR ) ;
}
2013-12-13 10:41:49 +04:00
2015-05-25 19:11:49 +03:00
mutex_unlock ( & atmel_pwm - > isr_lock ) ;
2013-12-13 10:41:49 +04:00
atmel_pwm_writel ( atmel_pwm , PWM_DIS , 1 < < pwm - > hwpwm ) ;
2016-05-13 14:09:37 +03:00
/*
* Wait for the PWM channel disable operation to be effective before
* stopping the clock .
*/
timeout = jiffies + 2 * HZ ;
while ( ( atmel_pwm_readl ( atmel_pwm , PWM_SR ) & ( 1 < < pwm - > hwpwm ) ) & &
time_before ( jiffies , timeout ) )
usleep_range ( 10 , 100 ) ;
2017-03-22 16:29:34 +03:00
if ( disable_clk )
clk_disable ( atmel_pwm - > clk ) ;
}
static int atmel_pwm_apply ( struct pwm_chip * chip , struct pwm_device * pwm ,
struct pwm_state * state )
{
struct atmel_pwm_chip * atmel_pwm = to_atmel_pwm_chip ( chip ) ;
struct pwm_state cstate ;
unsigned long cprd , cdty ;
u32 pres , val ;
int ret ;
pwm_get_state ( pwm , & cstate ) ;
if ( state - > enabled ) {
if ( cstate . enabled & &
cstate . polarity = = state - > polarity & &
cstate . period = = state - > period ) {
cprd = atmel_pwm_ch_readl ( atmel_pwm , pwm - > hwpwm ,
2019-02-25 19:44:33 +03:00
atmel_pwm - > data - > regs . period ) ;
2017-03-22 16:29:34 +03:00
atmel_pwm_calculate_cdty ( state , cprd , & cdty ) ;
atmel_pwm_update_cdty ( chip , pwm , cdty ) ;
return 0 ;
}
ret = atmel_pwm_calculate_cprd_and_pres ( chip , state , & cprd ,
& pres ) ;
if ( ret ) {
dev_err ( chip - > dev ,
" failed to calculate cprd and prescaler \n " ) ;
return ret ;
}
atmel_pwm_calculate_cdty ( state , cprd , & cdty ) ;
if ( cstate . enabled ) {
atmel_pwm_disable ( chip , pwm , false ) ;
} else {
ret = clk_enable ( atmel_pwm - > clk ) ;
if ( ret ) {
dev_err ( chip - > dev , " failed to enable clock \n " ) ;
return ret ;
}
}
/* It is necessary to preserve CPOL, inside CMR */
val = atmel_pwm_ch_readl ( atmel_pwm , pwm - > hwpwm , PWM_CMR ) ;
val = ( val & ~ PWM_CMR_CPRE_MSK ) | ( pres & PWM_CMR_CPRE_MSK ) ;
if ( state - > polarity = = PWM_POLARITY_NORMAL )
val & = ~ PWM_CMR_CPOL ;
else
val | = PWM_CMR_CPOL ;
atmel_pwm_ch_writel ( atmel_pwm , pwm - > hwpwm , PWM_CMR , val ) ;
atmel_pwm_set_cprd_cdty ( chip , pwm , cprd , cdty ) ;
mutex_lock ( & atmel_pwm - > isr_lock ) ;
atmel_pwm - > updated_pwms | = atmel_pwm_readl ( atmel_pwm , PWM_ISR ) ;
atmel_pwm - > updated_pwms & = ~ ( 1 < < pwm - > hwpwm ) ;
mutex_unlock ( & atmel_pwm - > isr_lock ) ;
atmel_pwm_writel ( atmel_pwm , PWM_ENA , 1 < < pwm - > hwpwm ) ;
} else if ( cstate . enabled ) {
atmel_pwm_disable ( chip , pwm , true ) ;
}
return 0 ;
2013-12-13 10:41:49 +04:00
}
static const struct pwm_ops atmel_pwm_ops = {
2017-03-22 16:29:34 +03:00
. apply = atmel_pwm_apply ,
2013-12-13 10:41:49 +04:00
. owner = THIS_MODULE ,
} ;
2019-02-25 19:44:41 +03:00
static const struct atmel_pwm_data atmel_sam9rl_pwm_data = {
2019-02-25 19:44:33 +03:00
. regs = {
. period = PWMV1_CPRD ,
. period_upd = PWMV1_CUPD ,
. duty = PWMV1_CDTY ,
. duty_upd = PWMV1_CUPD ,
} ,
2019-02-25 19:44:37 +03:00
. cfg = {
/* 16 bits to keep period and duty. */
2019-03-04 14:10:29 +03:00
. max_period = 0xffff ,
. max_pres = 10 ,
2019-02-25 19:44:37 +03:00
} ,
2013-12-13 10:41:49 +04:00
} ;
2019-02-25 19:44:41 +03:00
static const struct atmel_pwm_data atmel_sama5_pwm_data = {
2019-02-25 19:44:33 +03:00
. regs = {
. period = PWMV2_CPRD ,
. period_upd = PWMV2_CPRDUPD ,
. duty = PWMV2_CDTY ,
. duty_upd = PWMV2_CDTYUPD ,
} ,
2019-02-25 19:44:37 +03:00
. cfg = {
/* 16 bits to keep period and duty. */
2019-03-04 14:10:29 +03:00
. max_period = 0xffff ,
. max_pres = 10 ,
2019-02-25 19:44:37 +03:00
} ,
2013-12-13 10:41:49 +04:00
} ;
2019-02-25 19:44:45 +03:00
static const struct atmel_pwm_data mchp_sam9x60_pwm_data = {
. regs = {
. period = PWMV1_CPRD ,
. period_upd = PWMV1_CUPD ,
. duty = PWMV1_CDTY ,
. duty_upd = PWMV1_CUPD ,
} ,
. cfg = {
/* 32 bits to keep period and duty. */
2019-03-04 14:10:29 +03:00
. max_period = 0xffffffff ,
. max_pres = 10 ,
2019-02-25 19:44:45 +03:00
} ,
} ;
2013-12-13 10:41:49 +04:00
static const struct platform_device_id atmel_pwm_devtypes [ ] = {
{
. name = " at91sam9rl-pwm " ,
2019-02-25 19:44:41 +03:00
. driver_data = ( kernel_ulong_t ) & atmel_sam9rl_pwm_data ,
2013-12-13 10:41:49 +04:00
} , {
. name = " sama5d3-pwm " ,
2019-02-25 19:44:41 +03:00
. driver_data = ( kernel_ulong_t ) & atmel_sama5_pwm_data ,
2013-12-13 10:41:49 +04:00
} , {
/* sentinel */
} ,
} ;
MODULE_DEVICE_TABLE ( platform , atmel_pwm_devtypes ) ;
static const struct of_device_id atmel_pwm_dt_ids [ ] = {
{
. compatible = " atmel,at91sam9rl-pwm " ,
2019-02-25 19:44:41 +03:00
. data = & atmel_sam9rl_pwm_data ,
2013-12-13 10:41:49 +04:00
} , {
. compatible = " atmel,sama5d3-pwm " ,
2019-02-25 19:44:41 +03:00
. data = & atmel_sama5_pwm_data ,
2017-03-22 16:29:35 +03:00
} , {
. compatible = " atmel,sama5d2-pwm " ,
2019-02-25 19:44:41 +03:00
. data = & atmel_sama5_pwm_data ,
2019-02-25 19:44:45 +03:00
} , {
. compatible = " microchip,sam9x60-pwm " ,
. data = & mchp_sam9x60_pwm_data ,
2013-12-13 10:41:49 +04:00
} , {
/* sentinel */
} ,
} ;
MODULE_DEVICE_TABLE ( of , atmel_pwm_dt_ids ) ;
2019-02-25 19:44:33 +03:00
static inline const struct atmel_pwm_data *
2013-12-13 10:41:49 +04:00
atmel_pwm_get_driver_data ( struct platform_device * pdev )
{
2016-07-11 13:14:34 +03:00
const struct platform_device_id * id ;
2016-07-11 13:16:01 +03:00
if ( pdev - > dev . of_node )
return of_device_get_match_data ( & pdev - > dev ) ;
2013-12-13 10:41:49 +04:00
2016-07-11 13:14:34 +03:00
id = platform_get_device_id ( pdev ) ;
2013-12-13 10:41:49 +04:00
2019-02-25 19:44:33 +03:00
return ( struct atmel_pwm_data * ) id - > driver_data ;
2013-12-13 10:41:49 +04:00
}
static int atmel_pwm_probe ( struct platform_device * pdev )
{
2019-02-25 19:44:33 +03:00
const struct atmel_pwm_data * data ;
2013-12-13 10:41:49 +04:00
struct atmel_pwm_chip * atmel_pwm ;
struct resource * res ;
int ret ;
2019-02-25 19:44:33 +03:00
data = atmel_pwm_get_driver_data ( pdev ) ;
if ( ! data )
2013-12-13 10:41:49 +04:00
return - ENODEV ;
atmel_pwm = devm_kzalloc ( & pdev - > dev , sizeof ( * atmel_pwm ) , GFP_KERNEL ) ;
if ( ! atmel_pwm )
return - ENOMEM ;
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
atmel_pwm - > base = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( atmel_pwm - > base ) )
return PTR_ERR ( atmel_pwm - > base ) ;
atmel_pwm - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( atmel_pwm - > clk ) )
return PTR_ERR ( atmel_pwm - > clk ) ;
ret = clk_prepare ( atmel_pwm - > clk ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " failed to prepare PWM clock \n " ) ;
return ret ;
}
atmel_pwm - > chip . dev = & pdev - > dev ;
atmel_pwm - > chip . ops = & atmel_pwm_ops ;
if ( pdev - > dev . of_node ) {
atmel_pwm - > chip . of_xlate = of_pwm_xlate_with_flags ;
atmel_pwm - > chip . of_pwm_n_cells = 3 ;
}
atmel_pwm - > chip . base = - 1 ;
atmel_pwm - > chip . npwm = 4 ;
2019-02-25 19:44:33 +03:00
atmel_pwm - > data = data ;
2015-05-25 19:11:49 +03:00
atmel_pwm - > updated_pwms = 0 ;
mutex_init ( & atmel_pwm - > isr_lock ) ;
2013-12-13 10:41:49 +04:00
ret = pwmchip_add ( & atmel_pwm - > chip ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to add PWM chip %d \n " , ret ) ;
goto unprepare_clk ;
}
platform_set_drvdata ( pdev , atmel_pwm ) ;
2013-12-19 07:42:22 +04:00
return ret ;
2013-12-13 10:41:49 +04:00
unprepare_clk :
clk_unprepare ( atmel_pwm - > clk ) ;
return ret ;
}
static int atmel_pwm_remove ( struct platform_device * pdev )
{
struct atmel_pwm_chip * atmel_pwm = platform_get_drvdata ( pdev ) ;
clk_unprepare ( atmel_pwm - > clk ) ;
2015-05-25 19:11:49 +03:00
mutex_destroy ( & atmel_pwm - > isr_lock ) ;
2013-12-13 10:41:49 +04:00
return pwmchip_remove ( & atmel_pwm - > chip ) ;
}
static struct platform_driver atmel_pwm_driver = {
. driver = {
. name = " atmel-pwm " ,
. of_match_table = of_match_ptr ( atmel_pwm_dt_ids ) ,
} ,
. id_table = atmel_pwm_devtypes ,
. probe = atmel_pwm_probe ,
. remove = atmel_pwm_remove ,
} ;
module_platform_driver ( atmel_pwm_driver ) ;
MODULE_ALIAS ( " platform:atmel-pwm " ) ;
MODULE_AUTHOR ( " Bo Shen <voice.shen@atmel.com> " ) ;
MODULE_DESCRIPTION ( " Atmel PWM driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;