1. 启动过程从用户启动Mac OS X系统到出现登录窗口,Mac OS X执行了一套启动流程来使系统可被使用。如果你想为所有用户提供系统服务,你需要在系统启动的过程中运行一些代码。下面的章节介绍了基本启动流程和应该把你提供服务的代码放到何处。
1.1. BootROM当一台Macintosh被启动时,BootROM固件同时也被激活。BootROM(也是电脑的一个硬件)拥有两个主要职责:初始化(译者注,其实就是启动各个硬件,让其准备好被使用)系统的硬件和选择一个操作系统来启动。BootROM拥有两个部分来帮助它实现这两个职责:
在Intel架构的Macintosh电脑上,EFI来处理基本的硬件初始化,并且选择一个操作系统来启动。如果安装有多个可启动的Mac OS X系统,BootROM会选择那个最后被系统预置中启动磁盘设置项选择的操作系统。用户可以在电脑启动时按住Option键来自行选择启动的系统,这个做法会让Open Firmware或者EFI显示一个选择启动宗卷的界面。
注意,在一些遗留机型(译者注,老机型)上,有些版本的BootROM既可以启动Mac OS 9系统也可以启动Mac OS X。而大多数现在的机型只能启动Mac OS X。
1.2. BootX,boot.efi,系统初始化一旦BootROM的工作完成,并且选择了个Mac OS X分区被选择用其启动,控制权就被传递给BootX(PowerPC平台)或者boot.efi(Intel平台)启动引导器。启动引导器的主要工作就是加载内核环境(译者注,原文kernel environment),作此工作时,启动引导器会在屏幕上显示一个启动画面。我们可以在根分区的/System/Library/CoreServices目录下找到BootX和boot.efi。而且,可以找到/usr/standalone/i386/boot.efi这个boot.efi的拷贝。在“exotic”(译者注,字面意思是“外来的,外国的”,但显然这里不是只外接硬盘的意思。)模式下启动的时候,比如说从一个软RAID宗卷上启动的时候,一个启动引导器会保存在一个叫做独立的HFS+的”辅助“(译者注,原文“helper”)宗卷上来启动系统。在一些版本的Mac OS X上,内核和mkext缓存的拷贝亦会保存在辅助宗卷里。也就是,这种情况下,根分区下面的引导器和其他一些部件不起作用。
注意:不推荐用UFS格式做Mac OS X v10.5的引导宗卷。启动过程最开始,启动引导器尝试加载一个包含了所有设备驱动程序的预链接( 译者注,原文prelink,这个词的意思是,为了节省时间而预先把一些动态链接库的连接方式记录在执行档中)的内核版本,其位置在:/System/Library/Caches/com.apple.kernelcaches。预链接这些设备驱动程序可以缩短系统启动时间。如果预链接版本的内核丢失,已经过期,或者已损坏,启动引导器会加载一个包含了与预连接内核所列相同驱动的单独压缩包,此包被称作mkext缓存。如果mkext缓存包过期,丢失,或者已损坏,启动引导器会搜索/System/Library/Extensions,寻找OSBundleRequired属性根据启动类型(例如本地启动或者网络启动)所列的驱动列表来加载驱动。查看I/O Kit Fundamentals(I/O工具纲要)可以获得更多加载驱动的信息。一旦内核和所有为启动而必需的驱动加载成功,启动引导器就启动内核的初始化进程。此时应该有足够的驱动被加载,以找到根设备( 译者注,原文root device)来加载内核。也是在此时开始,PowerPC架构的Macintosh电脑上,Open Firmware不再被访问(静默状态)。内核初始化Mach和BSD数据结构( 译者注,原文data structure,我也不知道他这里到底指什么意思,应该就是指BSD那一层的东西。)以及I/O Kit。I/O Kit使用设备树( 译者注,device tree)作为根据来连接那些已经加载进内核的驱动。一旦内核找到根设备( 译者注,原文root device),BSD将把此为根*( 译者注,原文Once the kernel finds the root device, it roots(*) BSD off of it.)(来启动)。
注意:作为一个术语,“启动”(boot)这个词传统上讲是指加载启动引导器和启动硬盘或者分区上的内核。而最近几年,这个词的含义被扩展了第二层意义:指启动系统可以被最终用户使用的整个启动过程。在这里,“启动”这个词是指第一种含义。
在这里,“根”(root)这个词是指把一个分区作为根分区加载,或者是第一层,或者是文件系统(译者注,原文mounting a partition as the root, or top-level, filesystem,不知道是不是指这个逻辑关系。)。因此,系统在根分区上启动的时候,内核将会在运行启动脚本之前,把这个分区认为根分区(译者注,原文Thus, while the OS boots off of the root partition, the kernel roots the OS off of the partition before executing startup scripts from it.)。Mac OS X v10.4以前,系统初始化一直是交给mach_init和init来处理。初始化过程中,这两个处理进程调用不同的系统脚本(在/ect/rc目录下),启动程序,为用户准备系统。Mac OS X v10.4以后,这些系统脚本和守候进程的调用交由launchd来处理。也就是说,目前,launchd是系统根进程。除了初始化系统已外,launchd进程还协调系统守候进程秩序地调用。就像inetd进程一样,launchd按需启动守候进程( 译者注,原文launchd launches daemons on-demand,其实就是指那种launch-on-demand方式。)。以这种方式调用的守候进程可以在不活动期间被关闭或者按需要重启。(当一个服务的请求发送过来的时候,launchd会自动重新启动守候进程去处理请求。)这种方式可以释放守候进程占用的内存和其他相关资源,当守候进程在一段很长的时间闲置的时候,这样做非常值得。但更重要的,这样做相对于人工方式能更有效的处理守候进程间的依赖关系。然后,launchd 启动SystemStarter,后者用来启动那些非按需启动( 译者注,原文non-launch-on-demand)的守候进程。
注意,launchd也支持非按需启动的守候进程,但是这种用法并不推荐。launchd守候进程被设计用来消除守候进程之间的依赖关系。如果你不把你的守候进程设置成按需启动(译者注,原文launch-on-demand),你就需要用其他办法处理依赖问题,比如用那种遗留下来的Startup Item机制。查看Daemons可以获得更多关于按需启动( 原文launch-on-demand)和SystemStarter守候进程的信息。系统初始化的最后阶段,launchd启动loginwindow。loginwindow程序控制了用户会话( 原文user session)各自的外观( 原文aspect)和登录窗口的调整以及用户登录认证。
注意,默认情况下,Mac OS X随着一个图形化的启动屏幕启动。为了调试启动进程,我们也可以禁用它,而显示文字操作台信息。这种模式被称作verbose启动模式。要想使用verbose启动模式,只需要在启动的时候按住command-v键。
重要信息:注意,如果一个注销,重新启动或者关机event是被一个Classic环境中的应用程序发送的,这些event只会对Classic环境和其中的应用程序起作用。剩下的用户会话将会继续运行。
注意:Cocoa应用程序不支持kAEQuitApplication event,取而代之的,Application Kit通过调用delegate方法applicationShouldTerminalte来实现终止程序。要想取消此操作,可以实现这个方法并返回NSTerminateCancel;不然,应用程序的终止操作将会继续执行。如果一个前端应用程序响应失败或者超过45秒钟仍未终止,loginwindow自动取消终止操作。这个保护措施可以在很多情况下保护数据,例如当一个应用程序正在保存一个大文件到磁盘中并且此时不能在要求的时间内终止程序的时候。如果一个前端应用程序反应迟钝或者没有做任何事情,用户必须使用“强制退出”来强制退出(kill)该程序。对于使用了(link with)Carbon,Cocoa,Java的后端进程,操作方式将会有一点不一样。loginwindow程序通过发送Quit Application Apple event(kAEQuitApplication)通知进程将被终止。不同于对待前端进程那样,loginwindow不等待后端进程回应,它直接终止开放的后端进程,不管任何被返回的错误信息。用户注销的过程中,loginwindow不会终止root层的进程。这些进程不属于用户会话,他们只会在重新启动或者关机的时候被终止。loginwindow程序也不会杀死(kill)不依赖Carbon,Cocoa,Java的后端进程,尽管他们是用户层进程。(尽管在用户层面被启动,这些进程在用户登出的时候被系统接管。)Mac OS X在终止系统进程的时候不会向其发送任何通知。
注意:不是所有的用户进程都是WindowServer进程的子进程。那些root层启动的进程,特殊的系统进程,这些进程拥有者是用户,但他们是launchd的子进程。你可以通过活动监视器来判断系统中一个进程的拥有者和父进程。
启动过程:
1、 电源开启。
2、 执行固件中的代码。
3、 收集硬件信息并初始硬件。
4、 选择启动项(通常是选择 OS ,但有时会选择硬件测试等类似情况。)。用户可能会被提示进行启动选择。
5、 控制权交给 /System/Library/CoreServices/BootX (启动引导器)。 x86为boot.efi
6、 执行内核中的 init 例程。决定要启动之系统的根设备。从此刻起,将不再使用固件中的程序。
7、 由内核初始各种 Mach/BSD 数据结构。
8、 初始 I/O (输入输出)设备。
9、 内核开始运行 /sbin/mach_init ,Mach 服务命名(引导程序)后台。mach_init 为服务名和要准备访问其它服务所用的 Mach 端口提供映射。
到这步时,启动开始转为用户等级:
10、 mach_init 开始 /sbin/init,传统的 BSD 初始化(init)进程。初始化将决定运行等级,并运行 /etc/rc.boot (设置让机器能够运行单用户-single user 模式)。
在此步中,将执行: rc.boot 与其它 rc 脚本源程序 /etc/rc.common,一个包含实用功能的 shell 脚本,如 CheckForNetwork() (检查如网络已启动), GetPID(), purgedir() (仅删除目录内容,而非结构),等。
11、 rc.boot 会显示要启动的类型(多用户,安全模式,光驱,网络等等)。网络启动的情况下( sysctl 的变量 kern.netboot 将会为何种情况而设之为 1 ),其将用一个启动参数来运行 /etc/rc.netboot 。
/etc/rc.netboot 会处理网络启动的参数特征。例如:执行网络和(如有)本地挂载。其还会呼叫 /usr/bin/nbst 来关联当作根设备使用的磁盘镜像到一个影子文件(shadow file)。此方法是将那个希望处于本地存储器的文件(磁盘)重定向写入到影子文件。
12、 rc.boot 会在必须进行文件系统一致性检查(file system consistency check, fsck)时,显示图形。单用户模式和用光盘启动时不会运行 fsck。安全模式启动时总会运行 fsck。rc.boot 也会处理 fsck 的返回状态。
13、 如果 rc.boot 成功退出, /etc/rc 多用户启动脚本将会运行。如果正在从一个光驱启动,脚本将切换到 /etc/rc.cdrom (安装)。
14、 /etc/rc 挂载本地文件系统 (HFS+、HFS、UFS、/dev/fd、/.vol),确保目录 /private/var/tmp 存在,然后运行 /etc/rc.installer_cleanup 如果有(重启前,会由安装器离开)。
15、 /etc/rc.cleanup 运行。其将“清理”一定数量的 Unix 与 Mac 特殊目录/文件。
16、 启动缓存 (BootCache)开始。
17、 各种 sysctl 变量被设置(如:vnodes 的最大值、System V IPC 等)。如果 /etc/sysctl.conf 已存在 (在 Mac OS X Server 中为 /etc/sysctl-macosxserver.conf),它将读取和设置 sysctl 变量为已包含在其中的。
18、 syslogd 开始。
19、 创建机器检查符号文件(Mach symbol file)。
20、 /etc/rc 开始 kextd 后台进程,用来从内核或委托进程 (client processes)加载所需的内核扩展。
21、 /usr/libexec/register_mach_bootstrap_servers 将运行以加载包含在 /etc/mach_init.d 中的各种 Mach 引导程序所基于的服务。
22、 portmap 与 netinfo 开始。
23、 如 /System/Library/Extensions.mkext 旧于 /System/Library/Extensions, /etc/rc 将删除已存在的 mkext 并创建一个新的(不存在时,会创建)。
24、 /etc/rc 启动 /usr/sbin/update,一个后台程序,用来频繁地清空磁盘上的互联网文件系统缓存。
25、 /etc/rc 启动虚拟内存系统。 设置 /private/var/vm 为一个交换目录。/sbin/dynamic_pager 以适当的参数启动(交换文件名路径模板、已创建的交换文件大小、当创建额外交换文件或删除已存在文件时,指定高、低水平的警报切换开关。)
26、 /etc/rc 启动 /usr/libexec/fix_prebinding 以修复错误地预连编二进制文件 (prebound binaries)
27、 /etc/rc 执行 /etc/rc.cleanup 以清除并重置文件与设备。
28、 /etc/rc 最后将启动 /sbin/SystemStarter ,处理启动项从下列位置: /System/Library/StartupItems 与 /Library/StartupItems。一个启动项是一个程序、一个 shell 脚本、匹配一个文件夹名的名称。文件夹包含一个属性列表文件含有一些配对的关键值,如: Description、Provides、Requires、 OrderPreference、启动与停止信息等等。您可以运行 SystemStarter -n -D 以作为根用户 (root) 来进行程序打印调试与从属信息(不包含现在已经在运行的任何项目)。
29、 CoreGraphics 启动开始 Apple 类型服务后台(ATSServer) 和 Window 服务器 (WindowServer)。
默认下,loginwindow 程序 (loginwindow.app 位于 /System/Library/CoreServices 目录下) 已为控制设备执行。如果您不想运行到图形登录,可以修改 /etc/ttys 中相关的行。