gpt4 book ai didi

assembly - 如何在 ARM 汇编函数中访问 4 个以上的参数?

转载 作者:行者123 更新时间:2023-12-04 16:26:14 26 4
gpt4 key购买 nike

在我的汇编函数中,有 6 个参数。当我尝试访问第四个和第五个参数时,它们是错误的,这是我在 arm cortex-8a 上的代码

push {r4-r8,lr}
ldr r6, [sp]
ldr r7, [sp, #4]

我检查了 [sp] 内存,r4-r8 有错误的值。但是,如果有 3 个或更少的参数,[sp] 会给出正确的 r4-r8 值。我错过了什么吗?

最佳答案

为什么不试试呢?

unsigned int fun ( unsigned int, unsigned int, unsigned int, unsigned int, unsigned int );
unsigned int myfun ( void )
{
return(fun(1,2,3,4,5));
}

组装然后拆卸
> arm-none-eabi-gcc -O2 -c fun.c -o fun.o
> arm-none-eabi-objdump -D fun.o

汇编输出包含
00000000 <myfun>:
0: e52de004 push {lr} ; (str lr, [sp, #-4]!)
4: e3a03005 mov r3, #5
8: e24dd00c sub sp, sp, #12
c: e58d3000 str r3, [sp]
10: e3a01002 mov r1, #2
14: e3a02003 mov r2, #3
18: e3a03004 mov r3, #4
1c: e3a00001 mov r0, #1
20: ebfffffe bl 0 <fun>
24: e28dd00c add sp, sp, #12
28: e49de004 pop {lr} ; (ldr lr, [sp], #4)
2c: e12fff1e bx lr

正如预期的那样,前四个操作数在寄存器 r0-r3 中。尽管第五个操作数被放置在堆栈上。为什么编译器为操作数分配 12 个字节而不是 4 个字节,这是一个谜……也许看到该函数会更有意义:
unsigned int fun ( unsigned int a, unsigned int b, unsigned int c, unsigned int d, unsigned int e )
{
return(a+b+c+d-e);
}

组装和拆卸
arm-none-eabi-gcc -O2 -c fun.c -o fun.o
arm-none-eabi-objdump -D fun.o

00000000 <fun>:
0: e0811000 add r1, r1, r0
4: e0812002 add r2, r1, r2
8: e59d0000 ldr r0, [sp]
c: e0823003 add r3, r2, r3
10: e0600003 rsb r0, r0, r3
14: e12fff1e bx lr

所以被调用者只知道操作数是堆栈上的第一件事,而不关心调用者创建的堆栈帧。所以在这种情况下调用者分配 12 个字节而不是 4 个字节是一个谜。
arm-none-eabi-gcc --version
arm-none-eabi-gcc (GCC) 4.7.2
Copyright (C) 2012 Free Software Foundation, Inc.
This is free software; see the source for copying conditions. There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

看到编译器实际实现调用约定可以使调用约定本身的阅读更容易理解。或者,如果你在你感兴趣的编译器上为你感兴趣的特定函数原型(prototype)制作这样的例子,你不必阅读约定,你只需让你的调用者或被调用者无论你感兴趣,匹配编译器正在做什么为自己。

关于assembly - 如何在 ARM 汇编函数中访问 4 个以上的参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15071506/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com