2019-05-20 16:04:21 +02:00
// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2016-08-22 17:36:30 +02:00
/*
2019-06-12 21:59:11 +02:00
* PWM controller driver for Amlogic Meson SoCs .
*
* This PWM is only a set of Gates , Dividers and Counters :
* PWM output is achieved by calculating a clock that permits calculating
* two periods ( low and high ) . The counter then has to be set to switch after
* N cycles for the first half period .
* The hardware has no " polarity " setting . This driver reverses the period
* cycles ( the low length is inverted with the high length ) for
* PWM_POLARITY_INVERSED . This means that . get_state cannot read the polarity
* from the hardware .
* Setting the duty cycle will disable and re - enable the PWM output .
* Disabling the PWM stops the output immediately ( without waiting for the
* current period to complete first ) .
*
* The public S912 ( GXM ) datasheet contains some documentation for this PWM
* controller starting on page 543 :
* https : //dl.khadas.com/Hardware/VIM2/Datasheet/S912_Datasheet_V0.220170314publicversion-Wesion.pdf
* An updated version of this IP block is found in S922X ( G12B ) SoCs . The
* datasheet contains the description for this IP block revision starting at
* page 1084 :
* https : //dn.odroid.com/S922X/ODROID-N2/Datasheet/S922X_Public_Datasheet_V0.2.pdf
*
2016-08-22 17:36:30 +02:00
* Copyright ( c ) 2016 BayLibre , SAS .
* Author : Neil Armstrong < narmstrong @ baylibre . com >
* Copyright ( C ) 2014 Amlogic , Inc .
*/
2019-06-12 21:59:00 +02:00
# include <linux/bitfield.h>
# include <linux/bits.h>
2016-08-22 17:36:30 +02:00
# include <linux/clk.h>
# include <linux/clk-provider.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/kernel.h>
2019-06-12 21:59:07 +02:00
# include <linux/math64.h>
2016-08-22 17:36:30 +02:00
# include <linux/module.h>
# include <linux/of.h>
# include <linux/of_device.h>
# include <linux/platform_device.h>
# include <linux/pwm.h>
# include <linux/slab.h>
# include <linux/spinlock.h>
# define REG_PWM_A 0x0
# define REG_PWM_B 0x4
2019-06-12 21:59:00 +02:00
# define PWM_LOW_MASK GENMASK(15, 0)
# define PWM_HIGH_MASK GENMASK(31, 16)
2016-08-22 17:36:30 +02:00
# define REG_MISC_AB 0x8
2023-05-24 21:52:21 +02:00
# define MISC_B_CLK_EN_SHIFT 23
# define MISC_A_CLK_EN_SHIFT 15
# define MISC_CLK_DIV_WIDTH 7
2016-08-22 17:36:30 +02:00
# define MISC_B_CLK_DIV_SHIFT 16
# define MISC_A_CLK_DIV_SHIFT 8
# define MISC_B_CLK_SEL_SHIFT 6
# define MISC_A_CLK_SEL_SHIFT 4
2019-06-12 21:59:01 +02:00
# define MISC_CLK_SEL_MASK 0x3
2016-08-22 17:36:30 +02:00
# define MISC_B_EN BIT(1)
# define MISC_A_EN BIT(0)
2019-06-12 21:59:04 +02:00
# define MESON_NUM_PWMS 2
2023-05-24 21:50:47 +02:00
# define MESON_MAX_MUX_PARENTS 4
2019-06-12 21:59:04 +02:00
2019-06-12 21:59:05 +02:00
static struct meson_pwm_channel_data {
u8 reg_offset ;
u8 clk_sel_shift ;
u8 clk_div_shift ;
2023-05-24 21:52:21 +02:00
u8 clk_en_shift ;
2019-06-12 21:59:05 +02:00
u32 pwm_en_mask ;
} meson_pwm_per_channel_data [ MESON_NUM_PWMS ] = {
{
. reg_offset = REG_PWM_A ,
. clk_sel_shift = MISC_A_CLK_SEL_SHIFT ,
. clk_div_shift = MISC_A_CLK_DIV_SHIFT ,
2023-05-24 21:52:21 +02:00
. clk_en_shift = MISC_A_CLK_EN_SHIFT ,
2019-06-12 21:59:05 +02:00
. pwm_en_mask = MISC_A_EN ,
} ,
{
. reg_offset = REG_PWM_B ,
. clk_sel_shift = MISC_B_CLK_SEL_SHIFT ,
. clk_div_shift = MISC_B_CLK_DIV_SHIFT ,
2023-05-24 21:52:21 +02:00
. clk_en_shift = MISC_B_CLK_EN_SHIFT ,
2019-06-12 21:59:05 +02:00
. pwm_en_mask = MISC_B_EN ,
}
2016-08-22 17:36:30 +02:00
} ;
struct meson_pwm_channel {
2023-05-24 21:52:21 +02:00
unsigned long rate ;
2016-08-22 17:36:30 +02:00
unsigned int hi ;
unsigned int lo ;
struct clk_mux mux ;
2023-05-24 21:52:21 +02:00
struct clk_divider div ;
struct clk_gate gate ;
2016-08-22 17:36:30 +02:00
struct clk * clk ;
} ;
struct meson_pwm_data {
const char * const * parent_names ;
2017-06-08 14:24:15 +02:00
unsigned int num_parents ;
2016-08-22 17:36:30 +02:00
} ;
struct meson_pwm {
struct pwm_chip chip ;
const struct meson_pwm_data * data ;
2019-06-12 21:59:04 +02:00
struct meson_pwm_channel channels [ MESON_NUM_PWMS ] ;
2016-08-22 17:36:30 +02:00
void __iomem * base ;
2019-04-01 19:57:48 +02:00
/*
* Protects register ( write ) access to the REG_MISC_AB register
* that is shared between the two PWMs .
*/
2016-08-22 17:36:30 +02:00
spinlock_t lock ;
} ;
static inline struct meson_pwm * to_meson_pwm ( struct pwm_chip * chip )
{
return container_of ( chip , struct meson_pwm , chip ) ;
}
static int meson_pwm_request ( struct pwm_chip * chip , struct pwm_device * pwm )
{
2019-06-12 21:59:06 +02:00
struct meson_pwm * meson = to_meson_pwm ( chip ) ;
2021-11-08 14:46:24 +01:00
struct meson_pwm_channel * channel = & meson - > channels [ pwm - > hwpwm ] ;
2016-08-22 17:36:30 +02:00
struct device * dev = chip - > dev ;
int err ;
err = clk_prepare_enable ( channel - > clk ) ;
if ( err < 0 ) {
dev_err ( dev , " failed to enable clock %s: %d \n " ,
__clk_get_name ( channel - > clk ) , err ) ;
return err ;
}
2021-11-08 14:46:26 +01:00
return 0 ;
2016-08-22 17:36:30 +02:00
}
static void meson_pwm_free ( struct pwm_chip * chip , struct pwm_device * pwm )
{
2021-11-08 14:46:26 +01:00
struct meson_pwm * meson = to_meson_pwm ( chip ) ;
struct meson_pwm_channel * channel = & meson - > channels [ pwm - > hwpwm ] ;
2016-08-22 17:36:30 +02:00
2021-11-08 14:46:25 +01:00
clk_disable_unprepare ( channel - > clk ) ;
2016-08-22 17:36:30 +02:00
}
2019-06-12 21:59:03 +02:00
static int meson_pwm_calc ( struct meson_pwm * meson , struct pwm_device * pwm ,
2019-08-24 17:37:07 +02:00
const struct pwm_state * state )
2016-08-22 17:36:30 +02:00
{
2021-11-08 14:46:26 +01:00
struct meson_pwm_channel * channel = & meson - > channels [ pwm - > hwpwm ] ;
2023-05-24 21:52:21 +02:00
unsigned int cnt , duty_cnt ;
2020-04-02 12:08:57 +01:00
unsigned long fin_freq ;
2023-05-24 21:52:21 +02:00
u64 duty , period , freq ;
2016-08-22 17:36:30 +02:00
2019-06-12 21:59:02 +02:00
duty = state - > duty_cycle ;
period = state - > period ;
2023-03-22 22:45:44 +01:00
/*
* Note this is wrong . The result is an output wave that isn ' t really
* inverted and so is wrongly identified by . get_state as normal .
* Fixing this needs some care however as some machines might rely on
* this .
*/
2019-06-12 21:59:02 +02:00
if ( state - > polarity = = PWM_POLARITY_INVERSED )
2016-08-22 17:36:30 +02:00
duty = period - duty ;
2023-05-24 21:52:21 +02:00
freq = div64_u64 ( NSEC_PER_SEC * 0xffffULL , period ) ;
if ( freq > ULONG_MAX )
freq = ULONG_MAX ;
fin_freq = clk_round_rate ( channel - > clk , freq ) ;
2016-08-22 17:36:30 +02:00
if ( fin_freq = = 0 ) {
dev_err ( meson - > chip . dev , " invalid source clock frequency \n " ) ;
return - EINVAL ;
}
dev_dbg ( meson - > chip . dev , " fin_freq: %lu Hz \n " , fin_freq ) ;
2023-05-24 21:52:21 +02:00
cnt = div_u64 ( fin_freq * period , NSEC_PER_SEC ) ;
2019-06-12 21:59:07 +02:00
if ( cnt > 0xffff ) {
dev_err ( meson - > chip . dev , " unable to get period cnt \n " ) ;
return - EINVAL ;
}
2023-05-24 21:52:21 +02:00
dev_dbg ( meson - > chip . dev , " period=%llu cnt=%u \n " , period , cnt ) ;
2016-08-22 17:36:30 +02:00
if ( duty = = period ) {
channel - > hi = cnt ;
channel - > lo = 0 ;
} else if ( duty = = 0 ) {
channel - > hi = 0 ;
channel - > lo = cnt ;
} else {
2023-05-24 21:52:21 +02:00
duty_cnt = div_u64 ( fin_freq * duty , NSEC_PER_SEC ) ;
2016-08-22 17:36:30 +02:00
2023-05-24 21:52:21 +02:00
dev_dbg ( meson - > chip . dev , " duty=%llu duty_cnt=%u \n " , duty , duty_cnt ) ;
2016-08-22 17:36:30 +02:00
channel - > hi = duty_cnt ;
channel - > lo = cnt - duty_cnt ;
}
2023-05-24 21:52:21 +02:00
channel - > rate = fin_freq ;
2016-08-22 17:36:30 +02:00
return 0 ;
}
2019-06-12 21:58:58 +02:00
static void meson_pwm_enable ( struct meson_pwm * meson , struct pwm_device * pwm )
2016-08-22 17:36:30 +02:00
{
2021-11-08 14:46:26 +01:00
struct meson_pwm_channel * channel = & meson - > channels [ pwm - > hwpwm ] ;
2019-06-12 21:59:05 +02:00
struct meson_pwm_channel_data * channel_data ;
2019-04-01 19:57:48 +02:00
unsigned long flags ;
2019-06-12 21:59:05 +02:00
u32 value ;
2023-05-24 21:52:21 +02:00
int err ;
2016-08-22 17:36:30 +02:00
2019-06-12 21:59:05 +02:00
channel_data = & meson_pwm_per_channel_data [ pwm - > hwpwm ] ;
2016-08-22 17:36:30 +02:00
2023-05-24 21:52:21 +02:00
err = clk_set_rate ( channel - > clk , channel - > rate ) ;
if ( err )
dev_err ( meson - > chip . dev , " setting clock rate failed \n " ) ;
2019-04-01 19:57:48 +02:00
2023-05-24 21:52:21 +02:00
spin_lock_irqsave ( & meson - > lock , flags ) ;
2016-08-22 17:36:30 +02:00
2019-06-12 21:59:00 +02:00
value = FIELD_PREP ( PWM_HIGH_MASK , channel - > hi ) |
FIELD_PREP ( PWM_LOW_MASK , channel - > lo ) ;
2019-06-12 21:59:05 +02:00
writel ( value , meson - > base + channel_data - > reg_offset ) ;
2016-08-22 17:36:30 +02:00
value = readl ( meson - > base + REG_MISC_AB ) ;
2019-06-12 21:59:05 +02:00
value | = channel_data - > pwm_en_mask ;
2016-08-22 17:36:30 +02:00
writel ( value , meson - > base + REG_MISC_AB ) ;
2019-04-01 19:57:48 +02:00
spin_unlock_irqrestore ( & meson - > lock , flags ) ;
2016-08-22 17:36:30 +02:00
}
2019-06-12 21:58:58 +02:00
static void meson_pwm_disable ( struct meson_pwm * meson , struct pwm_device * pwm )
2016-08-22 17:36:30 +02:00
{
2019-04-01 19:57:48 +02:00
unsigned long flags ;
2019-06-12 21:59:05 +02:00
u32 value ;
2016-08-22 17:36:30 +02:00
2019-04-01 19:57:48 +02:00
spin_lock_irqsave ( & meson - > lock , flags ) ;
2016-08-22 17:36:30 +02:00
value = readl ( meson - > base + REG_MISC_AB ) ;
2019-06-12 21:59:05 +02:00
value & = ~ meson_pwm_per_channel_data [ pwm - > hwpwm ] . pwm_en_mask ;
2016-08-22 17:36:30 +02:00
writel ( value , meson - > base + REG_MISC_AB ) ;
2019-04-01 19:57:48 +02:00
spin_unlock_irqrestore ( & meson - > lock , flags ) ;
2016-08-22 17:36:30 +02:00
}
static int meson_pwm_apply ( struct pwm_chip * chip , struct pwm_device * pwm ,
2019-08-24 17:37:07 +02:00
const struct pwm_state * state )
2016-08-22 17:36:30 +02:00
{
struct meson_pwm * meson = to_meson_pwm ( chip ) ;
2021-11-08 14:46:26 +01:00
struct meson_pwm_channel * channel = & meson - > channels [ pwm - > hwpwm ] ;
2016-08-22 17:36:30 +02:00
int err = 0 ;
if ( ! state - > enabled ) {
2019-06-12 21:59:10 +02:00
if ( state - > polarity = = PWM_POLARITY_INVERSED ) {
/*
* This IP block revision doesn ' t have an " always high "
* setting which we can use for " inverted disabled " .
2023-05-24 21:52:21 +02:00
* Instead we achieve this by setting mux parent with
* highest rate and minimum divider value , resulting
* in the shortest possible duration for one " count "
* and " period == duty_cycle " . This results in a signal
2019-06-12 21:59:10 +02:00
* which is LOW for one " count " , while being HIGH for
* the rest of the ( so the signal is HIGH for slightly
* less than 100 % of the period , but this is the best
* we can achieve ) .
*/
2023-05-24 21:52:21 +02:00
channel - > rate = ULONG_MAX ;
2019-06-12 21:59:10 +02:00
channel - > hi = ~ 0 ;
channel - > lo = 0 ;
meson_pwm_enable ( meson , pwm ) ;
} else {
meson_pwm_disable ( meson , pwm ) ;
}
2019-06-12 21:59:09 +02:00
} else {
2019-06-12 21:59:03 +02:00
err = meson_pwm_calc ( meson , pwm , state ) ;
2016-08-22 17:36:30 +02:00
if ( err < 0 )
2019-04-01 19:57:48 +02:00
return err ;
2016-08-22 17:36:30 +02:00
2019-06-12 21:58:58 +02:00
meson_pwm_enable ( meson , pwm ) ;
2016-08-22 17:36:30 +02:00
}
2019-04-01 19:57:48 +02:00
return 0 ;
2016-08-22 17:36:30 +02:00
}
2023-05-24 21:52:21 +02:00
static u64 meson_pwm_cnt_to_ns ( struct pwm_chip * chip , struct pwm_device * pwm ,
u32 cnt )
2019-06-12 21:59:08 +02:00
{
struct meson_pwm * meson = to_meson_pwm ( chip ) ;
struct meson_pwm_channel * channel ;
unsigned long fin_freq ;
/* to_meson_pwm() can only be used after .get_state() is called */
channel = & meson - > channels [ pwm - > hwpwm ] ;
fin_freq = clk_get_rate ( channel - > clk ) ;
if ( fin_freq = = 0 )
return 0 ;
2023-05-24 21:52:21 +02:00
return div64_ul ( NSEC_PER_SEC * ( u64 ) cnt , fin_freq ) ;
2019-06-12 21:59:08 +02:00
}
2022-12-02 19:35:26 +01:00
static int meson_pwm_get_state ( struct pwm_chip * chip , struct pwm_device * pwm ,
struct pwm_state * state )
2016-08-22 17:36:30 +02:00
{
struct meson_pwm * meson = to_meson_pwm ( chip ) ;
2019-06-12 21:59:08 +02:00
struct meson_pwm_channel_data * channel_data ;
struct meson_pwm_channel * channel ;
2023-05-24 21:52:21 +02:00
u32 value ;
2016-08-22 17:36:30 +02:00
if ( ! state )
2022-12-02 19:35:26 +01:00
return 0 ;
2016-08-22 17:36:30 +02:00
2019-06-12 21:59:08 +02:00
channel = & meson - > channels [ pwm - > hwpwm ] ;
channel_data = & meson_pwm_per_channel_data [ pwm - > hwpwm ] ;
2016-08-22 17:36:30 +02:00
value = readl ( meson - > base + REG_MISC_AB ) ;
2023-05-24 21:52:21 +02:00
state - > enabled = value & channel_data - > pwm_en_mask ;
2019-06-12 21:59:08 +02:00
value = readl ( meson - > base + channel_data - > reg_offset ) ;
channel - > lo = FIELD_GET ( PWM_LOW_MASK , value ) ;
channel - > hi = FIELD_GET ( PWM_HIGH_MASK , value ) ;
2023-05-24 21:47:43 +02:00
state - > period = meson_pwm_cnt_to_ns ( chip , pwm , channel - > lo + channel - > hi ) ;
state - > duty_cycle = meson_pwm_cnt_to_ns ( chip , pwm , channel - > hi ) ;
2022-12-02 19:35:26 +01:00
2023-03-22 22:45:44 +01:00
state - > polarity = PWM_POLARITY_NORMAL ;
2022-12-02 19:35:26 +01:00
return 0 ;
2016-08-22 17:36:30 +02:00
}
static const struct pwm_ops meson_pwm_ops = {
. request = meson_pwm_request ,
. free = meson_pwm_free ,
. apply = meson_pwm_apply ,
. get_state = meson_pwm_get_state ,
. owner = THIS_MODULE ,
} ;
static const char * const pwm_meson8b_parent_names [ ] = {
2023-05-24 21:51:31 +02:00
" xtal " , NULL , " fclk_div4 " , " fclk_div3 "
2016-08-22 17:36:30 +02:00
} ;
static const struct meson_pwm_data pwm_meson8b_data = {
. parent_names = pwm_meson8b_parent_names ,
2017-06-08 14:24:15 +02:00
. num_parents = ARRAY_SIZE ( pwm_meson8b_parent_names ) ,
2016-08-22 17:36:30 +02:00
} ;
2017-06-08 14:24:15 +02:00
/*
* Only the 2 first inputs of the GXBB AO PWMs are valid
* The last 2 are grounded
*/
static const char * const pwm_gxbb_ao_parent_names [ ] = {
" xtal " , " clk81 "
} ;
static const struct meson_pwm_data pwm_gxbb_ao_data = {
. parent_names = pwm_gxbb_ao_parent_names ,
. num_parents = ARRAY_SIZE ( pwm_gxbb_ao_parent_names ) ,
2016-08-22 17:36:30 +02:00
} ;
2017-12-04 14:00:17 +08:00
static const char * const pwm_axg_ee_parent_names [ ] = {
" xtal " , " fclk_div5 " , " fclk_div4 " , " fclk_div3 "
} ;
static const struct meson_pwm_data pwm_axg_ee_data = {
. parent_names = pwm_axg_ee_parent_names ,
. num_parents = ARRAY_SIZE ( pwm_axg_ee_parent_names ) ,
} ;
static const char * const pwm_axg_ao_parent_names [ ] = {
2023-04-09 17:15:52 +02:00
" xtal " , " axg_ao_clk81 " , " fclk_div4 " , " fclk_div5 "
2017-12-04 14:00:17 +08:00
} ;
static const struct meson_pwm_data pwm_axg_ao_data = {
. parent_names = pwm_axg_ao_parent_names ,
. num_parents = ARRAY_SIZE ( pwm_axg_ao_parent_names ) ,
} ;
2019-06-20 16:46:55 +02:00
static const char * const pwm_g12a_ao_ab_parent_names [ ] = {
2023-04-11 07:34:11 +02:00
" xtal " , " g12a_ao_clk81 " , " fclk_div4 " , " fclk_div5 "
2019-06-20 16:46:55 +02:00
} ;
static const struct meson_pwm_data pwm_g12a_ao_ab_data = {
. parent_names = pwm_g12a_ao_ab_parent_names ,
. num_parents = ARRAY_SIZE ( pwm_g12a_ao_ab_parent_names ) ,
} ;
2019-04-23 15:36:45 +02:00
static const char * const pwm_g12a_ao_cd_parent_names [ ] = {
2023-04-11 07:34:11 +02:00
" xtal " , " g12a_ao_clk81 " ,
2019-04-23 15:36:45 +02:00
} ;
static const struct meson_pwm_data pwm_g12a_ao_cd_data = {
. parent_names = pwm_g12a_ao_cd_parent_names ,
. num_parents = ARRAY_SIZE ( pwm_g12a_ao_cd_parent_names ) ,
} ;
2016-08-22 17:36:30 +02:00
static const struct of_device_id meson_pwm_matches [ ] = {
2017-06-08 14:24:15 +02:00
{
. compatible = " amlogic,meson8b-pwm " ,
. data = & pwm_meson8b_data
} ,
{
. compatible = " amlogic,meson-gxbb-pwm " ,
2023-05-24 21:51:31 +02:00
. data = & pwm_meson8b_data
2017-06-08 14:24:15 +02:00
} ,
{
. compatible = " amlogic,meson-gxbb-ao-pwm " ,
. data = & pwm_gxbb_ao_data
} ,
2017-12-04 14:00:17 +08:00
{
. compatible = " amlogic,meson-axg-ee-pwm " ,
. data = & pwm_axg_ee_data
} ,
{
. compatible = " amlogic,meson-axg-ao-pwm " ,
. data = & pwm_axg_ao_data
} ,
2019-04-23 15:36:45 +02:00
{
. compatible = " amlogic,meson-g12a-ee-pwm " ,
2023-05-24 21:51:31 +02:00
. data = & pwm_meson8b_data
2019-04-23 15:36:45 +02:00
} ,
{
. compatible = " amlogic,meson-g12a-ao-pwm-ab " ,
2019-06-20 16:46:55 +02:00
. data = & pwm_g12a_ao_ab_data
2019-04-23 15:36:45 +02:00
} ,
{
. compatible = " amlogic,meson-g12a-ao-pwm-cd " ,
. data = & pwm_g12a_ao_cd_data
} ,
2016-08-22 17:36:30 +02:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , meson_pwm_matches ) ;
2019-06-12 21:59:04 +02:00
static int meson_pwm_init_channels ( struct meson_pwm * meson )
2016-08-22 17:36:30 +02:00
{
2023-05-24 21:50:47 +02:00
struct clk_parent_data mux_parent_data [ MESON_MAX_MUX_PARENTS ] = { } ;
2016-08-22 17:36:30 +02:00
struct device * dev = meson - > chip . dev ;
unsigned int i ;
char name [ 255 ] ;
int err ;
2023-05-24 21:50:47 +02:00
for ( i = 0 ; i < meson - > data - > num_parents ; i + + ) {
mux_parent_data [ i ] . index = - 1 ;
mux_parent_data [ i ] . name = meson - > data - > parent_names [ i ] ;
}
2016-08-22 17:36:30 +02:00
for ( i = 0 ; i < meson - > chip . npwm ; i + + ) {
2019-06-12 21:59:04 +02:00
struct meson_pwm_channel * channel = & meson - > channels [ i ] ;
2023-05-24 21:52:21 +02:00
struct clk_parent_data div_parent = { } , gate_parent = { } ;
2023-05-24 21:50:47 +02:00
struct clk_init_data init = { } ;
2016-08-22 17:36:30 +02:00
2018-08-01 12:57:20 +02:00
snprintf ( name , sizeof ( name ) , " %s#mux%u " , dev_name ( dev ) , i ) ;
2016-08-22 17:36:30 +02:00
init . name = name ;
init . ops = & clk_mux_ops ;
2019-04-25 10:57:37 -07:00
init . flags = 0 ;
2023-05-24 21:50:47 +02:00
init . parent_data = mux_parent_data ;
2017-06-08 14:24:15 +02:00
init . num_parents = meson - > data - > num_parents ;
2016-08-22 17:36:30 +02:00
channel - > mux . reg = meson - > base + REG_MISC_AB ;
2019-06-12 21:59:05 +02:00
channel - > mux . shift =
meson_pwm_per_channel_data [ i ] . clk_sel_shift ;
2019-06-12 21:59:01 +02:00
channel - > mux . mask = MISC_CLK_SEL_MASK ;
2016-08-22 17:36:30 +02:00
channel - > mux . flags = 0 ;
channel - > mux . lock = & meson - > lock ;
channel - > mux . table = NULL ;
channel - > mux . hw . init = & init ;
2023-05-24 21:52:21 +02:00
err = devm_clk_hw_register ( dev , & channel - > mux . hw ) ;
if ( err ) {
dev_err ( dev , " failed to register %s: %d \n " , name , err ) ;
return err ;
}
snprintf ( name , sizeof ( name ) , " %s#div%u " , dev_name ( dev ) , i ) ;
init . name = name ;
init . ops = & clk_divider_ops ;
init . flags = CLK_SET_RATE_PARENT ;
div_parent . index = - 1 ;
div_parent . hw = & channel - > mux . hw ;
init . parent_data = & div_parent ;
init . num_parents = 1 ;
channel - > div . reg = meson - > base + REG_MISC_AB ;
channel - > div . shift = meson_pwm_per_channel_data [ i ] . clk_div_shift ;
channel - > div . width = MISC_CLK_DIV_WIDTH ;
channel - > div . hw . init = & init ;
channel - > div . flags = 0 ;
channel - > div . lock = & meson - > lock ;
err = devm_clk_hw_register ( dev , & channel - > div . hw ) ;
if ( err ) {
2016-08-22 17:36:30 +02:00
dev_err ( dev , " failed to register %s: %d \n " , name , err ) ;
return err ;
}
2023-05-24 21:52:21 +02:00
snprintf ( name , sizeof ( name ) , " %s#gate%u " , dev_name ( dev ) , i ) ;
2016-08-22 17:36:30 +02:00
2023-05-24 21:52:21 +02:00
init . name = name ;
init . ops = & clk_gate_ops ;
init . flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED ;
gate_parent . index = - 1 ;
gate_parent . hw = & channel - > div . hw ;
init . parent_data = & gate_parent ;
init . num_parents = 1 ;
channel - > gate . reg = meson - > base + REG_MISC_AB ;
channel - > gate . bit_idx = meson_pwm_per_channel_data [ i ] . clk_en_shift ;
channel - > gate . hw . init = & init ;
channel - > gate . flags = 0 ;
channel - > gate . lock = & meson - > lock ;
err = devm_clk_hw_register ( dev , & channel - > gate . hw ) ;
if ( err ) {
dev_err ( dev , " failed to register %s: %d \n " , name , err ) ;
return err ;
}
channel - > clk = devm_clk_hw_get_clk ( dev , & channel - > gate . hw , NULL ) ;
if ( IS_ERR ( channel - > clk ) ) {
err = PTR_ERR ( channel - > clk ) ;
dev_err ( dev , " failed to register %s: %d \n " , name , err ) ;
return err ;
}
2016-08-22 17:36:30 +02:00
}
return 0 ;
}
static int meson_pwm_probe ( struct platform_device * pdev )
{
struct meson_pwm * meson ;
int err ;
meson = devm_kzalloc ( & pdev - > dev , sizeof ( * meson ) , GFP_KERNEL ) ;
if ( ! meson )
return - ENOMEM ;
2019-12-29 08:05:52 +00:00
meson - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2016-08-22 17:36:30 +02:00
if ( IS_ERR ( meson - > base ) )
return PTR_ERR ( meson - > base ) ;
2016-09-10 09:55:49 +08:00
spin_lock_init ( & meson - > lock ) ;
2016-08-22 17:36:30 +02:00
meson - > chip . dev = & pdev - > dev ;
meson - > chip . ops = & meson_pwm_ops ;
2019-06-12 21:59:04 +02:00
meson - > chip . npwm = MESON_NUM_PWMS ;
2016-08-22 17:36:30 +02:00
meson - > data = of_device_get_match_data ( & pdev - > dev ) ;
2019-06-12 21:59:04 +02:00
err = meson_pwm_init_channels ( meson ) ;
2016-08-22 17:36:30 +02:00
if ( err < 0 )
return err ;
2021-05-10 21:09:25 +02:00
err = devm_pwmchip_add ( & pdev - > dev , & meson - > chip ) ;
2016-08-22 17:36:30 +02:00
if ( err < 0 ) {
dev_err ( & pdev - > dev , " failed to register PWM chip: %d \n " , err ) ;
return err ;
}
return 0 ;
}
static struct platform_driver meson_pwm_driver = {
. driver = {
. name = " meson-pwm " ,
. of_match_table = meson_pwm_matches ,
} ,
. probe = meson_pwm_probe ,
} ;
module_platform_driver ( meson_pwm_driver ) ;
MODULE_DESCRIPTION ( " Amlogic Meson PWM Generator driver " ) ;
MODULE_AUTHOR ( " Neil Armstrong <narmstrong@baylibre.com> " ) ;
MODULE_LICENSE ( " Dual BSD/GPL " ) ;