/* * 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 #include #include #include "dmfs.h" #include "dm.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) { struct dmfs_i *dmi = DMFS_I(inode); if (dmi) { if (dmi->md) dm_remove(dmi->md); if (dmi->table) dm_table_destroy(dmi->table); if (dmi->dentry) dput(dmi->dentry); 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 dmfs_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; } int __exit dmfs_exit(void) { MOD_INC_USE_COUNT; /* So that it lands up being zero */ do_umount(dmfs_mnt, 0); unregister_filesystem(&dmfs_fstype); return 0; }