1*9356374aSAndroid Build Coastguard Worker // Copyright 2020 The Abseil Authors.
2*9356374aSAndroid Build Coastguard Worker //
3*9356374aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*9356374aSAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*9356374aSAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*9356374aSAndroid Build Coastguard Worker //
7*9356374aSAndroid Build Coastguard Worker // https://www.apache.org/licenses/LICENSE-2.0
8*9356374aSAndroid Build Coastguard Worker //
9*9356374aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*9356374aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*9356374aSAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*9356374aSAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*9356374aSAndroid Build Coastguard Worker // limitations under the License.
14*9356374aSAndroid Build Coastguard Worker
15*9356374aSAndroid Build Coastguard Worker #include "absl/base/optimization.h"
16*9356374aSAndroid Build Coastguard Worker
17*9356374aSAndroid Build Coastguard Worker #include "gtest/gtest.h"
18*9356374aSAndroid Build Coastguard Worker #include "absl/types/optional.h"
19*9356374aSAndroid Build Coastguard Worker
20*9356374aSAndroid Build Coastguard Worker namespace {
21*9356374aSAndroid Build Coastguard Worker
22*9356374aSAndroid Build Coastguard Worker // Tests for the ABSL_PREDICT_TRUE and ABSL_PREDICT_FALSE macros.
23*9356374aSAndroid Build Coastguard Worker // The tests only verify that the macros are functionally correct - i.e. code
24*9356374aSAndroid Build Coastguard Worker // behaves as if they weren't used. They don't try to check their impact on
25*9356374aSAndroid Build Coastguard Worker // optimization.
26*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,PredictTrue)27*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, PredictTrue) {
28*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(true));
29*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_TRUE(false));
30*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(1 == 1));
31*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_TRUE(1 == 2));
32*9356374aSAndroid Build Coastguard Worker
33*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_TRUE(false)) ADD_FAILURE();
34*9356374aSAndroid Build Coastguard Worker if (!ABSL_PREDICT_TRUE(true)) ADD_FAILURE();
35*9356374aSAndroid Build Coastguard Worker
36*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(true) && true);
37*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(true) || false);
38*9356374aSAndroid Build Coastguard Worker }
39*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,PredictFalse)40*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, PredictFalse) {
41*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(true));
42*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_FALSE(false));
43*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(1 == 1));
44*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_FALSE(1 == 2));
45*9356374aSAndroid Build Coastguard Worker
46*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_FALSE(false)) ADD_FAILURE();
47*9356374aSAndroid Build Coastguard Worker if (!ABSL_PREDICT_FALSE(true)) ADD_FAILURE();
48*9356374aSAndroid Build Coastguard Worker
49*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(true) && true);
50*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(true) || false);
51*9356374aSAndroid Build Coastguard Worker }
52*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,OneEvaluation)53*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, OneEvaluation) {
54*9356374aSAndroid Build Coastguard Worker // Verify that the expression is only evaluated once.
55*9356374aSAndroid Build Coastguard Worker int x = 0;
56*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_TRUE((++x) == 0)) ADD_FAILURE();
57*9356374aSAndroid Build Coastguard Worker EXPECT_EQ(x, 1);
58*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_FALSE((++x) == 0)) ADD_FAILURE();
59*9356374aSAndroid Build Coastguard Worker EXPECT_EQ(x, 2);
60*9356374aSAndroid Build Coastguard Worker }
61*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,OperatorOrder)62*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, OperatorOrder) {
63*9356374aSAndroid Build Coastguard Worker // Verify that operator order inside and outside the macro behaves well.
64*9356374aSAndroid Build Coastguard Worker // These would fail for a naive '#define ABSL_PREDICT_TRUE(x) x'
65*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(1 && 2) == true);
66*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(1 && 2) == true);
67*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(!ABSL_PREDICT_TRUE(1 == 2));
68*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(!ABSL_PREDICT_FALSE(1 == 2));
69*9356374aSAndroid Build Coastguard Worker }
70*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,Pointer)71*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, Pointer) {
72*9356374aSAndroid Build Coastguard Worker const int x = 3;
73*9356374aSAndroid Build Coastguard Worker const int *good_intptr = &x;
74*9356374aSAndroid Build Coastguard Worker const int *null_intptr = nullptr;
75*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(good_intptr));
76*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_TRUE(null_intptr));
77*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(good_intptr));
78*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_FALSE(null_intptr));
79*9356374aSAndroid Build Coastguard Worker }
80*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,Optional)81*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, Optional) {
82*9356374aSAndroid Build Coastguard Worker // Note: An optional's truth value is the value's existence, not its truth.
83*9356374aSAndroid Build Coastguard Worker absl::optional<bool> has_value(false);
84*9356374aSAndroid Build Coastguard Worker absl::optional<bool> no_value;
85*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_TRUE(has_value));
86*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_TRUE(no_value));
87*9356374aSAndroid Build Coastguard Worker EXPECT_TRUE(ABSL_PREDICT_FALSE(has_value));
88*9356374aSAndroid Build Coastguard Worker EXPECT_FALSE(ABSL_PREDICT_FALSE(no_value));
89*9356374aSAndroid Build Coastguard Worker }
90*9356374aSAndroid Build Coastguard Worker
91*9356374aSAndroid Build Coastguard Worker class ImplictlyConvertibleToBool {
92*9356374aSAndroid Build Coastguard Worker public:
ImplictlyConvertibleToBool(bool value)93*9356374aSAndroid Build Coastguard Worker explicit ImplictlyConvertibleToBool(bool value) : value_(value) {}
operator bool() const94*9356374aSAndroid Build Coastguard Worker operator bool() const { // NOLINT(google-explicit-constructor)
95*9356374aSAndroid Build Coastguard Worker return value_;
96*9356374aSAndroid Build Coastguard Worker }
97*9356374aSAndroid Build Coastguard Worker
98*9356374aSAndroid Build Coastguard Worker private:
99*9356374aSAndroid Build Coastguard Worker bool value_;
100*9356374aSAndroid Build Coastguard Worker };
101*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,ImplicitBoolConversion)102*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, ImplicitBoolConversion) {
103*9356374aSAndroid Build Coastguard Worker const ImplictlyConvertibleToBool is_true(true);
104*9356374aSAndroid Build Coastguard Worker const ImplictlyConvertibleToBool is_false(false);
105*9356374aSAndroid Build Coastguard Worker if (!ABSL_PREDICT_TRUE(is_true)) ADD_FAILURE();
106*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_TRUE(is_false)) ADD_FAILURE();
107*9356374aSAndroid Build Coastguard Worker if (!ABSL_PREDICT_FALSE(is_true)) ADD_FAILURE();
108*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_FALSE(is_false)) ADD_FAILURE();
109*9356374aSAndroid Build Coastguard Worker }
110*9356374aSAndroid Build Coastguard Worker
111*9356374aSAndroid Build Coastguard Worker class ExplictlyConvertibleToBool {
112*9356374aSAndroid Build Coastguard Worker public:
ExplictlyConvertibleToBool(bool value)113*9356374aSAndroid Build Coastguard Worker explicit ExplictlyConvertibleToBool(bool value) : value_(value) {}
operator bool() const114*9356374aSAndroid Build Coastguard Worker explicit operator bool() const { return value_; }
115*9356374aSAndroid Build Coastguard Worker
116*9356374aSAndroid Build Coastguard Worker private:
117*9356374aSAndroid Build Coastguard Worker bool value_;
118*9356374aSAndroid Build Coastguard Worker };
119*9356374aSAndroid Build Coastguard Worker
TEST(PredictTest,ExplicitBoolConversion)120*9356374aSAndroid Build Coastguard Worker TEST(PredictTest, ExplicitBoolConversion) {
121*9356374aSAndroid Build Coastguard Worker const ExplictlyConvertibleToBool is_true(true);
122*9356374aSAndroid Build Coastguard Worker const ExplictlyConvertibleToBool is_false(false);
123*9356374aSAndroid Build Coastguard Worker if (!ABSL_PREDICT_TRUE(is_true)) ADD_FAILURE();
124*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_TRUE(is_false)) ADD_FAILURE();
125*9356374aSAndroid Build Coastguard Worker if (!ABSL_PREDICT_FALSE(is_true)) ADD_FAILURE();
126*9356374aSAndroid Build Coastguard Worker if (ABSL_PREDICT_FALSE(is_false)) ADD_FAILURE();
127*9356374aSAndroid Build Coastguard Worker }
128*9356374aSAndroid Build Coastguard Worker
129*9356374aSAndroid Build Coastguard Worker } // namespace
130