]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
convert nfsctl
authorAl Viro <viro@zeniv.linux.org.uk>
Mon, 26 Feb 2024 06:37:54 +0000 (01:37 -0500)
committerAl Viro <viro@zeniv.linux.org.uk>
Tue, 18 Nov 2025 04:59:27 +0000 (23:59 -0500)
One instance per net-ns.  There's a fixed subset (several files in root,
an optional symlink in root + initially empty /clients/) + per-client
subdirectory in /clients/.  Clients can appear only after the filesystem
is there and they are all gone before it gets through ->kill_sb().

Fixed subset created in fill_super(), regular files by simple_fill_super(),
then a subdirectory and a symlink - manually.  It is removed by
kill_litter_super().

Per-client subdirectories are created by nfsd_client_mkdir() (populated
with client-supplied list of files in them).  Removed by nfsd_client_rmdir(),
which is simple_recursive_removal().

All dentries except for the ones from simple_fill_super() come from
* nfsd_mkdir() (subdirectory, dentry from simple_start_creating()).
  Called from fill_super() (creates initially empty /clients)
  and from nfsd_client_mkdir (creates a per-client subdirectory
  in /clients).
* _nfsd_symlink() (symlink, dentry from simple_start_creating()), called
  from fill_super().
* nfsdfs_create_files() (regulars, dentry from simple_start_creating()),
  called only from nfsd_client_mkdir().

Turn d_instatiate() + inode_unlock() into d_make_persistent() + simple_done_creating()
in nfsd_mkdir(), _nfsd_symlink() and nfsdfs_create_files() and we are done.

Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
fs/nfsd/nfsctl.c

index 2b79129703d54b139fa981d6121d953d0215d1e4..5ce9a49e76ba409eb3cd184b9232740d581e73e4 100644 (file)
@@ -1137,11 +1137,11 @@ static struct dentry *nfsd_mkdir(struct dentry *parent, struct nfsdfs_client *nc
                inode->i_private = ncl;
                kref_get(&ncl->cl_ref);
        }
-       d_instantiate(dentry, inode);
+       d_make_persistent(dentry, inode);
        inc_nlink(dir);
        fsnotify_mkdir(dir, dentry);
-       inode_unlock(dir);
-       return dentry;
+       simple_done_creating(dentry);
+       return dentry;  // borrowed
 }
 
 #if IS_ENABLED(CONFIG_SUNRPC_GSS)
@@ -1170,9 +1170,9 @@ static void _nfsd_symlink(struct dentry *parent, const char *name,
        inode->i_link = (char *)content;
        inode->i_size = strlen(content);
 
-       d_instantiate(dentry, inode);
+       d_make_persistent(dentry, inode);
        fsnotify_create(dir, dentry);
-       inode_unlock(dir);
+       simple_done_creating(dentry);
 }
 #else
 static inline void _nfsd_symlink(struct dentry *parent, const char *name,
@@ -1228,11 +1228,11 @@ static int nfsdfs_create_files(struct dentry *root,
                kref_get(&ncl->cl_ref);
                inode->i_fop = files->ops;
                inode->i_private = ncl;
-               d_instantiate(dentry, inode);
+               d_make_persistent(dentry, inode);
                fsnotify_create(dir, dentry);
                if (fdentries)
-                       fdentries[i] = dentry;
-               inode_unlock(dir);
+                       fdentries[i] = dentry; // borrowed
+               simple_done_creating(dentry);
        }
        return 0;
 }
@@ -1346,7 +1346,7 @@ static void nfsd_umount(struct super_block *sb)
 
        nfsd_shutdown_threads(net);
 
-       kill_litter_super(sb);
+       kill_anon_super(sb);
        put_net(net);
 }