;asm与c,手写print函数,c调用
开发环境
win10系统
vmware12
其他都在vmware内
centos7 x86_64 虚拟机内,有桌面环境,看qemu
ac.asm
[bits 16]
extern toprint ; void toprint(void) in c
[section .text]
global _start
global print
_start:
mov ah,0xe
mov bx,0x7
mov al,'c'
int 0x10
mov al,':'
int 0x10
call toprint
print:
push bp
mov bp,sp
mov ah,0xe
mov bx,0x7
mov al,[bp+4]
int 0x10
mov sp,bp
pop bp
ret
ac.asm–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:
nasm -f elf -o ac.o ac.asm
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