[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:
clemensh 2016-11-29 07:02:16 -08:00 committed by Commit bot
parent b1b7d19610
commit 8fcfe66f94
3 changed files with 43 additions and 33 deletions

View File

@ -14,9 +14,8 @@ namespace v8 {
namespace base { 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)
@ -29,11 +28,11 @@ 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)

View File

@ -55,13 +55,14 @@ namespace base {
// Helper macro for binary operators. // Helper macro for binary operators.
// 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 = \
(lhs), (rhs), #lhs " " #op " " #rhs)) { \ ::v8::base::Check##name##Impl<decltype(lhs), decltype(rhs)>( \
V8_Fatal(__FILE__, __LINE__, "Check failed: %s.", _msg->c_str()); \ (lhs), (rhs), #lhs " " #op " " #rhs)) { \
delete _msg; \ V8_Fatal(__FILE__, __LINE__, "Check failed: %s.", _msg->c_str()); \
} \ delete _msg; \
} \
} while (0) } while (0)
#else #else
@ -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, !=)

View File

@ -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