2019-05-29 07:17:56 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2010-12-23 13:11:21 +01:00
/*
2012-07-01 08:29:28 +08:00
* drivers / pwm / pwm - vt8500 . c
2010-12-23 13:11:21 +01:00
*
2012-10-27 14:49:57 +13:00
* Copyright ( C ) 2012 Tony Prisk < linux @ prisktech . co . nz >
* Copyright ( C ) 2010 Alexey Charkov < alchark @ gmail . com >
2010-12-23 13:11:21 +01:00
*/
2023-07-14 11:48:50 -06:00
# include <linux/mod_devicetable.h>
2010-12-23 13:11:21 +01:00
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/platform_device.h>
# include <linux/slab.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/pwm.h>
# include <linux/delay.h>
2012-10-27 14:49:57 +13:00
# include <linux/clk.h>
2010-12-23 13:11:21 +01:00
# include <asm/div64.h>
2012-10-27 14:49:57 +13:00
/*
* SoC architecture allocates register space for 4 PWMs but only
* 2 are currently implemented .
*/
# define VT8500_NR_PWMS 2
2010-12-23 13:11:21 +01:00
2013-01-03 08:44:15 +13:00
# define REG_CTRL(pwm) (((pwm) << 4) + 0x00)
# define REG_SCALAR(pwm) (((pwm) << 4) + 0x04)
# define REG_PERIOD(pwm) (((pwm) << 4) + 0x08)
# define REG_DUTY(pwm) (((pwm) << 4) + 0x0C)
# define REG_STATUS 0x40
# define CTRL_ENABLE BIT(0)
# define CTRL_INVERT BIT(1)
# define CTRL_AUTOLOAD BIT(2)
# define CTRL_STOP_IMM BIT(3)
# define CTRL_LOAD_PRESCALE BIT(4)
# define CTRL_LOAD_PERIOD BIT(5)
# define STATUS_CTRL_UPDATE BIT(0)
# define STATUS_SCALAR_UPDATE BIT(1)
# define STATUS_PERIOD_UPDATE BIT(2)
# define STATUS_DUTY_UPDATE BIT(3)
# define STATUS_ALL_UPDATE 0x0F
2012-03-15 10:04:37 +01:00
struct vt8500_chip {
struct pwm_chip chip ;
void __iomem * base ;
2012-10-27 14:49:57 +13:00
struct clk * clk ;
2010-12-23 13:11:21 +01:00
} ;
2012-03-15 10:04:37 +01:00
# define to_vt8500_chip(chip) container_of(chip, struct vt8500_chip, chip)
2010-12-23 13:11:21 +01:00
# define msecs_to_loops(t) (loops_per_jiffy / 1000 * HZ * t)
2021-11-02 10:28:04 +01:00
static inline void vt8500_pwm_busy_wait ( struct vt8500_chip * vt8500 , int nr , u8 bitmask )
2010-12-23 13:11:21 +01:00
{
int loops = msecs_to_loops ( 10 ) ;
2013-01-03 08:44:15 +13:00
u32 mask = bitmask < < ( nr < < 8 ) ;
while ( ( readl ( vt8500 - > base + REG_STATUS ) & mask ) & & - - loops )
2010-12-23 13:11:21 +01:00
cpu_relax ( ) ;
if ( unlikely ( ! loops ) )
2013-01-03 08:44:15 +13:00
dev_warn ( vt8500 - > chip . dev , " Waiting for status bits 0x%x to clear timed out \n " ,
mask ) ;
2010-12-23 13:11:21 +01:00
}
2012-03-15 10:04:37 +01:00
static int vt8500_pwm_config ( struct pwm_chip * chip , struct pwm_device * pwm ,
2021-11-02 10:28:03 +01:00
u64 duty_ns , u64 period_ns )
2010-12-23 13:11:21 +01:00
{
2012-03-15 10:04:37 +01:00
struct vt8500_chip * vt8500 = to_vt8500_chip ( chip ) ;
2010-12-23 13:11:21 +01:00
unsigned long long c ;
unsigned long period_cycles , prescale , pv , dc ;
2012-11-20 06:44:46 +13:00
int err ;
2013-01-03 08:44:15 +13:00
u32 val ;
2012-11-20 06:44:46 +13:00
err = clk_enable ( vt8500 - > clk ) ;
if ( err < 0 ) {
dev_err ( chip - > dev , " failed to enable clock \n " ) ;
return err ;
}
2010-12-23 13:11:21 +01:00
2012-10-27 14:49:57 +13:00
c = clk_get_rate ( vt8500 - > clk ) ;
2010-12-23 13:11:21 +01:00
c = c * period_ns ;
do_div ( c , 1000000000 ) ;
period_cycles = c ;
if ( period_cycles < 1 )
period_cycles = 1 ;
prescale = ( period_cycles - 1 ) / 4096 ;
pv = period_cycles / ( prescale + 1 ) - 1 ;
if ( pv > 4095 )
pv = 4095 ;
2012-11-20 06:44:46 +13:00
if ( prescale > 1023 ) {
clk_disable ( vt8500 - > clk ) ;
2010-12-23 13:11:21 +01:00
return - EINVAL ;
2012-11-20 06:44:46 +13:00
}
2010-12-23 13:11:21 +01:00
c = ( unsigned long long ) pv * duty_ns ;
2021-11-02 10:28:03 +01:00
dc = div64_u64 ( c , period_ns ) ;
2010-12-23 13:11:21 +01:00
2013-01-03 08:44:15 +13:00
writel ( prescale , vt8500 - > base + REG_SCALAR ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_SCALAR_UPDATE ) ;
2010-12-23 13:11:21 +01:00
2013-01-03 08:44:15 +13:00
writel ( pv , vt8500 - > base + REG_PERIOD ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_PERIOD_UPDATE ) ;
2010-12-23 13:11:21 +01:00
2013-01-03 08:44:15 +13:00
writel ( dc , vt8500 - > base + REG_DUTY ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_DUTY_UPDATE ) ;
2010-12-23 13:11:21 +01:00
2013-01-03 08:44:15 +13:00
val = readl ( vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
val | = CTRL_AUTOLOAD ;
writel ( val , vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_CTRL_UPDATE ) ;
2010-12-23 13:11:21 +01:00
2012-11-20 06:44:46 +13:00
clk_disable ( vt8500 - > clk ) ;
2010-12-23 13:11:21 +01:00
return 0 ;
}
2012-03-15 10:04:37 +01:00
static int vt8500_pwm_enable ( struct pwm_chip * chip , struct pwm_device * pwm )
2010-12-23 13:11:21 +01:00
{
2012-03-15 10:04:37 +01:00
struct vt8500_chip * vt8500 = to_vt8500_chip ( chip ) ;
2013-01-03 08:44:15 +13:00
int err ;
u32 val ;
2010-12-23 13:11:21 +01:00
2012-10-27 14:49:57 +13:00
err = clk_enable ( vt8500 - > clk ) ;
2012-11-20 06:44:45 +13:00
if ( err < 0 ) {
2012-10-27 14:49:57 +13:00
dev_err ( chip - > dev , " failed to enable clock \n " ) ;
return err ;
2012-11-20 06:44:46 +13:00
}
2012-10-27 14:49:57 +13:00
2013-01-03 08:44:15 +13:00
val = readl ( vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
val | = CTRL_ENABLE ;
writel ( val , vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_CTRL_UPDATE ) ;
2013-01-03 08:44:15 +13:00
2012-03-15 10:04:37 +01:00
return 0 ;
2010-12-23 13:11:21 +01:00
}
2012-03-15 10:04:37 +01:00
static void vt8500_pwm_disable ( struct pwm_chip * chip , struct pwm_device * pwm )
2010-12-23 13:11:21 +01:00
{
2012-03-15 10:04:37 +01:00
struct vt8500_chip * vt8500 = to_vt8500_chip ( chip ) ;
2013-01-03 08:44:15 +13:00
u32 val ;
2010-12-23 13:11:21 +01:00
2013-01-03 08:44:15 +13:00
val = readl ( vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
val & = ~ CTRL_ENABLE ;
writel ( val , vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_CTRL_UPDATE ) ;
2012-10-27 14:49:57 +13:00
clk_disable ( vt8500 - > clk ) ;
2010-12-23 13:11:21 +01:00
}
2013-01-03 08:44:16 +13:00
static int vt8500_pwm_set_polarity ( struct pwm_chip * chip ,
struct pwm_device * pwm ,
enum pwm_polarity polarity )
{
struct vt8500_chip * vt8500 = to_vt8500_chip ( chip ) ;
u32 val ;
val = readl ( vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
if ( polarity = = PWM_POLARITY_INVERSED )
val | = CTRL_INVERT ;
else
val & = ~ CTRL_INVERT ;
writel ( val , vt8500 - > base + REG_CTRL ( pwm - > hwpwm ) ) ;
2021-11-02 10:28:04 +01:00
vt8500_pwm_busy_wait ( vt8500 , pwm - > hwpwm , STATUS_CTRL_UPDATE ) ;
2013-01-03 08:44:16 +13:00
return 0 ;
}
2021-11-02 10:28:03 +01:00
static int vt8500_pwm_apply ( struct pwm_chip * chip , struct pwm_device * pwm ,
const struct pwm_state * state )
{
int err ;
bool enabled = pwm - > state . enabled ;
if ( state - > polarity ! = pwm - > state . polarity ) {
/*
* Changing the polarity of a running PWM is only allowed when
* the PWM driver implements - > apply ( ) .
*/
if ( enabled ) {
vt8500_pwm_disable ( chip , pwm ) ;
enabled = false ;
}
err = vt8500_pwm_set_polarity ( chip , pwm , state - > polarity ) ;
if ( err )
return err ;
}
if ( ! state - > enabled ) {
if ( enabled )
vt8500_pwm_disable ( chip , pwm ) ;
return 0 ;
}
/*
* We cannot skip calling - > config even if state - > period = =
* pwm - > state . period & & state - > duty_cycle = = pwm - > state . duty_cycle
* because we might have exited early in the last call to
* pwm_apply_state because of ! state - > enabled and so the two values in
* pwm - > state might not be configured in hardware .
*/
err = vt8500_pwm_config ( pwm - > chip , pwm , state - > duty_cycle , state - > period ) ;
if ( err )
return err ;
if ( ! enabled )
err = vt8500_pwm_enable ( chip , pwm ) ;
return err ;
}
2017-01-10 23:42:06 +05:30
static const struct pwm_ops vt8500_pwm_ops = {
2021-11-02 10:28:03 +01:00
. apply = vt8500_pwm_apply ,
2012-03-15 10:04:37 +01:00
. owner = THIS_MODULE ,
} ;
2010-12-23 13:11:21 +01:00
2012-10-27 14:49:57 +13:00
static const struct of_device_id vt8500_pwm_dt_ids [ ] = {
{ . compatible = " via,vt8500-pwm " , } ,
{ /* Sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , vt8500_pwm_dt_ids ) ;
static int vt8500_pwm_probe ( struct platform_device * pdev )
2010-12-23 13:11:21 +01:00
{
2022-02-11 10:37:41 +08:00
struct vt8500_chip * vt8500 ;
2012-10-27 14:49:57 +13:00
struct device_node * np = pdev - > dev . of_node ;
2012-03-15 10:04:37 +01:00
int ret ;
2010-12-23 13:11:21 +01:00
2012-10-27 14:49:57 +13:00
if ( ! np ) {
dev_err ( & pdev - > dev , " invalid devicetree node \n " ) ;
return - EINVAL ;
}
2022-02-11 10:37:41 +08:00
vt8500 = devm_kzalloc ( & pdev - > dev , sizeof ( * vt8500 ) , GFP_KERNEL ) ;
if ( vt8500 = = NULL )
2010-12-23 13:11:21 +01:00
return - ENOMEM ;
2022-02-11 10:37:41 +08:00
vt8500 - > chip . dev = & pdev - > dev ;
vt8500 - > chip . ops = & vt8500_pwm_ops ;
vt8500 - > chip . npwm = VT8500_NR_PWMS ;
2010-12-23 13:11:21 +01:00
2022-02-11 10:37:41 +08:00
vt8500 - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( vt8500 - > clk ) ) {
2012-10-27 14:49:57 +13:00
dev_err ( & pdev - > dev , " clock source not specified \n " ) ;
2022-02-11 10:37:41 +08:00
return PTR_ERR ( vt8500 - > clk ) ;
2012-10-27 14:49:57 +13:00
}
2022-02-11 10:37:41 +08:00
vt8500 - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
if ( IS_ERR ( vt8500 - > base ) )
return PTR_ERR ( vt8500 - > base ) ;
2010-12-23 13:11:21 +01:00
2022-02-11 10:37:41 +08:00
ret = clk_prepare ( vt8500 - > clk ) ;
2012-10-27 14:49:57 +13:00
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to prepare clock \n " ) ;
return ret ;
}
2022-02-11 10:37:41 +08:00
ret = pwmchip_add ( & vt8500 - > chip ) ;
2012-10-27 14:49:57 +13:00
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to add PWM chip \n " ) ;
2022-02-11 10:37:41 +08:00
clk_unprepare ( vt8500 - > clk ) ;
2012-07-01 08:29:28 +08:00
return ret ;
2012-10-27 14:49:57 +13:00
}
2010-12-23 13:11:21 +01:00
2022-02-11 10:37:41 +08:00
platform_set_drvdata ( pdev , vt8500 ) ;
2012-03-15 10:04:37 +01:00
return ret ;
2010-12-23 13:11:21 +01:00
}
2023-03-03 19:54:44 +01:00
static void vt8500_pwm_remove ( struct platform_device * pdev )
2010-12-23 13:11:21 +01:00
{
2022-02-11 10:37:41 +08:00
struct vt8500_chip * vt8500 = platform_get_drvdata ( pdev ) ;
2010-12-23 13:11:21 +01:00
2022-02-11 10:37:41 +08:00
pwmchip_remove ( & vt8500 - > chip ) ;
2010-12-23 13:11:21 +01:00
2022-02-11 10:37:41 +08:00
clk_unprepare ( vt8500 - > clk ) ;
2010-12-23 13:11:21 +01:00
}
2012-10-27 14:49:57 +13:00
static struct platform_driver vt8500_pwm_driver = {
. probe = vt8500_pwm_probe ,
2023-03-03 19:54:44 +01:00
. remove_new = vt8500_pwm_remove ,
2010-12-23 13:11:21 +01:00
. driver = {
. name = " vt8500-pwm " ,
2012-10-27 14:49:57 +13:00
. of_match_table = vt8500_pwm_dt_ids ,
2010-12-23 13:11:21 +01:00
} ,
} ;
2012-10-27 14:49:57 +13:00
module_platform_driver ( vt8500_pwm_driver ) ;
2010-12-23 13:11:21 +01:00
2012-10-27 14:49:57 +13:00
MODULE_DESCRIPTION ( " VT8500 PWM Driver " ) ;
MODULE_AUTHOR ( " Tony Prisk <linux@prisktech.co.nz> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;