2019-05-28 09:57:07 -07:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2013-04-06 02:40:36 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2007 Ben Dooks
|
|
|
|
* Copyright (c) 2008 Simtec Electronics
|
|
|
|
* Ben Dooks <ben@simtec.co.uk>, <ben-linux@fluff.org>
|
|
|
|
* Copyright (c) 2013 Tomasz Figa <tomasz.figa@gmail.com>
|
2017-04-24 12:01:08 +02:00
|
|
|
* Copyright (c) 2017 Samsung Electronics Co., Ltd.
|
2013-04-06 02:40:36 +02:00
|
|
|
*
|
|
|
|
* PWM driver for Samsung SoCs
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/bitops.h>
|
|
|
|
#include <linux/clk.h>
|
|
|
|
#include <linux/export.h>
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
2013-09-27 16:53:24 +05:30
|
|
|
#include <linux/of.h>
|
2013-04-06 02:40:36 +02:00
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/pwm.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/time.h>
|
|
|
|
|
|
|
|
/* For struct samsung_timer_variant and samsung_pwm_lock. */
|
|
|
|
#include <clocksource/samsung_pwm.h>
|
|
|
|
|
|
|
|
#define REG_TCFG0 0x00
|
|
|
|
#define REG_TCFG1 0x04
|
|
|
|
#define REG_TCON 0x08
|
|
|
|
|
|
|
|
#define REG_TCNTB(chan) (0x0c + ((chan) * 0xc))
|
|
|
|
#define REG_TCMPB(chan) (0x10 + ((chan) * 0xc))
|
|
|
|
|
|
|
|
#define TCFG0_PRESCALER_MASK 0xff
|
|
|
|
#define TCFG0_PRESCALER1_SHIFT 8
|
|
|
|
|
|
|
|
#define TCFG1_MUX_MASK 0xf
|
|
|
|
#define TCFG1_SHIFT(chan) (4 * (chan))
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Each channel occupies 4 bits in TCON register, but there is a gap of 4
|
|
|
|
* bits (one channel) after channel 0, so channels have different numbering
|
|
|
|
* when accessing TCON register. See to_tcon_channel() function.
|
|
|
|
*
|
|
|
|
* In addition, the location of autoreload bit for channel 4 (TCON channel 5)
|
|
|
|
* in its set of bits is 2 as opposed to 3 for other channels.
|
|
|
|
*/
|
|
|
|
#define TCON_START(chan) BIT(4 * (chan) + 0)
|
|
|
|
#define TCON_MANUALUPDATE(chan) BIT(4 * (chan) + 1)
|
|
|
|
#define TCON_INVERT(chan) BIT(4 * (chan) + 2)
|
|
|
|
#define _TCON_AUTORELOAD(chan) BIT(4 * (chan) + 3)
|
|
|
|
#define _TCON_AUTORELOAD4(chan) BIT(4 * (chan) + 2)
|
|
|
|
#define TCON_AUTORELOAD(chan) \
|
|
|
|
((chan < 5) ? _TCON_AUTORELOAD(chan) : _TCON_AUTORELOAD4(chan))
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct samsung_pwm_channel - private data of PWM channel
|
|
|
|
* @period_ns: current period in nanoseconds programmed to the hardware
|
|
|
|
* @duty_ns: current duty time in nanoseconds programmed to the hardware
|
|
|
|
* @tin_ns: time of one timer tick in nanoseconds with current timer rate
|
|
|
|
*/
|
|
|
|
struct samsung_pwm_channel {
|
|
|
|
u32 period_ns;
|
|
|
|
u32 duty_ns;
|
|
|
|
u32 tin_ns;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct samsung_pwm_chip - private data of PWM chip
|
|
|
|
* @variant: local copy of hardware variant data
|
|
|
|
* @inverter_mask: inverter status for all channels - one bit per channel
|
2017-04-24 12:01:08 +02:00
|
|
|
* @disabled_mask: disabled status for all channels - one bit per channel
|
2013-04-06 02:40:36 +02:00
|
|
|
* @base: base address of mapped PWM registers
|
|
|
|
* @base_clk: base clock used to drive the timers
|
|
|
|
* @tclk0: external clock 0 (can be ERR_PTR if not present)
|
|
|
|
* @tclk1: external clock 1 (can be ERR_PTR if not present)
|
2023-10-12 23:02:29 +02:00
|
|
|
* @channel: per channel driver data
|
2013-04-06 02:40:36 +02:00
|
|
|
*/
|
|
|
|
struct samsung_pwm_chip {
|
|
|
|
struct samsung_pwm_variant variant;
|
|
|
|
u8 inverter_mask;
|
2017-04-24 12:01:08 +02:00
|
|
|
u8 disabled_mask;
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
void __iomem *base;
|
|
|
|
struct clk *base_clk;
|
|
|
|
struct clk *tclk0;
|
|
|
|
struct clk *tclk1;
|
2023-07-05 10:06:44 +02:00
|
|
|
struct samsung_pwm_channel channel[SAMSUNG_PWM_NUM];
|
2013-04-06 02:40:36 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifndef CONFIG_CLKSRC_SAMSUNG_PWM
|
|
|
|
/*
|
|
|
|
* PWM block is shared between pwm-samsung and samsung_pwm_timer drivers
|
|
|
|
* and some registers need access synchronization. If both drivers are
|
|
|
|
* compiled in, the spinlock is defined in the clocksource driver,
|
|
|
|
* otherwise following definition is used.
|
|
|
|
*
|
|
|
|
* Currently we do not need any more complex synchronization method
|
|
|
|
* because all the supported SoCs contain only one instance of the PWM
|
|
|
|
* IP. Should this change, both drivers will need to be modified to
|
|
|
|
* properly synchronize accesses to particular instances.
|
|
|
|
*/
|
|
|
|
static DEFINE_SPINLOCK(samsung_pwm_lock);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static inline
|
|
|
|
struct samsung_pwm_chip *to_samsung_pwm_chip(struct pwm_chip *chip)
|
|
|
|
{
|
2024-02-14 10:32:28 +01:00
|
|
|
return pwmchip_get_drvdata(chip);
|
2013-04-06 02:40:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline unsigned int to_tcon_channel(unsigned int channel)
|
|
|
|
{
|
|
|
|
/* TCON register has a gap of 4 bits (1 channel) after channel 0 */
|
|
|
|
return (channel == 0) ? 0 : (channel + 1);
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
static void __pwm_samsung_manual_update(struct samsung_pwm_chip *our_chip,
|
2021-09-09 12:10:42 +02:00
|
|
|
struct pwm_device *pwm)
|
|
|
|
{
|
|
|
|
unsigned int tcon_chan = to_tcon_channel(pwm->hwpwm);
|
|
|
|
u32 tcon;
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
tcon = readl(our_chip->base + REG_TCON);
|
2021-09-09 12:10:42 +02:00
|
|
|
tcon |= TCON_MANUALUPDATE(tcon_chan);
|
2023-09-29 18:19:17 +02:00
|
|
|
writel(tcon, our_chip->base + REG_TCON);
|
2021-09-09 12:10:42 +02:00
|
|
|
|
|
|
|
tcon &= ~TCON_MANUALUPDATE(tcon_chan);
|
2023-09-29 18:19:17 +02:00
|
|
|
writel(tcon, our_chip->base + REG_TCON);
|
2021-09-09 12:10:42 +02:00
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
static void pwm_samsung_set_divisor(struct samsung_pwm_chip *our_chip,
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned int channel, u8 divisor)
|
|
|
|
{
|
|
|
|
u8 shift = TCFG1_SHIFT(channel);
|
|
|
|
unsigned long flags;
|
|
|
|
u32 reg;
|
|
|
|
u8 bits;
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
bits = (fls(divisor) - 1) - our_chip->variant.div_base;
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
spin_lock_irqsave(&samsung_pwm_lock, flags);
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
reg = readl(our_chip->base + REG_TCFG1);
|
2013-04-06 02:40:36 +02:00
|
|
|
reg &= ~(TCFG1_MUX_MASK << shift);
|
|
|
|
reg |= bits << shift;
|
2023-09-29 18:19:17 +02:00
|
|
|
writel(reg, our_chip->base + REG_TCFG1);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&samsung_pwm_lock, flags);
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
static int pwm_samsung_is_tdiv(struct samsung_pwm_chip *our_chip, unsigned int chan)
|
2013-04-06 02:40:36 +02:00
|
|
|
{
|
2023-09-29 18:19:17 +02:00
|
|
|
struct samsung_pwm_variant *variant = &our_chip->variant;
|
2013-04-06 02:40:36 +02:00
|
|
|
u32 reg;
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
reg = readl(our_chip->base + REG_TCFG1);
|
2013-04-06 02:40:36 +02:00
|
|
|
reg >>= TCFG1_SHIFT(chan);
|
|
|
|
reg &= TCFG1_MUX_MASK;
|
|
|
|
|
|
|
|
return (BIT(reg) & variant->tclk_mask) == 0;
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
static unsigned long pwm_samsung_get_tin_rate(struct samsung_pwm_chip *our_chip,
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned int chan)
|
|
|
|
{
|
|
|
|
unsigned long rate;
|
|
|
|
u32 reg;
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
rate = clk_get_rate(our_chip->base_clk);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
reg = readl(our_chip->base + REG_TCFG0);
|
2013-04-06 02:40:36 +02:00
|
|
|
if (chan >= 2)
|
|
|
|
reg >>= TCFG0_PRESCALER1_SHIFT;
|
|
|
|
reg &= TCFG0_PRESCALER_MASK;
|
|
|
|
|
|
|
|
return rate / (reg + 1);
|
|
|
|
}
|
|
|
|
|
2024-02-14 10:32:24 +01:00
|
|
|
static unsigned long pwm_samsung_calc_tin(struct pwm_chip *chip,
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned int chan, unsigned long freq)
|
|
|
|
{
|
2024-02-14 10:32:24 +01:00
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
2023-09-29 18:19:17 +02:00
|
|
|
struct samsung_pwm_variant *variant = &our_chip->variant;
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned long rate;
|
|
|
|
struct clk *clk;
|
|
|
|
u8 div;
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
if (!pwm_samsung_is_tdiv(our_chip, chan)) {
|
|
|
|
clk = (chan < 2) ? our_chip->tclk0 : our_chip->tclk1;
|
2013-04-06 02:40:36 +02:00
|
|
|
if (!IS_ERR(clk)) {
|
|
|
|
rate = clk_get_rate(clk);
|
|
|
|
if (rate)
|
|
|
|
return rate;
|
|
|
|
}
|
|
|
|
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_warn(pwmchip_parent(chip),
|
2013-04-06 02:40:36 +02:00
|
|
|
"tclk of PWM %d is inoperational, using tdiv\n", chan);
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
rate = pwm_samsung_get_tin_rate(our_chip, chan);
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_dbg(pwmchip_parent(chip), "tin parent at %lu\n", rate);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Compare minimum PWM frequency that can be achieved with possible
|
|
|
|
* divider settings and choose the lowest divisor that can generate
|
|
|
|
* frequencies lower than requested.
|
|
|
|
*/
|
2016-08-16 23:22:01 +09:00
|
|
|
if (variant->bits < 32) {
|
|
|
|
/* Only for s3c24xx */
|
|
|
|
for (div = variant->div_base; div < 4; ++div)
|
|
|
|
if ((rate >> (variant->bits + div)) < freq)
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* Other variants have enough counter bits to generate any
|
|
|
|
* requested rate, so no need to check higher divisors.
|
|
|
|
*/
|
|
|
|
div = variant->div_base;
|
|
|
|
}
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
pwm_samsung_set_divisor(our_chip, chan, BIT(div));
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
return rate >> div;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pwm_samsung_request(struct pwm_chip *chip, struct pwm_device *pwm)
|
|
|
|
{
|
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
|
|
|
|
|
|
|
if (!(our_chip->variant.output_mask & BIT(pwm->hwpwm))) {
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_warn(pwmchip_parent(chip),
|
2013-04-06 02:40:36 +02:00
|
|
|
"tried to request PWM channel %d without output\n",
|
|
|
|
pwm->hwpwm);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2023-07-05 10:06:44 +02:00
|
|
|
memset(&our_chip->channel[pwm->hwpwm], 0, sizeof(our_chip->channel[pwm->hwpwm]));
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pwm_samsung_enable(struct pwm_chip *chip, struct pwm_device *pwm)
|
|
|
|
{
|
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
|
|
|
unsigned int tcon_chan = to_tcon_channel(pwm->hwpwm);
|
|
|
|
unsigned long flags;
|
|
|
|
u32 tcon;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&samsung_pwm_lock, flags);
|
|
|
|
|
|
|
|
tcon = readl(our_chip->base + REG_TCON);
|
|
|
|
|
|
|
|
tcon &= ~TCON_START(tcon_chan);
|
|
|
|
tcon |= TCON_MANUALUPDATE(tcon_chan);
|
|
|
|
writel(tcon, our_chip->base + REG_TCON);
|
|
|
|
|
|
|
|
tcon &= ~TCON_MANUALUPDATE(tcon_chan);
|
|
|
|
tcon |= TCON_START(tcon_chan) | TCON_AUTORELOAD(tcon_chan);
|
|
|
|
writel(tcon, our_chip->base + REG_TCON);
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
our_chip->disabled_mask &= ~BIT(pwm->hwpwm);
|
|
|
|
|
2013-04-06 02:40:36 +02:00
|
|
|
spin_unlock_irqrestore(&samsung_pwm_lock, flags);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pwm_samsung_disable(struct pwm_chip *chip, struct pwm_device *pwm)
|
|
|
|
{
|
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
|
|
|
unsigned int tcon_chan = to_tcon_channel(pwm->hwpwm);
|
|
|
|
unsigned long flags;
|
|
|
|
u32 tcon;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&samsung_pwm_lock, flags);
|
|
|
|
|
|
|
|
tcon = readl(our_chip->base + REG_TCON);
|
|
|
|
tcon &= ~TCON_AUTORELOAD(tcon_chan);
|
|
|
|
writel(tcon, our_chip->base + REG_TCON);
|
|
|
|
|
2021-09-09 12:10:42 +02:00
|
|
|
/*
|
|
|
|
* In case the PWM is at 100% duty cycle, force a manual
|
|
|
|
* update to prevent the signal from staying high.
|
|
|
|
*/
|
|
|
|
if (readl(our_chip->base + REG_TCMPB(pwm->hwpwm)) == (u32)-1U)
|
|
|
|
__pwm_samsung_manual_update(our_chip, pwm);
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
our_chip->disabled_mask |= BIT(pwm->hwpwm);
|
|
|
|
|
2013-04-06 02:40:36 +02:00
|
|
|
spin_unlock_irqrestore(&samsung_pwm_lock, flags);
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
static void pwm_samsung_manual_update(struct samsung_pwm_chip *our_chip,
|
pwm: samsung: Fix output race on disabling
When disabling the Samsung PWM the output state remains at the level it
was at the end of a PWM cycle. In other words, calling pwm_disable()
when at 100% duty cycle will keep the output active, while at all other
settings the output will go/stay inactive. On top of that the Samsung
PWM settings are double-buffered, which means the new settings only get
applied at the start of a new PWM cycle.
This results in a race if the PWM is at 100% duty cycle and a driver
calls:
pwm_config(pwm, 0, period);
pwm_disable(pwm);
In this case the PWMs output will unexpectedly stay active, unless a new
PWM cycle happened to start between the register writes in pwm_config()
and pwm_disable(). As far as I can tell this is a regression introduced
by 3bdf878, before that a call to pwm_config() would call
pwm_samsung_enable() which, while heavy-handed, made sure the expected
settings were live.
To resolve this, while not re-introducing the issues 3bdf878 (flickering
as the PWM got reset while in a PWM cycle) fixed, only force an update
of the settings when at 100% duty cycle, which shouldn't have any
noticeable effect on the output but is enough to ensure the behaviour is
as expected on disable.
Signed-off-by: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
Reviewed-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Acked-by: Lukasz Majewski <l.majewski@samsung.com>
Signed-off-by: Thierry Reding <thierry.reding@gmail.com>
2015-03-05 09:14:03 +01:00
|
|
|
struct pwm_device *pwm)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&samsung_pwm_lock, flags);
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
__pwm_samsung_manual_update(our_chip, pwm);
|
pwm: samsung: Fix output race on disabling
When disabling the Samsung PWM the output state remains at the level it
was at the end of a PWM cycle. In other words, calling pwm_disable()
when at 100% duty cycle will keep the output active, while at all other
settings the output will go/stay inactive. On top of that the Samsung
PWM settings are double-buffered, which means the new settings only get
applied at the start of a new PWM cycle.
This results in a race if the PWM is at 100% duty cycle and a driver
calls:
pwm_config(pwm, 0, period);
pwm_disable(pwm);
In this case the PWMs output will unexpectedly stay active, unless a new
PWM cycle happened to start between the register writes in pwm_config()
and pwm_disable(). As far as I can tell this is a regression introduced
by 3bdf878, before that a call to pwm_config() would call
pwm_samsung_enable() which, while heavy-handed, made sure the expected
settings were live.
To resolve this, while not re-introducing the issues 3bdf878 (flickering
as the PWM got reset while in a PWM cycle) fixed, only force an update
of the settings when at 100% duty cycle, which shouldn't have any
noticeable effect on the output but is enough to ensure the behaviour is
as expected on disable.
Signed-off-by: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
Reviewed-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Acked-by: Lukasz Majewski <l.majewski@samsung.com>
Signed-off-by: Thierry Reding <thierry.reding@gmail.com>
2015-03-05 09:14:03 +01:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&samsung_pwm_lock, flags);
|
|
|
|
}
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
static int __pwm_samsung_config(struct pwm_chip *chip, struct pwm_device *pwm,
|
|
|
|
int duty_ns, int period_ns, bool force_period)
|
2013-04-06 02:40:36 +02:00
|
|
|
{
|
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
2023-07-05 10:06:44 +02:00
|
|
|
struct samsung_pwm_channel *chan = &our_chip->channel[pwm->hwpwm];
|
pwm: samsung: Fix output race on disabling
When disabling the Samsung PWM the output state remains at the level it
was at the end of a PWM cycle. In other words, calling pwm_disable()
when at 100% duty cycle will keep the output active, while at all other
settings the output will go/stay inactive. On top of that the Samsung
PWM settings are double-buffered, which means the new settings only get
applied at the start of a new PWM cycle.
This results in a race if the PWM is at 100% duty cycle and a driver
calls:
pwm_config(pwm, 0, period);
pwm_disable(pwm);
In this case the PWMs output will unexpectedly stay active, unless a new
PWM cycle happened to start between the register writes in pwm_config()
and pwm_disable(). As far as I can tell this is a regression introduced
by 3bdf878, before that a call to pwm_config() would call
pwm_samsung_enable() which, while heavy-handed, made sure the expected
settings were live.
To resolve this, while not re-introducing the issues 3bdf878 (flickering
as the PWM got reset while in a PWM cycle) fixed, only force an update
of the settings when at 100% duty cycle, which shouldn't have any
noticeable effect on the output but is enough to ensure the behaviour is
as expected on disable.
Signed-off-by: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
Reviewed-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Acked-by: Lukasz Majewski <l.majewski@samsung.com>
Signed-off-by: Thierry Reding <thierry.reding@gmail.com>
2015-03-05 09:14:03 +01:00
|
|
|
u32 tin_ns = chan->tin_ns, tcnt, tcmp, oldtcmp;
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
tcnt = readl(our_chip->base + REG_TCNTB(pwm->hwpwm));
|
pwm: samsung: Fix output race on disabling
When disabling the Samsung PWM the output state remains at the level it
was at the end of a PWM cycle. In other words, calling pwm_disable()
when at 100% duty cycle will keep the output active, while at all other
settings the output will go/stay inactive. On top of that the Samsung
PWM settings are double-buffered, which means the new settings only get
applied at the start of a new PWM cycle.
This results in a race if the PWM is at 100% duty cycle and a driver
calls:
pwm_config(pwm, 0, period);
pwm_disable(pwm);
In this case the PWMs output will unexpectedly stay active, unless a new
PWM cycle happened to start between the register writes in pwm_config()
and pwm_disable(). As far as I can tell this is a regression introduced
by 3bdf878, before that a call to pwm_config() would call
pwm_samsung_enable() which, while heavy-handed, made sure the expected
settings were live.
To resolve this, while not re-introducing the issues 3bdf878 (flickering
as the PWM got reset while in a PWM cycle) fixed, only force an update
of the settings when at 100% duty cycle, which shouldn't have any
noticeable effect on the output but is enough to ensure the behaviour is
as expected on disable.
Signed-off-by: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
Reviewed-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Acked-by: Lukasz Majewski <l.majewski@samsung.com>
Signed-off-by: Thierry Reding <thierry.reding@gmail.com>
2015-03-05 09:14:03 +01:00
|
|
|
oldtcmp = readl(our_chip->base + REG_TCMPB(pwm->hwpwm));
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
/* We need tick count for calculation, not last tick. */
|
|
|
|
++tcnt;
|
|
|
|
|
|
|
|
/* Check to see if we are changing the clock rate of the PWM. */
|
2017-04-24 12:01:08 +02:00
|
|
|
if (chan->period_ns != period_ns || force_period) {
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned long tin_rate;
|
|
|
|
u32 period;
|
|
|
|
|
|
|
|
period = NSEC_PER_SEC / period_ns;
|
|
|
|
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_dbg(pwmchip_parent(chip), "duty_ns=%d, period_ns=%d (%u)\n",
|
2013-04-06 02:40:36 +02:00
|
|
|
duty_ns, period_ns, period);
|
|
|
|
|
2024-02-14 10:32:24 +01:00
|
|
|
tin_rate = pwm_samsung_calc_tin(chip, pwm->hwpwm, period);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_dbg(pwmchip_parent(chip), "tin_rate=%lu\n", tin_rate);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
tin_ns = NSEC_PER_SEC / tin_rate;
|
|
|
|
tcnt = period_ns / tin_ns;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Period is too short. */
|
|
|
|
if (tcnt <= 1)
|
|
|
|
return -ERANGE;
|
|
|
|
|
|
|
|
/* Note that counters count down. */
|
|
|
|
tcmp = duty_ns / tin_ns;
|
|
|
|
|
|
|
|
/* 0% duty is not available */
|
|
|
|
if (!tcmp)
|
|
|
|
++tcmp;
|
|
|
|
|
|
|
|
tcmp = tcnt - tcmp;
|
|
|
|
|
|
|
|
/* Decrement to get tick numbers, instead of tick counts. */
|
|
|
|
--tcnt;
|
|
|
|
/* -1UL will give 100% duty. */
|
|
|
|
--tcmp;
|
|
|
|
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_dbg(pwmchip_parent(chip), "tin_ns=%u, tcmp=%u/%u\n", tin_ns, tcmp, tcnt);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
/* Update PWM registers. */
|
|
|
|
writel(tcnt, our_chip->base + REG_TCNTB(pwm->hwpwm));
|
|
|
|
writel(tcmp, our_chip->base + REG_TCMPB(pwm->hwpwm));
|
|
|
|
|
pwm: samsung: Fix output race on disabling
When disabling the Samsung PWM the output state remains at the level it
was at the end of a PWM cycle. In other words, calling pwm_disable()
when at 100% duty cycle will keep the output active, while at all other
settings the output will go/stay inactive. On top of that the Samsung
PWM settings are double-buffered, which means the new settings only get
applied at the start of a new PWM cycle.
This results in a race if the PWM is at 100% duty cycle and a driver
calls:
pwm_config(pwm, 0, period);
pwm_disable(pwm);
In this case the PWMs output will unexpectedly stay active, unless a new
PWM cycle happened to start between the register writes in pwm_config()
and pwm_disable(). As far as I can tell this is a regression introduced
by 3bdf878, before that a call to pwm_config() would call
pwm_samsung_enable() which, while heavy-handed, made sure the expected
settings were live.
To resolve this, while not re-introducing the issues 3bdf878 (flickering
as the PWM got reset while in a PWM cycle) fixed, only force an update
of the settings when at 100% duty cycle, which shouldn't have any
noticeable effect on the output but is enough to ensure the behaviour is
as expected on disable.
Signed-off-by: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
Reviewed-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Acked-by: Lukasz Majewski <l.majewski@samsung.com>
Signed-off-by: Thierry Reding <thierry.reding@gmail.com>
2015-03-05 09:14:03 +01:00
|
|
|
/*
|
|
|
|
* In case the PWM is currently at 100% duty cycle, force a manual
|
|
|
|
* update to prevent the signal staying high if the PWM is disabled
|
|
|
|
* shortly afer this update (before it autoreloaded the new values).
|
|
|
|
*/
|
|
|
|
if (oldtcmp == (u32) -1) {
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_dbg(pwmchip_parent(chip), "Forcing manual update");
|
pwm: samsung: Fix output race on disabling
When disabling the Samsung PWM the output state remains at the level it
was at the end of a PWM cycle. In other words, calling pwm_disable()
when at 100% duty cycle will keep the output active, while at all other
settings the output will go/stay inactive. On top of that the Samsung
PWM settings are double-buffered, which means the new settings only get
applied at the start of a new PWM cycle.
This results in a race if the PWM is at 100% duty cycle and a driver
calls:
pwm_config(pwm, 0, period);
pwm_disable(pwm);
In this case the PWMs output will unexpectedly stay active, unless a new
PWM cycle happened to start between the register writes in pwm_config()
and pwm_disable(). As far as I can tell this is a regression introduced
by 3bdf878, before that a call to pwm_config() would call
pwm_samsung_enable() which, while heavy-handed, made sure the expected
settings were live.
To resolve this, while not re-introducing the issues 3bdf878 (flickering
as the PWM got reset while in a PWM cycle) fixed, only force an update
of the settings when at 100% duty cycle, which shouldn't have any
noticeable effect on the output but is enough to ensure the behaviour is
as expected on disable.
Signed-off-by: Sjoerd Simons <sjoerd.simons@collabora.co.uk>
Reviewed-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Acked-by: Lukasz Majewski <l.majewski@samsung.com>
Signed-off-by: Thierry Reding <thierry.reding@gmail.com>
2015-03-05 09:14:03 +01:00
|
|
|
pwm_samsung_manual_update(our_chip, pwm);
|
|
|
|
}
|
|
|
|
|
2013-04-06 02:40:36 +02:00
|
|
|
chan->period_ns = period_ns;
|
|
|
|
chan->tin_ns = tin_ns;
|
|
|
|
chan->duty_ns = duty_ns;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
static int pwm_samsung_config(struct pwm_chip *chip, struct pwm_device *pwm,
|
|
|
|
int duty_ns, int period_ns)
|
|
|
|
{
|
|
|
|
return __pwm_samsung_config(chip, pwm, duty_ns, period_ns, false);
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
static void pwm_samsung_set_invert(struct samsung_pwm_chip *our_chip,
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned int channel, bool invert)
|
|
|
|
{
|
|
|
|
unsigned int tcon_chan = to_tcon_channel(channel);
|
|
|
|
unsigned long flags;
|
|
|
|
u32 tcon;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&samsung_pwm_lock, flags);
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
tcon = readl(our_chip->base + REG_TCON);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
if (invert) {
|
2023-09-29 18:19:17 +02:00
|
|
|
our_chip->inverter_mask |= BIT(channel);
|
2013-04-06 02:40:36 +02:00
|
|
|
tcon |= TCON_INVERT(tcon_chan);
|
|
|
|
} else {
|
2023-09-29 18:19:17 +02:00
|
|
|
our_chip->inverter_mask &= ~BIT(channel);
|
2013-04-06 02:40:36 +02:00
|
|
|
tcon &= ~TCON_INVERT(tcon_chan);
|
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
writel(tcon, our_chip->base + REG_TCON);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&samsung_pwm_lock, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pwm_samsung_set_polarity(struct pwm_chip *chip,
|
|
|
|
struct pwm_device *pwm,
|
|
|
|
enum pwm_polarity polarity)
|
|
|
|
{
|
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
|
|
|
bool invert = (polarity == PWM_POLARITY_NORMAL);
|
|
|
|
|
|
|
|
/* Inverted means normal in the hardware. */
|
|
|
|
pwm_samsung_set_invert(our_chip, pwm->hwpwm, invert);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-03-28 09:34:34 +02:00
|
|
|
static int pwm_samsung_apply(struct pwm_chip *chip, struct pwm_device *pwm,
|
|
|
|
const struct pwm_state *state)
|
|
|
|
{
|
|
|
|
int err, enabled = pwm->state.enabled;
|
|
|
|
|
|
|
|
if (state->polarity != pwm->state.polarity) {
|
|
|
|
if (enabled) {
|
|
|
|
pwm_samsung_disable(chip, pwm);
|
|
|
|
enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = pwm_samsung_set_polarity(chip, pwm, state->polarity);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!state->enabled) {
|
|
|
|
if (enabled)
|
|
|
|
pwm_samsung_disable(chip, pwm);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We currently avoid using 64bit arithmetic by using the
|
|
|
|
* fact that anything faster than 1Hz is easily representable
|
|
|
|
* by 32bits.
|
|
|
|
*/
|
|
|
|
if (state->period > NSEC_PER_SEC)
|
|
|
|
return -ERANGE;
|
|
|
|
|
|
|
|
err = pwm_samsung_config(chip, pwm, state->duty_cycle, state->period);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (!pwm->state.enabled)
|
|
|
|
err = pwm_samsung_enable(chip, pwm);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2013-04-06 02:40:36 +02:00
|
|
|
static const struct pwm_ops pwm_samsung_ops = {
|
|
|
|
.request = pwm_samsung_request,
|
2022-03-28 09:34:34 +02:00
|
|
|
.apply = pwm_samsung_apply,
|
2013-04-06 02:40:36 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_OF
|
|
|
|
static const struct samsung_pwm_variant s3c24xx_variant = {
|
|
|
|
.bits = 16,
|
|
|
|
.div_base = 1,
|
|
|
|
.has_tint_cstat = false,
|
|
|
|
.tclk_mask = BIT(4),
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct samsung_pwm_variant s3c64xx_variant = {
|
|
|
|
.bits = 32,
|
|
|
|
.div_base = 0,
|
|
|
|
.has_tint_cstat = true,
|
|
|
|
.tclk_mask = BIT(7) | BIT(6) | BIT(5),
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct samsung_pwm_variant s5p64x0_variant = {
|
|
|
|
.bits = 32,
|
|
|
|
.div_base = 0,
|
|
|
|
.has_tint_cstat = true,
|
|
|
|
.tclk_mask = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct samsung_pwm_variant s5pc100_variant = {
|
|
|
|
.bits = 32,
|
|
|
|
.div_base = 0,
|
|
|
|
.has_tint_cstat = true,
|
|
|
|
.tclk_mask = BIT(5),
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct of_device_id samsung_pwm_matches[] = {
|
|
|
|
{ .compatible = "samsung,s3c2410-pwm", .data = &s3c24xx_variant },
|
|
|
|
{ .compatible = "samsung,s3c6400-pwm", .data = &s3c64xx_variant },
|
|
|
|
{ .compatible = "samsung,s5p6440-pwm", .data = &s5p64x0_variant },
|
|
|
|
{ .compatible = "samsung,s5pc100-pwm", .data = &s5pc100_variant },
|
|
|
|
{ .compatible = "samsung,exynos4210-pwm", .data = &s5p64x0_variant },
|
|
|
|
{},
|
|
|
|
};
|
2015-05-14 02:32:31 +02:00
|
|
|
MODULE_DEVICE_TABLE(of, samsung_pwm_matches);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2024-02-14 10:32:24 +01:00
|
|
|
static int pwm_samsung_parse_dt(struct pwm_chip *chip)
|
2013-04-06 02:40:36 +02:00
|
|
|
{
|
2024-02-14 10:32:24 +01:00
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
2024-02-14 10:32:25 +01:00
|
|
|
struct device_node *np = pwmchip_parent(chip)->of_node;
|
2013-04-06 02:40:36 +02:00
|
|
|
const struct of_device_id *match;
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
match = of_match_node(samsung_pwm_matches, np);
|
|
|
|
if (!match)
|
|
|
|
return -ENODEV;
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
memcpy(&our_chip->variant, match->data, sizeof(our_chip->variant));
|
2013-04-06 02:40:36 +02:00
|
|
|
|
of: remove internal arguments from of_property_for_each_u32()
The of_property_for_each_u32() macro needs five parameters, two of which
are primarily meant as internal variables for the macro itself (in the
for() clause). Yet these two parameters are used by a few drivers, and this
can be considered misuse or at least bad practice.
Now that the kernel uses C11 to build, these two parameters can be avoided
by declaring them internally, thus changing this pattern:
struct property *prop;
const __be32 *p;
u32 val;
of_property_for_each_u32(np, "xyz", prop, p, val) { ... }
to this:
u32 val;
of_property_for_each_u32(np, "xyz", val) { ... }
However two variables cannot be declared in the for clause even with C11,
so declare one struct that contain the two variables we actually need. As
the variables inside this struct are not meant to be used by users of this
macro, give the struct instance the noticeable name "_it" so it is visible
during code reviews, helping to avoid new code to use it directly.
Most usages are trivially converted as they do not use those two
parameters, as expected. The non-trivial cases are:
- drivers/clk/clk.c, of_clk_get_parent_name(): easily doable anyway
- drivers/clk/clk-si5351.c, si5351_dt_parse(): this is more complex as the
checks had to be replicated in a different way, making code more verbose
and somewhat uglier, but I refrained from a full rework to keep as much
of the original code untouched having no hardware to test my changes
All the changes have been build tested. The few for which I have the
hardware have been runtime-tested too.
Reviewed-by: Andre Przywara <andre.przywara@arm.com> # drivers/clk/sunxi/clk-simple-gates.c, drivers/clk/sunxi/clk-sun8i-bus-gates.c
Acked-by: Bartosz Golaszewski <bartosz.golaszewski@linaro.org> # drivers/gpio/gpio-brcmstb.c
Acked-by: Nicolas Ferre <nicolas.ferre@microchip.com> # drivers/irqchip/irq-atmel-aic-common.c
Acked-by: Jonathan Cameron <Jonathan.Cameron@huawei.com> # drivers/iio/adc/ti_am335x_adc.c
Acked-by: Uwe Kleine-König <u.kleine-koenig@baylibre.com> # drivers/pwm/pwm-samsung.c
Acked-by: Richard Leitner <richard.leitner@linux.dev> # drivers/usb/misc/usb251xb.c
Acked-by: Mark Brown <broonie@kernel.org> # sound/soc/codecs/arizona.c
Reviewed-by: Richard Fitzgerald <rf@opensource.cirrus.com> # sound/soc/codecs/arizona.c
Acked-by: Michael Ellerman <mpe@ellerman.id.au> # arch/powerpc/sysdev/xive/spapr.c
Acked-by: Stephen Boyd <sboyd@kernel.org> # clk
Signed-off-by: Luca Ceresoli <luca.ceresoli@bootlin.com>
Acked-by: Lee Jones <lee@kernel.org>
Link: https://lore.kernel.org/r/20240724-of_property_for_each_u32-v3-1-bea82ce429e2@bootlin.com
Signed-off-by: Rob Herring (Arm) <robh@kernel.org>
2024-07-24 18:33:06 +02:00
|
|
|
of_property_for_each_u32(np, "samsung,pwm-outputs", val) {
|
2013-04-06 02:40:36 +02:00
|
|
|
if (val >= SAMSUNG_PWM_NUM) {
|
2024-02-14 10:32:25 +01:00
|
|
|
dev_err(pwmchip_parent(chip),
|
2013-04-06 02:40:36 +02:00
|
|
|
"%s: invalid channel index in samsung,pwm-outputs property\n",
|
|
|
|
__func__);
|
|
|
|
continue;
|
|
|
|
}
|
2023-09-29 18:19:17 +02:00
|
|
|
our_chip->variant.output_mask |= BIT(val);
|
2013-04-06 02:40:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else
|
2024-02-14 10:32:24 +01:00
|
|
|
static int pwm_samsung_parse_dt(struct pwm_chip *chip)
|
2013-04-06 02:40:36 +02:00
|
|
|
{
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int pwm_samsung_probe(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct device *dev = &pdev->dev;
|
2023-09-29 18:19:17 +02:00
|
|
|
struct samsung_pwm_chip *our_chip;
|
2024-02-14 10:32:24 +01:00
|
|
|
struct pwm_chip *chip;
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned int chan;
|
|
|
|
int ret;
|
|
|
|
|
2024-02-14 10:32:28 +01:00
|
|
|
chip = devm_pwmchip_alloc(&pdev->dev, SAMSUNG_PWM_NUM, sizeof(*our_chip));
|
|
|
|
if (IS_ERR(chip))
|
|
|
|
return PTR_ERR(chip);
|
|
|
|
our_chip = to_samsung_pwm_chip(chip);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2024-02-14 10:32:24 +01:00
|
|
|
chip->ops = &pwm_samsung_ops;
|
2023-09-29 18:19:17 +02:00
|
|
|
our_chip->inverter_mask = BIT(SAMSUNG_PWM_NUM) - 1;
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) {
|
2024-02-14 10:32:24 +01:00
|
|
|
ret = pwm_samsung_parse_dt(chip);
|
2013-04-06 02:40:36 +02:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
} else {
|
2024-02-14 10:32:27 +01:00
|
|
|
if (!pdev->dev.platform_data)
|
|
|
|
return dev_err_probe(&pdev->dev, -EINVAL,
|
|
|
|
"no platform data specified\n");
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
memcpy(&our_chip->variant, pdev->dev.platform_data,
|
|
|
|
sizeof(our_chip->variant));
|
2013-04-06 02:40:36 +02:00
|
|
|
}
|
|
|
|
|
2023-09-29 18:19:17 +02:00
|
|
|
our_chip->base = devm_platform_ioremap_resource(pdev, 0);
|
|
|
|
if (IS_ERR(our_chip->base))
|
|
|
|
return PTR_ERR(our_chip->base);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2024-02-14 10:32:26 +01:00
|
|
|
our_chip->base_clk = devm_clk_get_enabled(&pdev->dev, "timers");
|
2024-02-14 10:32:27 +01:00
|
|
|
if (IS_ERR(our_chip->base_clk))
|
|
|
|
return dev_err_probe(dev, PTR_ERR(our_chip->base_clk),
|
|
|
|
"failed to get timer base clk\n");
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
for (chan = 0; chan < SAMSUNG_PWM_NUM; ++chan)
|
2023-09-29 18:19:17 +02:00
|
|
|
if (our_chip->variant.output_mask & BIT(chan))
|
|
|
|
pwm_samsung_set_invert(our_chip, chan, true);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
/* Following clocks are optional. */
|
2023-09-29 18:19:17 +02:00
|
|
|
our_chip->tclk0 = devm_clk_get(&pdev->dev, "pwm-tclk0");
|
|
|
|
our_chip->tclk1 = devm_clk_get(&pdev->dev, "pwm-tclk1");
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2024-02-14 10:32:24 +01:00
|
|
|
platform_set_drvdata(pdev, chip);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2024-02-14 10:32:26 +01:00
|
|
|
ret = devm_pwmchip_add(&pdev->dev, chip);
|
2024-02-14 10:32:27 +01:00
|
|
|
if (ret < 0)
|
|
|
|
return dev_err_probe(dev, ret, "failed to register PWM chip\n");
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
dev_dbg(dev, "base_clk at %lu, tclk0 at %lu, tclk1 at %lu\n",
|
2023-09-29 18:19:17 +02:00
|
|
|
clk_get_rate(our_chip->base_clk),
|
|
|
|
!IS_ERR(our_chip->tclk0) ? clk_get_rate(our_chip->tclk0) : 0,
|
|
|
|
!IS_ERR(our_chip->tclk1) ? clk_get_rate(our_chip->tclk1) : 0);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
static int pwm_samsung_resume(struct device *dev)
|
2013-04-06 02:40:36 +02:00
|
|
|
{
|
2024-02-14 10:32:24 +01:00
|
|
|
struct pwm_chip *chip = dev_get_drvdata(dev);
|
|
|
|
struct samsung_pwm_chip *our_chip = to_samsung_pwm_chip(chip);
|
2013-04-06 02:40:36 +02:00
|
|
|
unsigned int i;
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
for (i = 0; i < SAMSUNG_PWM_NUM; i++) {
|
|
|
|
struct pwm_device *pwm = &chip->pwms[i];
|
2023-07-05 10:06:44 +02:00
|
|
|
struct samsung_pwm_channel *chan = &our_chip->channel[i];
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2023-10-25 14:57:34 +03:00
|
|
|
if (!test_bit(PWMF_REQUESTED, &pwm->flags))
|
2013-04-06 02:40:36 +02:00
|
|
|
continue;
|
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
if (our_chip->variant.output_mask & BIT(i))
|
|
|
|
pwm_samsung_set_invert(our_chip, i,
|
|
|
|
our_chip->inverter_mask & BIT(i));
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
if (chan->period_ns) {
|
|
|
|
__pwm_samsung_config(chip, pwm, chan->duty_ns,
|
|
|
|
chan->period_ns, true);
|
|
|
|
/* needed to make PWM disable work on Odroid-XU3 */
|
|
|
|
pwm_samsung_manual_update(our_chip, pwm);
|
|
|
|
}
|
2013-04-06 02:40:36 +02:00
|
|
|
|
2017-04-24 12:01:08 +02:00
|
|
|
if (our_chip->disabled_mask & BIT(i))
|
|
|
|
pwm_samsung_disable(chip, pwm);
|
|
|
|
else
|
|
|
|
pwm_samsung_enable(chip, pwm);
|
2013-04-06 02:40:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-10-23 19:46:24 +02:00
|
|
|
static DEFINE_SIMPLE_DEV_PM_OPS(pwm_samsung_pm_ops, NULL, pwm_samsung_resume);
|
2013-04-06 02:40:36 +02:00
|
|
|
|
|
|
|
static struct platform_driver pwm_samsung_driver = {
|
|
|
|
.driver = {
|
|
|
|
.name = "samsung-pwm",
|
2023-10-23 19:46:24 +02:00
|
|
|
.pm = pm_ptr(&pwm_samsung_pm_ops),
|
2013-04-06 02:40:36 +02:00
|
|
|
.of_match_table = of_match_ptr(samsung_pwm_matches),
|
|
|
|
},
|
|
|
|
.probe = pwm_samsung_probe,
|
|
|
|
};
|
|
|
|
module_platform_driver(pwm_samsung_driver);
|
|
|
|
|
2024-06-10 07:51:15 -07:00
|
|
|
MODULE_DESCRIPTION("Samsung Pulse Width Modulator driver");
|
2013-04-06 02:40:36 +02:00
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_AUTHOR("Tomasz Figa <tomasz.figa@gmail.com>");
|
|
|
|
MODULE_ALIAS("platform:samsung-pwm");
|