kernfs: Add support for always empty directories.
authorEric W. Biederman <ebiederm@xmission.com>
Wed, 13 May 2015 21:09:29 +0000 (16:09 -0500)
committerEric W. Biederman <ebiederm@xmission.com>
Wed, 1 Jul 2015 15:36:43 +0000 (10:36 -0500)
Add a new function kernfs_create_empty_dir that can be used to create
directory that can not be modified.

Update the code to use make_empty_dir_inode when reporting a
permanently empty directory to the vfs.

Update the code to not allow adding to permanently empty directories.

Cc: stable@vger.kernel.org
Signed-off-by: "Eric W. Biederman" <ebiederm@xmission.com>
fs/kernfs/dir.c
fs/kernfs/inode.c
include/linux/kernfs.h

index f131fc23ffc4c18f03a9764973fa998bf0e5f79e..47dc636d80edd76d6c65e3325e92359ed6f2e93d 100644 (file)
@@ -585,6 +585,9 @@ int kernfs_add_one(struct kernfs_node *kn)
                goto out_unlock;
 
        ret = -ENOENT;
+       if (parent->flags & KERNFS_EMPTY_DIR)
+               goto out_unlock;
+
        if ((parent->flags & KERNFS_ACTIVATED) && !kernfs_active(parent))
                goto out_unlock;
 
@@ -776,6 +779,38 @@ struct kernfs_node *kernfs_create_dir_ns(struct kernfs_node *parent,
        return ERR_PTR(rc);
 }
 
+/**
+ * kernfs_create_empty_dir - create an always empty directory
+ * @parent: parent in which to create a new directory
+ * @name: name of the new directory
+ *
+ * Returns the created node on success, ERR_PTR() value on failure.
+ */
+struct kernfs_node *kernfs_create_empty_dir(struct kernfs_node *parent,
+                                           const char *name)
+{
+       struct kernfs_node *kn;
+       int rc;
+
+       /* allocate */
+       kn = kernfs_new_node(parent, name, S_IRUGO|S_IXUGO|S_IFDIR, KERNFS_DIR);
+       if (!kn)
+               return ERR_PTR(-ENOMEM);
+
+       kn->flags |= KERNFS_EMPTY_DIR;
+       kn->dir.root = parent->dir.root;
+       kn->ns = NULL;
+       kn->priv = NULL;
+
+       /* link in */
+       rc = kernfs_add_one(kn);
+       if (!rc)
+               return kn;
+
+       kernfs_put(kn);
+       return ERR_PTR(rc);
+}
+
 static struct dentry *kernfs_iop_lookup(struct inode *dir,
                                        struct dentry *dentry,
                                        unsigned int flags)
@@ -1247,7 +1282,8 @@ int kernfs_rename_ns(struct kernfs_node *kn, struct kernfs_node *new_parent,
        mutex_lock(&kernfs_mutex);
 
        error = -ENOENT;
-       if (!kernfs_active(kn) || !kernfs_active(new_parent))
+       if (!kernfs_active(kn) || !kernfs_active(new_parent) ||
+           (new_parent->flags & KERNFS_EMPTY_DIR))
                goto out;
 
        error = 0;
index 2da8493a380b8a43a8eca65c04f4de245c4b0219..756dd56aaf60acd337fb251fb287cbd382d57740 100644 (file)
@@ -296,6 +296,8 @@ static void kernfs_init_inode(struct kernfs_node *kn, struct inode *inode)
        case KERNFS_DIR:
                inode->i_op = &kernfs_dir_iops;
                inode->i_fop = &kernfs_dir_fops;
+               if (kn->flags & KERNFS_EMPTY_DIR)
+                       make_empty_dir_inode(inode);
                break;
        case KERNFS_FILE:
                inode->i_size = kn->attr.size;
index 71ecdab1671b837b05f946ca44fa5f431f5f2859..29d1896c3ba55f1fc73eb96d512fa94117ab00e5 100644 (file)
@@ -45,6 +45,7 @@ enum kernfs_node_flag {
        KERNFS_LOCKDEP          = 0x0100,
        KERNFS_SUICIDAL         = 0x0400,
        KERNFS_SUICIDED         = 0x0800,
+       KERNFS_EMPTY_DIR        = 0x1000,
 };
 
 /* @flags for kernfs_create_root() */
@@ -285,6 +286,8 @@ void kernfs_destroy_root(struct kernfs_root *root);
 struct kernfs_node *kernfs_create_dir_ns(struct kernfs_node *parent,
                                         const char *name, umode_t mode,
                                         void *priv, const void *ns);
+struct kernfs_node *kernfs_create_empty_dir(struct kernfs_node *parent,
+                                           const char *name);
 struct kernfs_node *__kernfs_create_file(struct kernfs_node *parent,
                                         const char *name,
                                         umode_t mode, loff_t size,