Index: test/asan/TestCases/speculative_load.cc =================================================================== --- /dev/null +++ test/asan/TestCases/speculative_load.cc @@ -0,0 +1,50 @@ +// Verifies that speculative loads from unions do not happen under asan. +// RUN: %clangxx_asan -O0 %s -o %t && %run %t 2>&1 +// RUN: %clangxx_asan -O1 %s -o %t && %run %t 2>&1 +// RUN: %clangxx_asan -O2 %s -o %t && %run %t 2>&1 +// RUN: %clangxx_asan -O3 %s -o %t && %run %t 2>&1 + +#include + +struct S { + struct _long { + void* _pad; + const char* _ptr; + }; + + struct _short { + unsigned char _size; + char _ch[23]; + }; + + union { + _short _s; + _long _l; + } _data; + + S() { + _data._s._size = 0; + __asan_poison_memory_region(_data._s._ch, 23); + } + + bool is_long() const { + return _data._s._size & 1; + } + + const char* get_pointer() const { + return is_long() ? _data._l._ptr : _data._s._ch; + } +}; + + +inline void side_effect(const void *arg) { + __asm__ __volatile__("" : : "r" (arg) : "memory"); +} + +int main(int argc, char **argv) { + S s; + side_effect(&s); // optimizer is too smart otherwise + const char *ptr = s.get_pointer(); + side_effect(ptr); // force use ptr + return 0; +}