aboutsummaryrefslogtreecommitdiffstats
path: root/fs/proc
diff options
context:
space:
mode:
authorOleg Nesterov <oleg@redhat.com>2014-01-23 18:55:40 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2014-01-23 19:37:01 -0500
commit9f6e963f06c19a57a876cb77a9c87f6a56295b13 (patch)
treeed59763257b26d1faa606f582ccbfbec4a3cfafa /fs/proc
parentd855a4b79f49ea07d1827fc0591490a6a324148b (diff)
proc: fix ->f_pos overflows in first_tid()
1. proc_task_readdir()->first_tid() path truncates f_pos to int, this is wrong even on 64bit. We could check that f_pos < PID_MAX or even INT_MAX in proc_task_readdir(), but this patch simply checks the potential overflow in first_tid(), this check is nop on 64bit. We do not care if it was negative and the new unsigned value is huge, all we need to ensure is that we never wrongly return !NULL. 2. Remove the 2nd "nr != 0" check before get_nr_threads(), nr_threads == 0 is not distinguishable from !pid_task() above. Signed-off-by: Oleg Nesterov <oleg@redhat.com> Cc: "Eric W. Biederman" <ebiederm@xmission.com> Cc: Michal Hocko <mhocko@suse.cz> Cc: Sameer Nanda <snanda@chromium.org> Cc: Sergey Dyasly <dserrg@gmail.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'fs/proc')
-rw-r--r--fs/proc/base.c14
1 files changed, 9 insertions, 5 deletions
diff --git a/fs/proc/base.c b/fs/proc/base.c
index 9b423fec9732..51507065263b 100644
--- a/fs/proc/base.c
+++ b/fs/proc/base.c
@@ -3097,10 +3097,14 @@ out_no_task:
3097 * In the case of a seek we start with the leader and walk nr 3097 * In the case of a seek we start with the leader and walk nr
3098 * threads past it. 3098 * threads past it.
3099 */ 3099 */
3100static struct task_struct *first_tid(struct pid *pid, int tid, 3100static struct task_struct *first_tid(struct pid *pid, int tid, loff_t f_pos,
3101 int nr, struct pid_namespace *ns) 3101 struct pid_namespace *ns)
3102{ 3102{
3103 struct task_struct *pos, *task; 3103 struct task_struct *pos, *task;
3104 unsigned long nr = f_pos;
3105
3106 if (nr != f_pos) /* 32bit overflow? */
3107 return NULL;
3104 3108
3105 rcu_read_lock(); 3109 rcu_read_lock();
3106 task = pid_task(pid, PIDTYPE_PID); 3110 task = pid_task(pid, PIDTYPE_PID);
@@ -3108,14 +3112,14 @@ static struct task_struct *first_tid(struct pid *pid, int tid,
3108 goto fail; 3112 goto fail;
3109 3113
3110 /* Attempt to start with the tid of a thread */ 3114 /* Attempt to start with the tid of a thread */
3111 if (tid && (nr > 0)) { 3115 if (tid && nr) {
3112 pos = find_task_by_pid_ns(tid, ns); 3116 pos = find_task_by_pid_ns(tid, ns);
3113 if (pos && same_thread_group(pos, task)) 3117 if (pos && same_thread_group(pos, task))
3114 goto found; 3118 goto found;
3115 } 3119 }
3116 3120
3117 /* If nr exceeds the number of threads there is nothing todo */ 3121 /* If nr exceeds the number of threads there is nothing todo */
3118 if (nr && nr >= get_nr_threads(task)) 3122 if (nr >= get_nr_threads(task))
3119 goto fail; 3123 goto fail;
3120 3124
3121 /* If we haven't found our starting place yet start 3125 /* If we haven't found our starting place yet start
@@ -3123,7 +3127,7 @@ static struct task_struct *first_tid(struct pid *pid, int tid,
3123 */ 3127 */
3124 pos = task = task->group_leader; 3128 pos = task = task->group_leader;
3125 do { 3129 do {
3126 if (nr-- <= 0) 3130 if (!nr--)
3127 goto found; 3131 goto found;
3128 } while_each_thread(task, pos); 3132 } while_each_thread(task, pos);
3129fail: 3133fail: