linux进程管理(C语言)

理解进程控制的原理对于理解和修改fio project非常的重要。"fio
is an I/O tool meant to be used both for benchmark and stress/hardware verification."

进程

unix提供了大量的从c程序中操作系统的系统调用(别的语言应该也是有的吧)。

创建进程

每一个进程都有一个正数的id,叫做pid。getpid函数返回调用进程的pid,getppid函数返回它的父进程的pid。

fork

exit

父进程和子进程是并发运行的独立进程。内核能够以任意方式交替执行他们的逻辑控制流中的指令。

如果能够在fork函数在父进程和子进程中返回后立即暂停这两个进程,我们会看到每个进程的地址空间都是相同的。(每个进程有相同的用户栈,相同的本地变量值,相同的堆,相同的全局变量值以及相同的代码。然而他们都有自己独立的地址空间)

关于这个,fio这个程序利用到它的地方就是


while (todo) {

struct thread_data *map[REAL_MAX_JOBS];

struct timeval this_start;

int this_jobs = 0, left;

for_each_td(td, i) {

if (td->runstate != TD_NOT_CREATED)

continue;

if (td->o.start_delay) {

spent = utime_since_genesis();

if (td->o.start_delay > spent)

continue;

}

if (td->o.stonewall && (nr_started || nr_running)) {

dprint(FD_PROCESS, "%s: stonewall wait\n",

td->o.name);

break;

}

init_disk_util(td);

td->rusage_sem = fio_mutex_init(FIO_MUTEX_LOCKED);

td->update_rusage = 0;

/*

* Set state to created. Thread will transition

* to TD_INITIALIZED when it‘s done setting up.

*/

td_set_runstate(td, TD_CREATED);

map[this_jobs++] = td;

nr_started++;

...

if (td->o.use_thread) {

int ret;

dprint(FD_PROCESS, "will pthread_create\n");

ret = pthread_create(&td->thread, NULL,

thread_main, td);

if (ret) {

log_err("pthread_create: %s\n",

strerror(ret));

nr_started--;

break;

}

ret = pthread_detach(td->thread);

if (ret)

log_err("pthread_detach: %s",

strerror(ret));

} else {

pid_t pid;

dprint(FD_PROCESS, "will fork\n");

pid = fork();

if (!pid) {

int ret = fork_main(shm_id, i);

_exit(ret);

} else if (i == fio_debug_jobno)

*fio_debug_jobp = pid;

}

这就相当于这么编程:


#include <stdio.h>

int main()

{

int i=1;

int pid;

while((i--)>=0){

pid=fork();

if(pid==0){

i--;

printf("in the child process.\n");

}

else

printf("in the parent process.\n");

}

}

编译并执行上面那段程序的结果:


[email protected] ~]# ./a.out

in the parent process.

in the child process.

in the parent process.

in the child process.

一共创建了两个进程,只不过在Fio中的子进程的执行是由另外一个函数fork_main和thread_main来决定的。

note:thread main这个函数做了许多事情,会再分析。

回收子进程

当一个进程由于某种原因终止时,内核并不是立即把它从系统中清除。相反,进程被保持在一种已终止的状态中,直到被它的父进程reap。当父进程回收已经终止的子进程时,内核将子进程的退出状态(这是什么样的退出状态呢,留个心)传递给父进程,然后抛弃已终止的进程,从此时开始,该进程就不存在了。

一个终止了但未被回收的进程称为僵尸zombie。

如果父进程没有回收他的zombie就终止了,那么内核就会安排init进程来回收他们。长时间运行的程序,比如shell或者服务器,总是应该回收他们的zombie(总是在消耗系统的存储器资源)。

一个进程可以通过调用waitpid函数来等待它的子进程终止或者停止。

函数原型:


 #include<sys/types.h>

  #include<sys/wait.h>

  定义函数 pid_t waitpid(pid_t pid,int * status,int options);

waitpid - 函数说明

  waitpid()会暂时停止目前进程的执行,直到有信号来到或子进程

  结束。如果在调用 waitpid()时子进程已经结束,则 waitpid()会立即

  返回子进程结束状态值。 子进程的结束状态值会由参数 status 返回,

  而子进程的进程识别码也会一起返回。如果不在意结束状态值,则

  参数 status 可以设成 NULL。参数 pid 为欲等待的子进程识别码,

  其他数值意义如下:

  pid<-1 等待进程组识别码为 pid 绝对值的任何子进程。

  pid=-1 等待任何子进程,相当于 wait()。

  pid=0 等待进程组识别码与目前进程相同的任何子进程。

  pid>0 等待任何子进程识别码为 pid 的子进程。

  参数options提供了一些额外的选项来控制waitpid,参数 option 可以为 0 或可以用"|"运算符把它们连接起来使用,比如:

  ret=waitpid(-1,NULL,WNOHANG | WUNTRACED);

  如果我们不想使用它们,也可以把options设为0,如:

  ret=waitpid(-1,NULL,0);

  WNOHANG 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若结束,则返回该子进程的ID。

  WUNTRACED 若子进程进入暂停状态,则马上返回,但子进程的结束状态不予以理会。WIFSTOPPED(status)宏确定返回值是否对应与一个暂停子进程。

  子进程的结束状态返回后存于 status,底下有几个宏可判别结束情况:

  WIFEXITED(status)如果若为正常结束子进程返回的状态,则为真;对于这种情况可执行WEXITSTATUS(status),取子进程传给exit或_eixt的低8位。

  WEXITSTATUS(status)取得子进程 exit()返回的结束代码,一般会先用 WIFEXITED 来判断是否正常结束才能使用此宏。

  WIFSIGNALED(status)若为异常结束子进程返回的状态,则为真;对于这种情况可执行WTERMSIG(status),取使子进程结束的信号编号。

  WTERMSIG(status) 取得子进程因信号而中止的信号代码,一般会先用 WIFSIGNALED 来判断后才使用此宏。

  WIFSTOPPED(status) 若为当前暂停子进程返回的状态,则为真;对于这种情况可执行WSTOPSIG(status),取使子进程暂停的信号编号。

  WSTOPSIG(status) 取得引发子进程暂停的信号代码,一般会先用 WIFSTOPPED 来判断后才使用此宏。

  如果执行成功则返回子进程识别码(PID) ,如果有错误发生则返回

  返回值-1。失败原因存于 errno 中。

fio使用waitpid的例子


* Run over the job map and reap the threads that have exited, if any.

*/

static void reap_threads(unsigned int *nr_running, unsigned int *t_rate,

unsigned int *m_rate)

{

...

realthreads = pending = cputhreads = 0;

for_each_td(td, i) {

int flags = 0;

/*

* ->io_ops is NULL for a thread that has closed its

* io engine

*/

if (td->io_ops && !strcmp(td->io_ops->name, "cpuio"))

cputhreads++;

else

realthreads++;

...

flags = WNOHANG;

if (td->runstate == TD_EXITED)

flags = 0;

/*

* check if someone quit or got killed in an unusual way

*/

ret = waitpid(td->pid, &status, flags);

if (ret < 0) {

if (errno == ECHILD) {

log_err("fio: pid=%d disappeared %d\n",

(int) td->pid, td->runstate);

td->sig = ECHILD;

td_set_runstate(td, TD_REAPED);

goto reaped;

}

perror("waitpid");

} else if (ret == td->pid) {

if (WIFSIGNALED(status)) {

int sig = WTERMSIG(status);

if (sig != SIGTERM && sig != SIGUSR2)

log_err("fio: pid=%d, got signal=%d\n",

(int) td->pid, sig);

td->sig = sig;

td_set_runstate(td, TD_REAPED);

goto reaped;

}

if (WIFEXITED(status)) {

if (WEXITSTATUS(status) && !td->error)

td->error = WEXITSTATUS(status);

td_set_runstate(td, TD_REAPED);

goto reaped;

}

}

...

让进程休眠

sleep函数将一个进程挂起一段指定的时间。sleep函数返回0,否则返回剩下的要休眠的秒数(是可能的,因为可能被信号中断而过早的返回)。

另外一个有用的函数是pause,该函数让调用函数休眠,直到该进程收到一个信号。

进程间的信号传递

note:由于时间关系,有空再补充。

参考文献

Caapp:深入理解计算机系统

源码fio-2.1.10

时间: 2024-10-06 23:30:07

linux进程管理(C语言)的相关文章

Linux进程管理与调度-之-目录导航【转】

转自:http://blog.csdn.net/gatieme/article/details/51456569 版权声明:本文为博主原创文章 && 转载请著名出处 @ http://blog.csdn.net/gatieme 目录(?)[-] 项目链接 进程的描述 进程的创建 进程的加载与运行 进程的退出 进程的调度 调度普通进程-完全公平调度器CFS 日期 内核版本 架构 作者 GitHub CSDN 2016-07-21 Linux-4.6 X86 & arm gatieme

Linux进程管理简谈

Linux系统进程管理简谈 进程是什么? 简单来说进程是一个正在执行的程序的一个副本,存在生命周期,有段指令和代码在不断运行. linux内核存储信息的固定格式:task struct 进程的相关信息存储在链表中 多个任务的task struct组件的链表:task list 进程的创建:父进程创建子进程(内核创建init进程,剩余一切进程有init及其子进程进程创建) 父进程创建子进程时向内核调用fork()来创建子进程并且通过调用clone()复制父进程的信息给子进程 Linux进程的优先级

linux进程管理(5)---进程消亡

一.目的 本文将讲述进程是如何消亡的.一个进程既有父进程又有子进程,因此进程消亡时,既要通知父进程,也要安排好子进程. 当前进程消亡时主要做了三件大事:释放当前进程占用的资源:为当前进程的子进程重新寻找"养父":通知当前进程的父进程,释放当前进程剩下的资源. 当前进程释放掉大多数进程资源后,只保留内核栈.structtask_struct数据结构:剩下的资源由父进程负责释放. linux调用sys_exit().sys_wait4()实现进程的消亡,代码可以在kernel/exit.c

Linux性能及调优指南(翻译)之Linux进程管理

译文如下:1.1 Linux进程管理 进程管理是操作系统的最重要的功能之一.有效率的进程管理能保证一个程序平稳而高效地运行. Linux的进程管理与UNIX的进程管理相似.它包括进程调度.中断处理.信号.进程优先级.上下文切换.进程状态.进度内存等. 在本节中,我们将描述Linux进程管理的基本原理的实现.它将更好地帮助你理解Linux内核如何处理进程及其对系统性能的影响. 1.1.1 什么是进程?一个进程是一个运行在处理器的程序的一个实例.该进程使用Linux内核能够处理的任何资源来完成它的任

Linux进程管理知识整理

Linux进程管理知识整理 1.进程有哪些状态?什么是进程的可中断等待状态?进程退出后为什么要等待调度器删除其task_struct结构?进程的退出状态有哪些? TASK_RUNNING(可运行状态) TASK_INTERRUPTIBLE(可中断等待状态) TASK_UNINTERRUPTIBLE(不可中断等待状态) TASK_STOPPED(进程被其它进程设置为暂停状态) TASK_TRACED(进程被调试器设置为暂停状态) TASK_DEAD(退出状态) 进程由于所需资源得不到满足,从而进入

linux进程管理(四)

[教程主题]:进程管理 [1]进程介绍 程序和进程 程序是为了完成某种任务而设计的软件,比如OpenOffice是程序.什么是进程呢?进程就是运行中的程序. 一个运行着的程序,可能有多个进程. 比如自学it网所用的WWW服务器是apache服务器,当管理员启动服务后,可能会有好多人来访问,也就是说许多用户来同时请求httpd服务,apache服务器将会创建有多个httpd进程来对其进行服务. 进程分类: 进程一般分为交互进程.批处理进程和守护进程三类. 值得一提的是守护进程总是活跃的,一般是后台

Linux进程管理(3):总结

7. exit与_exit的差异    为了理解这两个系统调用的差异,先来讨论文件内存缓存区的问题. 在linux中,标准输入输出(I/O)函数都是作为文件来处理.对应于打开的每个文件,在内存中都有对应的缓存,每次读取文件时,会多读一些记录到缓存中,这样在下次读文件时,就在缓存中读取:同样,在写文件时也是写在文件对应的缓存中,并不是直接写入硬盘的文件中,等满足了一定条件(如达到一定数量,遇到换行符\n或文件结束标志EOF)才将数据真正的写入文件.这样做的好处就是加快了文件读写的速度.但这样也带来

linux进程管理命令总结

linux进程管理的相关命令: 1.pstree  --以树状形式显示当前系统进程,和命令tree相似. 2.ps命令 命令格式:ps [options] 常用组合:aux: u:以用户为中心组织的进程状态信息显示 a:与终端相关的进程 x:与终端无关的进程 #ps aux USER : 用户名 PID : 进程号 %CPU :占用的cpu百分比 %MEM  :内存的使用率 VSZ :  virtual memory size ,虚拟内存集 RSS : resident size ,常驻内存集

mschedule 简单linux进程管理(树莓派)

树莓派是神奇的机器,CPU和内存都少的可怜,但体积小功耗低,在上面搞些动搞些西其实也挺有意思,挺好玩的.装的是pidara,基本服务没有精简多少,先cat一下CPU和RAM. [[email protected] ~]$ cat /proc/cpuinfo Processor : ARMv6-compatible processor rev 7 (v6l) BogoMIPS : 464.48 Features : swp half thumb fastmult vfp edsp java tls

12个Linux进程管理命令介绍

12个Linux进程管理命令介绍 当程序可以执行文件存放在存储中,并且运行的时候,每个进程会被动态得分配系统资源.内存.安全属性和与之相关的状态.可以有多个进程关联到同一个程序,并同时执行不会互相干扰.Linux提供了许多命令来让用户来高效掌控上述的操作. 执行中的程序称作进程.当程序可以执行文件存放在存储中,并且运行的时候,每个进程会被动态得分配系统资源.内存.安全属性和与之相关的状态.可以有多个进程关联到同一个程序,并同时执行不会互相干扰.操作系统会有效地管理和追踪所有运行着的进程. 为了管