情景
一般情况下,32位程序是能在64位机子上正常运行的,64位对32位兼容,通过WOW64机制加载32位程序,不需要修改代码就行了;但不能使用64位的性能。如果你要使用64位的性能就要对32位代码进行移植了,如下面情况:
- 需要多于4GB的内存
- 使用的文件大小常大于2GB
- 密集浮点运算,需要利用64位架构的优势
- 能从64位平台的优化数学库中受益
运行64位配制
64位操作系统+64位cpu。
64位模式
数据模型
32位环境涉及"ILP32"数据模型,是因为C数据类型为32位的int、long、指针;现今所有64位的类Unix平台均使用LP64数据模型,而64位Windows使用LLP64数据模型。
"LP64"数据模型的long和指针已为64位。
LLP64数据模型指针为64位,LLP64指long long和pointer是64-bit的。
数据类型
注意:
- 在64位系统里,size_t、time_t 和 ptrdiff_t 是 64 位值;
- time_t在vs2005版本是32位,在vs2005更高版本为64位;
- 从 32 位迁移到 64 位时,增长的主要类型是指针和派生数据类型,如句柄。在 Windows 64 位中,目前的指针和派生类型是 64 位 long 类型。大小增加的其他一些类型还有:WPARAM、LPARAM、LRESULT 和 SIZE_T。其中一个原因是,它们作为参数使用,并且某些函数将指针作为参数使用
移植存在的问题
硬编码的数据大小
有些类型大小使用固定数字,比如int *ptr = (int *)malloc(32),在32位程序下,得到32位的指针,但在64位程序下,就被截取了。
常量有效性问题
那些以十六进制或二进制表示的常量,通常都是32位的。例如,无符号32位常量0xFFFFFFFF通常用来测试是否为-1;
#define INVALID_POINTER_VALUE 0xFFFFFFFF
然而,在64位系统中,这个值不是-1,而是4294967295;在64位系统中,-1正确的值应为0xFFFFFFFFFFFFFFFF。要避免这个问题,在声明常量时,使用const,并且带上signed或unsigned。
例如:
const signed int INVALID_POINTER_VALUE =0xFFFFFFFF;
上面一行代码将会在32位和64位系统上都运行正常。或者,根据需要适当地使用 “L” 或 “U” 来声明整型常量。
又比如对最高位的设置,通常我们的做法是定义如下的常量0x80000000,但是可移植性更好的方法是使用一个位移表达式:1L << ((sizeof(long) * 8) - 1);
int和指针混用
在64位系统里指针是64位,而在32位系统里是32位。如:
int *ptr;
int i;
ptr = (int *) i;
这样是会截取掉,应该使用指针精度数据类型,比如:
INT_PTR | 指针精度的有符号整型 |
LONG_PTR | 指针精度的有符号长类型 |
对齐
- 对齐问题影响性能,对不对齐的数据进行操作需要执行更多的存储访问操作,应该对结构的域重新安排顺序,按照域的大小来排序,最大放在前面,最小放在后面;
- 会对结构中域的位置的偏移,不能使用偏移量硬编码;
格式化输入输出
64位的数据用32的格式输出是有问题,要进行修改。
格式化情景 | 32位 | 64位 |
十六进制的输出,如打印的地址 | %x | %p |
long long或者指针整形输出 | %d | %I64d |
缺少原型的截断
缺少原型是指之前的编译器允许使用那些使用之前末声明的函数,但后面定义了。类型检测的时候会把返回值默认成int类型,如果实际返回的是long long 或者指针类型,那么会被截取。目前新的编译器这种写法是会报错的。
函数变换
32位 | 64位 |
GetClassgLong | GetClassLongPtr |
SetWindowLong | SetWindowLongPtr |
... | ... |
移植步骤
- 使用64位编译选项进行编译;
- 消除错误,可能有些函数已经变化了;
- 要重视警告,可能出现截取的情况;
- 链接的时候要用64位的库;
- 测试,检测运行时错误,主要检查会不会发现截取的情况,还有要注意对齐问题;
参考文章
http://www.microsoft.com/china/MSDN/library/Windev/64bit/MW6TWPchapter5.mspx?mfr=true
http://www.cnblogs.com/shiney/archive/2011/06/23/2088103.html
http://msdn.microsoft.com/zh-cn/library/3b2e7499%28v=vs.90%29.aspx
http://www.ibm.com/developerworks/cn/linux/l-port64.html
http://wenku.baidu.com/link?url=d5F9Nxja2rYFM3ubjzbrCyv9NjtJRsaScPavAP3rIvl085_PDEkLYHe1RnFgnGKZvFQND9xJdzurjyrH3sjSto-T41n1fR7Kw7w72dLSrtK