llvm/compiler-rt/test/hwasan/TestCases/use-after-scope-setjmp.cpp

// RUN: %clangxx_hwasan  -mllvm -hwasan-use-stack-safety=0 -O2 %s -o %t && \
// RUN:     %run %t 2>&1

// REQUIRES: aarch64-target-arch || riscv64-target-arch

#include <sanitizer/hwasan_interface.h>
#include <setjmp.h>
#include <stdlib.h>
#include <string.h>

#include <sys/types.h>
#include <unistd.h>

volatile const char *stackbuf = nullptr;
jmp_buf jbuf;

__attribute__((noinline)) bool jump() {
  // Fool the compiler so it cannot deduce noreturn.
  if (getpid() != 0) {
    longjmp(jbuf, 1);
    return true;
  }
  return false;
}

bool target() {
  switch (setjmp(jbuf)) {
  case 1:
    return false;
  default:
    break;
  }

  while (true) {
    char buf[4096];
    stackbuf = buf;
    if (!jump()) {
      break;
    };
  }
  return true;
}

int main() {
  target();

  void *untagged = __hwasan_tag_pointer(stackbuf, 0);
  if (stackbuf == untagged) {
    // The buffer wasn't tagged in the first place, so the test will not work
    // as expected.
    return 2;
  }
  if (__hwasan_test_shadow(untagged, 4096) != -1) {
    return 1;
  }

  return 0;
}