このコードの周りで説明できない厄介なバグがあります:
unsigned char bitmap[K_BITMAP_SIZE] = {0} ;
SetBit(bitmap, K_18); // Sets the bit #18 to 1
for(size_t i = 0; i < K_END; ++i)
{
if(TestBit(bitmap, i)) // true for 18
{
size_t i2 = getData(i); // for 18, will return 15
SetBit(bitmap, i2); // BUG: IS SUPPOSED TO set the bit #15 to 1
}
}
- Visual C++2010で発生します
- 32ビットビルドと64ビットビルドの両方で発生します
- これは、リリースビルドでのみ発生します(「最大速度(/ O2)」が設定されている場合)
- 「サイズの最小化(/ O1)」が設定されたリリースビルドでのみ発生しません
- これは、getData関数の場合にのみVisual C ++ 2008で発生し
__forceinline
ます(デフォルトでは、VC ++ 2008はその関数をインライン化しませんが、VC ++ 2010はインライン化します) - これは、おそらくループ内の大規模なインライン化が原因で、以下に示すコードで発生します
- ループを削除して、興味深い値を直接設定した場合は発生しません(18)
ボーナス情報:
1-BenJは、この問題はVisual C ++ 2012には表示されない、つまりこれはコンパイラのバグである可能性があるとコメントしました
unsigned char
2- Test / Set / ResetBit関数にキャストを追加すると、バグもなくなります
size_t TestBit(const unsigned char * bits, size_t pos) { return (((bits)[(pos) >> 3]) & (1 << (unsigned char)((pos) & 7))) ; }
size_t SetBit(unsigned char * bits, size_t pos) { return (((bits)[(pos) >> 3]) |= (1 << (unsigned char)((pos) & 7))) ; }
size_t ResetBit(unsigned char * bits, size_t pos) { return (((bits)[(pos) >> 3]) &= ~(1 << (unsigned char)((pos) & 7))) ; }
質問は:
このバグは、コードが未定義の動作に依存しているために発生しますか、それともVC ++ 2010コンパイラにバグがありますか?
次のソースは自給自足であり、お気に入りのコンパイラでそのままコンパイルできます。
#include <iostream>
const size_t K_UNKNOWN = (-1) ;
const size_t K_START = (0) ;
const size_t K_12 = (K_START + 12) ;
const size_t K_13 = (K_START + 13) ;
const size_t K_15 = (K_START + 15) ;
const size_t K_18 = (K_START + 18) ;
const size_t K_26 = (K_START + 26) ;
const size_t K_27 = (K_START + 27) ;
const size_t K_107 = (K_START + 107) ;
const size_t K_128 = (K_START + 128) ;
const size_t K_END = (K_START + 208) ;
const size_t K_BITMAP_SIZE = ((K_END/8) + 1) ;
size_t TestBit(const unsigned char * bits, size_t pos) { return (((bits)[(pos) >> 3]) & (1 << ((pos) & 7))) ; }
size_t SetBit(unsigned char * bits, size_t pos) { return (((bits)[(pos) >> 3]) |= (1 << ((pos) & 7))) ; }
size_t ResetBit(unsigned char * bits, size_t pos) { return (((bits)[(pos) >> 3]) &= ~(1 << ((pos) & 7))) ; }
size_t getData(size_t p_value)
{
size_t value = K_UNKNOWN;
switch(p_value)
{
case K_13: value = K_12; break;
case K_18: value = K_15; break;
case K_107: value = K_15; break;
case K_27: value = K_26; break;
case K_128: value = K_12; break;
default: value = p_value; break;
}
return value;
}
void testBug(const unsigned char * p_bitmap)
{
const size_t byte = p_bitmap[1] ;
const size_t bit = 1 << 7 ;
const size_t value = byte & bit ;
if(value == 0)
{
std::cout << "ERROR : The bit 15 should NOT be 0" << std::endl ;
}
else
{
std::cout << "Ok : The bit 15 is 1" << std::endl ;
}
}
int main(int argc, char * argv[])
{
unsigned char bitmap[K_BITMAP_SIZE] = {0} ;
SetBit(bitmap, K_18);
for(size_t i = 0; i < K_END; ++i)
{
if(TestBit(bitmap, i))
{
size_t i2 = getData(i);
SetBit(bitmap, i2);
}
}
testBug(bitmap) ;
return 0;
}
いくつかの背景情報:最初は:
- Test / Set/ResetBit関数はマクロでした。
- 定数は定義されました
- インデックスは
long
またはint
(Windows 32ビットでは同じサイズ)のいずれかでした
必要に応じて、できるだけ早くいくつかの情報を追加します(たとえば、両方の構成用に生成されたアセンブラー、g ++が問題を処理する方法を更新します)。