2 * NXP LPC18xx State Configurable Timer - Pulse Width Modulator driver
4 * Copyright (c) 2015 Ariel D'Alessandro <ariel@vanguardiasur.com>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License.
12 * NXP LPC18xx provides a State Configurable Timer (SCT) which can be configured
13 * as a Pulse Width Modulator.
15 * SCT supports 16 outputs, 16 events and 16 registers. Each event will be
16 * triggered when its related register matches the SCT counter value, and it
17 * will set or clear a selected output.
19 * One of the events is preselected to generate the period, thus the maximum
20 * number of simultaneous channels is limited to 15. Notice that period is
21 * global to all the channels, thus PWM driver will refuse setting different
22 * values to it, unless there's only one channel requested.
25 #include <linux/clk.h>
26 #include <linux/err.h>
28 #include <linux/module.h>
29 #include <linux/platform_device.h>
30 #include <linux/pwm.h>
32 /* LPC18xx SCT registers */
33 #define LPC18XX_PWM_CONFIG 0x000
34 #define LPC18XX_PWM_CONFIG_UNIFY BIT(0)
35 #define LPC18XX_PWM_CONFIG_NORELOAD BIT(7)
37 #define LPC18XX_PWM_CTRL 0x004
38 #define LPC18XX_PWM_CTRL_HALT BIT(2)
39 #define LPC18XX_PWM_BIDIR BIT(4)
40 #define LPC18XX_PWM_PRE_SHIFT 5
41 #define LPC18XX_PWM_PRE_MASK (0xff << LPC18XX_PWM_PRE_SHIFT)
42 #define LPC18XX_PWM_PRE(x) (x << LPC18XX_PWM_PRE_SHIFT)
44 #define LPC18XX_PWM_LIMIT 0x008
46 #define LPC18XX_PWM_RES_BASE 0x058
47 #define LPC18XX_PWM_RES_SHIFT(_ch) (_ch * 2)
48 #define LPC18XX_PWM_RES(_ch, _action) (_action << LPC18XX_PWM_RES_SHIFT(_ch))
49 #define LPC18XX_PWM_RES_MASK(_ch) (0x3 << LPC18XX_PWM_RES_SHIFT(_ch))
51 #define LPC18XX_PWM_MATCH_BASE 0x100
52 #define LPC18XX_PWM_MATCH(_ch) (LPC18XX_PWM_MATCH_BASE + _ch * 4)
54 #define LPC18XX_PWM_MATCHREL_BASE 0x200
55 #define LPC18XX_PWM_MATCHREL(_ch) (LPC18XX_PWM_MATCHREL_BASE + _ch * 4)
57 #define LPC18XX_PWM_EVSTATEMSK_BASE 0x300
58 #define LPC18XX_PWM_EVSTATEMSK(_ch) (LPC18XX_PWM_EVSTATEMSK_BASE + _ch * 8)
59 #define LPC18XX_PWM_EVSTATEMSK_ALL 0xffffffff
61 #define LPC18XX_PWM_EVCTRL_BASE 0x304
62 #define LPC18XX_PWM_EVCTRL(_ev) (LPC18XX_PWM_EVCTRL_BASE + _ev * 8)
64 #define LPC18XX_PWM_EVCTRL_MATCH(_ch) _ch
66 #define LPC18XX_PWM_EVCTRL_COMB_SHIFT 12
67 #define LPC18XX_PWM_EVCTRL_COMB_MATCH (0x1 << LPC18XX_PWM_EVCTRL_COMB_SHIFT)
69 #define LPC18XX_PWM_OUTPUTSET_BASE 0x500
70 #define LPC18XX_PWM_OUTPUTSET(_ch) (LPC18XX_PWM_OUTPUTSET_BASE + _ch * 8)
72 #define LPC18XX_PWM_OUTPUTCL_BASE 0x504
73 #define LPC18XX_PWM_OUTPUTCL(_ch) (LPC18XX_PWM_OUTPUTCL_BASE + _ch * 8)
75 /* LPC18xx SCT unified counter */
76 #define LPC18XX_PWM_TIMER_MAX 0xffffffff
78 /* LPC18xx SCT events */
79 #define LPC18XX_PWM_EVENT_PERIOD 0
80 #define LPC18XX_PWM_EVENT_MAX 16
82 /* SCT conflict resolution */
83 enum lpc18xx_pwm_res_action
{
86 LPC18XX_PWM_RES_CLEAR
,
87 LPC18XX_PWM_RES_TOGGLE
,
90 struct lpc18xx_pwm_data
{
91 unsigned int duty_event
;
94 struct lpc18xx_pwm_chip
{
99 unsigned long clk_rate
;
100 unsigned int period_ns
;
101 unsigned int min_period_ns
;
102 unsigned int max_period_ns
;
103 unsigned int period_event
;
104 unsigned long event_map
;
105 struct mutex res_lock
;
106 struct mutex period_lock
;
109 static inline struct lpc18xx_pwm_chip
*
110 to_lpc18xx_pwm_chip(struct pwm_chip
*chip
)
112 return container_of(chip
, struct lpc18xx_pwm_chip
, chip
);
115 static inline void lpc18xx_pwm_writel(struct lpc18xx_pwm_chip
*lpc18xx_pwm
,
118 writel(val
, lpc18xx_pwm
->base
+ reg
);
121 static inline u32
lpc18xx_pwm_readl(struct lpc18xx_pwm_chip
*lpc18xx_pwm
,
124 return readl(lpc18xx_pwm
->base
+ reg
);
127 static void lpc18xx_pwm_set_conflict_res(struct lpc18xx_pwm_chip
*lpc18xx_pwm
,
128 struct pwm_device
*pwm
,
129 enum lpc18xx_pwm_res_action action
)
133 mutex_lock(&lpc18xx_pwm
->res_lock
);
136 * Simultaneous set and clear may happen on an output, that is the case
137 * when duty_ns == period_ns. LPC18xx SCT allows to set a conflict
138 * resolution action to be taken in such a case.
140 val
= lpc18xx_pwm_readl(lpc18xx_pwm
, LPC18XX_PWM_RES_BASE
);
141 val
&= ~LPC18XX_PWM_RES_MASK(pwm
->hwpwm
);
142 val
|= LPC18XX_PWM_RES(pwm
->hwpwm
, action
);
143 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_RES_BASE
, val
);
145 mutex_unlock(&lpc18xx_pwm
->res_lock
);
148 static void lpc18xx_pwm_config_period(struct pwm_chip
*chip
, int period_ns
)
150 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
153 val
= (u64
)period_ns
* lpc18xx_pwm
->clk_rate
;
154 do_div(val
, NSEC_PER_SEC
);
156 lpc18xx_pwm_writel(lpc18xx_pwm
,
157 LPC18XX_PWM_MATCH(lpc18xx_pwm
->period_event
),
160 lpc18xx_pwm_writel(lpc18xx_pwm
,
161 LPC18XX_PWM_MATCHREL(lpc18xx_pwm
->period_event
),
165 static void lpc18xx_pwm_config_duty(struct pwm_chip
*chip
,
166 struct pwm_device
*pwm
, int duty_ns
)
168 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
169 struct lpc18xx_pwm_data
*lpc18xx_data
= pwm_get_chip_data(pwm
);
172 val
= (u64
)duty_ns
* lpc18xx_pwm
->clk_rate
;
173 do_div(val
, NSEC_PER_SEC
);
175 lpc18xx_pwm_writel(lpc18xx_pwm
,
176 LPC18XX_PWM_MATCH(lpc18xx_data
->duty_event
),
179 lpc18xx_pwm_writel(lpc18xx_pwm
,
180 LPC18XX_PWM_MATCHREL(lpc18xx_data
->duty_event
),
184 static int lpc18xx_pwm_config(struct pwm_chip
*chip
, struct pwm_device
*pwm
,
185 int duty_ns
, int period_ns
)
187 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
188 int requested_events
, i
;
190 if (period_ns
< lpc18xx_pwm
->min_period_ns
||
191 period_ns
> lpc18xx_pwm
->max_period_ns
) {
192 dev_err(chip
->dev
, "period %d not in range\n", period_ns
);
196 mutex_lock(&lpc18xx_pwm
->period_lock
);
198 requested_events
= bitmap_weight(&lpc18xx_pwm
->event_map
,
199 LPC18XX_PWM_EVENT_MAX
);
202 * The PWM supports only a single period for all PWM channels.
203 * Once the period is set, it can only be changed if no more than one
204 * channel is requested at that moment.
206 if (requested_events
> 2 && lpc18xx_pwm
->period_ns
!= period_ns
&&
207 lpc18xx_pwm
->period_ns
) {
208 dev_err(chip
->dev
, "conflicting period requested for PWM %u\n",
210 mutex_unlock(&lpc18xx_pwm
->period_lock
);
214 if ((requested_events
<= 2 && lpc18xx_pwm
->period_ns
!= period_ns
) ||
215 !lpc18xx_pwm
->period_ns
) {
216 lpc18xx_pwm
->period_ns
= period_ns
;
217 for (i
= 0; i
< chip
->npwm
; i
++)
218 pwm_set_period(&chip
->pwms
[i
], period_ns
);
219 lpc18xx_pwm_config_period(chip
, period_ns
);
222 mutex_unlock(&lpc18xx_pwm
->period_lock
);
224 lpc18xx_pwm_config_duty(chip
, pwm
, duty_ns
);
229 static int lpc18xx_pwm_set_polarity(struct pwm_chip
*chip
,
230 struct pwm_device
*pwm
,
231 enum pwm_polarity polarity
)
236 static int lpc18xx_pwm_enable(struct pwm_chip
*chip
, struct pwm_device
*pwm
)
238 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
239 struct lpc18xx_pwm_data
*lpc18xx_data
= pwm_get_chip_data(pwm
);
240 enum lpc18xx_pwm_res_action res_action
;
241 unsigned int set_event
, clear_event
;
243 lpc18xx_pwm_writel(lpc18xx_pwm
,
244 LPC18XX_PWM_EVCTRL(lpc18xx_data
->duty_event
),
245 LPC18XX_PWM_EVCTRL_MATCH(lpc18xx_data
->duty_event
) |
246 LPC18XX_PWM_EVCTRL_COMB_MATCH
);
248 lpc18xx_pwm_writel(lpc18xx_pwm
,
249 LPC18XX_PWM_EVSTATEMSK(lpc18xx_data
->duty_event
),
250 LPC18XX_PWM_EVSTATEMSK_ALL
);
252 if (pwm
->polarity
== PWM_POLARITY_NORMAL
) {
253 set_event
= lpc18xx_pwm
->period_event
;
254 clear_event
= lpc18xx_data
->duty_event
;
255 res_action
= LPC18XX_PWM_RES_SET
;
257 set_event
= lpc18xx_data
->duty_event
;
258 clear_event
= lpc18xx_pwm
->period_event
;
259 res_action
= LPC18XX_PWM_RES_CLEAR
;
262 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_OUTPUTSET(pwm
->hwpwm
),
264 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_OUTPUTCL(pwm
->hwpwm
),
266 lpc18xx_pwm_set_conflict_res(lpc18xx_pwm
, pwm
, res_action
);
271 static void lpc18xx_pwm_disable(struct pwm_chip
*chip
, struct pwm_device
*pwm
)
273 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
274 struct lpc18xx_pwm_data
*lpc18xx_data
= pwm_get_chip_data(pwm
);
276 lpc18xx_pwm_writel(lpc18xx_pwm
,
277 LPC18XX_PWM_EVCTRL(lpc18xx_data
->duty_event
), 0);
278 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_OUTPUTSET(pwm
->hwpwm
), 0);
279 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_OUTPUTCL(pwm
->hwpwm
), 0);
282 static int lpc18xx_pwm_request(struct pwm_chip
*chip
, struct pwm_device
*pwm
)
284 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
285 struct lpc18xx_pwm_data
*lpc18xx_data
= pwm_get_chip_data(pwm
);
288 event
= find_first_zero_bit(&lpc18xx_pwm
->event_map
,
289 LPC18XX_PWM_EVENT_MAX
);
291 if (event
>= LPC18XX_PWM_EVENT_MAX
) {
292 dev_err(lpc18xx_pwm
->dev
,
293 "maximum number of simultaneous channels reached\n");
297 set_bit(event
, &lpc18xx_pwm
->event_map
);
298 lpc18xx_data
->duty_event
= event
;
299 lpc18xx_pwm_config_duty(chip
, pwm
, pwm_get_duty_cycle(pwm
));
304 static void lpc18xx_pwm_free(struct pwm_chip
*chip
, struct pwm_device
*pwm
)
306 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= to_lpc18xx_pwm_chip(chip
);
307 struct lpc18xx_pwm_data
*lpc18xx_data
= pwm_get_chip_data(pwm
);
310 pwm_set_duty_cycle(pwm
, 0);
311 clear_bit(lpc18xx_data
->duty_event
, &lpc18xx_pwm
->event_map
);
314 static const struct pwm_ops lpc18xx_pwm_ops
= {
315 .config
= lpc18xx_pwm_config
,
316 .set_polarity
= lpc18xx_pwm_set_polarity
,
317 .enable
= lpc18xx_pwm_enable
,
318 .disable
= lpc18xx_pwm_disable
,
319 .request
= lpc18xx_pwm_request
,
320 .free
= lpc18xx_pwm_free
,
321 .owner
= THIS_MODULE
,
324 static const struct of_device_id lpc18xx_pwm_of_match
[] = {
325 { .compatible
= "nxp,lpc1850-sct-pwm" },
328 MODULE_DEVICE_TABLE(of
, lpc18xx_pwm_of_match
);
330 static int lpc18xx_pwm_probe(struct platform_device
*pdev
)
332 struct lpc18xx_pwm_chip
*lpc18xx_pwm
;
333 struct pwm_device
*pwm
;
334 struct resource
*res
;
338 lpc18xx_pwm
= devm_kzalloc(&pdev
->dev
, sizeof(*lpc18xx_pwm
),
343 lpc18xx_pwm
->dev
= &pdev
->dev
;
345 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
346 lpc18xx_pwm
->base
= devm_ioremap_resource(&pdev
->dev
, res
);
347 if (IS_ERR(lpc18xx_pwm
->base
))
348 return PTR_ERR(lpc18xx_pwm
->base
);
350 lpc18xx_pwm
->pwm_clk
= devm_clk_get(&pdev
->dev
, "pwm");
351 if (IS_ERR(lpc18xx_pwm
->pwm_clk
)) {
352 dev_err(&pdev
->dev
, "failed to get pwm clock\n");
353 return PTR_ERR(lpc18xx_pwm
->pwm_clk
);
356 ret
= clk_prepare_enable(lpc18xx_pwm
->pwm_clk
);
358 dev_err(&pdev
->dev
, "could not prepare or enable pwm clock\n");
362 lpc18xx_pwm
->clk_rate
= clk_get_rate(lpc18xx_pwm
->pwm_clk
);
364 mutex_init(&lpc18xx_pwm
->res_lock
);
365 mutex_init(&lpc18xx_pwm
->period_lock
);
367 val
= (u64
)NSEC_PER_SEC
* LPC18XX_PWM_TIMER_MAX
;
368 do_div(val
, lpc18xx_pwm
->clk_rate
);
369 lpc18xx_pwm
->max_period_ns
= val
;
371 lpc18xx_pwm
->min_period_ns
= DIV_ROUND_UP(NSEC_PER_SEC
,
372 lpc18xx_pwm
->clk_rate
);
374 lpc18xx_pwm
->chip
.dev
= &pdev
->dev
;
375 lpc18xx_pwm
->chip
.ops
= &lpc18xx_pwm_ops
;
376 lpc18xx_pwm
->chip
.base
= -1;
377 lpc18xx_pwm
->chip
.npwm
= 16;
378 lpc18xx_pwm
->chip
.of_xlate
= of_pwm_xlate_with_flags
;
379 lpc18xx_pwm
->chip
.of_pwm_n_cells
= 3;
381 /* SCT counter must be in unify (32 bit) mode */
382 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_CONFIG
,
383 LPC18XX_PWM_CONFIG_UNIFY
);
386 * Everytime the timer counter reaches the period value, the related
387 * event will be triggered and the counter reset to 0.
389 set_bit(LPC18XX_PWM_EVENT_PERIOD
, &lpc18xx_pwm
->event_map
);
390 lpc18xx_pwm
->period_event
= LPC18XX_PWM_EVENT_PERIOD
;
392 lpc18xx_pwm_writel(lpc18xx_pwm
,
393 LPC18XX_PWM_EVSTATEMSK(lpc18xx_pwm
->period_event
),
394 LPC18XX_PWM_EVSTATEMSK_ALL
);
396 val
= LPC18XX_PWM_EVCTRL_MATCH(lpc18xx_pwm
->period_event
) |
397 LPC18XX_PWM_EVCTRL_COMB_MATCH
;
398 lpc18xx_pwm_writel(lpc18xx_pwm
,
399 LPC18XX_PWM_EVCTRL(lpc18xx_pwm
->period_event
), val
);
401 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_LIMIT
,
402 BIT(lpc18xx_pwm
->period_event
));
404 ret
= pwmchip_add(&lpc18xx_pwm
->chip
);
406 dev_err(&pdev
->dev
, "pwmchip_add failed: %d\n", ret
);
410 for (i
= 0; i
< lpc18xx_pwm
->chip
.npwm
; i
++) {
411 pwm
= &lpc18xx_pwm
->chip
.pwms
[i
];
412 pwm
->chip_data
= devm_kzalloc(lpc18xx_pwm
->dev
,
413 sizeof(struct lpc18xx_pwm_data
),
415 if (!pwm
->chip_data
) {
421 platform_set_drvdata(pdev
, lpc18xx_pwm
);
423 val
= lpc18xx_pwm_readl(lpc18xx_pwm
, LPC18XX_PWM_CTRL
);
424 val
&= ~LPC18XX_PWM_BIDIR
;
425 val
&= ~LPC18XX_PWM_CTRL_HALT
;
426 val
&= ~LPC18XX_PWM_PRE_MASK
;
427 val
|= LPC18XX_PWM_PRE(0);
428 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_CTRL
, val
);
433 pwmchip_remove(&lpc18xx_pwm
->chip
);
435 clk_disable_unprepare(lpc18xx_pwm
->pwm_clk
);
439 static int lpc18xx_pwm_remove(struct platform_device
*pdev
)
441 struct lpc18xx_pwm_chip
*lpc18xx_pwm
= platform_get_drvdata(pdev
);
444 val
= lpc18xx_pwm_readl(lpc18xx_pwm
, LPC18XX_PWM_CTRL
);
445 lpc18xx_pwm_writel(lpc18xx_pwm
, LPC18XX_PWM_CTRL
,
446 val
| LPC18XX_PWM_CTRL_HALT
);
448 clk_disable_unprepare(lpc18xx_pwm
->pwm_clk
);
450 return pwmchip_remove(&lpc18xx_pwm
->chip
);
453 static struct platform_driver lpc18xx_pwm_driver
= {
455 .name
= "lpc18xx-sct-pwm",
456 .of_match_table
= lpc18xx_pwm_of_match
,
458 .probe
= lpc18xx_pwm_probe
,
459 .remove
= lpc18xx_pwm_remove
,
461 module_platform_driver(lpc18xx_pwm_driver
);
463 MODULE_AUTHOR("Ariel D'Alessandro <ariel@vanguardiasur.com.ar>");
464 MODULE_DESCRIPTION("NXP LPC18xx PWM driver");
465 MODULE_LICENSE("GPL v2");