1 // SPDX-License-Identifier: GPL-2.0-only
3 * drivers/pwm/pwm-vt8500.c
5 * Copyright (C) 2012 Tony Prisk <linux@prisktech.co.nz>
6 * Copyright (C) 2010 Alexey Charkov <alchark@gmail.com>
9 #include <linux/module.h>
10 #include <linux/kernel.h>
11 #include <linux/platform_device.h>
12 #include <linux/slab.h>
13 #include <linux/err.h>
15 #include <linux/pwm.h>
16 #include <linux/delay.h>
17 #include <linux/clk.h>
19 #include <asm/div64.h>
22 #include <linux/of_device.h>
23 #include <linux/of_address.h>
26 * SoC architecture allocates register space for 4 PWMs but only
27 * 2 are currently implemented.
29 #define VT8500_NR_PWMS 2
31 #define REG_CTRL(pwm) (((pwm) << 4) + 0x00)
32 #define REG_SCALAR(pwm) (((pwm) << 4) + 0x04)
33 #define REG_PERIOD(pwm) (((pwm) << 4) + 0x08)
34 #define REG_DUTY(pwm) (((pwm) << 4) + 0x0C)
35 #define REG_STATUS 0x40
37 #define CTRL_ENABLE BIT(0)
38 #define CTRL_INVERT BIT(1)
39 #define CTRL_AUTOLOAD BIT(2)
40 #define CTRL_STOP_IMM BIT(3)
41 #define CTRL_LOAD_PRESCALE BIT(4)
42 #define CTRL_LOAD_PERIOD BIT(5)
44 #define STATUS_CTRL_UPDATE BIT(0)
45 #define STATUS_SCALAR_UPDATE BIT(1)
46 #define STATUS_PERIOD_UPDATE BIT(2)
47 #define STATUS_DUTY_UPDATE BIT(3)
48 #define STATUS_ALL_UPDATE 0x0F
56 #define to_vt8500_chip(chip) container_of(chip, struct vt8500_chip, chip)
58 #define msecs_to_loops(t) (loops_per_jiffy / 1000 * HZ * t)
59 static inline void pwm_busy_wait(struct vt8500_chip
*vt8500
, int nr
, u8 bitmask
)
61 int loops
= msecs_to_loops(10);
62 u32 mask
= bitmask
<< (nr
<< 8);
64 while ((readl(vt8500
->base
+ REG_STATUS
) & mask
) && --loops
)
68 dev_warn(vt8500
->chip
.dev
, "Waiting for status bits 0x%x to clear timed out\n",
72 static int vt8500_pwm_config(struct pwm_chip
*chip
, struct pwm_device
*pwm
,
73 int duty_ns
, int period_ns
)
75 struct vt8500_chip
*vt8500
= to_vt8500_chip(chip
);
77 unsigned long period_cycles
, prescale
, pv
, dc
;
81 err
= clk_enable(vt8500
->clk
);
83 dev_err(chip
->dev
, "failed to enable clock\n");
87 c
= clk_get_rate(vt8500
->clk
);
89 do_div(c
, 1000000000);
92 if (period_cycles
< 1)
94 prescale
= (period_cycles
- 1) / 4096;
95 pv
= period_cycles
/ (prescale
+ 1) - 1;
99 if (prescale
> 1023) {
100 clk_disable(vt8500
->clk
);
104 c
= (unsigned long long)pv
* duty_ns
;
105 do_div(c
, period_ns
);
108 writel(prescale
, vt8500
->base
+ REG_SCALAR(pwm
->hwpwm
));
109 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_SCALAR_UPDATE
);
111 writel(pv
, vt8500
->base
+ REG_PERIOD(pwm
->hwpwm
));
112 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_PERIOD_UPDATE
);
114 writel(dc
, vt8500
->base
+ REG_DUTY(pwm
->hwpwm
));
115 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_DUTY_UPDATE
);
117 val
= readl(vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
118 val
|= CTRL_AUTOLOAD
;
119 writel(val
, vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
120 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_CTRL_UPDATE
);
122 clk_disable(vt8500
->clk
);
126 static int vt8500_pwm_enable(struct pwm_chip
*chip
, struct pwm_device
*pwm
)
128 struct vt8500_chip
*vt8500
= to_vt8500_chip(chip
);
132 err
= clk_enable(vt8500
->clk
);
134 dev_err(chip
->dev
, "failed to enable clock\n");
138 val
= readl(vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
140 writel(val
, vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
141 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_CTRL_UPDATE
);
146 static void vt8500_pwm_disable(struct pwm_chip
*chip
, struct pwm_device
*pwm
)
148 struct vt8500_chip
*vt8500
= to_vt8500_chip(chip
);
151 val
= readl(vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
153 writel(val
, vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
154 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_CTRL_UPDATE
);
156 clk_disable(vt8500
->clk
);
159 static int vt8500_pwm_set_polarity(struct pwm_chip
*chip
,
160 struct pwm_device
*pwm
,
161 enum pwm_polarity polarity
)
163 struct vt8500_chip
*vt8500
= to_vt8500_chip(chip
);
166 val
= readl(vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
168 if (polarity
== PWM_POLARITY_INVERSED
)
173 writel(val
, vt8500
->base
+ REG_CTRL(pwm
->hwpwm
));
174 pwm_busy_wait(vt8500
, pwm
->hwpwm
, STATUS_CTRL_UPDATE
);
179 static const struct pwm_ops vt8500_pwm_ops
= {
180 .enable
= vt8500_pwm_enable
,
181 .disable
= vt8500_pwm_disable
,
182 .config
= vt8500_pwm_config
,
183 .set_polarity
= vt8500_pwm_set_polarity
,
184 .owner
= THIS_MODULE
,
187 static const struct of_device_id vt8500_pwm_dt_ids
[] = {
188 { .compatible
= "via,vt8500-pwm", },
191 MODULE_DEVICE_TABLE(of
, vt8500_pwm_dt_ids
);
193 static int vt8500_pwm_probe(struct platform_device
*pdev
)
195 struct vt8500_chip
*chip
;
197 struct device_node
*np
= pdev
->dev
.of_node
;
201 dev_err(&pdev
->dev
, "invalid devicetree node\n");
205 chip
= devm_kzalloc(&pdev
->dev
, sizeof(*chip
), GFP_KERNEL
);
209 chip
->chip
.dev
= &pdev
->dev
;
210 chip
->chip
.ops
= &vt8500_pwm_ops
;
211 chip
->chip
.of_xlate
= of_pwm_xlate_with_flags
;
212 chip
->chip
.of_pwm_n_cells
= 3;
213 chip
->chip
.base
= -1;
214 chip
->chip
.npwm
= VT8500_NR_PWMS
;
216 chip
->clk
= devm_clk_get(&pdev
->dev
, NULL
);
217 if (IS_ERR(chip
->clk
)) {
218 dev_err(&pdev
->dev
, "clock source not specified\n");
219 return PTR_ERR(chip
->clk
);
222 r
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
223 chip
->base
= devm_ioremap_resource(&pdev
->dev
, r
);
224 if (IS_ERR(chip
->base
))
225 return PTR_ERR(chip
->base
);
227 ret
= clk_prepare(chip
->clk
);
229 dev_err(&pdev
->dev
, "failed to prepare clock\n");
233 ret
= pwmchip_add(&chip
->chip
);
235 dev_err(&pdev
->dev
, "failed to add PWM chip\n");
236 clk_unprepare(chip
->clk
);
240 platform_set_drvdata(pdev
, chip
);
244 static int vt8500_pwm_remove(struct platform_device
*pdev
)
246 struct vt8500_chip
*chip
;
248 chip
= platform_get_drvdata(pdev
);
252 clk_unprepare(chip
->clk
);
254 return pwmchip_remove(&chip
->chip
);
257 static struct platform_driver vt8500_pwm_driver
= {
258 .probe
= vt8500_pwm_probe
,
259 .remove
= vt8500_pwm_remove
,
261 .name
= "vt8500-pwm",
262 .of_match_table
= vt8500_pwm_dt_ids
,
265 module_platform_driver(vt8500_pwm_driver
);
267 MODULE_DESCRIPTION("VT8500 PWM Driver");
268 MODULE_AUTHOR("Tony Prisk <linux@prisktech.co.nz>");
269 MODULE_LICENSE("GPL v2");