У нас есть досадная ошибка, которую я не могу объяснить в этом фрагменте кода:
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
}
}
__forceinline
функция getData (по умолчанию VC ++ 2008 не включает эту функцию, а VC ++ 2010 -)Информация о бонусе:
1- BenJ прокомментировал, что проблема не появляется в Visual C ++ 2012, что означает, что это может быть ошибкой в компиляторе
2- Если мы добавим приведение к unsigned char
в функциях 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;
}
Некоторая справочная информация: Первоначально:
long
или же int
(в Windows 32-bit они имеют одинаковый размер)При необходимости я добавлю еще немного информации (например, сгенерированный ассемблер для обеих конфигураций, обновлю информацию о том, как g ++ решает проблему), как можно скорее.
Это ошибка оптимизатора кода. Он содержит как getData (), так и SetBit (). Комбинация кажется фатальной, она теряет значение 1 << ((Позы) & 7) и всегда выдает ноль.
Эта ошибка не возникает на VS2012. Обходной путь должен заставить одну из функций не быть встроенными. Учитывая код, вы, вероятно, хотите сделать это для getData ():
__declspec(noinline)
size_t getData(size_t p_value)
{
// etc..
}
Приложение 2
Наименьшая возможная часть кода ОП приведена ниже. Этот фрагмент приводит к указанной ошибке оптимизатора в VS2010 — зависит от содержимого inline-extended GetData()
, Даже если один объединяет два возвращения в GetData()
в одном ошибка «ушла». Кроме того, это не приводит к ошибке, если вы объединяете биты только в первом байте (например, char bitmap[1];
— вам нужно два байта).
Проблема не возникает под VS2012. Это ужасно, потому что MS исправила это в 2012 году, но не в 2010 году. WTF?
КСТАТИ:
Проверка ошибок оптимизатора VS2010:
#include <iostream>
const size_t B_5=5, B_9=9;
size_t GetBit(unsigned char * b, size_t p) { return b[p>>3] & (1 << (p & 7)); }
void SetBit(unsigned char * b, size_t p) { b[p>>3] |= (1 << (p & 7)); }
size_t GetData(size_t p) {
if (p == B_5) return B_9;
return 0;
}
/* SetBit-invocation will fail (write 0)
if inline-expanded in the vicinity of the GetData function, VS2010 */
int main(int argc, char * argv[])
{
unsigned char bitmap[2] = { 0, 0 };
SetBit(bitmap, B_5);
for(size_t i=0; i<2*8; ++i) {
if( GetBit(bitmap, i) ) // no difference if temporary variable used,
SetBit(bitmap, GetData(i)); // the optimizer will drop it anyway
}
const size_t byte=bitmap[1], bit=1<<1, value=byte & bit;
std::cout << (value == 0 ? "ERROR: The bit 9 should NOT be 0": "Ok: The bit 9 is 1") << std::endl;
return 0;
}
После некоторой проверки можно увидеть, что часть инициализации / обнуления не является частью этой конкретной проблемы.
Посмотрел еще раз после еды. Кажется, это ошибка распространения char / int. Можно вылечить, изменив функции маски (как уже было установлено ОП) на:
size_t TestBit (const unsigned char * bits, size_t pos) {
return (bits)[pos >> 3] & (1 << ( char(pos) & 7) ) ;
}
size_t SetBit (unsigned char * bits, size_t pos) {
return (bits)[pos >> 3] |= (1 << ( char(pos) & 7) ) ;
}
size_t ResetBit (unsigned char * bits, size_t pos) {
return (bits)[pos >> 3] &= ~(1 << ( char(pos) & 7) ) ;
}
бросая позицию int-size pos
до размера символа. Это заставит оптимизатор в VS2010 поступить правильно. Может быть, кто-то может прокомментировать.