《一个操作系统的实现》(五):1.ELF

首先展示一段汇编的“Hello, world!”的代码:

hello.asm

[section .data]	;数据在此
strHello	db	"Hello, world!", 0Ah
STRLEN		equ	$ - strHello

[section .text]	;代码在此

global _start	;必须导出_start这个入口以便链接器识别

_start:
	mov	edx, STRLEN
	mov	ecx, strHello
	mov	ebx, 1
	mov	eax, 4		;sys_write
	int	0x80		;系统调用
	mov	ebx, 0
	mov	eax, 1		;sys_exit
	int	0x80		;系统调用

用nasm编译:nasm -f elf hello.asm -o hello.o

其中-f elf指定输出文件格式为ELF格式

ld -s hello.o -o hello

其中-s可以去掉符号表等内容,可缩减生成的可执行代码

执行:./hello

输出Hello, world!

下面是汇编和C代码一块儿用的

foo.asm

extern choose	;int choose(int a, int b);

[section .data]	;数据在此

num1st	dd	3
num2nd	dd	4

[section .text]	;代码在此

global _start	;导出_start入口,以便让链接器识别
global myprint	;导出这个函数为了让bar.c使用

_start:
	push	dword [num2nd]
	push	dword [num1st]
	call	choose			;choose(num1st, num2nd);
	add		esp, 8

	mov	ebx, 0
	mov	eax, 1				;sys_exit
	int	0x80				;系统调用

;void myprint(char* msg, int len)
myprint:
	mov	edx, [esp+8]	;len
	mov	ecx, [esp+4]	;msg
	mov	ebx, 1
	mov eax, 4			;sys_write
	int	0x80			;系统调用
	ret

bar.c

void myPrint(char * msg, int len);

int choose(int a, int b)
{
	if(a >= b) {
		myprint("the first one\n", 13);
	}
	else {
		myprint("the second one\n", 13);
	}

	return 0;
}

编译和执行过程:

nasm -f elf -o foo.o foo.asm

gcc -c -o bar.o bar.c

ld -s -o foobar foo.o bar.o

./foobar

因为在bar.c中使用了foo.asm中的函数myprint,所以要用global导出。而foo.asm中调用了bar.c中的choose函数,所以要用extern声明。而且在参数入栈的时候,遵循C调用约定,后一个参数先入栈,并由调用者清理堆栈。


下面是ELF的内容——

ELF文件由4部分组成,分别是ELF头、程序头表、节、节头表。不过一个文件中不一定包含全部这些内容,而且只有ELF头的位置是固定的,其余各部分的位置、大小等信息由ELF头中的各项值来决定。ELF格式的详解见这篇文章,内容比较多...

ELF定义了一些数据类型(Elf32_Word、Elf32_Sword等),让文件格式与机器无关。(ELF文件力求支持从8位到32位不同架构的处理器)

程序头描述的是一个段在文件中的位置、大小以及它被放进内存后所在的位置和大小。这些信息正是我们想把一个文件加载进内存的重要信息。


这是第五章前三节的东西,内容比较少,后两节内容多一些

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值