| 1 | // RUN: %clangxx_tysan -O0 %s -c -o %t.o |
| 2 | // RUN: %clangxx_tysan -O0 %s -DPMAIN -c -o %tm.o |
| 3 | // RUN: %clangxx_tysan -O0 %t.o %tm.o -o %t |
| 4 | // RUN: %run %t >%t.out 2>&1 |
| 5 | // RUN: FileCheck %s < %t.out |
| 6 | |
| 7 | #include <iostream> |
| 8 | |
| 9 | // This test demonstrates that the types from anonymous namespaces are |
| 10 | // different in different translation units (while the char* type is the same). |
| 11 | |
| 12 | namespace { |
| 13 | struct X { |
| 14 | X(int i, int j) : a(i), b(j) {} |
| 15 | int a; |
| 16 | int b; |
| 17 | }; |
| 18 | } // namespace |
| 19 | |
| 20 | #ifdef PMAIN |
| 21 | void foo(void *context, int i); |
| 22 | char fbyte(void *context); |
| 23 | |
| 24 | int main() { |
| 25 | X x(5, 6); |
| 26 | foo((void *)&x, 8); |
| 27 | std::cout << "fbyte: " << fbyte((void *)&x) << "\n" ; |
| 28 | } |
| 29 | #else |
| 30 | void foo(void *context, int i) { |
| 31 | X *x = (X *)context; |
| 32 | x->b = i; |
| 33 | // CHECK: ERROR: TypeSanitizer: type-aliasing-violation |
| 34 | // CHECK: WRITE of size 4 at {{.*}} with type int (in (anonymous namespace)::X at offset 4) accesses an existing object of type int (in (anonymous namespace)::X at offset 4) |
| 35 | // CHECK: {{#0 0x.* in foo\(void\*, int\) .*anon-ns.cpp:}}[[@LINE-3]] |
| 36 | } |
| 37 | |
| 38 | char fbyte(void *context) { return *(char *)context; } |
| 39 | #endif |
| 40 | |
| 41 | // CHECK-NOT: ERROR: TypeSanitizer: type-aliasing-violation |
| 42 | |