Linux 内核进程管理 ( 进程相关系统调用源码分析 | fork() 源码 | vfork() 源码 | clone() 源码 | _do_fork() 源码 | do_fork() 源码 )

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux 内核进程管理 ( 进程相关系统调用源码分析 | fork() 源码 | vfork() 源码 | clone() 源码 | _do_fork() 源码 | do_fork() 源码 )相关的知识,希望对你有一定的参考价值。

文章目录


Linux 进程相关 " 系统调用 " 对应的源码在 linux-5.6.18\\kernel\\fork.c 源码中 , 下面开始对该源码的相关 " 系统调用 " 进行分析 ;





一、fork 系统调用源码



fork() 系统调用函数 , 最终返回的是 _do_fork() 函数执行结果 ;

#ifdef __ARCH_WANT_SYS_FORK
SYSCALL_DEFINE0(fork)

#ifdef CONFIG_MMU
	struct kernel_clone_args args = 
		.exit_signal = SIGCHLD,
	;

	return _do_fork(&args);
#else
	/* can not support in nommu mode */
	return -EINVAL;
#endif

#endif





二、vfork 系统调用源码



vfork() 系统调用函数 , 最终返回的是 _do_fork() 函数执行结果 ;

#ifdef __ARCH_WANT_SYS_VFORK
SYSCALL_DEFINE0(vfork)

	struct kernel_clone_args args = 
		.flags		= CLONE_VFORK | CLONE_VM,
		.exit_signal	= SIGCHLD,
	;

	return _do_fork(&args);

#endif





三、clone 系统调用源码



clone() 系统调用函数 , 最终返回的是 _do_fork() 函数执行结果 ;

#ifdef __ARCH_WANT_SYS_CLONE
#ifdef CONFIG_CLONE_BACKWARDS
SYSCALL_DEFINE5(clone, unsigned long, clone_flags, unsigned long, newsp,
		 int __user *, parent_tidptr,
		 unsigned long, tls,
		 int __user *, child_tidptr)
#elif defined(CONFIG_CLONE_BACKWARDS2)
SYSCALL_DEFINE5(clone, unsigned long, newsp, unsigned long, clone_flags,
		 int __user *, parent_tidptr,
		 int __user *, child_tidptr,
		 unsigned long, tls)
#elif defined(CONFIG_CLONE_BACKWARDS3)
SYSCALL_DEFINE6(clone, unsigned long, clone_flags, unsigned long, newsp,
		int, stack_size,
		int __user *, parent_tidptr,
		int __user *, child_tidptr,
		unsigned long, tls)
#else
SYSCALL_DEFINE5(clone, unsigned long, clone_flags, unsigned long, newsp,
		 int __user *, parent_tidptr,
		 int __user *, child_tidptr,
		 unsigned long, tls)
#endif

	struct kernel_clone_args args = 
		.flags		= (lower_32_bits(clone_flags) & ~CSIGNAL),
		.pidfd		= parent_tidptr,
		.child_tid	= child_tidptr,
		.parent_tid	= parent_tidptr,
		.exit_signal	= (lower_32_bits(clone_flags) & CSIGNAL),
		.stack		= newsp,
		.tls		= tls,
	;

	if (!legacy_clone_args_valid(&args))
		return -EINVAL;

	return _do_fork(&args);

#endif





四、_do_fork 函数源码



_do_fork() 函数中 , 调用了 copy_process() 函数 ;

/*
 *  Ok, this is the main fork-routine.
 *
 * It copies the process, and if successful kick-starts
 * it and waits for it to finish using the VM if required.
 *
 * args->exit_signal is expected to be checked for sanity by the caller.
 */
long _do_fork(struct kernel_clone_args *args)

	u64 clone_flags = args->flags;
	struct completion vfork;
	struct pid *pid;
	struct task_struct *p;
	int trace = 0;
	long nr;

	/*
	 * Determine whether and which event to report to ptracer.  When
	 * called from kernel_thread or CLONE_UNTRACED is explicitly
	 * requested, no event is reported; otherwise, report if the event
	 * for the type of forking is enabled.
	 */
	if (!(clone_flags & CLONE_UNTRACED)) 
		if (clone_flags & CLONE_VFORK)
			trace = PTRACE_EVENT_VFORK;
		else if (args->exit_signal != SIGCHLD)
			trace = PTRACE_EVENT_CLONE;
		else
			trace = PTRACE_EVENT_FORK;

		if (likely(!ptrace_event_enabled(current, trace)))
			trace = 0;
	

	p = copy_process(NULL, trace, NUMA_NO_NODE, args);
	add_latent_entropy();

	// ...
	return nr;





五、do_fork 函数源码



do_fork() 函数有 5 5 5 个参数 ,

unsigned long clone_flags 参数表示 创建进程 的 标志位 集合 ;

unsigned long stack_start 参数表示 用户空间 栈 起始地址 ;

unsigned long stack_size 参数表示 用户空间 栈 大小 , 通常为 0 0 0 ;

int __user *parent_tidptr 参数表示 指向 用户空间 地址的指针 , 该指针指向 父进程 的进程号 ;

int __user *child_tidptr 参数表示 指向 用户空间 地址的指针 , 该指针指向 子进程 的进程号 ;


#ifndef CONFIG_HAVE_COPY_THREAD_TLS
/* For compatibility with architectures that call do_fork directly rather than
 * using the syscall entry points below. */
long do_fork(unsigned long clone_flags,
	      unsigned long stack_start,
	      unsigned long stack_size,
	      int __user *parent_tidptr,
	      int __user *child_tidptr)

	struct kernel_clone_args args = 
		.flags		= (lower_32_bits(clone_flags) & ~CSIGNAL),
		.pidfd		= parent_tidptr,
		.child_tid	= child_tidptr,
		.parent_tid	= parent_tidptr,
		.exit_signal	= (lower_32_bits(clone_flags) & CSIGNAL),
		.stack		= stack_start,
		.stack_size	= stack_size,
	;

	if (!legacy_clone_args_valid(&args))
		return -EINVAL;

	return _do_fork(&args);

#endif

以上是关于Linux 内核进程管理 ( 进程相关系统调用源码分析 | fork() 源码 | vfork() 源码 | clone() 源码 | _do_fork() 源码 | do_fork() 源码 )的主要内容,如果未能解决你的问题,请参考以下文章

Linux(内核剖析):12---进程调度之与调度相关的系统调用

Linux 内核进程管理 ( 进程状态 | 进程创建 | 进程终止 | 调用 exit 系统调用函数主动退出 | main 函数返回自动退出 | kill 杀死进程 | 执行异常退出 )

Linux 内核 内存管理内存管理系统调用 ⑤ ( 代码示例 | 多进程共享 mmap 内存映射示例 )

《Linux内核设计与实现》读书笔记- 进程的调度

Linux系统调用及用户编程接口(API)

Linux 内核Linux 内核特性 ( 组织形式 | 进程调度 | 内核线程 | 多平台虚拟内存管理 | 虚拟文件系统 | 内核模块机制 | 定制系统调用 | 网络模块架构 )