forked from OSchip/llvm-project
99 lines
3.0 KiB
C++
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;
|
||
|
}
|