2014-04-25 22:31:12 +04:00
/*
* Copyright ( C ) 2014 Broadcom Corporation
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation version 2.
*
* This program is distributed " as is " WITHOUT ANY WARRANTY of any
* kind , whether express or implied ; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*/
# include <linux/clk.h>
# include <linux/delay.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/ioport.h>
# include <linux/math64.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/platform_device.h>
# include <linux/pwm.h>
# include <linux/slab.h>
# include <linux/types.h>
/*
* The Kona PWM has some unusual characteristics . Here are the main points .
*
* 1 ) There is no disable bit and the hardware docs advise programming a zero
* duty to achieve output equivalent to that of a normal disable operation .
*
* 2 ) Changes to prescale , duty , period , and polarity do not take effect until
* a subsequent rising edge of the trigger bit .
*
* 3 ) If the smooth bit and trigger bit are both low , the output is a constant
* high signal . Otherwise , the earlier waveform continues to be output .
*
* 4 ) If the smooth bit is set on the rising edge of the trigger bit , output
* will transition to the new settings on a period boundary ( which could be
* seconds away ) . If the smooth bit is clear , new settings will be applied
* as soon as possible ( the hardware always has a 400 ns delay ) .
*
* 5 ) When the external clock that feeds the PWM is disabled , output is pegged
* high or low depending on its state at that exact instant .
*/
2019-01-16 22:41:22 +03:00
# define PWM_CONTROL_OFFSET 0x00000000
2014-04-25 22:31:12 +04:00
# define PWM_CONTROL_SMOOTH_SHIFT(chan) (24 + (chan))
# define PWM_CONTROL_TYPE_SHIFT(chan) (16 + (chan))
# define PWM_CONTROL_POLARITY_SHIFT(chan) (8 + (chan))
# define PWM_CONTROL_TRIGGER_SHIFT(chan) (chan)
2019-01-16 22:41:22 +03:00
# define PRESCALE_OFFSET 0x00000004
2014-04-25 22:31:12 +04:00
# define PRESCALE_SHIFT(chan) ((chan) << 2)
# define PRESCALE_MASK(chan) (0x7 << PRESCALE_SHIFT(chan))
2019-01-16 22:41:22 +03:00
# define PRESCALE_MIN 0x00000000
# define PRESCALE_MAX 0x00000007
2014-04-25 22:31:12 +04:00
# define PERIOD_COUNT_OFFSET(chan) (0x00000008 + ((chan) << 3))
2019-01-16 22:41:22 +03:00
# define PERIOD_COUNT_MIN 0x00000002
# define PERIOD_COUNT_MAX 0x00ffffff
2014-04-25 22:31:12 +04:00
# define DUTY_CYCLE_HIGH_OFFSET(chan) (0x0000000c + ((chan) << 3))
2019-01-16 22:41:22 +03:00
# define DUTY_CYCLE_HIGH_MIN 0x00000000
# define DUTY_CYCLE_HIGH_MAX 0x00ffffff
2014-04-25 22:31:12 +04:00
struct kona_pwmc {
struct pwm_chip chip ;
void __iomem * base ;
struct clk * clk ;
} ;
static inline struct kona_pwmc * to_kona_pwmc ( struct pwm_chip * _chip )
{
return container_of ( _chip , struct kona_pwmc , chip ) ;
}
2015-06-16 00:21:01 +03:00
/*
* Clear trigger bit but set smooth bit to maintain old output .
*/
static void kona_pwmc_prepare_for_settings ( struct kona_pwmc * kp ,
unsigned int chan )
2014-04-25 22:31:12 +04:00
{
unsigned int value = readl ( kp - > base + PWM_CONTROL_OFFSET ) ;
value | = 1 < < PWM_CONTROL_SMOOTH_SHIFT ( chan ) ;
value & = ~ ( 1 < < PWM_CONTROL_TRIGGER_SHIFT ( chan ) ) ;
writel ( value , kp - > base + PWM_CONTROL_OFFSET ) ;
2015-06-16 00:21:01 +03:00
/*
* There must be a min 400 ns delay between clearing trigger and setting
* it . Failing to do this may result in no PWM signal .
*/
ndelay ( 400 ) ;
}
static void kona_pwmc_apply_settings ( struct kona_pwmc * kp , unsigned int chan )
{
unsigned int value = readl ( kp - > base + PWM_CONTROL_OFFSET ) ;
2014-04-25 22:31:12 +04:00
/* Set trigger bit and clear smooth bit to apply new settings */
value & = ~ ( 1 < < PWM_CONTROL_SMOOTH_SHIFT ( chan ) ) ;
value | = 1 < < PWM_CONTROL_TRIGGER_SHIFT ( chan ) ;
writel ( value , kp - > base + PWM_CONTROL_OFFSET ) ;
2015-06-16 00:21:01 +03:00
/* Trigger bit must be held high for at least 400 ns. */
ndelay ( 400 ) ;
2014-04-25 22:31:12 +04:00
}
static int kona_pwmc_config ( struct pwm_chip * chip , struct pwm_device * pwm ,
int duty_ns , int period_ns )
{
struct kona_pwmc * kp = to_kona_pwmc ( chip ) ;
u64 val , div , rate ;
unsigned long prescale = PRESCALE_MIN , pc , dc ;
unsigned int value , chan = pwm - > hwpwm ;
/*
* Find period count , duty count and prescale to suit duty_ns and
* period_ns . This is done according to formulas described below :
*
* period_ns = 10 ^ 9 * ( PRESCALE + 1 ) * PC / PWM_CLK_RATE
* duty_ns = 10 ^ 9 * ( PRESCALE + 1 ) * DC / PWM_CLK_RATE
*
* PC = ( PWM_CLK_RATE * period_ns ) / ( 10 ^ 9 * ( PRESCALE + 1 ) )
* DC = ( PWM_CLK_RATE * duty_ns ) / ( 10 ^ 9 * ( PRESCALE + 1 ) )
*/
rate = clk_get_rate ( kp - > clk ) ;
while ( 1 ) {
div = 1000000000 ;
div * = 1 + prescale ;
val = rate * period_ns ;
pc = div64_u64 ( val , div ) ;
val = rate * duty_ns ;
dc = div64_u64 ( val , div ) ;
/* If duty_ns or period_ns are not achievable then return */
2020-06-29 15:47:50 +03:00
if ( pc < PERIOD_COUNT_MIN )
2014-04-25 22:31:12 +04:00
return - EINVAL ;
/* If pc and dc are in bounds, the calculation is done */
if ( pc < = PERIOD_COUNT_MAX & & dc < = DUTY_CYCLE_HIGH_MAX )
break ;
/* Otherwise, increase prescale and recalculate pc and dc */
if ( + + prescale > PRESCALE_MAX )
return - EINVAL ;
}
2015-06-16 00:21:01 +03:00
/*
* Don ' t apply settings if disabled . The period and duty cycle are
* always calculated above to ensure the new values are
* validated immediately instead of on enable .
*/
2015-07-01 11:21:47 +03:00
if ( pwm_is_enabled ( pwm ) ) {
2015-06-16 00:21:01 +03:00
kona_pwmc_prepare_for_settings ( kp , chan ) ;
2014-04-25 22:31:12 +04:00
value = readl ( kp - > base + PRESCALE_OFFSET ) ;
value & = ~ PRESCALE_MASK ( chan ) ;
value | = prescale < < PRESCALE_SHIFT ( chan ) ;
writel ( value , kp - > base + PRESCALE_OFFSET ) ;
writel ( pc , kp - > base + PERIOD_COUNT_OFFSET ( chan ) ) ;
writel ( dc , kp - > base + DUTY_CYCLE_HIGH_OFFSET ( chan ) ) ;
kona_pwmc_apply_settings ( kp , chan ) ;
}
return 0 ;
}
static int kona_pwmc_set_polarity ( struct pwm_chip * chip , struct pwm_device * pwm ,
enum pwm_polarity polarity )
{
struct kona_pwmc * kp = to_kona_pwmc ( chip ) ;
unsigned int chan = pwm - > hwpwm ;
unsigned int value ;
int ret ;
ret = clk_prepare_enable ( kp - > clk ) ;
if ( ret < 0 ) {
dev_err ( chip - > dev , " failed to enable clock: %d \n " , ret ) ;
return ret ;
}
2015-06-16 00:21:01 +03:00
kona_pwmc_prepare_for_settings ( kp , chan ) ;
2014-04-25 22:31:12 +04:00
value = readl ( kp - > base + PWM_CONTROL_OFFSET ) ;
if ( polarity = = PWM_POLARITY_NORMAL )
value | = 1 < < PWM_CONTROL_POLARITY_SHIFT ( chan ) ;
else
value & = ~ ( 1 < < PWM_CONTROL_POLARITY_SHIFT ( chan ) ) ;
writel ( value , kp - > base + PWM_CONTROL_OFFSET ) ;
kona_pwmc_apply_settings ( kp , chan ) ;
clk_disable_unprepare ( kp - > clk ) ;
return 0 ;
}
static int kona_pwmc_enable ( struct pwm_chip * chip , struct pwm_device * pwm )
{
struct kona_pwmc * kp = to_kona_pwmc ( chip ) ;
int ret ;
ret = clk_prepare_enable ( kp - > clk ) ;
if ( ret < 0 ) {
dev_err ( chip - > dev , " failed to enable clock: %d \n " , ret ) ;
return ret ;
}
2015-07-01 11:21:50 +03:00
ret = kona_pwmc_config ( chip , pwm , pwm_get_duty_cycle ( pwm ) ,
pwm_get_period ( pwm ) ) ;
2014-04-25 22:31:12 +04:00
if ( ret < 0 ) {
clk_disable_unprepare ( kp - > clk ) ;
return ret ;
}
return 0 ;
}
static void kona_pwmc_disable ( struct pwm_chip * chip , struct pwm_device * pwm )
{
struct kona_pwmc * kp = to_kona_pwmc ( chip ) ;
unsigned int chan = pwm - > hwpwm ;
2015-06-16 00:21:01 +03:00
unsigned int value ;
kona_pwmc_prepare_for_settings ( kp , chan ) ;
2014-04-25 22:31:12 +04:00
/* Simulate a disable by configuring for zero duty */
writel ( 0 , kp - > base + DUTY_CYCLE_HIGH_OFFSET ( chan ) ) ;
2015-06-16 00:21:01 +03:00
writel ( 0 , kp - > base + PERIOD_COUNT_OFFSET ( chan ) ) ;
2014-04-25 22:31:12 +04:00
2015-06-16 00:21:01 +03:00
/* Set prescale to 0 for this channel */
value = readl ( kp - > base + PRESCALE_OFFSET ) ;
value & = ~ PRESCALE_MASK ( chan ) ;
writel ( value , kp - > base + PRESCALE_OFFSET ) ;
kona_pwmc_apply_settings ( kp , chan ) ;
2014-04-25 22:31:12 +04:00
clk_disable_unprepare ( kp - > clk ) ;
}
static const struct pwm_ops kona_pwm_ops = {
. config = kona_pwmc_config ,
. set_polarity = kona_pwmc_set_polarity ,
. enable = kona_pwmc_enable ,
. disable = kona_pwmc_disable ,
. owner = THIS_MODULE ,
} ;
static int kona_pwmc_probe ( struct platform_device * pdev )
{
struct kona_pwmc * kp ;
unsigned int chan ;
unsigned int value = 0 ;
int ret = 0 ;
kp = devm_kzalloc ( & pdev - > dev , sizeof ( * kp ) , GFP_KERNEL ) ;
if ( kp = = NULL )
return - ENOMEM ;
kp - > chip . dev = & pdev - > dev ;
kp - > chip . ops = & kona_pwm_ops ;
kp - > chip . npwm = 6 ;
2019-12-29 11:05:50 +03:00
kp - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2014-04-25 22:31:12 +04:00
if ( IS_ERR ( kp - > base ) )
return PTR_ERR ( kp - > base ) ;
kp - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( kp - > clk ) ) {
dev_err ( & pdev - > dev , " failed to get clock: %ld \n " ,
PTR_ERR ( kp - > clk ) ) ;
return PTR_ERR ( kp - > clk ) ;
}
ret = clk_prepare_enable ( kp - > clk ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to enable clock: %d \n " , ret ) ;
return ret ;
}
2015-05-26 23:08:17 +03:00
/* Set push/pull for all channels */
for ( chan = 0 ; chan < kp - > chip . npwm ; chan + + )
2014-04-25 22:31:12 +04:00
value | = ( 1 < < PWM_CONTROL_TYPE_SHIFT ( chan ) ) ;
writel ( value , kp - > base + PWM_CONTROL_OFFSET ) ;
clk_disable_unprepare ( kp - > clk ) ;
2021-07-07 19:28:01 +03:00
ret = devm_pwmchip_add ( & pdev - > dev , & kp - > chip ) ;
2014-04-25 22:31:12 +04:00
if ( ret < 0 )
dev_err ( & pdev - > dev , " failed to add PWM chip: %d \n " , ret ) ;
return ret ;
}
static const struct of_device_id bcm_kona_pwmc_dt [ ] = {
{ . compatible = " brcm,kona-pwm " } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , bcm_kona_pwmc_dt ) ;
static struct platform_driver kona_pwmc_driver = {
. driver = {
. name = " bcm-kona-pwm " ,
. of_match_table = bcm_kona_pwmc_dt ,
} ,
. probe = kona_pwmc_probe ,
} ;
module_platform_driver ( kona_pwmc_driver ) ;
MODULE_AUTHOR ( " Broadcom Corporation <bcm-kernel-feedback-list@broadcom.com> " ) ;
MODULE_AUTHOR ( " Tim Kryger <tkryger@broadcom.com> " ) ;
MODULE_DESCRIPTION ( " Broadcom Kona PWM driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;