9.1 终端登录
这里终端是本地的或者是远程的。登录unix系统都经由内核中的终端设备驱动程序。
BSD系统的过程:init读取文件/etc/ttys,对每一个允许登录的终端设备调用一次fork,它所生成的子进程则exec getty程序。getty打开终端设备,exec login程序。至此,登录用户的登录shell开始运行。
9.2 网络登录
网络登录时 在终端和计算机之间的连接不再是点到点的。在网络登录情况下,login仅仅是一种可用的网络服务。
为使同一个软件既能处理终端登录又能处理网络登录,系统使用了一种称为伪终端。
BSD系统过程:init调用shell脚本启动一个守护进程inetd。当一个TCP/IP连接请求到达主机时,inetd执行一次fork,生成的子进程exec适当的程序,如telnetd。获得登录shell连接到终端或伪终端。
9.3 进程组
每个进程还属于一个进程组。进程组是一个或多个进程的集合。通常它们是在同一作业中结合起来的 ,同一进程组中的各进程接收来
自同一终端的各种信号。每个进程组有一个唯一的进程组ID。
每个进程组有一个组长进程。组长进程的进程组ID等于其进程ID。进程组组长可以创建一个进程组、创建该组中的进程然后终止。从进程组创建开始到其中最后一个进程离开为止的时间区间称为进程组的生命期。
9.4 会话
会话(session)是一个或多个进程组的集合。
pid_t setsid(void);
pid_t getsid(pid_t pid);
终端设备
-
一个会话可以有一个控制终端,包括终端设备或伪终端。
-
建立与控制终端连接的会话首进程被称为控制进程。
-
一个会话中的几个进程组可被分成一个前台进程组和一个或多个后台进程组。
9.5 孤儿进程组
孤儿进程组:该组中每个成员的父进程要么是该组的一个成员,要么不是该组所属会话的成员(不属于同会话的其他进程组)。
#include "apue.h"
#include <errno.h>
static void sig_hup(int signo)
{
printf("SIGHUP received, pid = %ld\n", (long)getpid());
}
static void pr_ids(char *name)
{
printf("%s: pid = %ld, ppid = %ld, pgrp = %ld, tpgrp = %ld\n",
name, (long)getpid(), (long)getppid(), (long)getpgrp(),
(long)tcgetpgrp(STDIN_FILENO));
fflush(stdout);
}
int main(void)
{
char c;
pid_t pid;
pr_ids("parent");
if ((pid = fork()) < 0) {
err_sys("fork error");
} else if (pid > 0) { /* parent */
sleep(5); /* sleep to let child stop itself */
} else { /* child */
pr_ids("child");
signal(SIGHUP, sig_hup); /* establish signal handler */
kill(getpid(), SIGTSTP); /* stop ourself */
pr_ids("child"); /* prints only if we're continued */
if (read(STDIN_FILENO, &c, 1) != 1)
printf("read error %d on controlling TTY\n", errno);
}
exit(0);
}