六、GNU汇编的编译
1. 不含lds文件的编译
假设我们有以下代码,包括1个main.c文件,1个start.s文件:start.s
.global _start
_start: @汇编入口
ldr sp,=0x41000000
b main
.global mystrcopy
.text
mystrcopy: //参数dest->r0,src->r2
LDRB r2, [r1], #1
STRB r2, [r0], #1
CMP r2, #0 //判断是不是字符串尾
BNE mystrcopy
MOV pc, lr
stop:
b stop @死循环,防止跑飞 等价于while(1)
.end @汇编程序结束
main.c
extern void mystrcopy(char *d,const char *s);
int main(void)
{
const char *src ="yikoulinux";
char dest[20]={};
mystrcopy(dest,src);//调用汇编实现的mystrcopy函数
while(1);
return 0;
}
Makefile编写方法如下:
1. TARGET=start
2. TARGETC=main
3. all:
4. arm-none-linux-gnueabi-gcc -O0 -g -c -o $(TARGETC).o $(TARGETC).c
5. arm-none-linux-gnueabi-gcc -O0 -g -c -o $(TARGET).o $(TARGET).s
6. #arm-none-linux-gnueabi-gcc -O0 -g -S -o $(TARGETC).s $(TARGETC).c
7. arm-none-linux-gnueabi-ld $(TARGETC).o $(TARGET).o -Ttext 0x40008000 -o $(TARGET).elf
8. arm-none-linux-gnueabi-objcopy -O binary -S $(TARGET).elf $(TARGET).bin
9. clean:
10. rm -rf *.o *.elf *.dis *.bin
Makefile含义如下:
定义环境变量TARGET=start,start为汇编文件的文件名定义环境变量TARGETC=main,main为c语言文件目标:all,4~8行是该指令的指令语句将main.c编译生成main.o,$(TARGETC)会被替换成main将start.s编译生成start.o,$(TARGET)会被替换成start4-5也可以用该行1条指令实现通过ld命令将main.o、start.o链接生成start.elf,-Ttext 0x40008000表示设置代码段起始地址为0x40008000通过objcopy将start.elf转换成start.bin文件,-O binary (或--out-target=binary) 输出为原始的二进制文件,-S (或 --strip-all)输出文件中不要重定位信息和符号信息,缩小了文件尺寸,clean目标clean目标的执行语句,删除编译产生的临时文件
【补充】
gcc的代码优化级别,在 makefile 文件中的编译命令4级 O0 -- O3 数字越大,优化程度越高。O3最大优化volatile作用volatile修饰的变量,编译器不再进行优化,每次都真正访问内存地址空间。2. 依赖lds文件编译
实际的工程文件,段复杂程度远比我们这个要复杂的多,尤其Linux内核有几万个文件,段的分布及其复杂,所以这就需要我们借助lds文件来定义内存的分布。
文件列表
main.c和start.s和上一节一致。
map.lds
OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
OUTPUT_FORMAT("elf32-arm", "elf32-arm", "elf32-arm")
OUTPUT_ARCH(arm)
ENTRY(_start)
SECTIONS
{
. = 0x40008000;
. = ALIGN(4);
.text :
{
.start.o(.text)
*(.text)
}
. = ALIGN(4);
.rodata :
{ *(.rodata) }
. = ALIGN(4);
.data :
{ *(.data) }
. = ALIGN(4);
.bss :
{ *(.bss) }
}