From 4636c7421cce996b68329de63db2a47453e4c289 Mon Sep 17 00:00:00 2001 From: Stanislaw Halik Date: Tue, 16 Jan 2024 02:07:58 +0100 Subject: a --- serialize/packbits-write.cpp | 31 ++++++++++++++--------- serialize/packbits-write.hpp | 60 +++++++++++++++++++++++++------------------- 2 files changed, 53 insertions(+), 38 deletions(-) (limited to 'serialize') diff --git a/serialize/packbits-write.cpp b/serialize/packbits-write.cpp index a5c80b49..045d6283 100644 --- a/serialize/packbits-write.cpp +++ b/serialize/packbits-write.cpp @@ -3,30 +3,37 @@ namespace floormat::detail_Pack_output { using u32 = uint32_t; +using u16 = uint16_t; using u8 = uint8_t; template using f32 = output_field; +template using f16 = output_field; template using f8 = output_field; -static_assert(count_bits, f32<3>, f32<5> >> == 10); -static_assert(count_bits, f8<2>, f8<4> >> == 7); -static_assert(count_bits> == 0); -//static_assert(count_bits >> == 0); -//static_assert(count_bits, f8<2> >> == 9); - -template u32 write_(const std::tuple>&, output, output_bits<32>, std::index_sequence<0>); -static_assert(write_(std::tuple>{4242}, output{0}, output_bits<32>{}, std::index_sequence<0>{}) == 4242); +static_assert(write_(std::tuple>{4242}, output{0}, std::index_sequence<0>{}) == 4242u); static_assert(write_( std::tuple{f8<3>{7}, f8<2>{3}, f8<1>{1}}, - output{0}, - output_bits<8>{}, + output{0}, std::make_index_sequence<3>{} ) == (1 << 6) - 1); static_assert(write_( std::tuple{f32<2>{0b10}, f32<3>{0b011}, f32<3>{0b001}}, - output{0}, - output_bits<32>{}, + output{0}, make_reverse_index_sequence<3>{}) == 0b000101110); +static_assert(write(std::tuple{f32<2>{0b10}, f32<3>{0b011}, f32<3>{0b01}}) == 0b00101110); +//static_assert(write(std::tuple{f32<2>{0b10}, f32<3>{0b1011}, f32<3>{0b001}}) == 0b000101110); +static_assert(write(std::tuple{f8<2>{0b10}, f8<3>{0b011}, f8<3>{0b01}}) == 0b00101110); +//static_assert(write(std::tuple{f8<2>{0b10}, f8<3>{0b011}, f8<4>{0b01}}) == 0b00101110); +//static_assert(write(std::tuple{}) == 0); + +#if 0 // check disasembly +u32 foo1(u32 a, u32 b, u32 c); +u32 foo1(u32 a, u32 b, u32 c) +{ + return write(std::tuple{f32<2>{a}, f32<3>{b}, f32<3>{c}}); +} +#endif + } // namespace floormat::detail_Pack_output diff --git a/serialize/packbits-write.hpp b/serialize/packbits-write.hpp index 1a91cbcf..09ec820e 100644 --- a/serialize/packbits-write.hpp +++ b/serialize/packbits-write.hpp @@ -1,61 +1,69 @@ #pragma once #include "compat/assert.hpp" #include -#include +#include #include +#include namespace floormat::detail_Pack_output { -template -struct output_bits final -{ - static_assert(N > 0); - static_assert(N < sizeof(size_t)*8); - - static constexpr size_t length = N; -}; - -template +template struct output { static_assert(std::is_fundamental_v); + static_assert(CAPACITY > 0); static_assert(CAPACITY <= sizeof(T)*8); - static constexpr size_t Capacity = CAPACITY; - + static_assert(LEFT <= CAPACITY); + static constexpr size_t Capacity = CAPACITY, Left = LEFT; T value{0}; }; -template +template struct output_field { - T value; + static_assert(LENGTH > 0); static constexpr size_t Length = LENGTH; + T value; }; -template struct count_bits_; - -template -constexpr std::index_sequence reverse_index_sequence(std::index_sequence const&); - -template +template +constexpr std::index_sequence reverse_index_sequence(std::index_sequence); +template using make_reverse_index_sequence = decltype(reverse_index_sequence(std::make_index_sequence{})); -template -constexpr T write_(const Tuple& tuple, output st, output_bits, std::index_sequence) +template struct is_output_field : std::bool_constant {}; +template struct is_output_field> : std::bool_constant { static_assert(N > 0); }; + +template +constexpr CORRADE_ALWAYS_INLINE T write_(const Tuple& tuple, output st, std::index_sequence) { + static_assert(Capacity > 0); + static_assert(Left > 0); static_assert(Capacity <= sizeof(T)*8); static_assert(Left <= Capacity); + static_assert(is_output_field(tuple))>>{}); constexpr size_t N = std::tuple_element_t::Length; static_assert(N <= Left); + T x = std::get(tuple).value; + fm_assert((size_t)std::bit_width(x) <= N); T value = T(T(st.value << N) | x); - return write_(tuple, output{value}, output_bits{}, std::index_sequence{}); + return write_(tuple, output{value}, std::index_sequence{}); } -template -constexpr T write_(const Tuple&, output st, output_bits, std::index_sequence<>) +template +constexpr CORRADE_ALWAYS_INLINE T write_(const Tuple&, output st, std::index_sequence<>) { return st.value; } +template +constexpr T write(const std::tuple...>& tuple) +{ + constexpr size_t nbits = sizeof(T)*8; + return write_(tuple, output{T{0}}, make_reverse_index_sequence{}); +} + +constexpr uint8_t write(const std::tuple<>&) = delete; + } // namespace floormat::detail_Pack_output -- cgit v1.2.3