mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			Define std::byte for C++17 (P0298R3)
* doc/xml/manual/status_cxx2017.xml: Document std::byte support. * include/c_global/cstddef (std::byte): Define for C++17. * testsuite/18_support/byte/global_neg.cc: New test. * testsuite/18_support/byte/ops.cc: New test. * testsuite/18_support/byte/requirements.cc: New test. From-SVN: r246005
This commit is contained in:
		
							parent
							
								
									1f1fd3e272
								
							
						
					
					
						commit
						e0472d7e8c
					
				|  | @ -1,3 +1,11 @@ | |||
| 2017-03-09  Jonathan Wakely  <jwakely@redhat.com> | ||||
| 
 | ||||
| 	* doc/xml/manual/status_cxx2017.xml: Document std::byte support. | ||||
| 	* include/c_global/cstddef (std::byte): Define for C++17. | ||||
| 	* testsuite/18_support/byte/global_neg.cc: New test. | ||||
| 	* testsuite/18_support/byte/ops.cc: New test. | ||||
| 	* testsuite/18_support/byte/requirements.cc: New test. | ||||
| 
 | ||||
| 2017-03-05  Jonathan Wakely  <jwakely@redhat.com> | ||||
| 
 | ||||
| 	* doc/xml/manual/status_cxx2017.xml: Document P0156R2 status. | ||||
|  |  | |||
|  | @ -459,18 +459,6 @@ Feature-testing recommendations for C++</link>. | |||
|       <entry><code> __cpp_lib_has_unique_object_representations >= 201606 </code></entry> | ||||
|     </row> | ||||
| 
 | ||||
|     <row> | ||||
|       <?dbhtml bgcolor="#C8B0B0" ?> | ||||
|       <entry> Ordered by default </entry> | ||||
|       <entry> | ||||
| 	<link xmlns:xlink="http://www.w3.org/1999/xlink" xlink:href="http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/p0181r1.html"> | ||||
| 	P0181R1 | ||||
| 	</link> | ||||
|       </entry> | ||||
|       <entry align="center"> No </entry> | ||||
|       <entry><code> __cpp_lib_default_order >= 201606</code></entry> | ||||
|     </row> | ||||
| 
 | ||||
|     <row> | ||||
|       <entry> Polishing <code><chrono></code> </entry> | ||||
|       <entry> | ||||
|  | @ -729,6 +717,19 @@ Feature-testing recommendations for C++</link>. | |||
|       <entry><code> __cpp_lib_scoped_lock >= 201703 </code></entry> | ||||
|     </row> | ||||
| 
 | ||||
|     <row> | ||||
|       <?dbhtml bgcolor="#C8B0B0" ?> | ||||
|       <entry>  </entry> byte type definition | ||||
|       <entry> | ||||
| 	<link xmlns:xlink="http://www.w3.org/1999/xlink" xlink:href="http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2017/p0298r3.pdf"> | ||||
| 	P0298R3 | ||||
| 	</link> | ||||
|       </entry> | ||||
|       <entry align="center"> 7 </entry> | ||||
|       <entry><code> ??? </code></entry> | ||||
|     </row> | ||||
| 
 | ||||
| 
 | ||||
|   </tbody> | ||||
| </tgroup> | ||||
| </table> | ||||
|  |  | |||
|  | @ -57,4 +57,133 @@ namespace std | |||
| } | ||||
| #endif | ||||
| 
 | ||||
| #if __cplusplus > 201402L | ||||
| namespace std | ||||
| { | ||||
|   /// std::byte | ||||
|   enum class byte : unsigned char {}; | ||||
| 
 | ||||
|   template<typename _IntegerType> struct __byte_operand; | ||||
|   template<> struct __byte_operand<bool> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<char> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<signed char> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned char> { using __type = byte; }; | ||||
| #ifdef _GLIBCXX_USE_WCHAR_T | ||||
|   template<> struct __byte_operand<wchar_t> { using __type = byte; }; | ||||
| #endif | ||||
|   template<> struct __byte_operand<char16_t> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<char32_t> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<short> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned short> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<int> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned int> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<long> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned long> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<long long> { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned long long> { using __type = byte; }; | ||||
| #if defined(__GLIBCXX_TYPE_INT_N_0) | ||||
|   template<> struct __byte_operand<__GLIBCXX_TYPE_INT_N_0> | ||||
|   { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned __GLIBCXX_TYPE_INT_N_0> | ||||
|   { using __type = byte; }; | ||||
| #endif | ||||
| #if defined(__GLIBCXX_TYPE_INT_N_1) | ||||
|   template<> struct __byte_operand<__GLIBCXX_TYPE_INT_N_1> | ||||
|   { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned __GLIBCXX_TYPE_INT_N_1> | ||||
|   { using __type = byte; }; | ||||
| #endif | ||||
| #if defined(__GLIBCXX_TYPE_INT_N_2) | ||||
|   template<> struct __byte_operand<__GLIBCXX_TYPE_INT_N_2> | ||||
|   { using __type = byte; }; | ||||
|   template<> struct __byte_operand<unsigned __GLIBCXX_TYPE_INT_N_2> | ||||
|   { using __type = byte; }; | ||||
| #endif | ||||
|   template<typename _IntegerType> | ||||
|     struct __byte_operand<const _IntegerType> | ||||
|     : __byte_operand<_IntegerType> { }; | ||||
|   template<typename _IntegerType> | ||||
|     struct __byte_operand<volatile _IntegerType> | ||||
|     : __byte_operand<_IntegerType> { }; | ||||
|   template<typename _IntegerType> | ||||
|     struct __byte_operand<const volatile _IntegerType> | ||||
|     : __byte_operand<_IntegerType> { }; | ||||
| 
 | ||||
|   template<typename _IntegerType> | ||||
|     using __byte_op_t = typename __byte_operand<_IntegerType>::__type; | ||||
| 
 | ||||
|   template<typename _IntegerType> | ||||
|     constexpr __byte_op_t<_IntegerType>& | ||||
|     operator<<=(byte& __b, _IntegerType __shift) noexcept | ||||
|     { return __b = byte(static_cast<unsigned char>(__b) << __shift); } | ||||
| 
 | ||||
|   template<typename _IntegerType> | ||||
|     constexpr __byte_op_t<_IntegerType> | ||||
|     operator<<(byte __b, _IntegerType __shift) noexcept | ||||
|     { return byte(static_cast<unsigned char>(__b) << __shift); } | ||||
| 
 | ||||
|   template<typename _IntegerType> | ||||
|     constexpr __byte_op_t<_IntegerType>& | ||||
|     operator>>=(byte& __b, _IntegerType __shift) noexcept | ||||
|     { return __b = byte(static_cast<unsigned char>(__b) >> __shift); } | ||||
| 
 | ||||
|   template<typename _IntegerType> | ||||
|     constexpr __byte_op_t<_IntegerType> | ||||
|     operator>>(byte __b, _IntegerType __shift) noexcept | ||||
|     { return byte(static_cast<unsigned char>(__b) >> __shift); } | ||||
| 
 | ||||
|   constexpr byte& | ||||
|   operator|=(byte& __l, byte __r) noexcept | ||||
|   { | ||||
|     return __l = | ||||
|       byte(static_cast<unsigned char>(__l) | static_cast<unsigned char>(__r)); | ||||
|   } | ||||
| 
 | ||||
|   constexpr byte | ||||
|   operator|(byte __l, byte __r) noexcept | ||||
|   { | ||||
|     return | ||||
|       byte(static_cast<unsigned char>(__l) | static_cast<unsigned char>(__r)); | ||||
|   } | ||||
| 
 | ||||
|   constexpr byte& | ||||
|   operator&=(byte& __l, byte __r) noexcept | ||||
|   { | ||||
|    return __l = | ||||
|      byte(static_cast<unsigned char>(__l) & static_cast<unsigned char>(__r)); | ||||
|   } | ||||
| 
 | ||||
|   constexpr byte | ||||
|   operator&(byte __l, byte __r) noexcept | ||||
|   { | ||||
|     return | ||||
|       byte(static_cast<unsigned char>(__l) & static_cast<unsigned char>(__r)); | ||||
|   } | ||||
| 
 | ||||
|   constexpr byte& | ||||
|   operator^=(byte& __l, byte __r) noexcept | ||||
|   { | ||||
|     return __l = | ||||
|       byte(static_cast<unsigned char>(__l) ^ static_cast<unsigned char>(__r)); | ||||
|   } | ||||
| 
 | ||||
|   constexpr byte | ||||
|   operator^(byte __l, byte __r) noexcept | ||||
|   { | ||||
|     return | ||||
|       byte(static_cast<unsigned char>(__l) ^ static_cast<unsigned char>(__r)); | ||||
|   } | ||||
| 
 | ||||
|   constexpr byte | ||||
|   operator~(byte __b) noexcept | ||||
|   { return byte(~static_cast<unsigned char>(__b)); } | ||||
| 
 | ||||
|   template<typename _IntegerType> | ||||
|     constexpr _IntegerType | ||||
|     to_integer(__byte_op_t<_IntegerType> __b) noexcept | ||||
|     { return _IntegerType(__b); } | ||||
| 
 | ||||
| } // namespace std | ||||
| #endif | ||||
| 
 | ||||
| #endif // _GLIBCXX_CSTDDEF | ||||
|  |  | |||
|  | @ -0,0 +1,24 @@ | |||
| // Copyright (C) 2017 Free Software Foundation, Inc.
 | ||||
| //
 | ||||
| // This file is part of the GNU ISO C++ Library.  This library is free
 | ||||
| // software; you can redistribute it and/or modify it under the
 | ||||
| // terms of the GNU General Public License as published by the
 | ||||
| // Free Software Foundation; either version 3, or (at your option)
 | ||||
| // any later version.
 | ||||
| 
 | ||||
| // This library is distributed in the hope that it will be useful,
 | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||
| // GNU General Public License for more details.
 | ||||
| 
 | ||||
| // You should have received a copy of the GNU General Public License along
 | ||||
| // with this library; see the file COPYING3.  If not see
 | ||||
| // <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| // { dg-options "-std=gnu++17" }
 | ||||
| // { dg-do compile { target c++1z } }
 | ||||
| 
 | ||||
| #include <stddef.h> | ||||
| 
 | ||||
| byte b;			// { dg-error "does not name a type" }
 | ||||
| using std::byte;	// { dg-error "has not been declared" }
 | ||||
|  | @ -0,0 +1,224 @@ | |||
| // Copyright (C) 2017 Free Software Foundation, Inc.
 | ||||
| //
 | ||||
| // This file is part of the GNU ISO C++ Library.  This library is free
 | ||||
| // software; you can redistribute it and/or modify it under the
 | ||||
| // terms of the GNU General Public License as published by the
 | ||||
| // Free Software Foundation; either version 3, or (at your option)
 | ||||
| // any later version.
 | ||||
| 
 | ||||
| // This library is distributed in the hope that it will be useful,
 | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||
| // GNU General Public License for more details.
 | ||||
| 
 | ||||
| // You should have received a copy of the GNU General Public License along
 | ||||
| // with this library; see the file COPYING3.  If not see
 | ||||
| // <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| // { dg-options "-std=gnu++17" }
 | ||||
| // { dg-do compile { target c++1z } }
 | ||||
| 
 | ||||
| #include <cstddef> | ||||
| 
 | ||||
| constexpr bool is_byte(std::byte) { return true; } | ||||
| template<typename T> constexpr bool is_byte(T) { return false; } | ||||
| 
 | ||||
| template<typename T> | ||||
| constexpr bool test_lshift_assign(unsigned char c, T t) | ||||
| { | ||||
|   std::byte b{c}; | ||||
|   b <<= t; | ||||
|   return b == std::byte(c << t); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_lshift_assign(0, 1) ); | ||||
| static_assert( test_lshift_assign(0, 1u) ); | ||||
| static_assert( test_lshift_assign(0, 1ll) ); | ||||
| static_assert( test_lshift_assign(4, 1) ); | ||||
| static_assert( test_lshift_assign(9, 2u) ); | ||||
| static_assert( test_lshift_assign(16, 3ll) ); | ||||
| static_assert( test_lshift_assign(127, 1) ); | ||||
| static_assert( test_lshift_assign(255, 2u) ); | ||||
| static_assert( test_lshift_assign(63, 3ll) ); | ||||
| 
 | ||||
| template<typename T> | ||||
| constexpr bool test_lshift(unsigned char c, T t) | ||||
| { | ||||
|   const std::byte b{c}; | ||||
|   const std::byte b2 = b << t; | ||||
|   return b2 == std::byte(c << t); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_lshift(0, 1) ); | ||||
| static_assert( test_lshift(0, 1u) ); | ||||
| static_assert( test_lshift(0, 1ll) ); | ||||
| static_assert( test_lshift(4, 1) ); | ||||
| static_assert( test_lshift(9, 2u) ); | ||||
| static_assert( test_lshift(16, 3ll) ); | ||||
| static_assert( test_lshift(127, 1) ); | ||||
| static_assert( test_lshift(255, 2u) ); | ||||
| static_assert( test_lshift(63, 3ll) ); | ||||
| 
 | ||||
| template<typename T> | ||||
| constexpr bool test_rshift_assign(unsigned char c, T t) | ||||
| { | ||||
|   std::byte b{c}; | ||||
|   b >>= t; | ||||
|   return b == std::byte(c >> t); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_rshift_assign(0, 1) ); | ||||
| static_assert( test_rshift_assign(0, 1u) ); | ||||
| static_assert( test_rshift_assign(0, 1ll) ); | ||||
| static_assert( test_rshift_assign(4, 1) ); | ||||
| static_assert( test_rshift_assign(9, 2u) ); | ||||
| static_assert( test_rshift_assign(16, 3ll) ); | ||||
| static_assert( test_rshift_assign(127, 1) ); | ||||
| static_assert( test_rshift_assign(255, 2u) ); | ||||
| static_assert( test_rshift_assign(63, 3ll) ); | ||||
| 
 | ||||
| template<typename T> | ||||
| constexpr bool test_rshift(unsigned char c, T t) | ||||
| { | ||||
|   const std::byte b{c}; | ||||
|   const std::byte b2 = b >> t; | ||||
|   return b2 == std::byte(c >> t); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_rshift(0, 1) ); | ||||
| static_assert( test_rshift(0, 1u) ); | ||||
| static_assert( test_rshift(0, 1ll) ); | ||||
| static_assert( test_rshift(4, 1) ); | ||||
| static_assert( test_rshift(9, 2u) ); | ||||
| static_assert( test_rshift(16, 3ll) ); | ||||
| static_assert( test_rshift(127, 1) ); | ||||
| static_assert( test_rshift(255, 2u) ); | ||||
| static_assert( test_rshift(63, 3ll) ); | ||||
| 
 | ||||
| constexpr bool test_or_assign(unsigned char l, unsigned char r) | ||||
| { | ||||
|   std::byte b{l}; | ||||
|   b |= std::byte{r}; | ||||
|   return b == std::byte(l | r); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_or_assign(0, 1) ); | ||||
| static_assert( test_or_assign(4, 1) ); | ||||
| static_assert( test_or_assign(9, 2) ); | ||||
| static_assert( test_or_assign(16, 3) ); | ||||
| static_assert( test_or_assign(63, 3) ); | ||||
| static_assert( test_or_assign(127, 1) ); | ||||
| static_assert( test_or_assign(255, 2) ); | ||||
| 
 | ||||
| constexpr bool test_or(unsigned char l, unsigned char r) | ||||
| { | ||||
|   const std::byte b1{l}; | ||||
|   const std::byte b2{r}; | ||||
|   return (b1 | b2) == std::byte(l | r); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_or(0, 1) ); | ||||
| static_assert( test_or(0, 1u) ); | ||||
| static_assert( test_or(0, 1ll) ); | ||||
| static_assert( test_or(4, 1) ); | ||||
| static_assert( test_or(9, 2u) ); | ||||
| static_assert( test_or(16, 3ll) ); | ||||
| static_assert( test_or(127, 1) ); | ||||
| static_assert( test_or(255, 2u) ); | ||||
| static_assert( test_or(63, 3ll) ); | ||||
| 
 | ||||
| constexpr bool test_and_assign(unsigned char l, unsigned char r) | ||||
| { | ||||
|   std::byte b{l}; | ||||
|   b &= std::byte{r}; | ||||
|   return b == std::byte(l & r); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_and_assign(0, 1) ); | ||||
| static_assert( test_and_assign(0, 1u) ); | ||||
| static_assert( test_and_assign(0, 1ll) ); | ||||
| static_assert( test_and_assign(4, 1) ); | ||||
| static_assert( test_and_assign(9, 2u) ); | ||||
| static_assert( test_and_assign(16, 3ll) ); | ||||
| static_assert( test_and_assign(127, 1) ); | ||||
| static_assert( test_and_assign(255, 2u) ); | ||||
| static_assert( test_and_assign(63, 3ll) ); | ||||
| 
 | ||||
| constexpr bool test_and(unsigned char l, unsigned char r) | ||||
| { | ||||
|   const std::byte b1{l}; | ||||
|   const std::byte b2{r}; | ||||
|   return (b1 & b2) == std::byte(l & r); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_and(0, 1) ); | ||||
| static_assert( test_and(0, 1u) ); | ||||
| static_assert( test_and(0, 1ll) ); | ||||
| static_assert( test_and(4, 1) ); | ||||
| static_assert( test_and(9, 2u) ); | ||||
| static_assert( test_and(16, 3ll) ); | ||||
| static_assert( test_and(127, 1) ); | ||||
| static_assert( test_and(255, 2u) ); | ||||
| static_assert( test_and(63, 3ll) ); | ||||
| 
 | ||||
| constexpr bool test_xor_assign(unsigned char l, unsigned char r) | ||||
| { | ||||
|   std::byte b{l}; | ||||
|   b ^= std::byte{r}; | ||||
|   return b == std::byte(l ^ r); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_xor_assign(0, 1) ); | ||||
| static_assert( test_xor_assign(0, 1u) ); | ||||
| static_assert( test_xor_assign(0, 1ll) ); | ||||
| static_assert( test_xor_assign(4, 1) ); | ||||
| static_assert( test_xor_assign(9, 2u) ); | ||||
| static_assert( test_xor_assign(16, 3ll) ); | ||||
| static_assert( test_xor_assign(127, 1) ); | ||||
| static_assert( test_xor_assign(255, 2u) ); | ||||
| static_assert( test_xor_assign(63, 3ll) ); | ||||
| 
 | ||||
| constexpr bool test_xor(unsigned char l, unsigned char r) | ||||
| { | ||||
|   const std::byte b1{l}; | ||||
|   const std::byte b2{r}; | ||||
|   return (b1 ^ b2) == std::byte(l ^ r); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_xor(0, 1) ); | ||||
| static_assert( test_xor(0, 1u) ); | ||||
| static_assert( test_xor(0, 1ll) ); | ||||
| static_assert( test_xor(4, 1) ); | ||||
| static_assert( test_xor(9, 2u) ); | ||||
| static_assert( test_xor(16, 3ll) ); | ||||
| static_assert( test_xor(127, 1) ); | ||||
| static_assert( test_xor(255, 2u) ); | ||||
| static_assert( test_xor(63, 3ll) ); | ||||
| 
 | ||||
| constexpr bool test_complement(unsigned char c) | ||||
| { | ||||
|   const std::byte b{c}; | ||||
|   return ~b == std::byte(~c); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_complement(0) ); | ||||
| static_assert( test_complement(4) ); | ||||
| static_assert( test_complement(9) ); | ||||
| static_assert( test_complement(16) ); | ||||
| static_assert( test_complement(63) ); | ||||
| static_assert( test_complement(127) ); | ||||
| static_assert( test_complement(255) ); | ||||
| 
 | ||||
| template<typename T> | ||||
| constexpr bool test_to_integer(unsigned char c) | ||||
| { | ||||
|   std::byte b{c}; | ||||
|   return std::to_integer<T>(b) == T(c); | ||||
| } | ||||
| 
 | ||||
| static_assert( test_to_integer<int>(0) ); | ||||
| static_assert( test_to_integer<int>(255) ); | ||||
| static_assert( test_to_integer<signed char>(255) ); | ||||
| static_assert( test_to_integer<unsigned>(0) ); | ||||
| static_assert( test_to_integer<unsigned>(255) ); | ||||
| 
 | ||||
|  | @ -0,0 +1,27 @@ | |||
| // Copyright (C) 2017 Free Software Foundation, Inc.
 | ||||
| //
 | ||||
| // This file is part of the GNU ISO C++ Library.  This library is free
 | ||||
| // software; you can redistribute it and/or modify it under the
 | ||||
| // terms of the GNU General Public License as published by the
 | ||||
| // Free Software Foundation; either version 3, or (at your option)
 | ||||
| // any later version.
 | ||||
| 
 | ||||
| // This library is distributed in the hope that it will be useful,
 | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||
| // GNU General Public License for more details.
 | ||||
| 
 | ||||
| // You should have received a copy of the GNU General Public License along
 | ||||
| // with this library; see the file COPYING3.  If not see
 | ||||
| // <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| // { dg-options "-std=gnu++17" }
 | ||||
| // { dg-do compile { target c++1z } }
 | ||||
| 
 | ||||
| #include <cstddef> | ||||
| 
 | ||||
| static_assert( sizeof(std::byte) == sizeof(unsigned char) ); | ||||
| static_assert( alignof(std::byte) == alignof(unsigned char) ); | ||||
| 
 | ||||
| // Use the built-in to avoid depending on <type_traits>
 | ||||
| __underlying_type(std::byte)* p = (unsigned char*)nullptr; | ||||
		Loading…
	
		Reference in New Issue
	
	 Jonathan Wakely
						Jonathan Wakely