Wrapfs: superblock operations
authorErez Zadok <ezk@cs.sunysb.edu>
Tue, 5 Jan 2010 01:45:06 +0000 (20:45 -0500)
committerErez Zadok <ezk@cs.sunysb.edu>
Sun, 26 May 2013 02:47:47 +0000 (22:47 -0400)
Signed-off-by: Erez Zadok <ezk@cs.sunysb.edu>
fs/wrapfs/super.c [new file with mode: 0644]

diff --git a/fs/wrapfs/super.c b/fs/wrapfs/super.c
new file mode 100644 (file)
index 0000000..9a6e886
--- /dev/null
@@ -0,0 +1,166 @@
+/*
+ * Copyright (c) 1998-2010 Erez Zadok
+ * Copyright (c) 2009     Shrikar Archak
+ * Copyright (c) 2003-2010 Stony Brook University
+ * Copyright (c) 2003-2010 The Research Foundation of SUNY
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ */
+
+#include "wrapfs.h"
+
+/*
+ * The inode cache is used with alloc_inode for both our inode info and the
+ * vfs inode.
+ */
+static struct kmem_cache *wrapfs_inode_cachep;
+
+/* final actions when unmounting a file system */
+static void wrapfs_put_super(struct super_block *sb)
+{
+       struct wrapfs_sb_info *spd;
+       struct super_block *s;
+
+       spd = WRAPFS_SB(sb);
+       if (!spd)
+               return;
+
+       /* decrement lower super references */
+       s = wrapfs_lower_super(sb);
+       wrapfs_set_lower_super(sb, NULL);
+       atomic_dec(&s->s_active);
+
+       kfree(spd);
+       sb->s_fs_info = NULL;
+}
+
+static int wrapfs_statfs(struct dentry *dentry, struct kstatfs *buf)
+{
+       int err;
+       struct path lower_path;
+
+       wrapfs_get_lower_path(dentry, &lower_path);
+       err = vfs_statfs(lower_path.dentry, buf);
+       wrapfs_put_lower_path(dentry, &lower_path);
+
+       /* set return buf to our f/s to avoid confusing user-level utils */
+       buf->f_type = WRAPFS_SUPER_MAGIC;
+
+       return err;
+}
+
+/*
+ * @flags: numeric mount options
+ * @options: mount options string
+ */
+static int wrapfs_remount_fs(struct super_block *sb, int *flags, char *options)
+{
+       int err = 0;
+
+       /*
+        * The VFS will take care of "ro" and "rw" flags among others.  We
+        * can safely accept a few flags (RDONLY, MANDLOCK), and honor
+        * SILENT, but anything else left over is an error.
+        */
+       if ((*flags & ~(MS_RDONLY | MS_MANDLOCK | MS_SILENT)) != 0) {
+               printk(KERN_ERR
+                      "wrapfs: remount flags 0x%x unsupported\n", *flags);
+               err = -EINVAL;
+       }
+
+       return err;
+}
+
+/*
+ * Called by iput() when the inode reference count reached zero
+ * and the inode is not hashed anywhere.  Used to clear anything
+ * that needs to be, before the inode is completely destroyed and put
+ * on the inode free list.
+ */
+static void wrapfs_clear_inode(struct inode *inode)
+{
+       struct inode *lower_inode;
+
+       /*
+        * Decrement a reference to a lower_inode, which was incremented
+        * by our read_inode when it was created initially.
+        */
+       lower_inode = wrapfs_lower_inode(inode);
+       wrapfs_set_lower_inode(inode, NULL);
+       iput(lower_inode);
+}
+
+static struct inode *wrapfs_alloc_inode(struct super_block *sb)
+{
+       struct wrapfs_inode_info *i;
+
+       i = kmem_cache_alloc(wrapfs_inode_cachep, GFP_KERNEL);
+       if (!i)
+               return NULL;
+
+       /* memset everything up to the inode to 0 */
+       memset(i, 0, offsetof(struct wrapfs_inode_info, vfs_inode));
+
+       i->vfs_inode.i_version = 1;
+       return &i->vfs_inode;
+}
+
+static void wrapfs_destroy_inode(struct inode *inode)
+{
+       kmem_cache_free(wrapfs_inode_cachep, WRAPFS_I(inode));
+}
+
+/* wrapfs inode cache constructor */
+static void init_once(void *obj)
+{
+       struct wrapfs_inode_info *i = obj;
+
+       inode_init_once(&i->vfs_inode);
+}
+
+int wrapfs_init_inode_cache(void)
+{
+       int err = 0;
+
+       wrapfs_inode_cachep =
+               kmem_cache_create("wrapfs_inode_cache",
+                                 sizeof(struct wrapfs_inode_info), 0,
+                                 SLAB_RECLAIM_ACCOUNT, init_once);
+       if (!wrapfs_inode_cachep)
+               err = -ENOMEM;
+       return err;
+}
+
+/* wrapfs inode cache destructor */
+void wrapfs_destroy_inode_cache(void)
+{
+       if (wrapfs_inode_cachep)
+               kmem_cache_destroy(wrapfs_inode_cachep);
+}
+
+/*
+ * Used only in nfs, to kill any pending RPC tasks, so that subsequent
+ * code can actually succeed and won't leave tasks that need handling.
+ */
+static void wrapfs_umount_begin(struct super_block *sb)
+{
+       struct super_block *lower_sb;
+
+       lower_sb = wrapfs_lower_super(sb);
+       if (lower_sb && lower_sb->s_op && lower_sb->s_op->umount_begin)
+               lower_sb->s_op->umount_begin(lower_sb);
+}
+
+const struct super_operations wrapfs_sops = {
+       .put_super      = wrapfs_put_super,
+       .statfs         = wrapfs_statfs,
+       .remount_fs     = wrapfs_remount_fs,
+       .clear_inode    = wrapfs_clear_inode,
+       .umount_begin   = wrapfs_umount_begin,
+       .show_options   = generic_show_options,
+       .alloc_inode    = wrapfs_alloc_inode,
+       .destroy_inode  = wrapfs_destroy_inode,
+       .drop_inode     = generic_delete_inode,
+};