2005-04-16 15:20:36 -07:00
/*
* drivers / mtd / nand / h1910 . c
*
* Copyright ( C ) 2003 Joshua Wise ( joshua @ joshuawise . com )
*
* Derived from drivers / mtd / nand / edb7312 . c
* Copyright ( C ) 2002 Marius Gr <EFBFBD> ger ( mag @ sysgo . de )
* Copyright ( c ) 2001 Thomas Gleixner ( gleixner @ autronix . de )
*
2005-11-07 11:15:49 +00:00
* $ Id : h1910 . c , v 1.6 2005 / 11 / 07 11 : 14 : 30 gleixner Exp $
2005-04-16 15:20:36 -07:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* Overview :
* This is a device driver for the NAND flash device found on the
* iPAQ h1910 board which utilizes the Samsung K9F2808 part . This is
* a 128 Mibit ( 16 MiB x 8 bits ) NAND flash device .
*/
# include <linux/slab.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/mtd/mtd.h>
# include <linux/mtd/nand.h>
# include <linux/mtd/partitions.h>
# include <asm/io.h>
2006-05-13 18:07:53 +01:00
# include <asm/arch/hardware.h> /* for CLPS7111_VIRT_BASE */
2005-04-16 15:20:36 -07:00
# include <asm/sizes.h>
# include <asm/arch/h1900-gpio.h>
# include <asm/arch/ipaq.h>
/*
* MTD structure for EDB7312 board
*/
static struct mtd_info * h1910_nand_mtd = NULL ;
/*
* Module stuff
*/
# ifdef CONFIG_MTD_PARTITIONS
/*
* Define static partitions for flash device
*/
static struct mtd_partition partition_info [ ] = {
2006-05-13 18:07:53 +01:00
{ name : " h1910 NAND Flash " ,
offset : 0 ,
size : 16 * 1024 * 1024 }
2005-04-16 15:20:36 -07:00
} ;
2006-05-13 18:07:53 +01:00
2005-04-16 15:20:36 -07:00
# define NUM_PARTITIONS 1
# endif
2005-11-07 11:15:49 +00:00
/*
2005-04-16 15:20:36 -07:00
* hardware specific access to control - lines
*/
2005-11-07 11:15:49 +00:00
static void h1910_hwcontrol ( struct mtd_info * mtd , int cmd )
2005-04-16 15:20:36 -07:00
{
2006-05-13 18:07:53 +01:00
struct nand_chip * this = ( struct nand_chip * ) ( mtd - > priv ) ;
2005-11-07 11:15:49 +00:00
2006-05-13 18:07:53 +01:00
switch ( cmd ) {
2005-11-07 11:15:49 +00:00
case NAND_CTL_SETCLE :
2005-04-16 15:20:36 -07:00
this - > IO_ADDR_R | = ( 1 < < 2 ) ;
this - > IO_ADDR_W | = ( 1 < < 2 ) ;
break ;
2005-11-07 11:15:49 +00:00
case NAND_CTL_CLRCLE :
2005-04-16 15:20:36 -07:00
this - > IO_ADDR_R & = ~ ( 1 < < 2 ) ;
this - > IO_ADDR_W & = ~ ( 1 < < 2 ) ;
break ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
case NAND_CTL_SETALE :
this - > IO_ADDR_R | = ( 1 < < 3 ) ;
this - > IO_ADDR_W | = ( 1 < < 3 ) ;
break ;
case NAND_CTL_CLRALE :
this - > IO_ADDR_R & = ~ ( 1 < < 3 ) ;
this - > IO_ADDR_W & = ~ ( 1 < < 3 ) ;
break ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
case NAND_CTL_SETNCE :
break ;
case NAND_CTL_CLRNCE :
break ;
}
}
/*
* read device ready pin
*/
#if 0
static int h1910_device_ready ( struct mtd_info * mtd )
{
return ( GPLR ( 55 ) & GPIO_bit ( 55 ) ) ;
}
# endif
/*
* Main initialization routine
*/
2006-05-13 18:07:53 +01:00
static int __init h1910_init ( void )
2005-04-16 15:20:36 -07:00
{
struct nand_chip * this ;
const char * part_type = 0 ;
int mtd_parts_nb = 0 ;
struct mtd_partition * mtd_parts = 0 ;
void __iomem * nandaddr ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
if ( ! machine_is_h1900 ( ) )
return - ENODEV ;
2005-11-07 11:15:49 +00:00
2005-11-17 16:46:41 +00:00
nandaddr = ioremap ( 0x08000000 , 0x1000 ) ;
2005-04-16 15:20:36 -07:00
if ( ! nandaddr ) {
printk ( " Failed to ioremap nand flash. \n " ) ;
return - ENOMEM ;
}
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Allocate memory for MTD device structure and private data */
2006-05-13 18:07:53 +01:00
h1910_nand_mtd = kmalloc ( sizeof ( struct mtd_info ) + sizeof ( struct nand_chip ) , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( ! h1910_nand_mtd ) {
printk ( " Unable to allocate h1910 NAND MTD device structure. \n " ) ;
2006-05-13 18:07:53 +01:00
iounmap ( ( void * ) nandaddr ) ;
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
}
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Get pointer to private data */
2006-05-13 18:07:53 +01:00
this = ( struct nand_chip * ) ( & h1910_nand_mtd [ 1 ] ) ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Initialize structures */
2006-05-13 18:07:53 +01:00
memset ( h1910_nand_mtd , 0 , sizeof ( struct mtd_info ) ) ;
memset ( this , 0 , sizeof ( struct nand_chip ) ) ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Link the private data with the MTD structure */
h1910_nand_mtd - > priv = this ;
2006-05-14 01:20:46 +01:00
h1910_nand_mtd - > owner = THIS_MODULE ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/*
* Enable VPEN
*/
GPSR ( 37 ) = GPIO_bit ( 37 ) ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* insert callbacks */
this - > IO_ADDR_R = nandaddr ;
this - > IO_ADDR_W = nandaddr ;
this - > hwcontrol = h1910_hwcontrol ;
this - > dev_ready = NULL ; /* unknown whether that was correct or not so we will just do it like this */
/* 15 us command delay time */
this - > chip_delay = 50 ;
this - > eccmode = NAND_ECC_SOFT ;
this - > options = NAND_NO_AUTOINCR ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Scan to find existence of the device */
2006-05-13 18:07:53 +01:00
if ( nand_scan ( h1910_nand_mtd , 1 ) ) {
2005-04-16 15:20:36 -07:00
printk ( KERN_NOTICE " No NAND device - returning -ENXIO \n " ) ;
2006-05-13 18:07:53 +01:00
kfree ( h1910_nand_mtd ) ;
iounmap ( ( void * ) nandaddr ) ;
2005-04-16 15:20:36 -07:00
return - ENXIO ;
}
# ifdef CONFIG_MTD_CMDLINE_PARTS
2006-05-13 18:07:53 +01:00
mtd_parts_nb = parse_cmdline_partitions ( h1910_nand_mtd , & mtd_parts , " h1910-nand " ) ;
2005-04-16 15:20:36 -07:00
if ( mtd_parts_nb > 0 )
2006-05-13 18:07:53 +01:00
part_type = " command line " ;
2005-04-16 15:20:36 -07:00
else
2006-05-13 18:07:53 +01:00
mtd_parts_nb = 0 ;
2005-04-16 15:20:36 -07:00
# endif
2006-05-13 18:07:53 +01:00
if ( mtd_parts_nb = = 0 ) {
2005-04-16 15:20:36 -07:00
mtd_parts = partition_info ;
mtd_parts_nb = NUM_PARTITIONS ;
part_type = " static " ;
}
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Register the partitions */
printk ( KERN_NOTICE " Using %s partition definition \n " , part_type ) ;
add_mtd_partitions ( h1910_nand_mtd , mtd_parts , mtd_parts_nb ) ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Return happy */
return 0 ;
}
2006-05-13 18:07:53 +01:00
2005-04-16 15:20:36 -07:00
module_init ( h1910_init ) ;
/*
* Clean up routine
*/
2006-05-13 18:07:53 +01:00
static void __exit h1910_cleanup ( void )
2005-04-16 15:20:36 -07:00
{
2006-05-13 18:07:53 +01:00
struct nand_chip * this = ( struct nand_chip * ) & h1910_nand_mtd [ 1 ] ;
2005-11-07 11:15:49 +00:00
2005-04-16 15:20:36 -07:00
/* Release resources, unregister device */
2006-05-13 18:07:53 +01:00
nand_release ( h1910_nand_mtd ) ;
2005-04-16 15:20:36 -07:00
/* Release io resource */
2006-05-13 18:07:53 +01:00
iounmap ( ( void * ) this - > IO_ADDR_W ) ;
2005-04-16 15:20:36 -07:00
/* Free the MTD device structure */
2006-05-13 18:07:53 +01:00
kfree ( h1910_nand_mtd ) ;
2005-04-16 15:20:36 -07:00
}
2006-05-13 18:07:53 +01:00
2005-04-16 15:20:36 -07:00
module_exit ( h1910_cleanup ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Joshua Wise <joshua at joshuawise dot com> " ) ;
MODULE_DESCRIPTION ( " NAND flash driver for iPAQ h1910 " ) ;