Commit 5c05bc00 authored by Simon Arlott's avatar Simon Arlott Committed by Brian Norris

mtd: brcmnand: Request and enable the clock if present

Attempt to enable a clock named "nand" as some SoCs have a clock for the
controller that needs to be enabled.
Signed-off-by: default avatarSimon Arlott <simon@fire.lp0.eu>
Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarBrian Norris <computersforpeace@gmail.com>
parent d135d1c1
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
* GNU General Public License for more details. * GNU General Public License for more details.
*/ */
#include <linux/clk.h>
#include <linux/version.h> #include <linux/version.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/init.h> #include <linux/init.h>
...@@ -122,6 +123,9 @@ struct brcmnand_controller { ...@@ -122,6 +123,9 @@ struct brcmnand_controller {
/* Some SoCs provide custom interrupt status register(s) */ /* Some SoCs provide custom interrupt status register(s) */
struct brcmnand_soc *soc; struct brcmnand_soc *soc;
/* Some SoCs have a gateable clock for the controller */
struct clk *clk;
int cmd_pending; int cmd_pending;
bool dma_pending; bool dma_pending;
struct completion done; struct completion done;
...@@ -2127,10 +2131,24 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2127,10 +2131,24 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
if (IS_ERR(ctrl->nand_base)) if (IS_ERR(ctrl->nand_base))
return PTR_ERR(ctrl->nand_base); return PTR_ERR(ctrl->nand_base);
/* Enable clock before using NAND registers */
ctrl->clk = devm_clk_get(dev, "nand");
if (!IS_ERR(ctrl->clk)) {
ret = clk_prepare_enable(ctrl->clk);
if (ret)
return ret;
} else {
ret = PTR_ERR(ctrl->clk);
if (ret == -EPROBE_DEFER)
return ret;
ctrl->clk = NULL;
}
/* Initialize NAND revision */ /* Initialize NAND revision */
ret = brcmnand_revision_init(ctrl); ret = brcmnand_revision_init(ctrl);
if (ret) if (ret)
return ret; goto err;
/* /*
* Most chips have this cache at a fixed offset within 'nand' block. * Most chips have this cache at a fixed offset within 'nand' block.
...@@ -2139,8 +2157,10 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2139,8 +2157,10 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "nand-cache"); res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "nand-cache");
if (res) { if (res) {
ctrl->nand_fc = devm_ioremap_resource(dev, res); ctrl->nand_fc = devm_ioremap_resource(dev, res);
if (IS_ERR(ctrl->nand_fc)) if (IS_ERR(ctrl->nand_fc)) {
return PTR_ERR(ctrl->nand_fc); ret = PTR_ERR(ctrl->nand_fc);
goto err;
}
} else { } else {
ctrl->nand_fc = ctrl->nand_base + ctrl->nand_fc = ctrl->nand_base +
ctrl->reg_offsets[BRCMNAND_FC_BASE]; ctrl->reg_offsets[BRCMNAND_FC_BASE];
...@@ -2150,8 +2170,10 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2150,8 +2170,10 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "flash-dma"); res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "flash-dma");
if (res) { if (res) {
ctrl->flash_dma_base = devm_ioremap_resource(dev, res); ctrl->flash_dma_base = devm_ioremap_resource(dev, res);
if (IS_ERR(ctrl->flash_dma_base)) if (IS_ERR(ctrl->flash_dma_base)) {
return PTR_ERR(ctrl->flash_dma_base); ret = PTR_ERR(ctrl->flash_dma_base);
goto err;
}
flash_dma_writel(ctrl, FLASH_DMA_MODE, 1); /* linked-list */ flash_dma_writel(ctrl, FLASH_DMA_MODE, 1); /* linked-list */
flash_dma_writel(ctrl, FLASH_DMA_ERROR_STATUS, 0); flash_dma_writel(ctrl, FLASH_DMA_ERROR_STATUS, 0);
...@@ -2160,13 +2182,16 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2160,13 +2182,16 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
ctrl->dma_desc = dmam_alloc_coherent(dev, ctrl->dma_desc = dmam_alloc_coherent(dev,
sizeof(*ctrl->dma_desc), sizeof(*ctrl->dma_desc),
&ctrl->dma_pa, GFP_KERNEL); &ctrl->dma_pa, GFP_KERNEL);
if (!ctrl->dma_desc) if (!ctrl->dma_desc) {
return -ENOMEM; ret = -ENOMEM;
goto err;
}
ctrl->dma_irq = platform_get_irq(pdev, 1); ctrl->dma_irq = platform_get_irq(pdev, 1);
if ((int)ctrl->dma_irq < 0) { if ((int)ctrl->dma_irq < 0) {
dev_err(dev, "missing FLASH_DMA IRQ\n"); dev_err(dev, "missing FLASH_DMA IRQ\n");
return -ENODEV; ret = -ENODEV;
goto err;
} }
ret = devm_request_irq(dev, ctrl->dma_irq, ret = devm_request_irq(dev, ctrl->dma_irq,
...@@ -2175,7 +2200,7 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2175,7 +2200,7 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
if (ret < 0) { if (ret < 0) {
dev_err(dev, "can't allocate IRQ %d: error %d\n", dev_err(dev, "can't allocate IRQ %d: error %d\n",
ctrl->dma_irq, ret); ctrl->dma_irq, ret);
return ret; goto err;
} }
dev_info(dev, "enabling FLASH_DMA\n"); dev_info(dev, "enabling FLASH_DMA\n");
...@@ -2199,7 +2224,8 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2199,7 +2224,8 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
ctrl->irq = platform_get_irq(pdev, 0); ctrl->irq = platform_get_irq(pdev, 0);
if ((int)ctrl->irq < 0) { if ((int)ctrl->irq < 0) {
dev_err(dev, "no IRQ defined\n"); dev_err(dev, "no IRQ defined\n");
return -ENODEV; ret = -ENODEV;
goto err;
} }
/* /*
...@@ -2223,7 +2249,7 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2223,7 +2249,7 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
if (ret < 0) { if (ret < 0) {
dev_err(dev, "can't allocate IRQ %d: error %d\n", dev_err(dev, "can't allocate IRQ %d: error %d\n",
ctrl->irq, ret); ctrl->irq, ret);
return ret; goto err;
} }
for_each_available_child_of_node(dn, child) { for_each_available_child_of_node(dn, child) {
...@@ -2233,7 +2259,8 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2233,7 +2259,8 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
host = devm_kzalloc(dev, sizeof(*host), GFP_KERNEL); host = devm_kzalloc(dev, sizeof(*host), GFP_KERNEL);
if (!host) { if (!host) {
of_node_put(child); of_node_put(child);
return -ENOMEM; ret = -ENOMEM;
goto err;
} }
host->pdev = pdev; host->pdev = pdev;
host->ctrl = ctrl; host->ctrl = ctrl;
...@@ -2249,10 +2276,17 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc) ...@@ -2249,10 +2276,17 @@ int brcmnand_probe(struct platform_device *pdev, struct brcmnand_soc *soc)
} }
/* No chip-selects could initialize properly */ /* No chip-selects could initialize properly */
if (list_empty(&ctrl->host_list)) if (list_empty(&ctrl->host_list)) {
return -ENODEV; ret = -ENODEV;
goto err;
}
return 0; return 0;
err:
clk_disable_unprepare(ctrl->clk);
return ret;
} }
EXPORT_SYMBOL_GPL(brcmnand_probe); EXPORT_SYMBOL_GPL(brcmnand_probe);
...@@ -2264,6 +2298,8 @@ int brcmnand_remove(struct platform_device *pdev) ...@@ -2264,6 +2298,8 @@ int brcmnand_remove(struct platform_device *pdev)
list_for_each_entry(host, &ctrl->host_list, node) list_for_each_entry(host, &ctrl->host_list, node)
nand_release(&host->mtd); nand_release(&host->mtd);
clk_disable_unprepare(ctrl->clk);
dev_set_drvdata(&pdev->dev, NULL); dev_set_drvdata(&pdev->dev, NULL);
return 0; return 0;
......
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