Commit e353a20a authored by Lei Wen's avatar Lei Wen Committed by David Woodhouse

mtd: pxa3xx_nand: make scan procedure more clear

The previous probe function is some kind of big part.
This patch seperate the resource allocation to keep the probe process
more clear than before.
Signed-off-by: default avatarLei Wen <leiwen@marvell.com>
Signed-off-by: default avatarHaojian Zhuang <haojian.zhuang@marvell.com>
Signed-off-by: default avatarDavid Woodhouse <David.Woodhouse@intel.com>
parent e70727e4
...@@ -126,6 +126,7 @@ struct pxa3xx_nand_info { ...@@ -126,6 +126,7 @@ struct pxa3xx_nand_info {
unsigned int buf_start; unsigned int buf_start;
unsigned int buf_count; unsigned int buf_count;
struct mtd_info *mtd;
/* DMA information */ /* DMA information */
int drcmr_dat; int drcmr_dat;
int drcmr_cmd; int drcmr_cmd;
...@@ -1044,34 +1045,27 @@ static void pxa3xx_nand_init_mtd(struct mtd_info *mtd, ...@@ -1044,34 +1045,27 @@ static void pxa3xx_nand_init_mtd(struct mtd_info *mtd,
this->chip_delay = 25; this->chip_delay = 25;
} }
static int pxa3xx_nand_probe(struct platform_device *pdev) static
struct pxa3xx_nand_info *alloc_nand_resource(struct platform_device *pdev)
{ {
struct pxa3xx_nand_platform_data *pdata; struct pxa3xx_nand_platform_data *pdata = pdev->dev.platform_data;
struct pxa3xx_nand_info *info; struct pxa3xx_nand_info *info;
struct nand_chip *this;
struct mtd_info *mtd; struct mtd_info *mtd;
struct resource *r; struct resource *r;
int ret = 0, irq; int ret, irq;
pdata = pdev->dev.platform_data;
if (!pdata) {
dev_err(&pdev->dev, "no platform data defined\n");
return -ENODEV;
}
mtd = kzalloc(sizeof(struct mtd_info) + sizeof(struct pxa3xx_nand_info), mtd = kzalloc(sizeof(struct mtd_info) + sizeof(struct pxa3xx_nand_info),
GFP_KERNEL); GFP_KERNEL);
if (!mtd) { if (!mtd) {
dev_err(&pdev->dev, "failed to allocate memory\n"); dev_err(&pdev->dev, "failed to allocate memory\n");
return -ENOMEM; return NULL;
} }
info = (struct pxa3xx_nand_info *)(&mtd[1]); info = (struct pxa3xx_nand_info *)(&mtd[1]);
info->pdev = pdev; info->pdev = pdev;
this = &info->nand_chip;
mtd->priv = info; mtd->priv = info;
info->mtd = mtd;
mtd->owner = THIS_MODULE; mtd->owner = THIS_MODULE;
info->clk = clk_get(&pdev->dev, NULL); info->clk = clk_get(&pdev->dev, NULL);
...@@ -1149,31 +1143,9 @@ static int pxa3xx_nand_probe(struct platform_device *pdev) ...@@ -1149,31 +1143,9 @@ static int pxa3xx_nand_probe(struct platform_device *pdev)
} }
pxa3xx_nand_init_mtd(mtd, info); pxa3xx_nand_init_mtd(mtd, info);
platform_set_drvdata(pdev, info);
platform_set_drvdata(pdev, mtd); return info;
if (nand_scan(mtd, 1)) {
dev_err(&pdev->dev, "failed to scan nand\n");
ret = -ENXIO;
goto fail_free_irq;
}
#ifdef CONFIG_MTD_PARTITIONS
if (mtd_has_cmdlinepart()) {
static const char *probes[] = { "cmdlinepart", NULL };
struct mtd_partition *parts;
int nr_parts;
nr_parts = parse_mtd_partitions(mtd, probes, &parts, 0);
if (nr_parts)
return add_mtd_partitions(mtd, parts, nr_parts);
}
return add_mtd_partitions(mtd, pdata->parts, pdata->nr_parts);
#else
return 0;
#endif
fail_free_irq: fail_free_irq:
free_irq(irq, info); free_irq(irq, info);
...@@ -1193,13 +1165,13 @@ static int pxa3xx_nand_probe(struct platform_device *pdev) ...@@ -1193,13 +1165,13 @@ static int pxa3xx_nand_probe(struct platform_device *pdev)
clk_put(info->clk); clk_put(info->clk);
fail_free_mtd: fail_free_mtd:
kfree(mtd); kfree(mtd);
return ret; return NULL;
} }
static int pxa3xx_nand_remove(struct platform_device *pdev) static int pxa3xx_nand_remove(struct platform_device *pdev)
{ {
struct mtd_info *mtd = platform_get_drvdata(pdev); struct pxa3xx_nand_info *info = platform_get_drvdata(pdev);
struct pxa3xx_nand_info *info = mtd->priv; struct mtd_info *mtd = info->mtd;
struct resource *r; struct resource *r;
int irq; int irq;
...@@ -1230,11 +1202,50 @@ static int pxa3xx_nand_remove(struct platform_device *pdev) ...@@ -1230,11 +1202,50 @@ static int pxa3xx_nand_remove(struct platform_device *pdev)
return 0; return 0;
} }
static int pxa3xx_nand_probe(struct platform_device *pdev)
{
struct pxa3xx_nand_platform_data *pdata;
struct pxa3xx_nand_info *info;
pdata = pdev->dev.platform_data;
if (!pdata) {
dev_err(&pdev->dev, "no platform data defined\n");
return -ENODEV;
}
info = alloc_nand_resource(pdev);
if (info == NULL)
return -ENOMEM;
if (nand_scan(info->mtd, 1)) {
dev_err(&pdev->dev, "failed to scan nand\n");
pxa3xx_nand_remove(pdev);
return -ENODEV;
}
#ifdef CONFIG_MTD_PARTITIONS
if (mtd_has_cmdlinepart()) {
const char *probes[] = { "cmdlinepart", NULL };
struct mtd_partition *parts;
int nr_parts;
nr_parts = parse_mtd_partitions(info->mtd, probes, &parts, 0);
if (nr_parts)
return add_mtd_partitions(mtd, parts, nr_parts);
}
return add_mtd_partitions(mtd, pdata->parts, pdata->nr_parts);
#else
return 0;
#endif
}
#ifdef CONFIG_PM #ifdef CONFIG_PM
static int pxa3xx_nand_suspend(struct platform_device *pdev, pm_message_t state) static int pxa3xx_nand_suspend(struct platform_device *pdev, pm_message_t state)
{ {
struct mtd_info *mtd = (struct mtd_info *)platform_get_drvdata(pdev); struct pxa3xx_nand_info *info = platform_get_drvdata(pdev);
struct pxa3xx_nand_info *info = mtd->priv; struct mtd_info *mtd = info->mtd;
if (info->state != STATE_READY) { if (info->state != STATE_READY) {
dev_err(&pdev->dev, "driver busy, state = %d\n", info->state); dev_err(&pdev->dev, "driver busy, state = %d\n", info->state);
...@@ -1246,8 +1257,8 @@ static int pxa3xx_nand_suspend(struct platform_device *pdev, pm_message_t state) ...@@ -1246,8 +1257,8 @@ static int pxa3xx_nand_suspend(struct platform_device *pdev, pm_message_t state)
static int pxa3xx_nand_resume(struct platform_device *pdev) static int pxa3xx_nand_resume(struct platform_device *pdev)
{ {
struct mtd_info *mtd = (struct mtd_info *)platform_get_drvdata(pdev); struct pxa3xx_nand_info *info = platform_get_drvdata(pdev);
struct pxa3xx_nand_info *info = mtd->priv; struct mtd_info *mtd = info->mtd;
nand_writel(info, NDTR0CS0, info->ndtr0cs0); nand_writel(info, NDTR0CS0, info->ndtr0cs0);
nand_writel(info, NDTR1CS0, info->ndtr1cs0); nand_writel(info, NDTR1CS0, info->ndtr1cs0);
......
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