一年多以前,电子工业出版社的张毅编辑告诉我说,有一位年轻的程序员,正在写一本《自己动手写操作系统》的书。知道这个消息,我既有点好奇,又有些担忧。如果是在十年前,这样题材的书将会是读者争相传阅的对象,毕竟20世纪90年代是软件的理想主义年代。但是在理想褪尽、实务未兴的尴尬的这两年,这样一本书在市场上究竟会遇到怎样的待遇,确实让人不敢乐观。不过,在阅读了样章之后,我深为作者清新的文笔、流畅的思路和扎实的技术功底所折服,于是请张毅为我引见了这位作者,即本书的作者于渊。 于渊非常年轻,却有着高人一筹的表达能力和技术视野,我觉得他是难得的技术写作人才,就鼓励他在《程序员》杂志开辟了一个技术专栏,专门剖析操作系统相关的技术。一年来这个专栏陆续发表了一系列文章,获得了不少读者的正面反馈。 然而,事实证明,我最初的担忧并不是没有道理的。一年多来,不断有人表达过他们对这样一个题材的不同看法。他们认为,相对于90年代中后期,现在的软件产业已经务实了很多,今天的程序员更关心的是如何尽可能快、尽可能简单地用软件解决实际问题,创造实际价值,在一个既定的秩序中寻找自己的生存空间,而不是异想天开地憧憬能成为Linus Torvalds式的旧秩序的“破坏者”。因此诸如软件过程、开发方法、系统集成、应用架构等“高级”的话题受到关注和欢迎,而诸如操作系统、编译原理之类的基础技术,已经是关心者寥寥了。他们非常怀疑,这样的一本书,对于一线的开发者是否有实际的意义?对于尚在寻找自己职业发展方向的初学者是否构成一种误导?这个问题相当尖锐,必须面对。我想这样一本书,至少在以下几个方面是具有重要的正面意义的。 首先,对于正在大学里学习计算机科学的学生来说,“操作系统原理”是重要的专业基础课。为了达到大学阶段教育的标准,这方面的知识应当认真学习。一些比较严肃的学校鼓励学生在学习这门课程的同时自己动手开发一个具体而微的操作系统。
第1章 马上动手写一个最小的“操作系统”1<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />
1.1 准备工作1
1.2 10分钟完成的操作系统1
1.3 Boot Sector3
1.4 代码解释3
1.5 水面下的冰山5
1.6 回顾6
第2章 搭建你的工作环境7
2.1 虚拟计算机(Virtual PC)7
<?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" />2.1.1 Virtual PC初体验8
2.1.2 创建你的第一个Virtual PC9
2.1.3 虚拟软盘研究12
2.1.4 虚拟软盘实战14
2.2 编译器(NASM & GCC)18
2.3 安装虚拟Linux19
2.4 在虚拟Linux上访问Windows文件夹26
2.5 安装虚拟PCDOS26
2.6 其他要素29
2.7 Bochs29
2.7.1 Bochs vs. Virtual PC vs. VMware30
2.7.2 Bochs的使用方法31
2.7.3 用Bochs进行调试33
2.7.4 在Linux上开发34
2.8 总结与回顾36
第3章 保护模式(Protect Mode)37
3.1 认识保护模式37
3.1.1 GDT(Global Descriptor Table) 42
3.1.2 实模式到保护模式,不一般的jmp45
3.1.3 描述符属性47
3.2 保护模式进阶50
3.2.1 海阔凭鱼跃50
3.2.2 LDT(Local Descriptor Table)58
3.2.3 特权级62
3.3 页式存储82
3.3.1 分页机制概述83
3.3.2 编写代码启动分页机制84
3.3.3 PDE和PTE85
3.3.4 cr388
3.3.5 回头看代码88
3.3.6 克勤克俭用内存90
3.3.7 进一步体会分页机制100
3.4 中断和异常107
3.4.1 中断和异常机制109
3.4.2 外部中断111
3.4.3 编程操作8259A113
3.4.4 建立IDT116
3.4.5 实现一个中断117
3.4.6 时钟中断试验119
3.4.7 几点额外说明121
3.5 保护模式下的I/O122
3.5.1 IOPL122
3.5.2 I/O许可位图(I/O Permission Bitmap)123
3.6 保护模式小结123
第4章 让操作系统走进保护模式125
4.1 突破512字节的限制125
4.1.1 FAT12126
4.1.2 DOS可以识别的引导盘131
4.1.3 一个最简单的Loader132
4.1.4 加载Loader入内存133
4.1.5 向Loader交出控制权142
4.1.6 整理boot.asm142
4.2 保护模式下的“操作系统”144
第5章 内核雏形146
5.1 用NASM在Linux下写Hello World146
5.2 再进一步,汇编和C同步使用148
5.3 ELF(Executable and Linkable Format)150
5.4 从Loader到内核155
5.4.1 用Loader加载ELF155
5.4.2 跳入保护模式161
5.4.3 重新放置内核170
5.4.4 向内核交出控制权175
5.4.5 操作系统的调试方法176
5.5 扩充内核184
5.5.1 切换堆栈和GDT184
5.5.2 整理我们的文件夹191
5.5.3 Makefile191
5.5.4 添加中断处理200
5.5.5 两点说明218
5.6 小结219
第6章 进程221
6.1 迟到的进程221
6.2 概述222
6.2.1 进程介绍222
6.2.2 未雨绸缪——形成进程的必要考虑222
6.2.3 参考的代码224
6.3 最简单的进程224
6.3.1 简单进程的关键技术预测225
6.3.2 第一步——ring0→ring1227
6.3.3 第二步——丰富中断处理程序243
6.3.4 进程体设计技巧254
6.4 多进程256
6.4.1 添加一个进程体256
6.4.2 相关的变量和宏257
6.4.3 进程表初始化代码扩充258
6.4.4 LDT260
6.4.5 修改中断处理程序261
6.4.6 添加一个任务的步骤总结263
6.4.7 号外:Minix的中断处理265
6.4.8 代码回顾与整理269
6.5 系统调用280
6.5.1 实现一个简单的系统调用280
6.5.2 get_ticks的应用286
6.6 进程调度292
6.6.1 避免对称——进程的节奏感292
6.6.2 优先级调度总结300
第7章 输入/输出系统302
7.1 键盘302
7.1.1 从中断开始——键盘初体验302
7.1.2 AT、PS/2键盘304
7.1.3 键盘敲击的过程304
7.1.4 解析扫描码309
7.2 显示器325
7.2.1 初识TTY325
7.2.2 基本概念326
7.2.3 寄存器328
7.3 TTY任务332
7.3.1 TTY任务框架的搭建334
7.3.2 多控制台340
7.3.3 完善键盘处理346
7.3.4 TTY任务总结354
7.4 区分任务和用户进程354
7.5 printf357
7.5.1 为进程指定TTY357
7.5.2 printf()的实现358
7.5.3 系统调用write()361
7.5.4 使用printf()363
用户1035191 2010-3-22 11:26