]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
ksmbd: fix FSCTL permission bypass by adding a permission check for FSCTL_SET_SPARSE
authorSean Shen <grayhat@foxmail.com>
Tue, 26 May 2026 13:07:16 +0000 (22:07 +0900)
committerSteve French <stfrench@microsoft.com>
Wed, 27 May 2026 01:36:36 +0000 (20:36 -0500)
FSCTL_SET_SPARSE in fsctl_set_sparse() modifies the file's sparse
attribute and saves it through xattr without any permission checks.

This exposes two issues:

1) A client on a read-only share can change the sparse attribute
   on files it opened, even though the share is read-only.
   Other FSCTL write operations already check
   test_tree_conn_flag(work->tcon, KSMBD_TREE_CONN_FLAG_WRITABLE),
   but FSCTL_SET_SPARSE does not.

2) Even on writable shares, clients without FILE_WRITE_DATA or
   FILE_WRITE_ATTRIBUTES access should not modify the sparse
   attribute. Similar handle-level checks exist in other functions
   but are missing here.

Add both share-level writable check and per-handle access check.
Use goto out on error to avoid leaking file references.

Fixes: e2f34481b24d ("cifsd: add server-side procedures for SMB3")
Cc: Namjae Jeon <linkinjeon@kernel.org>
Cc: Sergey Senozhatsky <sergey.senozhatsky@gmail.com>
Cc: Steve French <smfrench@gmail.com>
Signed-off-by: Sean Shen <grayhat@foxmail.com>
Acked-by: Namjae Jeon <linkinjeon@kernel.org>
Signed-off-by: Steve French <stfrench@microsoft.com>
fs/smb/server/smb2pdu.c

index 5128a693aca6ca587f4526c7a31738984ca961cd..620bcfbbfd92e76e32a94ffe57071ac8190accff 100644 (file)
@@ -8202,9 +8202,20 @@ static inline int fsctl_set_sparse(struct ksmbd_work *work, u64 id,
        int ret = 0;
        __le32 old_fattr;
 
+       if (!test_tree_conn_flag(work->tcon, KSMBD_TREE_CONN_FLAG_WRITABLE)) {
+               ksmbd_debug(SMB, "User does not have write permission\n");
+               return -EACCES;
+       }
+
        fp = ksmbd_lookup_fd_fast(work, id);
        if (!fp)
                return -ENOENT;
+
+       if (!(fp->daccess & (FILE_WRITE_DATA_LE | FILE_WRITE_ATTRIBUTES_LE))) {
+               ret = -EACCES;
+               goto out;
+       }
+
        idmap = file_mnt_idmap(fp->filp);
 
        old_fattr = fp->f_ci->m_fattr;