2007-05-10 22:51:50 -07:00
/* MTD-based superblock management
*
* Copyright © 2001 - 2007 Red Hat , Inc . All Rights Reserved .
2010-08-08 20:58:20 +01:00
* Copyright © 2001 - 2010 David Woodhouse < dwmw2 @ infradead . org >
*
2007-05-10 22:51:50 -07:00
* Written by : David Howells < dhowells @ redhat . com >
* David Woodhouse < dwmw2 @ infradead . org >
*
* 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 .
*/
# include <linux/mtd/super.h>
# include <linux/namei.h>
# include <linux/ctype.h>
2010-04-22 14:11:43 +02:00
# include <linux/slab.h>
2007-05-10 22:51:50 -07:00
/*
* compare superblocks to see if they ' re equivalent
* - they are if the underlying MTD device is the same
*/
static int get_sb_mtd_compare ( struct super_block * sb , void * _mtd )
{
struct mtd_info * mtd = _mtd ;
if ( sb - > s_mtd = = mtd ) {
DEBUG ( 2 , " MTDSB: Match on device %d ( \" %s \" ) \n " ,
mtd - > index , mtd - > name ) ;
return 1 ;
}
DEBUG ( 2 , " MTDSB: No match, device %d ( \" %s \" ), device %d ( \" %s \" ) \n " ,
sb - > s_mtd - > index , sb - > s_mtd - > name , mtd - > index , mtd - > name ) ;
return 0 ;
}
/*
* mark the superblock by the MTD device it is using
* - set the device number to be the correct MTD block device for pesuperstence
* of NFS exports
*/
static int get_sb_mtd_set ( struct super_block * sb , void * _mtd )
{
struct mtd_info * mtd = _mtd ;
sb - > s_mtd = mtd ;
sb - > s_dev = MKDEV ( MTD_BLOCK_MAJOR , mtd - > index ) ;
2010-04-22 14:11:43 +02:00
sb - > s_bdi = mtd - > backing_dev_info ;
2007-05-10 22:51:50 -07:00
return 0 ;
}
/*
* get a superblock on an MTD - backed filesystem
*/
static int get_sb_mtd_aux ( struct file_system_type * fs_type , int flags ,
const char * dev_name , void * data ,
struct mtd_info * mtd ,
int ( * fill_super ) ( struct super_block * , void * , int ) ,
struct vfsmount * mnt )
{
struct super_block * sb ;
int ret ;
sb = sget ( fs_type , get_sb_mtd_compare , get_sb_mtd_set , mtd ) ;
if ( IS_ERR ( sb ) )
goto out_error ;
if ( sb - > s_root )
goto already_mounted ;
/* fresh new superblock */
DEBUG ( 1 , " MTDSB: New superblock for device %d ( \" %s \" ) \n " ,
mtd - > index , mtd - > name ) ;
2007-08-30 17:21:19 +01:00
sb - > s_flags = flags ;
2007-05-10 22:51:50 -07:00
ret = fill_super ( sb , data , flags & MS_SILENT ? 1 : 0 ) ;
if ( ret < 0 ) {
2009-05-06 01:34:22 -04:00
deactivate_locked_super ( sb ) ;
2007-05-10 22:51:50 -07:00
return ret ;
}
/* go */
sb - > s_flags | = MS_ACTIVE ;
2009-03-04 12:06:34 -08:00
simple_set_mnt ( mnt , sb ) ;
return 0 ;
2007-05-10 22:51:50 -07:00
/* new mountpoint for an already mounted superblock */
already_mounted :
DEBUG ( 1 , " MTDSB: Device %d ( \" %s \" ) is already mounted \n " ,
mtd - > index , mtd - > name ) ;
2009-03-04 12:06:34 -08:00
simple_set_mnt ( mnt , sb ) ;
ret = 0 ;
2007-05-10 22:51:50 -07:00
goto out_put ;
out_error :
ret = PTR_ERR ( sb ) ;
out_put :
put_mtd_device ( mtd ) ;
return ret ;
}
/*
* get a superblock on an MTD - backed filesystem by MTD device number
*/
static int get_sb_mtd_nr ( struct file_system_type * fs_type , int flags ,
const char * dev_name , void * data , int mtdnr ,
int ( * fill_super ) ( struct super_block * , void * , int ) ,
struct vfsmount * mnt )
{
struct mtd_info * mtd ;
mtd = get_mtd_device ( NULL , mtdnr ) ;
2007-06-01 19:21:59 +01:00
if ( IS_ERR ( mtd ) ) {
2007-05-10 22:51:50 -07:00
DEBUG ( 0 , " MTDSB: Device #%u doesn't appear to exist \n " , mtdnr ) ;
2007-06-01 19:21:59 +01:00
return PTR_ERR ( mtd ) ;
2007-05-10 22:51:50 -07:00
}
return get_sb_mtd_aux ( fs_type , flags , dev_name , data , mtd , fill_super ,
mnt ) ;
}
/*
* set up an MTD - based superblock
*/
int get_sb_mtd ( struct file_system_type * fs_type , int flags ,
const char * dev_name , void * data ,
int ( * fill_super ) ( struct super_block * , void * , int ) ,
struct vfsmount * mnt )
{
2008-08-02 00:01:21 +01:00
# ifdef CONFIG_BLOCK
2008-08-01 05:00:11 -04:00
struct block_device * bdev ;
2008-08-02 00:01:21 +01:00
int ret , major ;
# endif
int mtdnr ;
2007-05-10 22:51:50 -07:00
if ( ! dev_name )
return - EINVAL ;
DEBUG ( 2 , " MTDSB: dev_name \" %s \" \n " , dev_name ) ;
/* the preferred way of mounting in future; especially when
* CONFIG_BLOCK = n - we specify the underlying MTD device by number or
* by name , so that we don ' t require block device support to be present
* in the kernel . */
if ( dev_name [ 0 ] = = ' m ' & & dev_name [ 1 ] = = ' t ' & & dev_name [ 2 ] = = ' d ' ) {
if ( dev_name [ 3 ] = = ' : ' ) {
struct mtd_info * mtd ;
/* mount by MTD device name */
DEBUG ( 1 , " MTDSB: mtd:%%s, name \" %s \" \n " ,
dev_name + 4 ) ;
2010-01-29 20:57:18 +00:00
mtd = get_mtd_device_nm ( dev_name + 4 ) ;
if ( ! IS_ERR ( mtd ) )
return get_sb_mtd_aux (
fs_type , flags ,
dev_name , data , mtd ,
fill_super , mnt ) ;
2007-05-10 22:51:50 -07:00
printk ( KERN_NOTICE " MTD: "
" MTD device with name \" %s \" not found. \n " ,
dev_name + 4 ) ;
} else if ( isdigit ( dev_name [ 3 ] ) ) {
/* mount by MTD device number name */
char * endptr ;
mtdnr = simple_strtoul ( dev_name + 3 , & endptr , 0 ) ;
if ( ! * endptr ) {
/* It was a valid number */
DEBUG ( 1 , " MTDSB: mtd%%d, mtdnr %d \n " ,
mtdnr ) ;
return get_sb_mtd_nr ( fs_type , flags ,
dev_name , data ,
mtdnr , fill_super , mnt ) ;
}
}
}
2008-08-02 00:01:21 +01:00
# ifdef CONFIG_BLOCK
2007-05-10 22:51:50 -07:00
/* try the old way - the hack where we allowed users to mount
* / dev / mtdblock $ ( n ) but didn ' t actually _use_ the blockdev
*/
2008-08-01 05:00:11 -04:00
bdev = lookup_bdev ( dev_name ) ;
if ( IS_ERR ( bdev ) ) {
ret = PTR_ERR ( bdev ) ;
DEBUG ( 1 , " MTDSB: lookup_bdev() returned %d \n " , ret ) ;
2007-05-10 22:51:50 -07:00
return ret ;
}
2008-08-01 05:00:11 -04:00
DEBUG ( 1 , " MTDSB: lookup_bdev() returned 0 \n " ) ;
2007-05-10 22:51:50 -07:00
2008-08-01 05:00:11 -04:00
ret = - EINVAL ;
2007-05-10 22:51:50 -07:00
2008-08-02 00:01:21 +01:00
major = MAJOR ( bdev - > bd_dev ) ;
2008-08-01 05:00:11 -04:00
mtdnr = MINOR ( bdev - > bd_dev ) ;
bdput ( bdev ) ;
2007-05-10 22:51:50 -07:00
2008-08-02 00:01:21 +01:00
if ( major ! = MTD_BLOCK_MAJOR )
goto not_an_MTD_device ;
2007-05-10 22:51:50 -07:00
return get_sb_mtd_nr ( fs_type , flags , dev_name , data , mtdnr , fill_super ,
mnt ) ;
not_an_MTD_device :
2008-08-02 00:01:21 +01:00
# endif /* CONFIG_BLOCK */
2007-05-10 22:51:50 -07:00
if ( ! ( flags & MS_SILENT ) )
printk ( KERN_NOTICE
" MTD: Attempt to mount non-MTD device \" %s \" \n " ,
dev_name ) ;
2008-08-02 00:01:21 +01:00
return - EINVAL ;
2007-05-10 22:51:50 -07:00
}
EXPORT_SYMBOL_GPL ( get_sb_mtd ) ;
/*
* destroy an MTD - based superblock
*/
void kill_mtd_super ( struct super_block * sb )
{
generic_shutdown_super ( sb ) ;
put_mtd_device ( sb - > s_mtd ) ;
sb - > s_mtd = NULL ;
}
EXPORT_SYMBOL_GPL ( kill_mtd_super ) ;