diff mbox series

[v4] cifs: Fix lease buffer length error

Message ID 1554689950-30863-1-git-send-email-zhangxiaoxu5@huawei.com
State New
Headers show
Series [v4] cifs: Fix lease buffer length error | expand

Commit Message

zhangxiaoxu (A) April 8, 2019, 2:19 a.m. UTC
There is a KASAN slab-out-of-bounds:
BUG: KASAN: slab-out-of-bounds in _copy_from_iter_full+0x783/0xaa0
Read of size 80 at addr ffff88810c35e180 by task mount.cifs/539

CPU: 1 PID: 539 Comm: mount.cifs Not tainted 4.19 #10
Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS
            rel-1.12.0-0-ga698c8995f-prebuilt.qemu.org 04/01/2014
Call Trace:
 dump_stack+0xdd/0x12a
 print_address_description+0xa7/0x540
 kasan_report+0x1ff/0x550
 check_memory_region+0x2f1/0x310
 memcpy+0x2f/0x80
 _copy_from_iter_full+0x783/0xaa0
 tcp_sendmsg_locked+0x1840/0x4140
 tcp_sendmsg+0x37/0x60
 inet_sendmsg+0x18c/0x490
 sock_sendmsg+0xae/0x130
 smb_send_kvec+0x29c/0x520
 __smb_send_rqst+0x3ef/0xc60
 smb_send_rqst+0x25a/0x2e0
 compound_send_recv+0x9e8/0x2af0
 cifs_send_recv+0x24/0x30
 SMB2_open+0x35e/0x1620
 open_shroot+0x27b/0x490
 smb2_open_op_close+0x4e1/0x590
 smb2_query_path_info+0x2ac/0x650
 cifs_get_inode_info+0x1058/0x28f0
 cifs_root_iget+0x3bb/0xf80
 cifs_smb3_do_mount+0xe00/0x14c0
 cifs_do_mount+0x15/0x20
 mount_fs+0x5e/0x290
 vfs_kern_mount+0x88/0x460
 do_mount+0x398/0x31e0
 ksys_mount+0xc6/0x150
 __x64_sys_mount+0xea/0x190
 do_syscall_64+0x122/0x590
 entry_SYSCALL_64_after_hwframe+0x44/0xa9

It can be reproduced by the following step:
  1. samba configured with: server max protocol = SMB2_10
  2. mount -o vers=default

When parse the mount version parameter, the 'ops' and 'vals'
was setted to smb30,  if negotiate result is smb21, just
update the 'ops' to smb21, but the 'vals' is still smb30.
When add lease context, the iov_base is allocated with smb21
ops, but the iov_len is initiallited with the smb30. Because
the iov_len is longer than iov_base, when send the message,
copy array out of bounds.

we need to keep the 'ops' and 'vals' consistent.

Fixes: 9764c02fcbad(SMB3: Add support for multidialect negotiate (SMB2.1 and later))
Fixes: d5c7076b772a(smb3: add smb3.1.1 to default dialect list)

cc: stable@vger.kernel.org (v4.14+)
Signed-off-by: ZhangXiaoxu <zhangxiaoxu5@huawei.com>
---
 fs/cifs/smb2pdu.c | 5 ++++-
 1 file changed, 4 insertions(+), 1 deletion(-)

Comments

Pavel Shilovsky April 8, 2019, 11:09 p.m. UTC | #1
вс, 7 апр. 2019 г. в 19:15, ZhangXiaoxu <zhangxiaoxu5@huawei.com>:
>
> There is a KASAN slab-out-of-bounds:
> BUG: KASAN: slab-out-of-bounds in _copy_from_iter_full+0x783/0xaa0
> Read of size 80 at addr ffff88810c35e180 by task mount.cifs/539
>
> CPU: 1 PID: 539 Comm: mount.cifs Not tainted 4.19 #10
> Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS
>             rel-1.12.0-0-ga698c8995f-prebuilt.qemu.org 04/01/2014
> Call Trace:
>  dump_stack+0xdd/0x12a
>  print_address_description+0xa7/0x540
>  kasan_report+0x1ff/0x550
>  check_memory_region+0x2f1/0x310
>  memcpy+0x2f/0x80
>  _copy_from_iter_full+0x783/0xaa0
>  tcp_sendmsg_locked+0x1840/0x4140
>  tcp_sendmsg+0x37/0x60
>  inet_sendmsg+0x18c/0x490
>  sock_sendmsg+0xae/0x130
>  smb_send_kvec+0x29c/0x520
>  __smb_send_rqst+0x3ef/0xc60
>  smb_send_rqst+0x25a/0x2e0
>  compound_send_recv+0x9e8/0x2af0
>  cifs_send_recv+0x24/0x30
>  SMB2_open+0x35e/0x1620
>  open_shroot+0x27b/0x490
>  smb2_open_op_close+0x4e1/0x590
>  smb2_query_path_info+0x2ac/0x650
>  cifs_get_inode_info+0x1058/0x28f0
>  cifs_root_iget+0x3bb/0xf80
>  cifs_smb3_do_mount+0xe00/0x14c0
>  cifs_do_mount+0x15/0x20
>  mount_fs+0x5e/0x290
>  vfs_kern_mount+0x88/0x460
>  do_mount+0x398/0x31e0
>  ksys_mount+0xc6/0x150
>  __x64_sys_mount+0xea/0x190
>  do_syscall_64+0x122/0x590
>  entry_SYSCALL_64_after_hwframe+0x44/0xa9
>
> It can be reproduced by the following step:
>   1. samba configured with: server max protocol = SMB2_10
>   2. mount -o vers=default
>
> When parse the mount version parameter, the 'ops' and 'vals'
> was setted to smb30,  if negotiate result is smb21, just
> update the 'ops' to smb21, but the 'vals' is still smb30.
> When add lease context, the iov_base is allocated with smb21
> ops, but the iov_len is initiallited with the smb30. Because
> the iov_len is longer than iov_base, when send the message,
> copy array out of bounds.
>
> we need to keep the 'ops' and 'vals' consistent.
>
> Fixes: 9764c02fcbad(SMB3: Add support for multidialect negotiate (SMB2.1 and later))
> Fixes: d5c7076b772a(smb3: add smb3.1.1 to default dialect list)
>
> cc: stable@vger.kernel.org (v4.14+)
> Signed-off-by: ZhangXiaoxu <zhangxiaoxu5@huawei.com>
> ---
>  fs/cifs/smb2pdu.c | 5 ++++-
>  1 file changed, 4 insertions(+), 1 deletion(-)
>
> diff --git a/fs/cifs/smb2pdu.c b/fs/cifs/smb2pdu.c
> index 21ad01d..0f14471 100644
> --- a/fs/cifs/smb2pdu.c
> +++ b/fs/cifs/smb2pdu.c
> @@ -832,8 +832,11 @@ SMB2_negotiate(const unsigned int xid, struct cifs_ses *ses)
>                 } else if (rsp->DialectRevision == cpu_to_le16(SMB21_PROT_ID)) {
>                         /* ops set to 3.0 by default for default so update */
>                         ses->server->ops = &smb21_operations;
> -               } else if (rsp->DialectRevision == cpu_to_le16(SMB311_PROT_ID))
> +                       ses->server->vals = &smb21_values;
> +               } else if (rsp->DialectRevision == cpu_to_le16(SMB311_PROT_ID)) {
>                         ses->server->ops = &smb311_operations;
> +                       ses->server->vals = &smb311_values;
> +               }
>         } else if (le16_to_cpu(rsp->DialectRevision) !=
>                                 ses->server->vals->protocol_id) {
>                 /* if requested single dialect ensure returned dialect matched */
> --
> 2.7.4
>

Reviewed-by: Pavel Shilovsky <pshilov@microsoft.com>

--
Best regards,
Pavel Shilovsky
diff mbox series

Patch

diff --git a/fs/cifs/smb2pdu.c b/fs/cifs/smb2pdu.c
index 21ad01d..0f14471 100644
--- a/fs/cifs/smb2pdu.c
+++ b/fs/cifs/smb2pdu.c
@@ -832,8 +832,11 @@  SMB2_negotiate(const unsigned int xid, struct cifs_ses *ses)
 		} else if (rsp->DialectRevision == cpu_to_le16(SMB21_PROT_ID)) {
 			/* ops set to 3.0 by default for default so update */
 			ses->server->ops = &smb21_operations;
-		} else if (rsp->DialectRevision == cpu_to_le16(SMB311_PROT_ID))
+			ses->server->vals = &smb21_values;
+		} else if (rsp->DialectRevision == cpu_to_le16(SMB311_PROT_ID)) {
 			ses->server->ops = &smb311_operations;
+			ses->server->vals = &smb311_values;
+		}
 	} else if (le16_to_cpu(rsp->DialectRevision) !=
 				ses->server->vals->protocol_id) {
 		/* if requested single dialect ensure returned dialect matched */