Index: libcxx/include/concepts =================================================================== --- libcxx/include/concepts +++ libcxx/include/concepts @@ -163,6 +163,20 @@ concept derived_from = __is_base_of(_Bp, _Dp) && __is_convertible_to(const volatile _Dp*, const volatile _Bp*); +// [concepts.arithmetic] + +template +concept integral = __is_integral(_Tp); + +template +concept signed_integral = integral<_Tp> && __is_signed(_Tp); + +template +concept unsigned_integral = integral<_Tp> && !signed_integral<_Tp>; + +template +concept floating_point = __is_floating_point(_Tp); + #endif //_LIBCPP_STD_VER > 17 && defined(__cpp_concepts) && __cpp_concepts >= 201811L _LIBCPP_END_NAMESPACE_STD Index: libcxx/test/std/concepts/lang/arithmetic.pass.cpp =================================================================== --- /dev/null +++ libcxx/test/std/concepts/lang/arithmetic.pass.cpp @@ -0,0 +1,386 @@ +//===----------------------------------------------------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03, c++11, c++14, c++17 +// UNSUPPORTED: libcpp-no-concepts + +// template +// concept integral = // see below + +// template +// concept signed_integral = // see below + +// template +// concept unsigned_integral = // see below + +// template +// concept floating_point = // see below + +#include +#include + +namespace { +constexpr void CheckIntegral() { + // conventional integral types + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + + // integrals that mightn't necessarily be expected as integral + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + static_assert(std::integral); + + // types that aren't integral + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + static_assert(!std::integral); + + enum ClassicEnum { a, b, c }; + static_assert(!std::integral); + + enum class ScopedEnum { x, y, z }; + static_assert(!std::integral); + + struct EmptyStruct {}; + static_assert(!std::integral); + + struct WithMember { + int Sneaky; + }; + static_assert(!std::integral); + static_assert(!std::integral); + + struct WithMemberFunction { + int Sneaky(); + }; + static_assert(!std::integral); + + struct WithConstMemFunc { + int Sneaky() const; + }; + static_assert(!std::integral); + + struct WithLvalMemberFunction { + int Sneaky() &; + }; + static_assert(!std::integral); + + struct WithLvalConstMemFunc { + int Sneaky() const&; + }; + static_assert(!std::integral); + + struct WithRvalMemberFunction { + int Sneaky() &&; + }; + static_assert(!std::integral); + + struct WithRvalConstMemFunc { + int Sneaky() const&&; + }; + static_assert(!std::integral); +} + +constexpr void CheckSignedIntegral() { + // conventional signed integral types + static_assert(std::signed_integral); + static_assert(std::signed_integral); + static_assert(std::signed_integral); + static_assert(std::signed_integral); + static_assert(std::signed_integral); + static_assert(std::signed_integral); + static_assert(std::signed_integral); + static_assert(std::signed_integral); + + // integrals that mightn't necessarily be expected as signed integral + static_assert(std::signed_integral == std::is_signed_v); + static_assert(std::signed_integral == std::is_signed_v); + static_assert(std::signed_integral == std::is_signed_v); + static_assert(std::signed_integral == std::is_signed_v); + static_assert(std::signed_integral == std::is_signed_v); + static_assert(std::signed_integral == std::is_signed_v); + + // types that aren't signed integrals + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + + enum ClassicEnum { a, b, c }; + static_assert(!std::signed_integral); + + enum class ScopedEnum { x, y, z }; + static_assert(!std::signed_integral); + + struct EmptyStruct {}; + static_assert(!std::signed_integral); + + struct WithMember { + int Sneaky; + }; + static_assert(!std::signed_integral); + static_assert(!std::signed_integral); + + struct WithMemberFunction { + int Sneaky(); + }; + static_assert(!std::signed_integral); + + struct WithConstMemFunc { + int Sneaky() const; + }; + static_assert(!std::signed_integral); + + struct WithLvalMemberFunction { + int Sneaky() &; + }; + static_assert(!std::signed_integral); + + struct WithLvalConstMemFunc { + int Sneaky() const&; + }; + static_assert(!std::signed_integral); + + struct WithRvalMemberFunction { + int Sneaky() &&; + }; + static_assert(!std::signed_integral); + + struct WithRvalConstMemFunc { + int Sneaky() const&&; + }; + static_assert(!std::signed_integral); +} + +constexpr void CheckUnsignedIntegral() { + // conventional unsigned integral types + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + static_assert(std::unsigned_integral); + + // integrals that mightn't necessarily be expected as unsigned integral + static_assert(std::unsigned_integral == !std::is_signed_v); + static_assert(std::unsigned_integral == !std::is_signed_v); + static_assert(std::unsigned_integral == !std::is_signed_v); + static_assert(std::unsigned_integral == !std::is_signed_v); + static_assert(std::unsigned_integral == + !std::is_signed_v); + static_assert(std::unsigned_integral == + !std::is_signed_v); + + // types that aren't unsigned integrals + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + + enum ClassicEnum { a, b, c }; + static_assert(!std::unsigned_integral); + + enum class ScopedEnum { x, y, z }; + static_assert(!std::unsigned_integral); + + struct EmptyStruct {}; + static_assert(!std::unsigned_integral); + + struct WithMember { + int Sneaky; + }; + static_assert(!std::unsigned_integral); + static_assert(!std::unsigned_integral); + + struct WithMemberFunction { + int Sneaky(); + }; + static_assert(!std::unsigned_integral); + + struct WithConstMemFunc { + int Sneaky() const; + }; + static_assert(!std::unsigned_integral); + + struct WithLvalMemberFunction { + int Sneaky() &; + }; + static_assert(!std::unsigned_integral); + + struct WithLvalConstMemFunc { + int Sneaky() const&; + }; + static_assert(!std::unsigned_integral); + + struct WithRvalMemberFunction { + int Sneaky() &&; + }; + static_assert(!std::unsigned_integral); + + struct WithRvalConstMemFunc { + int Sneaky() const&&; + }; + static_assert(!std::unsigned_integral); +} + +constexpr void CheckFloatingPoint() { + // floating-point types + static_assert(std::floating_point); + static_assert(std::floating_point); + static_assert(std::floating_point); + static_assert(std::floating_point); + static_assert(std::floating_point); + static_assert(std::floating_point); + + // types that aren't floating-point + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + static_assert(!std::floating_point); + + enum ClassicEnum { a, b, c }; + static_assert(!std::floating_point); + + enum class ScopedEnum { x, y, z }; + static_assert(!std::floating_point); + + struct EmptyStruct {}; + static_assert(!std::floating_point); + + struct WithMember { + int Sneaky; + }; + static_assert(!std::floating_point); + static_assert(!std::floating_point); + + struct WithMemberFunction { + int Sneaky(); + }; + static_assert(!std::floating_point); + + struct WithConstMemFunc { + int Sneaky() const; + }; + static_assert(!std::floating_point); + + struct WithLvalMemberFunction { + int Sneaky() &; + }; + static_assert(!std::floating_point); + + struct WithLvalConstMemFunc { + int Sneaky() const&; + }; + static_assert(!std::floating_point); + + struct WithRvalMemberFunction { + int Sneaky() &&; + }; + static_assert(!std::floating_point); + + struct WithRvalConstMemFunc { + int Sneaky() const&&; + }; + static_assert(!std::floating_point); +} +} // namespace + +int main(int, char**) { + CheckIntegral(); + CheckSignedIntegral(); + CheckUnsignedIntegral(); + CheckFloatingPoint(); + return 0; +}