diff --git a/Makefile.am b/Makefile.am index 9866d2936..e1a5415d8 100644 --- a/Makefile.am +++ b/Makefile.am @@ -450,6 +450,8 @@ libpcm_a_SOURCES = \ src/pcm/PcmChannels.cxx src/pcm/PcmChannels.hxx \ src/pcm/PcmPack.cxx src/pcm/PcmPack.hxx \ src/pcm/PcmFormat.cxx src/pcm/PcmFormat.hxx \ + src/pcm/FloatConvert.hxx \ + src/pcm/ShiftConvert.hxx \ src/pcm/FormatConverter.cxx src/pcm/FormatConverter.hxx \ src/pcm/ChannelsConverter.cxx src/pcm/ChannelsConverter.hxx \ src/pcm/Resampler.hxx \ diff --git a/src/pcm/FloatConvert.hxx b/src/pcm/FloatConvert.hxx new file mode 100644 index 000000000..93e867159 --- /dev/null +++ b/src/pcm/FloatConvert.hxx @@ -0,0 +1,64 @@ +/* + * Copyright (C) 2003-2014 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program 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 2 of the License, or + * (at your option) any later version. + * + * This program 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 this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#ifndef MPD_PCM_FLOAT_CONVERT_HXX +#define MPD_PCM_FLOAT_CONVERT_HXX + +#include "Traits.hxx" + +/** + * Convert from float to an integer sample format. + */ +template> +struct FloatToIntegerSampleConvert { + typedef SampleTraits SrcTraits; + typedef Traits DstTraits; + + typedef typename SrcTraits::value_type SV; + typedef typename SrcTraits::long_type SL; + typedef typename DstTraits::value_type DV; + + static constexpr SV factor = 1 << (DstTraits::BITS - 1); + + gcc_const + static DV Convert(SV src) { + return PcmClamp(SL(src * factor)); + } +}; + +/** + * Convert from an integer sample format to float. + */ +template> +struct IntegerToFloatSampleConvert { + typedef SampleTraits DstTraits; + typedef Traits SrcTraits; + + typedef typename SrcTraits::value_type SV; + typedef typename DstTraits::value_type DV; + + static constexpr DV factor = 0.5 / (1 << (SrcTraits::BITS - 2)); + + gcc_const + static DV Convert(SV src) { + return DV(src) * factor; + } +}; + +#endif diff --git a/src/pcm/PcmFormat.cxx b/src/pcm/PcmFormat.cxx index 96f515f84..ca9f2e177 100644 --- a/src/pcm/PcmFormat.cxx +++ b/src/pcm/PcmFormat.cxx @@ -22,6 +22,8 @@ #include "PcmBuffer.hxx" #include "PcmUtils.hxx" #include "Traits.hxx" +#include "FloatConvert.hxx" +#include "ShiftConvert.hxx" #include "util/ConstBuffer.hxx" #include "util/WritableBuffer.hxx" @@ -34,73 +36,92 @@ ToConst(WritableBuffer b) return { b.data, b.size }; } -static void -pcm_convert_8_to_16(int16_t *out, const int8_t *in, size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] << 8; -} +/** + * Wrapper for a class that converts one sample at a time into one + * that converts a buffer at a time. + */ +template +struct PerSampleConvert : C { + typedef typename C::SrcTraits SrcTraits; + typedef typename C::DstTraits DstTraits; -static void -pcm_convert_24_to_16(PcmDither &dither, - int16_t *out, const int32_t *in, const int32_t *in_end) -{ - dither.Dither24To16(out, in, in_end); -} + void Convert(typename DstTraits::pointer_type gcc_restrict out, + typename SrcTraits::const_pointer_type gcc_restrict in, + size_t n) const { + for (size_t i = 0; i != n; ++i) + out[i] = C::Convert(in[i]); + } +}; -static void -pcm_convert_32_to_16(PcmDither &dither, - int16_t *out, const int32_t *in, const int32_t *in_end) -{ - dither.Dither32To16(out, in, in_end); -} +struct Convert8To16 + : PerSampleConvert> {}; + +struct Convert24To16 { + typedef SampleTraits SrcTraits; + typedef SampleTraits DstTraits; + + PcmDither &dither; + + Convert24To16(PcmDither &_dither):dither(_dither) {} + + void Convert(int16_t *out, const int32_t *in, size_t n) { + dither.Dither24To16(out, in, in + n); + } +}; + +struct Convert32To16 { + typedef SampleTraits SrcTraits; + typedef SampleTraits DstTraits; + + PcmDither &dither; + + Convert32To16(PcmDither &_dither):dither(_dither) {} + + void Convert(int16_t *out, const int32_t *in, size_t n) { + dither.Dither32To16(out, in, in + n); + } +}; template> -static void -ConvertFromFloat(typename Traits::pointer_type dest, - const float *src, size_t n) -{ - const float factor = 1 << (Traits::BITS - 1); +struct FloatToInteger + : PerSampleConvert> {}; - for (size_t i = 0; i != n; ++i) { - typename Traits::long_type sample(src[i] * factor); - dest[i] = PcmClamp(sample); - } +template +static WritableBuffer +AllocateConvert(PcmBuffer &buffer, C convert, + ConstBuffer src) +{ + auto dest = buffer.GetT(src.size); + convert.Convert(dest, src.data, src.size); + return { dest, src.size }; } template> static WritableBuffer AllocateFromFloat(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - ConvertFromFloat(dest, src.data, src.size); - return { dest, src.size }; + return AllocateConvert(buffer, FloatToInteger(), src); } static ConstBuffer pcm_allocate_8_to_16(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_8_to_16(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert8To16(), src)); } static ConstBuffer pcm_allocate_24p32_to_16(PcmBuffer &buffer, PcmDither &dither, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_24_to_16(dither, dest, src.data, src.end()); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert24To16(dither), src)); } static ConstBuffer pcm_allocate_32_to_16(PcmBuffer &buffer, PcmDither &dither, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_32_to_16(dither, dest, src.data, src.end()); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert32To16(dither), src)); } static ConstBuffer @@ -141,51 +162,34 @@ pcm_convert_to_16(PcmBuffer &buffer, PcmDither &dither, return nullptr; } -static void -pcm_convert_8_to_24(int32_t *out, const int8_t *in, size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] << 16; -} +struct Convert8To24 + : PerSampleConvert> {}; -static void -pcm_convert_16_to_24(int32_t *out, const int16_t *in, size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] << 8; -} - -static void -pcm_convert_32_to_24(int32_t *gcc_restrict out, - const int32_t *gcc_restrict in, - size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] >> 8; -} +struct Convert16To24 + : PerSampleConvert> {}; static ConstBuffer pcm_allocate_8_to_24(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_8_to_24(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert8To24(), src)); } static ConstBuffer pcm_allocate_16_to_24(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_16_to_24(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert16To24(), src)); } +struct Convert32To24 + : PerSampleConvert> {}; + static ConstBuffer pcm_allocate_32_to_24(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_32_to_24(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert32To24(), src)); } static WritableBuffer @@ -226,51 +230,34 @@ pcm_convert_to_24(PcmBuffer &buffer, return nullptr; } -static void -pcm_convert_8_to_32(int32_t *out, const int8_t *in, size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] << 24; -} +struct Convert8To32 + : PerSampleConvert> {}; -static void -pcm_convert_16_to_32(int32_t *out, const int16_t *in, size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] << 16; -} +struct Convert16To32 + : PerSampleConvert> {}; -static void -pcm_convert_24_to_32(int32_t *gcc_restrict out, - const int32_t *gcc_restrict in, - size_t n) -{ - for (size_t i = 0; i != n; ++i) - out[i] = in[i] << 8; -} +struct Convert24To32 + : PerSampleConvert> {}; static ConstBuffer pcm_allocate_8_to_32(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_8_to_32(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert8To32(), src)); } static ConstBuffer pcm_allocate_16_to_32(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_16_to_32(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert16To32(), src)); } static ConstBuffer pcm_allocate_24p32_to_32(PcmBuffer &buffer, ConstBuffer src) { - auto dest = buffer.GetT(src.size); - pcm_convert_24_to_32(dest, src.data, src.size); - return { dest, src.size }; + return ToConst(AllocateConvert(buffer, Convert24To32(), src)); } static ConstBuffer @@ -280,7 +267,7 @@ pcm_allocate_float_to_32(PcmBuffer &buffer, ConstBuffer src) auto dest = pcm_allocate_float_to_24(buffer, src); /* convert to 32 bit in-place */ - pcm_convert_24_to_32(dest.data, dest.data, src.size); + Convert24To32().Convert(dest.data, dest.data, src.size); return ToConst(dest); } @@ -316,49 +303,40 @@ pcm_convert_to_32(PcmBuffer &buffer, return nullptr; } -template> -static void -ConvertToFloat(float *dest, - typename Traits::const_pointer_type src, - size_t n) -{ - constexpr float factor = 0.5 / (1 << (Traits::BITS - 2)); - for (size_t i = 0; i != n; ++i) - dest[i] = float(src[i]) * factor; -} +struct Convert8ToFloat + : PerSampleConvert> {}; -template> -static ConstBuffer -AllocateToFloat(PcmBuffer &buffer, - ConstBuffer src) -{ - float *dest = buffer.GetT(src.size); - ConvertToFloat(dest, src.data, src.size); - return { dest, src.size }; -} +struct Convert16ToFloat + : PerSampleConvert> {}; + +struct Convert24ToFloat + : PerSampleConvert> {}; + +struct Convert32ToFloat + : PerSampleConvert> {}; static ConstBuffer pcm_allocate_8_to_float(PcmBuffer &buffer, ConstBuffer src) { - return AllocateToFloat(buffer, src); + return ToConst(AllocateConvert(buffer, Convert8ToFloat(), src)); } static ConstBuffer pcm_allocate_16_to_float(PcmBuffer &buffer, ConstBuffer src) { - return AllocateToFloat(buffer, src); + return ToConst(AllocateConvert(buffer, Convert16ToFloat(), src)); } static ConstBuffer pcm_allocate_24p32_to_float(PcmBuffer &buffer, ConstBuffer src) { - return AllocateToFloat(buffer, src); + return ToConst(AllocateConvert(buffer, Convert24ToFloat(), src)); } static ConstBuffer pcm_allocate_32_to_float(PcmBuffer &buffer, ConstBuffer src) { - return AllocateToFloat(buffer, src); + return ToConst(AllocateConvert(buffer, Convert32ToFloat(), src)); } ConstBuffer diff --git a/src/pcm/ShiftConvert.hxx b/src/pcm/ShiftConvert.hxx new file mode 100644 index 000000000..92f96b7ba --- /dev/null +++ b/src/pcm/ShiftConvert.hxx @@ -0,0 +1,69 @@ +/* + * Copyright (C) 2003-2014 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program 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 2 of the License, or + * (at your option) any later version. + * + * This program 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 this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#ifndef MPD_PCM_SHIFT_CONVERT_HXX +#define MPD_PCM_SHIFT_CONVERT_HXX + +#include "Traits.hxx" + +/** + * Convert from one integer sample format to another by shifting bits + * to the left. + */ +template, + class DT=SampleTraits> +struct LeftShiftSampleConvert { + typedef ST SrcTraits; + typedef DT DstTraits; + + typedef typename SrcTraits::value_type SV; + typedef typename DstTraits::value_type DV; + + static_assert(SrcTraits::BITS < DstTraits::BITS, + "Source format must be smaller than destination format"); + + constexpr static DV Convert(SV src) { + return DV(src) << (DstTraits::BITS - SrcTraits::BITS); + } +}; + +/** + * Convert from one integer sample format to another by shifting bits + * to the right. + */ +template, + class DT=SampleTraits> +struct RightShiftSampleConvert { + typedef ST SrcTraits; + typedef DT DstTraits; + + typedef typename SrcTraits::value_type SV; + typedef typename DstTraits::value_type DV; + + static_assert(SrcTraits::BITS > DstTraits::BITS, + "Source format must be smaller than destination format"); + + constexpr static DV Convert(SV src) { + return src >> (SrcTraits::BITS - DstTraits::BITS); + } +}; + +#endif