Skip to content

Commit 197e7e5

Browse files
committed
Sanitize 'move_pages()' permission checks
The 'move_paghes()' system call was introduced long long ago with the same permission checks as for sending a signal (except using CAP_SYS_NICE instead of CAP_SYS_KILL for the overriding capability). That turns out to not be a great choice - while the system call really only moves physical page allocations around (and you need other capabilities to do a lot of it), you can check the return value to map out some the virtual address choices and defeat ASLR of a binary that still shares your uid. So change the access checks to the more common 'ptrace_may_access()' model instead. This tightens the access checks for the uid, and also effectively changes the CAP_SYS_NICE check to CAP_SYS_PTRACE, but it's unlikely that anybody really _uses_ this legacy system call any more (we hav ebetter NUMA placement models these days), so I expect nobody to notice. Famous last words. Reported-by: Otto Ebeling <otto.ebeling@iki.fi> Acked-by: Eric W. Biederman <ebiederm@xmission.com> Cc: Willy Tarreau <w@1wt.eu> Cc: stable@kernel.org Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
1 parent 7f680d7 commit 197e7e5

File tree

1 file changed

+3
-8
lines changed

1 file changed

+3
-8
lines changed

mm/migrate.c

Lines changed: 3 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@
4141
#include <linux/page_idle.h>
4242
#include <linux/page_owner.h>
4343
#include <linux/sched/mm.h>
44+
#include <linux/ptrace.h>
4445

4546
#include <asm/tlbflush.h>
4647

@@ -1652,7 +1653,6 @@ SYSCALL_DEFINE6(move_pages, pid_t, pid, unsigned long, nr_pages,
16521653
const int __user *, nodes,
16531654
int __user *, status, int, flags)
16541655
{
1655-
const struct cred *cred = current_cred(), *tcred;
16561656
struct task_struct *task;
16571657
struct mm_struct *mm;
16581658
int err;
@@ -1676,14 +1676,9 @@ SYSCALL_DEFINE6(move_pages, pid_t, pid, unsigned long, nr_pages,
16761676

16771677
/*
16781678
* Check if this process has the right to modify the specified
1679-
* process. The right exists if the process has administrative
1680-
* capabilities, superuser privileges or the same
1681-
* userid as the target process.
1679+
* process. Use the regular "ptrace_may_access()" checks.
16821680
*/
1683-
tcred = __task_cred(task);
1684-
if (!uid_eq(cred->euid, tcred->suid) && !uid_eq(cred->euid, tcred->uid) &&
1685-
!uid_eq(cred->uid, tcred->suid) && !uid_eq(cred->uid, tcred->uid) &&
1686-
!capable(CAP_SYS_NICE)) {
1681+
if (!ptrace_may_access(task, PTRACE_MODE_READ_REALCREDS)) {
16871682
rcu_read_unlock();
16881683
err = -EPERM;
16891684
goto out;

0 commit comments

Comments
 (0)