llvm/compiler-rt/test/asan/TestCases/Posix/ignore_free_hook.cpp

// RUN: %clangxx_asan -O2 %s -o %t -DTEST=basic_hook_works && not %run %t \
// RUN:   2>&1 | FileCheck %s -check-prefix=CHECK-BASIC
// RUN: %clangxx_asan -O2 %s -o %t -DTEST=ignore && %run %t \
// RUN:   2>&1 | FileCheck %s -check-prefix=CHECK-IGNORE
// RUN: %clangxx_asan -O2 %s -o %t -DTEST=ignore_twice && not %run %t \
// RUN:   2>&1 | FileCheck %s -check-prefix=CHECK-IGNORE-2
// RUN: %clangxx_asan -O2 %s -o %t -DTEST=mismatch && %env_asan_opts=alloc_dealloc_mismatch=1 not %run %t \
// RUN:   2>&1 | FileCheck %s -check-prefix=CHECK-MISMATCH
// RUN: %clangxx_asan -O2 %s -o %t -DTEST=ignore_mismatch && %env_asan_opts=alloc_dealloc_mismatch=1 %run %t \
// RUN:   2>&1 | FileCheck %s -check-prefix=CHECK-IGNORE-MISMATCH
// RUN: %clangxx_asan -O2 %s -o %t -DTEST=double_delete && not %run %t \
// RUN:   2>&1 | FileCheck %s -check-prefix=CHECK-DOUBLE-DELETE

#include <stdio.h>
#include <stdlib.h>

static char *volatile glob_ptr;
bool ignore_free = false;

#if (__APPLE__)
// Required for dyld macOS 12.0+
#  define WEAK_ON_APPLE __attribute__((weak))
#else // !(__APPLE__)
#  define WEAK_ON_APPLE
#endif // (__APPLE__)

extern "C" {
WEAK_ON_APPLE void __sanitizer_free_hook(const volatile void *ptr) {
  if (ptr == glob_ptr)
    fprintf(stderr, "Free Hook\n");
}

WEAK_ON_APPLE int __sanitizer_ignore_free_hook(const volatile void *ptr) {
  if (ptr != glob_ptr)
    return 0;
  fprintf(stderr, ignore_free ? "Free Ignored\n" : "Free Respected\n");
  return ignore_free;
}
} // extern "C"

void allocate() { glob_ptr = reinterpret_cast<char *volatile>(malloc(100)); }
void deallocate() { free(reinterpret_cast<void *>(glob_ptr)); }

void basic_hook_works() {
  allocate();
  deallocate();  // CHECK-BASIC-NOT: Free Ignored
                 // CHECK-BASIC:     Free Respected
                 // CHECK-BASIC:     Free Hook
  *glob_ptr = 0; // CHECK-BASIC:     AddressSanitizer: heap-use-after-free
}

void ignore() {
  allocate();
  ignore_free = true;
  deallocate();
  // CHECK-IGNORE:     Free Ignored
  // CHECK-IGNORE-NOT: Free Respected
  // CHECK-IGNORE-NOT: Free Hook
  // CHECK-IGNORE-NOT: AddressSanitizer
  *glob_ptr = 0;
}

void ignore_twice() {
  allocate();
  ignore_free = true;
  deallocate(); // CHECK-IGNORE-2: Free Ignored
  *glob_ptr = 0;
  ignore_free = false;
  deallocate();  // CHECK-IGNORE-2-NOT: Free Ignored
                 // CHECK-IGNORE-2:     Free Respected
                 // CHECK-IGNORE-2:     Free Hook
  *glob_ptr = 0; // CHECK-IGNORE-2:     AddressSanitizer: heap-use-after-free
}

void ignore_a_lot() {
  allocate();
  ignore_free = true;
  for (int i = 0; i < 10000; ++i) {
    deallocate(); // CHECK-IGNORE-3: Free Ignored
    *glob_ptr = 0;
  }
  ignore_free = false;
  deallocate();  // CHECK-IGNORE-3: Free Respected
                 // CHECK-IGNORE-3: Free Hook
  *glob_ptr = 0; // CHECK-IGNORE-3: AddressSanitizer: heap-use-after-free
}

void mismatch() {
  glob_ptr = new char;
  deallocate(); // CHECK-MISMATCH: AddressSanitizer: alloc-dealloc-mismatch
}

void ignore_mismatch() {
  glob_ptr = new char;
  ignore_free = true;
  // Mismatch isn't detected when the free() is ignored.
  deallocate();
  deallocate();
  ignore_free = false;
  // And also isn't detected when the memory is free()-d for real.
  deallocate(); // CHECK-IGNORE-MISMATCH-NOT: AddressSanitizer: alloc-dealloc-mismatch
}

void double_delete() {
  allocate();
  ignore_free = true;
  deallocate(); // CHECK-DOUBLE-DELETE: Free Ignored
  deallocate(); // CHECK-DOUBLE-DELETE: Free Ignored
  ignore_free = false;
  deallocate(); // CHECK-DOUBLE-DELETE: Free Respected
                // CHECK-DOUBLE-DELETE: Free Hook
  deallocate(); // CHECK-DOUBLE-DELETE: AddressSanitizer: attempting double-free
}

int main() {
  TEST();
  return 0;
}