llvm-capstone/compiler-rt/test/msan/dtor-multiple-inheritance.cc
Naomi Musgrave 36597fa128 Explicit reference to bug highlighted by
test/msan/dtor-trivial.cpp. Runtime testing for poisoning
vtable pointer in dtor.

Summary: Runtime testing for vtable ptr poisoning in dtor.

Reviewers: eugenis, kcc

Subscribers: llvm-commits

Differential Revision: http://reviews.llvm.org/D12713

Clean test case & comments.

Update tests for vptr poisoning order.

Simplify test to rely upon globals.

Assertions verify that vtable still accessible from dtors.

Testing linear inheritance and multiple inheritance for vtable poisoning.

Macros for testing expected failing functions.

Rename macros.

Removed xfail, modified FileCheck commands, to expect test to crash.

llvm-svn: 247763
2015-09-16 00:41:28 +00:00

99 lines
3.0 KiB
C++

// Defines diamond multiple inheritance structure
// A
// / \
// B C
// \ /
// Derived
// RUN: %clangxx_msan %s -O0 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t >%t.out 2>&1
// RUN: %clangxx_msan %s -O1 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t >%t.out 2>&1
// RUN: %clangxx_msan %s -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t >%t.out 2>&1
#include <sanitizer/msan_interface.h>
#include <assert.h>
int *temp_x;
int *temp_y;
int *temp_z;
int *temp_w;
class A {
public:
int x;
A() { x = 5; }
virtual ~A() {
assert(__msan_test_shadow(&this->x, sizeof(this->x) == -1));
// Memory owned by subclasses is poisoned.
assert(__msan_test_shadow(temp_y, sizeof(*temp_y)) != -1);
assert(__msan_test_shadow(temp_z, sizeof(*temp_z)) != -1);
assert(__msan_test_shadow(temp_w, sizeof(*temp_w)) != -1);
}
};
struct B : virtual public A {
public:
int y;
B() { y = 10; }
virtual ~B() {
assert(__msan_test_shadow(&this->y, sizeof(this->y)) == -1);
// Memory accessible via vtable still reachable.
assert(__msan_test_shadow(&this->x, sizeof(this->x)) == -1);
// Memory in sibling and subclass is poisoned.
assert(__msan_test_shadow(temp_z, sizeof(*temp_z)) != -1);
assert(__msan_test_shadow(temp_w, sizeof(*temp_w)) != -1);
}
};
struct C : virtual public A {
public:
int z;
C() { z = 15; }
virtual ~C() {
assert(__msan_test_shadow(&this->z, sizeof(this->z)) == -1);
// Memory accessible via vtable still reachable.
assert(__msan_test_shadow(&this->x, sizeof(this->x)) == -1);
// Sibling class is unpoisoned.
assert(__msan_test_shadow(temp_y, sizeof(*temp_y)) == -1);
// Memory in subclasses is poisoned.
assert(__msan_test_shadow(temp_w, sizeof(*temp_w)) != -1);
}
};
class Derived : public B, public C {
public:
int w;
Derived() { w = 10; }
~Derived() {
assert(__msan_test_shadow(&this->x, sizeof(this->x)) == -1);
// Members accessed through the vtable are still accessible.
assert(__msan_test_shadow(&this->y, sizeof(this->y)) == -1);
assert(__msan_test_shadow(&this->z, sizeof(this->z)) == -1);
assert(__msan_test_shadow(&this->w, sizeof(this->w)) == -1);
}
};
int main() {
Derived *d = new Derived();
// Keep track of members inherited from virtual bases,
// since the virtual base table is inaccessible after destruction.
temp_x = &d->x;
temp_y = &d->y;
temp_z = &d->z;
temp_w = &d->w;
// Order of destruction: Derived, C, B, A
d->~Derived();
// Verify that local pointer is unpoisoned, and that the object's
// members are.
assert(__msan_test_shadow(&d, sizeof(d)) == -1);
assert(__msan_test_shadow(temp_x, sizeof(*temp_x)) != -1);
assert(__msan_test_shadow(temp_y, sizeof(*temp_y)) != -1);
assert(__msan_test_shadow(temp_z, sizeof(*temp_z)) != -1);
assert(__msan_test_shadow(temp_w, sizeof(*temp_w)) != -1);
return 0;
}