1.html
来自「linux 0.11中文版 有注释」· HTML 代码 · 共 387 行 · 第 1/2 页
HTML
387 行
<html>
<head>
<title>boot/bootsect.s</title>
<meta name='robots' content='noindex,nofollow'>
<meta name='generator' content='GLOBAL-5.4.1'>
</head>
<body text='#191970' bgcolor='#f5f5dc' vlink='gray'>
<a name='TOP'><h2><a href='../mains.html'>root</a>/<a href='../files/97.html'>boot</a>/bootsect.s</h2>
<i><font color='green'>/* [<][>][^][v][top]<a href='#BOTTOM'>[bottom]</a><a href='../mains.html'>[index]</a><a href='../help.html'>[help]</a> */</font></i>
<hr>
<pre>
<a name='L1'>!
<a name='L2'>! SYS_SIZE is the number of clicks (16 bytes) to be loaded.
<a name='L3'>! 0x3000 is 0x30000 bytes = 196kB, more than enough for current
<a name='L4'>! versions of linux ! SYS_SIZE 是要加载的节数(16 字节为1 节)。0x3000 共为
<a name='L5'>1 2 3 4 5 6
<a name='L6'>0x7c00
<a name='L7'>0x0000
<a name='L8'>0x90000
<a name='L9'>0x10000
<a name='L10'>0xA0000
<a name='L11'>system 模块
<a name='L12'>代码执行位置线路
<a name='L13'>0x90200
<a name='L14'>! 0x30000 字节=192 kB(上面Linus 估算错了),对于当前的版本空间已足够了。
<a name='L15'>!
<a name='L16'>SYSSIZE = 0x3000 ! 指编译连接后system 模块的大小。参见列表1.2 中第92 的说明。
<a name='L17'>! 这里给出了一个最大默认值。
<a name='L18'>!
<a name='L19'>! bootsect.s (C) 1991 Linus Torvalds
<a name='L20'>!
<a name='L21'>! bootsect.s is loaded at 0x7c00 by the bios-startup routines, and moves
<a name='L22'>! iself out of the way to address 0x90000, and jumps there.
<a name='L23'>!
<a name='L24'>! It then loads 'setup' directly after itself (0x90200), and the system
<a name='L25'>! at 0x10000, using BIOS interrupts.
<a name='L26'>!
<a name='L27'>! NOTE! currently system is at most 8*65536 bytes long. This should be no
<a name='L28'>! problem, even in the future. I want to keep it simple. This 512 kB
<a name='L29'>! kernel size should be enough, especially as this doesn't contain the
<a name='L30'>! buffer cache as in minix
<a name='L31'>!
<a name='L32'>! The loader has been made as simple as possible, and continuos
<a name='L33'>! read errors will result in a unbreakable loop. Reboot by hand. It
<a name='L34'>! loads pretty fast by getting whole sectors at a time whenever possible.
<a name='L35'>!
<a name='L36'>! 以下是前面这些文字的翻译:
<a name='L37'>! bootsect.s (C) 1991 Linus Torvalds 版权所有
<a name='L38'>!
<a name='L39'>! bootsect.s 被bios-启动子程序加载至0x7c00 (31k)处,并将自己
<a name='L40'>! 移到了地址0x90000 (576k)处,并跳转至那里。
<a name='L41'>!
<a name='L42'>! 它然后使用BIOS 中断将'setup'直接加载到自己的后面(0x90200)(576.5k),
<a name='L43'>! 并将system 加载到地址0x10000 处。
<a name='L44'>!
<a name='L45'>! 注意! 目前的内核系统最大长度限制为(8*65536)(512k)字节,即使是在
<a name='L46'>! 将来这也应该没有问题的。我想让它保持简单明了。这样512k 的最大内核长度应该
<a name='L47'>! 足够了,尤其是这里没有象minix 中一样包含缓冲区高速缓冲。
<a name='L48'>!
<a name='L49'>! 加载程序已经做的够简单了,所以持续的读出错将导致死循环。只能手工重启。
<a name='L50'>! 只要可能,通过一次取取所有的扇区,加载过程可以做的很快的。
<a name='L51'>
<a name='L52'>.globl begtext, begdata, begbss, endtext, enddata, endbss ! 定义了6 个全局标识符;
<a name='L53'>.text ! 文本段;
<a name='L54'>begtext:
<a name='L55'>.data ! 数据段;
<a name='L56'>begdata:
<a name='L57'>.bss ! 堆栈段;
<a name='L58'>begbss:
<a name='L59'>.text ! 文本段;
<a name='L60'>
<a name='L61'>SETUPLEN = 4 ! nr of setup-sectors
<a name='L62'>! setup 程序的扇区数(setup-sectors)值;
<a name='L63'>BOOTSEG = 0x07c0 ! original address of boot-sector
<a name='L64'>! bootsect 的原始地址(是段地址,以下同);
<a name='L65'>INITSEG = 0x9000 ! we move boot here - out of the way
<a name='L66'>! 将bootsect 移到这里 -- 避开;
<a name='L67'>SETUPSEG = 0x9020 ! setup starts here
<a name='L68'>! setup 程序从这里开始;
<a name='L69'>SYSSEG = 0x1000 ! system loaded at 0x10000 (65536).
<a name='L70'>! system 模块加载到0x10000(64 kB)处;
<a name='L71'>ENDSEG = SYSSEG + SYSSIZE ! where to stop loading
<a name='L72'>! 停止加载的段地址;
<a name='L73'>
<a name='L74'>! ROOT_DEV: 0x000 - same type of floppy as boot.
<a name='L75'>! 根文件系统设备使用与引导时同样的软驱设备;
<a name='L76'>! 0x301 - first partition on first drive etc
<a name='L77'>! 根文件系统设备在第一个硬盘的第一个分区上,等等;
<a name='L78'>ROOT_DEV = 0x306 ! 指定根文件系统设备是第2 个硬盘的第1 个分区。这是Linux 老式的硬盘命名
<a name='L79'>! 方式,具体值的含义如下:
<a name='L80'>! 设备号=主设备号*256 + 次设备号(也即dev_no = (major<<8) + minor )
<a name='L81'>! (主设备号:1-内存,2-磁盘,3-硬盘,4-ttyx,5-tty,6-并行口,7-非命名管道)
<a name='L82'>! 0x300 - /dev/hd0 - 代表整个第1 个硬盘;
<a name='L83'>! 0x301 - /dev/hd1 - 第1 个盘的第1 个分区;
<a name='L84'>! …
<a name='L85'>! 0x304 - /dev/hd4 - 第1 个盘的第4 个分区;
<a name='L86'>! 0x305 - /dev/hd5 - 代表整个第2 个硬盘盘;
<a name='L87'>! 0x306 - /dev/hd6 - 第2 个盘的第1 个分区;
<a name='L88'>! …
<a name='L89'>! 0x309 - /dev/hd9 - 第2 个盘的第4 个分区;
<a name='L90'>! 从linux 内核0.95 版后已经使用与现在相同的命名方法了。
<a name='L91'>
<a name='L92'>entry start ! 告知连接程序,程序从start 标号开始执行。
<a name='L93'>start: ! 47--56 行作用是将自身(bootsect)从目前段位置0x07c0(31k)
<a name='L94'>! 移动到0x9000(576k)处,共256 字(512 字节),然后跳转到
<a name='L95'>! 移动后代码的go 标号处,也即本程序的下一语句处。
<a name='L96'>mov ax,#BOOTSEG ! 将ds 段寄存器置为0x7C0;
<a name='L97'>mov ds,ax
<a name='L98'>mov ax,#INITSEG ! 将es 段寄存器置为0x9000;
<a name='L99'>mov es,ax
<a name='L100'>mov cx,#256 ! 移动计数值=256 字;
<a name='L101'>sub si,si ! 源地址 ds:si = 0x07C0:0x0000
<a name='L102'>sub di,di ! 目的地址 es:di = 0x9000:0x0000
<a name='L103'>rep ! 重复执行,直到cx = 0
<a name='L104'>movw ! 移动1 个字;
<a name='L105'>jmpi go,INITSEG ! 间接跳转。这里INITSEG 指出跳转到的段地址。
<a name='L106'>go: mov ax,cs ! 将ds、es 和ss 都置成移动后代码所在的段处(0x9000)。
<a name='L107'>mov ds,ax !由于程序中有堆栈操作(push,pop,call),因此必须设置堆栈。
<a name='L108'>mov es,ax
<a name='L109'>! put stack at 0x9ff00. ! 将堆栈指针sp 指向0x9ff00(即0x9000:0xff00)处
<a name='L110'>mov ss,ax
<a name='L111'>mov sp,#0xFF00 ! arbitrary value >>512
<a name='L112'>! 由于代码段移动过了,所以要重新设置堆栈段的位置。
<a name='L113'>! sp 只要指向远大于512 偏移(即地址0x90200)处
<a name='L114'>! 都可以。因为从0x90200 地址开始处还要放置setup 程序,
<a name='L115'>! 而此时setup 程序大约为4 个扇区,因此sp 要指向大
<a name='L116'>! 于(0x200 + 0x200 * 4 + 堆栈大小)处。
<a name='L117'>
<a name='L118'>! load the setup-sectors directly after the bootblock.
<a name='L119'>! Note that 'es' is already set up.
<a name='L120'>! 在bootsect 程序块后紧根着加载setup 模块的代码数据。
<a name='L121'>! 注意es 已经设置好了。(在移动代码时es 已经指向目的段地址处0x9000)。
<a name='L122'>
<a name='L123'>load_setup:
<a name='L124'>! 68--77 行的用途是利用BIOS 中断INT 0x13 将setup 模块从磁盘第2 个扇区
<a name='L125'>! 开始读到0x90200 开始处,共读4 个扇区。如果读出错,则复位驱动器,并
<a name='L126'>! 重试,没有退路。INT 0x13 的使用方法如下:
<a name='L127'>! 读扇区:
<a name='L128'>! ah = 0x02 - 读磁盘扇区到内存;al = 需要读出的扇区数量;
<a name='L129'>! ch = 磁道(柱面)号的低8 位; cl = 开始扇区(0-5 位),磁道号高2 位(6-7);
<a name='L130'>! dh = 磁头号; dl = 驱动器号(如果是硬盘则要置位7);
<a name='L131'>! es:bx ??指向数据缓冲区; 如果出错则CF 标志置位。
<a name='L132'>mov dx,#0x0000 ! drive 0, head 0
<a name='L133'>mov cx,#0x0002 ! sector 2, track 0
<a name='L134'>mov bx,#0x0200 ! address = 512, in INITSEG
<a name='L135'>mov ax,#0x0200+SETUPLEN ! service 2, nr of sectors
<a name='L136'>int 0x13 ! read it
<a name='L137'>jnc ok_load_setup ! ok - continue
<a name='L138'>mov dx,#0x0000
<a name='L139'>mov ax,#0x0000 ! reset the diskette
<a name='L140'>int 0x13
<a name='L141'>j load_setup
<a name='L142'>
<a name='L143'>ok_load_setup:
<a name='L144'>
<a name='L145'>! Get disk drive parameters, specifically nr of sectors/track
<a name='L146'>! 取磁盘驱动器的参数,特别是每道的扇区数量。
<a name='L147'>! 取磁盘驱动器参数INT 0x13 调用格式和返回信息如下:
<a name='L148'>! ah = 0x08 dl = 驱动器号(如果是硬盘则要置位7 为1)。
<a name='L149'>! 返回信息:
<a name='L150'>! 如果出错则CF 置位,并且ah = 状态码。
<a name='L151'>! ah = 0, al = 0, bl = 驱动器类型(AT/PS2)
<a name='L152'>! ch = 最大磁道号的低8 位,cl = 每磁道最大扇区数(位0-5),最大磁道号高2 位(位6-7)
<a name='L153'>! dh = 最大磁头数, dl = 驱动器数量,
<a name='L154'>! es:di -?? 软驱磁盘参数表。
<a name='L155'>
<a name='L156'>mov dl,#0x00
<a name='L157'>mov ax,#0x0800 ! AH=8 is get drive parameters
<a name='L158'>int 0x13
<a name='L159'>mov ch,#0x00
<a name='L160'>seg cs ! 表示下一条语句的操作数在cs 段寄存器所指的段中。
<a name='L161'>mov sectors,cx ! 保存每磁道扇区数。
<a name='L162'>mov ax,#INITSEG
<a name='L163'>mov es,ax ! 因为上面取磁盘参数中断改掉了es 的值,这里重新改回。
<a name='L164'>
<a name='L165'>! Print some inane message ! 在显示一些信息('Loading system ...'回车换行,共24 个字符)。
<a name='L166'>
<a name='L167'>mov ah,#0x03 ! read cursor pos
<a name='L168'>xor bh,bh ! 读光标位置。
<a name='L169'>int 0x10
<a name='L170'>
<a name='L171'>mov cx,#24 ! 共24 个字符。
<a name='L172'>mov bx,#0x0007 ! page 0, attribute 7 (normal)
<a name='L173'>mov bp,#msg1 ! 指向要显示的字符串。
<a name='L174'>mov ax,#0x1301 ! write string, move cursor
<a name='L175'>int 0x10 ! 写字符串并移动光标。
<a name='L176'>
<a name='L177'>! ok, we've written the message, now
<a name='L178'>! we want to load the system (at 0x10000) ! 现在开始将system 模块加载到0x10000(64k)处。
<a name='L179'>
<a name='L180'>mov ax,#SYSSEG
<a name='L181'>mov es,ax ! segment of 0x010000 ! es = 存放system 的段地址。
<a name='L182'>call read_it ! 读磁盘上system 模块,es 为输入参数。
<a name='L183'>call kill_motor ! 关闭驱动器马达,这样就可以知道驱动器的状态了。
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?