[windows下32位移植到64位]

情景

一般情况下,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_ttime_tptrdiff_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位
GetClassgLongGetClassLongPtr
SetWindowLongSetWindowLongPtr
......


移植步骤

  1. 使用64位编译选项进行编译;
  2. 消除错误,可能有些函数已经变化了;
  3. 要重视警告,可能出现截取的情况;
  4. 链接的时候要用64位的库;
  5. 测试,检测运行时错误,主要检查会不会发现截取的情况,还有要注意对齐问题;


参考文章

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


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值