首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

fork系统调用分析

除了这个函数,新进程的诞生还可以分别通过vfork()和clone() fork、vfork和clone三个API函数均由glibc库提供,它们分别在C库中封装了与其同名的系统调用fork() 这几个函数调用对应不同场景...clone函数创建子进程时灵活度比较大,因为它可以通过传递不同的参数来选择性的复制父进程的资源 系统调用fork、vfork和clone在内核中对应的服务例程分别为sys_fork(),sys_vfork...通过分析调用过程如下,其中我分析的是最新版4.X Linux源码,在i386体系结构中,采取0x80中断调用syscall: image.png 从图中可以看到do_fork()和copy_process...如果使用vfork系统调用来创建子进程,那么必然是子进程先执行。原因就是此处vfork完成量所起到的作用:当子进程调用exec函数或退出时就向父进程发出信号。此时,父进程才会被唤醒;否则一直等待。...这也就是为什么使用fork系统调用时父进程会返回子进程pid的原因。

96530
您找到你想要的搜索结果了吗?
是的
没有找到

Linux系统编程-进程创建(fork)、外部程序调用(exec)

1. fork函数介绍 在linuxfork函数是非常重要的函数,它可以从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。...fork函数的返回值如下: 1、在父进程中,fork返回新创建的子进程的PID号。 2、在子进程中,fork返回0; 3、如果出现错误,fork返回一个负值。...fork函数创建子进程的过程: 使用fork函数得到的子进程是父进程的一个复制品,它从父进程继承了进程的所有资源,相当于就是父进程的一个副本。...#include pid_t fork(void); 制作分身 函数功能: 创建新的子进程. 子进程是父的进程一个副本....思路: 父进程扫描目录,得到目录下的文件名称,在传递给子进程,子进程调用eog命令实现图片显示,父进程里2秒钟之后就杀死子进程,再读取目录下下一个文件,再传递给子进程……… #include <stdio.h

2.8K20

linux0.11系统调用过程和fork源码解析

) \ { \ long __res; \ __asm__ volatile ("int $0x80" \ // 输如输出都是eax,输入是系统调用函数在系统调用表的序号 : "=a"...下面是操作系统执行系统调用前,在内核栈里保存的寄存器,这个压入的寄存器和iret中断返回指令出栈的寄存器是对应的。其中ip指向的是调用系统调用返回后的下一句代码。 ?...,%eax ja bad_sys_call // 寄存器压栈,保存现场和用户传递的参数 push %ds push %es push %fs // 执行系统调用的函数时用户传入的三个参数...,每个函数地址4个字节 call _sys_call_table(,%eax,4) // 系统调用的返回值,压栈保存,因为下面需要用eax pushl %eax // 把当前进程的...call _sys_call_table(,%eax,4) pushl %eax 最后,通过iret中断返回指令弹出五个寄存器,回到系统调用前的ip处执行。父进程返回值是eax,即子进程id。

1.4K40

一文看懂 | fork 系统调用

前言 Unix标准的复制进程的系统调用fork(即分叉),但是Linux,BSD等操作系统并不止实现这一个,确切的说linux实现了三个,fork,vfork,clone(确切说vfork创造出来的是轻量级进程...,而且由vfork()创建的子进程将先于父进程运行 clone Linux上创建线程一般使用的是pthread库 实际上linux也给我们提供了创建线程的系统调用,就是clone fork, vfork...在进行系统调用之前,系统调用的参数被写入CPU的寄存器,而在实际调用系统服务例程之前,内核将CPU寄存器的内容拷贝到内核堆栈中,实现参数的传递。...负责进程的复制 即不同的体系结构可能需要采用不同的方式或者寄存器来存储函数调用的参数, 因此linux在设计系统调用的时候, 将其划分成体系结构相关的层次和体系结构无关的层次, 前者复杂提取出依赖与体系结构的特定的参数...fork, vfork, clone系统调用的实现 关于do_fork和_do_frok linux2.5.32以后, 添加了TLS(Thread Local Storage)机制, clone的标识CLONE_SETTLS

95520

一文看懂 | fork 系统调用

前言 Unix标准的复制进程的系统调用fork(即分叉),但是Linux,BSD等操作系统并不止实现这一个,确切的说linux实现了三个,fork,vfork,clone(确切说vfork创造出来的是轻量级进程...,而且由vfork()创建的子进程将先于父进程运行 clone Linux上创建线程一般使用的是pthread库 实际上linux也给我们提供了创建线程的系统调用,就是clone fork, vfork...在进行系统调用之前,系统调用的参数被写入CPU的寄存器,而在实际调用系统服务例程之前,内核将CPU寄存器的内容拷贝到内核堆栈中,实现参数的传递。...负责进程的复制 即不同的体系结构可能需要采用不同的方式或者寄存器来存储函数调用的参数, 因此linux在设计系统调用的时候, 将其划分成体系结构相关的层次和体系结构无关的层次, 前者复杂提取出依赖与体系结构的特定的参数...fork, vfork, clone系统调用的实现 关于do_fork和_do_frok linux2.5.32以后, 添加了TLS(Thread Local Storage)机制, clone的标识CLONE_SETTLS

2.2K30

Linux系统编程fork详解

使用fork函数会创建一个和父进程相同的子进程。...在调用fork函数后,会先为子进程申请一个PID号,然后申请一个PCB结构,然后将父进程的PCB结构复制过来,对于父进程的虚拟空间内的内容用到了读时共享,写时复制的机制(下面会讲)。        ...#include        #include        pid_t fork(void);        对于fork函数没有参数,会返回一个...最开始的linux的创建子进程的实现方法是在子进程创建时就直接将父进程的所有内容复制到子进程中,但是这一操作会造成不必要的资源和时间的消耗。所以就有了读时共享,写时复制的机制。...系统会为子进程创建其自己的4G的虚拟内存,而虚拟内存又分为内核内存和用户内存,大小比为1:3。子进程的虚拟地址映射了父进程的虚拟地址所指向的物理内存,所以父子进程实际上共享了同一块物理内存。

2.2K30

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

文章目录 一、fork 系统调用源码 二、vfork 系统调用源码 三、clone 系统调用源码 四、_do_fork 函数源码 五、do_fork 函数源码 Linux 进程相关 " 系统调用 " 对应的源码在...linux-5.6.18\kernel\fork.c 源码中 , 下面开始对该源码的相关 " 系统调用 " 进行分析 ; 一、fork 系统调用源码 ---- fork() 系统调用函数 , 最终返回的是...mode */ return -EINVAL; #endif } #endif 二、vfork 系统调用源码 ---- vfork() 系统调用函数 , 最终返回的是 _do_fork() 函数执行结果...(&args); } #endif 三、clone 系统调用源码 ---- clone() 系统调用函数 , 最终返回的是 _do_fork() 函数执行结果 ; #ifdef __ARCH_WANT_SYS_CLONE...---- 在 _do_fork() 函数中 , 调用了 copy_process() 函数 ; /* * Ok, this is the main fork-routine

4.6K10

Linux系统编程】通过系统调用获取进程标识符 及 创建子进程(fork

可以的: 我们可以通过一个系统调用来获取,这个系统调用叫做getpid 我们可以通过man手册学习一下 getpid没有参数,直接调用即可获取(返回)当前进程的pid,返回值是pid_t类型...通过系统调用创建进程-fork初识 经过之前的学习我们知道我们可以通过运行一个程序使之变成进程,那有没有其它产生新进程的方法呢? 有的,我们可以通过系统调用来创建进程。...这个系统调用叫做fork 那我们先来学习一个fork怎么用: man fork 它在当前进程的基础上创建一个新的子进程 3.1 批量化注释 那我们再重新写一段代码 把之前的注释掉,那这里再教大家一下如何批量化注释...失败的话,-1在父进程中返回,不会创建任何子进程,并且正确设置了errno(C语言中一个用于表示错误码的全局变量,Linux内核是C语言写的)。 也就是说fork成功的话,返回值会有两个。...那对于fork来说: 它是一个系统调用,那其实就是操作系统提供的一个函数嘛。 那在fork最后将要return的时候,那它的主体功能即创建子进程当然已经完成了。

22310

LinuxLinux系统调用

Linux系统调用 前言 操作系统——管理计算机硬件与软件资源的软件,是用户和系统交互的操作接口,为它上面运行的程序提供服务。...操作系统内核——操作系统的内核,负责管理系统的进程、内存、设备驱动程序、文件和网络系统。一个内核不是一套完整的操作系统。例如LinuxLinux操作系统——基于Linux内核的操作系统。...Linux的运行空间: Linux的运行空间:内核空间+用户空间 ---- 内核空间——存放的是整个内核代码和所有内核模块,以及内核所维护的数据。 用户空间——用户程序的代码和数据。...---- 系统调用的实现 通过软件中断实现。 **软件中断:**它是通过软件指令触发的中断。Linux系统内核响应软件中断,从用户态切换到内核态,执行相应的系统调用。...调用相应的执行程序来处理系统调用。 从系统调用返回。 系统调用号: 每个系统调用被赋予一个系统调用号,与具体的系统调用相关联。

27.8K10

Linux内核分析》之分析fork函数对应的系统调用处理过程

-s -S 3、打开gdb进行远程调试 ①相关配置 gdb file linux-3.18.6/vmlinux target remote:1234 ②设置断点 b sys_clone b do_fork...代码及分析 tast_struct xref: /linux-3.18.6/include/linux/sched.h struct task_struct { volatile long state...; //说明了该进程是否可以执行,还是可中断等信息 unsigned long flags; //进程号,在调用fork()时给出 int sigpending...cpus_allowed; struct list_head run_list; //指向运行队列的指针 unsigned long sleep_time; //进程的睡眠时间 //用于将系统中所有的进程连成一个双向循环链表...n"); } }  创建一个新进程在内核中的执行过程 fork、vfork和clone三个系统调用都可以创建一个新进程,而且都是通过调用do_fork来实现进程的创建; Linux通过复制父进程来创建一个新进程

1K10

Linux 系统调用

Linux 中,系统调用是用户空间访问内核的唯一手段﹔除异常和陷入外,它们是内核唯一的合法入口。实际上,其他的像设备文件和/proc之类的方式,最终也还是要通过系统调用进行访问的。...而有趣的是,Linux 提供的系统调用却比大部分操作系统都少得多。 要访问系统调用(在 Linux 中常称作 syscall),通常通过C库中定义的函数调用来进行。...系统调用在出现错误的时候C库会把错误码写人errno全局变量。通过调用perror()库函数,可以把该变量翻译成用户可以理解的错误字符串。 在 Linux 中,每个系统调用被赋予一个系统调用号。...这样,通过这个独一无二的号就可以关联系统调用。当用户空间的进程执行一个系统调用的时候,这个系统调用号就用来指明到底是要执行哪个系统调用;进程不会提及系统调用的名称。...假设系统调用在内核空间定义为 sys_ioctl,那么该系统调用的用户空间接口为 ioctl Linux kernel-5.18.8 有 440 个系统调用,这些系统调用讲究通用性,一旦固定,很少修改,

9.8K20

Linux系统调用原理

一、什么是系统调用 系统调用 跟用户自定义函数一样也是一个函数,不同的是 系统调用 运行在内核态,而用户自定义函数运行在用户态。...系统调用Linux 内核提供的一段代码(函数),其实现了一些特定的功能,用户可以通过 int 0x80 中断(x86 CPU)或者 syscall 指令(x64 CPU)来调用 系统调用。...二、进入系统调用 本文主要介绍的是 x86 CPU 进入系统调用的方式 Linux 提供了 int 0x80 中断来让用户程序进入 系统调用,我们来看看 Linux 对 int 0x80 中断的处理初始化过程...三、系统调用实现 当用户要调用 系统调用 时,需要通过向 eax 寄存器写入要调用系统调用 编号。...而 Linux 进入中断处理程序时,会把这些寄存器的值保存到内核栈中,这样 系统调用 就能通过内核栈来获取到参数。

4.2K30

Linux多进程(fork)

进程概念: 一个进程是一次程序执行的过程,它和程序不同,程序是静态的,它是一些保存在磁盘上可执行的代码和数据的集合,而进程是一个动态概念,也是操作系统分配资源的最小单位 fork和exec是两个重要的系统调用...,fork的作用是根据现有的进程复制出一个新的进程,原来的进程称为父进程,新的进程成为子进程, 系统中运行着很多进程,这些进程都是从开始的一个进程一个一个复制出来的。...#include #include pid_t fork(void); fork调用失败返回-1,调用成功在父子进程中的返回值不一样,子进程中返回0,父进程中返回的数值大于...0 #include //基本系统数据类型的头文件 #include //包含了许多UNIX系统服务的函数原型 getpid函数 #include //输入输出函数 int main(void){ pid_t pid; char * message; int n; pid = fork(); if(pid

2K30

Linuxfork使用

Linuxfork使用 fork函数可以算是Linux里有点不好明白的函数了,调用一次,返回两次,虽然在平时的写法中,有基本固定的写法,但是有时候看起来还是有些让人头疼的。...失败的两个主要原因是: 1)当前的进程数已经达到了系统规定的上限,这时 errno 的值被设置为 EAGAIN。 2)系统内存不足,这时 errno 的值被设置为 ENOMEM。...因此,使用 fork()函数的代价是很大的。 ? 日常使用fork 简单来说, 一个进程调用 fork() 函数后,系统先给新的进程分配资源,例如存储数据和代码的空间。...实际上,更准确来说,Linuxfork() 使用是通过写时拷贝 (copy- on-write) 实现。写时拷贝是一种可以推迟甚至避免拷贝数据的技术。...其他子进程 cout<<"这是父进程: "<<getpid()<<endl; } } 正确的使用Linux中的用fork()由一个父进程创建同时多个子进程 的格式如下: int

3.6K41

linux fork函数浅析

对子进程来说,之所以fork返回0给它,是由于它随时能够调用getpid()来获取自己的pid;也能够调用getppid()来获取父进程的id。...fork之后,操作系统会复制一个与父进程全然同样的子进程,虽说是父子关系,可是在操作系统看来,他们更像兄弟关系,这2个进程共享代码空间,可是数据空间是互相独立的,子进程数据空间中的内容是父进程的完整拷贝...此时程序寄存器pc,在父、子进程的上下文中都声称,这个进程眼下运行到fork调用即将返回(此时子进程不占有CPU,子进程的pc不是真正保存在寄存器中,而是作为进程上下文保存在进程表中的相应表项内)。...(如果父进程一直占领CPU,实际情况非常可能不一样)父进程继续运行,操作系统fork的实现,使这个调用在父进程中返回刚刚创建的子进程的pid(一个正整数),所以以下的swtich语句中运行了default...所以输出I am the parent process… 子进程在之后的某个时候得到调度,它的上下文被换入,占领 CPU,操作系统fork的实现,使得子进程中fork调用返回0,所以在这个进程(注意这不是父进程了哦

1.3K20
领券