Skip to content

Commit b5687db

Browse files
Abseil Teamvslashg
Abseil Team
authored andcommitted
Googletest export
Improve DoubleNearPredFormat output on bad epsilons DoubleNearPredFormat will happily accept epsilon values (abs_error) that are so small that they are meaningless. This turns EXPECT_NEAR into a complicated and non-obvious version of EXPECT_EQ. This change modifies DoubleNearPredFormat) so that when there is a failure it calculates the smallest meaningful epsilon value, given the input values, and then prints a message which explains what happened. If a true equality test is wanted either pass a literal 0.0 as abs_error or use EXPECT_EQ. If a check for being almost equal is wanted consider using EXPECT_DOUBLE_EQ which, contrary to its name, verifies that the two numbers are *almost* equal (within four ULPs). With this change the flaky test mentioned in crbug.com/786046 gives this output: The difference between 4.2934311416234112e+18 and 4.2934311416234107e+18 is 512, where 4.2934311416234112e+18 evaluates to 4.2934311416234112e+18, 4.2934311416234107e+18 evaluates to 4.2934311416234107e+18. The abs_error parameter 1.0 evaluates to 1 which is smaller than the minimum distance between doubles for numbers of this magnitude which is 512, thus making this EXPECT_NEAR check equivalent to EXPECT_EQUAL. Consider using EXPECT_DOUBLE_EQ instead. Tested: I confirmed that this change detects the bad epsilon value that caused crbug.com/786046 in Chromium and added a test for the desired output. PiperOrigin-RevId: 332946880
1 parent 7aca844 commit b5687db

File tree

2 files changed

+31
-0
lines changed

2 files changed

+31
-0
lines changed

googletest/src/gtest.cc

+24
Original file line numberDiff line numberDiff line change
@@ -1516,6 +1516,30 @@ AssertionResult DoubleNearPredFormat(const char* expr1,
15161516
const double diff = fabs(val1 - val2);
15171517
if (diff <= abs_error) return AssertionSuccess();
15181518

1519+
// Find the value which is closest to zero.
1520+
const double min_abs = std::min(fabs(val1), fabs(val2));
1521+
// Find the distance to the next double from that value.
1522+
const double epsilon =
1523+
nextafter(min_abs, std::numeric_limits<double>::infinity()) - min_abs;
1524+
// Detect the case where abs_error is so small that EXPECT_NEAR is
1525+
// effectively the same as EXPECT_EQUAL, and give an informative error
1526+
// message so that the situation can be more easily understood without
1527+
// requiring exotic floating-point knowledge.
1528+
// Don't do an epsilon check if abs_error is zero because that implies
1529+
// that an equality check was actually intended.
1530+
if (!isnan(val1) && !isnan(val2) && abs_error > 0 && abs_error < epsilon) {
1531+
return AssertionFailure()
1532+
<< "The difference between " << expr1 << " and " << expr2 << " is "
1533+
<< diff << ", where\n"
1534+
<< expr1 << " evaluates to " << val1 << ",\n"
1535+
<< expr2 << " evaluates to " << val2 << ".\nThe abs_error parameter "
1536+
<< abs_error_expr << " evaluates to " << abs_error
1537+
<< " which is smaller than the minimum distance between doubles for "
1538+
"numbers of this magnitude which is "
1539+
<< epsilon
1540+
<< ", thus making this EXPECT_NEAR check equivalent to "
1541+
"EXPECT_EQUAL. Consider using EXPECT_DOUBLE_EQ instead.";
1542+
}
15191543
return AssertionFailure()
15201544
<< "The difference between " << expr1 << " and " << expr2
15211545
<< " is " << diff << ", which exceeds " << abs_error_expr << ", where\n"

googletest/test/gtest_unittest.cc

+7
Original file line numberDiff line numberDiff line change
@@ -3084,6 +3084,13 @@ TEST_F(DoubleTest, EXPECT_NEAR) {
30843084
EXPECT_NONFATAL_FAILURE(EXPECT_NEAR(1.0, 1.5, 0.25), // NOLINT
30853085
"The difference between 1.0 and 1.5 is 0.5, "
30863086
"which exceeds 0.25");
3087+
// At this magnitude adjacent doubles are 512.0 apart, so this triggers a
3088+
// slightly different failure reporting path.
3089+
EXPECT_NONFATAL_FAILURE(
3090+
EXPECT_NEAR(4.2934311416234112e+18, 4.2934311416234107e+18, 1.0),
3091+
"The abs_error parameter 1.0 evaluates to 1 which is smaller than the "
3092+
"minimum distance between doubles for numbers of this magnitude which is "
3093+
"512");
30873094
}
30883095

30893096
// Tests ASSERT_NEAR.

0 commit comments

Comments
 (0)