aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/plat-nomadik
diff options
context:
space:
mode:
authorRabin Vincent <rabin.vincent@stericsson.com>2010-03-02 22:52:34 -0500
committerRussell King <rmk+kernel@arm.linux.org.uk>2010-03-19 14:29:32 -0400
commit3e3c62ca5395df48319b808379bc9fd487ff3c29 (patch)
treee4f9146aefc7401fc5e201c2f7bb1c0f92ea65f1 /arch/arm/plat-nomadik
parentaaedaa2b5c610ae97f863078075d8d3c6ef91575 (diff)
ARM: 5972/1: nomadik-gpio: convert to platform driver
On the U8500 platform there are four GPIO blocks, each with a 4K address space, including the peripheral identification. However, each of these blocks have a varying number of banks, each of which have 32 GPIOs and an interrupt line. The current nomadik-gpio driver implementation can handle each of these sub-banks easily with one instance each, but cannot as-is be hooked up to them because it is an AMBA driver and it expects to see a peripheral with the appropriate peripheral ids but having only one bank and only one interrupt. Solve this by converting the driver to a platform driver. Acked-by: Alessandro Rubini <rubini@unipv.it> Acked-by: Linus Walleij <linus.walleij@stericsson.com> Signed-off-by: Rabin Vincent <rabin.vincent@stericsson.com> Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Diffstat (limited to 'arch/arm/plat-nomadik')
-rw-r--r--arch/arm/plat-nomadik/gpio.c74
-rw-r--r--arch/arm/plat-nomadik/include/plat/gpio.h1
2 files changed, 43 insertions, 32 deletions
diff --git a/arch/arm/plat-nomadik/gpio.c b/arch/arm/plat-nomadik/gpio.c
index 4c3ea1a922ac..092f380063b3 100644
--- a/arch/arm/plat-nomadik/gpio.c
+++ b/arch/arm/plat-nomadik/gpio.c
@@ -13,7 +13,7 @@
13#include <linux/module.h> 13#include <linux/module.h>
14#include <linux/init.h> 14#include <linux/init.h>
15#include <linux/device.h> 15#include <linux/device.h>
16#include <linux/amba/bus.h> 16#include <linux/platform_device.h>
17#include <linux/io.h> 17#include <linux/io.h>
18#include <linux/gpio.h> 18#include <linux/gpio.h>
19#include <linux/spinlock.h> 19#include <linux/spinlock.h>
@@ -303,30 +303,48 @@ static struct gpio_chip nmk_gpio_template = {
303 .can_sleep = 0, 303 .can_sleep = 0,
304}; 304};
305 305
306static int __init nmk_gpio_probe(struct amba_device *dev, struct amba_id *id) 306static int __init nmk_gpio_probe(struct platform_device *dev)
307{ 307{
308 struct nmk_gpio_platform_data *pdata; 308 struct nmk_gpio_platform_data *pdata = dev->dev.platform_data;
309 struct nmk_gpio_chip *nmk_chip; 309 struct nmk_gpio_chip *nmk_chip;
310 struct gpio_chip *chip; 310 struct gpio_chip *chip;
311 struct resource *res;
312 int irq;
311 int ret; 313 int ret;
312 314
313 pdata = dev->dev.platform_data; 315 if (!pdata)
314 ret = amba_request_regions(dev, pdata->name); 316 return -ENODEV;
315 if (ret) 317
316 return ret; 318 res = platform_get_resource(dev, IORESOURCE_MEM, 0);
319 if (!res) {
320 ret = -ENOENT;
321 goto out;
322 }
323
324 irq = platform_get_irq(dev, 0);
325 if (irq < 0) {
326 ret = irq;
327 goto out;
328 }
329
330 if (request_mem_region(res->start, resource_size(res),
331 dev_name(&dev->dev)) == NULL) {
332 ret = -EBUSY;
333 goto out;
334 }
317 335
318 nmk_chip = kzalloc(sizeof(*nmk_chip), GFP_KERNEL); 336 nmk_chip = kzalloc(sizeof(*nmk_chip), GFP_KERNEL);
319 if (!nmk_chip) { 337 if (!nmk_chip) {
320 ret = -ENOMEM; 338 ret = -ENOMEM;
321 goto out_amba; 339 goto out_release;
322 } 340 }
323 /* 341 /*
324 * The virt address in nmk_chip->addr is in the nomadik register space, 342 * The virt address in nmk_chip->addr is in the nomadik register space,
325 * so we can simply convert the resource address, without remapping 343 * so we can simply convert the resource address, without remapping
326 */ 344 */
327 nmk_chip->addr = io_p2v(dev->res.start); 345 nmk_chip->addr = io_p2v(res->start);
328 nmk_chip->chip = nmk_gpio_template; 346 nmk_chip->chip = nmk_gpio_template;
329 nmk_chip->parent_irq = pdata->parent_irq; 347 nmk_chip->parent_irq = irq;
330 spin_lock_init(&nmk_chip->lock); 348 spin_lock_init(&nmk_chip->lock);
331 349
332 chip = &nmk_chip->chip; 350 chip = &nmk_chip->chip;
@@ -339,7 +357,7 @@ static int __init nmk_gpio_probe(struct amba_device *dev, struct amba_id *id)
339 if (ret) 357 if (ret)
340 goto out_free; 358 goto out_free;
341 359
342 amba_set_drvdata(dev, nmk_chip); 360 platform_set_drvdata(dev, nmk_chip);
343 361
344 nmk_gpio_init_irq(nmk_chip); 362 nmk_gpio_init_irq(nmk_chip);
345 363
@@ -347,51 +365,45 @@ static int __init nmk_gpio_probe(struct amba_device *dev, struct amba_id *id)
347 nmk_chip->chip.base, nmk_chip->chip.base+31, nmk_chip->addr); 365 nmk_chip->chip.base, nmk_chip->chip.base+31, nmk_chip->addr);
348 return 0; 366 return 0;
349 367
350 out_free: 368out_free:
351 kfree(nmk_chip); 369 kfree(nmk_chip);
352 out_amba: 370out_release:
353 amba_release_regions(dev); 371 release_mem_region(res->start, resource_size(res));
372out:
354 dev_err(&dev->dev, "Failure %i for GPIO %i-%i\n", ret, 373 dev_err(&dev->dev, "Failure %i for GPIO %i-%i\n", ret,
355 pdata->first_gpio, pdata->first_gpio+31); 374 pdata->first_gpio, pdata->first_gpio+31);
356 return ret; 375 return ret;
357} 376}
358 377
359static int nmk_gpio_remove(struct amba_device *dev) 378static int __exit nmk_gpio_remove(struct platform_device *dev)
360{ 379{
361 struct nmk_gpio_chip *nmk_chip; 380 struct nmk_gpio_chip *nmk_chip;
381 struct resource *res;
382
383 res = platform_get_resource(dev, IORESOURCE_MEM, 0);
362 384
363 nmk_chip = amba_get_drvdata(dev); 385 nmk_chip = platform_get_drvdata(dev);
364 gpiochip_remove(&nmk_chip->chip); 386 gpiochip_remove(&nmk_chip->chip);
365 kfree(nmk_chip); 387 kfree(nmk_chip);
366 amba_release_regions(dev); 388 release_mem_region(res->start, resource_size(res));
367 return 0; 389 return 0;
368} 390}
369 391
370 392
371/* We have 0x1f080060 and 0x1f180060, accept both using the mask */ 393static struct platform_driver nmk_gpio_driver = {
372static struct amba_id nmk_gpio_ids[] = { 394 .driver = {
373 {
374 .id = 0x1f080060,
375 .mask = 0xffefffff,
376 },
377 {0, 0},
378};
379
380static struct amba_driver nmk_gpio_driver = {
381 .drv = {
382 .owner = THIS_MODULE, 395 .owner = THIS_MODULE,
383 .name = "gpio", 396 .name = "gpio",
384 }, 397 },
385 .probe = nmk_gpio_probe, 398 .probe = nmk_gpio_probe,
386 .remove = nmk_gpio_remove, 399 .remove = __exit_p(nmk_gpio_remove),
387 .suspend = NULL, /* to be done */ 400 .suspend = NULL, /* to be done */
388 .resume = NULL, 401 .resume = NULL,
389 .id_table = nmk_gpio_ids,
390}; 402};
391 403
392static int __init nmk_gpio_init(void) 404static int __init nmk_gpio_init(void)
393{ 405{
394 return amba_driver_register(&nmk_gpio_driver); 406 return platform_driver_register(&nmk_gpio_driver);
395} 407}
396 408
397arch_initcall(nmk_gpio_init); 409arch_initcall(nmk_gpio_init);
diff --git a/arch/arm/plat-nomadik/include/plat/gpio.h b/arch/arm/plat-nomadik/include/plat/gpio.h
index 1d665a0abb87..4200811249ca 100644
--- a/arch/arm/plat-nomadik/include/plat/gpio.h
+++ b/arch/arm/plat-nomadik/include/plat/gpio.h
@@ -65,7 +65,6 @@ struct nmk_gpio_platform_data {
65 char *name; 65 char *name;
66 int first_gpio; 66 int first_gpio;
67 int first_irq; 67 int first_irq;
68 int parent_irq;
69}; 68};
70 69
71#endif /* __ASM_PLAT_GPIO_H */ 70#endif /* __ASM_PLAT_GPIO_H */