diff options
Diffstat (limited to 'kernel/fork.c')
| -rw-r--r-- | kernel/fork.c | 56 | 
1 files changed, 39 insertions, 17 deletions
| diff --git a/kernel/fork.c b/kernel/fork.c index d277e83ed3e0..103d78fd8f75 100644 --- a/kernel/fork.c +++ b/kernel/fork.c @@ -699,6 +699,26 @@ void __mmdrop(struct mm_struct *mm)  }  EXPORT_SYMBOL_GPL(__mmdrop); +static inline void __mmput(struct mm_struct *mm) +{ +	VM_BUG_ON(atomic_read(&mm->mm_users)); + +	uprobe_clear_state(mm); +	exit_aio(mm); +	ksm_exit(mm); +	khugepaged_exit(mm); /* must run before exit_mmap */ +	exit_mmap(mm); +	set_mm_exe_file(mm, NULL); +	if (!list_empty(&mm->mmlist)) { +		spin_lock(&mmlist_lock); +		list_del(&mm->mmlist); +		spin_unlock(&mmlist_lock); +	} +	if (mm->binfmt) +		module_put(mm->binfmt->module); +	mmdrop(mm); +} +  /*   * Decrement the use count and release all resources for an mm.   */ @@ -706,24 +726,24 @@ void mmput(struct mm_struct *mm)  {  	might_sleep(); +	if (atomic_dec_and_test(&mm->mm_users)) +		__mmput(mm); +} +EXPORT_SYMBOL_GPL(mmput); + +static void mmput_async_fn(struct work_struct *work) +{ +	struct mm_struct *mm = container_of(work, struct mm_struct, async_put_work); +	__mmput(mm); +} + +void mmput_async(struct mm_struct *mm) +{  	if (atomic_dec_and_test(&mm->mm_users)) { -		uprobe_clear_state(mm); -		exit_aio(mm); -		ksm_exit(mm); -		khugepaged_exit(mm); /* must run before exit_mmap */ -		exit_mmap(mm); -		set_mm_exe_file(mm, NULL); -		if (!list_empty(&mm->mmlist)) { -			spin_lock(&mmlist_lock); -			list_del(&mm->mmlist); -			spin_unlock(&mmlist_lock); -		} -		if (mm->binfmt) -			module_put(mm->binfmt->module); -		mmdrop(mm); +		INIT_WORK(&mm->async_put_work, mmput_async_fn); +		schedule_work(&mm->async_put_work);  	}  } -EXPORT_SYMBOL_GPL(mmput);  /**   * set_mm_exe_file - change a reference to the mm's executable file @@ -1470,7 +1490,7 @@ static struct task_struct *copy_process(unsigned long clone_flags,  		pid = alloc_pid(p->nsproxy->pid_ns_for_children);  		if (IS_ERR(pid)) {  			retval = PTR_ERR(pid); -			goto bad_fork_cleanup_io; +			goto bad_fork_cleanup_thread;  		}  	} @@ -1494,7 +1514,7 @@ static struct task_struct *copy_process(unsigned long clone_flags,  	 * sigaltstack should be cleared when sharing the same VM  	 */  	if ((clone_flags & (CLONE_VM|CLONE_VFORK)) == CLONE_VM) -		p->sas_ss_sp = p->sas_ss_size = 0; +		sas_ss_reset(p);  	/*  	 * Syscall tracing and stepping should be turned off in the @@ -1632,6 +1652,8 @@ bad_fork_cancel_cgroup:  bad_fork_free_pid:  	if (pid != &init_struct_pid)  		free_pid(pid); +bad_fork_cleanup_thread: +	exit_thread(p);  bad_fork_cleanup_io:  	if (p->io_context)  		exit_io_context(p); | 
