namespace folly { namespace detail {
namespace {
+void writeStderr(const char* s, size_t len) {
+ writeFull(STDERR_FILENO, s, len);
+}
void writeStderr(const char* s) {
- writeFull(STDERR_FILENO, s, strlen(s));
+ writeStderr(s, strlen(s));
}
} // namespace
void assertionFailure(const char* expr, const char* msg, const char* file,
unsigned int line, const char* function) {
writeStderr("\n\nAssertion failure: ");
- writeStderr(expr);
+ writeStderr(expr + 1, strlen(expr) - 2);
writeStderr("\nMessage: ");
writeStderr(msg);
writeStderr("\nFile: ");
* (containing msg) to stderr and abort()s. Just like CHECK(), but only
* logs to stderr and only does async-signal-safe calls.
*/
-#define FOLLY_SAFE_CHECK(expr, msg) \
+#define FOLLY_SAFE_CHECK_IMPL(expr, expr_s, msg) \
((expr) ? static_cast<void>(0) : \
::folly::detail::assertionFailure( \
- FB_STRINGIZE(expr), (msg), __FILE__, __LINE__, __PRETTY_FUNCTION__))
+ FB_STRINGIZE(expr_s), (msg), __FILE__, __LINE__, __PRETTY_FUNCTION__))
+#define FOLLY_SAFE_CHECK(expr, msg) FOLLY_SAFE_CHECK_IMPL((expr), (expr), (msg))
/**
* In debug mode, verify that the expression is true. Otherwise, do nothing
- * (do not even evaluate expr). Just like assert() or DCHECK(), but only
- * logs to stderr and only does async-signal-safe calls.
+ * (do not even evaluate expr). Just like DCHECK(), but only logs to stderr and
+ * only does async-signal-safe calls.
*/
-#ifdef NDEBUG
-#define FOLLY_SAFE_DCHECK(expr, msg) (static_cast<void>(0))
-#else
-#define FOLLY_SAFE_DCHECK FOLLY_SAFE_CHECK
-#endif
+#define FOLLY_SAFE_DCHECK(expr, msg) \
+ FOLLY_SAFE_CHECK_IMPL(!folly::kIsDebug || (expr), (expr), (msg))
namespace folly { namespace detail {
using namespace folly;
void fail() {
- FOLLY_SAFE_CHECK(0, "hello");
+ FOLLY_SAFE_CHECK(0 + 0, "hello");
}
void succeed() {
TEST(SafeAssert, AssertionFailure) {
succeed();
- EXPECT_DEATH(fail(), ".*Assertion failure:.*hello.*");
+ EXPECT_DEATH(fail(), "Assertion failure: 0 \\+ 0");
+ EXPECT_DEATH(fail(), "Message: hello");
}