NetBSD-5.0.2/external/gpl2/libdevmapper/dist/kernel/fs/dmfs-super.c

Compare this file to the similar file:
Show the results in this format:

/*
 * dmfs-super.c
 *
 * Copyright (C) 2001 Sistina Software
 *
 * This software 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, or (at
 * your option) any later version.
 *
 * This software 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.
 *
 * You should have received a copy of the GNU General Public License
 * along with GNU CC; see the file COPYING.  If not, write to
 * the Free Software Foundation, 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 */

#include "dm.h"
#include "dmfs.h"

#include <linux/init.h>
#include <linux/kmod.h>

#define DMFS_MAGIC 0x444D4653

extern struct inode *dmfs_create_root(struct super_block *sb, int);

static int dmfs_statfs(struct super_block *sb, struct statfs *buf)
{
	buf->f_type = sb->s_magic;
	buf->f_bsize = sb->s_blocksize;
	buf->f_namelen = DM_NAME_LEN - 1;

	return 0;
}

static void dmfs_delete_inode(struct inode *inode)
{
	if (S_ISDIR(inode->i_mode)) {
		struct dmfs_i *dmi = DMFS_I(inode);

		if (dmi) {
			if (dmi->md)
				BUG();
			if (!list_empty(&dmi->errors))
				dmfs_zap_errors(inode);
			kfree(dmi);
			MOD_DEC_USE_COUNT;	/* Don't remove */
		}
	}

	inode->u.generic_ip = NULL;
	clear_inode(inode);
}

static struct super_operations dmfs_super_operations = {
	statfs:		dmfs_statfs,
	put_inode:	force_delete,
	delete_inode:	dmfs_delete_inode,
};

static struct super_block *dmfs_read_super(struct super_block *sb, void *data,
					   int silent)
{
	struct inode *inode;
	struct dentry *root;

	sb->s_blocksize = PAGE_CACHE_SIZE;
	sb->s_blocksize_bits = PAGE_CACHE_SHIFT;
	sb->s_magic = DMFS_MAGIC;
	sb->s_op = &dmfs_super_operations;
	sb->s_maxbytes = MAX_NON_LFS;

	inode = dmfs_create_root(sb, 0755);
	if (IS_ERR(inode))
		return NULL;
	root = d_alloc_root(inode);
	if (!root) {
		iput(inode);
		return NULL;
	}
	sb->s_root = root;

	return sb;
}

struct inode *dmfs_new_inode(struct super_block *sb, int mode)
{
	struct inode *inode = new_inode(sb);

	if (inode) {
		inode->i_mode = mode;
		inode->i_uid = current->fsuid;
		inode->i_gid = current->fsgid;
		inode->i_blksize = PAGE_CACHE_SIZE;
		inode->i_blocks = 0;
		inode->i_rdev = NODEV;
		inode->i_atime = inode->i_ctime = inode->i_mtime = CURRENT_TIME;
	}

	return inode;
}

struct inode *dmfs_new_private_inode(struct super_block *sb, int mode)
{
	struct inode *inode = dmfs_new_inode(sb, mode);
	struct dmfs_i *dmi;

	if (inode) {
		dmi = kmalloc(sizeof(struct dmfs_i), GFP_KERNEL);
		if (dmi == NULL) {
			iput(inode);
			return NULL;
		}
		memset(dmi, 0, sizeof(struct dmfs_i));
		init_MUTEX(&dmi->sem);
		INIT_LIST_HEAD(&dmi->errors);
		inode->u.generic_ip = dmi;
		MOD_INC_USE_COUNT;	/* Don't remove */
	}

	return inode;
}

static DECLARE_FSTYPE(dmfs_fstype, "dmfs", dmfs_read_super, FS_SINGLE);
static struct vfsmount *dmfs_mnt;

int __init dm_interface_init(void)
{
	int ret;

	ret = register_filesystem(&dmfs_fstype);
	if (ret < 0)
		goto out;

	dmfs_mnt = kern_mount(&dmfs_fstype);
	if (IS_ERR(dmfs_mnt)) {
		ret = PTR_ERR(dmfs_mnt);
		unregister_filesystem(&dmfs_fstype);
	} else 
		MOD_DEC_USE_COUNT;	/* Yes, this really is correct... */
	
      out:
	return ret;
}

void __exit dm_interface_exit(void)
{
	MOD_INC_USE_COUNT;	/* So that it lands up being zero */

	do_umount(dmfs_mnt, 0);

	unregister_filesystem(&dmfs_fstype);
}