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