2019-06-11 11:14:44 +05:30
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright ( C ) 2017 - 2018 SiFive
* For SiFive ' s PWM IP block documentation please refer Chapter 14 of
* Reference Manual : https : //static.dev.sifive.com/FU540-C000-v1.0.pdf
*
* Limitations :
* - When changing both duty cycle and period , we cannot prevent in
* software that the output might produce a period with mixed
* settings ( new period length and old duty cycle ) .
* - The hardware cannot generate a 100 % duty cycle .
* - The hardware generates only inverted output .
*/
# include <linux/clk.h>
# include <linux/io.h>
2023-07-14 11:48:50 -06:00
# include <linux/mod_devicetable.h>
2019-06-11 11:14:44 +05:30
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/pwm.h>
# include <linux/slab.h>
# include <linux/bitfield.h>
/* Register offsets */
# define PWM_SIFIVE_PWMCFG 0x0
# define PWM_SIFIVE_PWMCOUNT 0x8
# define PWM_SIFIVE_PWMS 0x10
2022-07-21 12:31:23 +02:00
# define PWM_SIFIVE_PWMCMP(i) (0x20 + 4 * (i))
2019-06-11 11:14:44 +05:30
/* PWMCFG fields */
# define PWM_SIFIVE_PWMCFG_SCALE GENMASK(3, 0)
# define PWM_SIFIVE_PWMCFG_STICKY BIT(8)
# define PWM_SIFIVE_PWMCFG_ZERO_CMP BIT(9)
# define PWM_SIFIVE_PWMCFG_DEGLITCH BIT(10)
# define PWM_SIFIVE_PWMCFG_EN_ALWAYS BIT(12)
# define PWM_SIFIVE_PWMCFG_EN_ONCE BIT(13)
# define PWM_SIFIVE_PWMCFG_CENTER BIT(16)
# define PWM_SIFIVE_PWMCFG_GANG BIT(24)
# define PWM_SIFIVE_PWMCFG_IP BIT(28)
# define PWM_SIFIVE_CMPWIDTH 16
# define PWM_SIFIVE_DEFAULT_PERIOD 10000000
struct pwm_sifive_ddata {
struct pwm_chip chip ;
2022-07-21 12:31:25 +02:00
struct mutex lock ; /* lock to protect user_count and approx_period */
2019-06-11 11:14:44 +05:30
struct notifier_block notifier ;
struct clk * clk ;
void __iomem * regs ;
unsigned int real_period ;
unsigned int approx_period ;
int user_count ;
} ;
static inline
2023-07-14 22:56:21 +02:00
struct pwm_sifive_ddata * pwm_sifive_chip_to_ddata ( struct pwm_chip * chip )
2019-06-11 11:14:44 +05:30
{
2023-07-14 22:56:21 +02:00
return container_of ( chip , struct pwm_sifive_ddata , chip ) ;
2019-06-11 11:14:44 +05:30
}
static int pwm_sifive_request ( struct pwm_chip * chip , struct pwm_device * pwm )
{
struct pwm_sifive_ddata * ddata = pwm_sifive_chip_to_ddata ( chip ) ;
mutex_lock ( & ddata - > lock ) ;
ddata - > user_count + + ;
mutex_unlock ( & ddata - > lock ) ;
return 0 ;
}
static void pwm_sifive_free ( struct pwm_chip * chip , struct pwm_device * pwm )
{
struct pwm_sifive_ddata * ddata = pwm_sifive_chip_to_ddata ( chip ) ;
mutex_lock ( & ddata - > lock ) ;
ddata - > user_count - - ;
mutex_unlock ( & ddata - > lock ) ;
}
2022-07-21 12:31:25 +02:00
/* Called holding ddata->lock */
2019-06-11 11:14:44 +05:30
static void pwm_sifive_update_clock ( struct pwm_sifive_ddata * ddata ,
unsigned long rate )
{
unsigned long long num ;
unsigned long scale_pow ;
int scale ;
u32 val ;
/*
* The PWM unit is used with pwmzerocmp = 0 , so the only way to modify the
* period length is using pwmscale which provides the number of bits the
* counter is shifted before being feed to the comparators . A period
* lasts ( 1 < < ( PWM_SIFIVE_CMPWIDTH + pwmscale ) ) clock ticks .
* ( 1 < < ( PWM_SIFIVE_CMPWIDTH + scale ) ) * 10 ^ 9 / rate = period
*/
scale_pow = div64_ul ( ddata - > approx_period * ( u64 ) rate , NSEC_PER_SEC ) ;
scale = clamp ( ilog2 ( scale_pow ) - PWM_SIFIVE_CMPWIDTH , 0 , 0xf ) ;
val = PWM_SIFIVE_PWMCFG_EN_ALWAYS |
FIELD_PREP ( PWM_SIFIVE_PWMCFG_SCALE , scale ) ;
writel ( val , ddata - > regs + PWM_SIFIVE_PWMCFG ) ;
/* As scale <= 15 the shift operation cannot overflow. */
num = ( unsigned long long ) NSEC_PER_SEC < < ( PWM_SIFIVE_CMPWIDTH + scale ) ;
ddata - > real_period = div64_ul ( num , rate ) ;
dev_dbg ( ddata - > chip . dev ,
" New real_period = %u ns \n " , ddata - > real_period ) ;
}
2022-12-02 19:35:26 +01:00
static int pwm_sifive_get_state ( struct pwm_chip * chip , struct pwm_device * pwm ,
struct pwm_state * state )
2019-06-11 11:14:44 +05:30
{
struct pwm_sifive_ddata * ddata = pwm_sifive_chip_to_ddata ( chip ) ;
u32 duty , val ;
2022-07-21 12:31:23 +02:00
duty = readl ( ddata - > regs + PWM_SIFIVE_PWMCMP ( pwm - > hwpwm ) ) ;
2019-06-11 11:14:44 +05:30
state - > enabled = duty > 0 ;
val = readl ( ddata - > regs + PWM_SIFIVE_PWMCFG ) ;
if ( ! ( val & PWM_SIFIVE_PWMCFG_EN_ALWAYS ) )
state - > enabled = false ;
state - > period = ddata - > real_period ;
state - > duty_cycle =
( u64 ) duty * ddata - > real_period > > PWM_SIFIVE_CMPWIDTH ;
state - > polarity = PWM_POLARITY_INVERSED ;
2022-12-02 19:35:26 +01:00
return 0 ;
2019-06-11 11:14:44 +05:30
}
static int pwm_sifive_apply ( struct pwm_chip * chip , struct pwm_device * pwm ,
2019-08-24 17:37:07 +02:00
const struct pwm_state * state )
2019-06-11 11:14:44 +05:30
{
struct pwm_sifive_ddata * ddata = pwm_sifive_chip_to_ddata ( chip ) ;
struct pwm_state cur_state ;
unsigned int duty_cycle ;
unsigned long long num ;
bool enabled ;
int ret = 0 ;
u32 frac ;
if ( state - > polarity ! = PWM_POLARITY_INVERSED )
return - EINVAL ;
cur_state = pwm - > state ;
enabled = cur_state . enabled ;
duty_cycle = state - > duty_cycle ;
if ( ! state - > enabled )
duty_cycle = 0 ;
/*
* The problem of output producing mixed setting as mentioned at top ,
* occurs here . To minimize the window for this problem , we are
* calculating the register values first and then writing them
* consecutively
*/
num = ( u64 ) duty_cycle * ( 1U < < PWM_SIFIVE_CMPWIDTH ) ;
2020-06-02 15:31:12 -07:00
frac = DIV64_U64_ROUND_CLOSEST ( num , state - > period ) ;
2019-06-11 11:14:44 +05:30
/* The hardware cannot generate a 100% duty cycle */
frac = min ( frac , ( 1U < < PWM_SIFIVE_CMPWIDTH ) - 1 ) ;
2022-07-21 12:31:25 +02:00
mutex_lock ( & ddata - > lock ) ;
2019-06-11 11:14:44 +05:30
if ( state - > period ! = ddata - > approx_period ) {
2022-11-09 12:37:24 +01:00
/*
* Don ' t let a 2 nd user change the period underneath the 1 st user .
* However if ddate - > approx_period = = 0 this is the first time we set
* any period , so let whoever gets here first set the period so other
* users who agree on the period won ' t fail .
*/
if ( ddata - > user_count ! = 1 & & ddata - > approx_period ) {
2022-07-21 12:31:25 +02:00
mutex_unlock ( & ddata - > lock ) ;
2022-07-21 12:31:26 +02:00
return - EBUSY ;
2019-06-11 11:14:44 +05:30
}
ddata - > approx_period = state - > period ;
pwm_sifive_update_clock ( ddata , clk_get_rate ( ddata - > clk ) ) ;
}
2022-07-21 12:31:25 +02:00
mutex_unlock ( & ddata - > lock ) ;
2019-06-11 11:14:44 +05:30
2022-07-21 12:31:27 +02:00
/*
* If the PWM is enabled the clk is already on . So only enable it
* conditionally to have it on exactly once afterwards independent of
* the PWM state .
*/
if ( ! enabled ) {
ret = clk_enable ( ddata - > clk ) ;
if ( ret ) {
dev_err ( ddata - > chip . dev , " Enable clk failed \n " ) ;
return ret ;
}
2022-07-21 12:31:26 +02:00
}
2022-07-21 12:31:23 +02:00
writel ( frac , ddata - > regs + PWM_SIFIVE_PWMCMP ( pwm - > hwpwm ) ) ;
2019-06-11 11:14:44 +05:30
2022-07-21 12:31:27 +02:00
if ( ! state - > enabled )
clk_disable ( ddata - > clk ) ;
2019-06-11 11:14:44 +05:30
2022-07-21 12:31:26 +02:00
return 0 ;
2019-06-11 11:14:44 +05:30
}
static const struct pwm_ops pwm_sifive_ops = {
. request = pwm_sifive_request ,
. free = pwm_sifive_free ,
. get_state = pwm_sifive_get_state ,
. apply = pwm_sifive_apply ,
} ;
static int pwm_sifive_clock_notifier ( struct notifier_block * nb ,
unsigned long event , void * data )
{
struct clk_notifier_data * ndata = data ;
struct pwm_sifive_ddata * ddata =
container_of ( nb , struct pwm_sifive_ddata , notifier ) ;
2022-12-02 19:35:05 +01:00
if ( event = = POST_RATE_CHANGE ) {
mutex_lock ( & ddata - > lock ) ;
2019-06-11 11:14:44 +05:30
pwm_sifive_update_clock ( ddata , ndata - > new_rate ) ;
2022-12-02 19:35:05 +01:00
mutex_unlock ( & ddata - > lock ) ;
}
2019-06-11 11:14:44 +05:30
return NOTIFY_OK ;
}
static int pwm_sifive_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct pwm_sifive_ddata * ddata ;
struct pwm_chip * chip ;
int ret ;
2022-07-21 12:31:28 +02:00
u32 val ;
unsigned int enabled_pwms = 0 , enabled_clks = 1 ;
2019-06-11 11:14:44 +05:30
ddata = devm_kzalloc ( dev , sizeof ( * ddata ) , GFP_KERNEL ) ;
if ( ! ddata )
return - ENOMEM ;
mutex_init ( & ddata - > lock ) ;
chip = & ddata - > chip ;
chip - > dev = dev ;
chip - > ops = & pwm_sifive_ops ;
chip - > npwm = 4 ;
2019-12-29 08:06:09 +00:00
ddata - > regs = devm_platform_ioremap_resource ( pdev , 0 ) ;
2019-07-18 15:51:11 +08:00
if ( IS_ERR ( ddata - > regs ) )
2019-06-11 11:14:44 +05:30
return PTR_ERR ( ddata - > regs ) ;
2023-04-18 22:21:02 +02:00
ddata - > clk = devm_clk_get_prepared ( dev , NULL ) ;
2020-08-26 16:47:45 +02:00
if ( IS_ERR ( ddata - > clk ) )
return dev_err_probe ( dev , PTR_ERR ( ddata - > clk ) ,
" Unable to find controller clock \n " ) ;
2019-06-11 11:14:44 +05:30
2023-04-18 22:21:02 +02:00
ret = clk_enable ( ddata - > clk ) ;
2019-06-11 11:14:44 +05:30
if ( ret ) {
dev_err ( dev , " failed to enable clock for pwm: %d \n " , ret ) ;
return ret ;
}
2022-07-21 12:31:28 +02:00
val = readl ( ddata - > regs + PWM_SIFIVE_PWMCFG ) ;
if ( val & PWM_SIFIVE_PWMCFG_EN_ALWAYS ) {
unsigned int i ;
for ( i = 0 ; i < chip - > npwm ; + + i ) {
val = readl ( ddata - > regs + PWM_SIFIVE_PWMCMP ( i ) ) ;
if ( val > 0 )
+ + enabled_pwms ;
}
}
/* The clk should be on once for each running PWM. */
if ( enabled_pwms ) {
while ( enabled_clks < enabled_pwms ) {
/* This is not expected to fail as the clk is already on */
ret = clk_enable ( ddata - > clk ) ;
if ( unlikely ( ret ) ) {
dev_err_probe ( dev , ret , " Failed to enable clk \n " ) ;
goto disable_clk ;
}
+ + enabled_clks ;
}
} else {
clk_disable ( ddata - > clk ) ;
enabled_clks = 0 ;
}
2019-06-11 11:14:44 +05:30
/* Watch for changes to underlying clock frequency */
ddata - > notifier . notifier_call = pwm_sifive_clock_notifier ;
ret = clk_notifier_register ( ddata - > clk , & ddata - > notifier ) ;
if ( ret ) {
dev_err ( dev , " failed to register clock notifier: %d \n " , ret ) ;
goto disable_clk ;
}
ret = pwmchip_add ( chip ) ;
if ( ret < 0 ) {
dev_err ( dev , " cannot register PWM: %d \n " , ret ) ;
goto unregister_clk ;
}
platform_set_drvdata ( pdev , ddata ) ;
dev_dbg ( dev , " SiFive PWM chip registered %d PWMs \n " , chip - > npwm ) ;
return 0 ;
unregister_clk :
clk_notifier_unregister ( ddata - > clk , & ddata - > notifier ) ;
disable_clk :
2022-07-21 12:31:28 +02:00
while ( enabled_clks ) {
clk_disable ( ddata - > clk ) ;
- - enabled_clks ;
}
2019-06-11 11:14:44 +05:30
return ret ;
}
2023-03-03 19:54:35 +01:00
static void pwm_sifive_remove ( struct platform_device * dev )
2019-06-11 11:14:44 +05:30
{
struct pwm_sifive_ddata * ddata = platform_get_drvdata ( dev ) ;
struct pwm_device * pwm ;
2021-07-07 18:28:33 +02:00
int ch ;
2019-06-11 11:14:44 +05:30
2022-07-21 12:31:29 +02:00
pwmchip_remove ( & ddata - > chip ) ;
clk_notifier_unregister ( ddata - > clk , & ddata - > notifier ) ;
2019-06-11 11:14:44 +05:30
for ( ch = 0 ; ch < ddata - > chip . npwm ; ch + + ) {
pwm = & ddata - > chip . pwms [ ch ] ;
2022-07-21 12:31:28 +02:00
if ( pwm - > state . enabled )
clk_disable ( ddata - > clk ) ;
2019-06-11 11:14:44 +05:30
}
}
static const struct of_device_id pwm_sifive_of_match [ ] = {
{ . compatible = " sifive,pwm0 " } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , pwm_sifive_of_match ) ;
static struct platform_driver pwm_sifive_driver = {
. probe = pwm_sifive_probe ,
2023-03-03 19:54:35 +01:00
. remove_new = pwm_sifive_remove ,
2019-06-11 11:14:44 +05:30
. driver = {
. name = " pwm-sifive " ,
. of_match_table = pwm_sifive_of_match ,
} ,
} ;
module_platform_driver ( pwm_sifive_driver ) ;
MODULE_DESCRIPTION ( " SiFive PWM driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;