1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (C) 2014 Free Electrons
4 * Copyright (C) 2014 Atmel
6 * Author: Boris BREZILLON <boris.brezillon@free-electrons.com>
10 #include <linux/delay.h>
11 #include <linux/mfd/atmel-hlcdc.h>
12 #include <linux/module.h>
14 #include <linux/platform_device.h>
15 #include <linux/pwm.h>
16 #include <linux/regmap.h>
18 #define ATMEL_HLCDC_PWMCVAL_MASK GENMASK(15, 8)
19 #define ATMEL_HLCDC_PWMCVAL(x) (((x) << 8) & ATMEL_HLCDC_PWMCVAL_MASK)
20 #define ATMEL_HLCDC_PWMPOL BIT(4)
21 #define ATMEL_HLCDC_PWMPS_MASK GENMASK(2, 0)
22 #define ATMEL_HLCDC_PWMPS_MAX 0x6
23 #define ATMEL_HLCDC_PWMPS(x) ((x) & ATMEL_HLCDC_PWMPS_MASK)
25 struct atmel_hlcdc_pwm_errata
{
26 bool slow_clk_erratum
;
27 bool div1_clk_erratum
;
30 struct atmel_hlcdc_pwm
{
31 struct atmel_hlcdc
*hlcdc
;
33 const struct atmel_hlcdc_pwm_errata
*errata
;
36 static inline struct atmel_hlcdc_pwm
*to_atmel_hlcdc_pwm(struct pwm_chip
*chip
)
38 return pwmchip_get_drvdata(chip
);
41 static int atmel_hlcdc_pwm_apply(struct pwm_chip
*chip
, struct pwm_device
*pwm
,
42 const struct pwm_state
*state
)
44 struct atmel_hlcdc_pwm
*atmel
= to_atmel_hlcdc_pwm(chip
);
45 struct atmel_hlcdc
*hlcdc
= atmel
->hlcdc
;
50 struct clk
*new_clk
= hlcdc
->slow_clk
;
51 u64 pwmcval
= state
->duty_cycle
* 256;
52 unsigned long clk_freq
;
57 if (!atmel
->errata
|| !atmel
->errata
->slow_clk_erratum
) {
58 clk_freq
= clk_get_rate(new_clk
);
62 clk_period_ns
= (u64
)NSEC_PER_SEC
* 256;
63 do_div(clk_period_ns
, clk_freq
);
66 /* Errata: cannot use slow clk on some IP revisions */
67 if ((atmel
->errata
&& atmel
->errata
->slow_clk_erratum
) ||
68 clk_period_ns
> state
->period
) {
69 new_clk
= hlcdc
->sys_clk
;
70 clk_freq
= clk_get_rate(new_clk
);
74 clk_period_ns
= (u64
)NSEC_PER_SEC
* 256;
75 do_div(clk_period_ns
, clk_freq
);
78 for (pres
= 0; pres
<= ATMEL_HLCDC_PWMPS_MAX
; pres
++) {
79 /* Errata: cannot divide by 1 on some IP revisions */
80 if (!pres
&& atmel
->errata
&&
81 atmel
->errata
->div1_clk_erratum
)
84 if ((clk_period_ns
<< pres
) >= state
->period
)
88 if (pres
> ATMEL_HLCDC_PWMPS_MAX
)
91 pwmcfg
= ATMEL_HLCDC_PWMPS(pres
);
93 if (new_clk
!= atmel
->cur_clk
) {
97 ret
= clk_prepare_enable(new_clk
);
101 clk_disable_unprepare(atmel
->cur_clk
);
102 atmel
->cur_clk
= new_clk
;
104 if (new_clk
== hlcdc
->sys_clk
)
105 gencfg
= ATMEL_HLCDC_CLKPWMSEL
;
107 ret
= regmap_update_bits(hlcdc
->regmap
,
109 ATMEL_HLCDC_CLKPWMSEL
,
115 do_div(pwmcval
, state
->period
);
118 * The PWM duty cycle is configurable from 0/256 to 255/256 of
119 * the period cycle. Hence we can't set a duty cycle occupying
120 * the whole period cycle if we're asked to.
121 * Set it to 255 if pwmcval is greater than 256.
126 pwmcfg
|= ATMEL_HLCDC_PWMCVAL(pwmcval
);
128 if (state
->polarity
== PWM_POLARITY_NORMAL
)
129 pwmcfg
|= ATMEL_HLCDC_PWMPOL
;
131 ret
= regmap_update_bits(hlcdc
->regmap
, ATMEL_HLCDC_CFG(6),
132 ATMEL_HLCDC_PWMCVAL_MASK
|
133 ATMEL_HLCDC_PWMPS_MASK
|
139 ret
= regmap_write(hlcdc
->regmap
, ATMEL_HLCDC_EN
,
144 ret
= regmap_read_poll_timeout(hlcdc
->regmap
, ATMEL_HLCDC_SR
,
146 status
& ATMEL_HLCDC_PWM
,
151 ret
= regmap_write(hlcdc
->regmap
, ATMEL_HLCDC_DIS
,
156 ret
= regmap_read_poll_timeout(hlcdc
->regmap
, ATMEL_HLCDC_SR
,
158 !(status
& ATMEL_HLCDC_PWM
),
163 clk_disable_unprepare(atmel
->cur_clk
);
164 atmel
->cur_clk
= NULL
;
170 static const struct pwm_ops atmel_hlcdc_pwm_ops
= {
171 .apply
= atmel_hlcdc_pwm_apply
,
174 static const struct atmel_hlcdc_pwm_errata atmel_hlcdc_pwm_at91sam9x5_errata
= {
175 .slow_clk_erratum
= true,
178 static const struct atmel_hlcdc_pwm_errata atmel_hlcdc_pwm_sama5d3_errata
= {
179 .div1_clk_erratum
= true,
182 static int atmel_hlcdc_pwm_suspend(struct device
*dev
)
184 struct pwm_chip
*chip
= dev_get_drvdata(dev
);
185 struct atmel_hlcdc_pwm
*atmel
= to_atmel_hlcdc_pwm(chip
);
186 struct pwm_device
*pwm
= &chip
->pwms
[0];
188 /* Keep the periph clock enabled if the PWM is still running. */
189 if (!pwm
->state
.enabled
)
190 clk_disable_unprepare(atmel
->hlcdc
->periph_clk
);
195 static int atmel_hlcdc_pwm_resume(struct device
*dev
)
197 struct pwm_chip
*chip
= dev_get_drvdata(dev
);
198 struct atmel_hlcdc_pwm
*atmel
= to_atmel_hlcdc_pwm(chip
);
199 struct pwm_device
*pwm
= &chip
->pwms
[0];
202 /* Re-enable the periph clock it was stopped during suspend. */
203 if (!pwm
->state
.enabled
) {
204 ret
= clk_prepare_enable(atmel
->hlcdc
->periph_clk
);
209 return atmel_hlcdc_pwm_apply(chip
, pwm
, &pwm
->state
);
212 static DEFINE_SIMPLE_DEV_PM_OPS(atmel_hlcdc_pwm_pm_ops
,
213 atmel_hlcdc_pwm_suspend
, atmel_hlcdc_pwm_resume
);
215 static const struct of_device_id atmel_hlcdc_dt_ids
[] = {
217 .compatible
= "atmel,at91sam9n12-hlcdc",
218 /* 9n12 has same errata as 9x5 HLCDC PWM */
219 .data
= &atmel_hlcdc_pwm_at91sam9x5_errata
,
222 .compatible
= "atmel,at91sam9x5-hlcdc",
223 .data
= &atmel_hlcdc_pwm_at91sam9x5_errata
,
226 .compatible
= "atmel,sama5d2-hlcdc",
229 .compatible
= "atmel,sama5d3-hlcdc",
230 .data
= &atmel_hlcdc_pwm_sama5d3_errata
,
233 .compatible
= "atmel,sama5d4-hlcdc",
234 .data
= &atmel_hlcdc_pwm_sama5d3_errata
,
236 { .compatible
= "microchip,sam9x60-hlcdc", },
239 MODULE_DEVICE_TABLE(of
, atmel_hlcdc_dt_ids
);
241 static int atmel_hlcdc_pwm_probe(struct platform_device
*pdev
)
243 const struct of_device_id
*match
;
244 struct device
*dev
= &pdev
->dev
;
245 struct pwm_chip
*chip
;
246 struct atmel_hlcdc_pwm
*atmel
;
247 struct atmel_hlcdc
*hlcdc
;
250 hlcdc
= dev_get_drvdata(dev
->parent
);
252 chip
= devm_pwmchip_alloc(dev
, 1, sizeof(*atmel
));
254 return PTR_ERR(chip
);
255 atmel
= to_atmel_hlcdc_pwm(chip
);
257 ret
= clk_prepare_enable(hlcdc
->periph_clk
);
261 match
= of_match_node(atmel_hlcdc_dt_ids
, dev
->parent
->of_node
);
263 atmel
->errata
= match
->data
;
265 atmel
->hlcdc
= hlcdc
;
266 chip
->ops
= &atmel_hlcdc_pwm_ops
;
268 ret
= pwmchip_add(chip
);
270 clk_disable_unprepare(hlcdc
->periph_clk
);
274 platform_set_drvdata(pdev
, chip
);
279 static void atmel_hlcdc_pwm_remove(struct platform_device
*pdev
)
281 struct pwm_chip
*chip
= platform_get_drvdata(pdev
);
282 struct atmel_hlcdc_pwm
*atmel
= to_atmel_hlcdc_pwm(chip
);
284 pwmchip_remove(chip
);
286 clk_disable_unprepare(atmel
->hlcdc
->periph_clk
);
289 static const struct of_device_id atmel_hlcdc_pwm_dt_ids
[] = {
290 { .compatible
= "atmel,hlcdc-pwm" },
293 MODULE_DEVICE_TABLE(of
, atmel_hlcdc_pwm_dt_ids
);
295 static struct platform_driver atmel_hlcdc_pwm_driver
= {
297 .name
= "atmel-hlcdc-pwm",
298 .of_match_table
= atmel_hlcdc_pwm_dt_ids
,
299 .pm
= pm_ptr(&atmel_hlcdc_pwm_pm_ops
),
301 .probe
= atmel_hlcdc_pwm_probe
,
302 .remove
= atmel_hlcdc_pwm_remove
,
304 module_platform_driver(atmel_hlcdc_pwm_driver
);
306 MODULE_ALIAS("platform:atmel-hlcdc-pwm");
307 MODULE_AUTHOR("Boris Brezillon <boris.brezillon@free-electrons.com>");
308 MODULE_DESCRIPTION("Atmel HLCDC PWM driver");
309 MODULE_LICENSE("GPL v2");