From: Pavel Shilovsky <piastryyy@xxxxxxxxx> Signed-off-by: Pavel Shilovsky <piastryyy@xxxxxxxxx> --- fs/cifs/smb2inode.c | 90 ++++++++++++++++++++++++++++++++++++++++++++++++++- fs/cifs/smb2proto.h | 1 + 2 files changed, 90 insertions(+), 1 deletions(-) diff --git a/fs/cifs/smb2inode.c b/fs/cifs/smb2inode.c index 3d3df06..1c1c749 100644 --- a/fs/cifs/smb2inode.c +++ b/fs/cifs/smb2inode.c @@ -39,7 +39,7 @@ const struct inode_operations smb2_dir_inode_ops = { .create = cifs_create, .lookup = cifs_lookup, .getattr = cifs_getattr, - .unlink = cifs_unlink, + .unlink = smb2_unlink, .link = cifs_hardlink, .mkdir = smb2_mkdir, .rmdir = smb2_rmdir, @@ -439,3 +439,91 @@ rmdir_exit: FreeXid(xid); return rc; } + +int smb2_unlink(struct inode *dir, struct dentry *dentry) +{ + int rc = 0; + int xid; + struct inode *inode = dentry->d_inode; + struct cifsInodeInfo *cifs_inode; + struct super_block *sb = dir->i_sb; + struct cifs_sb_info *cifs_sb = CIFS_SB(sb); + struct tcon_link *tlink; + struct cifs_tcon *tcon; + struct iattr *attrs = NULL; + __le16 *ucs_path = NULL; + __u32 dosattr = 0, origattr = 0; + + cFYI(1, "smb2_unlink, dir=0x%p, dentry=0x%p", dir, dentry); + + tlink = cifs_sb_tlink(cifs_sb); + if (IS_ERR(tlink)) + return PTR_ERR(tlink); + tcon = tlink_tcon(tlink); + + xid = GetXid(); + + /* Unlink can be called from rename so we can not take the + * sb->s_vfs_rename_mutex here */ + ucs_path = build_ucspath_from_dentry(dentry); + if (ucs_path == NULL) { + rc = -ENOMEM; + goto unlink_out; + } + + rc = smb2_open_op_close(xid, tcon, ucs_path, FILE_DELETE_LE, + FILE_OPEN_LE, 0, FILE_DELETE_ON_CLOSE_LE, NULL, + SMB2_OP_DELETE); + if (!rc) { + if (inode) + drop_nlink(inode); + } else if (rc == -ENOENT) { + d_drop(dentry); + } else if (rc == -ETXTBSY) { + /*rc = cifs_rename_pending_delete(full_path, dentry, xid); + if (rc == 0) + drop_nlink(inode);*/ + } else if ((rc == -EACCES) && (dosattr == 0) && inode) { + attrs = kzalloc(sizeof(*attrs), GFP_KERNEL); + if (attrs == NULL) { + rc = -ENOMEM; + goto out_reval; + } + + /* try to reset dos attributes */ + cifs_inode = CIFS_I(inode); + origattr = cifs_inode->cifsAttrs; + if (origattr == 0) + origattr |= ATTR_NORMAL; + dosattr = origattr & ~ATTR_READONLY; + if (dosattr == 0) + dosattr |= ATTR_NORMAL; + dosattr |= ATTR_HIDDEN; + +/* rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr); + if (rc != 0) + goto out_reval; + + goto retry_std_delete;*/ + } + + /* undo the setattr if we errored out and it's needed */ +/* if (rc != 0 && dosattr != 0) + cifs_set_file_info(inode, attrs, xid, full_path, origattr);*/ +out_reval: + if (inode) { + cifs_inode = CIFS_I(inode); + cifs_inode->time = 0; /* will force revalidate to get info + when needed */ + inode->i_ctime = current_fs_time(sb); + } + dir->i_ctime = dir->i_mtime = current_fs_time(sb); + cifs_inode = CIFS_I(dir); + CIFS_I(dir)->time = 0; /* force revalidate of dir as well */ +unlink_out: + kfree(attrs); + kfree(ucs_path); + FreeXid(xid); + cifs_put_tlink(tlink); + return rc; +} diff --git a/fs/cifs/smb2proto.h b/fs/cifs/smb2proto.h index c35f0a0..df8d421 100644 --- a/fs/cifs/smb2proto.h +++ b/fs/cifs/smb2proto.h @@ -103,6 +103,7 @@ extern int smb2_flush(struct file *file, fl_owner_t id); extern int smb2_mkdir(struct inode *inode, struct dentry *direntry, int mode); extern int smb2_rmdir(struct inode *inode, struct dentry *direntry); +extern int smb2_unlink(struct inode *dir, struct dentry *dentry); /* extern char *smb2_compose_mount_options(const char *sb_mountdata, const char *fullpath, const struct dfs_info3_param *ref, -- 1.7.1 -- To unsubscribe from this list: send the line "unsubscribe linux-cifs" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html