]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
smb: client: fix hardlinking of reparse points
authorPaulo Alcantara <pc@manguebit.com>
Sun, 26 Nov 2023 02:55:07 +0000 (23:55 -0300)
committerSteve French <stfrench@microsoft.com>
Sun, 7 Jan 2024 21:46:06 +0000 (15:46 -0600)
The client was sending an SMB2_CREATE request without setting
OPEN_REPARSE_POINT flag thus failing the entire hardlink operation.

Fix this by setting OPEN_REPARSE_POINT in create options for
SMB2_CREATE request when the source inode is a repase point.

Signed-off-by: Paulo Alcantara (SUSE) <pc@manguebit.com>
Signed-off-by: Steve French <stfrench@microsoft.com>
fs/smb/client/cifsglob.h
fs/smb/client/cifsproto.h
fs/smb/client/cifssmb.c
fs/smb/client/link.c
fs/smb/client/smb2inode.c
fs/smb/client/smb2proto.h

index ad5cf3e31d7342cd7ed9911550a31f33d9672868..d5d6aedef55812e53df5e8896321a37743bca58c 100644 (file)
@@ -405,9 +405,11 @@ struct smb_version_operations {
                      const char *from_name, const char *to_name,
                      struct cifs_sb_info *cifs_sb);
        /* send create hardlink request */
-       int (*create_hardlink)(const unsigned int, struct cifs_tcon *,
-                              const char *, const char *,
-                              struct cifs_sb_info *);
+       int (*create_hardlink)(const unsigned int xid,
+                              struct cifs_tcon *tcon,
+                              struct dentry *source_dentry,
+                              const char *from_name, const char *to_name,
+                              struct cifs_sb_info *cifs_sb);
        /* query symlink target */
        int (*query_symlink)(const unsigned int xid,
                             struct cifs_tcon *tcon,
index e680fe46d4e86deea8c6ad527280214fdf9e77b5..afbab86331a1f3970b7dc7b58deadd28eda3cff6 100644 (file)
@@ -447,9 +447,11 @@ extern int CIFSSMBRenameOpenFile(const unsigned int xid, struct cifs_tcon *tcon,
                                 int netfid, const char *target_name,
                                 const struct nls_table *nls_codepage,
                                 int remap_special_chars);
-extern int CIFSCreateHardLink(const unsigned int xid, struct cifs_tcon *tcon,
-                             const char *from_name, const char *to_name,
-                             struct cifs_sb_info *cifs_sb);
+int CIFSCreateHardLink(const unsigned int xid,
+                      struct cifs_tcon *tcon,
+                      struct dentry *source_dentry,
+                      const char *from_name, const char *to_name,
+                      struct cifs_sb_info *cifs_sb);
 extern int CIFSUnixCreateHardLink(const unsigned int xid,
                        struct cifs_tcon *tcon,
                        const char *fromName, const char *toName,
index 5bdea01919e8bd693701e39fe197a402df589fc9..e9e33b0b3ac472536db1a53f72cbb2635312087c 100644 (file)
@@ -2530,10 +2530,11 @@ createHardLinkRetry:
        return rc;
 }
 
-int
-CIFSCreateHardLink(const unsigned int xid, struct cifs_tcon *tcon,
-                  const char *from_name, const char *to_name,
-                  struct cifs_sb_info *cifs_sb)
+int CIFSCreateHardLink(const unsigned int xid,
+                      struct cifs_tcon *tcon,
+                      struct dentry *source_dentry,
+                      const char *from_name, const char *to_name,
+                      struct cifs_sb_info *cifs_sb)
 {
        int rc = 0;
        NT_RENAME_REQ *pSMB = NULL;
index 5ce0f74be4ecd2c40dd728545091332f710b685e..691f43a1ec2bcebd326e5a9f9542f80087b9ae9e 100644 (file)
@@ -510,8 +510,8 @@ cifs_hardlink(struct dentry *old_file, struct inode *inode,
                        rc = -ENOSYS;
                        goto cifs_hl_exit;
                }
-               rc = server->ops->create_hardlink(xid, tcon, from_name, to_name,
-                                                 cifs_sb);
+               rc = server->ops->create_hardlink(xid, tcon, old_file,
+                                                 from_name, to_name, cifs_sb);
                if ((rc == -EIO) || (rc == -EINVAL))
                        rc = -EOPNOTSUPP;
        }
index 2d9cdc1944031ca4e73a03915edeb7107026b0f6..919f778be84d0ba69e39d135097a470987e6d0ae 100644 (file)
@@ -44,6 +44,18 @@ static struct reparse_data_buffer *reparse_buf_ptr(struct kvec *iov)
        return buf;
 }
 
+static inline __u32 file_create_options(struct dentry *dentry)
+{
+       struct cifsInodeInfo *ci;
+
+       if (dentry) {
+               ci = CIFS_I(d_inode(dentry));
+               if (ci->cifsAttrs & ATTR_REPARSE)
+                       return OPEN_REPARSE_POINT;
+       }
+       return 0;
+}
+
 /*
  * note: If cfile is passed, the reference to it is dropped here.
  * So make sure that you do not reuse cfile after return from this func.
@@ -920,15 +932,9 @@ int smb2_rename_path(const unsigned int xid,
                     const char *from_name, const char *to_name,
                     struct cifs_sb_info *cifs_sb)
 {
-       struct cifsInodeInfo *ci;
        struct cifsFileInfo *cfile;
-       __u32 co = 0;
+       __u32 co = file_create_options(source_dentry);
 
-       if (source_dentry) {
-               ci = CIFS_I(d_inode(source_dentry));
-               if (ci->cifsAttrs & ATTR_REPARSE)
-                       co |= OPEN_REPARSE_POINT;
-       }
        drop_cached_dir_by_name(xid, tcon, from_name, cifs_sb);
        cifs_get_writable_path(tcon, from_name, FIND_WR_WITH_DELETE, &cfile);
 
@@ -936,13 +942,16 @@ int smb2_rename_path(const unsigned int xid,
                                  co, DELETE, SMB2_OP_RENAME, cfile);
 }
 
-int
-smb2_create_hardlink(const unsigned int xid, struct cifs_tcon *tcon,
-                    const char *from_name, const char *to_name,
-                    struct cifs_sb_info *cifs_sb)
+int smb2_create_hardlink(const unsigned int xid,
+                        struct cifs_tcon *tcon,
+                        struct dentry *source_dentry,
+                        const char *from_name, const char *to_name,
+                        struct cifs_sb_info *cifs_sb)
 {
+       __u32 co = file_create_options(source_dentry);
+
        return smb2_set_path_attr(xid, tcon, from_name, to_name,
-                                 cifs_sb, 0, FILE_READ_ATTRIBUTES,
+                                 cifs_sb, co, FILE_READ_ATTRIBUTES,
                                  SMB2_OP_HARDLINK, NULL);
 }
 
index aef50316670acdccb398a7812e7f07514de811de..efa2f8fe23449ad7fea349949a3ba3c77c93a9e1 100644 (file)
@@ -91,9 +91,11 @@ int smb2_rename_path(const unsigned int xid,
                     struct dentry *source_dentry,
                     const char *from_name, const char *to_name,
                     struct cifs_sb_info *cifs_sb);
-extern int smb2_create_hardlink(const unsigned int xid, struct cifs_tcon *tcon,
-                               const char *from_name, const char *to_name,
-                               struct cifs_sb_info *cifs_sb);
+int smb2_create_hardlink(const unsigned int xid,
+                        struct cifs_tcon *tcon,
+                        struct dentry *source_dentry,
+                        const char *from_name, const char *to_name,
+                        struct cifs_sb_info *cifs_sb);
 extern int smb3_create_mf_symlink(unsigned int xid, struct cifs_tcon *tcon,
                        struct cifs_sb_info *cifs_sb, const unsigned char *path,
                        char *pbuf, unsigned int *pbytes_written);