forked from eden-emu/eden
		
	common/bitfield: make it endianness-aware
This commit is contained in:
		
							parent
							
								
									bed349ba51
								
							
						
					
					
						commit
						a2fa901591
					
				
					 3 changed files with 100 additions and 3 deletions
				
			
		|  | @ -34,6 +34,7 @@ | |||
| #include <limits> | ||||
| #include <type_traits> | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/swap.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Abstract bitfield class | ||||
|  | @ -108,7 +109,7 @@ | |||
|  * symptoms. | ||||
|  */ | ||||
| #pragma pack(1) | ||||
| template <std::size_t Position, std::size_t Bits, typename T> | ||||
| template <std::size_t Position, std::size_t Bits, typename T, typename EndianTag = LETag> | ||||
| struct BitField { | ||||
| private: | ||||
|     // We hide the copy assigment operator here, because the default copy
 | ||||
|  | @ -127,6 +128,8 @@ private: | |||
|     // We store the value as the unsigned type to avoid undefined behaviour on value shifting
 | ||||
|     using StorageType = std::make_unsigned_t<UnderlyingType>; | ||||
| 
 | ||||
|     using StorageTypeWithEndian = typename AddEndian<StorageType, EndianTag>::type; | ||||
| 
 | ||||
| public: | ||||
|     /// Constants to allow limited introspection of fields if needed
 | ||||
|     static constexpr std::size_t position = Position; | ||||
|  | @ -172,7 +175,7 @@ public: | |||
|     } | ||||
| 
 | ||||
|     constexpr FORCE_INLINE void Assign(const T& value) { | ||||
|         storage = (storage & ~mask) | FormatValue(value); | ||||
|         storage = (static_cast<StorageType>(storage) & ~mask) | FormatValue(value); | ||||
|     } | ||||
| 
 | ||||
|     constexpr T Value() const { | ||||
|  | @ -184,7 +187,7 @@ public: | |||
|     } | ||||
| 
 | ||||
| private: | ||||
|     StorageType storage; | ||||
|     StorageTypeWithEndian storage; | ||||
| 
 | ||||
|     static_assert(bits + position <= 8 * sizeof(T), "Bitfield out of range"); | ||||
| 
 | ||||
|  | @ -195,3 +198,6 @@ private: | |||
|     static_assert(std::is_trivially_copyable_v<T>, "T must be trivially copyable in a BitField"); | ||||
| }; | ||||
| #pragma pack() | ||||
| 
 | ||||
| template <std::size_t Position, std::size_t Bits, typename T> | ||||
| using BitFieldBE = BitField<Position, Bits, T, BETag>; | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Weiyi Wang
						Weiyi Wang