162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *  linux/fs/nfs/unlink.c
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * nfs sillydelete handling
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/slab.h>
1062306a36Sopenharmony_ci#include <linux/string.h>
1162306a36Sopenharmony_ci#include <linux/dcache.h>
1262306a36Sopenharmony_ci#include <linux/sunrpc/sched.h>
1362306a36Sopenharmony_ci#include <linux/sunrpc/clnt.h>
1462306a36Sopenharmony_ci#include <linux/nfs_fs.h>
1562306a36Sopenharmony_ci#include <linux/sched.h>
1662306a36Sopenharmony_ci#include <linux/wait.h>
1762306a36Sopenharmony_ci#include <linux/namei.h>
1862306a36Sopenharmony_ci#include <linux/fsnotify.h>
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#include "internal.h"
2162306a36Sopenharmony_ci#include "nfs4_fs.h"
2262306a36Sopenharmony_ci#include "iostat.h"
2362306a36Sopenharmony_ci#include "delegation.h"
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci#include "nfstrace.h"
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci/**
2862306a36Sopenharmony_ci * nfs_free_unlinkdata - release data from a sillydelete operation.
2962306a36Sopenharmony_ci * @data: pointer to unlink structure.
3062306a36Sopenharmony_ci */
3162306a36Sopenharmony_cistatic void
3262306a36Sopenharmony_cinfs_free_unlinkdata(struct nfs_unlinkdata *data)
3362306a36Sopenharmony_ci{
3462306a36Sopenharmony_ci	put_cred(data->cred);
3562306a36Sopenharmony_ci	kfree(data->args.name.name);
3662306a36Sopenharmony_ci	kfree(data);
3762306a36Sopenharmony_ci}
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_ci/**
4062306a36Sopenharmony_ci * nfs_async_unlink_done - Sillydelete post-processing
4162306a36Sopenharmony_ci * @task: rpc_task of the sillydelete
4262306a36Sopenharmony_ci * @calldata: pointer to nfs_unlinkdata
4362306a36Sopenharmony_ci *
4462306a36Sopenharmony_ci * Do the directory attribute update.
4562306a36Sopenharmony_ci */
4662306a36Sopenharmony_cistatic void nfs_async_unlink_done(struct rpc_task *task, void *calldata)
4762306a36Sopenharmony_ci{
4862306a36Sopenharmony_ci	struct nfs_unlinkdata *data = calldata;
4962306a36Sopenharmony_ci	struct inode *dir = d_inode(data->dentry->d_parent);
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	trace_nfs_sillyrename_unlink(data, task->tk_status);
5262306a36Sopenharmony_ci	if (!NFS_PROTO(dir)->unlink_done(task, dir))
5362306a36Sopenharmony_ci		rpc_restart_call_prepare(task);
5462306a36Sopenharmony_ci}
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci/**
5762306a36Sopenharmony_ci * nfs_async_unlink_release - Release the sillydelete data.
5862306a36Sopenharmony_ci * @calldata: struct nfs_unlinkdata to release
5962306a36Sopenharmony_ci *
6062306a36Sopenharmony_ci * We need to call nfs_put_unlinkdata as a 'tk_release' task since the
6162306a36Sopenharmony_ci * rpc_task would be freed too.
6262306a36Sopenharmony_ci */
6362306a36Sopenharmony_cistatic void nfs_async_unlink_release(void *calldata)
6462306a36Sopenharmony_ci{
6562306a36Sopenharmony_ci	struct nfs_unlinkdata	*data = calldata;
6662306a36Sopenharmony_ci	struct dentry *dentry = data->dentry;
6762306a36Sopenharmony_ci	struct super_block *sb = dentry->d_sb;
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_ci	up_read_non_owner(&NFS_I(d_inode(dentry->d_parent))->rmdir_sem);
7062306a36Sopenharmony_ci	d_lookup_done(dentry);
7162306a36Sopenharmony_ci	nfs_free_unlinkdata(data);
7262306a36Sopenharmony_ci	dput(dentry);
7362306a36Sopenharmony_ci	nfs_sb_deactive(sb);
7462306a36Sopenharmony_ci}
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_cistatic void nfs_unlink_prepare(struct rpc_task *task, void *calldata)
7762306a36Sopenharmony_ci{
7862306a36Sopenharmony_ci	struct nfs_unlinkdata *data = calldata;
7962306a36Sopenharmony_ci	struct inode *dir = d_inode(data->dentry->d_parent);
8062306a36Sopenharmony_ci	NFS_PROTO(dir)->unlink_rpc_prepare(task, data);
8162306a36Sopenharmony_ci}
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_cistatic const struct rpc_call_ops nfs_unlink_ops = {
8462306a36Sopenharmony_ci	.rpc_call_done = nfs_async_unlink_done,
8562306a36Sopenharmony_ci	.rpc_release = nfs_async_unlink_release,
8662306a36Sopenharmony_ci	.rpc_call_prepare = nfs_unlink_prepare,
8762306a36Sopenharmony_ci};
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cistatic void nfs_do_call_unlink(struct inode *inode, struct nfs_unlinkdata *data)
9062306a36Sopenharmony_ci{
9162306a36Sopenharmony_ci	struct rpc_message msg = {
9262306a36Sopenharmony_ci		.rpc_argp = &data->args,
9362306a36Sopenharmony_ci		.rpc_resp = &data->res,
9462306a36Sopenharmony_ci		.rpc_cred = data->cred,
9562306a36Sopenharmony_ci	};
9662306a36Sopenharmony_ci	struct rpc_task_setup task_setup_data = {
9762306a36Sopenharmony_ci		.rpc_message = &msg,
9862306a36Sopenharmony_ci		.callback_ops = &nfs_unlink_ops,
9962306a36Sopenharmony_ci		.callback_data = data,
10062306a36Sopenharmony_ci		.workqueue = nfsiod_workqueue,
10162306a36Sopenharmony_ci		.flags = RPC_TASK_ASYNC | RPC_TASK_CRED_NOREF,
10262306a36Sopenharmony_ci	};
10362306a36Sopenharmony_ci	struct rpc_task *task;
10462306a36Sopenharmony_ci	struct inode *dir = d_inode(data->dentry->d_parent);
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	if (nfs_server_capable(inode, NFS_CAP_MOVEABLE))
10762306a36Sopenharmony_ci		task_setup_data.flags |= RPC_TASK_MOVEABLE;
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_ci	nfs_sb_active(dir->i_sb);
11062306a36Sopenharmony_ci	data->args.fh = NFS_FH(dir);
11162306a36Sopenharmony_ci	nfs_fattr_init(data->res.dir_attr);
11262306a36Sopenharmony_ci
11362306a36Sopenharmony_ci	NFS_PROTO(dir)->unlink_setup(&msg, data->dentry, inode);
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci	task_setup_data.rpc_client = NFS_CLIENT(dir);
11662306a36Sopenharmony_ci	task = rpc_run_task(&task_setup_data);
11762306a36Sopenharmony_ci	if (!IS_ERR(task))
11862306a36Sopenharmony_ci		rpc_put_task_async(task);
11962306a36Sopenharmony_ci}
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_cistatic int nfs_call_unlink(struct dentry *dentry, struct inode *inode, struct nfs_unlinkdata *data)
12262306a36Sopenharmony_ci{
12362306a36Sopenharmony_ci	struct inode *dir = d_inode(dentry->d_parent);
12462306a36Sopenharmony_ci	struct dentry *alias;
12562306a36Sopenharmony_ci
12662306a36Sopenharmony_ci	down_read_non_owner(&NFS_I(dir)->rmdir_sem);
12762306a36Sopenharmony_ci	alias = d_alloc_parallel(dentry->d_parent, &data->args.name, &data->wq);
12862306a36Sopenharmony_ci	if (IS_ERR(alias)) {
12962306a36Sopenharmony_ci		up_read_non_owner(&NFS_I(dir)->rmdir_sem);
13062306a36Sopenharmony_ci		return 0;
13162306a36Sopenharmony_ci	}
13262306a36Sopenharmony_ci	if (!d_in_lookup(alias)) {
13362306a36Sopenharmony_ci		int ret;
13462306a36Sopenharmony_ci		void *devname_garbage = NULL;
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci		/*
13762306a36Sopenharmony_ci		 * Hey, we raced with lookup... See if we need to transfer
13862306a36Sopenharmony_ci		 * the sillyrename information to the aliased dentry.
13962306a36Sopenharmony_ci		 */
14062306a36Sopenharmony_ci		spin_lock(&alias->d_lock);
14162306a36Sopenharmony_ci		if (d_really_is_positive(alias) &&
14262306a36Sopenharmony_ci		    !nfs_compare_fh(NFS_FH(inode), NFS_FH(d_inode(alias))) &&
14362306a36Sopenharmony_ci		    !(alias->d_flags & DCACHE_NFSFS_RENAMED)) {
14462306a36Sopenharmony_ci			devname_garbage = alias->d_fsdata;
14562306a36Sopenharmony_ci			alias->d_fsdata = data;
14662306a36Sopenharmony_ci			alias->d_flags |= DCACHE_NFSFS_RENAMED;
14762306a36Sopenharmony_ci			ret = 1;
14862306a36Sopenharmony_ci		} else
14962306a36Sopenharmony_ci			ret = 0;
15062306a36Sopenharmony_ci		spin_unlock(&alias->d_lock);
15162306a36Sopenharmony_ci		dput(alias);
15262306a36Sopenharmony_ci		up_read_non_owner(&NFS_I(dir)->rmdir_sem);
15362306a36Sopenharmony_ci		/*
15462306a36Sopenharmony_ci		 * If we'd displaced old cached devname, free it.  At that
15562306a36Sopenharmony_ci		 * point dentry is definitely not a root, so we won't need
15662306a36Sopenharmony_ci		 * that anymore.
15762306a36Sopenharmony_ci		 */
15862306a36Sopenharmony_ci		kfree(devname_garbage);
15962306a36Sopenharmony_ci		return ret;
16062306a36Sopenharmony_ci	}
16162306a36Sopenharmony_ci	data->dentry = alias;
16262306a36Sopenharmony_ci	nfs_do_call_unlink(inode, data);
16362306a36Sopenharmony_ci	return 1;
16462306a36Sopenharmony_ci}
16562306a36Sopenharmony_ci
16662306a36Sopenharmony_ci/**
16762306a36Sopenharmony_ci * nfs_async_unlink - asynchronous unlinking of a file
16862306a36Sopenharmony_ci * @dentry: parent directory of dentry
16962306a36Sopenharmony_ci * @name: name of dentry to unlink
17062306a36Sopenharmony_ci */
17162306a36Sopenharmony_cistatic int
17262306a36Sopenharmony_cinfs_async_unlink(struct dentry *dentry, const struct qstr *name)
17362306a36Sopenharmony_ci{
17462306a36Sopenharmony_ci	struct nfs_unlinkdata *data;
17562306a36Sopenharmony_ci	int status = -ENOMEM;
17662306a36Sopenharmony_ci	void *devname_garbage = NULL;
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_ci	data = kzalloc(sizeof(*data), GFP_KERNEL);
17962306a36Sopenharmony_ci	if (data == NULL)
18062306a36Sopenharmony_ci		goto out;
18162306a36Sopenharmony_ci	data->args.name.name = kstrdup(name->name, GFP_KERNEL);
18262306a36Sopenharmony_ci	if (!data->args.name.name)
18362306a36Sopenharmony_ci		goto out_free;
18462306a36Sopenharmony_ci	data->args.name.len = name->len;
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_ci	data->cred = get_current_cred();
18762306a36Sopenharmony_ci	data->res.dir_attr = &data->dir_attr;
18862306a36Sopenharmony_ci	init_waitqueue_head(&data->wq);
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_ci	status = -EBUSY;
19162306a36Sopenharmony_ci	spin_lock(&dentry->d_lock);
19262306a36Sopenharmony_ci	if (dentry->d_flags & DCACHE_NFSFS_RENAMED)
19362306a36Sopenharmony_ci		goto out_unlock;
19462306a36Sopenharmony_ci	dentry->d_flags |= DCACHE_NFSFS_RENAMED;
19562306a36Sopenharmony_ci	devname_garbage = dentry->d_fsdata;
19662306a36Sopenharmony_ci	dentry->d_fsdata = data;
19762306a36Sopenharmony_ci	spin_unlock(&dentry->d_lock);
19862306a36Sopenharmony_ci	/*
19962306a36Sopenharmony_ci	 * If we'd displaced old cached devname, free it.  At that
20062306a36Sopenharmony_ci	 * point dentry is definitely not a root, so we won't need
20162306a36Sopenharmony_ci	 * that anymore.
20262306a36Sopenharmony_ci	 */
20362306a36Sopenharmony_ci	kfree(devname_garbage);
20462306a36Sopenharmony_ci	return 0;
20562306a36Sopenharmony_ciout_unlock:
20662306a36Sopenharmony_ci	spin_unlock(&dentry->d_lock);
20762306a36Sopenharmony_ci	put_cred(data->cred);
20862306a36Sopenharmony_ci	kfree(data->args.name.name);
20962306a36Sopenharmony_ciout_free:
21062306a36Sopenharmony_ci	kfree(data);
21162306a36Sopenharmony_ciout:
21262306a36Sopenharmony_ci	return status;
21362306a36Sopenharmony_ci}
21462306a36Sopenharmony_ci
21562306a36Sopenharmony_ci/**
21662306a36Sopenharmony_ci * nfs_complete_unlink - Initialize completion of the sillydelete
21762306a36Sopenharmony_ci * @dentry: dentry to delete
21862306a36Sopenharmony_ci * @inode: inode
21962306a36Sopenharmony_ci *
22062306a36Sopenharmony_ci * Since we're most likely to be called by dentry_iput(), we
22162306a36Sopenharmony_ci * only use the dentry to find the sillydelete. We then copy the name
22262306a36Sopenharmony_ci * into the qstr.
22362306a36Sopenharmony_ci */
22462306a36Sopenharmony_civoid
22562306a36Sopenharmony_cinfs_complete_unlink(struct dentry *dentry, struct inode *inode)
22662306a36Sopenharmony_ci{
22762306a36Sopenharmony_ci	struct nfs_unlinkdata	*data;
22862306a36Sopenharmony_ci
22962306a36Sopenharmony_ci	spin_lock(&dentry->d_lock);
23062306a36Sopenharmony_ci	dentry->d_flags &= ~DCACHE_NFSFS_RENAMED;
23162306a36Sopenharmony_ci	data = dentry->d_fsdata;
23262306a36Sopenharmony_ci	dentry->d_fsdata = NULL;
23362306a36Sopenharmony_ci	spin_unlock(&dentry->d_lock);
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	if (NFS_STALE(inode) || !nfs_call_unlink(dentry, inode, data))
23662306a36Sopenharmony_ci		nfs_free_unlinkdata(data);
23762306a36Sopenharmony_ci}
23862306a36Sopenharmony_ci
23962306a36Sopenharmony_ci/* Cancel a queued async unlink. Called when a sillyrename run fails. */
24062306a36Sopenharmony_cistatic void
24162306a36Sopenharmony_cinfs_cancel_async_unlink(struct dentry *dentry)
24262306a36Sopenharmony_ci{
24362306a36Sopenharmony_ci	spin_lock(&dentry->d_lock);
24462306a36Sopenharmony_ci	if (dentry->d_flags & DCACHE_NFSFS_RENAMED) {
24562306a36Sopenharmony_ci		struct nfs_unlinkdata *data = dentry->d_fsdata;
24662306a36Sopenharmony_ci
24762306a36Sopenharmony_ci		dentry->d_flags &= ~DCACHE_NFSFS_RENAMED;
24862306a36Sopenharmony_ci		dentry->d_fsdata = NULL;
24962306a36Sopenharmony_ci		spin_unlock(&dentry->d_lock);
25062306a36Sopenharmony_ci		nfs_free_unlinkdata(data);
25162306a36Sopenharmony_ci		return;
25262306a36Sopenharmony_ci	}
25362306a36Sopenharmony_ci	spin_unlock(&dentry->d_lock);
25462306a36Sopenharmony_ci}
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_ci/**
25762306a36Sopenharmony_ci * nfs_async_rename_done - Sillyrename post-processing
25862306a36Sopenharmony_ci * @task: rpc_task of the sillyrename
25962306a36Sopenharmony_ci * @calldata: nfs_renamedata for the sillyrename
26062306a36Sopenharmony_ci *
26162306a36Sopenharmony_ci * Do the directory attribute updates and the d_move
26262306a36Sopenharmony_ci */
26362306a36Sopenharmony_cistatic void nfs_async_rename_done(struct rpc_task *task, void *calldata)
26462306a36Sopenharmony_ci{
26562306a36Sopenharmony_ci	struct nfs_renamedata *data = calldata;
26662306a36Sopenharmony_ci	struct inode *old_dir = data->old_dir;
26762306a36Sopenharmony_ci	struct inode *new_dir = data->new_dir;
26862306a36Sopenharmony_ci	struct dentry *old_dentry = data->old_dentry;
26962306a36Sopenharmony_ci
27062306a36Sopenharmony_ci	trace_nfs_sillyrename_rename(old_dir, old_dentry,
27162306a36Sopenharmony_ci			new_dir, data->new_dentry, task->tk_status);
27262306a36Sopenharmony_ci	if (!NFS_PROTO(old_dir)->rename_done(task, old_dir, new_dir)) {
27362306a36Sopenharmony_ci		rpc_restart_call_prepare(task);
27462306a36Sopenharmony_ci		return;
27562306a36Sopenharmony_ci	}
27662306a36Sopenharmony_ci
27762306a36Sopenharmony_ci	if (data->complete)
27862306a36Sopenharmony_ci		data->complete(task, data);
27962306a36Sopenharmony_ci}
28062306a36Sopenharmony_ci
28162306a36Sopenharmony_ci/**
28262306a36Sopenharmony_ci * nfs_async_rename_release - Release the sillyrename data.
28362306a36Sopenharmony_ci * @calldata: the struct nfs_renamedata to be released
28462306a36Sopenharmony_ci */
28562306a36Sopenharmony_cistatic void nfs_async_rename_release(void *calldata)
28662306a36Sopenharmony_ci{
28762306a36Sopenharmony_ci	struct nfs_renamedata	*data = calldata;
28862306a36Sopenharmony_ci	struct super_block *sb = data->old_dir->i_sb;
28962306a36Sopenharmony_ci
29062306a36Sopenharmony_ci	if (d_really_is_positive(data->old_dentry))
29162306a36Sopenharmony_ci		nfs_mark_for_revalidate(d_inode(data->old_dentry));
29262306a36Sopenharmony_ci
29362306a36Sopenharmony_ci	/* The result of the rename is unknown. Play it safe by
29462306a36Sopenharmony_ci	 * forcing a new lookup */
29562306a36Sopenharmony_ci	if (data->cancelled) {
29662306a36Sopenharmony_ci		spin_lock(&data->old_dir->i_lock);
29762306a36Sopenharmony_ci		nfs_force_lookup_revalidate(data->old_dir);
29862306a36Sopenharmony_ci		spin_unlock(&data->old_dir->i_lock);
29962306a36Sopenharmony_ci		if (data->new_dir != data->old_dir) {
30062306a36Sopenharmony_ci			spin_lock(&data->new_dir->i_lock);
30162306a36Sopenharmony_ci			nfs_force_lookup_revalidate(data->new_dir);
30262306a36Sopenharmony_ci			spin_unlock(&data->new_dir->i_lock);
30362306a36Sopenharmony_ci		}
30462306a36Sopenharmony_ci	}
30562306a36Sopenharmony_ci
30662306a36Sopenharmony_ci	dput(data->old_dentry);
30762306a36Sopenharmony_ci	dput(data->new_dentry);
30862306a36Sopenharmony_ci	iput(data->old_dir);
30962306a36Sopenharmony_ci	iput(data->new_dir);
31062306a36Sopenharmony_ci	nfs_sb_deactive(sb);
31162306a36Sopenharmony_ci	put_cred(data->cred);
31262306a36Sopenharmony_ci	kfree(data);
31362306a36Sopenharmony_ci}
31462306a36Sopenharmony_ci
31562306a36Sopenharmony_cistatic void nfs_rename_prepare(struct rpc_task *task, void *calldata)
31662306a36Sopenharmony_ci{
31762306a36Sopenharmony_ci	struct nfs_renamedata *data = calldata;
31862306a36Sopenharmony_ci	NFS_PROTO(data->old_dir)->rename_rpc_prepare(task, data);
31962306a36Sopenharmony_ci}
32062306a36Sopenharmony_ci
32162306a36Sopenharmony_cistatic const struct rpc_call_ops nfs_rename_ops = {
32262306a36Sopenharmony_ci	.rpc_call_done = nfs_async_rename_done,
32362306a36Sopenharmony_ci	.rpc_release = nfs_async_rename_release,
32462306a36Sopenharmony_ci	.rpc_call_prepare = nfs_rename_prepare,
32562306a36Sopenharmony_ci};
32662306a36Sopenharmony_ci
32762306a36Sopenharmony_ci/**
32862306a36Sopenharmony_ci * nfs_async_rename - perform an asynchronous rename operation
32962306a36Sopenharmony_ci * @old_dir: directory that currently holds the dentry to be renamed
33062306a36Sopenharmony_ci * @new_dir: target directory for the rename
33162306a36Sopenharmony_ci * @old_dentry: original dentry to be renamed
33262306a36Sopenharmony_ci * @new_dentry: dentry to which the old_dentry should be renamed
33362306a36Sopenharmony_ci * @complete: Function to run on successful completion
33462306a36Sopenharmony_ci *
33562306a36Sopenharmony_ci * It's expected that valid references to the dentries and inodes are held
33662306a36Sopenharmony_ci */
33762306a36Sopenharmony_cistruct rpc_task *
33862306a36Sopenharmony_cinfs_async_rename(struct inode *old_dir, struct inode *new_dir,
33962306a36Sopenharmony_ci		 struct dentry *old_dentry, struct dentry *new_dentry,
34062306a36Sopenharmony_ci		 void (*complete)(struct rpc_task *, struct nfs_renamedata *))
34162306a36Sopenharmony_ci{
34262306a36Sopenharmony_ci	struct nfs_renamedata *data;
34362306a36Sopenharmony_ci	struct rpc_message msg = { };
34462306a36Sopenharmony_ci	struct rpc_task_setup task_setup_data = {
34562306a36Sopenharmony_ci		.rpc_message = &msg,
34662306a36Sopenharmony_ci		.callback_ops = &nfs_rename_ops,
34762306a36Sopenharmony_ci		.workqueue = nfsiod_workqueue,
34862306a36Sopenharmony_ci		.rpc_client = NFS_CLIENT(old_dir),
34962306a36Sopenharmony_ci		.flags = RPC_TASK_ASYNC | RPC_TASK_CRED_NOREF,
35062306a36Sopenharmony_ci	};
35162306a36Sopenharmony_ci
35262306a36Sopenharmony_ci	if (nfs_server_capable(old_dir, NFS_CAP_MOVEABLE) &&
35362306a36Sopenharmony_ci	    nfs_server_capable(new_dir, NFS_CAP_MOVEABLE))
35462306a36Sopenharmony_ci		task_setup_data.flags |= RPC_TASK_MOVEABLE;
35562306a36Sopenharmony_ci
35662306a36Sopenharmony_ci	data = kzalloc(sizeof(*data), GFP_KERNEL);
35762306a36Sopenharmony_ci	if (data == NULL)
35862306a36Sopenharmony_ci		return ERR_PTR(-ENOMEM);
35962306a36Sopenharmony_ci	task_setup_data.task = &data->task;
36062306a36Sopenharmony_ci	task_setup_data.callback_data = data;
36162306a36Sopenharmony_ci
36262306a36Sopenharmony_ci	data->cred = get_current_cred();
36362306a36Sopenharmony_ci
36462306a36Sopenharmony_ci	msg.rpc_argp = &data->args;
36562306a36Sopenharmony_ci	msg.rpc_resp = &data->res;
36662306a36Sopenharmony_ci	msg.rpc_cred = data->cred;
36762306a36Sopenharmony_ci
36862306a36Sopenharmony_ci	/* set up nfs_renamedata */
36962306a36Sopenharmony_ci	data->old_dir = old_dir;
37062306a36Sopenharmony_ci	ihold(old_dir);
37162306a36Sopenharmony_ci	data->new_dir = new_dir;
37262306a36Sopenharmony_ci	ihold(new_dir);
37362306a36Sopenharmony_ci	data->old_dentry = dget(old_dentry);
37462306a36Sopenharmony_ci	data->new_dentry = dget(new_dentry);
37562306a36Sopenharmony_ci	nfs_fattr_init(&data->old_fattr);
37662306a36Sopenharmony_ci	nfs_fattr_init(&data->new_fattr);
37762306a36Sopenharmony_ci	data->complete = complete;
37862306a36Sopenharmony_ci
37962306a36Sopenharmony_ci	/* set up nfs_renameargs */
38062306a36Sopenharmony_ci	data->args.old_dir = NFS_FH(old_dir);
38162306a36Sopenharmony_ci	data->args.old_name = &old_dentry->d_name;
38262306a36Sopenharmony_ci	data->args.new_dir = NFS_FH(new_dir);
38362306a36Sopenharmony_ci	data->args.new_name = &new_dentry->d_name;
38462306a36Sopenharmony_ci
38562306a36Sopenharmony_ci	/* set up nfs_renameres */
38662306a36Sopenharmony_ci	data->res.old_fattr = &data->old_fattr;
38762306a36Sopenharmony_ci	data->res.new_fattr = &data->new_fattr;
38862306a36Sopenharmony_ci
38962306a36Sopenharmony_ci	nfs_sb_active(old_dir->i_sb);
39062306a36Sopenharmony_ci
39162306a36Sopenharmony_ci	NFS_PROTO(data->old_dir)->rename_setup(&msg, old_dentry, new_dentry);
39262306a36Sopenharmony_ci
39362306a36Sopenharmony_ci	return rpc_run_task(&task_setup_data);
39462306a36Sopenharmony_ci}
39562306a36Sopenharmony_ci
39662306a36Sopenharmony_ci/*
39762306a36Sopenharmony_ci * Perform tasks needed when a sillyrename is done such as cancelling the
39862306a36Sopenharmony_ci * queued async unlink if it failed.
39962306a36Sopenharmony_ci */
40062306a36Sopenharmony_cistatic void
40162306a36Sopenharmony_cinfs_complete_sillyrename(struct rpc_task *task, struct nfs_renamedata *data)
40262306a36Sopenharmony_ci{
40362306a36Sopenharmony_ci	struct dentry *dentry = data->old_dentry;
40462306a36Sopenharmony_ci
40562306a36Sopenharmony_ci	if (task->tk_status != 0) {
40662306a36Sopenharmony_ci		nfs_cancel_async_unlink(dentry);
40762306a36Sopenharmony_ci		return;
40862306a36Sopenharmony_ci	}
40962306a36Sopenharmony_ci}
41062306a36Sopenharmony_ci
41162306a36Sopenharmony_ci#define SILLYNAME_PREFIX ".nfs"
41262306a36Sopenharmony_ci#define SILLYNAME_PREFIX_LEN ((unsigned)sizeof(SILLYNAME_PREFIX) - 1)
41362306a36Sopenharmony_ci#define SILLYNAME_FILEID_LEN ((unsigned)sizeof(u64) << 1)
41462306a36Sopenharmony_ci#define SILLYNAME_COUNTER_LEN ((unsigned)sizeof(unsigned int) << 1)
41562306a36Sopenharmony_ci#define SILLYNAME_LEN (SILLYNAME_PREFIX_LEN + \
41662306a36Sopenharmony_ci		SILLYNAME_FILEID_LEN + \
41762306a36Sopenharmony_ci		SILLYNAME_COUNTER_LEN)
41862306a36Sopenharmony_ci
41962306a36Sopenharmony_ci/**
42062306a36Sopenharmony_ci * nfs_sillyrename - Perform a silly-rename of a dentry
42162306a36Sopenharmony_ci * @dir: inode of directory that contains dentry
42262306a36Sopenharmony_ci * @dentry: dentry to be sillyrenamed
42362306a36Sopenharmony_ci *
42462306a36Sopenharmony_ci * NFSv2/3 is stateless and the server doesn't know when the client is
42562306a36Sopenharmony_ci * holding a file open. To prevent application problems when a file is
42662306a36Sopenharmony_ci * unlinked while it's still open, the client performs a "silly-rename".
42762306a36Sopenharmony_ci * That is, it renames the file to a hidden file in the same directory,
42862306a36Sopenharmony_ci * and only performs the unlink once the last reference to it is put.
42962306a36Sopenharmony_ci *
43062306a36Sopenharmony_ci * The final cleanup is done during dentry_iput.
43162306a36Sopenharmony_ci *
43262306a36Sopenharmony_ci * (Note: NFSv4 is stateful, and has opens, so in theory an NFSv4 server
43362306a36Sopenharmony_ci * could take responsibility for keeping open files referenced.  The server
43462306a36Sopenharmony_ci * would also need to ensure that opened-but-deleted files were kept over
43562306a36Sopenharmony_ci * reboots.  However, we may not assume a server does so.  (RFC 5661
43662306a36Sopenharmony_ci * does provide an OPEN4_RESULT_PRESERVE_UNLINKED flag that a server can
43762306a36Sopenharmony_ci * use to advertise that it does this; some day we may take advantage of
43862306a36Sopenharmony_ci * it.))
43962306a36Sopenharmony_ci */
44062306a36Sopenharmony_ciint
44162306a36Sopenharmony_cinfs_sillyrename(struct inode *dir, struct dentry *dentry)
44262306a36Sopenharmony_ci{
44362306a36Sopenharmony_ci	static unsigned int sillycounter;
44462306a36Sopenharmony_ci	unsigned char silly[SILLYNAME_LEN + 1];
44562306a36Sopenharmony_ci	unsigned long long fileid;
44662306a36Sopenharmony_ci	struct dentry *sdentry;
44762306a36Sopenharmony_ci	struct inode *inode = d_inode(dentry);
44862306a36Sopenharmony_ci	struct rpc_task *task;
44962306a36Sopenharmony_ci	int            error = -EBUSY;
45062306a36Sopenharmony_ci
45162306a36Sopenharmony_ci	dfprintk(VFS, "NFS: silly-rename(%pd2, ct=%d)\n",
45262306a36Sopenharmony_ci		dentry, d_count(dentry));
45362306a36Sopenharmony_ci	nfs_inc_stats(dir, NFSIOS_SILLYRENAME);
45462306a36Sopenharmony_ci
45562306a36Sopenharmony_ci	/*
45662306a36Sopenharmony_ci	 * We don't allow a dentry to be silly-renamed twice.
45762306a36Sopenharmony_ci	 */
45862306a36Sopenharmony_ci	if (dentry->d_flags & DCACHE_NFSFS_RENAMED)
45962306a36Sopenharmony_ci		goto out;
46062306a36Sopenharmony_ci
46162306a36Sopenharmony_ci	fileid = NFS_FILEID(d_inode(dentry));
46262306a36Sopenharmony_ci
46362306a36Sopenharmony_ci	sdentry = NULL;
46462306a36Sopenharmony_ci	do {
46562306a36Sopenharmony_ci		int slen;
46662306a36Sopenharmony_ci		dput(sdentry);
46762306a36Sopenharmony_ci		sillycounter++;
46862306a36Sopenharmony_ci		slen = scnprintf(silly, sizeof(silly),
46962306a36Sopenharmony_ci				SILLYNAME_PREFIX "%0*llx%0*x",
47062306a36Sopenharmony_ci				SILLYNAME_FILEID_LEN, fileid,
47162306a36Sopenharmony_ci				SILLYNAME_COUNTER_LEN, sillycounter);
47262306a36Sopenharmony_ci
47362306a36Sopenharmony_ci		dfprintk(VFS, "NFS: trying to rename %pd to %s\n",
47462306a36Sopenharmony_ci				dentry, silly);
47562306a36Sopenharmony_ci
47662306a36Sopenharmony_ci		sdentry = lookup_one_len(silly, dentry->d_parent, slen);
47762306a36Sopenharmony_ci		/*
47862306a36Sopenharmony_ci		 * N.B. Better to return EBUSY here ... it could be
47962306a36Sopenharmony_ci		 * dangerous to delete the file while it's in use.
48062306a36Sopenharmony_ci		 */
48162306a36Sopenharmony_ci		if (IS_ERR(sdentry))
48262306a36Sopenharmony_ci			goto out;
48362306a36Sopenharmony_ci	} while (d_inode(sdentry) != NULL); /* need negative lookup */
48462306a36Sopenharmony_ci
48562306a36Sopenharmony_ci	ihold(inode);
48662306a36Sopenharmony_ci
48762306a36Sopenharmony_ci	/* queue unlink first. Can't do this from rpc_release as it
48862306a36Sopenharmony_ci	 * has to allocate memory
48962306a36Sopenharmony_ci	 */
49062306a36Sopenharmony_ci	error = nfs_async_unlink(dentry, &sdentry->d_name);
49162306a36Sopenharmony_ci	if (error)
49262306a36Sopenharmony_ci		goto out_dput;
49362306a36Sopenharmony_ci
49462306a36Sopenharmony_ci	/* run the rename task, undo unlink if it fails */
49562306a36Sopenharmony_ci	task = nfs_async_rename(dir, dir, dentry, sdentry,
49662306a36Sopenharmony_ci					nfs_complete_sillyrename);
49762306a36Sopenharmony_ci	if (IS_ERR(task)) {
49862306a36Sopenharmony_ci		error = -EBUSY;
49962306a36Sopenharmony_ci		nfs_cancel_async_unlink(dentry);
50062306a36Sopenharmony_ci		goto out_dput;
50162306a36Sopenharmony_ci	}
50262306a36Sopenharmony_ci
50362306a36Sopenharmony_ci	/* wait for the RPC task to complete, unless a SIGKILL intervenes */
50462306a36Sopenharmony_ci	error = rpc_wait_for_completion_task(task);
50562306a36Sopenharmony_ci	if (error == 0)
50662306a36Sopenharmony_ci		error = task->tk_status;
50762306a36Sopenharmony_ci	switch (error) {
50862306a36Sopenharmony_ci	case 0:
50962306a36Sopenharmony_ci		/* The rename succeeded */
51062306a36Sopenharmony_ci		nfs_set_verifier(dentry, nfs_save_change_attribute(dir));
51162306a36Sopenharmony_ci		spin_lock(&inode->i_lock);
51262306a36Sopenharmony_ci		NFS_I(inode)->attr_gencount = nfs_inc_attr_generation_counter();
51362306a36Sopenharmony_ci		nfs_set_cache_invalid(inode, NFS_INO_INVALID_CHANGE |
51462306a36Sopenharmony_ci						     NFS_INO_INVALID_CTIME |
51562306a36Sopenharmony_ci						     NFS_INO_REVAL_FORCED);
51662306a36Sopenharmony_ci		spin_unlock(&inode->i_lock);
51762306a36Sopenharmony_ci		d_move(dentry, sdentry);
51862306a36Sopenharmony_ci		break;
51962306a36Sopenharmony_ci	case -ERESTARTSYS:
52062306a36Sopenharmony_ci		/* The result of the rename is unknown. Play it safe by
52162306a36Sopenharmony_ci		 * forcing a new lookup */
52262306a36Sopenharmony_ci		d_drop(dentry);
52362306a36Sopenharmony_ci		d_drop(sdentry);
52462306a36Sopenharmony_ci	}
52562306a36Sopenharmony_ci	rpc_put_task(task);
52662306a36Sopenharmony_ciout_dput:
52762306a36Sopenharmony_ci	iput(inode);
52862306a36Sopenharmony_ci	dput(sdentry);
52962306a36Sopenharmony_ciout:
53062306a36Sopenharmony_ci	return error;
53162306a36Sopenharmony_ci}
532