Blame view
arch/arm/plat-pxa/pwm.c
6.16 KB
75540c1ac [ARM] pxa: Add PX... |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
/* * linux/arch/arm/mach-pxa/pwm.c * * simple driver for PWM (Pulse Width Modulator) controller * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * 2008-02-13 initial version * eric miao <eric.miao@marvell.com> */ #include <linux/module.h> #include <linux/kernel.h> #include <linux/platform_device.h> |
5a0e3ad6a include cleanup: ... |
17 |
#include <linux/slab.h> |
75540c1ac [ARM] pxa: Add PX... |
18 19 20 21 22 23 |
#include <linux/err.h> #include <linux/clk.h> #include <linux/io.h> #include <linux/pwm.h> #include <asm/div64.h> |
75540c1ac [ARM] pxa: Add PX... |
24 |
|
3d2a98cd5 [ARM] pxa: simpli... |
25 |
#define HAS_SECONDARY_PWM 0x10 |
a757ad8b3 [ARM] pxa: allow ... |
26 |
#define PWM_ID_BASE(d) ((d) & 0xf) |
3d2a98cd5 [ARM] pxa: simpli... |
27 28 29 30 |
static const struct platform_device_id pwm_id_table[] = { /* PWM has_secondary_pwm? */ { "pxa25x-pwm", 0 }, |
a757ad8b3 [ARM] pxa: allow ... |
31 |
{ "pxa27x-pwm", 0 | HAS_SECONDARY_PWM }, |
a27ba768a [ARM] pxa: add PW... |
32 33 |
{ "pxa168-pwm", 1 }, { "pxa910-pwm", 1 }, |
3d2a98cd5 [ARM] pxa: simpli... |
34 35 36 |
{ }, }; MODULE_DEVICE_TABLE(platform, pwm_id_table); |
75540c1ac [ARM] pxa: Add PX... |
37 38 39 40 41 42 43 44 45 46 |
/* PWM registers and bits definitions */ #define PWMCR (0x00) #define PWMDCR (0x04) #define PWMPCR (0x08) #define PWMCR_SD (1 << 6) #define PWMDCR_FD (1 << 10) struct pwm_device { struct list_head node; |
3d2a98cd5 [ARM] pxa: simpli... |
47 48 |
struct pwm_device *secondary; struct platform_device *pdev; |
75540c1ac [ARM] pxa: Add PX... |
49 50 51 |
const char *label; struct clk *clk; |
c860d701c [ARM] 5087/1: Get... |
52 |
int clk_enabled; |
75540c1ac [ARM] pxa: Add PX... |
53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 |
void __iomem *mmio_base; unsigned int use_count; unsigned int pwm_id; }; /* * period_ns = 10^9 * (PRESCALE + 1) * (PV + 1) / PWM_CLK_RATE * duty_ns = 10^9 * (PRESCALE + 1) * DC / PWM_CLK_RATE */ int pwm_config(struct pwm_device *pwm, int duty_ns, int period_ns) { unsigned long long c; unsigned long period_cycles, prescale, pv, dc; if (pwm == NULL || period_ns == 0 || duty_ns > period_ns) return -EINVAL; c = clk_get_rate(pwm->clk); c = c * period_ns; do_div(c, 1000000000); period_cycles = c; |
71a35d756 [ARM] 5303/1: per... |
75 |
if (period_cycles < 1) |
75540c1ac [ARM] pxa: Add PX... |
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 |
period_cycles = 1; prescale = (period_cycles - 1) / 1024; pv = period_cycles / (prescale + 1) - 1; if (prescale > 63) return -EINVAL; if (duty_ns == period_ns) dc = PWMDCR_FD; else dc = (pv + 1) * duty_ns / period_ns; /* NOTE: the clock to PWM has to be enabled first * before writing to the registers */ clk_enable(pwm->clk); __raw_writel(prescale, pwm->mmio_base + PWMCR); __raw_writel(dc, pwm->mmio_base + PWMDCR); __raw_writel(pv, pwm->mmio_base + PWMPCR); clk_disable(pwm->clk); return 0; } EXPORT_SYMBOL(pwm_config); int pwm_enable(struct pwm_device *pwm) { |
c860d701c [ARM] 5087/1: Get... |
103 104 105 106 107 108 109 110 |
int rc = 0; if (!pwm->clk_enabled) { rc = clk_enable(pwm->clk); if (!rc) pwm->clk_enabled = 1; } return rc; |
75540c1ac [ARM] pxa: Add PX... |
111 112 113 114 115 |
} EXPORT_SYMBOL(pwm_enable); void pwm_disable(struct pwm_device *pwm) { |
c860d701c [ARM] 5087/1: Get... |
116 117 118 119 |
if (pwm->clk_enabled) { clk_disable(pwm->clk); pwm->clk_enabled = 0; } |
75540c1ac [ARM] pxa: Add PX... |
120 121 122 123 124 125 126 127 128 129 130 131 132 133 |
} EXPORT_SYMBOL(pwm_disable); static DEFINE_MUTEX(pwm_lock); static LIST_HEAD(pwm_list); struct pwm_device *pwm_request(int pwm_id, const char *label) { struct pwm_device *pwm; int found = 0; mutex_lock(&pwm_lock); list_for_each_entry(pwm, &pwm_list, node) { |
43bda1a6d [ARM] 5141/1: PWM... |
134 |
if (pwm->pwm_id == pwm_id) { |
75540c1ac [ARM] pxa: Add PX... |
135 136 137 138 |
found = 1; break; } } |
43bda1a6d [ARM] 5141/1: PWM... |
139 140 141 142 143 144 145 146 |
if (found) { if (pwm->use_count == 0) { pwm->use_count++; pwm->label = label; } else pwm = ERR_PTR(-EBUSY); } else pwm = ERR_PTR(-ENOENT); |
75540c1ac [ARM] pxa: Add PX... |
147 |
|
43bda1a6d [ARM] 5141/1: PWM... |
148 149 |
mutex_unlock(&pwm_lock); return pwm; |
75540c1ac [ARM] pxa: Add PX... |
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 |
} EXPORT_SYMBOL(pwm_request); void pwm_free(struct pwm_device *pwm) { mutex_lock(&pwm_lock); if (pwm->use_count) { pwm->use_count--; pwm->label = NULL; } else pr_warning("PWM device already freed "); mutex_unlock(&pwm_lock); } EXPORT_SYMBOL(pwm_free); static inline void __add_pwm(struct pwm_device *pwm) { mutex_lock(&pwm_lock); list_add_tail(&pwm->node, &pwm_list); mutex_unlock(&pwm_lock); } |
3d2a98cd5 [ARM] pxa: simpli... |
174 |
static int __devinit pwm_probe(struct platform_device *pdev) |
75540c1ac [ARM] pxa: Add PX... |
175 |
{ |
b3282ab11 ARM: pxa: Make id... |
176 |
const struct platform_device_id *id = platform_get_device_id(pdev); |
3d2a98cd5 [ARM] pxa: simpli... |
177 |
struct pwm_device *pwm, *secondary = NULL; |
75540c1ac [ARM] pxa: Add PX... |
178 179 180 181 182 183 184 |
struct resource *r; int ret = 0; pwm = kzalloc(sizeof(struct pwm_device), GFP_KERNEL); if (pwm == NULL) { dev_err(&pdev->dev, "failed to allocate memory "); |
3d2a98cd5 [ARM] pxa: simpli... |
185 |
return -ENOMEM; |
75540c1ac [ARM] pxa: Add PX... |
186 |
} |
e0d8b13ae [ARM] pxa: don't ... |
187 |
pwm->clk = clk_get(&pdev->dev, NULL); |
75540c1ac [ARM] pxa: Add PX... |
188 189 190 191 |
if (IS_ERR(pwm->clk)) { ret = PTR_ERR(pwm->clk); goto err_free; } |
c860d701c [ARM] 5087/1: Get... |
192 |
pwm->clk_enabled = 0; |
75540c1ac [ARM] pxa: Add PX... |
193 194 |
pwm->use_count = 0; |
a757ad8b3 [ARM] pxa: allow ... |
195 |
pwm->pwm_id = PWM_ID_BASE(id->driver_data) + pdev->id; |
75540c1ac [ARM] pxa: Add PX... |
196 |
pwm->pdev = pdev; |
75540c1ac [ARM] pxa: Add PX... |
197 198 199 200 201 202 203 |
r = platform_get_resource(pdev, IORESOURCE_MEM, 0); if (r == NULL) { dev_err(&pdev->dev, "no memory resource defined "); ret = -ENODEV; goto err_free_clk; } |
1692231cf [ARM] pxa: use re... |
204 |
r = request_mem_region(r->start, resource_size(r), pdev->name); |
75540c1ac [ARM] pxa: Add PX... |
205 206 207 208 209 210 |
if (r == NULL) { dev_err(&pdev->dev, "failed to request memory resource "); ret = -EBUSY; goto err_free_clk; } |
1692231cf [ARM] pxa: use re... |
211 |
pwm->mmio_base = ioremap(r->start, resource_size(r)); |
75540c1ac [ARM] pxa: Add PX... |
212 213 214 215 216 217 |
if (pwm->mmio_base == NULL) { dev_err(&pdev->dev, "failed to ioremap() registers "); ret = -ENODEV; goto err_free_mem; } |
3d2a98cd5 [ARM] pxa: simpli... |
218 219 220 221 222 223 224 225 226 227 228 229 230 231 |
if (id->driver_data & HAS_SECONDARY_PWM) { secondary = kzalloc(sizeof(struct pwm_device), GFP_KERNEL); if (secondary == NULL) { ret = -ENOMEM; goto err_free_mem; } *secondary = *pwm; pwm->secondary = secondary; /* registers for the second PWM has offset of 0x10 */ secondary->mmio_base = pwm->mmio_base + 0x10; secondary->pwm_id = pdev->id + 2; } |
75540c1ac [ARM] pxa: Add PX... |
232 |
__add_pwm(pwm); |
3d2a98cd5 [ARM] pxa: simpli... |
233 234 |
if (secondary) __add_pwm(secondary); |
75540c1ac [ARM] pxa: Add PX... |
235 |
platform_set_drvdata(pdev, pwm); |
3d2a98cd5 [ARM] pxa: simpli... |
236 |
return 0; |
75540c1ac [ARM] pxa: Add PX... |
237 238 |
err_free_mem: |
1692231cf [ARM] pxa: use re... |
239 |
release_mem_region(r->start, resource_size(r)); |
75540c1ac [ARM] pxa: Add PX... |
240 241 242 243 |
err_free_clk: clk_put(pwm->clk); err_free: kfree(pwm); |
3d2a98cd5 [ARM] pxa: simpli... |
244 |
return ret; |
75540c1ac [ARM] pxa: Add PX... |
245 246 247 248 249 250 251 252 253 254 255 256 |
} static int __devexit pwm_remove(struct platform_device *pdev) { struct pwm_device *pwm; struct resource *r; pwm = platform_get_drvdata(pdev); if (pwm == NULL) return -ENODEV; mutex_lock(&pwm_lock); |
3d2a98cd5 [ARM] pxa: simpli... |
257 258 259 260 261 |
if (pwm->secondary) { list_del(&pwm->secondary->node); kfree(pwm->secondary); } |
75540c1ac [ARM] pxa: Add PX... |
262 263 264 265 266 267 |
list_del(&pwm->node); mutex_unlock(&pwm_lock); iounmap(pwm->mmio_base); r = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
1692231cf [ARM] pxa: use re... |
268 |
release_mem_region(r->start, resource_size(r)); |
75540c1ac [ARM] pxa: Add PX... |
269 270 271 272 273 |
clk_put(pwm->clk); kfree(pwm); return 0; } |
3d2a98cd5 [ARM] pxa: simpli... |
274 |
static struct platform_driver pwm_driver = { |
75540c1ac [ARM] pxa: Add PX... |
275 276 |
.driver = { .name = "pxa25x-pwm", |
3d2a98cd5 [ARM] pxa: simpli... |
277 |
.owner = THIS_MODULE, |
75540c1ac [ARM] pxa: Add PX... |
278 |
}, |
3d2a98cd5 [ARM] pxa: simpli... |
279 |
.probe = pwm_probe, |
75540c1ac [ARM] pxa: Add PX... |
280 |
.remove = __devexit_p(pwm_remove), |
3d2a98cd5 [ARM] pxa: simpli... |
281 |
.id_table = pwm_id_table, |
75540c1ac [ARM] pxa: Add PX... |
282 283 284 285 |
}; static int __init pwm_init(void) { |
3d2a98cd5 [ARM] pxa: simpli... |
286 |
return platform_driver_register(&pwm_driver); |
75540c1ac [ARM] pxa: Add PX... |
287 288 289 290 291 |
} arch_initcall(pwm_init); static void __exit pwm_exit(void) { |
3d2a98cd5 [ARM] pxa: simpli... |
292 |
platform_driver_unregister(&pwm_driver); |
75540c1ac [ARM] pxa: Add PX... |
293 294 |
} module_exit(pwm_exit); |
b5f0228af [ARM] 5078/1: pxa... |
295 296 |
MODULE_LICENSE("GPL v2"); |