[ox/mc] Fix decoding of int64s
This commit is contained in:
		
							
								
								
									
										28
									
								
								deps/ox/src/ox/mc/intops.hpp
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								deps/ox/src/ox/mc/intops.hpp
									
									
									
									
										vendored
									
									
								
							| @@ -8,6 +8,7 @@ | |||||||
|  |  | ||||||
| #pragma once | #pragma once | ||||||
|  |  | ||||||
|  | #include <ox/std/array.hpp> | ||||||
| #include <ox/std/assert.hpp> | #include <ox/std/assert.hpp> | ||||||
| #include <ox/std/bit.hpp> | #include <ox/std/bit.hpp> | ||||||
| #include <ox/std/byteswap.hpp> | #include <ox/std/byteswap.hpp> | ||||||
| @@ -55,7 +56,7 @@ static_assert(highestBit(uint64_t(1) << 31) == 31); | |||||||
| static_assert(highestBit(uint64_t(1) << 63) == 63); | static_assert(highestBit(uint64_t(1) << 63) == 63); | ||||||
|  |  | ||||||
| struct McInt { | struct McInt { | ||||||
| 	uint8_t data[9] = {0}; | 	uint8_t data[9] = {}; | ||||||
| 	// length of integer in bytes | 	// length of integer in bytes | ||||||
| 	std::size_t length = 0; | 	std::size_t length = 0; | ||||||
| }; | }; | ||||||
| @@ -113,7 +114,7 @@ constexpr McInt encodeInteger(I input) noexcept { | |||||||
|  * length integer. |  * length integer. | ||||||
|  */ |  */ | ||||||
| [[nodiscard]] | [[nodiscard]] | ||||||
| static constexpr std::size_t countBytes(uint8_t b) noexcept { | static constexpr std::size_t countBytes(unsigned b) noexcept { | ||||||
| 	std::size_t i = 0; | 	std::size_t i = 0; | ||||||
| 	while ((b >> i) & 1) ++i; | 	while ((b >> i) & 1) ++i; | ||||||
| 	return i + 1; | 	return i + 1; | ||||||
| @@ -130,13 +131,13 @@ static_assert(countBytes(0b0111'1111) == 8); | |||||||
| static_assert(countBytes(0b1111'1111) == 9); | static_assert(countBytes(0b1111'1111) == 9); | ||||||
|  |  | ||||||
| template<typename I> | template<typename I> | ||||||
| Result<I> decodeInteger(const uint8_t buff[9], std::size_t buffLen, std::size_t *bytesRead) noexcept { | constexpr Result<I> decodeInteger(const uint8_t buff[9], std::size_t buffLen, std::size_t *bytesRead) noexcept { | ||||||
| 	const auto bytes = countBytes(buff[0]); | 	const auto bytes = countBytes(buff[0]); | ||||||
| 	if (bytes == 9) { | 	if (bytes == 9) { | ||||||
| 		*bytesRead = bytes; | 		*bytesRead = bytes; | ||||||
| 		I out = 0; | 		I out = 0; | ||||||
| 		ox_memcpy(&out, &buff[1], sizeof(I)); | 		ox_memcpy(&out, &buff[1], sizeof(I)); | ||||||
| 		return static_cast<I>(LittleEndian<I>(out)); | 		return fromLittleEndian<I>(out); | ||||||
| 	} else if (buffLen >= bytes) { | 	} else if (buffLen >= bytes) { | ||||||
| 		*bytesRead = bytes; | 		*bytesRead = bytes; | ||||||
| 		uint64_t decoded = 0; | 		uint64_t decoded = 0; | ||||||
| @@ -150,28 +151,33 @@ Result<I> decodeInteger(const uint8_t buff[9], std::size_t buffLen, std::size_t | |||||||
| 			if (negative) { | 			if (negative) { | ||||||
| 				// fill in all bits between encoded sign and real sign with 1s | 				// fill in all bits between encoded sign and real sign with 1s | ||||||
| 				// split it up because the 32-bit ARM can't shift more than 32 bits | 				// split it up because the 32-bit ARM can't shift more than 32 bits | ||||||
| 				uint32_t *d = reinterpret_cast<uint32_t*>(&decoded); | 				ox::Array<uint32_t, 2> d = {}; | ||||||
|  | 				ox_memcpy(d.data(), &decoded, sizeof(decoded)); | ||||||
| 				auto bit = negBit; | 				auto bit = negBit; | ||||||
| 				for (; bit < ox::min<std::size_t>(Bits<I>, 32); ++bit) { | 				for (; bit < ox::min<std::size_t>(Bits<I>, 32); ++bit) { | ||||||
| 					d[0] |= 1 << bit; | 					d[0] |= 1 << bit; | ||||||
| 				} | 				} | ||||||
|  | 				bit -= 32; | ||||||
| 				for (; bit < Bits<I>; ++bit) { | 				for (; bit < Bits<I>; ++bit) { | ||||||
| 					d[1] |= 1 << bit; | 					d[1] |= 1 << bit; | ||||||
| 				} | 				} | ||||||
| 			} |  | ||||||
| 				I out = 0; | 				I out = 0; | ||||||
| 			ox_memcpy(&out, &decoded, sizeof(out)); | 				if constexpr(ox::defines::BigEndian) { | ||||||
| 			return out; | 					const auto d0Tmp = d[0]; | ||||||
| 		} else { | 					d[0] = d[1]; | ||||||
| 			auto out = static_cast<I>(decoded); | 					d[1] = d0Tmp; | ||||||
|  | 				} | ||||||
|  | 				ox_memcpy(&out, d.data(), sizeof(out)); | ||||||
| 				return out; | 				return out; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 		return static_cast<I>(decoded); | ||||||
|  | 	} | ||||||
| 	return OxError(1); | 	return OxError(1); | ||||||
| } | } | ||||||
|  |  | ||||||
| template<typename I> | template<typename I> | ||||||
| Result<I> decodeInteger(McInt m) noexcept { | constexpr Result<I> decodeInteger(McInt m) noexcept { | ||||||
| 	std::size_t bytesRead; | 	std::size_t bytesRead; | ||||||
| 	return decodeInteger<I>(m.data, 9, &bytesRead); | 	return decodeInteger<I>(m.data, 9, &bytesRead); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								deps/ox/src/ox/std/byteswap.hpp
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								deps/ox/src/ox/std/byteswap.hpp
									
									
									
									
										vendored
									
									
								
							| @@ -64,13 +64,27 @@ constexpr T conditionalByteSwap(T i) noexcept { | |||||||
| } | } | ||||||
|  |  | ||||||
| template<typename T> | template<typename T> | ||||||
|  | [[nodiscard]] | ||||||
|  | constexpr T fromLittleEndian(T i) noexcept { | ||||||
|  | 	return conditionalByteSwap<T, defines::BigEndian>(i); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<typename T> | ||||||
|  | [[nodiscard]] | ||||||
|  | constexpr T fromBigEndian(T i) noexcept { | ||||||
|  | 	return conditionalByteSwap<T, defines::LittleEndian>(i); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<typename T> | ||||||
|  | [[nodiscard]] | ||||||
| constexpr T toLittleEndian(T i) noexcept { | constexpr T toLittleEndian(T i) noexcept { | ||||||
| 	return conditionalByteSwap<T, defines::BigEndian>(i); | 	return conditionalByteSwap<T, defines::BigEndian>(i); | ||||||
| } | } | ||||||
|  |  | ||||||
| template<typename T> | template<typename T> | ||||||
|  | [[nodiscard]] | ||||||
| constexpr T toBigEndian(T i) noexcept { | constexpr T toBigEndian(T i) noexcept { | ||||||
| 	return conditionalByteSwap<T, defines::BigEndian>(i); | 	return conditionalByteSwap<T, defines::LittleEndian>(i); | ||||||
| } | } | ||||||
|  |  | ||||||
| template<typename T, bool byteSwap> | template<typename T, bool byteSwap> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user