Amd64 and Va_arg
Made of Bugs
Amd64 and Va_arg
OCT 3RD, 2010
A while back, I was poking around LLVM bugs, and discovered, to my surprise, that LLVM doesn't supportthe va_arg
intrinsic, used by functions to accept multiple arguments, at all on amd64. It turns out that clang
and llvm-gcc
, the compilers that backend to LLVM, have their own implementations in the frontend, so this isn't as big a deal as it might sound, but it was still a surprise to me.
Figuring that this might just be something no one got around to, and couldn't actually be that hard, I pulled out my copy of the amd64 ABI specification, figuring that maybe I could throw together a patch and fix this issue.
Maybe half an hour of reading later, I stopped in terror and gave up, repenting of my foolish ways to go work on something else. va_arg
on amd64 is a hairy, hairy beast, and probably not something I was going to hack together in an evening. And so instead I decided to blog about it.
The problem: Argument passing on amd64
On i386, because of the dearth of general-purpose registers, the calling convention passes all arguments on the stack. This makes the va_arg implementation easy – A va_list
is simply a pointer into the stack, and va_arg
just adds the size of the type to be retrieved to the va_list
, and returns the old value. In fact, the i386 ABI reference simply specifies va_arg
in terms of a single line of code:
#define va_arg(list, mode) ((mode *)(list = (char *)list + sizeof(mode)))[-1]
On amd64, the problem is much more complicated. To start, amd64 specifies that up to 6 integer arguments and up to 8 floating-point arguments are passed to functions in registers, to take advantage of amd64's larger number of registers. So, for a start, va_arg
will have to deal with the fact that some arguments may have been passed in registers, and some on the stack.
(One could imagine simplifying the problem by stipulating a different calling convention for variadic functions, but unfortunately, for historical reasons and otherwise, C requires that code be able to call functions even if their prototype is not visible, which means the compiler doesn't necessarily know if it's calling a variadic function at any given call site. [edited to add: caf points out in the comments that C99 actually explicitly does not require this property. But I speculate that the ABI designers wanted to preserve this property from i386 because it has historically worked, and so existing code depended on it]).
That's not all, however. Not only can integer arguments be passed by registers, but small struct
s (16 bytes or fewer) can also be passed in registers. A sufficiently small struct, for the purposes of the calling convention, is essentially broken up into its component members, which are passed as though they were separate arguments – unless only some of them would fit into registers, in which case the whole struct is passed on the stack.
So va_arg
, given a struct
as an argument, has to be able to figure out whether it was passed in registers or on the stack, and possibly even re-assemble it into temporary space.
The implementation
Given all those constraints, the required implementation is fairly straightforward, but incredibly complex compared to any other platform I know of.
To start, any function that is known to use va_start
is required to, at the start of the function, save all registers that may have been used to pass arguments onto the stack, into the "register save area", for future access by va_start
and va_arg
. This is an obvious step, and I believe pretty standard on any platform with a register calling convention. The registers are saved as integer registers followed by floating point registers. As an optimization, during a function call, %rax
is required to hold the number of SSE registers used to hold arguments, to allow a varargs caller to avoid touching the FPU at all if there are no floating point arguments.
va_list
, instead of being a pointer, is a structure that keeps track of four different things:
typedef struct {
unsigned int gp_offset;
unsigned int fp_offset;
void *overflow_arg_area;
void *reg_save_area;
} va_list[1];
reg_save_area
points at the base of the register save area initialized at the start of the function. fp_offset
and gp_offset
are offsets into that register save area, indicating the next unused floating point and general-purpose register, respectively. Finally, overflow_arg_area
points at the next stack-passed argument to the function, for arguments that didn't fit into registers.
Here's an ASCII art diagram of the stack frame during the execution of a varargs function, after the register save area has been established. Note that the spec allows functions to put the register save area anywhere in its frame it wants, so I've shown potential storage both above and below it.
| ... | [high addresses]
+----------------+
| argument |
| passed |
| on stack (2) |
+----------------+ <---- overflow_arg_area
| argument |
| passed |
| on stack (1) |
+----------------+
| return address |
+----------------+
| ... | (possible local storage for func)
+----------------+
| %xmm15 | \
+----------------+ |
| %xmm14 | | ___
+----------------+ | |
| ... | \ register
+----------------+ }save|
| %xmm0 | / area|
+----------------+ | |
| %r9 | | |
+----------------+ | | fp_offset
| %r8 | | ___ |
+----------------+ | | |
| ... | | | |
+----------------+ | | gp_offset
| %rsi | | | |
+----------------+ | | |
| %rdi | / | |
+----------------+ <----+--+--- reg_save_area
| ... | (potentially more storage)
+----------------+ <----------- %esp
| ... | [low addresses]
Because va_arg
must tell determine whether the requested type was passed in registers, it needs compiler support, and can't be implemented as a simple macro like on i386. The amd64 ABI reference specifies va_arg
using a list of eleven different steps that the macro must perform. I'll try to summarize them here.
First off, va_arg
determines whether the requested type could be passed in registers. If not, va_arg
behaves much like it does on i386
, using the overflow_arg_area
member of the va_list
(Plus some complexity to deal with alignment values).
Next, assuming the argument can be passed in registers, va_arg
determines how many floating-point and general-purpose registers would be used to pass the requested type. It compares those values with the gp_offset
and fp_offset
fields in the va_list
. If the additional registers would cause either value to overflow the number of registers used for parameter-passing for that type, then the argument was passed on the stack, and va_arg
bails out and uses overflow_arg_area
.
If we've made it this far, the argument was passed in registers. va_arg
fetches the argument using reg_save_area
and the appropriate offsets, and then updates gp_offset
and fp_offset
as appropriate.
Note that if the argument was passed in a mix of floating-point and general-purpose registers, or requires a large alignment, this means that va_arg
must copy it out of the register save area onto temporary space in order to assemble the value.
So, in the worst case, va_arg
on a type that embeds both a floating-point and an integer type must do two comparisons, a conditional branch, and then update two fields in the va_list
and copy multiple values out of the register save area into a temporary object to return. That's quite a lot more work than the i386 version does. Note that I don't mean to suggest this is a performance concern – I don't have any benchmarks to back this up, but I would be shocked if this is measurable in any reasonable code. But I was surprised by how complex this operation is.
Amd64 and Va_arg的更多相关文章
- AMD64和i386的区别
下载Debian系统时,出现两个选项:ADM64和i386,那么这两者的区别是什么? i386=Intel 80386.其实i386通常被用来作为对Intel(英特尔)32位微处理器的统称. AMD6 ...
- debian7 请把标有“Debian GNU/Linux 7.1.0 _Wheezy_ - Official amd64 DVD Binary-1 20130615-23:06”的盘片插入驱动器“/media/cdrom/”再按回车键
有时候,在通过apt-get install 安装软件的时候,会出现: 更换介质:请把标有“Debian GNU/Linux 7.1.0 _Wheezy_ - Official amd64 DVD B ...
- AMD64与IA64的区别
其实很多人从字面上,都以为AMD64就是针对AMD CPU的,IA64是针对INTEL CPU,其实是错的,我最初也是这样认为,其实不然: 你在市面上买的到的intel 64位 CPU都属于amd64 ...
- C++省略参数(va_list va_start va_arg va_end)的简单应用
原文参考自:http://www.cnblogs.com/hanyonglu/archive/2011/05/07/2039916.html #include <iostream> #in ...
- va_list/va_start/va_arg/va_end深入分析【转】
转自:http://www.cnblogs.com/justinzhang/archive/2011/09/29/2195969.html va_list/va_start/va_arg/va_end ...
- 对C语言中va_list,va_start,va_arg和va_end的一点理解
这几个函数和变量是针对可变参数函数的,什么是可变参数函数呢,最经典的莫过于printf和scanf,这两个函数的声明如下: int printf(const char *format, ...); i ...
- Compiling Xen-4.4 From Source And Installing It On Ubuntu Server (Amd-64)
First of all, you should install a clean Ubuntu Server (Amd-64) on your server. (Version 14.04 is st ...
- 可变长参数列表误区与陷阱——va_arg不可接受的类型
可变长参数列表误区与陷阱--va_arg不可接受的类型 实现一个有可变长参数列表函数的时候,会使用到stdarg.h(这里不讨论varargs.h)中提供的宏. 例如,我们要实现一个简易的my_pri ...
- i386 和amd64 的意思
首先可以简化一个概念,i386=Intel 80386.其实i386通常被用来作为对Intel(英特尔)32位微处理器的统称. Windows NT类系统的安装盘上,通常i386是其根上的一个文件夹, ...
随机推荐
- Galera集群server.cnf参数调整--前言
文档安排: 前言部分会简述下galera集群,正文中会针对我们线上的环境,在不断业务的情况下,进行参数调整的话,有些参数不能够进行配置,会以#***的形式写入配置文件中,文档也会进行进一步说明. 如果 ...
- git常用功能
- JavaScript 面向对象(二) —— 案例篇
看案例前可以先看看基础篇:JavaScript 面向对象(一) —— 基础篇 案例——面向对象的选项卡:把面向过程的程序一步步改成面向对象的形式,使其能够更加的通用(但是通用的东西,一般会比较臃肿). ...
- 隔壁信概大作业xjb写——同化棋ATAXX
话说泥萌北大信科啊,助教是有多懒...去年黑白棋今年同化棋,顺带打ai都不用自己写标程... 好吧..我知道泥萌重点在各种sb的辅助操作上..什么悲剧的可以随时暂停载入...有毒吧 [据说泥萌上课没讲 ...
- Gulp的使用教程
- Oracle查询
1.普通查询 select * from 表格 查询所有内容 select 列名,列名 from 表格查询某几列 2.条件查询 select * from 表格 where 条件 条件查询 selec ...
- JSON简单介绍
//JSON是一种数据格式//JSON比较像php里面的关联数组,它里面存的内容也是key和value成对存在的 JSON写法格式 var js = { "one":"h ...
- pct xcode7
1.) 打开你的Xcode工程. 在Supporting Files目录下,选择 File > New > File > iOS > Other > PCH File 然 ...
- 【Python扩展阅读【转】】字符串的方法及注释
capitalize() 把字符串的第一个字符改为大写 casefold() 把整个字符串的所有字符改为小写 center(width) 将字符串居中,并使用空格填充至长度wi ...
- 爬虫:selenium + phantomjs 解决js抓取问题(一)
selenium模块主要用来做测试,模拟键盘.鼠标来操作浏览器. phantomjs 就像一个无界面的浏览器一样. 两个结合能很好的解决js抓取的问题. 测试代码: #coding=utf-8 fro ...