只用gcc实现底层纯汇编与c调用显示手写print函数,没有操作系统与c库

;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
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值