android init.rc中service console option的含义、作用
void service_start(struct service *svc, const char *dynamic_args) needs_console = (svc->flags & SVC_CONSOLE) ? 1 : 0; /*如果是需要控制台环境但是没有控制台,设置SVC_DISABLED标志位后返回*/ if (needs_console && (!have_console)) { ERROR("service '%s' requires console ", svc->name); svc->flags |= SVC_DISABLED; return; } if (needs_console) { /* 使进程独立开来,摆脱源会话、源进程组、源控制终端,使其成为一个新的进程组长 */ setsid(); open_console(); } else { zap_stdio(); }
一个会话单独占据一个控制终端,不可能出现两个会话同时共享一个控制终端。
这也是为什么 setsid()函数能摆脱控制终端的原因,因为,一个控制终端只能被一个会话所拥有,而,setsid()函数调用成功之后,会创建一个新的会话,因此这时,出现了俩个会话,而这俩个会话不可能同时
共用一个控制终端,所以,新创建的会话会自动脱离原来的控制终端。
会话 (session)
更进一步,在shell支持工作控制(job control)的前提下,多个进程组还可以构成一个会话 (session)。bash(Bourne-Again shell)支持工作控制,而sh(Bourne shell)并不支持。
会话是由其中的进程建立的,该进程叫做会话的领导进程(session leader)。会话领导进程的PID成为识别会话的SID(session ID)。会话中的每个进程组称为一个工作(job)。会话可以有一个进程组成为会话的前台工作(foreground),而其他的进程组是后台工作(background)。每个会话可以连接一个控制终端(control terminal)。当控制终端有输入输出时,都传递给该会话的前台进程组。由终端产生的信号,比如CTRL+Z, CTRL+,会传递到前台进程组。
会话的意义在于将多个工作囊括在一个终端,并取其中的一个工作作为前台,来直接接收该终端的输入输出以及终端信号。 其他工作在后台运行。