;s与c,手写print函数,c调用
开发环境
win10系统
vmware12
其他都在vmware内
centos7 x86_64 虚拟机内,有桌面环境,看qemu
ac.s
如果加了
.= 0x7c00
会使编译的object增大至32kb,就是从0开始填充到0x7c00后的地址
.code16
.text
.extern toprint
.global print
.global _start
_start:
mov $0xe,%ah
mov $0x7,%bx
mov $0x43,%al
int $0x10
mov $0x3a,%al
int $0x10
call toprint
print:
push %bp
mov %sp,%bp
mov $0xe,%ah
mov $0x7,%bx
mov 4(%bp),%al
int $0x10
mov %bp,%sp
pop %bp
ret
ac.s–end
toprint.c
__asm__(".code16\n");
extern void print(const char* pStr);
void toprint(void)
{
const char *msg ="string in c!";
while(*msg)
{
print(*msg);
msg++;
}
for(;;){}
}
toprint.c–end
magic.asm
times 510-($-$$) db 0
db 0x55, 0xaa
magic.asm–end
创建1m的img镜像文件
qemu-img create -f qcow disk.img 1M
或者(等效)
dd if=/dev/zero of=disk.img bs=512 count=2048
编译magic.asm,生成1扇区,标准55aa,只是为了创建mbr第1扇区启动标志,也许可以使用ld script
nasm -f bin -o magic.bin magic.asm
dd conv=notrunc if=magic.bin of=disk.img
使用makefile编译ac.asm与toprint.c,链接生成ac的二进制写入创建的img第一扇区
如果生成的ac太大(一般小于512-2(55aa)-16(mbr分区表))就不能这样直接在启动扇区运行,需要bootloader加载后执行
Makefile ;all下面的行前是tab键
all:
gcc -ffreestanding -m32 -c ac.s
gcc -ffreestanding -m32 -c toprint.c
ld -Ttext 0x7c00 --oformat binary -m elf_i386 -s -o ac ac.o toprint.o
dd conv=notrunc if=ac of=disk.img
Makefile–end
执行编译
make
qemu虚拟机启动
qemu-system-x86_64 disk.img
看到输出c:string in c!
c:是ac.asm使用bios int10h中断输出
string in c!是ac.asm调用toprint.c的toprint函数,toprint再调用ac.asm中的print实现输出
如果只看到c:说明有问题可以使用gdb单指令instruction运行qemu
1.可能是最后形成的指令有前缀66,67导致指令出错,66,67主要是指令在16或32位模式使用寄存器或内存地址长度不同,导致实际指令错位
2.可能是ac.asm中的print函数接收参数的地址不对,mov al,[bp+4],涉及函数调用规则calling convention
查看生成的ac.o的反汇编指令
objdump -m i386 -D -Maddr16,data16,intel ac.o
查看生成的ac的反汇编指令
objdump -b binary -m i386 -D -Maddr16,data16,intel ac
调试qemu
qemu-system-x86_64 -s -S disk.img
新开一个终端,gdb
gdb -q
target remote :1234 连接qemu远程调试
set architecture i8086
set disassembly-flavor intel
display /5i $cs * 0x10 + $pc
b *0x7c00 设置断点
c 运行到断点0x7c00
运行一条指令
si