ARM 汇编与C之间 的调用

时间:2022-02-26 11:50:49

一. 汇编调用 C

  1. 初始化栈

  2. 初始化BSS段 (BSS 段是C语言存放未初始化的全局变量,或者初始化为0 的全局变量) 

  3 .使用 r0 ,r1, r2, r3 给函数传参,如果多于  4 个参数使用栈的方式传参

  4. 返回值放在 r0, r1 中,最大32位数据, 一般使用 r0 就可以满足要求

例子: 汇编文件 start.s

.text  @ 表示汇编文件开始

/****初始化栈*****/
ldr sp, =0x40000100
mov r0, #0x5
mov r1, #0x6
    
/***初始化bss段***/
@ bss段地址由链接时编译器分配
ldr r2, =__bss_start         @ BSS 起始地址
ldr r3, =__bss_end__         @ BSS 结束地址
mov r4, #0x00000000     

clbss_l:  
    teq r2, r3         @ r2, r3 中存放的是BSS段起始和结束地址
    strne r4, [r2]    @ 把r4 中的数据写入 r2中存放的地址中,teq结果不成立写入,也就是把bss段写0
    addne r2, r2, #1  @ 起始地址开始偏移一个字节,teq不成立的清苦下
    bne clbss_l   @ teq 不成立,表示还没有清除完bss 段,
    
    bl _main   @ 跳转到执行的C 函数名

.end @ 伪操作 , 表示汇编文件结束

main.c

int aa;
int _main(int a, int b)
{
	int t=0;
	t = a+ b;
	return t;
}

链接文件: map.lds

OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
/*OUTPUT_FORMAT("elf32-arm", "elf32-arm", "elf32-arm")*/
OUTPUT_ARCH(arm)
ENTRY(_start)
SECTIONS
{
    . = 0x00000000;
    . = ALIGN(4);
    .text      :
    {
        ./Objects/start.o(.text)
        *(.text)
    }
    . = ALIGN(4);
    .rodata : 
    { *(.rodata) }
    . = ALIGN(4);
    .data : 
    { *(.data) }
    . = ALIGN(4);
    __bss_start = .; 
    .bss :
     { *(.bss) }
    __bss_end__ = .;
}

测试: 申请一个全局变量,int aa; 占4字节的BSS 段所以起始和结束地址相差4 字节,所以会在清楚BSS 执行循环4次

ARM 汇编与C之间 的调用

 

二. C调用汇编

C 语言调用 汇编时 只要汇编实现的函数通过 .global 声明成全局函数即可在 C语言中调用

关于参数的传递:

  C 函数的参数,传递给汇编放在寄存器 r0,r1,r2,r3 中;汇编会使用 寄存器 r0 接受调用汇编函数的返回值

  

例子: start.s汇编文件

.text

/****初始化栈*****/
ldr sp, =0x40000100
mov r0, #0x5
mov r1, #0x6

/*   汇编跳转到 C 中   */
/*    初始化bss段    */
@ bss段地址由链接时编译器分配
ldr r2, =__bss_start         @ BSS 起始地址
ldr r3, =__bss_end__         @ BSS 结束地址
mov r4, #0x00000000     

clbss_l:  
    teq r2, r3         @ r2, r3 中存放的是BSS段起始和结束地址
    strne r4, [r2]    @ 把r4 中的数据写入 r2中存放的地址中,teq结果不成立写入,也就是把bss段写0
    addne r2, r2, #1  @ 起始地址开始偏移一个字节,teq不成立的清苦下
    bne clbss_l   @ teq 不成立,表示还没有清除完bss 段,
    
    bl _main   @ 跳转到执行的C 函数名 , lr 中会保存下一条指令的地址, 也就是
    b loop

/*    汇编函数, 在C 中调用   */
.global add_fun @ 声明为全局函数,使外部可调用
add_fun:
    add r2, r1, r0   @ 把 r1和r0中的数据相加 放在r2 中
    mov r0, r2    @ 使用寄存器 r0 存放调用汇编函数时返回的结果
    mov pc, lr    @ lr 寄存器中保存 C 跳转到汇编时的下一条指令地址 , 重新付给 PC 继续执行loop:
    b loop

.end

测试:

ARM 汇编与C之间 的调用