以前也琢磨过OS源码,不过没整明白OS怎么在bochs上跑起来
我看的是《操作系统真相还原》,讲得还比较详细
使用的是2.6.9版本bochs
系统ubuntu
从sourceforge上下载bochs源代码
sudo apt-get install vgabios bochs-x bochsbios bochs-doc xorg-dev bochs-sdl #依赖关系
解压缩bochs后
./configure \
--prefix=/your/path/bochs \ 个人安装目录
--enable-debugger \ 打开bochs自己的调试器
--enable-disasm \ 使bochs支持反汇编
--enable-iodebug \ 启用io接口调试器
--enable-x86-debugger \ 支持x86调试器
--with-x \ 使用x windows
--with-x11 使用x11图形用户接口
make
make install
bochs配置文件:
因为用的是2.6.9版本,所以和书上不一样
放在安装目录下,bochsrc.disk:
###############################################################
# Configuration file for Bochs
###############################################################
# how much memory the emulated machine will have
megs: 32
# 对应真实机器的BIOS和VGA BIOS
romimage: file=/usr/share/bochs/BIOS-bochs-latest
vgaromimage: file=/usr/share/bochs/VGABIOS-lgpl-latest
# choose the boot disk.
# 默认是软盘,注释掉,改为disk
#boot: floppy
boot: disk
# where do we send log messages?
log: bochs.out
# disable the mouse
mouse: enabled=0
# enable key mapping, using US layout as default.
keyboard: keymap=/home/songyf/bochOS/share/bochs/keymaps/x11-pc-us.map
ata0: enabled=1, ioaddr1=0x1f0, ioaddr2=0x3f0, irq=14
#这一句是根据bximage生成的,后面会解释。
ata0-master: type=disk, path="/home/dante/workspace/os/hd60M.img", mode=flat, cylinders=121, heads=16, spt=63
利用自带工具bin/bximage创建虚拟硬盘
bin/bximage -hd -mode="flat" -size=60 -q hd60M.img
#这里创建失败了,直接运行bin/bximage按1按照提示一步步创建即可
此时硬盘里什么东西都没有,需要添加内容
mbr.S
;主引导程序
;------------------------------------------------------------
SECTION MBR vstart=0x7c00
mov ax,cs
mov ds,ax
mov es,ax
mov ss,ax
mov fs,ax
mov sp,0x7c00
; 清屏 利用0x06号功能,上卷全部行,则可清屏。
; -----------------------------------------------------------
;INT 0x10 功能号:0x06 功能描述:上卷窗口
;------------------------------------------------------
;输入:
;AH 功能号= 0x06
;AL = 上卷的行数(如果为0,表示全部)
;BH = 上卷行属性
;(CL,CH) = 窗口左上角的(X,Y)位置
;(DL,DH) = 窗口右下角的(X,Y)位置
;无返回值:
mov ax, 0x600
mov bx, 0x700
mov cx, 0 ; 左上角: (0, 0)
mov dx, 0x184f ; 右下角: (80,25),
; VGA文本模式中,一行只能容纳80个字符,共25行。
; 下标从0开始,所以0x18=24,0x4f=79
int 0x10 ; int 0x10
;;;;;;;;; 下面这三行代码是获取光标位置 ;;;;;;;;;
;.get_cursor获取当前光标位置,在光标位置处打印字符.
mov ah, 3 ; 输入: 3号子功能是获取光标位置,需要存入ah寄存器
mov bh, 0 ; bh寄存器存储的是待获取光标的页号
int 0x10 ; 输出: ch=光标开始行,cl=光标结束行
; dh=光标所在行号,dl=光标所在列号
;;;;;;;;; 获取光标位置结束 ;;;;;;;;;;;;;;;;
;;;;;;;;; 打印字符串 ;;;;;;;;;;;
;还是用10h中断,不过这次是调用13号子功能打印字符串
mov ax, message
mov bp, ax ; es:bp 为串首地址, es此时同cs一致,
; 开头时已经为sreg初始化
; 光标位置要用到dx寄存器中内容,cx中的光标位置可忽略
mov cx, 5 ; cx 为串长度,不包括结束符0的字符个数
mov ax, 0x1301 ; 子功能号13是显示字符及属性,要存入ah寄存器,
; al设置写字符方式 ah=01: 显示字符串,光标跟随移动
mov bx, 0x2 ; bh存储要显示的页号,此处是第0页,
; bl中是字符属性, 属性黑底绿字(bl = 02h)
int 0x10 ; 执行BIOS 0x10 号中断
;;;;;;;;; 打字字符串结束 ;;;;;;;;;;;;;;;
jmp $ ; 使程序悬停在此
message db "1 MBR" ; db,定义初始化常量值
times 510-($-$$) db 0
db 0x55,0xaa ; just the byte 0x55
将上述汇编代码用NASM编译成bin并放入虚拟硬盘中
bin是纯二进制文件(单片机有bin和hex两种文件格式)
elf或pe格式的二进制可执行文件,掺杂了程序的内存布局,位置等信息,是给操作系统的程序加载器用的。
nasm -o mbr.bin mbr.S #512字节
dd if=/your_path/mbr.bin of=/your_path/hd60M.img bs=512 count=1 conv=notrunc
#写入0盘0道1扇区
#参数,if:指定要读取的文件 of:目标文件 bs:指定输入块和输出块大小(字节) count:拷贝的块数 seek:块输出时跳过块大小 conv:在追加数据时,conv最好用notrunc方式,不打断文件
运行
bin/bochs -f bochsrc.disk
回车 c
出现绿色的“1 MBR”字样