在C语言中,在调用一个函数func()时,如果func()接受不超过4个的参数,则调用者直接将实参从左到右依次赋值给r0, r1, r2, r3寄存器(如果是指针则传入地址),然后调用func()。
进入func()函数后,func()将自己的栈紧接着caller的栈顶向下扩展,从r0, r1, r2, r3寄存器中拿参数,然后开始做事。
而如果func()有超过4个参数,例如:
int func(int a, int *b, int c, int d, struct sh_info * e, int f) {}
则前4个参数仍然通过寄存器传递,其余参数e和f则放在栈里面让func()来取。就如下图:
也就是说,caller函数在调用func()之前,将参数e和f放到自己栈顶的位置。在编译阶段caller函数知道在函数栈里为这两个参数留出空间,而func()也根据自己参数列表的数量知道有8bytes的arg要从栈里面去拿。
所以存放参数的栈空间是由caller函数开辟的,因为caller才知道自己是如何传参的,尤其对于可变长参数的函数,如printf,caller函数将所有参数入栈,c

本文探讨了C语言中函数调用时的参数传递方式,特别是在使用ARM架构时,当函数参数不超过4个,它们会被依次放入r0至r3寄存器。若参数超过4个,额外的参数将被压入调用函数的栈中。在调用函数func()时,caller函数会为所有参数预留栈空间,而func()则根据其参数列表从栈中获取超出寄存器承载部分的参数。对于可变长参数的函数,如printf,caller函数会将所有参数入栈,callee函数则通过解析格式化字符串或使用va_arg获取参数。
最低0.47元/天 解锁文章
486

被折叠的 条评论
为什么被折叠?



