Skip to content

Commit

Permalink
Explicit reference to bug highlighted by
Browse files Browse the repository at this point in the history
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
  • Loading branch information
Naomi Musgrave committed Sep 16, 2015
1 parent 703835c commit 36597fa
Show file tree
Hide file tree
Showing 4 changed files with 177 additions and 33 deletions.
66 changes: 34 additions & 32 deletions compiler-rt/test/msan/dtor-multiple-inheritance.cc
Expand Up @@ -14,24 +14,21 @@
#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;
int *y_ptr;
int *z_ptr;
int *w_ptr;
A() { x = 5; }
void set_ptrs(int *y_ptr, int *z_ptr, int *w_ptr) {
this->y_ptr = y_ptr;
this->z_ptr = z_ptr;
this->w_ptr = w_ptr;
}
virtual ~A() {
assert(__msan_test_shadow(&this->x, sizeof(this->x) == -1));
// bad access subclass member
assert(__msan_test_shadow(this->y_ptr, sizeof(*this->y_ptr)) != -1);
assert(__msan_test_shadow(this->z_ptr, sizeof(*this->z_ptr)) != -1);
assert(__msan_test_shadow(this->w_ptr, sizeof(*this->w_ptr)) != -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);
}
};

Expand All @@ -40,13 +37,12 @@ struct B : virtual public A {
int y;
B() { y = 10; }
virtual ~B() {
assert(__msan_test_shadow(&this->x, sizeof(this->x)) == -1);
assert(__msan_test_shadow(&this->y, sizeof(this->y)) == -1);
assert(__msan_test_shadow(this->y_ptr, sizeof(*this->y_ptr)) == -1);

// memory in subclasses is poisoned
assert(__msan_test_shadow(this->z_ptr, sizeof(*this->z_ptr)) != -1);
assert(__msan_test_shadow(this->w_ptr, sizeof(*this->w_ptr)) != -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);
}
};

Expand All @@ -55,13 +51,13 @@ struct C : virtual public A {
int z;
C() { z = 15; }
virtual ~C() {
assert(__msan_test_shadow(&this->x, sizeof(this->x)) == -1);
assert(__msan_test_shadow(&this->z, sizeof(this->z)) == -1);
assert(__msan_test_shadow(this->y_ptr, sizeof(*this->y_ptr)) == -1);
assert(__msan_test_shadow(this->z_ptr, sizeof(*this->z_ptr) == -1));

// memory in subclasses is poisoned
assert(__msan_test_shadow(this->w_ptr, sizeof(*this->w_ptr)) != -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);
}
};

Expand All @@ -71,26 +67,32 @@ class Derived : public B, public C {
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();
d->set_ptrs(&d->y, &d->z, &d->w);

// 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(&d->x, sizeof(d->x)) != -1);
assert(__msan_test_shadow(&d->y, sizeof(d->y)) != -1);
assert(__msan_test_shadow(&d->z, sizeof(d->z)) != -1);
assert(__msan_test_shadow(&d->w, sizeof(d->w)) != -1);
assert(__msan_test_shadow(&d->y_ptr, sizeof(d->y_ptr)) != -1);
assert(__msan_test_shadow(&d->z_ptr, sizeof(d->z_ptr)) != -1);
assert(__msan_test_shadow(&d->w_ptr, sizeof(d->w_ptr)) != -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;
}
4 changes: 3 additions & 1 deletion compiler-rt/test/msan/dtor-trivial.cpp
Expand Up @@ -4,7 +4,9 @@

// 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

// TODO Success pending on resolution of 596
// TODO Success pending on resolution of
// https://github.com/google/sanitizers/issues/596

// XFAIL: *

#include <assert.h>
Expand Down
72 changes: 72 additions & 0 deletions compiler-rt/test/msan/dtor-vtable-multiple-inheritance.cc
@@ -0,0 +1,72 @@
// RUN: %clangxx_msan %s -O0 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t

// RUN: %clangxx_msan %s -O1 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t

// RUN: %clangxx_msan %s -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t

// RUN: %clangxx_msan %s -DCVPTR=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// RUN: %clangxx_msan %s -DEAVPTR=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// RUN: %clangxx_msan %s -DEDVPTR=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// Expected to quit due to invalid access when invoking
// function using vtable.

class A {
public:
int x;
virtual ~A() {
// Should succeed
this->A_Foo();
}
virtual void A_Foo() {}
};

class B : public virtual A {
public:
int y;
virtual ~B() {}
virtual void A_Foo() {}
};

class C : public B {
public:
int z;
~C() {}
};

class D {
public:
int w;
~D() {}
virtual void D_Foo() {}
};

class E : public virtual A, public virtual D {
public:
int u;
~E() {}
void A_Foo() {}
};

int main() {
// Simple linear inheritance
C *c = new C();
c->~C();
// This fails
#ifdef CVPTR
c->A_Foo();
#endif

// Multiple inheritance, so has multiple vtables
E *e = new E();
e->~E();
// Both of these fail
#ifdef EAVPTR
e->A_Foo();
#endif
#ifdef EDVPTR
e->D_Foo();
#endif
}
68 changes: 68 additions & 0 deletions compiler-rt/test/msan/dtor-vtable.cc
@@ -0,0 +1,68 @@
// RUN: %clangxx_msan %s -O0 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t

// RUN: %clangxx_msan %s -O1 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t

// RUN: %clangxx_msan %s -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 %run %t

// RUN: %clangxx_msan %s -DVPTRA=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// RUN: %clangxx_msan %s -DVPTRCA=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// RUN: %clangxx_msan %s -DVPTRCB=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// RUN: %clangxx_msan %s -DVPTRC=1 -O2 -fsanitize=memory -fsanitize-memory-use-after-dtor -o %t && MSAN_OPTIONS=poison_in_dtor=1 not %run %t

// Expected to quit due to invalid access when invoking
// function using vtable.

#include <sanitizer/msan_interface.h>
#include <stdio.h>
#include <assert.h>

class A {
public:
int x;
~A() {}
virtual void A_Foo() {}
};

class B {
public:
int y;
~B() {}
virtual void B_Foo() {}
};

class C : public A, public B {
public:
int z;
~C() {}
virtual void C_Foo() {}
};

int main() {
A *a = new A();
a->~A();

// Shouldn't be allowed to invoke function via vtable.
#ifdef VPTRA
a->A_Foo();
#endif

C *c = new C();
c->~C();

#ifdef VPTRCA
c->A_Foo();
#endif

#ifdef VPTRCB
c->B_Foo();
#endif

#ifdef VPTRC
c->C_Foo();
#endif

return 0;
}

0 comments on commit 36597fa

Please sign in to comment.