C语言可变参数的实现

x86函数调用惯例

在开始说明可变参数的实现之前,我们要说说x86函数调用的一些惯例。函数参数传递是从右向左的入栈顺序,在x86体系中,用户栈是从用户地址空间的顶端开始向下拓展的。

va_list

可变参数由类型 va_list,一组宏va_start,va_arg,va_end实现。宏的实现根据编译器而有所不同。
下面以VC6.0为例说明va_list的实现。

假设一个函数调用foo( foo(fmt, …) ),并向该函数传递参数fmt, arg2, arg1. 那么根据x86的函数调用惯例,该函数的栈帧如下图所示。那么如果我们要获取参数arg1和arg2,只需将fmt的地址增加相应的偏移量即可。

宏#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) 将指针指向可变参数集合的第一个参数。其接收两个参数ap是用来接受第一个参数地址的变量,v是函数参数栈顶的变量。

宏#define va_arg(ap,t) ( (t )( ( ap += _INTSIZEOF(t) ) - _INTSIZEOF(t)) ) 将ap指向下一个参数,并且获取指针当前指向的参数的值。

宏#define va_end(ap) ( ap = (va_list)0 ) 清空ap指针

这里写图片描述


typedef char * va_list; 

#define _INTSIZEOF(n) ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) ) 

#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) 

#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) 

#define va_end(ap) ( ap = (va_list)0 ) //清空va_list,即结束变参的获取

代码实例

static void err_sys(const char *fmt, ...)
{
        va_list ap;
        va_start(ap, fmt);
        err_doit(1, errno, ap, fmt);
        va_end(ap);
        exit(1);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值