1 | // RUN: %clangxx -std=c++11 -frtti -fsanitize=vptr -g %s -O3 -o %t |
2 | // RUN: %run %t &> %t.log |
3 | // RUN: cat %t.log | not count 0 && FileCheck --input-file %t.log %s || cat %t.log | count 0 |
4 | |
5 | // REQUIRES: cxxabi |
6 | |
7 | #include <sys/mman.h> |
8 | #include <unistd.h> |
9 | |
10 | class Base { |
11 | public: |
12 | int i; |
13 | virtual void print() {} |
14 | }; |
15 | |
16 | class Derived : public Base { |
17 | public: |
18 | void print() {} |
19 | }; |
20 | |
21 | |
22 | int main() { |
23 | int page_size = getpagesize(); |
24 | |
25 | void *non_accessible = mmap(addr: nullptr, len: page_size * 2, PROT_NONE, |
26 | MAP_PRIVATE | MAP_ANONYMOUS, fd: -1, offset: 0); |
27 | |
28 | if (non_accessible == MAP_FAILED) |
29 | return 0; |
30 | |
31 | void *accessible = mmap(addr: (char*)non_accessible + page_size, len: page_size, |
32 | PROT_READ | PROT_WRITE, |
33 | MAP_FIXED | MAP_PRIVATE | MAP_ANONYMOUS, fd: -1, offset: 0); |
34 | if (accessible == MAP_FAILED) |
35 | return 0; |
36 | |
37 | char *c = new char[sizeof(Derived)]; |
38 | |
39 | // The goal is to trigger a condition when Vptr points to accessible memory, |
40 | // but VptrPrefix does not. That has been triggering SIGSEGV in UBSan code. |
41 | void **vtable_ptr = reinterpret_cast<void **>(c); |
42 | *vtable_ptr = (void*)accessible; |
43 | |
44 | Derived *list = (Derived *)c; |
45 | |
46 | // CHECK: PR33221.cpp:[[@LINE+2]]:19: runtime error: member access within address {{.*}} which does not point to an object of type 'Base' |
47 | // CHECK-NEXT: invalid vptr |
48 | int foo = list->i; |
49 | return 0; |
50 | } |
51 | |