📄 build.c
字号:
/** linux/tools/build.c** (C) 1991 Linus Torvalds*//** This file builds a disk-image from three different files:** - bootsect: max 510 bytes of 8086 machine code, loads the rest* - setup: max 4 sectors of 8086 machine code, sets up system parm* - system: 80386 code for actual system** It does some checking that all files are of the correct type, and* just writes the result to stdout, removing headers and padding to* the right amount. It also writes some system data to stderr.*//** 该程序从三个不同的程序中创建磁盘映象文件:** - bootsect:该文件的8086 机器码最长为510 字节,用于加载其它程序。* - setup:该文件的8086 机器码最长为4 个磁盘扇区,用于设置系统参数。* - system:实际系统的80386 代码。** 该程序首先检查所有程序模块的类型是否正确,并将检查结果在终端上显示出来,* 然后删除模块头部并扩充大正确的长度。该程序也会将一些系统数据写到stderr。*//** Changes by tytso to allow root device specification*//** tytso 对该程序作了修改,以允许指定根文件设备。*/#include <stdio.h> /* fprintf */ /* 使用其中的fprintf() */#include <string.h> /* 字符串操作 */#include <stdlib.h> /* contains exit */ /* 含有exit() */#include <sys/types.h> /* unistd.h needs this */ /* 供unistd.h 使用 */#include <sys/stat.h> /* 文件状态信息结构 */#include <linux/fs.h> /* 文件系统 */#include <unistd.h> /* contains read/write */ /* 含有read()/write() */#include <fcntl.h> /* 文件操作模式符号常数 */#define MINIX_HEADER 32 // minix 二进制模块头部长度为32 字节。#define GCC_HEADER 1024 // GCC 头部信息长度为1024 字节。#define SYS_SIZE 0x2000 // system 文件最长节数(字节数为SYS_SIZE*16=128KB)。#define DEFAULT_MAJOR_ROOT 3 // 默认根设备主设备号 - 3(硬盘)。#define DEFAULT_MINOR_ROOT 6 // 默认根设备次设备号 - 6(第2 个硬盘的第1 分区)。/* max nr of sectors of setup: don't change unless you also change* bootsect etc *//* 下面指定setup 模块占的最大扇区数:不要改变该值,除非也改变bootsect 等相应文件。#define SETUP_SECTS 4 // setup 最大长度为4 个扇区(4*512 字节)。*/#define STRINGIFY(x) #x // 用于出错时显示语句中表示扇区数。//// 显示出错信息,并终止程序。voiddie (char *str){ fprintf (stderr, "%s\n", str); exit (1);}// 显示程序使用方法,并退出。voidusage (void){ die ("Usage: build bootsect setup system [rootdev] [> image]");}intmain (int argc, char **argv){ int i, c, id; char buf[1024]; char major_root, minor_root; struct stat sb;// 如果程序命令行参数不是4 或5 个,则显示程序用法并退出。 if ((argc != 4) && (argc != 5)) usage ();// 如果参数是5 个,则说明带有根设备名。 if (argc == 5) {// 如果根设备名是软盘("FLOPPY"),则取该设备文件的状态信息,若出错则显示信息,退出。 if (strcmp (argv[4], "FLOPPY")) { if (stat (argv[4], &sb)) { perror (argv[4]); die ("Couldn't stat root device."); }// 若成功则取该设备名状态结构中的主设备号和次设备号。 major_root = MAJOR (sb.st_rdev); minor_root = MINOR (sb.st_rdev); } else {// 否则让主设备号和次设备号取0。 major_root = 0; minor_root = 0; }// 若参数只有4 个,则让主设备号和次设备号等于系统默认的根设备。 } else { major_root = DEFAULT_MAJOR_ROOT; minor_root = DEFAULT_MINOR_ROOT; }// 在标准错误终端上显示所选择的根设备主、次设备号。 fprintf (stderr, "Root device is (%d, %d)\n", major_root, minor_root);// 如果主设备号不等于2(软盘)或3(硬盘),也不等于0(取系统默认根设备),则显示出错信息,退出。 if ((major_root != 2) && (major_root != 3) && (major_root != 0)) { fprintf (stderr, "Illegal root device (major = %d)\n", major_root); die ("Bad root device --- major #"); }// 初始化buf 缓冲区,全置0。 for (i = 0; i < sizeof buf; i++) buf[i] = 0;// 以只读方式打开参数1 指定的文件(bootsect),若出错则显示出错信息,退出。 if ((id = open (argv[1], O_RDONLY, 0)) < 0) die ("Unable to open 'boot'");// 读取文件中的minix 执行头部信息(参见列表后说明),若出错则显示出错信息,退出。 if (read (id, buf, MINIX_HEADER) != MINIX_HEADER) die ("Unable to read header of 'boot'");// 0x0301 - minix 头部a_magic 魔数;0x10 - a_flag 可执行;0x04 - a_cpu, Intel 8086 机器码。 if (((long *) buf)[0] != 0x04100301) die ("Non-Minix header of 'boot'");// 判断头部长度字段a_hdrlen(字节)是否正确。(后三字节正好没有用,是0) if (((long *) buf)[1] != MINIX_HEADER) die ("Non-Minix header of 'boot'");// 判断数据段长a_data 字段(long)内容是否为0。 if (((long *) buf)[3] != 0) die ("Illegal data segment in 'boot'");// 判断堆a_bss 字段(long)内容是否为0。 if (((long *) buf)[4] != 0) die ("Illegal bss in 'boot'");// 判断执行点a_entry 字段(long)内容是否为0。 if (((long *) buf)[5] != 0) die ("Non-Minix header of 'boot'");// 判断符号表长字段a_sym 的内容是否为0。 if (((long *) buf)[7] != 0) die ("Illegal symbol table in 'boot'");// 读取实际代码数据,应该返回读取字节数为512 字节。 i = read (id, buf, sizeof buf); fprintf (stderr, "Boot sector %d bytes.\n", i); if (i != 512) die ("Boot block must be exactly 512 bytes");// 判断boot 块0x510 处是否有可引导标志0xAA55。 if ((*(unsigned short *) (buf + 510)) != 0xAA55) die ("Boot block hasn't got boot flag (0xAA55)");// 引导块的508,509 偏移处存放的是根设备号。 buf[508] = (char) minor_root; buf[509] = (char) major_root;// 将该boot 块512 字节的数据写到标准输出stdout,若写出字节数不对,则显示出错信息,退出。 i = write (1, buf, 512); if (i != 512) die ("Write call failed");// 最后关闭bootsect 模块文件。 close (id);// 现在开始处理setup 模块。首先以只读方式打开该模块,若出错则显示出错信息,退出。 if ((id = open (argv[2], O_RDONLY, 0)) < 0) die ("Unable to open 'setup'");// 读取该文件中的minix 执行头部信息(32 字节),若出错则显示出错信息,退出。 if (read (id, buf, MINIX_HEADER) != MINIX_HEADER) die ("Unable to read header of 'setup'");// 0x0301 - minix 头部a_magic 魔数;0x10 - a_flag 可执行;0x04 - a_cpu, Intel 8086 机器码。 if (((long *) buf)[0] != 0x04100301) die ("Non-Minix header of 'setup'");// 判断头部长度字段a_hdrlen(字节)是否正确。(后三字节正好没有用,是0) if (((long *) buf)[1] != MINIX_HEADER) die ("Non-Minix header of 'setup'");// 判断数据段长a_data 字段(long)内容是否为0。 if (((long *) buf)[3] != 0) die ("Illegal data segment in 'setup'");// 判断堆a_bss 字段(long)内容是否为0。 if (((long *) buf)[4] != 0) die ("Illegal bss in 'setup'");// 判断执行点a_entry 字段(long)内容是否为0。 if (((long *) buf)[5] != 0) die ("Non-Minix header of 'setup'");// 判断符号表长字段a_sym 的内容是否为0。 if (((long *) buf)[7] != 0) die ("Illegal symbol table in 'setup'");// 读取随后的执行代码数据,并写到标准输出stdout。 for (i = 0; (c = read (id, buf, sizeof buf)) > 0; i += c) if (write (1, buf, c) != c) die ("Write call failed");//关闭setup 模块文件。 close (id);// 若setup 模块长度大于4 个扇区,则算出错,显示出错信息,退出。 if (i > SETUP_SECTS * 512) die ("Setup exceeds " STRINGIFY (SETUP_SECTS) " sectors - rewrite build/boot/setup");// 在标准错误stderr 显示setup 文件的长度值。 fprintf (stderr, "Setup is %d bytes.\n", i);// 将缓冲区buf 清零。 for (c = 0; c < sizeof (buf); c++) buf[c] = '\0';// 若setup 长度小于4*512 字节,则用\0 将setup 填足为4*512 字节。 while (i < SETUP_SECTS * 512) { c = SETUP_SECTS * 512 - i; if (c > sizeof (buf)) c = sizeof (buf); if (write (1, buf, c) != c) die ("Write call failed"); i += c; }// 下面处理system 模块。首先以只读方式打开该文件。 if ((id = open (argv[3], O_RDONLY, 0)) < 0) die ("Unable to open 'system'");// system 模块是GCC 格式的文件,先读取GCC 格式的头部结构信息(linux 的执行文件也采用该格式)。 if (read (id, buf, GCC_HEADER) != GCC_HEADER) die ("Unable to read header of 'system'");// 该结构中的执行代码入口点字段a_entry 值应为0。 if (((long *) buf)[5] != 0) die ("Non-GCC header of 'system'");// 读取随后的执行代码数据,并写到标准输出stdout。 for (i = 0; (c = read (id, buf, sizeof buf)) > 0; i += c) if (write (1, buf, c) != c) die ("Write call failed");// 关闭system 文件,并向stderr 上打印system 的字节数。 close (id); fprintf (stderr, "System is %d bytes.\n", i);// 若system 代码数据长度超过SYS_SIZE 节(或128KB 字节),则显示出错信息,退出。 if (i > SYS_SIZE * 16) die ("System is too big"); return (0);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -