有人可以提供一个示例,由于未对齐而将指针从一种类型转换为另一种类型失败吗?
在对这个答案的评论中,bothie 说做类似的事情
char * foo = ...;
int bar = *(int *)foo;
如果启用对齐检查,即使在 x86 上也可能导致错误。
通过在 GDB 中设置对齐检查标志后,我尝试产生错误条件set $ps |= (1<<18)
,但没有发生任何事情。
工作(即不工作;))示例是什么样的?
答案中的所有代码片段都不会在我的系统上失败 - 我稍后会在不同的编译器版本和不同的电脑上尝试它。
顺便说一句,我自己的测试代码看起来像这样(现在也使用 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
和movl $0xf1f2f3f4,-0x17(%ebp)
.
那么设置是否会AC
触发SIGBUS
?我无法让它在 Windows XP 下的英特尔双核笔记本电脑上运行,而我测试的任何 GCC 版本(MinGW-3.4.5、MinGW-4.3.0、Cygwin-3.4.4)都没有,而 codelogic 和 Jonathan Leffler提到x86上的失败......