| 
									
										
										
										
											2008-11-14 10:39:21 +11:00
										 |  |  | /* Function to determine if a thread group is single threaded or not
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2008 Red Hat, Inc. All Rights Reserved. | 
					
						
							|  |  |  |  * Written by David Howells (dhowells@redhat.com) | 
					
						
							|  |  |  |  * - Derived from security/selinux/hooks.c | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public Licence | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version | 
					
						
							|  |  |  |  * 2 of the Licence, or (at your option) any later version. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/sched.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-17 09:09:36 +10:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Returns true if the task does not share ->mm with another thread/process. | 
					
						
							| 
									
										
										
										
											2008-11-14 10:39:21 +11:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-07-10 03:48:23 +02:00
										 |  |  | bool current_is_single_threaded(void) | 
					
						
							| 
									
										
										
										
											2008-11-14 10:39:21 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-10 03:48:23 +02:00
										 |  |  | 	struct task_struct *task = current; | 
					
						
							| 
									
										
										
										
											2009-07-17 09:09:36 +10:00
										 |  |  | 	struct mm_struct *mm = task->mm; | 
					
						
							|  |  |  | 	struct task_struct *p, *t; | 
					
						
							|  |  |  | 	bool ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (atomic_read(&task->signal->live) != 1) | 
					
						
							|  |  |  | 		return false; | 
					
						
							| 
									
										
										
										
											2008-11-14 10:39:21 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-17 09:09:36 +10:00
										 |  |  | 	if (atomic_read(&mm->mm_users) == 1) | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = false; | 
					
						
							|  |  |  | 	rcu_read_lock(); | 
					
						
							|  |  |  | 	for_each_process(p) { | 
					
						
							|  |  |  | 		if (unlikely(p->flags & PF_KTHREAD)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		if (unlikely(p == task->group_leader)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t = p; | 
					
						
							|  |  |  | 		do { | 
					
						
							|  |  |  | 			if (unlikely(t->mm == mm)) | 
					
						
							|  |  |  | 				goto found; | 
					
						
							|  |  |  | 			if (likely(t->mm)) | 
					
						
							|  |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2009-07-09 23:28:49 +02:00
										 |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * t->mm == NULL. Make sure next_thread/next_task | 
					
						
							|  |  |  | 			 * will see other CLONE_VM tasks which might be | 
					
						
							|  |  |  | 			 * forked before exiting. | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			smp_rmb(); | 
					
						
							| 
									
										
										
										
											2009-07-17 09:09:36 +10:00
										 |  |  | 		} while_each_thread(p, t); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ret = true; | 
					
						
							|  |  |  | found: | 
					
						
							|  |  |  | 	rcu_read_unlock(); | 
					
						
							| 
									
										
										
										
											2008-11-14 10:39:21 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-17 09:09:36 +10:00
										 |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2008-11-14 10:39:21 +11:00
										 |  |  | } |