2009-07-29 18:51:56 +02:00
/*
* drivers / mtd / nand / nomadik_nand . c
*
* Overview :
* Driver for on - board NAND flash on Nomadik Platforms
*
* Copyright © 2007 STMicroelectronics Pvt . Ltd .
* Author : Sachin Verma < sachin . verma @ st . com >
*
* Copyright © 2009 Alessandro Rubini
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/types.h>
# include <linux/mtd/mtd.h>
# include <linux/mtd/nand.h>
# include <linux/mtd/nand_ecc.h>
# include <linux/platform_device.h>
# include <linux/mtd/partitions.h>
# include <linux/io.h>
# include <mach/nand.h>
# include <mach/fsmc.h>
# include <mtd/mtd-abi.h>
struct nomadik_nand_host {
struct mtd_info mtd ;
struct nand_chip nand ;
void __iomem * data_va ;
void __iomem * cmd_va ;
void __iomem * addr_va ;
struct nand_bbt_descr * bbt_desc ;
} ;
static struct nand_ecclayout nomadik_ecc_layout = {
. eccbytes = 3 * 4 ,
. eccpos = { /* each subpage has 16 bytes: pos 2,3,4 hosts ECC */
0x02 , 0x03 , 0x04 ,
0x12 , 0x13 , 0x14 ,
0x22 , 0x23 , 0x24 ,
0x32 , 0x33 , 0x34 } ,
/* let's keep bytes 5,6,7 for us, just in case we change ECC algo */
. oobfree = { { 0x08 , 0x08 } , { 0x18 , 0x08 } , { 0x28 , 0x08 } , { 0x38 , 0x08 } } ,
} ;
static void nomadik_ecc_control ( struct mtd_info * mtd , int mode )
{
/* No need to enable hw ecc, it's on by default */
}
static void nomadik_cmd_ctrl ( struct mtd_info * mtd , int cmd , unsigned int ctrl )
{
struct nand_chip * nand = mtd - > priv ;
struct nomadik_nand_host * host = nand - > priv ;
if ( cmd = = NAND_CMD_NONE )
return ;
if ( ctrl & NAND_CLE )
writeb ( cmd , host - > cmd_va ) ;
else
writeb ( cmd , host - > addr_va ) ;
}
static int nomadik_nand_probe ( struct platform_device * pdev )
{
struct nomadik_nand_platform_data * pdata = pdev - > dev . platform_data ;
struct nomadik_nand_host * host ;
struct mtd_info * mtd ;
struct nand_chip * nand ;
struct resource * res ;
int ret = 0 ;
/* Allocate memory for the device structure (and zero it) */
host = kzalloc ( sizeof ( struct nomadik_nand_host ) , GFP_KERNEL ) ;
if ( ! host ) {
dev_err ( & pdev - > dev , " Failed to allocate device structure. \n " ) ;
return - ENOMEM ;
}
/* Call the client's init function, if any */
if ( pdata - > init )
ret = pdata - > init ( ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Init function failed \n " ) ;
goto err ;
}
/* ioremap three regions */
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " nand_addr " ) ;
if ( ! res ) {
ret = - EIO ;
goto err_unmap ;
}
host - > addr_va = ioremap ( res - > start , res - > end - res - > start + 1 ) ;
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " nand_data " ) ;
if ( ! res ) {
ret = - EIO ;
goto err_unmap ;
}
host - > data_va = ioremap ( res - > start , res - > end - res - > start + 1 ) ;
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " nand_cmd " ) ;
if ( ! res ) {
ret = - EIO ;
goto err_unmap ;
}
host - > cmd_va = ioremap ( res - > start , res - > end - res - > start + 1 ) ;
if ( ! host - > addr_va | | ! host - > data_va | | ! host - > cmd_va ) {
ret = - ENOMEM ;
goto err_unmap ;
}
/* Link all private pointers */
mtd = & host - > mtd ;
nand = & host - > nand ;
mtd - > priv = nand ;
nand - > priv = host ;
host - > mtd . owner = THIS_MODULE ;
nand - > IO_ADDR_R = host - > data_va ;
nand - > IO_ADDR_W = host - > data_va ;
nand - > cmd_ctrl = nomadik_cmd_ctrl ;
/*
* This stanza declares ECC_HW but uses soft routines . It ' s because
* HW claims to make the calculation but not the correction . However ,
* I haven ' t managed to get the desired data out of it until now .
*/
nand - > ecc . mode = NAND_ECC_SOFT ;
nand - > ecc . layout = & nomadik_ecc_layout ;
nand - > ecc . hwctl = nomadik_ecc_control ;
nand - > ecc . size = 512 ;
nand - > ecc . bytes = 3 ;
nand - > options = pdata - > options ;
/*
* Scan to find existance of the device
*/
if ( nand_scan ( & host - > mtd , 1 ) ) {
ret = - ENXIO ;
goto err_unmap ;
}
# ifdef CONFIG_MTD_PARTITIONS
add_mtd_partitions ( & host - > mtd , pdata - > parts , pdata - > nparts ) ;
# else
pr_info ( " Registering %s as whole device \n " , mtd - > name ) ;
add_mtd_device ( mtd ) ;
# endif
platform_set_drvdata ( pdev , host ) ;
return 0 ;
err_unmap :
if ( host - > cmd_va )
iounmap ( host - > cmd_va ) ;
if ( host - > data_va )
iounmap ( host - > data_va ) ;
if ( host - > addr_va )
iounmap ( host - > addr_va ) ;
err :
kfree ( host ) ;
return ret ;
}
/*
* Clean up routine
*/
static int nomadik_nand_remove ( struct platform_device * pdev )
{
struct nomadik_nand_host * host = platform_get_drvdata ( pdev ) ;
struct nomadik_nand_platform_data * pdata = pdev - > dev . platform_data ;
if ( pdata - > exit )
pdata - > exit ( ) ;
if ( host ) {
iounmap ( host - > cmd_va ) ;
iounmap ( host - > data_va ) ;
iounmap ( host - > addr_va ) ;
kfree ( host ) ;
}
return 0 ;
}
static int nomadik_nand_suspend ( struct device * dev )
{
struct nomadik_nand_host * host = dev_get_drvdata ( dev ) ;
int ret = 0 ;
if ( host )
ret = host - > mtd . suspend ( & host - > mtd ) ;
return ret ;
}
static int nomadik_nand_resume ( struct device * dev )
{
struct nomadik_nand_host * host = dev_get_drvdata ( dev ) ;
if ( host )
host - > mtd . resume ( & host - > mtd ) ;
return 0 ;
}
2009-12-14 18:00:08 -08:00
static const struct dev_pm_ops nomadik_nand_pm_ops = {
2009-07-29 18:51:56 +02:00
. suspend = nomadik_nand_suspend ,
. resume = nomadik_nand_resume ,
} ;
static struct platform_driver nomadik_nand_driver = {
. probe = nomadik_nand_probe ,
. remove = nomadik_nand_remove ,
. driver = {
. owner = THIS_MODULE ,
. name = " nomadik_nand " ,
. pm = & nomadik_nand_pm_ops ,
} ,
} ;
static int __init nand_nomadik_init ( void )
{
pr_info ( " Nomadik NAND driver \n " ) ;
return platform_driver_register ( & nomadik_nand_driver ) ;
}
static void __exit nand_nomadik_exit ( void )
{
platform_driver_unregister ( & nomadik_nand_driver ) ;
}
module_init ( nand_nomadik_init ) ;
module_exit ( nand_nomadik_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " ST Microelectronics (sachin.verma@st.com) " ) ;
MODULE_DESCRIPTION ( " NAND driver for Nomadik Platform " ) ;