[base] Pass scalar arguments by value in CHECK/DCHECK
This not only potentially improves performance, but also avoids weird linker errors, like the one below, where I used Smi::kMinValue in a DCHECK_LE. > [421/649] LINK ./mksnapshot > FAILED: mksnapshot > src/base/logging.h|178| error: undefined reference to 'v8::internal::Smi::kMinValue' R=bmeurer@chromium.org, ishell@chromium.org Committed: https://crrev.com/76723502528c5af003fdffc3520632ea2a13fef3 Review-Url: https://codereview.chromium.org/2524093002 Cr-Original-Commit-Position: refs/heads/master@{#41273} Cr-Commit-Position: refs/heads/master@{#41363}
This commit is contained in:
parent
b1b7d19610
commit
8fcfe66f94
@ -15,8 +15,7 @@ namespace base {
|
|||||||
|
|
||||||
// Explicit instantiations for commonly used comparisons.
|
// Explicit instantiations for commonly used comparisons.
|
||||||
#define DEFINE_MAKE_CHECK_OP_STRING(type) \
|
#define DEFINE_MAKE_CHECK_OP_STRING(type) \
|
||||||
template std::string* MakeCheckOpString<type, type>( \
|
template std::string* MakeCheckOpString<type, type>(type, type, char const*);
|
||||||
type const&, type const&, char const*);
|
|
||||||
DEFINE_MAKE_CHECK_OP_STRING(int)
|
DEFINE_MAKE_CHECK_OP_STRING(int)
|
||||||
DEFINE_MAKE_CHECK_OP_STRING(long) // NOLINT(runtime/int)
|
DEFINE_MAKE_CHECK_OP_STRING(long) // NOLINT(runtime/int)
|
||||||
DEFINE_MAKE_CHECK_OP_STRING(long long) // NOLINT(runtime/int)
|
DEFINE_MAKE_CHECK_OP_STRING(long long) // NOLINT(runtime/int)
|
||||||
@ -30,10 +29,10 @@ DEFINE_MAKE_CHECK_OP_STRING(void const*)
|
|||||||
|
|
||||||
// Explicit instantiations for floating point checks.
|
// Explicit instantiations for floating point checks.
|
||||||
#define DEFINE_CHECK_OP_IMPL(NAME) \
|
#define DEFINE_CHECK_OP_IMPL(NAME) \
|
||||||
template std::string* Check##NAME##Impl<float, float>( \
|
template std::string* Check##NAME##Impl<float, float>(float lhs, float rhs, \
|
||||||
float const& lhs, float const& rhs, char const* msg); \
|
char const* msg); \
|
||||||
template std::string* Check##NAME##Impl<double, double>( \
|
template std::string* Check##NAME##Impl<double, double>( \
|
||||||
double const& lhs, double const& rhs, char const* msg);
|
double lhs, double rhs, char const* msg);
|
||||||
DEFINE_CHECK_OP_IMPL(EQ)
|
DEFINE_CHECK_OP_IMPL(EQ)
|
||||||
DEFINE_CHECK_OP_IMPL(NE)
|
DEFINE_CHECK_OP_IMPL(NE)
|
||||||
DEFINE_CHECK_OP_IMPL(LE)
|
DEFINE_CHECK_OP_IMPL(LE)
|
||||||
|
@ -57,7 +57,8 @@ namespace base {
|
|||||||
// Don't use this macro directly in your code, use CHECK_EQ et al below.
|
// Don't use this macro directly in your code, use CHECK_EQ et al below.
|
||||||
#define CHECK_OP(name, op, lhs, rhs) \
|
#define CHECK_OP(name, op, lhs, rhs) \
|
||||||
do { \
|
do { \
|
||||||
if (std::string* _msg = ::v8::base::Check##name##Impl( \
|
if (std::string* _msg = \
|
||||||
|
::v8::base::Check##name##Impl<decltype(lhs), decltype(rhs)>( \
|
||||||
(lhs), (rhs), #lhs " " #op " " #rhs)) { \
|
(lhs), (rhs), #lhs " " #op " " #rhs)) { \
|
||||||
V8_Fatal(__FILE__, __LINE__, "Check failed: %s.", _msg->c_str()); \
|
V8_Fatal(__FILE__, __LINE__, "Check failed: %s.", _msg->c_str()); \
|
||||||
delete _msg; \
|
delete _msg; \
|
||||||
@ -73,13 +74,22 @@ namespace base {
|
|||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Helper to determine how to pass values: Pass scalars and arrays by value,
|
||||||
|
// others by const reference. std::decay<T> provides the type which should be
|
||||||
|
// used to pass T by value, e.g. converts array to pointer and removes const,
|
||||||
|
// volatile and reference.
|
||||||
|
template <typename T>
|
||||||
|
struct PassType : public std::conditional<
|
||||||
|
std::is_scalar<typename std::decay<T>::type>::value,
|
||||||
|
typename std::decay<T>::type, T const&> {};
|
||||||
|
|
||||||
// Build the error message string. This is separate from the "Impl"
|
// Build the error message string. This is separate from the "Impl"
|
||||||
// function template because it is not performance critical and so can
|
// function template because it is not performance critical and so can
|
||||||
// be out of line, while the "Impl" code should be inline. Caller
|
// be out of line, while the "Impl" code should be inline. Caller
|
||||||
// takes ownership of the returned string.
|
// takes ownership of the returned string.
|
||||||
template <typename Lhs, typename Rhs>
|
template <typename Lhs, typename Rhs>
|
||||||
std::string* MakeCheckOpString(Lhs const& lhs, Rhs const& rhs,
|
std::string* MakeCheckOpString(typename PassType<Lhs>::type lhs,
|
||||||
|
typename PassType<Rhs>::type rhs,
|
||||||
char const* msg) {
|
char const* msg) {
|
||||||
std::ostringstream ss;
|
std::ostringstream ss;
|
||||||
ss << msg << " (" << lhs << " vs. " << rhs << ")";
|
ss << msg << " (" << lhs << " vs. " << rhs << ")";
|
||||||
@ -90,7 +100,7 @@ std::string* MakeCheckOpString(Lhs const& lhs, Rhs const& rhs,
|
|||||||
// in logging.cc.
|
// in logging.cc.
|
||||||
#define DEFINE_MAKE_CHECK_OP_STRING(type) \
|
#define DEFINE_MAKE_CHECK_OP_STRING(type) \
|
||||||
extern template V8_BASE_EXPORT std::string* MakeCheckOpString<type, type>( \
|
extern template V8_BASE_EXPORT std::string* MakeCheckOpString<type, type>( \
|
||||||
type const&, type const&, char const*);
|
type, type, char const*);
|
||||||
DEFINE_MAKE_CHECK_OP_STRING(int)
|
DEFINE_MAKE_CHECK_OP_STRING(int)
|
||||||
DEFINE_MAKE_CHECK_OP_STRING(long) // NOLINT(runtime/int)
|
DEFINE_MAKE_CHECK_OP_STRING(long) // NOLINT(runtime/int)
|
||||||
DEFINE_MAKE_CHECK_OP_STRING(long long) // NOLINT(runtime/int)
|
DEFINE_MAKE_CHECK_OP_STRING(long long) // NOLINT(runtime/int)
|
||||||
@ -101,27 +111,21 @@ DEFINE_MAKE_CHECK_OP_STRING(char const*)
|
|||||||
DEFINE_MAKE_CHECK_OP_STRING(void const*)
|
DEFINE_MAKE_CHECK_OP_STRING(void const*)
|
||||||
#undef DEFINE_MAKE_CHECK_OP_STRING
|
#undef DEFINE_MAKE_CHECK_OP_STRING
|
||||||
|
|
||||||
|
|
||||||
// Helper functions for CHECK_OP macro.
|
// Helper functions for CHECK_OP macro.
|
||||||
// The (int, int) specialization works around the issue that the compiler
|
|
||||||
// will not instantiate the template version of the function on values of
|
|
||||||
// unnamed enum type - see comment below.
|
|
||||||
// The (float, float) and (double, double) instantiations are explicitly
|
// The (float, float) and (double, double) instantiations are explicitly
|
||||||
// externialized to ensure proper 32/64-bit comparisons on x86.
|
// externialized to ensure proper 32/64-bit comparisons on x86.
|
||||||
#define DEFINE_CHECK_OP_IMPL(NAME, op) \
|
#define DEFINE_CHECK_OP_IMPL(NAME, op) \
|
||||||
template <typename Lhs, typename Rhs> \
|
template <typename Lhs, typename Rhs> \
|
||||||
V8_INLINE std::string* Check##NAME##Impl(Lhs const& lhs, Rhs const& rhs, \
|
V8_INLINE std::string* Check##NAME##Impl(typename PassType<Lhs>::type lhs, \
|
||||||
|
typename PassType<Rhs>::type rhs, \
|
||||||
char const* msg) { \
|
char const* msg) { \
|
||||||
return V8_LIKELY(lhs op rhs) ? nullptr : MakeCheckOpString(lhs, rhs, msg); \
|
return V8_LIKELY(lhs op rhs) ? nullptr \
|
||||||
} \
|
: MakeCheckOpString<Lhs, Rhs>(lhs, rhs, msg); \
|
||||||
V8_INLINE std::string* Check##NAME##Impl(int lhs, int rhs, \
|
|
||||||
char const* msg) { \
|
|
||||||
return V8_LIKELY(lhs op rhs) ? nullptr : MakeCheckOpString(lhs, rhs, msg); \
|
|
||||||
} \
|
} \
|
||||||
extern template V8_BASE_EXPORT std::string* Check##NAME##Impl<float, float>( \
|
extern template V8_BASE_EXPORT std::string* Check##NAME##Impl<float, float>( \
|
||||||
float const& lhs, float const& rhs, char const* msg); \
|
float lhs, float rhs, char const* msg); \
|
||||||
extern template V8_BASE_EXPORT std::string* \
|
extern template V8_BASE_EXPORT std::string* \
|
||||||
Check##NAME##Impl<double, double>(double const& lhs, double const& rhs, \
|
Check##NAME##Impl<double, double>(double lhs, double rhs, \
|
||||||
char const* msg);
|
char const* msg);
|
||||||
DEFINE_CHECK_OP_IMPL(EQ, ==)
|
DEFINE_CHECK_OP_IMPL(EQ, ==)
|
||||||
DEFINE_CHECK_OP_IMPL(NE, !=)
|
DEFINE_CHECK_OP_IMPL(NE, !=)
|
||||||
|
@ -8,11 +8,18 @@
|
|||||||
namespace v8 {
|
namespace v8 {
|
||||||
namespace base {
|
namespace base {
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
template <typename Lhs, typename Rhs>
|
||||||
|
std::string *CallCheckEQ(Lhs lhs, Rhs rhs, const char *msg) {
|
||||||
|
return CheckEQImpl<Lhs, Rhs>(lhs, rhs, msg);
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
TEST(LoggingTest, CheckEQImpl) {
|
TEST(LoggingTest, CheckEQImpl) {
|
||||||
EXPECT_EQ(nullptr, CheckEQImpl(0.0, 0.0, ""));
|
EXPECT_EQ(nullptr, CallCheckEQ(0.0, 0.0, ""));
|
||||||
EXPECT_EQ(nullptr, CheckEQImpl(0.0, -0.0, ""));
|
EXPECT_EQ(nullptr, CallCheckEQ(0.0, -0.0, ""));
|
||||||
EXPECT_EQ(nullptr, CheckEQImpl(-0.0, 0.0, ""));
|
EXPECT_EQ(nullptr, CallCheckEQ(-0.0, 0.0, ""));
|
||||||
EXPECT_EQ(nullptr, CheckEQImpl(-0.0, -0.0, ""));
|
EXPECT_EQ(nullptr, CallCheckEQ(-0.0, -0.0, ""));
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace base
|
} // namespace base
|
||||||
|
Loading…
Reference in New Issue
Block a user