Blame SOURCES/CVE-2020-36385.patch

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