mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-18 22:14:16 +00:00 
			
		
		
		
	 b4d0d230cc
			
		
	
	
		b4d0d230cc
		
	
	
	
	
		
			
			Based on 1 normalized pattern(s): 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 extracted by the scancode license scanner the SPDX license identifier GPL-2.0-or-later has been chosen to replace the boilerplate/reference in 114 file(s). Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Reviewed-by: Allison Randal <allison@lohutok.net> Reviewed-by: Kate Stewart <kstewart@linuxfoundation.org> Cc: linux-spdx@vger.kernel.org Link: https://lkml.kernel.org/r/20190520170857.552531963@linutronix.de Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
		
			
				
	
	
		
			54 lines
		
	
	
	
		
			1.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			54 lines
		
	
	
	
		
			1.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| /* 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
 | |
|  */
 | |
| #include <linux/sched/signal.h>
 | |
| #include <linux/sched/task.h>
 | |
| #include <linux/sched/mm.h>
 | |
| 
 | |
| /*
 | |
|  * Returns true if the task does not share ->mm with another thread/process.
 | |
|  */
 | |
| bool current_is_single_threaded(void)
 | |
| {
 | |
| 	struct task_struct *task = current;
 | |
| 	struct mm_struct *mm = task->mm;
 | |
| 	struct task_struct *p, *t;
 | |
| 	bool ret;
 | |
| 
 | |
| 	if (atomic_read(&task->signal->live) != 1)
 | |
| 		return false;
 | |
| 
 | |
| 	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;
 | |
| 
 | |
| 		for_each_thread(p, t) {
 | |
| 			if (unlikely(t->mm == mm))
 | |
| 				goto found;
 | |
| 			if (likely(t->mm))
 | |
| 				break;
 | |
| 			/*
 | |
| 			 * t->mm == NULL. Make sure next_thread/next_task
 | |
| 			 * will see other CLONE_VM tasks which might be
 | |
| 			 * forked before exiting.
 | |
| 			 */
 | |
| 			smp_rmb();
 | |
| 		}
 | |
| 	}
 | |
| 	ret = true;
 | |
| found:
 | |
| 	rcu_read_unlock();
 | |
| 
 | |
| 	return ret;
 | |
| }
 |