110 lines
4.0 KiB
Diff
110 lines
4.0 KiB
Diff
From c027b463e9389e4421b21f170128f35c0f2941b9 Mon Sep 17 00:00:00 2001
|
|
From: Paulo Alcantara <pc@manguebit.org>
|
|
Date: Thu, 31 Jul 2025 20:46:43 -0300
|
|
Subject: smb: client: fix creating symlinks under POSIX mounts
|
|
|
|
SMB3.1.1 POSIX mounts support native symlinks that are created with
|
|
IO_REPARSE_TAG_SYMLINK reparse points, so skip the checking of
|
|
FILE_SUPPORTS_REPARSE_POINTS as some servers might not have it set.
|
|
|
|
Cc: linux-cifs@vger.kernel.org
|
|
Cc: Ralph Boehme <slow@samba.org>
|
|
Cc: David Howells <dhowells@redhat.com>
|
|
Cc: <stable@vger.kernel.org>
|
|
Reported-by: Matthew Richardson <m.richardson@ed.ac.uk>
|
|
Closes: https://marc.info/?i=1124e7cd-6a46-40a6-9f44-b7664a66654b@ed.ac.uk
|
|
Signed-off-by: Paulo Alcantara (Red Hat) <pc@manguebit.org>
|
|
Signed-off-by: Steve French <stfrench@microsoft.com>
|
|
---
|
|
fs/smb/client/cifsglob.h | 5 +++++
|
|
fs/smb/client/cifssmb.c | 4 ++--
|
|
fs/smb/client/link.c | 2 +-
|
|
fs/smb/client/smb1ops.c | 2 +-
|
|
fs/smb/client/smb2inode.c | 5 ++---
|
|
fs/smb/client/smb2ops.c | 5 ++---
|
|
6 files changed, 13 insertions(+), 10 deletions(-)
|
|
|
|
--- a/fs/smb/client/cifsglob.h
|
|
+++ b/fs/smb/client/cifsglob.h
|
|
@@ -2377,4 +2377,9 @@ static inline bool cifs_netbios_name(con
|
|
return ret;
|
|
}
|
|
|
|
+#define CIFS_REPARSE_SUPPORT(tcon) \
|
|
+ ((tcon)->posix_extensions || \
|
|
+ (le32_to_cpu((tcon)->fsAttrInfo.Attributes) & \
|
|
+ FILE_SUPPORTS_REPARSE_POINTS))
|
|
+
|
|
#endif /* _CIFS_GLOB_H */
|
|
--- a/fs/smb/client/cifssmb.c
|
|
+++ b/fs/smb/client/cifssmb.c
|
|
@@ -2751,7 +2751,7 @@ int cifs_query_reparse_point(const unsig
|
|
if (cap_unix(tcon->ses))
|
|
return -EOPNOTSUPP;
|
|
|
|
- if (!(le32_to_cpu(tcon->fsAttrInfo.Attributes) & FILE_SUPPORTS_REPARSE_POINTS))
|
|
+ if (!CIFS_REPARSE_SUPPORT(tcon))
|
|
return -EOPNOTSUPP;
|
|
|
|
oparms = (struct cifs_open_parms) {
|
|
@@ -2879,7 +2879,7 @@ struct inode *cifs_create_reparse_inode(
|
|
* attempt to create reparse point. This will prevent creating unusable
|
|
* empty object on the server.
|
|
*/
|
|
- if (!(le32_to_cpu(tcon->fsAttrInfo.Attributes) & FILE_SUPPORTS_REPARSE_POINTS))
|
|
+ if (!CIFS_REPARSE_SUPPORT(tcon))
|
|
return ERR_PTR(-EOPNOTSUPP);
|
|
|
|
#ifndef CONFIG_CIFS_XATTR
|
|
--- a/fs/smb/client/link.c
|
|
+++ b/fs/smb/client/link.c
|
|
@@ -635,7 +635,7 @@ cifs_symlink(struct mnt_idmap *idmap, st
|
|
case CIFS_SYMLINK_TYPE_NATIVE:
|
|
case CIFS_SYMLINK_TYPE_NFS:
|
|
case CIFS_SYMLINK_TYPE_WSL:
|
|
- if (le32_to_cpu(pTcon->fsAttrInfo.Attributes) & FILE_SUPPORTS_REPARSE_POINTS) {
|
|
+ if (CIFS_REPARSE_SUPPORT(pTcon)) {
|
|
rc = create_reparse_symlink(xid, inode, direntry, pTcon,
|
|
full_path, symname);
|
|
goto symlink_exit;
|
|
--- a/fs/smb/client/smb1ops.c
|
|
+++ b/fs/smb/client/smb1ops.c
|
|
@@ -1272,7 +1272,7 @@ cifs_make_node(unsigned int xid, struct
|
|
*/
|
|
return cifs_sfu_make_node(xid, inode, dentry, tcon,
|
|
full_path, mode, dev);
|
|
- } else if (le32_to_cpu(tcon->fsAttrInfo.Attributes) & FILE_SUPPORTS_REPARSE_POINTS) {
|
|
+ } else if (CIFS_REPARSE_SUPPORT(tcon)) {
|
|
/*
|
|
* mknod via reparse points requires server support for
|
|
* storing reparse points, which is available since
|
|
--- a/fs/smb/client/smb2inode.c
|
|
+++ b/fs/smb/client/smb2inode.c
|
|
@@ -1345,9 +1345,8 @@ struct inode *smb2_create_reparse_inode(
|
|
* attempt to create reparse point. This will prevent creating unusable
|
|
* empty object on the server.
|
|
*/
|
|
- if (!(le32_to_cpu(tcon->fsAttrInfo.Attributes) & FILE_SUPPORTS_REPARSE_POINTS))
|
|
- if (!tcon->posix_extensions)
|
|
- return ERR_PTR(-EOPNOTSUPP);
|
|
+ if (!CIFS_REPARSE_SUPPORT(tcon))
|
|
+ return ERR_PTR(-EOPNOTSUPP);
|
|
|
|
oparms = CIFS_OPARMS(cifs_sb, tcon, full_path,
|
|
SYNCHRONIZE | DELETE |
|
|
--- a/fs/smb/client/smb2ops.c
|
|
+++ b/fs/smb/client/smb2ops.c
|
|
@@ -5269,10 +5269,9 @@ static int smb2_make_node(unsigned int x
|
|
if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL) {
|
|
rc = cifs_sfu_make_node(xid, inode, dentry, tcon,
|
|
full_path, mode, dev);
|
|
- } else if ((le32_to_cpu(tcon->fsAttrInfo.Attributes) & FILE_SUPPORTS_REPARSE_POINTS)
|
|
- || (tcon->posix_extensions)) {
|
|
+ } else if (CIFS_REPARSE_SUPPORT(tcon)) {
|
|
rc = mknod_reparse(xid, inode, dentry, tcon,
|
|
- full_path, mode, dev);
|
|
+ full_path, mode, dev);
|
|
}
|
|
return rc;
|
|
}
|