468663ddbb
Former-commit-id: 1d6753294b2993e1fbf92de9366bb9544db4189b
180 lines
8.2 KiB
C++
180 lines
8.2 KiB
C++
// RUN: %check_clang_tidy %s hicpp-exception-baseclass %t -- -- -fcxx-exceptions
|
|
|
|
namespace std {
|
|
class exception {};
|
|
} // namespace std
|
|
|
|
class derived_exception : public std::exception {};
|
|
class deep_hierarchy : public derived_exception {};
|
|
class non_derived_exception {};
|
|
class terrible_idea : public non_derived_exception, public derived_exception {};
|
|
|
|
// FIXME: More complicated kinds of inheritance should be checked later, but there is
|
|
// currently no way use ASTMatchers for this kind of task.
|
|
#if 0
|
|
class bad_inheritance : private std::exception {};
|
|
class no_good_inheritance : protected std::exception {};
|
|
class really_creative : public non_derived_exception, private std::exception {};
|
|
#endif
|
|
|
|
void problematic() {
|
|
try {
|
|
throw int(42);
|
|
// CHECK-MESSAGES: [[@LINE-1]]:11: warning: throwing an exception whose type 'int' is not derived from 'std::exception'
|
|
} catch (int e) {
|
|
}
|
|
throw int(42);
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'int' is not derived from 'std::exception'
|
|
|
|
try {
|
|
throw 12;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:11: warning: throwing an exception whose type 'int' is not derived from 'std::exception'
|
|
} catch (...) {
|
|
throw; // Ok, even if the type is not known, conforming code can never rethrow a non-std::exception object.
|
|
}
|
|
|
|
try {
|
|
throw non_derived_exception();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:11: warning: throwing an exception whose type 'non_derived_exception' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 9:1: note: type defined here
|
|
} catch (non_derived_exception &e) {
|
|
}
|
|
throw non_derived_exception();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'non_derived_exception' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 9:1: note: type defined here
|
|
|
|
// FIXME: More complicated kinds of inheritance should be checked later, but there is
|
|
// currently no way use ASTMatchers for this kind of task.
|
|
#if 0
|
|
// Handle private inheritance cases correctly.
|
|
try {
|
|
throw bad_inheritance();
|
|
// CHECK MESSAGES: [[@LINE-1]]:11: warning: throwing an exception whose type 'bad_inheritance' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 10:1: note: type defined here
|
|
throw no_good_inheritance();
|
|
// CHECK MESSAGES: [[@LINE-1]]:11: warning: throwing an exception whose type 'no_good_inheritance' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 11:1: note: type defined here
|
|
throw really_creative();
|
|
// CHECK MESSAGES: [[@LINE-1]]:11: warning: throwing an exception whose type 'really_creative' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 12:1: note: type defined here
|
|
} catch (...) {
|
|
}
|
|
throw bad_inheritance();
|
|
// CHECK MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'bad_inheritance' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 10:1: note: type defined here
|
|
throw no_good_inheritance();
|
|
// CHECK MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'no_good_inheritance' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 11:1: note: type defined here
|
|
throw really_creative();
|
|
// CHECK MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'really_creative' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 12:1: note: type defined here
|
|
#endif
|
|
}
|
|
|
|
void allowed_throws() {
|
|
try {
|
|
throw std::exception(); // Ok
|
|
} catch (std::exception &e) { // Ok
|
|
}
|
|
throw std::exception();
|
|
|
|
try {
|
|
throw derived_exception(); // Ok
|
|
} catch (derived_exception &e) { // Ok
|
|
}
|
|
throw derived_exception(); // Ok
|
|
|
|
try {
|
|
throw deep_hierarchy(); // Ok, multiple levels of inheritance
|
|
} catch (deep_hierarchy &e) { // Ok
|
|
}
|
|
throw deep_hierarchy(); // Ok
|
|
|
|
try {
|
|
throw terrible_idea(); // Ok, but multiple inheritance isn't clean
|
|
} catch (std::exception &e) { // Can be caught as std::exception, even with multiple inheritance
|
|
}
|
|
throw terrible_idea(); // Ok, but multiple inheritance
|
|
}
|
|
|
|
// Templated function that throws exception based on template type
|
|
template <typename T>
|
|
void ThrowException() { throw T(); }
|
|
// CHECK-MESSAGES: [[@LINE-1]]:31: warning: throwing an exception whose type 'bad_generic_exception<int>' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 120:1: note: type defined here
|
|
// CHECK-MESSAGES: [[@LINE-3]]:31: warning: throwing an exception whose type 'bad_generic_exception<std::exception>' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 120:1: note: type defined here
|
|
// CHECK-MESSAGES: [[@LINE-5]]:31: warning: throwing an exception whose type 'exotic_exception<non_derived_exception>' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 123:1: note: type defined here
|
|
// CHECK-MESSAGES: [[@LINE-7]]:31: warning: throwing an exception whose type 'int' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: [[@LINE-8]]:31: warning: throwing an exception whose type 'non_derived_exception' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 9:1: note: type defined here
|
|
#define THROW_EXCEPTION(CLASS) ThrowException<CLASS>()
|
|
#define THROW_BAD_EXCEPTION throw int(42);
|
|
#define THROW_GOOD_EXCEPTION throw std::exception();
|
|
#define THROW_DERIVED_EXCEPTION throw deep_hierarchy();
|
|
|
|
template <typename T>
|
|
class generic_exception : std::exception {};
|
|
|
|
template <typename T>
|
|
class bad_generic_exception {};
|
|
|
|
template <typename T>
|
|
class exotic_exception : public T {};
|
|
|
|
void generic_exceptions() {
|
|
THROW_EXCEPTION(int);
|
|
// CHECK MESSAGES: [[@LINE-1]]:3: warning: throwing an exception whose type 'int' is not derived from 'std::exception'
|
|
THROW_EXCEPTION(non_derived_exception);
|
|
// CHECK MESSAGES: [[@LINE-1]]:3: warning: throwing an exception whose type 'non_derived_exception' is not derived from 'std::exception'
|
|
// CHECK MESSAGES: 9:1: note: type defined here
|
|
THROW_EXCEPTION(std::exception); // Ok
|
|
THROW_EXCEPTION(derived_exception); // Ok
|
|
THROW_EXCEPTION(deep_hierarchy); // Ok
|
|
|
|
THROW_BAD_EXCEPTION;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:3: warning: throwing an exception whose type 'int' is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: [[@LINE-25]]:35: note: expanded from macro 'THROW_BAD_EXCEPTION'
|
|
THROW_GOOD_EXCEPTION;
|
|
THROW_DERIVED_EXCEPTION;
|
|
|
|
throw generic_exception<int>(); // Ok,
|
|
THROW_EXCEPTION(generic_exception<float>); // Ok
|
|
|
|
throw bad_generic_exception<int>();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'bad_generic_exception<int>' is not derived from 'std::exception'
|
|
throw bad_generic_exception<std::exception>();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'bad_generic_exception<std::exception>' is not derived from 'std::exception'
|
|
THROW_EXCEPTION(bad_generic_exception<int>);
|
|
// CHECK MESSAGES: [[@LINE-1]]:3: warning: throwing an exception whose type 'bad_generic_exception<int>' is not derived from 'std::exception'
|
|
THROW_EXCEPTION(bad_generic_exception<std::exception>);
|
|
// CHECK MESSAGES: [[@LINE-1]]:3: warning: throwing an exception whose type 'bad_generic_exception<std::exception>' is not derived from 'std::exception'
|
|
|
|
throw exotic_exception<non_derived_exception>();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'exotic_exception<non_derived_exception>' is not derived from 'std::exception'
|
|
THROW_EXCEPTION(exotic_exception<non_derived_exception>);
|
|
// CHECK MESSAGES: [[@LINE-1]]:3: warning: throwing an exception whose type 'exotic_exception<non_derived_exception>' is not derived from 'std::exception'
|
|
|
|
throw exotic_exception<derived_exception>(); // Ok
|
|
THROW_EXCEPTION(exotic_exception<derived_exception>); // Ok
|
|
}
|
|
|
|
// Test for typedefed exception types
|
|
typedef int TypedefedBad;
|
|
typedef derived_exception TypedefedGood;
|
|
using UsingBad = int;
|
|
using UsingGood = deep_hierarchy;
|
|
|
|
void typedefed() {
|
|
throw TypedefedBad();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'TypedefedBad' (aka 'int') is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 164:1: note: type defined here
|
|
throw TypedefedGood(); // Ok
|
|
|
|
throw UsingBad();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:9: warning: throwing an exception whose type 'UsingBad' (aka 'int') is not derived from 'std::exception'
|
|
// CHECK-MESSAGES: 166:1: note: type defined here
|
|
throw UsingGood(); // Ok
|
|
}
|