shell 查看会话、进程组
ps -o pid,ppid,pgid,sid,comm
1、控制终端
会话和进程组有一些特性:
1). 一个会话可以有一个控制终端(controlling terminal),会话首进程打开一个终端之后, 该终端就成为该会话的控制终端。
2). 建立与控制终端连接的会话首进程被称为控制进程(controlling process)。
3). 一个会话中的几个进程组可被分成一个前台进程组(forkground process group)和几个后台进程组(background process group)。
4). 如果一个会话有一个控制终端,则它有一个前台进程组。
5). 无论何时键入终端的中断键(DELETE或Ctrl+C),就会将中断信号发送给前台进程组的所有进程。
6). 无论何时键入终端的退出键(Ctrl+/),就会将退出信号发送给前台进程组的所有进程。
7). 如果终端检测到调制解调器(或网络)已经断开连接,则将挂断信号发送给控制进程(会话首进程)。
需要有一种方法来通知内核哪一个进程组是前台进程组,这样,终端设备驱动程序就能了解将终端输入和终端产生的信号送到何处。
- #include <unistd.h>
- pid_t tcgetpgrp(int filedes);
返回值:成功返回前台进程组的进程组ID,出错返回-1
- int tcsetpgrp(int filedes, pid_t pgrpid);
返回值:成功返回0,出错返回-1
函数tcgetpgrp返回前台进程组的进程组ID,该前台进程组与在filedes上打开的终端相关联。如果进程有一个控制终端,则该进程可以调用tcsetpgrp将前台进程组ID设置为pgrpid。pgrpid的值应当是在同一会话中的一个进程组的ID。filedes必须引用该会话的控制终端。大多数应用程序并不直接调用这两个函数,它们通常由作业控制shell调用。
- #include <termios.h>
- pid_t tcgetsid(int filedes);
返回值:成功返回会话首进程的进程组ID,出错返回-1。
给出控制TTY的文件描述符,应用程序就能获得会话首进程的进程组ID。需要管理控制终端的应用程序可以调用tcgetsid函数识别出控制终端的会话首进程的会话ID。
通常,我们不必关心控制终端,登录时,将自动建立控制终端。
例如,在shell中:
$proc1 | proc2 &
$proc3 | proc4 | proc5
那么此时,session中就会有三个进程组存在,分别是{登陆shell(session leader)},{proc1, proc2}, {proc3, proc4, proc5}。
这里要注意:与终端连接的进程为“控制进程”,与终端连接的进程组为“控制进程组”。
上图,“{proc3, proc4, proc5}”为前台进程组,也为控制进程,当此进程组运行完毕,shell将回收,替代为控制进程,此时shell为会话首进程,亦为控制进程(详见:shell 前台进程组的选择)。
同一会话中的进程通过该会话的领头进程和一个终端相连,该终端作为这个会话的控制终端。一个会话只能有一个控制终端,而一个控制终端只能控制一个会话。用户通过控制终端,可以向该控制终端所控制的会话中的进程发送键盘信号。
同一会话中只能有一个前台进程组,属于前台进程组的进程可从控制终端获得输入,而其他进程均是后台进程,可能分属于不同的后台进程组。
2. Linux中的实现举例,用以验证上述规则: