我有一种情况,位掩码和/或类型转换无法按预期工作。我正在使用一个非常大的分布式系统,其中一些部分是用 C 编写的。
让我困惑的几行代码是:
uint16_t
mask(uint32_t nr)
{
return (uint16_t)(nr & 0x1ff);
}
...
int nr = 65536;
int index;
index = mask(nr);
...
在这种情况下,掩码的返回是 65536 而不是预期的 0。这个例子中有一些不一致的类型转换,但我不明白结果怎么可能是错误的。我们基本上屏蔽了 32 位整数的低 9 位,并将结果转换为 16 位整数。我最好的猜测是,有一些编译器优化会改变前 16 位,但对高 16 位没有任何作用,因为我们只想返回 16 位。由于在这种情况下“索引”是一个 32 位整数,这还包括“nr”的第 17 位,它没有在位掩码中修改,因此第 17 位仍然设置为 1 并将 65536 分配给索引.
有没有人对这种行为有解释?
编辑:我忘记了一些重要的参数。正确包含掩码函数的原型。该系统是为 Linux x86 构建的,带有 gcc 4.3.2。
编辑2:汇编代码:
0805b640 <mask>:
805b640: 55 push %ebp
805b641: 89 e5 mov %esp,%ebp
805b643: 83 ec 18 sub $0x18,%esp
805b646: 65 a1 14 00 00 00 mov %gs:0x14,%eax
805b64c: 89 45 fc mov %eax,0xfffffffc(%ebp)
805b64f: 31 c0 xor %eax,%eax
805b651: 8b 45 08 mov 0x8(%ebp),%eax
805b654: 66 25 ff 01 and $0x1ff,%ax
805b658: 8b 55 fc mov 0xfffffffc(%ebp),%edx
805b65b: 65 33 15 14 00 00 00 xor %gs:0x14,%edx
805b662: 75 02 jne 805b666 <maks+0x26>
805b664: c9 leave
805b665: c3 ret
805b666: e8 a5 ac 04 00 call 80a6310 <__stack_chk_fail_local>
805b66b: 90 nop
805b66c: 8d 74 26 00 lea 0x0(%esi),%esi