diff options
Diffstat (limited to 'kernel/exit.c')
| -rw-r--r-- | kernel/exit.c | 33 | 
1 files changed, 21 insertions, 12 deletions
| diff --git a/kernel/exit.c b/kernel/exit.c index f7864ac2ecc1..5962d7ccf243 100644 --- a/kernel/exit.c +++ b/kernel/exit.c @@ -49,6 +49,7 @@  #include <linux/init_task.h>  #include <linux/perf_event.h>  #include <trace/events/sched.h> +#include <linux/hw_breakpoint.h>  #include <asm/uaccess.h>  #include <asm/unistd.h> @@ -110,9 +111,9 @@ static void __exit_signal(struct task_struct *tsk)  		 * We won't ever get here for the group leader, since it  		 * will have been the last reference on the signal_struct.  		 */ -		sig->utime = cputime_add(sig->utime, task_utime(tsk)); -		sig->stime = cputime_add(sig->stime, task_stime(tsk)); -		sig->gtime = cputime_add(sig->gtime, task_gtime(tsk)); +		sig->utime = cputime_add(sig->utime, tsk->utime); +		sig->stime = cputime_add(sig->stime, tsk->stime); +		sig->gtime = cputime_add(sig->gtime, tsk->gtime);  		sig->min_flt += tsk->min_flt;  		sig->maj_flt += tsk->maj_flt;  		sig->nvcsw += tsk->nvcsw; @@ -932,7 +933,7 @@ NORET_TYPE void do_exit(long code)  	 * an exiting task cleaning up the robust pi futexes.  	 */  	smp_mb(); -	spin_unlock_wait(&tsk->pi_lock); +	raw_spin_unlock_wait(&tsk->pi_lock);  	if (unlikely(in_atomic()))  		printk(KERN_INFO "note: %s[%d] exited with preempt_count %d\n", @@ -970,7 +971,7 @@ NORET_TYPE void do_exit(long code)  	exit_thread();  	cgroup_exit(tsk, 1); -	if (group_dead && tsk->signal->leader) +	if (group_dead)  		disassociate_ctty(1);  	module_put(task_thread_info(tsk)->exec_domain->module); @@ -978,6 +979,10 @@ NORET_TYPE void do_exit(long code)  	proc_exit_connector(tsk);  	/* +	 * FIXME: do that only when needed, using sched_exit tracepoint +	 */ +	flush_ptrace_hw_breakpoint(tsk); +	/*  	 * Flush inherited counters to the parent - before the parent  	 * gets woken up by child-exit notifications.  	 */ @@ -1004,7 +1009,7 @@ NORET_TYPE void do_exit(long code)  	tsk->flags |= PF_EXITPIDONE;  	if (tsk->io_context) -		exit_io_context(); +		exit_io_context(tsk);  	if (tsk->splice_pipe)  		__free_pipe_info(tsk->splice_pipe); @@ -1205,6 +1210,7 @@ static int wait_task_zombie(struct wait_opts *wo, struct task_struct *p)  		struct signal_struct *psig;  		struct signal_struct *sig;  		unsigned long maxrss; +		cputime_t tgutime, tgstime;  		/*  		 * The resource counters for the group leader are in its @@ -1220,20 +1226,23 @@ static int wait_task_zombie(struct wait_opts *wo, struct task_struct *p)  		 * need to protect the access to parent->signal fields,  		 * as other threads in the parent group can be right  		 * here reaping other children at the same time. +		 * +		 * We use thread_group_times() to get times for the thread +		 * group, which consolidates times for all threads in the +		 * group including the group leader.  		 */ +		thread_group_times(p, &tgutime, &tgstime);  		spin_lock_irq(&p->real_parent->sighand->siglock);  		psig = p->real_parent->signal;  		sig = p->signal;  		psig->cutime =  			cputime_add(psig->cutime, -			cputime_add(p->utime, -			cputime_add(sig->utime, -				    sig->cutime))); +			cputime_add(tgutime, +				    sig->cutime));  		psig->cstime =  			cputime_add(psig->cstime, -			cputime_add(p->stime, -			cputime_add(sig->stime, -				    sig->cstime))); +			cputime_add(tgstime, +				    sig->cstime));  		psig->cgtime =  			cputime_add(psig->cgtime,  			cputime_add(p->gtime, |