llvm/clang/test/CodeGenCXX/dynamic-cast-always-null.cpp

// RUN: %clang_cc1 -I%S %s -triple x86_64-apple-darwin10 -emit-llvm -fcxx-exceptions -fexceptions -std=c++11 -o - | FileCheck %s
struct A { virtual ~A(); };
struct B final : A { };
struct C { virtual ~C(); int c; };

// CHECK: @_Z8nonnull1P1C
A *nonnull1(C* c) {
  // CHECK: call {{.*}} @__dynamic_cast
  return dynamic_cast<A*>(c);
}

// CHECK: @_Z8nonnull2P1A
C *nonnull2(A* a) {
  // CHECK: call {{.*}} @__dynamic_cast
  return dynamic_cast<C*>(a);
}

// CHECK: @_Z1fP1B
C *f(B* b) {
  // CHECK-NOT: call {{.*}} @__dynamic_cast
  // CHECK: ret ptr null
  return dynamic_cast<C*>(b);
}

// CHECK: @_Z1fR1B
C &f(B& b) {
  // CHECK-NOT: call {{.*}} @__dynamic_cast
  // CHECK: call void @__cxa_bad_cast() [[NR:#[0-9]+]]
  // CHECK: unreachable
  //
  // CHECK: ret ptr poison
  return dynamic_cast<C&>(b);
}

// CHECK: @_Z1gP1C
B *g(C* c) {
  // CHECK-NOT: call {{.*}} @__dynamic_cast
  // CHECK: ret ptr null
  return dynamic_cast<B*>(c);
}

// CHECK: @_Z1gR1C
B &g(C& c) {
  // CHECK-NOT: call {{.*}} @__dynamic_cast
  // CHECK: call void @__cxa_bad_cast() [[NR:#[0-9]+]]
  // CHECK: unreachable
  //
  // CHECK: ret ptr poison
  return dynamic_cast<B&>(c);
}

void dont_crash() {
  (void) dynamic_cast<void*>((A*)0);
  (void) dynamic_cast<void*>((B*)0);
}

// CHECK: attributes [[NR]] = { noreturn }