Index: include/llvm/IR/Value.h =================================================================== --- include/llvm/IR/Value.h +++ include/llvm/IR/Value.h @@ -802,19 +802,19 @@ template <> struct isa_impl { static inline bool doit(const Value &Val) { - return isa(Val) || isa(Val); + return isa>(Val); } }; template <> struct isa_impl { static inline bool doit(const Value &Val) { - return isa(Val) || isa(Val); + return isa>(Val); } }; template <> struct isa_impl { static inline bool doit(const Value &Val) { - return isa(Val) || isa(Val); + return isa>(Val); } }; Index: include/llvm/Support/Casting.h =================================================================== --- include/llvm/Support/Casting.h +++ include/llvm/Support/Casting.h @@ -132,14 +132,37 @@ } }; +template struct anyof {}; + +template inline bool isa_impl_dispatcher(const Y &Val, X *) { + return isa_impl_wrap::SimpleType>::doit(Val); +} + +template +inline bool isa_impl_dispatcher(const Y &Val, anyof *) { + return isa_impl_dispatcher(Val, (X *)nullptr) || + isa_impl_dispatcher(Val, (anyof *)nullptr); +} +template inline bool isa_impl_dispatcher(const Y &Val, anyof<> *) { + return false; +} + // isa - Return true if the parameter to the template is an instance of the // template type argument. Used like this: // // if (isa(myVal)) { ... } // +// If type is of the form anyof as in: +// +// if (isa>(myVal)) { ... } +// +// Then the construct is equivalent to: +// +// if (isa(myVal) || isa(myVal) || isa(myVal)) { ... } +// template LLVM_NODISCARD inline bool isa(const Y &Val) { - return isa_impl_wrap::SimpleType>::doit(Val); + return isa_impl_dispatcher(Val, (X *)nullptr); } //===----------------------------------------------------------------------===//