第7章-初始化程序

    • 初始化阶段
      • 首先如何分配使用系统物理内存
      • 调用内核各个部分的初始化函数分别对内存管理、中断处理、块设备、字符设备、进程管理以及硬盘和软盘等硬件进行初始化
      • 在完成这些初始化后,系统各部分以及处于可运行状态。
      • 程序手动移到进程0运行
      • 进程0调用fork()创建init进程
    • init功能分为四个部分
      • 安装根文件系统
      • 显示系统信息
      • 运行系统初始资源配置rc中的命令
      • 执行登录shell程序

    环境初始化工作

    1528858957294.png

    • getty(get teletype writer),取得电传打字机。
    • 终端是一种字符型设备,它有多种类型,通常使用tty来简称各种类型的终端设备。tty是Teletype的缩写。Teletype是最早出现的一种终端设备,很象电传打字机,是由Teletype公司生产的。在Linux系统的设备特殊文件目录/dev/下

    1. 串口端口终端(/dev/ttySn)
    2. 控制终端(/dev/tty)
    3. 伪终端(/dev/pty/)
    4. 控制台终端(/dev/ttyn, /dev/console)
    5. 虚拟终端(/dev/pts/n)
    • tty(终端设备的统称)
    • pty(虚拟终端pseudo-terminal)
    • pts/ptmx(pts/ptmx结合使用,进而实现pty)
      1. pts(pseudo-terminal slave)是pty的实现方法,与ptmx(pseudo-terminal master)配合使用实现pty
    • 如果当前进程有控制终端(Controlling Terminal)的话,那么/dev/tty就是当前进程的控制终端的设备特殊文件。
    • 可以使用命令”ps –ax”来查看进程与哪个控制终端相连。对于你登录的shell,/dev/tty就是你使用的终端,设备号是(5,0)。
    • 使用命令 ”tty”可以查看它具体对应哪个实际终端设备。/dev/tty有些类似于到实际所使用终端设备的一个联接。
    1. /dev/ttyn
    2. /dev/console
    • 在UNIX系统中,计算机显示器通常被称为控制台终端(Console)。它仿真了类型为Linux的一种终端(TERM=Linux),并且有一些设备特殊文件与之相关联:tty0、tty1、tty2等。
    • 当你在控制台上登录时,使用的是tty1。使用Alt+[F1—F6]组合键时,我们就可以切换到tty2、tty3等上面去。
    • tty1 –tty6等称为虚拟终端,而tty0则是当前所使用虚拟终端的一个别名,系统所产生的信息会发送到该终端上。
    • 因此不管当前正在使用哪个虚拟终端,系统信息都会发送到控制台终端上。
    • 你可以登录到不同的虚拟终端上去,因而可以让系统同时有几个不同的会话期存在。
    • 只有系统或超级用户root可以向/dev/tty0进行写操作,
    • 伪终端(Pseudo Terminal)是成对的逻辑终端设备,例如/dev/ptyp3和/dev/ttyp3(或着在设备文件系统中分别是 /dev/pty/m3和/dev/pty/s3)。
    • 它们与实际物理设备并不直接相关。如果一个程序把ttyp3看作是一个串行端口设备,则它对该端口的读/写操作会反映在该逻辑终端设备对的另一个上面(ttyp3), 而ttyp3则是另一个程序用于读写操作的逻辑设备。这样,两个程序就可以通过这种逻辑设备进行互相交流,而其中一个使用ttyp3的程序则认为自己正在与一个串行端口进行通信。这很象是逻辑设备对之间的管道操作。
    • 对于ttyp3(s3),任何设计成使用一个串行端口设备的程序都可以使用该逻辑设备。但对于使用ptyp3的程序,则需要专门设计来使用ptyp3(m3)逻辑设备。
    • 例如,如果某人在网上使用telnet程序连接到你的计算机上,则telnet程序就可能会开始连接到设备ptyp2(m2)上(一个伪终端端口上)。此时一个getty程序就应该运行在对应的ttyp2(s2)端口上。当telnet从远端获取了一个字符时,该字符就会通过m2、s2传递给 getty程序,而getty程序就会通过s2、m2和telnet程序往网络上返回”login:”字符串信息。这样,登录程序与telnet程序就通过“伪终端”进行通信。通过使用适当的软件,就可以把两个甚至多个伪终端设备连接到同一个物理串行端口上。
    • 可见在重定向/dev/ptmx的时候在/dev/pts目录下多了个设备节点11,而当上面这个shell结束的时候再次ls /dev/pts目录,设备节点11又消失了。每open /dev/ptmx就会得到一个新的文件描述符,并且在/dev/pts/目录下生成一个与这个文件描述符对应的新的设备节点
    • 串行端口终端(Serial Port Terminal)是使用计算机串行端口连接的终端设备。
    • 计算机把每个串行端口都看作是一个字符设备。有段时间这些串行端口设备通常被称为终端设备,因为那时它的最大用途就是用来连接终端。
    • 这些串行端口所对应的设备名称是/dev/tts/0(或/dev /ttyS0)、/dev/tts/1(或/dev/ttyS1)等,设备号分别是(4,0)、(4,1)等,分别对应于DOS系统下的COM1、 COM2等。
    • 若要向一个端口发送数据,可以在命令行上把标准输出重定向到这些特殊文件名上即可。例如,在命令行提示符下键入如下命令,会把单词”test”发送到连接在ttyS1(COM2)端口的设备上。

      echo test > /dev/ttyS1   

    /etc/issue和/etc/motd的问题

    • /etc/motd
    1. /etc/motdmessageoftoday(布告栏信息),每次用户登录时,/etc/motd文件的内容会显示在用户的终端.系统管理员可以在文件中编辑系统活动消息,例如:管理员通知用户系统何时进行软件或硬件的升级、何时进行系统维护等.如果shell支持中文,还可以使用中文,这样看起来更易于了解.
    2. 注:window操作系统也有相关的功能,有的公司员工的工作机是统一管理,用的是windows ,当员工早上来登入工作机可能看到一些如提示信息,/etc/motd实现的作用跟它差不多。
    • /etc/issue
    1. issue 内的各代码意义
    2. 本地端时间的日期;
    3. /l 显示第几个终端机接口;
    4. /m 显示硬件的等级 (i386/i486/i586/i686...);
    5. /o 显示 domain name
    6. /r 操作系统的版本 (相当于 uname -r)
    7. /t 显示本地端时间的时间;
    8. /s 操作系统的名称;
    9. /v 操作系统的版本.
    • 首先剧透,printf = vsprintf + write。
      • vsprintf进行字符串格式化
      • write写到终端
    • 这里还需要编译器配合,本质上是编译器帮忙搞了一波内建功能buildin。虽然说参数可变,但是你编译的时候,参数四不四已经固定了?因为格式化字符串的内容和参数列表不可能在运行的时候改变的哦,这个要分清。就好像万精油一样,不管什么输出都可以用它,但是前提是,必须在编译之前用,编译后就固定了。这点其实就是泛型
    • 本质上是编译器帮忙处理,识别了传递的参数个数啊,参数类型啊,然后帮忙字节对齐访问参数。之所以可变,全都是编译器累死累活帮忙。

    1531124177671.png

    1. int vsprintf(char *buf, const char *fmt, va_list args)
    2. {
    3. int len;
    4. int i;
    5. char * str;
    6. char *s;
    7. int *ip;
    8. int flags; /* flags to number() */
    9. int field_width; /* width of output field */
    10. int precision; /* min. # of digits for integers; max
    11. number of chars for from string */
    12. int qualifier; /* 'h', 'l', or 'L' for integer fields */
    13. for (str=buf ; *fmt ; ++fmt) {
    14. if (*fmt != '%') {
    15. *str++ = *fmt;
    16. continue;
    17. }
    18. /* process flags */
    19. flags = 0;
    20. repeat:
    21. ++fmt; /* this also skips first '%' */
    22. switch (*fmt) {
    23. case '-': flags |= LEFT; goto repeat;
    24. case '+': flags |= PLUS; goto repeat;
    25. case ' ': flags |= SPACE; goto repeat;
    26. case '#': flags |= SPECIAL; goto repeat;
    27. case '0': flags |= ZEROPAD; goto repeat;
    28. }
    29. /* get field width */
    30. field_width = -1;
    31. field_width = skip_atoi(&fmt);
    32. else if (*fmt == '*') {
    33. /* it's the next argument */
    34. field_width = va_arg(args, int);
    35. if (field_width < 0) {
    36. field_width = -field_width;
    37. flags |= LEFT;
    38. }
    39. }
    40. /* get the precision */
    41. precision = -1;
    42. if (*fmt == '.') {
    43. ++fmt;
    44. if (is_digit(*fmt))
    45. precision = skip_atoi(&fmt);
    46. else if (*fmt == '*') {
    47. /* it's the next argument */
    48. precision = va_arg(args, int);
    49. }
    50. if (precision < 0)
    51. precision = 0;
    52. }
    53. /* get the conversion qualifier */
    54. qualifier = -1;
    55. if (*fmt == 'h' || *fmt == 'l' || *fmt == 'L') {
    56. qualifier = *fmt;
    57. ++fmt;
    58. switch (*fmt) {
    59. case 'c':
    60. if (!(flags & LEFT))
    61. while (--field_width > 0)
    62. *str++ = ' ';
    63. *str++ = (unsigned char) va_arg(args, int);
    64. while (--field_width > 0)
    65. *str++ = ' ';
    66. break;
    67. case 's':
    68. s = va_arg(args, char *);
    69. len = strlen(s);
    70. if (precision < 0)
    71. precision = len;
    72. else if (len > precision)
    73. len = precision;
    74. if (!(flags & LEFT))
    75. while (len < field_width--)
    76. *str++ = ' ';
    77. *str++ = *s++;
    78. while (len < field_width--)
    79. *str++ = ' ';
    80. break;
    81. case 'o':
    82. str = number(str, va_arg(args, unsigned long), 8,
    83. field_width, precision, flags);
    84. break;
    85. case 'p':
    86. if (field_width == -1) {
    87. field_width = 8;
    88. flags |= ZEROPAD;
    89. }
    90. str = number(str,
    91. (unsigned long) va_arg(args, void *), 16,
    92. field_width, precision, flags);
    93. break;
    94. case 'x':
    95. flags |= SMALL;
    96. case 'X':
    97. str = number(str, va_arg(args, unsigned long), 16,
    98. field_width, precision, flags);
    99. break;
    100. case 'd':
    101. case 'i':
    102. flags |= SIGN;
    103. case 'u':
    104. str = number(str, va_arg(args, unsigned long), 10,
    105. field_width, precision, flags);
    106. break;
    107. case 'n':
    108. ip = va_arg(args, int *);
    109. *ip = (str - buf);
    110. break;
    111. default:
    112. if (*fmt != '%')
    113. *str++ = '%';
    114. if (*fmt)
    115. *str++ = *fmt;
    116. else
    117. --fmt;
    118. break;
    119. }
    120. }
    121. *str = '\0';
    122. return str-buf;
    • 参考一波博客文章,例子很简单,上面的函数实现比较实用,所以长的一笔

    1531126205752.png

    1531126245021.png

    https://www.cnblogs.com/cpoint/p/3368993.html

    总结