Commit a0fa0b66 authored by Michael Opdenacker's avatar Michael Opdenacker Committed by Brian Norris

mtd: orion_nand: fix error code path in probe

This replaces kzalloc() and ioremap() calls by devm_ functions
in the probe() routine, which automatically release the corresponding
resources when probe() fails or when the device is removed.

This simplifies simplifies the error management code, and brings
the below improvements or changes:

A. Fixing a bug reported by "make coccicheck":

If "board = devm_kzalloc()" fails, the probe() function jumps
incorrectly to label "no_res" and therefore returns without
running iounmap().

B. Requesting the memory region

Using devm_ioremap_resource() makes the probe() function request
the corresponding memory region before running ioremap(), as
it is supposed to do.

C. Standardizing the error codes:

The use of devm_ioremap_resource() changes the return value:
 * -ENOMEM instead of -EIO in case of ioremap() failure,
 * -EINVAL instead of -ENODEV in case of platform_get_resource()
   failure.
Signed-off-by: default avatarMichael Opdenacker <michael.opdenacker@free-electrons.com>
Reviewed-by: default avatarJingoo Han <jg1.han@samsung.com>
Acked-by: default avatarAndrew Lunn <andrew@lunn.ch>
Signed-off-by: default avatarBrian Norris <computersforpeace@gmail.com>
parent 59af5c7a
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
#include <linux/mtd/partitions.h> #include <linux/mtd/partitions.h>
#include <linux/clk.h> #include <linux/clk.h>
#include <linux/err.h> #include <linux/err.h>
#include <asm/io.h> #include <linux/io.h>
#include <asm/sizes.h> #include <asm/sizes.h>
#include <linux/platform_data/mtd-orion_nand.h> #include <linux/platform_data/mtd-orion_nand.h>
...@@ -85,33 +85,24 @@ static int __init orion_nand_probe(struct platform_device *pdev) ...@@ -85,33 +85,24 @@ static int __init orion_nand_probe(struct platform_device *pdev)
int ret = 0; int ret = 0;
u32 val = 0; u32 val = 0;
nc = kzalloc(sizeof(struct nand_chip) + sizeof(struct mtd_info), GFP_KERNEL); nc = devm_kzalloc(&pdev->dev,
if (!nc) { sizeof(struct nand_chip) + sizeof(struct mtd_info),
ret = -ENOMEM; GFP_KERNEL);
goto no_res; if (!nc)
} return -ENOMEM;
mtd = (struct mtd_info *)(nc + 1); mtd = (struct mtd_info *)(nc + 1);
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!res) { io_base = devm_ioremap_resource(&pdev->dev, res);
ret = -ENODEV;
goto no_res;
}
io_base = ioremap(res->start, resource_size(res)); if (IS_ERR(io_base))
if (!io_base) { return PTR_ERR(io_base);
dev_err(&pdev->dev, "ioremap failed\n");
ret = -EIO;
goto no_res;
}
if (pdev->dev.of_node) { if (pdev->dev.of_node) {
board = devm_kzalloc(&pdev->dev, sizeof(struct orion_nand_data), board = devm_kzalloc(&pdev->dev, sizeof(struct orion_nand_data),
GFP_KERNEL); GFP_KERNEL);
if (!board) { if (!board)
ret = -ENOMEM; return -ENOMEM;
goto no_res;
}
if (!of_property_read_u32(pdev->dev.of_node, "cle", &val)) if (!of_property_read_u32(pdev->dev.of_node, "cle", &val))
board->cle = (u8)val; board->cle = (u8)val;
else else
...@@ -185,9 +176,6 @@ static int __init orion_nand_probe(struct platform_device *pdev) ...@@ -185,9 +176,6 @@ static int __init orion_nand_probe(struct platform_device *pdev)
clk_disable_unprepare(clk); clk_disable_unprepare(clk);
clk_put(clk); clk_put(clk);
} }
iounmap(io_base);
no_res:
kfree(nc);
return ret; return ret;
} }
...@@ -195,15 +183,10 @@ static int __init orion_nand_probe(struct platform_device *pdev) ...@@ -195,15 +183,10 @@ static int __init orion_nand_probe(struct platform_device *pdev)
static int orion_nand_remove(struct platform_device *pdev) static int orion_nand_remove(struct platform_device *pdev)
{ {
struct mtd_info *mtd = platform_get_drvdata(pdev); struct mtd_info *mtd = platform_get_drvdata(pdev);
struct nand_chip *nc = mtd->priv;
struct clk *clk; struct clk *clk;
nand_release(mtd); nand_release(mtd);
iounmap(nc->IO_ADDR_W);
kfree(nc);
clk = clk_get(&pdev->dev, NULL); clk = clk_get(&pdev->dev, NULL);
if (!IS_ERR(clk)) { if (!IS_ERR(clk)) {
clk_disable_unprepare(clk); clk_disable_unprepare(clk);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment