summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPavel Labath <pavel@labath.sk>2019-03-11 09:06:18 +0000
committerPavel Labath <pavel@labath.sk>2019-03-11 09:06:18 +0000
commit8b911934f3dedb1802edc62009a13aec067e98ef (patch)
treefd2e2728034a903cfb1eef866de7a9fc3aaf5987
parent5bca7ff98a103968ac63f9248b0d7b07e472f006 (diff)
downloadbcm5719-llvm-8b911934f3dedb1802edc62009a13aec067e98ef.tar.gz
bcm5719-llvm-8b911934f3dedb1802edc62009a13aec067e98ef.zip
[Support/Endian] Add support for endian-specific enums
Summary: Binary formats often include various enumerations or bitsets, but using endian-specific types for accessing them is tricky because they currently only support integral types. This is particularly true for scoped enums (enum class), as these are not implicitly convertible to integral types, and so one has to perform two casts just to read the enum value. This fixes that support by adding first-class support for enumeration types to endian-specific types. The support for them was already almost working -- all I needed to do was overload getSwappedBytes for enumeration types (which casts the enum to its underlying type and performs the conversion there). I also add some convenience template aliases to simplify declaring endian-specific enums. Reviewers: Bigcheese, zturner Subscribers: kristina, llvm-commits Tags: #llvm Differential Revision: https://reviews.llvm.org/D59141 llvm-svn: 355812
-rw-r--r--llvm/include/llvm/Support/Endian.h11
-rw-r--r--llvm/include/llvm/Support/SwapByteOrder.h7
-rw-r--r--llvm/unittests/Support/EndianTest.cpp9
3 files changed, 27 insertions, 0 deletions
diff --git a/llvm/include/llvm/Support/Endian.h b/llvm/include/llvm/Support/Endian.h
index 5c1e89ae793..d8be94427d7 100644
--- a/llvm/include/llvm/Support/Endian.h
+++ b/llvm/include/llvm/Support/Endian.h
@@ -338,6 +338,17 @@ using unaligned_int32_t =
using unaligned_int64_t =
detail::packed_endian_specific_integral<int64_t, native, unaligned>;
+template <typename T>
+using little_t = detail::packed_endian_specific_integral<T, little, unaligned>;
+template <typename T>
+using big_t = detail::packed_endian_specific_integral<T, big, unaligned>;
+
+template <typename T>
+using aligned_little_t =
+ detail::packed_endian_specific_integral<T, little, aligned>;
+template <typename T>
+using aligned_big_t = detail::packed_endian_specific_integral<T, big, aligned>;
+
namespace endian {
template <typename T> inline T read(const void *P, endianness E) {
diff --git a/llvm/include/llvm/Support/SwapByteOrder.h b/llvm/include/llvm/Support/SwapByteOrder.h
index f58b4d911d4..380ca0a781f 100644
--- a/llvm/include/llvm/Support/SwapByteOrder.h
+++ b/llvm/include/llvm/Support/SwapByteOrder.h
@@ -115,6 +115,13 @@ inline double getSwappedBytes(double C) {
return out.d;
}
+template <typename T>
+inline typename std::enable_if<std::is_enum<T>::value, T>::type
+getSwappedBytes(T C) {
+ return static_cast<T>(
+ getSwappedBytes(static_cast<typename std::underlying_type<T>::type>(C)));
+}
+
template<typename T>
inline void swapByteOrder(T &Value) {
Value = getSwappedBytes(Value);
diff --git a/llvm/unittests/Support/EndianTest.cpp b/llvm/unittests/Support/EndianTest.cpp
index c2b29288537..b5e4a9c8d14 100644
--- a/llvm/unittests/Support/EndianTest.cpp
+++ b/llvm/unittests/Support/EndianTest.cpp
@@ -200,4 +200,13 @@ TEST(Endian, PackedEndianSpecificIntegral) {
EXPECT_EQ(*big_val, *little_val);
}
+TEST(Endian, PacketEndianSpecificIntegralAsEnum) {
+ enum class Test : uint16_t { ONETWO = 0x0102, TWOONE = 0x0201 };
+ unsigned char bytes[] = {0x01, 0x02};
+ using LittleTest = little_t<Test>;
+ using BigTest = big_t<Test>;
+ EXPECT_EQ(Test::TWOONE, *reinterpret_cast<LittleTest *>(bytes));
+ EXPECT_EQ(Test::ONETWO, *reinterpret_cast<BigTest *>(bytes));
+}
+
} // end anon namespace
OpenPOWER on IntegriCloud