众所周知,早期的GNU汇编只能支持32代码的编写,不支持16为的代码。所以用GNU汇编写boot文件是很困难的事情。
后来GNU经过改进,终于支持16位的代码编写了,但是GNU汇编写的代码编译出的纯二进制文件还是默认为32位的。
例如一下代码:
.section .text
.globl _start
_start:
movl $0x10, %eax
保存为1.S
as -o 1.o 1.S
ld --oformat binary -o 1.bin 1.o
而后反汇编1.bin:ndisasm -b 16 1.bin
得出一下结果:
b81000 mov ax, 0x10
0000 add [bx+si], al
由此可见,一句汇编指令被拆分成两条了,原因是gnu默认支持的是32位代码。
我们编写的是16位代码,而gnu却以32为代码的模式去编译这个文件,所以会有上面的奇怪现象。
如果我们要编写16位代码,则必须在代码前面加上.code16
切换回32位代码则必须在代码前面加上.code32