第七周笔记##
1.内核gdb调试报错
(gdb) target remote:1234
:1234: u8fdeu63a5u8d85u65f6.
原因是没有冻结窗口,落了一步,绝了
qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -S -s
2.听说macro参数status并不同于wait的参数-int型指针status,咋不同啦?
活动笔记
wait
pid_t wait(int *status)
进程调用了wait,立即阻塞父进程,wait分析当前进程的某个子进程是否已经退出。
case1:成功,找到僵尸子进程,wait就会收集子进程信息,彻底销毁,返回;
case2:失败,没有找到僵尸子进程,wait一直阻塞在这里,直到有一个出现。
参数status用来保存被收集进程退出时的一些状态,它是一个指向int类型的指针,可以设定这个参数为NULL。
case1:wait成功,返回被收集的子进程的进程ID;
case2:调用失败,调用进程没有子进程,wait返回-1,同时errno被置为ECHILD。
示例:
/* wait1.c */
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdlib.h>
main()
{
pid_t pc,pr;
pc=fork();
if(pc<0) /* 如果出错 */
printf("error ocurred!/n");
else if(pc==0){ /* 如果是子进程 */
printf("This is child process with pid of %d/n",getpid());
sleep(10); /* 睡眠10秒钟 */
}
else{ /* 如果是父进程 */
pr=wait(NULL); /* 在这里等待 */
printf("I catched a child process with pid of %d/n"),pr);
}
exit(0);
}
编译并运行:
$ gcc wait1.c -o wait1
$ ./wait1
宏macro:
1.WIFEXITED(status):子进程是否为正常退出的。
若是,返回一个非零值(请注意,虽然名字一样,这里的参数status并不同于wait唯一的参数---指向整数的指针status,而是那个指针所指向的整数,切记不要搞混了)
2、WEXITSTATUS(status) :子进程返回值。
case1:返回非零值,提取子进程的返回值;
case2:如果子进程调用exit(5)退出,返回5;
case3:如果子进程调用exit(7),返回7;
case4:不是正常退出的,返回0无意义。
示例:
/* wait2.c */
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
main()
{
int status;
pid_t pc,pr;
pc=fork();
if(pc<0) /* 如果出错 */
printf("error ocurred!/n");
else if(pc==0){ /* 子进程 */
printf("This is child process with pid of %d./n",getpid());
exit(3); /* 子进程返回3 */
}
else{ /* 父进程 */
pr=wait(&status);
if(WIFEXITED(status)){ /* 如果WIFEXITED返回非零值 */
printf("the child process %d exit normally./n",pr);
printf("the return code is %d./n",WEXITSTATUS(status));
}else /* 如果WIFEXITED返回零 */
printf("the child process %d exit abnormally./n",pr);
}
}
编译并运行:
$ gcc wait2.c -o wait2
$ ./wait2
waitpid
waitpid多出了两个可由用户控制的参数pid和options.
pid:
pid>0,只等待进程ID等于pid的子进程;
pid=0,等待同一个进程组中的任何子进程(如果子进程已经加入了别的进程组,waitpid不care);
pid=-1,等待任何一个子进程退出;
pid<-1,等待指定进程组中的任何子进程,进程组的ID等于pid的绝对值。
options:
Linux中只支持WNOHANG和WUNTRACED两个选项,这是两个常数,可以用"|"运算符把它们连接起来使用。
WNOHANG参数调用waitpid,即使没有子进程退出,它也会立即返回,不会像wait那样永远等下去。
WUNTRACED参数用的很少。
示例
ret=waitpid(-1,NULL,WNOHANG | WUNTRACED);
可以把options设为0.
ret=waitpid(-1,NULL,0);
static inline pid_t wait(int * wait_stat)
{
return waitpid(-1,wait_stat,0);
}
返回值和错误:
case1:waitpid返回收集到的子进程的进程ID;
case2:设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
case3:调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;
case4:pid所指示的子进程不存在,或此进程存在,但不是调用进程的子进程,waitpid就会出错返回,这时errno被设置为ECHILD;
示例
/* waitpid.c */
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
main()
{
pid_t pc, pr;
pc=fork();
if(pc<0) /* 如果fork出错 */
printf("Error occured on forking./n");
else if(pc==0){ /* 如果是子进程 */
sleep(10); /* 睡眠10秒 */
exit(0);
}
/* 如果是父进程 */
do{
pr=waitpid(pc, NULL, WNOHANG); /* 使用了WNOHANG参数,waitpid不会在这里等待 */
if(pr==0){ /* 如果没有收集到子进程 */
printf("No child exited/n");
sleep(1);
}
}while(pr==0); /* 没有收集到子进程,就回去继续尝试 */
if(pr==pc)
printf("successfully get child %d/n", pr);
else
printf("some error occured/n");
}
编译并运行:
$ cc waitpid.c -o waitpid
$ ./waitp