有人可以提供一个示例,将指针从一种类型转换为另一种类型由于未对齐而失败吗?
在评论中这个答案 https://stackoverflow.com/questions/544928/reading-integer-size-bytes-from-a-char-array/544964#544964,博蒂说做类似的事情
char * foo = ...;
int bar = *(int *)foo;
如果启用对齐检查,即使在 x86 上也可能会导致错误。
我试图通过设置对齐检查标志后产生错误条件set $ps |= (1<<18)
在GDB中,但什么也没发生。
工作(即非工作;))示例是什么样的?
答案中的代码片段在我的系统上都不会失败 - 我稍后将使用不同的编译器版本并在不同的电脑上尝试它。
顺便说一句,我自己的测试代码看起来像这样(现在也使用 asm 来设置AC
标志和未对齐的读写):
#include <assert.h>
int main(void)
{
#ifndef NOASM
__asm__(
"pushf\n"
"orl $(1<<18),(%esp)\n"
"popf\n"
);
#endif
volatile unsigned char foo[] = { 1, 2, 3, 4, 5, 6 };
volatile unsigned int bar = 0;
bar = *(int *)(foo + 1);
assert(bar == 0x05040302);
bar = *(int *)(foo + 2);
assert(bar == 0x06050403);
*(int *)(foo + 1) = 0xf1f2f3f4;
assert(foo[1] == 0xf4 && foo[2] == 0xf3 && foo[3] == 0xf2 &&
foo[4] == 0xf1);
return 0;
}
即使生成的代码肯定包含未对齐的访问,断言也可以顺利通过mov -0x17(%ebp), %edx
and movl $0xf1f2f3f4,-0x17(%ebp)
.
设置也会如此AC
触发一个SIGBUS
或不?我无法让它在 Windows XP 下的 Intel 双核笔记本电脑上运行,并且没有我测试过的 GCC 版本(MinGW-3.4.5、MinGW-4.3.0、Cygwin-3.4.4),而 codelogic 和 Jonathan Leffler提到 x86 上的失败...