forked from luck/tmp_suning_uos_patched
[PATCH] pidhash: kill switch_exec_pids
switch_exec_pids is only called from de_thread by way of exec, and it is only called when we are exec'ing from a non thread group leader. Currently switch_exec_pids gives the leader the pid of the thread and unhashes and rehashes all of the process groups. The leader is already in the EXIT_DEAD state so no one cares about it's pids. The only concern for the leader is that __unhash_process called from release_task will function correctly. If we don't touch the leader at all we know that __unhash_process will work fine so there is no need to touch the leader. For the task becomming the thread group leader, we just need to give it the pid of the old thread group leader, add it to the task list, and attach it to the session and the process group of the thread group. Currently de_thread is also adding the task to the task list which is just silly. Currently the only leader of __detach_pid besides detach_pid is switch_exec_pids because of the ugly extra work that was being performed. So this patch removes switch_exec_pids because it is doing too much, it is creating an unnecessary special case in pid.c, duing work duplicated in de_thread, and generally obscuring what it is going on. The necessary work is added to de_thread, and it seems to be a little clearer there what is going on. Signed-off-by: Eric W. Biederman <ebiederm@xmission.com> Cc: Oleg Nesterov <oleg@tv-sign.ru> Cc: Kirill Korotaev <dev@sw.ru> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
parent
652486fb84
commit
d73d65293e
14
fs/exec.c
14
fs/exec.c
|
@ -708,7 +708,17 @@ static int de_thread(struct task_struct *tsk)
|
||||||
remove_parent(current);
|
remove_parent(current);
|
||||||
remove_parent(leader);
|
remove_parent(leader);
|
||||||
|
|
||||||
switch_exec_pids(leader, current);
|
|
||||||
|
/* Become a process group leader with the old leader's pid.
|
||||||
|
* Note: The old leader also uses thispid until release_task
|
||||||
|
* is called. Odd but simple and correct.
|
||||||
|
*/
|
||||||
|
detach_pid(current, PIDTYPE_PID);
|
||||||
|
current->pid = leader->pid;
|
||||||
|
attach_pid(current, PIDTYPE_PID, current->pid);
|
||||||
|
attach_pid(current, PIDTYPE_PGID, current->signal->pgrp);
|
||||||
|
attach_pid(current, PIDTYPE_SID, current->signal->session);
|
||||||
|
list_add_tail(¤t->tasks, &init_task.tasks);
|
||||||
|
|
||||||
current->parent = current->real_parent = leader->real_parent;
|
current->parent = current->real_parent = leader->real_parent;
|
||||||
leader->parent = leader->real_parent = child_reaper;
|
leader->parent = leader->real_parent = child_reaper;
|
||||||
|
@ -722,8 +732,6 @@ static int de_thread(struct task_struct *tsk)
|
||||||
__ptrace_link(current, parent);
|
__ptrace_link(current, parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
list_del(¤t->tasks);
|
|
||||||
list_add_tail(¤t->tasks, &init_task.tasks);
|
|
||||||
current->exit_signal = SIGCHLD;
|
current->exit_signal = SIGCHLD;
|
||||||
|
|
||||||
BUG_ON(leader->exit_state != EXIT_ZOMBIE);
|
BUG_ON(leader->exit_state != EXIT_ZOMBIE);
|
||||||
|
|
|
@ -38,7 +38,6 @@ extern struct pid *FASTCALL(find_pid(enum pid_type, int));
|
||||||
|
|
||||||
extern int alloc_pidmap(void);
|
extern int alloc_pidmap(void);
|
||||||
extern void FASTCALL(free_pidmap(int));
|
extern void FASTCALL(free_pidmap(int));
|
||||||
extern void switch_exec_pids(struct task_struct *leader, struct task_struct *thread);
|
|
||||||
|
|
||||||
#define do_each_task_pid(who, type, task) \
|
#define do_each_task_pid(who, type, task) \
|
||||||
if ((task = find_task_by_pid_type(type, who))) { \
|
if ((task = find_task_by_pid_type(type, who))) { \
|
||||||
|
|
30
kernel/pid.c
30
kernel/pid.c
|
@ -217,36 +217,6 @@ task_t *find_task_by_pid_type(int type, int nr)
|
||||||
|
|
||||||
EXPORT_SYMBOL(find_task_by_pid_type);
|
EXPORT_SYMBOL(find_task_by_pid_type);
|
||||||
|
|
||||||
/*
|
|
||||||
* This function switches the PIDs if a non-leader thread calls
|
|
||||||
* sys_execve() - this must be done without releasing the PID.
|
|
||||||
* (which a detach_pid() would eventually do.)
|
|
||||||
*/
|
|
||||||
void switch_exec_pids(task_t *leader, task_t *thread)
|
|
||||||
{
|
|
||||||
__detach_pid(leader, PIDTYPE_PID);
|
|
||||||
__detach_pid(leader, PIDTYPE_TGID);
|
|
||||||
__detach_pid(leader, PIDTYPE_PGID);
|
|
||||||
__detach_pid(leader, PIDTYPE_SID);
|
|
||||||
|
|
||||||
__detach_pid(thread, PIDTYPE_PID);
|
|
||||||
__detach_pid(thread, PIDTYPE_TGID);
|
|
||||||
|
|
||||||
leader->pid = leader->tgid = thread->pid;
|
|
||||||
thread->pid = thread->tgid;
|
|
||||||
|
|
||||||
attach_pid(thread, PIDTYPE_PID, thread->pid);
|
|
||||||
attach_pid(thread, PIDTYPE_TGID, thread->tgid);
|
|
||||||
attach_pid(thread, PIDTYPE_PGID, thread->signal->pgrp);
|
|
||||||
attach_pid(thread, PIDTYPE_SID, thread->signal->session);
|
|
||||||
list_add_tail(&thread->tasks, &init_task.tasks);
|
|
||||||
|
|
||||||
attach_pid(leader, PIDTYPE_PID, leader->pid);
|
|
||||||
attach_pid(leader, PIDTYPE_TGID, leader->tgid);
|
|
||||||
attach_pid(leader, PIDTYPE_PGID, leader->signal->pgrp);
|
|
||||||
attach_pid(leader, PIDTYPE_SID, leader->signal->session);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The pid hash table is scaled according to the amount of memory in the
|
* The pid hash table is scaled according to the amount of memory in the
|
||||||
* machine. From a minimum of 16 slots up to 4096 slots at one gigabyte or
|
* machine. From a minimum of 16 slots up to 4096 slots at one gigabyte or
|
||||||
|
|
Loading…
Reference in New Issue
Block a user