/* vim:set ts=2 sw=2 sts=2 et: */ /** * \author Marcus Holland-Moritz (github@mhxnet.de) * \copyright Copyright (c) Marcus Holland-Moritz * * This file is part of dwarfs. * * dwarfs is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * dwarfs is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with dwarfs. If not, see . */ #include #include #include #include #include #include namespace dwarfs { namespace { template class basic_pcm_sample_transformer { public: using uint_type = std::make_unsigned_t; template static constexpr void unpack(UnpackedType* dst, uint8_t const* src, int bits) { uint_type tmp; if constexpr (End == pcm_sample_endianness::Big) { if constexpr (Bytes == 1) { tmp = (static_cast(src[0]) << 0); } if constexpr (Bytes == 2) { tmp = (static_cast(src[0]) << 8) | (static_cast(src[1]) << 0); } if constexpr (Bytes == 3) { tmp = (static_cast(src[0]) << 16) | (static_cast(src[1]) << 8) | (static_cast(src[2]) << 0); } if constexpr (Bytes == 4) { tmp = (static_cast(src[0]) << 24) | (static_cast(src[1]) << 16) | (static_cast(src[2]) << 8) | (static_cast(src[3]) << 0); } } else { if constexpr (Bytes == 1) { tmp = (static_cast(src[0]) << 0); } if constexpr (Bytes == 2) { tmp = (static_cast(src[0]) << 0) | (static_cast(src[1]) << 8); } if constexpr (Bytes == 3) { tmp = (static_cast(src[0]) << 0) | (static_cast(src[1]) << 8) | (static_cast(src[2]) << 16); } if constexpr (Bytes == 4) { tmp = (static_cast(src[0]) << 0) | (static_cast(src[1]) << 8) | (static_cast(src[2]) << 16) | (static_cast(src[3]) << 24); } } *dst = unpack_native(tmp, bits); } template static constexpr void pack(uint8_t* dst, UnpackedType const* src, int bits) { auto tmp = pack_native(*src, bits); if constexpr (End == pcm_sample_endianness::Big) { if constexpr (Bytes == 1) { dst[0] = static_cast((tmp >> 0) & 0xFF); } if constexpr (Bytes == 2) { dst[0] = static_cast((tmp >> 8) & 0xFF); dst[1] = static_cast((tmp >> 0) & 0xFF); } if constexpr (Bytes == 3) { dst[0] = static_cast((tmp >> 16) & 0xFF); dst[1] = static_cast((tmp >> 8) & 0xFF); dst[2] = static_cast((tmp >> 0) & 0xFF); } if constexpr (Bytes == 4) { dst[0] = static_cast((tmp >> 24) & 0xFF); dst[1] = static_cast((tmp >> 16) & 0xFF); dst[2] = static_cast((tmp >> 8) & 0xFF); dst[3] = static_cast((tmp >> 0) & 0xFF); } } else { if constexpr (Bytes == 1) { dst[0] = static_cast((tmp >> 0) & 0xFF); } if constexpr (Bytes == 2) { dst[0] = static_cast((tmp >> 0) & 0xFF); dst[1] = static_cast((tmp >> 8) & 0xFF); } if constexpr (Bytes == 3) { dst[0] = static_cast((tmp >> 0) & 0xFF); dst[1] = static_cast((tmp >> 8) & 0xFF); dst[2] = static_cast((tmp >> 16) & 0xFF); } if constexpr (Bytes == 4) { dst[0] = static_cast((tmp >> 0) & 0xFF); dst[1] = static_cast((tmp >> 8) & 0xFF); dst[2] = static_cast((tmp >> 16) & 0xFF); dst[3] = static_cast((tmp >> 24) & 0xFF); } } } private: template static constexpr UnpackedType unpack_native(uint_type src, int bits) { if constexpr (Pad == pcm_sample_padding::Lsb) { src >>= (8 * Bytes - bits); } if constexpr (Sig == pcm_sample_signedness::Signed) { if (bits < 8 * static_cast(sizeof(uint_type))) { if (src & (1 << (bits - 1))) { src |= (~static_cast(0)) << bits; } } return static_cast(src); } else { return static_cast(src) - (1 << (bits - 1)); } } template static constexpr uint_type pack_native(UnpackedType src, int bits) { if constexpr (Sig == pcm_sample_signedness::Unsigned) { src += (1 << (bits - 1)); } if constexpr (Pad == pcm_sample_padding::Lsb) { return static_cast(src << (8 * Bytes - bits)); } else { return static_cast(src); } } }; template class pcm_sample_transformer_fixed final : public pcm_sample_transformer::impl { public: using basic_transformer = basic_pcm_sample_transformer; void unpack(std::span dst, std::span src) const override { assert(Bytes * dst.size() == src.size()); for (size_t i = 0; i < dst.size(); ++i) { basic_transformer::template unpack( &dst[i], &src[Bytes * i], Bits); } } void pack(std::span dst, std::span src) const override { assert(dst.size() == Bytes * src.size()); for (size_t i = 0; i < src.size(); ++i) { basic_transformer::template pack(&dst[Bytes * i], &src[i], Bits); } } }; template class pcm_sample_transformer_generic final : public pcm_sample_transformer::impl { public: using basic_transformer = basic_pcm_sample_transformer; explicit pcm_sample_transformer_generic(int bits) : bits_{bits} {} void unpack(std::span dst, std::span src) const override { assert(Bytes * dst.size() == src.size()); for (size_t i = 0; i < dst.size(); ++i) { basic_transformer::template unpack( &dst[i], &src[Bytes * i], bits_); } } void pack(std::span dst, std::span src) const override { assert(dst.size() == Bytes * src.size()); for (size_t i = 0; i < src.size(); ++i) { basic_transformer::template pack(&dst[Bytes * i], &src[i], bits_); } } private: int bits_; }; template std::unique_ptr::impl> make_pcm_sample_transformer(int bits) { static_assert(1 <= Bytes && Bytes <= 4); if constexpr (Bytes == 1) { if (bits == 8) { return std::make_unique< pcm_sample_transformer_fixed>(); } return std::make_unique< pcm_sample_transformer_generic>(bits); } if constexpr (Bytes == 2) { if (bits == 16) { return std::make_unique< pcm_sample_transformer_fixed>(); } return std::make_unique< pcm_sample_transformer_generic>(bits); } if constexpr (Bytes == 3) { if (bits == 20) { return std::make_unique< pcm_sample_transformer_fixed>(); } if (bits == 24) { return std::make_unique< pcm_sample_transformer_fixed>(); } return std::make_unique< pcm_sample_transformer_generic>(bits); } if constexpr (Bytes == 4) { if (bits == 20) { return std::make_unique< pcm_sample_transformer_fixed>(); } if (bits == 24) { return std::make_unique< pcm_sample_transformer_fixed>(); } if (bits == 32) { return std::make_unique< pcm_sample_transformer_fixed>(); } return std::make_unique< pcm_sample_transformer_generic>(bits); } } template std::unique_ptr::impl> make_pcm_sample_transformer(int bytes, int bits) { switch (bytes) { case 1: return make_pcm_sample_transformer(bits); case 2: return make_pcm_sample_transformer(bits); case 3: return make_pcm_sample_transformer(bits); case 4: return make_pcm_sample_transformer(bits); default: throw std::runtime_error( fmt::format("unsupported number of bytes per sample: {}", bytes)); } } template std::unique_ptr::impl> make_pcm_sample_transformer(pcm_sample_padding pad, int bytes, int bits) { switch (pad) { case pcm_sample_padding::Lsb: return make_pcm_sample_transformer(bytes, bits); case pcm_sample_padding::Msb: return make_pcm_sample_transformer(bytes, bits); } folly::assume_unreachable(); } template std::unique_ptr::impl> make_pcm_sample_transformer(pcm_sample_signedness sig, pcm_sample_padding pad, int bytes, int bits) { switch (sig) { case pcm_sample_signedness::Signed: return make_pcm_sample_transformer( pad, bytes, bits); case pcm_sample_signedness::Unsigned: return make_pcm_sample_transformer( pad, bytes, bits); } folly::assume_unreachable(); } template std::unique_ptr::impl> make_pcm_sample_transformer(pcm_sample_endianness end, pcm_sample_signedness sig, pcm_sample_padding pad, int bytes, int bits) { assert(bits <= 8 * bytes); switch (end) { case pcm_sample_endianness::Big: return make_pcm_sample_transformer(sig, pad, bytes, bits); case pcm_sample_endianness::Little: return make_pcm_sample_transformer( sig, pad, bytes, bits); } folly::assume_unreachable(); } } // namespace template pcm_sample_transformer::pcm_sample_transformer( pcm_sample_endianness end, pcm_sample_signedness sig, pcm_sample_padding pad, int bytes, int bits) : impl_{make_pcm_sample_transformer(end, sig, pad, bytes, bits)} {} template class pcm_sample_transformer; std::ostream& operator<<(std::ostream& os, pcm_sample_endianness e) { os << (e == pcm_sample_endianness::Big ? "big-endian" : "little-endian"); return os; } std::ostream& operator<<(std::ostream& os, pcm_sample_signedness s) { os << (s == pcm_sample_signedness::Signed ? "signed" : "unsigned"); return os; } std::ostream& operator<<(std::ostream& os, pcm_sample_padding p) { os << (p == pcm_sample_padding::Lsb ? "lsb-padded" : "msb-padded"); return os; } } // namespace dwarfs