Commit ace1ba1c authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'pwm/for-6.5-rc1' of...

Merge tag 'pwm/for-6.5-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/thierry.reding/linux-pwm

Pull pwm updates from Thierry Reding:
 "There's a little bit of everything in here: we've got various
  improvements and cleanups to drivers, some fixes across the board and
  a bit of new hardware support"

* tag 'pwm/for-6.5-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/thierry.reding/linux-pwm: (22 commits)
  dt-bindings: pwm: convert pwm-bcm2835 bindings to YAML
  pwm: Add Renesas RZ/G2L MTU3a PWM driver
  pwm: mtk_disp: Fix the disable flow of disp_pwm
  dt-bindings: pwm: restrict node name suffixes
  pwm: pca9685: Switch i2c driver back to use .probe()
  pwm: ab8500: Fix error code in probe()
  MAINTAINERS: add pwm to PolarFire SoC entry
  pwm: add microchip soft ip corePWM driver
  pwm: sysfs: Do not apply state to already disabled PWMs
  pwm: imx-tpm: force 'real_period' to be zero in suspend
  pwm: meson: make full use of common clock framework
  pwm: meson: don't use hdmi/video clock as mux parent
  pwm: meson: switch to using struct clk_parent_data for mux parents
  pwm: meson: remove not needed check in meson_pwm_calc
  pwm: meson: fix handling of period/duty if greater than UINT_MAX
  pwm: meson: modify and simplify calculation in meson_pwm_get_state
  dt-bindings: pwm: Add R-Car V3U device tree bindings
  dt-bindings: pwm: imx: add i.MX8QXP compatible
  pwm: mediatek: Add support for MT7981
  dt-bindings: pwm: mediatek: Add mediatek,mt7981 compatible
  ...
parents b9861581 92554cdd
...@@ -43,6 +43,7 @@ properties: ...@@ -43,6 +43,7 @@ properties:
- fsl,imx8mn-pwm - fsl,imx8mn-pwm
- fsl,imx8mp-pwm - fsl,imx8mp-pwm
- fsl,imx8mq-pwm - fsl,imx8mq-pwm
- fsl,imx8qxp-pwm
- const: fsl,imx27-pwm - const: fsl,imx27-pwm
reg: reg:
...@@ -61,6 +62,9 @@ properties: ...@@ -61,6 +62,9 @@ properties:
interrupts: interrupts:
maxItems: 1 maxItems: 1
power-domains:
maxItems: 1
required: required:
- compatible - compatible
- reg - reg
......
...@@ -22,6 +22,7 @@ properties: ...@@ -22,6 +22,7 @@ properties:
- mediatek,mt7623-pwm - mediatek,mt7623-pwm
- mediatek,mt7628-pwm - mediatek,mt7628-pwm
- mediatek,mt7629-pwm - mediatek,mt7629-pwm
- mediatek,mt7981-pwm
- mediatek,mt7986-pwm - mediatek,mt7986-pwm
- mediatek,mt8183-pwm - mediatek,mt8183-pwm
- mediatek,mt8365-pwm - mediatek,mt8365-pwm
......
BCM2835 PWM controller (Raspberry Pi controller)
Required properties:
- compatible: should be "brcm,bcm2835-pwm"
- reg: physical base address and length of the controller's registers
- clocks: This clock defines the base clock frequency of the PWM hardware
system, the period and the duty_cycle of the PWM signal is a multiple of
the base period.
- #pwm-cells: Should be 3. See pwm.yaml in this directory for a description of
the cells format.
Examples:
pwm@2020c000 {
compatible = "brcm,bcm2835-pwm";
reg = <0x2020c000 0x28>;
clocks = <&clk_pwm>;
#pwm-cells = <3>;
};
clocks {
....
clk_pwm: pwm {
compatible = "fixed-clock";
reg = <3>;
#clock-cells = <0>;
clock-frequency = <9200000>;
};
....
};
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
%YAML 1.2
---
$id: http://devicetree.org/schemas/pwm/pwm-bcm2835.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: BCM2835 PWM controller (Raspberry Pi controller)
maintainers:
- Stefan Wahren <stefan.wahren@i2se.com>
allOf:
- $ref: pwm.yaml#
properties:
compatible:
const: brcm,bcm2835-pwm
reg:
maxItems: 1
clocks:
maxItems: 1
"#pwm-cells":
const: 3
required:
- compatible
- reg
- clocks
- "#pwm-cells"
additionalProperties: false
examples:
- |
pwm@2020c000 {
compatible = "brcm,bcm2835-pwm";
reg = <0x2020c000 0x28>;
clocks = <&clk_pwm>;
#pwm-cells = <3>;
};
...@@ -13,7 +13,7 @@ select: false ...@@ -13,7 +13,7 @@ select: false
properties: properties:
$nodename: $nodename:
pattern: "^pwm(@.*|-[0-9a-f])*$" pattern: "^pwm(@.*|-([0-9]|[1-9][0-9]+))?$"
"#pwm-cells": "#pwm-cells":
description: description:
......
...@@ -35,6 +35,7 @@ properties: ...@@ -35,6 +35,7 @@ properties:
- renesas,pwm-r8a77980 # R-Car V3H - renesas,pwm-r8a77980 # R-Car V3H
- renesas,pwm-r8a77990 # R-Car E3 - renesas,pwm-r8a77990 # R-Car E3
- renesas,pwm-r8a77995 # R-Car D3 - renesas,pwm-r8a77995 # R-Car D3
- renesas,pwm-r8a779a0 # R-Car V3U
- renesas,pwm-r8a779g0 # R-Car V4H - renesas,pwm-r8a779g0 # R-Car V4H
- const: renesas,pwm-rcar - const: renesas,pwm-rcar
......
...@@ -18337,6 +18337,7 @@ F: drivers/clk/microchip/clk-mpfs*.c ...@@ -18337,6 +18337,7 @@ F: drivers/clk/microchip/clk-mpfs*.c
F: drivers/i2c/busses/i2c-microchip-corei2c.c F: drivers/i2c/busses/i2c-microchip-corei2c.c
F: drivers/mailbox/mailbox-mpfs.c F: drivers/mailbox/mailbox-mpfs.c
F: drivers/pci/controller/pcie-microchip-host.c F: drivers/pci/controller/pcie-microchip-host.c
F: drivers/pwm/pwm-microchip-core.c
F: drivers/reset/reset-mpfs.c F: drivers/reset/reset-mpfs.c
F: drivers/rtc/rtc-mpfs.c F: drivers/rtc/rtc-mpfs.c
F: drivers/soc/microchip/mpfs-sys-controller.c F: drivers/soc/microchip/mpfs-sys-controller.c
......
...@@ -405,6 +405,16 @@ config PWM_MEDIATEK ...@@ -405,6 +405,16 @@ config PWM_MEDIATEK
To compile this driver as a module, choose M here: the module To compile this driver as a module, choose M here: the module
will be called pwm-mediatek. will be called pwm-mediatek.
config PWM_MICROCHIP_CORE
tristate "Microchip corePWM PWM support"
depends on SOC_MICROCHIP_POLARFIRE || COMPILE_TEST
depends on HAS_IOMEM && OF
help
PWM driver for Microchip FPGA soft IP core.
To compile this driver as a module, choose M here: the module
will be called pwm-microchip-core.
config PWM_MXS config PWM_MXS
tristate "Freescale MXS PWM support" tristate "Freescale MXS PWM support"
depends on ARCH_MXS || COMPILE_TEST depends on ARCH_MXS || COMPILE_TEST
...@@ -493,6 +503,17 @@ config PWM_ROCKCHIP ...@@ -493,6 +503,17 @@ config PWM_ROCKCHIP
Generic PWM framework driver for the PWM controller found on Generic PWM framework driver for the PWM controller found on
Rockchip SoCs. Rockchip SoCs.
config PWM_RZ_MTU3
tristate "Renesas RZ/G2L MTU3a PWM Timer support"
depends on RZ_MTU3 || COMPILE_TEST
depends on HAS_IOMEM
help
This driver exposes the MTU3a PWM Timer controller found in Renesas
RZ/G2L like chips through the PWM API.
To compile this driver as a module, choose M here: the module
will be called pwm-rz-mtu3.
config PWM_SAMSUNG config PWM_SAMSUNG
tristate "Samsung PWM support" tristate "Samsung PWM support"
depends on PLAT_SAMSUNG || ARCH_S5PV210 || ARCH_EXYNOS || COMPILE_TEST depends on PLAT_SAMSUNG || ARCH_S5PV210 || ARCH_EXYNOS || COMPILE_TEST
......
...@@ -35,6 +35,7 @@ obj-$(CONFIG_PWM_LPSS_PCI) += pwm-lpss-pci.o ...@@ -35,6 +35,7 @@ obj-$(CONFIG_PWM_LPSS_PCI) += pwm-lpss-pci.o
obj-$(CONFIG_PWM_LPSS_PLATFORM) += pwm-lpss-platform.o obj-$(CONFIG_PWM_LPSS_PLATFORM) += pwm-lpss-platform.o
obj-$(CONFIG_PWM_MESON) += pwm-meson.o obj-$(CONFIG_PWM_MESON) += pwm-meson.o
obj-$(CONFIG_PWM_MEDIATEK) += pwm-mediatek.o obj-$(CONFIG_PWM_MEDIATEK) += pwm-mediatek.o
obj-$(CONFIG_PWM_MICROCHIP_CORE) += pwm-microchip-core.o
obj-$(CONFIG_PWM_MTK_DISP) += pwm-mtk-disp.o obj-$(CONFIG_PWM_MTK_DISP) += pwm-mtk-disp.o
obj-$(CONFIG_PWM_MXS) += pwm-mxs.o obj-$(CONFIG_PWM_MXS) += pwm-mxs.o
obj-$(CONFIG_PWM_NTXEC) += pwm-ntxec.o obj-$(CONFIG_PWM_NTXEC) += pwm-ntxec.o
...@@ -45,6 +46,7 @@ obj-$(CONFIG_PWM_RASPBERRYPI_POE) += pwm-raspberrypi-poe.o ...@@ -45,6 +46,7 @@ obj-$(CONFIG_PWM_RASPBERRYPI_POE) += pwm-raspberrypi-poe.o
obj-$(CONFIG_PWM_RCAR) += pwm-rcar.o obj-$(CONFIG_PWM_RCAR) += pwm-rcar.o
obj-$(CONFIG_PWM_RENESAS_TPU) += pwm-renesas-tpu.o obj-$(CONFIG_PWM_RENESAS_TPU) += pwm-renesas-tpu.o
obj-$(CONFIG_PWM_ROCKCHIP) += pwm-rockchip.o obj-$(CONFIG_PWM_ROCKCHIP) += pwm-rockchip.o
obj-$(CONFIG_PWM_RZ_MTU3) += pwm-rz-mtu3.o
obj-$(CONFIG_PWM_SAMSUNG) += pwm-samsung.o obj-$(CONFIG_PWM_SAMSUNG) += pwm-samsung.o
obj-$(CONFIG_PWM_SIFIVE) += pwm-sifive.o obj-$(CONFIG_PWM_SIFIVE) += pwm-sifive.o
obj-$(CONFIG_PWM_SL28CPLD) += pwm-sl28cpld.o obj-$(CONFIG_PWM_SL28CPLD) += pwm-sl28cpld.o
......
...@@ -190,7 +190,7 @@ static int ab8500_pwm_probe(struct platform_device *pdev) ...@@ -190,7 +190,7 @@ static int ab8500_pwm_probe(struct platform_device *pdev)
int err; int err;
if (pdev->id < 1 || pdev->id > 31) if (pdev->id < 1 || pdev->id > 31)
return dev_err_probe(&pdev->dev, EINVAL, "Invalid device id %d\n", pdev->id); return dev_err_probe(&pdev->dev, -EINVAL, "Invalid device id %d\n", pdev->id);
/* /*
* Nothing to be done in probe, this is required to get the * Nothing to be done in probe, this is required to get the
......
...@@ -89,7 +89,7 @@ static int pwm_clk_probe(struct platform_device *pdev) ...@@ -89,7 +89,7 @@ static int pwm_clk_probe(struct platform_device *pdev)
if (!pcchip) if (!pcchip)
return -ENOMEM; return -ENOMEM;
pcchip->clk = devm_clk_get(&pdev->dev, NULL); pcchip->clk = devm_clk_get_prepared(&pdev->dev, NULL);
if (IS_ERR(pcchip->clk)) if (IS_ERR(pcchip->clk))
return dev_err_probe(&pdev->dev, PTR_ERR(pcchip->clk), return dev_err_probe(&pdev->dev, PTR_ERR(pcchip->clk),
"Failed to get clock\n"); "Failed to get clock\n");
...@@ -98,15 +98,9 @@ static int pwm_clk_probe(struct platform_device *pdev) ...@@ -98,15 +98,9 @@ static int pwm_clk_probe(struct platform_device *pdev)
pcchip->chip.ops = &pwm_clk_ops; pcchip->chip.ops = &pwm_clk_ops;
pcchip->chip.npwm = 1; pcchip->chip.npwm = 1;
ret = clk_prepare(pcchip->clk);
if (ret < 0)
return dev_err_probe(&pdev->dev, ret, "Failed to prepare clock\n");
ret = pwmchip_add(&pcchip->chip); ret = pwmchip_add(&pcchip->chip);
if (ret < 0) { if (ret < 0)
clk_unprepare(pcchip->clk);
return dev_err_probe(&pdev->dev, ret, "Failed to add pwm chip\n"); return dev_err_probe(&pdev->dev, ret, "Failed to add pwm chip\n");
}
platform_set_drvdata(pdev, pcchip); platform_set_drvdata(pdev, pcchip);
return 0; return 0;
...@@ -120,8 +114,6 @@ static void pwm_clk_remove(struct platform_device *pdev) ...@@ -120,8 +114,6 @@ static void pwm_clk_remove(struct platform_device *pdev)
if (pcchip->clk_enabled) if (pcchip->clk_enabled)
clk_disable(pcchip->clk); clk_disable(pcchip->clk);
clk_unprepare(pcchip->clk);
} }
static const struct of_device_id pwm_clk_dt_ids[] = { static const struct of_device_id pwm_clk_dt_ids[] = {
......
...@@ -397,6 +397,13 @@ static int __maybe_unused pwm_imx_tpm_suspend(struct device *dev) ...@@ -397,6 +397,13 @@ static int __maybe_unused pwm_imx_tpm_suspend(struct device *dev)
if (tpm->enable_count > 0) if (tpm->enable_count > 0)
return -EBUSY; return -EBUSY;
/*
* Force 'real_period' to be zero to force period update code
* can be executed after system resume back, since suspend causes
* the period related registers to become their reset values.
*/
tpm->real_period = 0;
clk_disable_unprepare(tpm->clk); clk_disable_unprepare(tpm->clk);
return 0; return 0;
......
...@@ -38,6 +38,7 @@ struct pwm_mediatek_of_data { ...@@ -38,6 +38,7 @@ struct pwm_mediatek_of_data {
unsigned int num_pwms; unsigned int num_pwms;
bool pwm45_fixup; bool pwm45_fixup;
bool has_ck_26m_sel; bool has_ck_26m_sel;
const unsigned int *reg_offset;
}; };
/** /**
...@@ -59,10 +60,14 @@ struct pwm_mediatek_chip { ...@@ -59,10 +60,14 @@ struct pwm_mediatek_chip {
const struct pwm_mediatek_of_data *soc; const struct pwm_mediatek_of_data *soc;
}; };
static const unsigned int pwm_mediatek_reg_offset[] = { static const unsigned int mtk_pwm_reg_offset_v1[] = {
0x0010, 0x0050, 0x0090, 0x00d0, 0x0110, 0x0150, 0x0190, 0x0220 0x0010, 0x0050, 0x0090, 0x00d0, 0x0110, 0x0150, 0x0190, 0x0220
}; };
static const unsigned int mtk_pwm_reg_offset_v2[] = {
0x0080, 0x00c0, 0x0100, 0x0140, 0x0180, 0x01c0, 0x0200, 0x0240
};
static inline struct pwm_mediatek_chip * static inline struct pwm_mediatek_chip *
to_pwm_mediatek_chip(struct pwm_chip *chip) to_pwm_mediatek_chip(struct pwm_chip *chip)
{ {
...@@ -111,7 +116,7 @@ static inline void pwm_mediatek_writel(struct pwm_mediatek_chip *chip, ...@@ -111,7 +116,7 @@ static inline void pwm_mediatek_writel(struct pwm_mediatek_chip *chip,
unsigned int num, unsigned int offset, unsigned int num, unsigned int offset,
u32 value) u32 value)
{ {
writel(value, chip->regs + pwm_mediatek_reg_offset[num] + offset); writel(value, chip->regs + chip->soc->reg_offset[num] + offset);
} }
static int pwm_mediatek_config(struct pwm_chip *chip, struct pwm_device *pwm, static int pwm_mediatek_config(struct pwm_chip *chip, struct pwm_device *pwm,
...@@ -285,60 +290,77 @@ static const struct pwm_mediatek_of_data mt2712_pwm_data = { ...@@ -285,60 +290,77 @@ static const struct pwm_mediatek_of_data mt2712_pwm_data = {
.num_pwms = 8, .num_pwms = 8,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = false, .has_ck_26m_sel = false,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt6795_pwm_data = { static const struct pwm_mediatek_of_data mt6795_pwm_data = {
.num_pwms = 7, .num_pwms = 7,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = false, .has_ck_26m_sel = false,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt7622_pwm_data = { static const struct pwm_mediatek_of_data mt7622_pwm_data = {
.num_pwms = 6, .num_pwms = 6,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = true, .has_ck_26m_sel = true,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt7623_pwm_data = { static const struct pwm_mediatek_of_data mt7623_pwm_data = {
.num_pwms = 5, .num_pwms = 5,
.pwm45_fixup = true, .pwm45_fixup = true,
.has_ck_26m_sel = false, .has_ck_26m_sel = false,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt7628_pwm_data = { static const struct pwm_mediatek_of_data mt7628_pwm_data = {
.num_pwms = 4, .num_pwms = 4,
.pwm45_fixup = true, .pwm45_fixup = true,
.has_ck_26m_sel = false, .has_ck_26m_sel = false,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt7629_pwm_data = { static const struct pwm_mediatek_of_data mt7629_pwm_data = {
.num_pwms = 1, .num_pwms = 1,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = false, .has_ck_26m_sel = false,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt8183_pwm_data = { static const struct pwm_mediatek_of_data mt7981_pwm_data = {
.num_pwms = 4, .num_pwms = 3,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = true, .has_ck_26m_sel = true,
.reg_offset = mtk_pwm_reg_offset_v2,
}; };
static const struct pwm_mediatek_of_data mt8365_pwm_data = { static const struct pwm_mediatek_of_data mt7986_pwm_data = {
.num_pwms = 3, .num_pwms = 2,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = true, .has_ck_26m_sel = true,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt7986_pwm_data = { static const struct pwm_mediatek_of_data mt8183_pwm_data = {
.num_pwms = 2, .num_pwms = 4,
.pwm45_fixup = false,
.has_ck_26m_sel = true,
.reg_offset = mtk_pwm_reg_offset_v1,
};
static const struct pwm_mediatek_of_data mt8365_pwm_data = {
.num_pwms = 3,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = true, .has_ck_26m_sel = true,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct pwm_mediatek_of_data mt8516_pwm_data = { static const struct pwm_mediatek_of_data mt8516_pwm_data = {
.num_pwms = 5, .num_pwms = 5,
.pwm45_fixup = false, .pwm45_fixup = false,
.has_ck_26m_sel = true, .has_ck_26m_sel = true,
.reg_offset = mtk_pwm_reg_offset_v1,
}; };
static const struct of_device_id pwm_mediatek_of_match[] = { static const struct of_device_id pwm_mediatek_of_match[] = {
...@@ -348,6 +370,7 @@ static const struct of_device_id pwm_mediatek_of_match[] = { ...@@ -348,6 +370,7 @@ static const struct of_device_id pwm_mediatek_of_match[] = {
{ .compatible = "mediatek,mt7623-pwm", .data = &mt7623_pwm_data }, { .compatible = "mediatek,mt7623-pwm", .data = &mt7623_pwm_data },
{ .compatible = "mediatek,mt7628-pwm", .data = &mt7628_pwm_data }, { .compatible = "mediatek,mt7628-pwm", .data = &mt7628_pwm_data },
{ .compatible = "mediatek,mt7629-pwm", .data = &mt7629_pwm_data }, { .compatible = "mediatek,mt7629-pwm", .data = &mt7629_pwm_data },
{ .compatible = "mediatek,mt7981-pwm", .data = &mt7981_pwm_data },
{ .compatible = "mediatek,mt7986-pwm", .data = &mt7986_pwm_data }, { .compatible = "mediatek,mt7986-pwm", .data = &mt7986_pwm_data },
{ .compatible = "mediatek,mt8183-pwm", .data = &mt8183_pwm_data }, { .compatible = "mediatek,mt8183-pwm", .data = &mt8183_pwm_data },
{ .compatible = "mediatek,mt8365-pwm", .data = &mt8365_pwm_data }, { .compatible = "mediatek,mt8365-pwm", .data = &mt8365_pwm_data },
......
This diff is collapsed.
This diff is collapsed.
...@@ -79,14 +79,11 @@ static int mtk_disp_pwm_apply(struct pwm_chip *chip, struct pwm_device *pwm, ...@@ -79,14 +79,11 @@ static int mtk_disp_pwm_apply(struct pwm_chip *chip, struct pwm_device *pwm,
if (state->polarity != PWM_POLARITY_NORMAL) if (state->polarity != PWM_POLARITY_NORMAL)
return -EINVAL; return -EINVAL;
if (!state->enabled) { if (!state->enabled && mdp->enabled) {
mtk_disp_pwm_update_bits(mdp, DISP_PWM_EN, mdp->data->enable_mask, mtk_disp_pwm_update_bits(mdp, DISP_PWM_EN,
0x0); mdp->data->enable_mask, 0x0);
clk_disable_unprepare(mdp->clk_mm);
if (mdp->enabled) { clk_disable_unprepare(mdp->clk_main);
clk_disable_unprepare(mdp->clk_mm);
clk_disable_unprepare(mdp->clk_main);
}
mdp->enabled = false; mdp->enabled = false;
return 0; return 0;
......
...@@ -665,7 +665,7 @@ static struct i2c_driver pca9685_i2c_driver = { ...@@ -665,7 +665,7 @@ static struct i2c_driver pca9685_i2c_driver = {
.of_match_table = of_match_ptr(pca9685_dt_ids), .of_match_table = of_match_ptr(pca9685_dt_ids),
.pm = &pca9685_pwm_pm, .pm = &pca9685_pwm_pm,
}, },
.probe_new = pca9685_pwm_probe, .probe = pca9685_pwm_probe,
.remove = pca9685_pwm_remove, .remove = pca9685_pwm_remove,
.id_table = pca9685_id, .id_table = pca9685_id,
}; };
......
This diff is collapsed.
...@@ -244,12 +244,12 @@ static int pwm_sifive_probe(struct platform_device *pdev) ...@@ -244,12 +244,12 @@ static int pwm_sifive_probe(struct platform_device *pdev)
if (IS_ERR(ddata->regs)) if (IS_ERR(ddata->regs))
return PTR_ERR(ddata->regs); return PTR_ERR(ddata->regs);
ddata->clk = devm_clk_get(dev, NULL); ddata->clk = devm_clk_get_prepared(dev, NULL);
if (IS_ERR(ddata->clk)) if (IS_ERR(ddata->clk))
return dev_err_probe(dev, PTR_ERR(ddata->clk), return dev_err_probe(dev, PTR_ERR(ddata->clk),
"Unable to find controller clock\n"); "Unable to find controller clock\n");
ret = clk_prepare_enable(ddata->clk); ret = clk_enable(ddata->clk);
if (ret) { if (ret) {
dev_err(dev, "failed to enable clock for pwm: %d\n", ret); dev_err(dev, "failed to enable clock for pwm: %d\n", ret);
return ret; return ret;
...@@ -308,7 +308,6 @@ static int pwm_sifive_probe(struct platform_device *pdev) ...@@ -308,7 +308,6 @@ static int pwm_sifive_probe(struct platform_device *pdev)
clk_disable(ddata->clk); clk_disable(ddata->clk);
--enabled_clks; --enabled_clks;
} }
clk_unprepare(ddata->clk);
return ret; return ret;
} }
...@@ -327,8 +326,6 @@ static void pwm_sifive_remove(struct platform_device *dev) ...@@ -327,8 +326,6 @@ static void pwm_sifive_remove(struct platform_device *dev)
if (pwm->state.enabled) if (pwm->state.enabled)
clk_disable(ddata->clk); clk_disable(ddata->clk);
} }
clk_unprepare(ddata->clk);
} }
static const struct of_device_id pwm_sifive_of_match[] = { static const struct of_device_id pwm_sifive_of_match[] = {
......
...@@ -424,6 +424,13 @@ static int pwm_class_resume_npwm(struct device *parent, unsigned int npwm) ...@@ -424,6 +424,13 @@ static int pwm_class_resume_npwm(struct device *parent, unsigned int npwm)
if (!export) if (!export)
continue; continue;
/* If pwmchip was not enabled before suspend, do nothing. */
if (!export->suspend.enabled) {
/* release lock taken in pwm_class_get_state */
mutex_unlock(&export->lock);
continue;
}
state.enabled = export->suspend.enabled; state.enabled = export->suspend.enabled;
ret = pwm_class_apply_state(export, pwm, &state); ret = pwm_class_apply_state(export, pwm, &state);
if (ret < 0) if (ret < 0)
...@@ -448,7 +455,17 @@ static int pwm_class_suspend(struct device *parent) ...@@ -448,7 +455,17 @@ static int pwm_class_suspend(struct device *parent)
if (!export) if (!export)
continue; continue;
/*
* If pwmchip was not enabled before suspend, save
* state for resume time and do nothing else.
*/
export->suspend = state; export->suspend = state;
if (!state.enabled) {
/* release lock taken in pwm_class_get_state */
mutex_unlock(&export->lock);
continue;
}
state.enabled = false; state.enabled = false;
ret = pwm_class_apply_state(export, pwm, &state); ret = pwm_class_apply_state(export, pwm, &state);
if (ret < 0) { if (ret < 0) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment