2005-04-17 02:20:36 +04:00
/*
2005-11-07 14:15:26 +03:00
* $ Id : mtdchar . c , v 1.76 2005 / 11 / 07 11 : 14 : 20 gleixner Exp $
2005-04-17 02:20:36 +04:00
*
* Character - device access to raw MTD devices .
*
*/
# include <linux/config.h>
2005-11-07 02:14:42 +03:00
# include <linux/device.h>
# include <linux/fs.h>
# include <linux/init.h>
2005-04-17 02:20:36 +04:00
# include <linux/kernel.h>
# include <linux/module.h>
2005-11-07 02:14:42 +03:00
# include <linux/slab.h>
# include <linux/sched.h>
2005-04-17 02:20:36 +04:00
# include <linux/mtd/mtd.h>
# include <linux/mtd/compatmac.h>
2005-11-07 02:14:42 +03:00
# include <asm/uaccess.h>
2005-06-30 04:23:27 +04:00
static struct class * mtd_class ;
2005-04-17 02:20:36 +04:00
static void mtd_notify_add ( struct mtd_info * mtd )
{
if ( ! mtd )
return ;
2005-10-28 09:25:43 +04:00
class_device_create ( mtd_class , NULL , MKDEV ( MTD_CHAR_MAJOR , mtd - > index * 2 ) ,
2005-06-30 04:23:27 +04:00
NULL , " mtd%d " , mtd - > index ) ;
2005-11-07 14:15:26 +03:00
2005-10-28 09:25:43 +04:00
class_device_create ( mtd_class , NULL ,
2005-06-30 04:23:27 +04:00
MKDEV ( MTD_CHAR_MAJOR , mtd - > index * 2 + 1 ) ,
NULL , " mtd%dro " , mtd - > index ) ;
2005-04-17 02:20:36 +04:00
}
static void mtd_notify_remove ( struct mtd_info * mtd )
{
if ( ! mtd )
return ;
2005-06-30 04:23:27 +04:00
class_device_destroy ( mtd_class , MKDEV ( MTD_CHAR_MAJOR , mtd - > index * 2 ) ) ;
class_device_destroy ( mtd_class , MKDEV ( MTD_CHAR_MAJOR , mtd - > index * 2 + 1 ) ) ;
2005-04-17 02:20:36 +04:00
}
static struct mtd_notifier notifier = {
. add = mtd_notify_add ,
. remove = mtd_notify_remove ,
} ;
2005-02-08 22:12:53 +03:00
/*
* We use file - > private_data to store a pointer to the MTDdevice .
* Since alighment is at least 32 bits , we have 2 bits free for OTP
* modes as well .
*/
# define TO_MTD(file) (struct mtd_info *)((long)((file)->private_data) & ~3L)
2005-02-08 20:45:55 +03:00
2005-02-08 22:12:53 +03:00
# define MTD_MODE_OTP_FACT 1
# define MTD_MODE_OTP_USER 2
# define MTD_MODE(file) ((long)((file)->private_data) & 3)
# define SET_MTD_MODE(file, mode) \
do { long __p = ( long ) ( ( file ) - > private_data ) ; \
( file ) - > private_data = ( void * ) ( ( __p & ~ 3L ) | mode ) ; } while ( 0 )
2005-02-08 20:45:55 +03:00
2005-04-17 02:20:36 +04:00
static loff_t mtd_lseek ( struct file * file , loff_t offset , int orig )
{
2005-02-08 22:12:53 +03:00
struct mtd_info * mtd = TO_MTD ( file ) ;
2005-04-17 02:20:36 +04:00
switch ( orig ) {
case 0 :
/* SEEK_SET */
break ;
case 1 :
/* SEEK_CUR */
2005-08-04 05:05:51 +04:00
offset + = file - > f_pos ;
2005-04-17 02:20:36 +04:00
break ;
case 2 :
/* SEEK_END */
2005-08-04 05:05:51 +04:00
offset + = mtd - > size ;
2005-04-17 02:20:36 +04:00
break ;
default :
return - EINVAL ;
}
2005-08-04 05:05:51 +04:00
if ( offset > = 0 & & offset < mtd - > size )
return file - > f_pos = offset ;
2005-04-17 02:20:36 +04:00
2005-08-04 05:05:51 +04:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
static int mtd_open ( struct inode * inode , struct file * file )
{
int minor = iminor ( inode ) ;
int devnum = minor > > 1 ;
struct mtd_info * mtd ;
DEBUG ( MTD_DEBUG_LEVEL0 , " MTD_open \n " ) ;
if ( devnum > = MAX_MTD_DEVICES )
return - ENODEV ;
/* You can't open the RO devices RW */
if ( ( file - > f_mode & 2 ) & & ( minor & 1 ) )
return - EACCES ;
mtd = get_mtd_device ( NULL , devnum ) ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( ! mtd )
return - ENODEV ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( MTD_ABSENT = = mtd - > type ) {
put_mtd_device ( mtd ) ;
return - ENODEV ;
}
file - > private_data = mtd ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
/* You can't open it RW if it's not a writeable device */
if ( ( file - > f_mode & 2 ) & & ! ( mtd - > flags & MTD_WRITEABLE ) ) {
put_mtd_device ( mtd ) ;
return - EACCES ;
}
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
return 0 ;
} /* mtd_open */
/*====================================================================*/
static int mtd_close ( struct inode * inode , struct file * file )
{
struct mtd_info * mtd ;
DEBUG ( MTD_DEBUG_LEVEL0 , " MTD_close \n " ) ;
2005-02-08 22:12:53 +03:00
mtd = TO_MTD ( file ) ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( mtd - > sync )
mtd - > sync ( mtd ) ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
put_mtd_device ( mtd ) ;
return 0 ;
} /* mtd_close */
/* FIXME: This _really_ needs to die. In 2.5, we should lock the
userspace buffer down and use it directly with readv / writev .
*/
# define MAX_KMALLOC_SIZE 0x20000
static ssize_t mtd_read ( struct file * file , char __user * buf , size_t count , loff_t * ppos )
{
2005-02-08 22:12:53 +03:00
struct mtd_info * mtd = TO_MTD ( file ) ;
2005-04-17 02:20:36 +04:00
size_t retlen = 0 ;
size_t total_retlen = 0 ;
int ret = 0 ;
int len ;
char * kbuf ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
DEBUG ( MTD_DEBUG_LEVEL0 , " MTD_read \n " ) ;
if ( * ppos + count > mtd - > size )
count = mtd - > size - * ppos ;
if ( ! count )
return 0 ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
/* FIXME: Use kiovec in 2.5 to lock down the user's buffers
and pass them directly to the MTD functions */
while ( count ) {
2005-11-07 14:15:26 +03:00
if ( count > MAX_KMALLOC_SIZE )
2005-04-17 02:20:36 +04:00
len = MAX_KMALLOC_SIZE ;
else
len = count ;
kbuf = kmalloc ( len , GFP_KERNEL ) ;
if ( ! kbuf )
return - ENOMEM ;
2005-11-07 14:15:26 +03:00
2005-02-08 22:12:53 +03:00
switch ( MTD_MODE ( file ) ) {
2005-02-08 20:45:55 +03:00
case MTD_MODE_OTP_FACT :
ret = mtd - > read_fact_prot_reg ( mtd , * ppos , len , & retlen , kbuf ) ;
break ;
case MTD_MODE_OTP_USER :
ret = mtd - > read_user_prot_reg ( mtd , * ppos , len , & retlen , kbuf ) ;
break ;
default :
ret = MTD_READ ( mtd , * ppos , len , & retlen , kbuf ) ;
}
2005-04-17 02:20:36 +04:00
/* Nand returns -EBADMSG on ecc errors, but it returns
* the data . For our userspace tools it is important
2005-11-07 14:15:26 +03:00
* to dump areas with ecc errors !
2005-04-17 02:20:36 +04:00
* Userspace software which accesses NAND this way
* must be aware of the fact that it deals with NAND
*/
if ( ! ret | | ( ret = = - EBADMSG ) ) {
* ppos + = retlen ;
if ( copy_to_user ( buf , kbuf , retlen ) ) {
kfree ( kbuf ) ;
return - EFAULT ;
}
else
total_retlen + = retlen ;
count - = retlen ;
buf + = retlen ;
2005-02-08 20:45:55 +03:00
if ( retlen = = 0 )
count = 0 ;
2005-04-17 02:20:36 +04:00
}
else {
kfree ( kbuf ) ;
return ret ;
}
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
kfree ( kbuf ) ;
}
return total_retlen ;
} /* mtd_read */
static ssize_t mtd_write ( struct file * file , const char __user * buf , size_t count , loff_t * ppos )
{
2005-02-08 22:12:53 +03:00
struct mtd_info * mtd = TO_MTD ( file ) ;
2005-04-17 02:20:36 +04:00
char * kbuf ;
size_t retlen ;
size_t total_retlen = 0 ;
int ret = 0 ;
int len ;
DEBUG ( MTD_DEBUG_LEVEL0 , " MTD_write \n " ) ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( * ppos = = mtd - > size )
return - ENOSPC ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( * ppos + count > mtd - > size )
count = mtd - > size - * ppos ;
if ( ! count )
return 0 ;
while ( count ) {
2005-11-07 14:15:26 +03:00
if ( count > MAX_KMALLOC_SIZE )
2005-04-17 02:20:36 +04:00
len = MAX_KMALLOC_SIZE ;
else
len = count ;
kbuf = kmalloc ( len , GFP_KERNEL ) ;
if ( ! kbuf ) {
printk ( " kmalloc is null \n " ) ;
return - ENOMEM ;
}
if ( copy_from_user ( kbuf , buf , len ) ) {
kfree ( kbuf ) ;
return - EFAULT ;
}
2005-11-07 14:15:26 +03:00
2005-02-08 22:12:53 +03:00
switch ( MTD_MODE ( file ) ) {
2005-02-08 20:45:55 +03:00
case MTD_MODE_OTP_FACT :
ret = - EROFS ;
break ;
case MTD_MODE_OTP_USER :
if ( ! mtd - > write_user_prot_reg ) {
ret = - EOPNOTSUPP ;
break ;
}
ret = mtd - > write_user_prot_reg ( mtd , * ppos , len , & retlen , kbuf ) ;
break ;
default :
ret = ( * ( mtd - > write ) ) ( mtd , * ppos , len , & retlen , kbuf ) ;
}
2005-04-17 02:20:36 +04:00
if ( ! ret ) {
* ppos + = retlen ;
total_retlen + = retlen ;
count - = retlen ;
buf + = retlen ;
}
else {
kfree ( kbuf ) ;
return ret ;
}
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
kfree ( kbuf ) ;
}
return total_retlen ;
} /* mtd_write */
/*======================================================================
IOCTL calls for getting device parameters .
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
static void mtdchar_erase_callback ( struct erase_info * instr )
{
wake_up ( ( wait_queue_head_t * ) instr - > priv ) ;
}
static int mtd_ioctl ( struct inode * inode , struct file * file ,
u_int cmd , u_long arg )
{
2005-02-08 22:12:53 +03:00
struct mtd_info * mtd = TO_MTD ( file ) ;
2005-04-17 02:20:36 +04:00
void __user * argp = ( void __user * ) arg ;
int ret = 0 ;
u_long size ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
DEBUG ( MTD_DEBUG_LEVEL0 , " MTD_ioctl \n " ) ;
size = ( cmd & IOCSIZE_MASK ) > > IOCSIZE_SHIFT ;
if ( cmd & IOC_IN ) {
if ( ! access_ok ( VERIFY_READ , argp , size ) )
return - EFAULT ;
}
if ( cmd & IOC_OUT ) {
if ( ! access_ok ( VERIFY_WRITE , argp , size ) )
return - EFAULT ;
}
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
switch ( cmd ) {
case MEMGETREGIONCOUNT :
if ( copy_to_user ( argp , & ( mtd - > numeraseregions ) , sizeof ( int ) ) )
return - EFAULT ;
break ;
case MEMGETREGIONINFO :
{
struct region_info_user ur ;
if ( copy_from_user ( & ur , argp , sizeof ( struct region_info_user ) ) )
return - EFAULT ;
if ( ur . regionindex > = mtd - > numeraseregions )
return - EINVAL ;
if ( copy_to_user ( argp , & ( mtd - > eraseregions [ ur . regionindex ] ) ,
sizeof ( struct mtd_erase_region_info ) ) )
return - EFAULT ;
break ;
}
case MEMGETINFO :
if ( copy_to_user ( argp , mtd , sizeof ( struct mtd_info_user ) ) )
return - EFAULT ;
break ;
case MEMERASE :
{
struct erase_info * erase ;
if ( ! ( file - > f_mode & 2 ) )
return - EPERM ;
erase = kmalloc ( sizeof ( struct erase_info ) , GFP_KERNEL ) ;
if ( ! erase )
ret = - ENOMEM ;
else {
wait_queue_head_t waitq ;
DECLARE_WAITQUEUE ( wait , current ) ;
init_waitqueue_head ( & waitq ) ;
memset ( erase , 0 , sizeof ( struct erase_info ) ) ;
if ( copy_from_user ( & erase - > addr , argp ,
sizeof ( struct erase_info_user ) ) ) {
kfree ( erase ) ;
return - EFAULT ;
}
erase - > mtd = mtd ;
erase - > callback = mtdchar_erase_callback ;
erase - > priv = ( unsigned long ) & waitq ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
/*
FIXME : Allow INTERRUPTIBLE . Which means
not having the wait_queue head on the stack .
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
If the wq_head is on the stack , and we
leave because we got interrupted , then the
wq_head is no longer there when the
callback routine tries to wake us up .
*/
ret = mtd - > erase ( mtd , erase ) ;
if ( ! ret ) {
set_current_state ( TASK_UNINTERRUPTIBLE ) ;
add_wait_queue ( & waitq , & wait ) ;
if ( erase - > state ! = MTD_ERASE_DONE & &
erase - > state ! = MTD_ERASE_FAILED )
schedule ( ) ;
remove_wait_queue ( & waitq , & wait ) ;
set_current_state ( TASK_RUNNING ) ;
ret = ( erase - > state = = MTD_ERASE_FAILED ) ? - EIO : 0 ;
}
kfree ( erase ) ;
}
break ;
}
case MEMWRITEOOB :
{
struct mtd_oob_buf buf ;
void * databuf ;
ssize_t retlen ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( ! ( file - > f_mode & 2 ) )
return - EPERM ;
if ( copy_from_user ( & buf , argp , sizeof ( struct mtd_oob_buf ) ) )
return - EFAULT ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( buf . length > 0x4096 )
return - EINVAL ;
if ( ! mtd - > write_oob )
ret = - EOPNOTSUPP ;
else
ret = access_ok ( VERIFY_READ , buf . ptr ,
buf . length ) ? 0 : EFAULT ;
if ( ret )
return ret ;
databuf = kmalloc ( buf . length , GFP_KERNEL ) ;
if ( ! databuf )
return - ENOMEM ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( copy_from_user ( databuf , buf . ptr , buf . length ) ) {
kfree ( databuf ) ;
return - EFAULT ;
}
ret = ( mtd - > write_oob ) ( mtd , buf . start , buf . length , & retlen , databuf ) ;
if ( copy_to_user ( argp + sizeof ( uint32_t ) , & retlen , sizeof ( uint32_t ) ) )
ret = - EFAULT ;
kfree ( databuf ) ;
break ;
}
case MEMREADOOB :
{
struct mtd_oob_buf buf ;
void * databuf ;
ssize_t retlen ;
if ( copy_from_user ( & buf , argp , sizeof ( struct mtd_oob_buf ) ) )
return - EFAULT ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( buf . length > 0x4096 )
return - EINVAL ;
if ( ! mtd - > read_oob )
ret = - EOPNOTSUPP ;
else
ret = access_ok ( VERIFY_WRITE , buf . ptr ,
buf . length ) ? 0 : - EFAULT ;
if ( ret )
return ret ;
databuf = kmalloc ( buf . length , GFP_KERNEL ) ;
if ( ! databuf )
return - ENOMEM ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
ret = ( mtd - > read_oob ) ( mtd , buf . start , buf . length , & retlen , databuf ) ;
if ( put_user ( retlen , ( uint32_t __user * ) argp ) )
ret = - EFAULT ;
else if ( retlen & & copy_to_user ( buf . ptr , databuf , retlen ) )
ret = - EFAULT ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
kfree ( databuf ) ;
break ;
}
case MEMLOCK :
{
struct erase_info_user info ;
if ( copy_from_user ( & info , argp , sizeof ( info ) ) )
return - EFAULT ;
if ( ! mtd - > lock )
ret = - EOPNOTSUPP ;
else
ret = mtd - > lock ( mtd , info . start , info . length ) ;
break ;
}
case MEMUNLOCK :
{
struct erase_info_user info ;
if ( copy_from_user ( & info , argp , sizeof ( info ) ) )
return - EFAULT ;
if ( ! mtd - > unlock )
ret = - EOPNOTSUPP ;
else
ret = mtd - > unlock ( mtd , info . start , info . length ) ;
break ;
}
case MEMSETOOBSEL :
{
if ( copy_from_user ( & mtd - > oobinfo , argp , sizeof ( struct nand_oobinfo ) ) )
return - EFAULT ;
break ;
}
case MEMGETOOBSEL :
{
if ( copy_to_user ( argp , & ( mtd - > oobinfo ) , sizeof ( struct nand_oobinfo ) ) )
return - EFAULT ;
break ;
}
case MEMGETBADBLOCK :
{
loff_t offs ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
if ( copy_from_user ( & offs , argp , sizeof ( loff_t ) ) )
return - EFAULT ;
if ( ! mtd - > block_isbad )
ret = - EOPNOTSUPP ;
else
return mtd - > block_isbad ( mtd , offs ) ;
break ;
}
case MEMSETBADBLOCK :
{
loff_t offs ;
if ( copy_from_user ( & offs , argp , sizeof ( loff_t ) ) )
return - EFAULT ;
if ( ! mtd - > block_markbad )
ret = - EOPNOTSUPP ;
else
return mtd - > block_markbad ( mtd , offs ) ;
break ;
}
2005-02-08 20:45:55 +03:00
# ifdef CONFIG_MTD_OTP
case OTPSELECT :
{
int mode ;
if ( copy_from_user ( & mode , argp , sizeof ( int ) ) )
return - EFAULT ;
2005-02-08 22:12:53 +03:00
SET_MTD_MODE ( file , 0 ) ;
2005-02-08 20:45:55 +03:00
switch ( mode ) {
case MTD_OTP_FACTORY :
if ( ! mtd - > read_fact_prot_reg )
ret = - EOPNOTSUPP ;
else
2005-02-08 22:12:53 +03:00
SET_MTD_MODE ( file , MTD_MODE_OTP_FACT ) ;
2005-02-08 20:45:55 +03:00
break ;
case MTD_OTP_USER :
if ( ! mtd - > read_fact_prot_reg )
ret = - EOPNOTSUPP ;
else
2005-02-08 22:12:53 +03:00
SET_MTD_MODE ( file , MTD_MODE_OTP_USER ) ;
2005-02-08 20:45:55 +03:00
break ;
default :
ret = - EINVAL ;
case MTD_OTP_OFF :
break ;
}
2005-04-01 19:36:15 +04:00
file - > f_pos = 0 ;
2005-02-08 20:45:55 +03:00
break ;
}
case OTPGETREGIONCOUNT :
case OTPGETREGIONINFO :
{
struct otp_info * buf = kmalloc ( 4096 , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
ret = - EOPNOTSUPP ;
2005-02-08 22:12:53 +03:00
switch ( MTD_MODE ( file ) ) {
2005-02-08 20:45:55 +03:00
case MTD_MODE_OTP_FACT :
if ( mtd - > get_fact_prot_info )
ret = mtd - > get_fact_prot_info ( mtd , buf , 4096 ) ;
break ;
case MTD_MODE_OTP_USER :
if ( mtd - > get_user_prot_info )
ret = mtd - > get_user_prot_info ( mtd , buf , 4096 ) ;
break ;
}
if ( ret > = 0 ) {
if ( cmd = = OTPGETREGIONCOUNT ) {
int nbr = ret / sizeof ( struct otp_info ) ;
ret = copy_to_user ( argp , & nbr , sizeof ( int ) ) ;
} else
ret = copy_to_user ( argp , buf , ret ) ;
if ( ret )
ret = - EFAULT ;
}
kfree ( buf ) ;
break ;
}
case OTPLOCK :
{
struct otp_info info ;
2005-02-08 22:12:53 +03:00
if ( MTD_MODE ( file ) ! = MTD_MODE_OTP_USER )
2005-02-08 20:45:55 +03:00
return - EINVAL ;
if ( copy_from_user ( & info , argp , sizeof ( info ) ) )
return - EFAULT ;
if ( ! mtd - > lock_user_prot_reg )
return - EOPNOTSUPP ;
ret = mtd - > lock_user_prot_reg ( mtd , info . start , info . length ) ;
break ;
}
# endif
2005-04-17 02:20:36 +04:00
default :
ret = - ENOTTY ;
}
return ret ;
} /* memory_ioctl */
static struct file_operations mtd_fops = {
. owner = THIS_MODULE ,
. llseek = mtd_lseek ,
. read = mtd_read ,
. write = mtd_write ,
. ioctl = mtd_ioctl ,
. open = mtd_open ,
. release = mtd_close ,
} ;
static int __init init_mtdchar ( void )
{
if ( register_chrdev ( MTD_CHAR_MAJOR , " mtd " , & mtd_fops ) ) {
printk ( KERN_NOTICE " Can't allocate major number %d for Memory Technology Devices. \n " ,
MTD_CHAR_MAJOR ) ;
return - EAGAIN ;
}
2005-06-30 04:23:27 +04:00
mtd_class = class_create ( THIS_MODULE , " mtd " ) ;
if ( IS_ERR ( mtd_class ) ) {
printk ( KERN_ERR " Error creating mtd class. \n " ) ;
unregister_chrdev ( MTD_CHAR_MAJOR , " mtd " ) ;
2005-07-04 21:15:28 +04:00
return PTR_ERR ( mtd_class ) ;
2005-06-30 04:23:27 +04:00
}
register_mtd_user ( & notifier ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static void __exit cleanup_mtdchar ( void )
{
2005-06-30 04:23:27 +04:00
unregister_mtd_user ( & notifier ) ;
class_destroy ( mtd_class ) ;
2005-04-17 02:20:36 +04:00
unregister_chrdev ( MTD_CHAR_MAJOR , " mtd " ) ;
}
module_init ( init_mtdchar ) ;
module_exit ( cleanup_mtdchar ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " David Woodhouse <dwmw2@infradead.org> " ) ;
MODULE_DESCRIPTION ( " Direct character-device access to MTD devices " ) ;