|
|
69eca8 |
From e77f229ecaf387f9f54430dbd277baf8c60b2716 Mon Sep 17 00:00:00 2001
|
|
|
69eca8 |
From: Joe Lawrence <joe.lawrence@redhat.com>
|
|
|
69eca8 |
Date: Wed, 27 Oct 2021 13:17:13 -0400
|
|
|
69eca8 |
Subject: [KPATCH CVE-2020-36385] RDMA/ucma: kpatch fixes for CVE-2020-36385
|
|
|
69eca8 |
|
|
|
69eca8 |
Kernels:
|
|
|
69eca8 |
3.10.0-1160.6.1.el7
|
|
|
69eca8 |
3.10.0-1160.11.1.el7
|
|
|
69eca8 |
3.10.0-1160.15.2.el7
|
|
|
69eca8 |
3.10.0-1160.21.1.el7
|
|
|
69eca8 |
3.10.0-1160.24.1.el7
|
|
|
69eca8 |
3.10.0-1160.25.1.el7
|
|
|
69eca8 |
3.10.0-1160.31.1.el7
|
|
|
69eca8 |
3.10.0-1160.36.2.el7
|
|
|
69eca8 |
3.10.0-1160.41.1.el7
|
|
|
69eca8 |
3.10.0-1160.42.2.el7
|
|
|
69eca8 |
3.10.0-1160.45.1.el7
|
|
|
69eca8 |
|
|
|
69eca8 |
Changes since last build:
|
|
|
69eca8 |
arches: x86_64 ppc64le
|
|
|
69eca8 |
ucma.o: changed function: ucma_migrate_id
|
|
|
69eca8 |
---------------------------
|
|
|
69eca8 |
|
|
|
69eca8 |
Kpatch-MR: https://gitlab.com/redhat/prdsc/rhel/src/kpatch/rhel-7/-/merge_requests/12
|
|
|
69eca8 |
Approved-by: Artem Savkov (@artem.savkov)
|
|
|
69eca8 |
Modifications:
|
|
|
69eca8 |
- Avoid the complications of reworking all the locks (and preceding
|
|
|
69eca8 |
commits) and apply a minimal patch to avoid the CVE condition.
|
|
|
69eca8 |
- Always inline ucma_unlock_files() to avoid new function on x64_64
|
|
|
69eca8 |
|
|
|
69eca8 |
Z-MR: https://gitlab.com/redhat/rhel/src/kernel/rhel-7/-/merge_requests/231
|
|
|
69eca8 |
|
|
|
69eca8 |
KT0 test PASS: https://beaker.engineering.redhat.com/jobs/5948342
|
|
|
69eca8 |
for kpatch-patch-3_10_0-1160_6_1-1-11.el7 scratch build:
|
|
|
69eca8 |
https://brewweb.engineering.redhat.com/brew/taskinfo?taskID=40661892
|
|
|
69eca8 |
|
|
|
69eca8 |
commit c71835cc23a3793651a693ea6cb1100e0eb9a0b1
|
|
|
69eca8 |
Author: Kamal Heib <kheib@redhat.com>
|
|
|
69eca8 |
Date: Sun Aug 1 10:49:07 2021 +0300
|
|
|
69eca8 |
|
|
|
69eca8 |
RDMA/ucma: Rework ucma_migrate_id() to avoid races with destroy
|
|
|
69eca8 |
|
|
|
69eca8 |
Bugzilla: https://bugzilla.redhat.com/show_bug.cgi?id=1978075
|
|
|
69eca8 |
CVE: CVE-2020-36385
|
|
|
69eca8 |
Conflicts:
|
|
|
69eca8 |
Adjust the patch to use "mut" mutext instead xa_lock due to the missing
|
|
|
69eca8 |
of: afcafe07af0e ("ucma: Convert ctx_idr to XArray").
|
|
|
69eca8 |
|
|
|
69eca8 |
commit f5449e74802c1112dea984aec8af7a33c4516af1
|
|
|
69eca8 |
Author: Jason Gunthorpe <jgg@nvidia.com>
|
|
|
69eca8 |
Date: Mon Sep 14 08:59:56 2020 -0300
|
|
|
69eca8 |
|
|
|
69eca8 |
RDMA/ucma: Rework ucma_migrate_id() to avoid races with destroy
|
|
|
69eca8 |
|
|
|
69eca8 |
ucma_destroy_id() assumes that all things accessing the ctx will do so via
|
|
|
69eca8 |
the xarray. This assumption violated only in the case the FD is being
|
|
|
69eca8 |
closed, then the ctx is reached via the ctx_list. Normally this is OK
|
|
|
69eca8 |
since ucma_destroy_id() cannot run concurrenty with release(), however
|
|
|
69eca8 |
with ucma_migrate_id() is involved this can violated as the close of the
|
|
|
69eca8 |
2nd FD can run concurrently with destroy on the first:
|
|
|
69eca8 |
|
|
|
69eca8 |
CPU0 CPU1
|
|
|
69eca8 |
ucma_destroy_id(fda)
|
|
|
69eca8 |
ucma_migrate_id(fda -> fdb)
|
|
|
69eca8 |
ucma_get_ctx()
|
|
|
69eca8 |
xa_lock()
|
|
|
69eca8 |
_ucma_find_context()
|
|
|
69eca8 |
xa_erase()
|
|
|
69eca8 |
xa_unlock()
|
|
|
69eca8 |
xa_lock()
|
|
|
69eca8 |
ctx->file = new_file
|
|
|
69eca8 |
list_move()
|
|
|
69eca8 |
xa_unlock()
|
|
|
69eca8 |
ucma_put_ctx()
|
|
|
69eca8 |
|
|
|
69eca8 |
ucma_close(fdb)
|
|
|
69eca8 |
_destroy_id()
|
|
|
69eca8 |
kfree(ctx)
|
|
|
69eca8 |
|
|
|
69eca8 |
_destroy_id()
|
|
|
69eca8 |
wait_for_completion()
|
|
|
69eca8 |
// boom, ctx was freed
|
|
|
69eca8 |
|
|
|
69eca8 |
The ctx->file must be modified under the handler and xa_lock, and prior to
|
|
|
69eca8 |
modification the ID must be rechecked that it is still reachable from
|
|
|
69eca8 |
cur_file, ie there is no parallel destroy or migrate.
|
|
|
69eca8 |
|
|
|
69eca8 |
To make this work remove the double locking and streamline the control
|
|
|
69eca8 |
flow. The double locking was obsoleted by the handler lock now directly
|
|
|
69eca8 |
preventing new uevents from being created, and the ctx_list cannot be read
|
|
|
69eca8 |
while holding fgets on both files. Removing the double locking also
|
|
|
69eca8 |
removes the need to check for the same file.
|
|
|
69eca8 |
|
|
|
69eca8 |
Fixes: 88314e4dda1e ("RDMA/cma: add support for rdma_migrate_id()")
|
|
|
69eca8 |
Link: https://lore.kernel.org/r/0-v1-05c5a4090305+3a872-ucma_syz_migrate_jgg@nvidia.com
|
|
|
69eca8 |
Reported-and-tested-by: syzbot+cc6fc752b3819e082d0c@syzkaller.appspotmail.com
|
|
|
69eca8 |
Signed-off-by: Jason Gunthorpe <jgg@nvidia.com>
|
|
|
69eca8 |
|
|
|
69eca8 |
Signed-off-by: Kamal Heib <kheib@redhat.com>
|
|
|
69eca8 |
|
|
|
69eca8 |
Signed-off-by: Joe Lawrence <joe.lawrence@redhat.com>
|
|
|
69eca8 |
---
|
|
|
69eca8 |
drivers/infiniband/core/ucma.c | 11 ++++++++++-
|
|
|
69eca8 |
1 file changed, 10 insertions(+), 1 deletion(-)
|
|
|
69eca8 |
|
|
|
69eca8 |
diff --git a/drivers/infiniband/core/ucma.c b/drivers/infiniband/core/ucma.c
|
|
|
69eca8 |
index 608a780d9ebb..72e7eb893d03 100644
|
|
|
69eca8 |
--- a/drivers/infiniband/core/ucma.c
|
|
|
69eca8 |
+++ b/drivers/infiniband/core/ucma.c
|
|
|
69eca8 |
@@ -1547,7 +1547,7 @@ static void ucma_lock_files(struct ucma_file *file1, struct ucma_file *file2)
|
|
|
69eca8 |
}
|
|
|
69eca8 |
}
|
|
|
69eca8 |
|
|
|
69eca8 |
-static void ucma_unlock_files(struct ucma_file *file1, struct ucma_file *file2)
|
|
|
69eca8 |
+static __always_inline void ucma_unlock_files(struct ucma_file *file1, struct ucma_file *file2)
|
|
|
69eca8 |
{
|
|
|
69eca8 |
if (file1 < file2) {
|
|
|
69eca8 |
mutex_unlock(&file2->mut);
|
|
|
69eca8 |
@@ -1610,6 +1610,14 @@ static ssize_t ucma_migrate_id(struct ucma_file *new_file,
|
|
|
69eca8 |
ucma_lock_files(cur_file, new_file);
|
|
|
69eca8 |
mutex_lock(&mut;;
|
|
|
69eca8 |
|
|
|
69eca8 |
+ /* CVE-2020-36385 kpatch: double check the context one last time */
|
|
|
69eca8 |
+ if (_ucma_find_context(cmd.id, cur_file) != ctx) {
|
|
|
69eca8 |
+ mutex_unlock(&mut;;
|
|
|
69eca8 |
+ ucma_unlock_files(cur_file, new_file);
|
|
|
69eca8 |
+ ret = -ENOENT;
|
|
|
69eca8 |
+ goto err_unlock;
|
|
|
69eca8 |
+ }
|
|
|
69eca8 |
+
|
|
|
69eca8 |
list_move_tail(&ctx->list, &new_file->ctx_list);
|
|
|
69eca8 |
ucma_move_events(ctx, new_file);
|
|
|
69eca8 |
ctx->file = new_file;
|
|
|
69eca8 |
@@ -1623,6 +1631,7 @@ response:
|
|
|
69eca8 |
&resp, sizeof(resp)))
|
|
|
69eca8 |
ret = -EFAULT;
|
|
|
69eca8 |
|
|
|
69eca8 |
+err_unlock:
|
|
|
69eca8 |
ucma_put_ctx(ctx);
|
|
|
69eca8 |
file_put:
|
|
|
69eca8 |
fdput(f);
|
|
|
69eca8 |
--
|
|
|
69eca8 |
2.26.3
|
|
|
69eca8 |
|
|
|
69eca8 |
|