233 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* MTD-based superblock management
 | 
						|
 *
 | 
						|
 * Copyright © 2001-2007 Red Hat, Inc. All Rights Reserved.
 | 
						|
 * 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>
 | 
						|
 | 
						|
/*
 | 
						|
 * 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);
 | 
						|
	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);
 | 
						|
 | 
						|
	sb->s_flags = flags;
 | 
						|
 | 
						|
	ret = fill_super(sb, data, flags & MS_SILENT ? 1 : 0);
 | 
						|
	if (ret < 0) {
 | 
						|
		deactivate_locked_super(sb);
 | 
						|
		return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	/* go */
 | 
						|
	sb->s_flags |= MS_ACTIVE;
 | 
						|
	simple_set_mnt(mnt, sb);
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
	/* new mountpoint for an already mounted superblock */
 | 
						|
already_mounted:
 | 
						|
	DEBUG(1, "MTDSB: Device %d (\"%s\") is already mounted\n",
 | 
						|
	      mtd->index, mtd->name);
 | 
						|
	simple_set_mnt(mnt, sb);
 | 
						|
	ret = 0;
 | 
						|
	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);
 | 
						|
	if (IS_ERR(mtd)) {
 | 
						|
		DEBUG(0, "MTDSB: Device #%u doesn't appear to exist\n", mtdnr);
 | 
						|
		return PTR_ERR(mtd);
 | 
						|
	}
 | 
						|
 | 
						|
	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)
 | 
						|
{
 | 
						|
#ifdef CONFIG_BLOCK
 | 
						|
	struct block_device *bdev;
 | 
						|
	int ret, major;
 | 
						|
#endif
 | 
						|
	int mtdnr;
 | 
						|
 | 
						|
	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);
 | 
						|
 | 
						|
			for (mtdnr = 0; mtdnr < MAX_MTD_DEVICES; mtdnr++) {
 | 
						|
				mtd = get_mtd_device(NULL, mtdnr);
 | 
						|
				if (!IS_ERR(mtd)) {
 | 
						|
					if (!strcmp(mtd->name, dev_name + 4))
 | 
						|
						return get_sb_mtd_aux(
 | 
						|
							fs_type, flags,
 | 
						|
							dev_name, data, mtd,
 | 
						|
							fill_super, mnt);
 | 
						|
 | 
						|
					put_mtd_device(mtd);
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			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);
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
#ifdef CONFIG_BLOCK
 | 
						|
	/* try the old way - the hack where we allowed users to mount
 | 
						|
	 * /dev/mtdblock$(n) but didn't actually _use_ the blockdev
 | 
						|
	 */
 | 
						|
	bdev = lookup_bdev(dev_name);
 | 
						|
	if (IS_ERR(bdev)) {
 | 
						|
		ret = PTR_ERR(bdev);
 | 
						|
		DEBUG(1, "MTDSB: lookup_bdev() returned %d\n", ret);
 | 
						|
		return ret;
 | 
						|
	}
 | 
						|
	DEBUG(1, "MTDSB: lookup_bdev() returned 0\n");
 | 
						|
 | 
						|
	ret = -EINVAL;
 | 
						|
 | 
						|
	major = MAJOR(bdev->bd_dev);
 | 
						|
	mtdnr = MINOR(bdev->bd_dev);
 | 
						|
	bdput(bdev);
 | 
						|
 | 
						|
	if (major != MTD_BLOCK_MAJOR)
 | 
						|
		goto not_an_MTD_device;
 | 
						|
 | 
						|
	return get_sb_mtd_nr(fs_type, flags, dev_name, data, mtdnr, fill_super,
 | 
						|
			     mnt);
 | 
						|
 | 
						|
not_an_MTD_device:
 | 
						|
#endif /* CONFIG_BLOCK */
 | 
						|
 | 
						|
	if (!(flags & MS_SILENT))
 | 
						|
		printk(KERN_NOTICE
 | 
						|
		       "MTD: Attempt to mount non-MTD device \"%s\"\n",
 | 
						|
		       dev_name);
 | 
						|
	return -EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
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);
 |