mystifyingyouth / rpms / polkit

Forked from rpms/polkit 2 years ago
Clone

Blame SOURCES/polkit-0.115-CVE-2019-6133.patch

b415b0
From 6cc6aafee135ba44ea748250d7d29b562ca190e3 Mon Sep 17 00:00:00 2001
b415b0
From: Colin Walters <walters@verbum.org>
b415b0
Date: Fri, 4 Jan 2019 14:24:48 -0500
b415b0
Subject: [PATCH] backend: Compare PolkitUnixProcess uids for temporary
b415b0
 authorizations
b415b0
b415b0
It turns out that the combination of `(pid, start time)` is not
b415b0
enough to be unique.  For temporary authorizations, we can avoid
b415b0
separate users racing on pid reuse by simply comparing the uid.
b415b0
b415b0
https://bugs.chromium.org/p/project-zero/issues/detail?id=1692
b415b0
b415b0
And the above original email report is included in full in a new comment.
b415b0
b415b0
Reported-by: Jann Horn <jannh@google.com>
b415b0
b415b0
Closes: https://gitlab.freedesktop.org/polkit/polkit/issues/75
b415b0
---
b415b0
 src/polkit/polkitsubject.c                    |  2 +
b415b0
 src/polkit/polkitunixprocess.c                | 71 ++++++++++++++++++-
b415b0
 .../polkitbackendinteractiveauthority.c       | 39 +++++++++-
b415b0
 3 files changed, 110 insertions(+), 2 deletions(-)
b415b0
b415b0
diff --git a/src/polkit/polkitsubject.c b/src/polkit/polkitsubject.c
b415b0
index d4c1182..ccabd0a 100644
b415b0
--- a/src/polkit/polkitsubject.c
b415b0
+++ b/src/polkit/polkitsubject.c
b415b0
@@ -99,6 +99,8 @@ polkit_subject_hash (PolkitSubject *subject)
b415b0
  * @b: A #PolkitSubject.
b415b0
  *
b415b0
  * Checks if @a and @b are equal, ie. represent the same subject.
b415b0
+ * However, avoid calling polkit_subject_equal() to compare two processes;
b415b0
+ * for more information see the `PolkitUnixProcess` documentation.
b415b0
  *
b415b0
  * This function can be used in e.g. g_hash_table_new().
b415b0
  *
b415b0
diff --git a/src/polkit/polkitunixprocess.c b/src/polkit/polkitunixprocess.c
b415b0
index b02b258..78d7251 100644
b415b0
--- a/src/polkit/polkitunixprocess.c
b415b0
+++ b/src/polkit/polkitunixprocess.c
b415b0
@@ -51,7 +51,10 @@
b415b0
  * @title: PolkitUnixProcess
b415b0
  * @short_description: Unix processs
b415b0
  *
b415b0
- * An object for representing a UNIX process.
b415b0
+ * An object for representing a UNIX process.  NOTE: This object as
b415b0
+ * designed is now known broken; a mechanism to exploit a delay in
b415b0
+ * start time in the Linux kernel was identified.  Avoid
b415b0
+ * calling polkit_subject_equal() to compare two processes.
b415b0
  *
b415b0
  * To uniquely identify processes, both the process id and the start
b415b0
  * time of the process (a monotonic increasing value representing the
b415b0
@@ -66,6 +69,72 @@
b415b0
  * polkit_unix_process_new_for_owner() with trusted data.
b415b0
  */
b415b0
 
b415b0
+/* See https://gitlab.freedesktop.org/polkit/polkit/issues/75
b415b0
+
b415b0
+  But quoting the original email in full here to ensure it's preserved:
b415b0
+
b415b0
+  From: Jann Horn <jannh@google.com>
b415b0
+  Subject: [SECURITY] polkit: temporary auth hijacking via PID reuse and non-atomic fork
b415b0
+  Date: Wednesday, October 10, 2018 5:34 PM
b415b0
+
b415b0
+When a (non-root) user attempts to e.g. control systemd units in the system
b415b0
+instance from an active session over DBus, the access is gated by a polkit
b415b0
+policy that requires "auth_admin_keep" auth. This results in an auth prompt
b415b0
+being shown to the user, asking the user to confirm the action by entering the
b415b0
+password of an administrator account.
b415b0
+
b415b0
+After the action has been confirmed, the auth decision for "auth_admin_keep" is
b415b0
+cached for up to five minutes. Subject to some restrictions, similar actions can
b415b0
+then be performed in this timespan without requiring re-auth:
b415b0
+
b415b0
+ - The PID of the DBus client requesting the new action must match the PID of
b415b0
+   the DBus client requesting the old action (based on SO_PEERCRED information
b415b0
+   forwarded by the DBus daemon).
b415b0
+ - The "start time" of the client's PID (as seen in /proc/$pid/stat, field 22)
b415b0
+   must not have changed. The granularity of this timestamp is in the
b415b0
+   millisecond range.
b415b0
+ - polkit polls every two seconds whether a process with the expected start time
b415b0
+   still exists. If not, the temporary auth entry is purged.
b415b0
+
b415b0
+Without the start time check, this would obviously be buggy because an attacker
b415b0
+could simply wait for the legitimate client to disappear, then create a new
b415b0
+client with the same PID.
b415b0
+
b415b0
+Unfortunately, the start time check is bypassable because fork() is not atomic.
b415b0
+Looking at the source code of copy_process() in the kernel:
b415b0
+
b415b0
+        p->start_time = ktime_get_ns();
b415b0
+        p->real_start_time = ktime_get_boot_ns();
b415b0
+        [...]
b415b0
+        retval = copy_thread_tls(clone_flags, stack_start, stack_size, p, tls);
b415b0
+        if (retval)
b415b0
+                goto bad_fork_cleanup_io;
b415b0
+
b415b0
+        if (pid != &init_struct_pid) {
b415b0
+                pid = alloc_pid(p->nsproxy->pid_ns_for_children);
b415b0
+                if (IS_ERR(pid)) {
b415b0
+                        retval = PTR_ERR(pid);
b415b0
+                        goto bad_fork_cleanup_thread;
b415b0
+                }
b415b0
+        }
b415b0
+
b415b0
+The ktime_get_boot_ns() call is where the "start time" of the process is
b415b0
+recorded. The alloc_pid() call is where a free PID is allocated. In between
b415b0
+these, some time passes; and because the copy_thread_tls() call between them can
b415b0
+access userspace memory when sys_clone() is invoked through the 32-bit syscall
b415b0
+entry point, an attacker can even stall the kernel arbitrarily long at this
b415b0
+point (by supplying a pointer into userspace memory that is associated with a
b415b0
+userfaultfd or is backed by a custom FUSE filesystem).
b415b0
+
b415b0
+This means that an attacker can immediately call sys_clone() when the victim
b415b0
+process is created, often resulting in a process that has the exact same start
b415b0
+time reported in procfs; and then the attacker can delay the alloc_pid() call
b415b0
+until after the victim process has died and the PID assignment has cycled
b415b0
+around. This results in an attacker process that polkit can't distinguish from
b415b0
+the victim process.
b415b0
+*/
b415b0
+
b415b0
+
b415b0
 /**
b415b0
  * PolkitUnixProcess:
b415b0
  *
b415b0
diff --git a/src/polkitbackend/polkitbackendinteractiveauthority.c b/src/polkitbackend/polkitbackendinteractiveauthority.c
b415b0
index a1630b9..80e8141 100644
b415b0
--- a/src/polkitbackend/polkitbackendinteractiveauthority.c
b415b0
+++ b/src/polkitbackend/polkitbackendinteractiveauthority.c
b415b0
@@ -3031,6 +3031,43 @@ temporary_authorization_store_free (TemporaryAuthorizationStore *store)
b415b0
   g_free (store);
b415b0
 }
b415b0
 
b415b0
+/* See the comment at the top of polkitunixprocess.c */
b415b0
+static gboolean
b415b0
+subject_equal_for_authz (PolkitSubject *a,
b415b0
+                         PolkitSubject *b)
b415b0
+{
b415b0
+  if (!polkit_subject_equal (a, b))
b415b0
+    return FALSE;
b415b0
+
b415b0
+  /* Now special case unix processes, as we want to protect against
b415b0
+   * pid reuse by including the UID.
b415b0
+   */
b415b0
+  if (POLKIT_IS_UNIX_PROCESS (a) && POLKIT_IS_UNIX_PROCESS (b)) {
b415b0
+    PolkitUnixProcess *ap = (PolkitUnixProcess*)a;
b415b0
+    int uid_a = polkit_unix_process_get_uid ((PolkitUnixProcess*)a);
b415b0
+    PolkitUnixProcess *bp = (PolkitUnixProcess*)b;
b415b0
+    int uid_b = polkit_unix_process_get_uid ((PolkitUnixProcess*)b);
b415b0
+
b415b0
+    if (uid_a != -1 && uid_b != -1)
b415b0
+      {
b415b0
+        if (uid_a == uid_b)
b415b0
+          {
b415b0
+            return TRUE;
b415b0
+          }
b415b0
+        else
b415b0
+          {
b415b0
+            g_printerr ("denying slowfork; pid %d uid %d != %d!\n",
b415b0
+                        polkit_unix_process_get_pid (ap),
b415b0
+                        uid_a, uid_b);
b415b0
+            return FALSE;
b415b0
+          }
b415b0
+      }
b415b0
+    /* Fall through; one of the uids is unset so we can't reliably compare */
b415b0
+  }
b415b0
+
b415b0
+  return TRUE;
b415b0
+}
b415b0
+
b415b0
 static gboolean
b415b0
 temporary_authorization_store_has_authorization (TemporaryAuthorizationStore *store,
b415b0
                                                  PolkitSubject               *subject,
b415b0
@@ -3073,7 +3110,7 @@ temporary_authorization_store_has_authorization (TemporaryAuthorizationStore *st
b415b0
     TemporaryAuthorization *authorization = l->data;
b415b0
 
b415b0
     if (strcmp (action_id, authorization->action_id) == 0 &&
b415b0
-        polkit_subject_equal (subject_to_use, authorization->subject))
b415b0
+        subject_equal_for_authz (subject_to_use, authorization->subject))
b415b0
       {
b415b0
         ret = TRUE;
b415b0
         if (out_tmp_authz_id != NULL)
b415b0
-- 
b415b0
2.19.2
b415b0