forked from Minki/linux
2795343705
The new clock subsystem was merged in linux-3.4 without any users, this now moves the first three platforms over to it: imx, mxs and spear. The series also contains the changes for the clock subsystem itself, since Mike preferred to have it together with the platforms that require these changes, in order to avoid interdependencies and conflicts. -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.11 (GNU/Linux) iQIcBAABAgAGBQJPuexPAAoJEIwa5zzehBx3YBsP/0nFhXjb5t1PdLfFzGKtcZVB j4zXWXMHQ1fA7wIfEpZF3Nnco6MQkufF5wJPoPdn1+wmkzCn3D6IwNVWVtW4U5i9 VGyShSbgusAAYXUe/9yYj8eN+bbRQSvdN4eWYWU6+rRXShGZ5dZZmp+IPNl54dnW 6F8uCnHX0cnIMCpGqV+41zZgZ/4wL2k9gdqu0LO6pi07o4tGd0Z4gcySgUFAnn1R kofNHueYIP4UgOg8DREoBzVKlpRqMou3S2kSZUfMeb3Q9ryF7UIvaGqIILyi7PKL kWd3nptg0EPavfL21SwXHiGpnDpB/Gj/F70kcPLus5RYujB24C9bvBmc26z68NZx Sz9mbElkkIU5duZsl1nxBWJ8IZ/tSWdtmC2xQMznmV7gHyGgVwr4j47f4Uv5sBvM 14JHDO7mqN6E6FnTFZu/oPAN5pDjgL+TVNK5BU6Wkq0zitrA6eyKDqCvBCqkO6Nn tNzOuyRDzMOwM7HzqXhxqtzJWXylO1Mldc4bM8X4Cocf4pnLna/X6uP6dgE6A+JY azVYx4I/0NdEPerDTzIcEhBDgZeBVROhUQr+kHxc4rf6WzUUbu/wEo1UKXWV66oW 1jb1yAFFWqYjkQuQc2PD4JSx35sFJaoSaoneRtmzBzRDfzSr5KjKj1E0e1skyMFq 7ZVLCqZD0cB9DhmMDkWP =rwFF -----END PGP SIGNATURE----- Merge tag 'clock' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc Pull arm-soc clock driver changes from Olof Johansson: "The new clock subsystem was merged in linux-3.4 without any users, this now moves the first three platforms over to it: imx, mxs and spear. The series also contains the changes for the clock subsystem itself, since Mike preferred to have it together with the platforms that require these changes, in order to avoid interdependencies and conflicts." Fix up trivial conflicts in arch/arm/mach-kirkwood/common.c (code removed in one branch, added OF support in another) and drivers/dma/imx-sdma.c (independent changes next to each other). * tag 'clock' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc: (97 commits) clk: Fix CLK_SET_RATE_GATE flag validation in clk_set_rate(). clk: Provide dummy clk_unregister() SPEAr: Update defconfigs SPEAr: Add SMI NOR partition info in dts files SPEAr: Switch to common clock framework SPEAr: Call clk_prepare() before calling clk_enable SPEAr: clk: Add General Purpose Timer Synthesizer clock SPEAr: clk: Add Fractional Synthesizer clock SPEAr: clk: Add Auxiliary Synthesizer clock SPEAr: clk: Add VCO-PLL Synthesizer clock SPEAr: Add DT bindings for SPEAr's timer ARM i.MX: remove now unused clock files ARM: i.MX6: implement clocks using common clock framework ARM i.MX35: implement clocks using common clock framework ARM i.MX5: implement clocks using common clock framework ARM: Kirkwood: Replace clock gating ARM: Orion: Audio: Add clk/clkdev support ARM: Orion: PCIE: Add support for clk ARM: Orion: XOR: Add support for clk ARM: Orion: CESA: Add support for clk ...
248 lines
5.4 KiB
C
248 lines
5.4 KiB
C
/*
|
|
* drivers/mtd/nand/orion_nand.c
|
|
*
|
|
* NAND support for Marvell Orion SoC platforms
|
|
*
|
|
* Tzachi Perelstein <tzachi@marvell.com>
|
|
*
|
|
* This file is licensed under the terms of the GNU General Public
|
|
* License version 2. This program is licensed "as is" without any
|
|
* warranty of any kind, whether express or implied.
|
|
*/
|
|
|
|
#include <linux/slab.h>
|
|
#include <linux/module.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/of.h>
|
|
#include <linux/mtd/mtd.h>
|
|
#include <linux/mtd/nand.h>
|
|
#include <linux/mtd/partitions.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/err.h>
|
|
#include <asm/io.h>
|
|
#include <asm/sizes.h>
|
|
#include <mach/hardware.h>
|
|
#include <plat/orion_nand.h>
|
|
|
|
static void orion_nand_cmd_ctrl(struct mtd_info *mtd, int cmd, unsigned int ctrl)
|
|
{
|
|
struct nand_chip *nc = mtd->priv;
|
|
struct orion_nand_data *board = nc->priv;
|
|
u32 offs;
|
|
|
|
if (cmd == NAND_CMD_NONE)
|
|
return;
|
|
|
|
if (ctrl & NAND_CLE)
|
|
offs = (1 << board->cle);
|
|
else if (ctrl & NAND_ALE)
|
|
offs = (1 << board->ale);
|
|
else
|
|
return;
|
|
|
|
if (nc->options & NAND_BUSWIDTH_16)
|
|
offs <<= 1;
|
|
|
|
writeb(cmd, nc->IO_ADDR_W + offs);
|
|
}
|
|
|
|
static void orion_nand_read_buf(struct mtd_info *mtd, uint8_t *buf, int len)
|
|
{
|
|
struct nand_chip *chip = mtd->priv;
|
|
void __iomem *io_base = chip->IO_ADDR_R;
|
|
uint64_t *buf64;
|
|
int i = 0;
|
|
|
|
while (len && (unsigned long)buf & 7) {
|
|
*buf++ = readb(io_base);
|
|
len--;
|
|
}
|
|
buf64 = (uint64_t *)buf;
|
|
while (i < len/8) {
|
|
/*
|
|
* Since GCC has no proper constraint (PR 43518)
|
|
* force x variable to r2/r3 registers as ldrd instruction
|
|
* requires first register to be even.
|
|
*/
|
|
register uint64_t x asm ("r2");
|
|
|
|
asm volatile ("ldrd\t%0, [%1]" : "=&r" (x) : "r" (io_base));
|
|
buf64[i++] = x;
|
|
}
|
|
i *= 8;
|
|
while (i < len)
|
|
buf[i++] = readb(io_base);
|
|
}
|
|
|
|
static int __init orion_nand_probe(struct platform_device *pdev)
|
|
{
|
|
struct mtd_info *mtd;
|
|
struct mtd_part_parser_data ppdata = {};
|
|
struct nand_chip *nc;
|
|
struct orion_nand_data *board;
|
|
struct resource *res;
|
|
struct clk *clk;
|
|
void __iomem *io_base;
|
|
int ret = 0;
|
|
u32 val = 0;
|
|
|
|
nc = kzalloc(sizeof(struct nand_chip) + sizeof(struct mtd_info), GFP_KERNEL);
|
|
if (!nc) {
|
|
printk(KERN_ERR "orion_nand: failed to allocate device structure.\n");
|
|
ret = -ENOMEM;
|
|
goto no_res;
|
|
}
|
|
mtd = (struct mtd_info *)(nc + 1);
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
if (!res) {
|
|
ret = -ENODEV;
|
|
goto no_res;
|
|
}
|
|
|
|
io_base = ioremap(res->start, resource_size(res));
|
|
if (!io_base) {
|
|
printk(KERN_ERR "orion_nand: ioremap failed\n");
|
|
ret = -EIO;
|
|
goto no_res;
|
|
}
|
|
|
|
if (pdev->dev.of_node) {
|
|
board = devm_kzalloc(&pdev->dev, sizeof(struct orion_nand_data),
|
|
GFP_KERNEL);
|
|
if (!board) {
|
|
printk(KERN_ERR "orion_nand: failed to allocate board structure.\n");
|
|
ret = -ENOMEM;
|
|
goto no_res;
|
|
}
|
|
if (!of_property_read_u32(pdev->dev.of_node, "cle", &val))
|
|
board->cle = (u8)val;
|
|
else
|
|
board->cle = 0;
|
|
if (!of_property_read_u32(pdev->dev.of_node, "ale", &val))
|
|
board->ale = (u8)val;
|
|
else
|
|
board->ale = 1;
|
|
if (!of_property_read_u32(pdev->dev.of_node,
|
|
"bank-width", &val))
|
|
board->width = (u8)val * 8;
|
|
else
|
|
board->width = 8;
|
|
if (!of_property_read_u32(pdev->dev.of_node,
|
|
"chip-delay", &val))
|
|
board->chip_delay = (u8)val;
|
|
} else
|
|
board = pdev->dev.platform_data;
|
|
|
|
mtd->priv = nc;
|
|
mtd->owner = THIS_MODULE;
|
|
|
|
nc->priv = board;
|
|
nc->IO_ADDR_R = nc->IO_ADDR_W = io_base;
|
|
nc->cmd_ctrl = orion_nand_cmd_ctrl;
|
|
nc->read_buf = orion_nand_read_buf;
|
|
nc->ecc.mode = NAND_ECC_SOFT;
|
|
|
|
if (board->chip_delay)
|
|
nc->chip_delay = board->chip_delay;
|
|
|
|
WARN(board->width > 16,
|
|
"%d bit bus width out of range",
|
|
board->width);
|
|
|
|
if (board->width == 16)
|
|
nc->options |= NAND_BUSWIDTH_16;
|
|
|
|
if (board->dev_ready)
|
|
nc->dev_ready = board->dev_ready;
|
|
|
|
platform_set_drvdata(pdev, mtd);
|
|
|
|
/* Not all platforms can gate the clock, so it is not
|
|
an error if the clock does not exists. */
|
|
clk = clk_get(&pdev->dev, NULL);
|
|
if (!IS_ERR(clk)) {
|
|
clk_prepare_enable(clk);
|
|
clk_put(clk);
|
|
}
|
|
|
|
if (nand_scan(mtd, 1)) {
|
|
ret = -ENXIO;
|
|
goto no_dev;
|
|
}
|
|
|
|
mtd->name = "orion_nand";
|
|
ppdata.of_node = pdev->dev.of_node;
|
|
ret = mtd_device_parse_register(mtd, NULL, &ppdata,
|
|
board->parts, board->nr_parts);
|
|
if (ret) {
|
|
nand_release(mtd);
|
|
goto no_dev;
|
|
}
|
|
|
|
return 0;
|
|
|
|
no_dev:
|
|
platform_set_drvdata(pdev, NULL);
|
|
iounmap(io_base);
|
|
no_res:
|
|
kfree(nc);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int __devexit orion_nand_remove(struct platform_device *pdev)
|
|
{
|
|
struct mtd_info *mtd = platform_get_drvdata(pdev);
|
|
struct nand_chip *nc = mtd->priv;
|
|
struct clk *clk;
|
|
|
|
nand_release(mtd);
|
|
|
|
iounmap(nc->IO_ADDR_W);
|
|
|
|
kfree(nc);
|
|
|
|
clk = clk_get(&pdev->dev, NULL);
|
|
if (!IS_ERR(clk)) {
|
|
clk_disable_unprepare(clk);
|
|
clk_put(clk);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_OF
|
|
static struct of_device_id orion_nand_of_match_table[] = {
|
|
{ .compatible = "mrvl,orion-nand", },
|
|
{},
|
|
};
|
|
#endif
|
|
|
|
static struct platform_driver orion_nand_driver = {
|
|
.remove = __devexit_p(orion_nand_remove),
|
|
.driver = {
|
|
.name = "orion_nand",
|
|
.owner = THIS_MODULE,
|
|
.of_match_table = of_match_ptr(orion_nand_of_match_table),
|
|
},
|
|
};
|
|
|
|
static int __init orion_nand_init(void)
|
|
{
|
|
return platform_driver_probe(&orion_nand_driver, orion_nand_probe);
|
|
}
|
|
|
|
static void __exit orion_nand_exit(void)
|
|
{
|
|
platform_driver_unregister(&orion_nand_driver);
|
|
}
|
|
|
|
module_init(orion_nand_init);
|
|
module_exit(orion_nand_exit);
|
|
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_AUTHOR("Tzachi Perelstein");
|
|
MODULE_DESCRIPTION("NAND glue for Orion platforms");
|
|
MODULE_ALIAS("platform:orion_nand");
|