通过纯汇编调用c语言代码实现手写print函数,没有操作系统与c库,真正底层开发

;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

你可能感兴趣的:(计算机)