|  | // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/template_util.h" | 
|  |  | 
|  | #include <string> | 
|  |  | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  |  | 
|  | namespace base { | 
|  | namespace { | 
|  |  | 
|  | enum SimpleEnum { SIMPLE_ENUM }; | 
|  | enum EnumWithExplicitType : uint64_t { ENUM_WITH_EXPLICIT_TYPE }; | 
|  | enum class ScopedEnum { SCOPED_ENUM }; | 
|  | enum class ScopedEnumWithOperator { SCOPED_ENUM_WITH_OPERATOR }; | 
|  | std::ostream& operator<<(std::ostream& os, ScopedEnumWithOperator v) { | 
|  | return os; | 
|  | } | 
|  | struct SimpleStruct {}; | 
|  | struct StructWithOperator {}; | 
|  | std::ostream& operator<<(std::ostream& os, const StructWithOperator& v) { | 
|  | return os; | 
|  | } | 
|  |  | 
|  | // is_non_const_reference<Type> | 
|  | static_assert(!is_non_const_reference<int>::value, "IsNonConstReference"); | 
|  | static_assert(!is_non_const_reference<const int&>::value, | 
|  | "IsNonConstReference"); | 
|  | static_assert(is_non_const_reference<int&>::value, "IsNonConstReference"); | 
|  |  | 
|  | class AssignParent {}; | 
|  | class AssignChild : AssignParent {}; | 
|  |  | 
|  | // is_assignable<Type1, Type2> | 
|  | static_assert(!is_assignable<int, int>::value, "IsAssignable");  // 1 = 1; | 
|  | static_assert(!is_assignable<int, double>::value, "IsAssignable"); | 
|  | static_assert(is_assignable<int&, int>::value, "IsAssignable"); | 
|  | static_assert(is_assignable<int&, double>::value, "IsAssignable"); | 
|  | static_assert(is_assignable<int&, int&>::value, "IsAssignable"); | 
|  | static_assert(is_assignable<int&, int const&>::value, "IsAssignable"); | 
|  | static_assert(!is_assignable<int const&, int>::value, "IsAssignable"); | 
|  | static_assert(!is_assignable<AssignParent&, AssignChild>::value, | 
|  | "IsAssignable"); | 
|  | static_assert(!is_assignable<AssignChild&, AssignParent>::value, | 
|  | "IsAssignable"); | 
|  |  | 
|  | struct AssignCopy {}; | 
|  | struct AssignNoCopy { | 
|  | AssignNoCopy& operator=(AssignNoCopy&&) { return *this; } | 
|  | AssignNoCopy& operator=(const AssignNoCopy&) = delete; | 
|  | }; | 
|  | struct AssignNoMove { | 
|  | AssignNoMove& operator=(AssignNoMove&&) = delete; | 
|  | AssignNoMove& operator=(const AssignNoMove&) = delete; | 
|  | }; | 
|  |  | 
|  | static_assert(is_copy_assignable<AssignCopy>::value, "IsCopyAssignable"); | 
|  | static_assert(!is_copy_assignable<AssignNoCopy>::value, "IsCopyAssignable"); | 
|  |  | 
|  | static_assert(is_move_assignable<AssignCopy>::value, "IsMoveAssignable"); | 
|  | static_assert(is_move_assignable<AssignNoCopy>::value, "IsMoveAssignable"); | 
|  | static_assert(!is_move_assignable<AssignNoMove>::value, "IsMoveAssignable"); | 
|  |  | 
|  | // A few standard types that definitely support printing. | 
|  | static_assert(internal::SupportsOstreamOperator<int>::value, | 
|  | "ints should be printable"); | 
|  | static_assert(internal::SupportsOstreamOperator<const char*>::value, | 
|  | "C strings should be printable"); | 
|  | static_assert(internal::SupportsOstreamOperator<std::string>::value, | 
|  | "std::string should be printable"); | 
|  |  | 
|  | // Various kinds of enums operator<< support. | 
|  | static_assert(internal::SupportsOstreamOperator<SimpleEnum>::value, | 
|  | "simple enum should be printable by value"); | 
|  | static_assert(internal::SupportsOstreamOperator<const SimpleEnum&>::value, | 
|  | "simple enum should be printable by const ref"); | 
|  | static_assert(internal::SupportsOstreamOperator<EnumWithExplicitType>::value, | 
|  | "enum with explicit type should be printable by value"); | 
|  | static_assert( | 
|  | internal::SupportsOstreamOperator<const EnumWithExplicitType&>::value, | 
|  | "enum with explicit type should be printable by const ref"); | 
|  | static_assert(!internal::SupportsOstreamOperator<ScopedEnum>::value, | 
|  | "scoped enum should not be printable by value"); | 
|  | static_assert(!internal::SupportsOstreamOperator<const ScopedEnum&>::value, | 
|  | "simple enum should not be printable by const ref"); | 
|  | static_assert(internal::SupportsOstreamOperator<ScopedEnumWithOperator>::value, | 
|  | "scoped enum with operator<< should be printable by value"); | 
|  | static_assert( | 
|  | internal::SupportsOstreamOperator<const ScopedEnumWithOperator&>::value, | 
|  | "scoped enum with operator<< should be printable by const ref"); | 
|  |  | 
|  | // operator<< support on structs. | 
|  | static_assert(!internal::SupportsOstreamOperator<SimpleStruct>::value, | 
|  | "simple struct should not be printable by value"); | 
|  | static_assert(!internal::SupportsOstreamOperator<const SimpleStruct&>::value, | 
|  | "simple struct should not be printable by const ref"); | 
|  | static_assert(internal::SupportsOstreamOperator<StructWithOperator>::value, | 
|  | "struct with operator<< should be printable by value"); | 
|  | static_assert( | 
|  | internal::SupportsOstreamOperator<const StructWithOperator&>::value, | 
|  | "struct with operator<< should be printable by const ref"); | 
|  |  | 
|  | // underlying type of enums | 
|  | static_assert(std::is_integral<underlying_type<SimpleEnum>::type>::value, | 
|  | "simple enum must have some integral type"); | 
|  | static_assert( | 
|  | std::is_same<underlying_type<EnumWithExplicitType>::type, uint64_t>::value, | 
|  | "explicit type must be detected"); | 
|  | static_assert(std::is_same<underlying_type<ScopedEnum>::type, int>::value, | 
|  | "scoped enum defaults to int"); | 
|  |  | 
|  | struct TriviallyDestructible { | 
|  | int field; | 
|  | }; | 
|  |  | 
|  | class NonTriviallyDestructible { | 
|  | ~NonTriviallyDestructible() {} | 
|  | }; | 
|  |  | 
|  | static_assert(is_trivially_destructible<int>::value, "IsTriviallyDestructible"); | 
|  | static_assert(is_trivially_destructible<TriviallyDestructible>::value, | 
|  | "IsTriviallyDestructible"); | 
|  | static_assert(!is_trivially_destructible<NonTriviallyDestructible>::value, | 
|  | "IsTriviallyDestructible"); | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace base |