mirror of
https://github.com/torvalds/linux.git
synced 2024-11-27 22:51:35 +00:00
spi/pl022: use more managed resources
This switches the PL022 SPI driver to use devm_* managed resources for IRQ, clocks, ioremap and GPIO. Prior to this, the GPIOs would even leak. Signed-off-by: Linus Walleij <linus.walleij@stericsson.com> Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
This commit is contained in:
parent
8e5d0661b3
commit
aeef9915b9
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* A driver for the ARM PL022 PrimeCell SSP/SPI bus master.
|
* A driver for the ARM PL022 PrimeCell SSP/SPI bus master.
|
||||||
*
|
*
|
||||||
* Copyright (C) 2008-2009 ST-Ericsson AB
|
* Copyright (C) 2008-2012 ST-Ericsson AB
|
||||||
* Copyright (C) 2006 STMicroelectronics Pvt. Ltd.
|
* Copyright (C) 2006 STMicroelectronics Pvt. Ltd.
|
||||||
*
|
*
|
||||||
* Author: Linus Walleij <linus.walleij@stericsson.com>
|
* Author: Linus Walleij <linus.walleij@stericsson.com>
|
||||||
@ -2074,24 +2074,21 @@ pl022_probe(struct amba_device *adev, const struct amba_id *id)
|
|||||||
|
|
||||||
if (!platform_info) {
|
if (!platform_info) {
|
||||||
dev_err(dev, "probe: no platform data defined\n");
|
dev_err(dev, "probe: no platform data defined\n");
|
||||||
status = -ENODEV;
|
return -ENODEV;
|
||||||
goto err_no_pdata;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (platform_info->num_chipselect) {
|
if (platform_info->num_chipselect) {
|
||||||
num_cs = platform_info->num_chipselect;
|
num_cs = platform_info->num_chipselect;
|
||||||
} else {
|
} else {
|
||||||
dev_err(dev, "probe: no chip select defined\n");
|
dev_err(dev, "probe: no chip select defined\n");
|
||||||
status = -ENODEV;
|
return -ENODEV;
|
||||||
goto err_no_pdata;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate master with space for data */
|
/* Allocate master with space for data */
|
||||||
master = spi_alloc_master(dev, sizeof(struct pl022));
|
master = spi_alloc_master(dev, sizeof(struct pl022));
|
||||||
if (master == NULL) {
|
if (master == NULL) {
|
||||||
dev_err(&adev->dev, "probe - cannot alloc SPI master\n");
|
dev_err(&adev->dev, "probe - cannot alloc SPI master\n");
|
||||||
status = -ENOMEM;
|
return -ENOMEM;
|
||||||
goto err_no_master;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pl022 = spi_master_get_devdata(master);
|
pl022 = spi_master_get_devdata(master);
|
||||||
@ -2153,7 +2150,7 @@ pl022_probe(struct amba_device *adev, const struct amba_id *id)
|
|||||||
pl022->chipselects[i] = cs_gpio;
|
pl022->chipselects[i] = cs_gpio;
|
||||||
|
|
||||||
if (gpio_is_valid(cs_gpio)) {
|
if (gpio_is_valid(cs_gpio)) {
|
||||||
if (gpio_request(cs_gpio, "ssp-pl022"))
|
if (devm_gpio_request(dev, cs_gpio, "ssp-pl022"))
|
||||||
dev_err(&adev->dev,
|
dev_err(&adev->dev,
|
||||||
"could not request %d gpio\n",
|
"could not request %d gpio\n",
|
||||||
cs_gpio);
|
cs_gpio);
|
||||||
@ -2180,7 +2177,8 @@ pl022_probe(struct amba_device *adev, const struct amba_id *id)
|
|||||||
goto err_no_ioregion;
|
goto err_no_ioregion;
|
||||||
|
|
||||||
pl022->phybase = adev->res.start;
|
pl022->phybase = adev->res.start;
|
||||||
pl022->virtbase = ioremap(adev->res.start, resource_size(&adev->res));
|
pl022->virtbase = devm_ioremap(dev, adev->res.start,
|
||||||
|
resource_size(&adev->res));
|
||||||
if (pl022->virtbase == NULL) {
|
if (pl022->virtbase == NULL) {
|
||||||
status = -ENOMEM;
|
status = -ENOMEM;
|
||||||
goto err_no_ioremap;
|
goto err_no_ioremap;
|
||||||
@ -2190,7 +2188,7 @@ pl022_probe(struct amba_device *adev, const struct amba_id *id)
|
|||||||
|
|
||||||
pm_runtime_resume(dev);
|
pm_runtime_resume(dev);
|
||||||
|
|
||||||
pl022->clk = clk_get(&adev->dev, NULL);
|
pl022->clk = devm_clk_get(&adev->dev, NULL);
|
||||||
if (IS_ERR(pl022->clk)) {
|
if (IS_ERR(pl022->clk)) {
|
||||||
status = PTR_ERR(pl022->clk);
|
status = PTR_ERR(pl022->clk);
|
||||||
dev_err(&adev->dev, "could not retrieve SSP/SPI bus clock\n");
|
dev_err(&adev->dev, "could not retrieve SSP/SPI bus clock\n");
|
||||||
@ -2218,8 +2216,8 @@ pl022_probe(struct amba_device *adev, const struct amba_id *id)
|
|||||||
SSP_CR1(pl022->virtbase));
|
SSP_CR1(pl022->virtbase));
|
||||||
load_ssp_default_config(pl022);
|
load_ssp_default_config(pl022);
|
||||||
|
|
||||||
status = request_irq(adev->irq[0], pl022_interrupt_handler, 0, "pl022",
|
status = devm_request_irq(dev, adev->irq[0], pl022_interrupt_handler,
|
||||||
pl022);
|
0, "pl022", pl022);
|
||||||
if (status < 0) {
|
if (status < 0) {
|
||||||
dev_err(&adev->dev, "probe - cannot get IRQ (%d)\n", status);
|
dev_err(&adev->dev, "probe - cannot get IRQ (%d)\n", status);
|
||||||
goto err_no_irq;
|
goto err_no_irq;
|
||||||
@ -2259,24 +2257,18 @@ pl022_probe(struct amba_device *adev, const struct amba_id *id)
|
|||||||
err_spi_register:
|
err_spi_register:
|
||||||
if (platform_info->enable_dma)
|
if (platform_info->enable_dma)
|
||||||
pl022_dma_remove(pl022);
|
pl022_dma_remove(pl022);
|
||||||
|
|
||||||
free_irq(adev->irq[0], pl022);
|
|
||||||
err_no_irq:
|
err_no_irq:
|
||||||
clk_disable(pl022->clk);
|
clk_disable(pl022->clk);
|
||||||
err_no_clk_en:
|
err_no_clk_en:
|
||||||
clk_unprepare(pl022->clk);
|
clk_unprepare(pl022->clk);
|
||||||
err_clk_prep:
|
err_clk_prep:
|
||||||
clk_put(pl022->clk);
|
|
||||||
err_no_clk:
|
err_no_clk:
|
||||||
iounmap(pl022->virtbase);
|
|
||||||
err_no_ioremap:
|
err_no_ioremap:
|
||||||
amba_release_regions(adev);
|
amba_release_regions(adev);
|
||||||
err_no_ioregion:
|
err_no_ioregion:
|
||||||
err_no_gpio:
|
err_no_gpio:
|
||||||
err_no_pinctrl:
|
err_no_pinctrl:
|
||||||
spi_master_put(master);
|
spi_master_put(master);
|
||||||
err_no_master:
|
|
||||||
err_no_pdata:
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2298,12 +2290,9 @@ pl022_remove(struct amba_device *adev)
|
|||||||
if (pl022->master_info->enable_dma)
|
if (pl022->master_info->enable_dma)
|
||||||
pl022_dma_remove(pl022);
|
pl022_dma_remove(pl022);
|
||||||
|
|
||||||
free_irq(adev->irq[0], pl022);
|
|
||||||
clk_disable(pl022->clk);
|
clk_disable(pl022->clk);
|
||||||
clk_unprepare(pl022->clk);
|
clk_unprepare(pl022->clk);
|
||||||
clk_put(pl022->clk);
|
|
||||||
pm_runtime_disable(&adev->dev);
|
pm_runtime_disable(&adev->dev);
|
||||||
iounmap(pl022->virtbase);
|
|
||||||
amba_release_regions(adev);
|
amba_release_regions(adev);
|
||||||
tasklet_disable(&pl022->pump_transfers);
|
tasklet_disable(&pl022->pump_transfers);
|
||||||
spi_unregister_master(pl022->master);
|
spi_unregister_master(pl022->master);
|
||||||
|
Loading…
Reference in New Issue
Block a user