summaryrefslogtreecommitdiff
path: root/test/asan/TestCases/coverage-caller-callee-total-count.cc
blob: ac6d2486e462463de712868c5ba0085ed8e2b0ad (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
// Test __sanitizer_get_total_unique_coverage for caller-callee coverage

// RUN: %clangxx_asan -fsanitize-coverage=edge,indirect-calls %s -o %t
// RUN: env ASAN_OPTIONS=$ASAN_OPTIONS:coverage=1 %run %t
// RUN: rm -f caller-callee*.sancov
//
// REQUIRES: asan-64-bits

#include <sanitizer/coverage_interface.h>
#include <stdio.h>
#include <assert.h>
int P = 0;
struct Foo {virtual void f() {if (P) printf("Foo::f()\n");}};
struct Foo1 : Foo {virtual void f() {if (P) printf("%d\n", __LINE__);}};
struct Foo2 : Foo {virtual void f() {if (P) printf("%d\n", __LINE__);}};

Foo *foo[3] = {new Foo, new Foo1, new Foo2};

uintptr_t CheckNewTotalUniqueCoverageIsLargerAndReturnIt(uintptr_t old_total) {
  uintptr_t new_total = __sanitizer_get_total_unique_coverage();
  assert(new_total > old_total);
  return new_total;
}

int main(int argc, char **argv) {
  uintptr_t total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(0);
  foo[0]->f();
  total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(total);
  foo[1]->f();
  total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(total);
  foo[2]->f();
  total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(total);
  // Ok, called every function once.
  // Now call them again from another call site. Should get new coverage.
  foo[0]->f();
  total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(total);
  foo[1]->f();
  total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(total);
  foo[2]->f();
  total = CheckNewTotalUniqueCoverageIsLargerAndReturnIt(total);
}