1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0 2274dcf55SMike Marshall /* 3274dcf55SMike Marshall * (C) 2001 Clemson University and The University of Chicago 4274dcf55SMike Marshall * 5274dcf55SMike Marshall * See COPYING in top-level directory. 6274dcf55SMike Marshall */ 7274dcf55SMike Marshall 8274dcf55SMike Marshall /* 9274dcf55SMike Marshall * Linux VFS namei operations. 10274dcf55SMike Marshall */ 11274dcf55SMike Marshall 12274dcf55SMike Marshall #include "protocol.h" 13575e9461SMike Marshall #include "orangefs-kernel.h" 14274dcf55SMike Marshall 15274dcf55SMike Marshall /* 16274dcf55SMike Marshall * Get a newly allocated inode to go with a negative dentry. 17274dcf55SMike Marshall */ 188bb8aefdSYi Liu static int orangefs_create(struct inode *dir, 19274dcf55SMike Marshall struct dentry *dentry, 20274dcf55SMike Marshall umode_t mode, 21274dcf55SMike Marshall bool exclusive) 22274dcf55SMike Marshall { 238bb8aefdSYi Liu struct orangefs_inode_s *parent = ORANGEFS_I(dir); 248bb8aefdSYi Liu struct orangefs_kernel_op_s *new_op; 25*db0267e7SMartin Brandenburg struct orangefs_object_kref ref; 26274dcf55SMike Marshall struct inode *inode; 27a55f2d86SMartin Brandenburg struct iattr iattr; 28274dcf55SMike Marshall int ret; 29274dcf55SMike Marshall 30f66debf1SAl Viro gossip_debug(GOSSIP_NAME_DEBUG, "%s: %pd\n", 315253487eSMike Marshall __func__, 32f66debf1SAl Viro dentry); 33274dcf55SMike Marshall 348bb8aefdSYi Liu new_op = op_alloc(ORANGEFS_VFS_OP_CREATE); 35274dcf55SMike Marshall if (!new_op) 36274dcf55SMike Marshall return -ENOMEM; 37274dcf55SMike Marshall 38274dcf55SMike Marshall new_op->upcall.req.create.parent_refn = parent->refn; 39274dcf55SMike Marshall 40274dcf55SMike Marshall fill_default_sys_attrs(new_op->upcall.req.create.attributes, 418bb8aefdSYi Liu ORANGEFS_TYPE_METAFILE, mode); 42274dcf55SMike Marshall 43274dcf55SMike Marshall strncpy(new_op->upcall.req.create.d_name, 4447b4948fSMartin Brandenburg dentry->d_name.name, ORANGEFS_NAME_MAX); 45274dcf55SMike Marshall 46274dcf55SMike Marshall ret = service_operation(new_op, __func__, get_interruptible_flag(dir)); 47274dcf55SMike Marshall 48274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 49f66debf1SAl Viro "%s: %pd: handle:%pU: fsid:%d: new_op:%p: ret:%d:\n", 505253487eSMike Marshall __func__, 51f66debf1SAl Viro dentry, 52274dcf55SMike Marshall &new_op->downcall.resp.create.refn.khandle, 535253487eSMike Marshall new_op->downcall.resp.create.refn.fs_id, 545253487eSMike Marshall new_op, 555253487eSMike Marshall ret); 56274dcf55SMike Marshall 575253487eSMike Marshall if (ret < 0) 58274dcf55SMike Marshall goto out; 59274dcf55SMike Marshall 60*db0267e7SMartin Brandenburg ref = new_op->downcall.resp.create.refn; 61*db0267e7SMartin Brandenburg op_release(new_op); 62*db0267e7SMartin Brandenburg 63*db0267e7SMartin Brandenburg inode = orangefs_new_inode(dir->i_sb, dir, S_IFREG | mode, 0, &ref); 64274dcf55SMike Marshall if (IS_ERR(inode)) { 65f66debf1SAl Viro gossip_err("%s: Failed to allocate inode for file :%pd:\n", 665253487eSMike Marshall __func__, 67f66debf1SAl Viro dentry); 68274dcf55SMike Marshall ret = PTR_ERR(inode); 69274dcf55SMike Marshall goto out; 70274dcf55SMike Marshall } 71274dcf55SMike Marshall 72274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 73f66debf1SAl Viro "%s: Assigned inode :%pU: for file :%pd:\n", 745253487eSMike Marshall __func__, 755253487eSMike Marshall get_khandle_from_ino(inode), 76f66debf1SAl Viro dentry); 77274dcf55SMike Marshall 78274dcf55SMike Marshall d_instantiate(dentry, inode); 79274dcf55SMike Marshall unlock_new_inode(inode); 80804b1737SMiklos Szeredi orangefs_set_timeout(dentry); 818bbb20a8SMartin Brandenburg ORANGEFS_I(inode)->getattr_time = jiffies - 1; 8268a24a6cSMartin Brandenburg ORANGEFS_I(inode)->getattr_mask = STATX_BASIC_STATS; 83274dcf55SMike Marshall 84274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 85f66debf1SAl Viro "%s: dentry instantiated for %pd\n", 865253487eSMike Marshall __func__, 87f66debf1SAl Viro dentry); 88274dcf55SMike Marshall 89c2050a45SDeepa Dinamani dir->i_mtime = dir->i_ctime = current_time(dir); 90a55f2d86SMartin Brandenburg memset(&iattr, 0, sizeof iattr); 91a55f2d86SMartin Brandenburg iattr.ia_valid |= ATTR_MTIME; 92a55f2d86SMartin Brandenburg orangefs_inode_setattr(dir, &iattr); 93274dcf55SMike Marshall mark_inode_dirty_sync(dir); 94274dcf55SMike Marshall ret = 0; 95274dcf55SMike Marshall out: 965253487eSMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 97f66debf1SAl Viro "%s: %pd: returning %d\n", 985253487eSMike Marshall __func__, 99f66debf1SAl Viro dentry, 1005253487eSMike Marshall ret); 101274dcf55SMike Marshall return ret; 102274dcf55SMike Marshall } 103274dcf55SMike Marshall 104274dcf55SMike Marshall /* 105274dcf55SMike Marshall * Attempt to resolve an object name (dentry->d_name), parent handle, and 106274dcf55SMike Marshall * fsid into a handle for the object. 107274dcf55SMike Marshall */ 1088bb8aefdSYi Liu static struct dentry *orangefs_lookup(struct inode *dir, struct dentry *dentry, 109274dcf55SMike Marshall unsigned int flags) 110274dcf55SMike Marshall { 1118bb8aefdSYi Liu struct orangefs_inode_s *parent = ORANGEFS_I(dir); 1128bb8aefdSYi Liu struct orangefs_kernel_op_s *new_op; 113274dcf55SMike Marshall struct inode *inode; 114274dcf55SMike Marshall struct dentry *res; 115274dcf55SMike Marshall int ret = -EINVAL; 116274dcf55SMike Marshall 117274dcf55SMike Marshall /* 118274dcf55SMike Marshall * in theory we could skip a lookup here (if the intent is to 119274dcf55SMike Marshall * create) in order to avoid a potentially failed lookup, but 120274dcf55SMike Marshall * leaving it in can skip a valid lookup and try to create a file 121274dcf55SMike Marshall * that already exists (e.g. the vfs already handles checking for 122274dcf55SMike Marshall * -EEXIST on O_EXCL opens, which is broken if we skip this lookup 123274dcf55SMike Marshall * in the create path) 124274dcf55SMike Marshall */ 125f66debf1SAl Viro gossip_debug(GOSSIP_NAME_DEBUG, "%s called on %pd\n", 126f66debf1SAl Viro __func__, dentry); 127274dcf55SMike Marshall 12847b4948fSMartin Brandenburg if (dentry->d_name.len > (ORANGEFS_NAME_MAX - 1)) 129274dcf55SMike Marshall return ERR_PTR(-ENAMETOOLONG); 130274dcf55SMike Marshall 1318bb8aefdSYi Liu new_op = op_alloc(ORANGEFS_VFS_OP_LOOKUP); 132274dcf55SMike Marshall if (!new_op) 133274dcf55SMike Marshall return ERR_PTR(-ENOMEM); 134274dcf55SMike Marshall 1357cec28e9SMike Marshall new_op->upcall.req.lookup.sym_follow = ORANGEFS_LOOKUP_LINK_NO_FOLLOW; 136274dcf55SMike Marshall 137274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, "%s:%s:%d using parent %pU\n", 138274dcf55SMike Marshall __FILE__, 139274dcf55SMike Marshall __func__, 140274dcf55SMike Marshall __LINE__, 141274dcf55SMike Marshall &parent->refn.khandle); 142274dcf55SMike Marshall new_op->upcall.req.lookup.parent_refn = parent->refn; 143274dcf55SMike Marshall 144274dcf55SMike Marshall strncpy(new_op->upcall.req.lookup.d_name, dentry->d_name.name, 14547b4948fSMartin Brandenburg ORANGEFS_NAME_MAX); 146274dcf55SMike Marshall 147274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 1486ceaf781SMartin Brandenburg "%s: doing lookup on %s under %pU,%d\n", 149274dcf55SMike Marshall __func__, 150274dcf55SMike Marshall new_op->upcall.req.lookup.d_name, 151274dcf55SMike Marshall &new_op->upcall.req.lookup.parent_refn.khandle, 1526ceaf781SMartin Brandenburg new_op->upcall.req.lookup.parent_refn.fs_id); 153274dcf55SMike Marshall 154274dcf55SMike Marshall ret = service_operation(new_op, __func__, get_interruptible_flag(dir)); 155274dcf55SMike Marshall 156274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 157274dcf55SMike Marshall "Lookup Got %pU, fsid %d (ret=%d)\n", 158274dcf55SMike Marshall &new_op->downcall.resp.lookup.refn.khandle, 159274dcf55SMike Marshall new_op->downcall.resp.lookup.refn.fs_id, 160274dcf55SMike Marshall ret); 161274dcf55SMike Marshall 162274dcf55SMike Marshall if (ret < 0) { 163274dcf55SMike Marshall if (ret == -ENOENT) { 164274dcf55SMike Marshall /* 165274dcf55SMike Marshall * if no inode was found, add a negative dentry to 166274dcf55SMike Marshall * dcache anyway; if we don't, we don't hold expected 167274dcf55SMike Marshall * lookup semantics and we most noticeably break 168274dcf55SMike Marshall * during directory renames. 169274dcf55SMike Marshall * 170274dcf55SMike Marshall * however, if the operation failed or exited, do not 171274dcf55SMike Marshall * add the dentry (e.g. in the case that a touch is 172274dcf55SMike Marshall * issued on a file that already exists that was 173274dcf55SMike Marshall * interrupted during this lookup -- no need to add 174274dcf55SMike Marshall * another negative dentry for an existing file) 175274dcf55SMike Marshall */ 176274dcf55SMike Marshall 177274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 1788bb8aefdSYi Liu "orangefs_lookup: Adding *negative* dentry " 179f66debf1SAl Viro "%p for %pd\n", 180274dcf55SMike Marshall dentry, 181f66debf1SAl Viro dentry); 182274dcf55SMike Marshall 183274dcf55SMike Marshall d_add(dentry, NULL); 184274dcf55SMike Marshall res = NULL; 185274dcf55SMike Marshall goto out; 186274dcf55SMike Marshall } 187274dcf55SMike Marshall 188274dcf55SMike Marshall /* must be a non-recoverable error */ 189274dcf55SMike Marshall res = ERR_PTR(ret); 190274dcf55SMike Marshall goto out; 191274dcf55SMike Marshall } 192274dcf55SMike Marshall 193804b1737SMiklos Szeredi orangefs_set_timeout(dentry); 19431b7c1abSMartin Brandenburg 1958bb8aefdSYi Liu inode = orangefs_iget(dir->i_sb, &new_op->downcall.resp.lookup.refn); 196274dcf55SMike Marshall if (IS_ERR(inode)) { 197274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 198274dcf55SMike Marshall "error %ld from iget\n", PTR_ERR(inode)); 199274dcf55SMike Marshall res = ERR_CAST(inode); 200274dcf55SMike Marshall goto out; 201274dcf55SMike Marshall } 202274dcf55SMike Marshall 203274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 204274dcf55SMike Marshall "%s:%s:%d " 205274dcf55SMike Marshall "Found good inode [%lu] with count [%d]\n", 206274dcf55SMike Marshall __FILE__, 207274dcf55SMike Marshall __func__, 208274dcf55SMike Marshall __LINE__, 209274dcf55SMike Marshall inode->i_ino, 210274dcf55SMike Marshall (int)atomic_read(&inode->i_count)); 211274dcf55SMike Marshall 212274dcf55SMike Marshall /* update dentry/inode pair into dcache */ 213274dcf55SMike Marshall res = d_splice_alias(inode, dentry); 214274dcf55SMike Marshall 215274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 216274dcf55SMike Marshall "Lookup success (inode ct = %d)\n", 217274dcf55SMike Marshall (int)atomic_read(&inode->i_count)); 218274dcf55SMike Marshall out: 219274dcf55SMike Marshall op_release(new_op); 220274dcf55SMike Marshall return res; 221274dcf55SMike Marshall } 222274dcf55SMike Marshall 223274dcf55SMike Marshall /* return 0 on success; non-zero otherwise */ 2248bb8aefdSYi Liu static int orangefs_unlink(struct inode *dir, struct dentry *dentry) 225274dcf55SMike Marshall { 226274dcf55SMike Marshall struct inode *inode = dentry->d_inode; 2278bb8aefdSYi Liu struct orangefs_inode_s *parent = ORANGEFS_I(dir); 2288bb8aefdSYi Liu struct orangefs_kernel_op_s *new_op; 229a55f2d86SMartin Brandenburg struct iattr iattr; 230274dcf55SMike Marshall int ret; 231274dcf55SMike Marshall 232274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 233f66debf1SAl Viro "%s: called on %pd\n" 234274dcf55SMike Marshall " (inode %pU): Parent is %pU | fs_id %d\n", 235274dcf55SMike Marshall __func__, 236f66debf1SAl Viro dentry, 237274dcf55SMike Marshall get_khandle_from_ino(inode), 238274dcf55SMike Marshall &parent->refn.khandle, 239274dcf55SMike Marshall parent->refn.fs_id); 240274dcf55SMike Marshall 2418bb8aefdSYi Liu new_op = op_alloc(ORANGEFS_VFS_OP_REMOVE); 242274dcf55SMike Marshall if (!new_op) 243274dcf55SMike Marshall return -ENOMEM; 244274dcf55SMike Marshall 245274dcf55SMike Marshall new_op->upcall.req.remove.parent_refn = parent->refn; 246274dcf55SMike Marshall strncpy(new_op->upcall.req.remove.d_name, dentry->d_name.name, 24747b4948fSMartin Brandenburg ORANGEFS_NAME_MAX); 248274dcf55SMike Marshall 2498bb8aefdSYi Liu ret = service_operation(new_op, "orangefs_unlink", 250274dcf55SMike Marshall get_interruptible_flag(inode)); 251274dcf55SMike Marshall 2525253487eSMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 2535253487eSMike Marshall "%s: service_operation returned:%d:\n", 2545253487eSMike Marshall __func__, 2555253487eSMike Marshall ret); 2565253487eSMike Marshall 257274dcf55SMike Marshall op_release(new_op); 258274dcf55SMike Marshall 259274dcf55SMike Marshall if (!ret) { 260274dcf55SMike Marshall drop_nlink(inode); 261274dcf55SMike Marshall 262c2050a45SDeepa Dinamani dir->i_mtime = dir->i_ctime = current_time(dir); 263a55f2d86SMartin Brandenburg memset(&iattr, 0, sizeof iattr); 264a55f2d86SMartin Brandenburg iattr.ia_valid |= ATTR_MTIME; 265a55f2d86SMartin Brandenburg orangefs_inode_setattr(dir, &iattr); 266274dcf55SMike Marshall mark_inode_dirty_sync(dir); 267274dcf55SMike Marshall } 268274dcf55SMike Marshall return ret; 269274dcf55SMike Marshall } 270274dcf55SMike Marshall 2718bb8aefdSYi Liu static int orangefs_symlink(struct inode *dir, 272274dcf55SMike Marshall struct dentry *dentry, 273274dcf55SMike Marshall const char *symname) 274274dcf55SMike Marshall { 2758bb8aefdSYi Liu struct orangefs_inode_s *parent = ORANGEFS_I(dir); 2768bb8aefdSYi Liu struct orangefs_kernel_op_s *new_op; 277*db0267e7SMartin Brandenburg struct orangefs_object_kref ref; 278274dcf55SMike Marshall struct inode *inode; 279a55f2d86SMartin Brandenburg struct iattr iattr; 280274dcf55SMike Marshall int mode = 755; 281274dcf55SMike Marshall int ret; 282274dcf55SMike Marshall 283274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, "%s: called\n", __func__); 284274dcf55SMike Marshall 285274dcf55SMike Marshall if (!symname) 286274dcf55SMike Marshall return -EINVAL; 287274dcf55SMike Marshall 288c62da585SMartin Brandenburg if (strlen(symname)+1 > ORANGEFS_NAME_MAX) 289c62da585SMartin Brandenburg return -ENAMETOOLONG; 290c62da585SMartin Brandenburg 2918bb8aefdSYi Liu new_op = op_alloc(ORANGEFS_VFS_OP_SYMLINK); 292274dcf55SMike Marshall if (!new_op) 293274dcf55SMike Marshall return -ENOMEM; 294274dcf55SMike Marshall 295274dcf55SMike Marshall new_op->upcall.req.sym.parent_refn = parent->refn; 296274dcf55SMike Marshall 297274dcf55SMike Marshall fill_default_sys_attrs(new_op->upcall.req.sym.attributes, 2988bb8aefdSYi Liu ORANGEFS_TYPE_SYMLINK, 299274dcf55SMike Marshall mode); 300274dcf55SMike Marshall 301274dcf55SMike Marshall strncpy(new_op->upcall.req.sym.entry_name, 302274dcf55SMike Marshall dentry->d_name.name, 30347b4948fSMartin Brandenburg ORANGEFS_NAME_MAX); 30447b4948fSMartin Brandenburg strncpy(new_op->upcall.req.sym.target, symname, ORANGEFS_NAME_MAX); 305274dcf55SMike Marshall 306274dcf55SMike Marshall ret = service_operation(new_op, __func__, get_interruptible_flag(dir)); 307274dcf55SMike Marshall 308274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 3098bb8aefdSYi Liu "Symlink Got ORANGEFS handle %pU on fsid %d (ret=%d)\n", 310274dcf55SMike Marshall &new_op->downcall.resp.sym.refn.khandle, 311274dcf55SMike Marshall new_op->downcall.resp.sym.refn.fs_id, ret); 312274dcf55SMike Marshall 313274dcf55SMike Marshall if (ret < 0) { 314274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 315274dcf55SMike Marshall "%s: failed with error code %d\n", 316274dcf55SMike Marshall __func__, ret); 317274dcf55SMike Marshall goto out; 318274dcf55SMike Marshall } 319274dcf55SMike Marshall 320*db0267e7SMartin Brandenburg ref = new_op->downcall.resp.sym.refn; 321*db0267e7SMartin Brandenburg op_release(new_op); 322*db0267e7SMartin Brandenburg 323*db0267e7SMartin Brandenburg inode = orangefs_new_inode(dir->i_sb, dir, S_IFLNK | mode, 0, &ref); 324274dcf55SMike Marshall if (IS_ERR(inode)) { 325274dcf55SMike Marshall gossip_err 3268bb8aefdSYi Liu ("*** Failed to allocate orangefs symlink inode\n"); 327274dcf55SMike Marshall ret = PTR_ERR(inode); 328274dcf55SMike Marshall goto out; 329274dcf55SMike Marshall } 330274dcf55SMike Marshall 331274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 332274dcf55SMike Marshall "Assigned symlink inode new number of %pU\n", 333274dcf55SMike Marshall get_khandle_from_ino(inode)); 334274dcf55SMike Marshall 335274dcf55SMike Marshall d_instantiate(dentry, inode); 336274dcf55SMike Marshall unlock_new_inode(inode); 337804b1737SMiklos Szeredi orangefs_set_timeout(dentry); 3388bbb20a8SMartin Brandenburg ORANGEFS_I(inode)->getattr_time = jiffies - 1; 33968a24a6cSMartin Brandenburg ORANGEFS_I(inode)->getattr_mask = STATX_BASIC_STATS; 340274dcf55SMike Marshall 341274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 342f66debf1SAl Viro "Inode (Symlink) %pU -> %pd\n", 343274dcf55SMike Marshall get_khandle_from_ino(inode), 344f66debf1SAl Viro dentry); 345274dcf55SMike Marshall 346c2050a45SDeepa Dinamani dir->i_mtime = dir->i_ctime = current_time(dir); 347a55f2d86SMartin Brandenburg memset(&iattr, 0, sizeof iattr); 348a55f2d86SMartin Brandenburg iattr.ia_valid |= ATTR_MTIME; 349a55f2d86SMartin Brandenburg orangefs_inode_setattr(dir, &iattr); 350274dcf55SMike Marshall mark_inode_dirty_sync(dir); 351274dcf55SMike Marshall ret = 0; 352274dcf55SMike Marshall out: 353274dcf55SMike Marshall return ret; 354274dcf55SMike Marshall } 355274dcf55SMike Marshall 3568bb8aefdSYi Liu static int orangefs_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode) 357274dcf55SMike Marshall { 3588bb8aefdSYi Liu struct orangefs_inode_s *parent = ORANGEFS_I(dir); 3598bb8aefdSYi Liu struct orangefs_kernel_op_s *new_op; 360*db0267e7SMartin Brandenburg struct orangefs_object_kref ref; 361274dcf55SMike Marshall struct inode *inode; 362a55f2d86SMartin Brandenburg struct iattr iattr; 363274dcf55SMike Marshall int ret; 364274dcf55SMike Marshall 3658bb8aefdSYi Liu new_op = op_alloc(ORANGEFS_VFS_OP_MKDIR); 366274dcf55SMike Marshall if (!new_op) 367274dcf55SMike Marshall return -ENOMEM; 368274dcf55SMike Marshall 369274dcf55SMike Marshall new_op->upcall.req.mkdir.parent_refn = parent->refn; 370274dcf55SMike Marshall 371274dcf55SMike Marshall fill_default_sys_attrs(new_op->upcall.req.mkdir.attributes, 3728bb8aefdSYi Liu ORANGEFS_TYPE_DIRECTORY, mode); 373274dcf55SMike Marshall 374274dcf55SMike Marshall strncpy(new_op->upcall.req.mkdir.d_name, 37547b4948fSMartin Brandenburg dentry->d_name.name, ORANGEFS_NAME_MAX); 376274dcf55SMike Marshall 377274dcf55SMike Marshall ret = service_operation(new_op, __func__, get_interruptible_flag(dir)); 378274dcf55SMike Marshall 379274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 3808bb8aefdSYi Liu "Mkdir Got ORANGEFS handle %pU on fsid %d\n", 381274dcf55SMike Marshall &new_op->downcall.resp.mkdir.refn.khandle, 382274dcf55SMike Marshall new_op->downcall.resp.mkdir.refn.fs_id); 383274dcf55SMike Marshall 384274dcf55SMike Marshall if (ret < 0) { 385274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 386274dcf55SMike Marshall "%s: failed with error code %d\n", 387274dcf55SMike Marshall __func__, ret); 388274dcf55SMike Marshall goto out; 389274dcf55SMike Marshall } 390274dcf55SMike Marshall 391*db0267e7SMartin Brandenburg ref = new_op->downcall.resp.mkdir.refn; 392*db0267e7SMartin Brandenburg op_release(new_op); 393*db0267e7SMartin Brandenburg 394*db0267e7SMartin Brandenburg inode = orangefs_new_inode(dir->i_sb, dir, S_IFDIR | mode, 0, &ref); 395274dcf55SMike Marshall if (IS_ERR(inode)) { 3968bb8aefdSYi Liu gossip_err("*** Failed to allocate orangefs dir inode\n"); 397274dcf55SMike Marshall ret = PTR_ERR(inode); 398274dcf55SMike Marshall goto out; 399274dcf55SMike Marshall } 400274dcf55SMike Marshall 401274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 402274dcf55SMike Marshall "Assigned dir inode new number of %pU\n", 403274dcf55SMike Marshall get_khandle_from_ino(inode)); 404274dcf55SMike Marshall 405274dcf55SMike Marshall d_instantiate(dentry, inode); 406274dcf55SMike Marshall unlock_new_inode(inode); 407804b1737SMiklos Szeredi orangefs_set_timeout(dentry); 4088bbb20a8SMartin Brandenburg ORANGEFS_I(inode)->getattr_time = jiffies - 1; 40968a24a6cSMartin Brandenburg ORANGEFS_I(inode)->getattr_mask = STATX_BASIC_STATS; 410274dcf55SMike Marshall 411274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 412f66debf1SAl Viro "Inode (Directory) %pU -> %pd\n", 413274dcf55SMike Marshall get_khandle_from_ino(inode), 414f66debf1SAl Viro dentry); 415274dcf55SMike Marshall 416274dcf55SMike Marshall /* 417274dcf55SMike Marshall * NOTE: we have no good way to keep nlink consistent for directories 418274dcf55SMike Marshall * across clients; keep constant at 1. 419274dcf55SMike Marshall */ 420c2050a45SDeepa Dinamani dir->i_mtime = dir->i_ctime = current_time(dir); 421a55f2d86SMartin Brandenburg memset(&iattr, 0, sizeof iattr); 422a55f2d86SMartin Brandenburg iattr.ia_valid |= ATTR_MTIME; 423a55f2d86SMartin Brandenburg orangefs_inode_setattr(dir, &iattr); 424274dcf55SMike Marshall mark_inode_dirty_sync(dir); 425274dcf55SMike Marshall out: 426274dcf55SMike Marshall return ret; 427274dcf55SMike Marshall } 428274dcf55SMike Marshall 4298bb8aefdSYi Liu static int orangefs_rename(struct inode *old_dir, 430274dcf55SMike Marshall struct dentry *old_dentry, 431274dcf55SMike Marshall struct inode *new_dir, 4321cd66c93SMiklos Szeredi struct dentry *new_dentry, 4331cd66c93SMiklos Szeredi unsigned int flags) 434274dcf55SMike Marshall { 4358bb8aefdSYi Liu struct orangefs_kernel_op_s *new_op; 436274dcf55SMike Marshall int ret; 437274dcf55SMike Marshall 4381cd66c93SMiklos Szeredi if (flags) 4391cd66c93SMiklos Szeredi return -EINVAL; 4401cd66c93SMiklos Szeredi 441274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 44296b0cffbSAl Viro "orangefs_rename: called (%pd2 => %pd2) ct=%d\n", 44396b0cffbSAl Viro old_dentry, new_dentry, d_count(new_dentry)); 444274dcf55SMike Marshall 4458bbb20a8SMartin Brandenburg ORANGEFS_I(new_dentry->d_parent->d_inode)->getattr_time = jiffies - 1; 44671680c18SMartin Brandenburg 4478bb8aefdSYi Liu new_op = op_alloc(ORANGEFS_VFS_OP_RENAME); 448274dcf55SMike Marshall if (!new_op) 449274dcf55SMike Marshall return -EINVAL; 450274dcf55SMike Marshall 4518bb8aefdSYi Liu new_op->upcall.req.rename.old_parent_refn = ORANGEFS_I(old_dir)->refn; 4528bb8aefdSYi Liu new_op->upcall.req.rename.new_parent_refn = ORANGEFS_I(new_dir)->refn; 453274dcf55SMike Marshall 454274dcf55SMike Marshall strncpy(new_op->upcall.req.rename.d_old_name, 455274dcf55SMike Marshall old_dentry->d_name.name, 45647b4948fSMartin Brandenburg ORANGEFS_NAME_MAX); 457274dcf55SMike Marshall strncpy(new_op->upcall.req.rename.d_new_name, 458274dcf55SMike Marshall new_dentry->d_name.name, 45947b4948fSMartin Brandenburg ORANGEFS_NAME_MAX); 460274dcf55SMike Marshall 461274dcf55SMike Marshall ret = service_operation(new_op, 4628bb8aefdSYi Liu "orangefs_rename", 463274dcf55SMike Marshall get_interruptible_flag(old_dentry->d_inode)); 464274dcf55SMike Marshall 465274dcf55SMike Marshall gossip_debug(GOSSIP_NAME_DEBUG, 4668bb8aefdSYi Liu "orangefs_rename: got downcall status %d\n", 467274dcf55SMike Marshall ret); 468274dcf55SMike Marshall 469274dcf55SMike Marshall if (new_dentry->d_inode) 470c2050a45SDeepa Dinamani new_dentry->d_inode->i_ctime = current_time(new_dentry->d_inode); 471274dcf55SMike Marshall 472274dcf55SMike Marshall op_release(new_op); 473274dcf55SMike Marshall return ret; 474274dcf55SMike Marshall } 475274dcf55SMike Marshall 4768bb8aefdSYi Liu /* ORANGEFS implementation of VFS inode operations for directories */ 4776f3fc107SAl Viro const struct inode_operations orangefs_dir_inode_operations = { 4788bb8aefdSYi Liu .lookup = orangefs_lookup, 4798bb8aefdSYi Liu .get_acl = orangefs_get_acl, 4808bb8aefdSYi Liu .set_acl = orangefs_set_acl, 4818bb8aefdSYi Liu .create = orangefs_create, 4828bb8aefdSYi Liu .unlink = orangefs_unlink, 4838bb8aefdSYi Liu .symlink = orangefs_symlink, 4848bb8aefdSYi Liu .mkdir = orangefs_mkdir, 4858bb8aefdSYi Liu .rmdir = orangefs_unlink, 4868bb8aefdSYi Liu .rename = orangefs_rename, 4878bb8aefdSYi Liu .setattr = orangefs_setattr, 4888bb8aefdSYi Liu .getattr = orangefs_getattr, 4898bb8aefdSYi Liu .listxattr = orangefs_listxattr, 490933287daSMartin Brandenburg .permission = orangefs_permission, 491a55f2d86SMartin Brandenburg .update_time = orangefs_update_time, 492274dcf55SMike Marshall }; 493