llvm/clang/test/Analysis/enum.cpp

// RUN: %clang_analyze_cc1 -std=c++11 -analyzer-checker=debug.ExprInspection %s

void clang_analyzer_eval(bool);

enum class Foo {
  Zero
};

bool pr15703(int x) {
  return Foo::Zero == (Foo)x; // don't crash
}

void testCasting(int i) {
  Foo f = static_cast<Foo>(i);
  int j = static_cast<int>(f);
  if (i == 0)
  {
    clang_analyzer_eval(f == Foo::Zero); // expected-warning{{TRUE}}
    clang_analyzer_eval(j == 0); // expected-warning{{TRUE}}
  }
  else
  {
    clang_analyzer_eval(f == Foo::Zero); // expected-warning{{FALSE}}
    clang_analyzer_eval(j == 0); // expected-warning{{FALSE}}
  }
}

enum class EnumBool : bool {
  F = false,
  T = true
};

bool testNoCrashOnSwitchEnumBool(EnumBool E) {
  switch (E) {
  case EnumBool::F:
    return false;
  }
  return true;
}

bool testNoCrashOnSwitchEnumBoolConstant() {
  EnumBool E = EnumBool::F;
  switch (E) {
    case EnumBool::F:
      return false; 
  }
  return true; 
}

typedef __INTPTR_TYPE__ intptr_t;
bool testNoCrashOnSwitchEnumBoolConstantCastedFromNullptr() {
  EnumBool E = static_cast<EnumBool>((intptr_t)nullptr);
  switch (E) {
  case EnumBool::F:
    return false;
  }
  return true;
}

bool testNoCrashOnSwitchEnumBoolConstantCastedFromPtr() {
  int X;
  intptr_t P = (intptr_t)&X;
  EnumBool E = static_cast<EnumBool>(P);
  switch (E) {
  case EnumBool::F:
    return false;
  }
  return true;
}