diff options
author | David Howells <dhowells@redhat.com> | 2008-11-13 18:39:19 -0500 |
---|---|---|
committer | James Morris <jmorris@namei.org> | 2008-11-13 18:39:19 -0500 |
commit | c69e8d9c01db2adc503464993c358901c9af9de4 (patch) | |
tree | bed94aaa9aeb7a7834d1c880f72b62a11a752c78 /mm/migrate.c | |
parent | 86a264abe542cfececb4df129bc45a0338d8cdb9 (diff) |
CRED: Use RCU to access another task's creds and to release a task's own creds
Use RCU to access another task's creds and to release a task's own creds.
This means that it will be possible for the credentials of a task to be
replaced without another task (a) requiring a full lock to read them, and (b)
seeing deallocated memory.
Signed-off-by: David Howells <dhowells@redhat.com>
Acked-by: James Morris <jmorris@namei.org>
Acked-by: Serge Hallyn <serue@us.ibm.com>
Signed-off-by: James Morris <jmorris@namei.org>
Diffstat (limited to 'mm/migrate.c')
-rw-r--r-- | mm/migrate.c | 8 |
1 files changed, 5 insertions, 3 deletions
diff --git a/mm/migrate.c b/mm/migrate.c index 794443da1b4f..142284229ce2 100644 --- a/mm/migrate.c +++ b/mm/migrate.c | |||
@@ -1045,7 +1045,7 @@ asmlinkage long sys_move_pages(pid_t pid, unsigned long nr_pages, | |||
1045 | const int __user *nodes, | 1045 | const int __user *nodes, |
1046 | int __user *status, int flags) | 1046 | int __user *status, int flags) |
1047 | { | 1047 | { |
1048 | struct cred *cred, *tcred; | 1048 | const struct cred *cred = current_cred(), *tcred; |
1049 | struct task_struct *task; | 1049 | struct task_struct *task; |
1050 | struct mm_struct *mm; | 1050 | struct mm_struct *mm; |
1051 | int err; | 1051 | int err; |
@@ -1076,14 +1076,16 @@ asmlinkage long sys_move_pages(pid_t pid, unsigned long nr_pages, | |||
1076 | * capabilities, superuser privileges or the same | 1076 | * capabilities, superuser privileges or the same |
1077 | * userid as the target process. | 1077 | * userid as the target process. |
1078 | */ | 1078 | */ |
1079 | cred = current->cred; | 1079 | rcu_read_lock(); |
1080 | tcred = task->cred; | 1080 | tcred = __task_cred(task); |
1081 | if (cred->euid != tcred->suid && cred->euid != tcred->uid && | 1081 | if (cred->euid != tcred->suid && cred->euid != tcred->uid && |
1082 | cred->uid != tcred->suid && cred->uid != tcred->uid && | 1082 | cred->uid != tcred->suid && cred->uid != tcred->uid && |
1083 | !capable(CAP_SYS_NICE)) { | 1083 | !capable(CAP_SYS_NICE)) { |
1084 | rcu_read_unlock(); | ||
1084 | err = -EPERM; | 1085 | err = -EPERM; |
1085 | goto out; | 1086 | goto out; |
1086 | } | 1087 | } |
1088 | rcu_read_unlock(); | ||
1087 | 1089 | ||
1088 | err = security_task_movememory(task); | 1090 | err = security_task_movememory(task); |
1089 | if (err) | 1091 | if (err) |