Сегодня я столкнулся с некоторым кодом, который примерно похож на следующий фрагмент. И valgrind
, и UndefinedBehaviorSanitizer
обнаружены чтения неинициализированных данных.
template <typename T>
void foo(const T& x)
{
static_assert(std::is_pod_v<T> && sizeof(T) > 1);
auto p = reinterpret_cast<const char*>(&x);
std::size_t i = 1;
for(; i < sizeof(T); ++i)
{
if(p[i] != p[0]) { break; }
}
// ...
}
Вышеупомянутые инструменты жаловались на сравнение p[i] != p[0]
, когда
объект, содержащий байты заполнения, был передан в foo
. Пример:
struct obj { char c; int* i; };
foo(obj{'b', nullptr});
Это поведение undefined для чтения байтов заполнения из типа POD и сравнения их с чем-то еще? Я не смог найти окончательный ответ ни в стандарте, ни в StackOverflow.