[PATCH] D35472: Implement P0463R1: "Endian just Endian"
Marshall Clow via Phabricator via cfe-commits
cfe-commits at lists.llvm.org
Sun Jul 16 20:08:12 PDT 2017
mclow.lists created this revision.
Implement the C++2a feature "A compile time endian-ness detection idiom"
Howard's suggested implementation is:
enum class endian
{
#ifdef _WIN32
little = 0,
big = 1,
native = little
#else
little = __ORDER_LITTLE_ENDIAN__,
big = __ORDER_BIG_ENDIAN__
native = __BYTE_ORDER__,
#endif
};
but libc++ has it's own macros that have done the work to detect this.
The other option would be to rip out `_LIBCPP_LITTLE_ENDIAN` and `_LIBCPP_BIG_ENDIAN`, which is tempting, but that would entail changes in code that has to compile for earlier C++ language versions.
https://reviews.llvm.org/D35472
Files:
include/type_traits
test/std/utilities/meta/meta.type.synop/endian.pass.cpp
Index: test/std/utilities/meta/meta.type.synop/endian.pass.cpp
===================================================================
--- test/std/utilities/meta/meta.type.synop/endian.pass.cpp
+++ test/std/utilities/meta/meta.type.synop/endian.pass.cpp
@@ -0,0 +1,46 @@
+//===----------------------------------------------------------------------===//
+//
+// The LLVM Compiler Infrastructure
+//
+// This file is dual licensed under the MIT and the University of Illinois Open
+// Source Licenses. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+// UNSUPPORTED: c++98, c++03, c++14, c++17
+
+// enum class endian;
+
+#include <type_traits>
+#include <cassert>
+
+#include "test_macros.h"
+
+int main() {
+ typedef std::endian E;
+ static_assert(std::is_enum<std::endian>::value, "");
+
+// Check that E is a scoped enum by checking for conversions.
+ typedef std::underlying_type<std::endian>::type UT;
+ static_assert(!std::is_convertible<std::endian, UT>::value, "");
+
+// test that the enumeration values exist
+ static_assert( std::endian::little == std::endian::little );
+ static_assert( std::endian::big == std::endian::big );
+ static_assert( std::endian::native == std::endian::native );
+ static_assert( std::endian::little != std::endian::big );
+
+// Technically not required, but true on all existing machines
+ static_assert( std::endian::native == std::endian::little ||
+ std::endian::native == std::endian::big );
+
+// Try to check at runtime
+ {
+ union {
+ uint32_t i;
+ char c[4];
+ } u = {0x01020304};
+
+ assert ((u.c[0] == 1) == (std::endian::native == std::endian::big));
+ }
+}
Index: include/type_traits
===================================================================
--- include/type_traits
+++ include/type_traits
@@ -4737,6 +4737,21 @@
#endif
+#if _LIBCPP_STD_VER > 14
+enum class endian
+{
+ little = 0xDEAD,
+ big = 0xFACE,
+#if _LIBCPP_LITTLE_ENDIAN
+ native = little
+#elif _LIBCPP_BIG_ENDIAN
+ native = big
+#else
+ native = 0xCAFE
+#endif
+};
+#endif
+
_LIBCPP_END_NAMESPACE_STD
#if _LIBCPP_STD_VER > 14
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D35472.106831.patch
Type: text/x-patch
Size: 2245 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/cfe-commits/attachments/20170717/68a60011/attachment.bin>
More information about the cfe-commits
mailing list