blob: e3776e2bdc8a58c66bd49e8917813337891f12f8 [file] [log] [blame]
/*
* Copyright 2015 Facebook, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <folly/Benchmark.h>
#include <folly/Traits.h>
#include <gflags/gflags.h>
#include <gtest/gtest.h>
using namespace folly;
using namespace std;
struct T1 {}; // old-style IsRelocatable, below
struct T2 {}; // old-style IsRelocatable, below
struct T3 { typedef std::true_type IsRelocatable; };
struct T4 { typedef std::true_type IsTriviallyCopyable; };
struct T5 : T3 {};
struct F1 {};
struct F2 { typedef int IsRelocatable; };
struct F3 : T3 { typedef std::false_type IsRelocatable; };
struct F4 : T1 {};
namespace folly {
template <> struct IsRelocatable<T1> : std::true_type {};
template <> FOLLY_ASSUME_RELOCATABLE(T2);
}
TEST(Traits, scalars) {
EXPECT_TRUE(IsRelocatable<int>::value);
EXPECT_TRUE(IsRelocatable<bool>::value);
EXPECT_TRUE(IsRelocatable<double>::value);
EXPECT_TRUE(IsRelocatable<void*>::value);
}
TEST(Traits, containers) {
EXPECT_TRUE (IsRelocatable<vector<F1>>::value);
EXPECT_FALSE((IsRelocatable<pair<F1, F1>>::value));
EXPECT_TRUE ((IsRelocatable<pair<T1, T2>>::value));
}
TEST(Traits, original) {
EXPECT_TRUE(IsRelocatable<T1>::value);
EXPECT_TRUE(IsRelocatable<T2>::value);
}
TEST(Traits, typedefd) {
EXPECT_TRUE (IsRelocatable<T3>::value);
EXPECT_TRUE (IsRelocatable<T5>::value);
EXPECT_FALSE(IsRelocatable<F2>::value);
EXPECT_FALSE(IsRelocatable<F3>::value);
}
TEST(Traits, unset) {
EXPECT_FALSE(IsRelocatable<F1>::value);
EXPECT_FALSE(IsRelocatable<F4>::value);
}
TEST(Traits, bitprop) {
EXPECT_TRUE(IsTriviallyCopyable<T4>::value);
EXPECT_TRUE(IsRelocatable<T4>::value);
}
TEST(Traits, bitAndInit) {
EXPECT_TRUE (IsTriviallyCopyable<int>::value);
EXPECT_FALSE(IsTriviallyCopyable<vector<int>>::value);
EXPECT_TRUE (IsZeroInitializable<int>::value);
EXPECT_FALSE(IsZeroInitializable<vector<int>>::value);
}
TEST(Traits, is_negative) {
EXPECT_TRUE(folly::is_negative(-1));
EXPECT_FALSE(folly::is_negative(0));
EXPECT_FALSE(folly::is_negative(1));
EXPECT_FALSE(folly::is_negative(0u));
EXPECT_FALSE(folly::is_negative(1u));
EXPECT_TRUE(folly::is_non_positive(-1));
EXPECT_TRUE(folly::is_non_positive(0));
EXPECT_FALSE(folly::is_non_positive(1));
EXPECT_TRUE(folly::is_non_positive(0u));
EXPECT_FALSE(folly::is_non_positive(1u));
}
TEST(Traits, relational) {
// We test, especially, the edge cases to make sure we don't
// trip -Wtautological-comparisons
EXPECT_FALSE((folly::less_than<uint8_t, 0u, uint8_t>(0u)));
EXPECT_FALSE((folly::less_than<uint8_t, 0u, uint8_t>(254u)));
EXPECT_FALSE((folly::less_than<uint8_t, 255u, uint8_t>(255u)));
EXPECT_TRUE( (folly::less_than<uint8_t, 255u, uint8_t>(254u)));
EXPECT_FALSE((folly::greater_than<uint8_t, 0u, uint8_t>(0u)));
EXPECT_TRUE( (folly::greater_than<uint8_t, 0u, uint8_t>(254u)));
EXPECT_FALSE((folly::greater_than<uint8_t, 255u, uint8_t>(255u)));
EXPECT_FALSE((folly::greater_than<uint8_t, 255u, uint8_t>(254u)));
}
struct membership_no {};
struct membership_yes { using x = void; };
FOLLY_CREATE_HAS_MEMBER_TYPE_TRAITS(has_member_type_x, x);
TEST(Traits, has_member_type) {
EXPECT_FALSE(bool(has_member_type_x<membership_no>::value));
EXPECT_TRUE(bool(has_member_type_x<membership_yes>::value));
}
int main(int argc, char ** argv) {
testing::InitGoogleTest(&argc, argv);
gflags::ParseCommandLineFlags(&argc, &argv, true);
if (FLAGS_benchmark) {
folly::runBenchmarks();
}
return RUN_ALL_TESTS();
}