| // Test -fsanitize-memory-use-after-dtor |
| // RUN: %clang_cc1 -fsanitize=memory -fsanitize-memory-use-after-dtor -std=c++11 -triple=x86_64-pc-linux -emit-llvm -o - %s | FileCheck %s |
| |
| // The no_sanitize_memory attribute, when applied to a destructor, |
| // represses emission of sanitizing callback |
| |
| template <class T> class Vector { |
| public: |
| int size; |
| ~Vector() {} |
| }; |
| |
| struct No_San { |
| Vector<int> v; |
| int x; |
| No_San() { } |
| __attribute__((no_sanitize_memory)) ~No_San() = default; |
| }; |
| |
| int main() { |
| No_San *ns = new No_San(); |
| ns->~No_San(); |
| return 0; |
| } |
| |
| // Repressing the sanitization attribute results in no msan |
| // instrumentation of the destructor |
| // CHECK: define {{.*}}No_SanD1Ev{{.*}} [[ATTRIBUTE:#[0-9]+]] |
| // CHECK-NOT: call void {{.*}}sanitizer_dtor_callback |
| // CHECK: ret void |
| |
| // CHECK: define {{.*}}No_SanD2Ev{{.*}} [[ATTRIBUTE:#[0-9]+]] |
| // CHECK-NOT: call void {{.*}}sanitizer_dtor_callback |
| // CHECK: call void {{.*}}VectorIiED2Ev |
| // CHECK-NOT: call void {{.*}}sanitizer_dtor_callback |
| // CHECK: ret void |
| |
| // CHECK: define {{.*}}VectorIiED2Ev |
| // CHECK: call void {{.*}}sanitizer_dtor_callback |
| // CHECK: ret void |
| |
| // When attribute is repressed, the destructor does not emit any tail calls |
| // CHECK-NOT: attributes [[ATTRIBUTE]] = {{.*}} sanitize_memory |