diff options
Diffstat (limited to 'akode/lib/converter.cpp')
-rw-r--r-- | akode/lib/converter.cpp | 148 |
1 files changed, 148 insertions, 0 deletions
diff --git a/akode/lib/converter.cpp b/akode/lib/converter.cpp new file mode 100644 index 0000000..2169868 --- /dev/null +++ b/akode/lib/converter.cpp @@ -0,0 +1,148 @@ +/* aKode: Converter + + Copyright (C) 2005,2006 Allan Sandfeld Jensen <kde@carewolf.com> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, + Boston, MA 02110-1301, USA. +*/ + +#include "arithmetic.h" +#include "audioframe.h" +#include "converter.h" + +namespace aKode { + +Converter::Converter(int sample_width) : m_sample_width(sample_width) {} + +template<typename T, typename S, template<typename T> class ArithmT, template<typename S> class ArithmS> +static bool __doFrameFP(AudioFrame* in, AudioFrame* out, int sample_width) +{ + AudioConfiguration config = *in; + config.sample_width = sample_width; + + if (out) + out->reserveSpace(&config, in->length); + else + out = in; + + // ### Use doubles if double sized or 32bit samples are used. + float scale = 1.0; + scale = ArithmS<S>::max(sample_width)/(float)ArithmT<T>::max(in->sample_width); + + int channels = in->channels; + int length = in->length; + + T** indata = (T**)in->data; + S** outdata = (S**)out->data; + for(int i=0; i<channels; i++) + for(int j=0; j<length; j++) + outdata[i][j] = (S)(scale*indata[i][j]); + + return true; +} + +template<typename T, typename S> +static bool __doFrame(AudioFrame* in, AudioFrame* out, int sample_width) +{ + AudioConfiguration config = *in; + config.sample_width = sample_width; + + if (out) + out->reserveSpace(&config, in->length); + else + out = in; + + int width_T = sizeof(T)*8; + int shift = width_T - sample_width; + + int channels = in->channels; + uint32_t length = in->length; + + T** indata = (T**)in->data; + S** outdata = (S**)out->data; + for(int i=0; i<channels; i++) + for(uint32_t j=0; j<length; j++) + outdata[i][j] = (S)(indata[i][j] >> shift); + + out->sample_width = sample_width; + return true; +} + +template<typename T> +static inline bool _doFrameFP(AudioFrame* in, AudioFrame* out, int sample_width) +{ + if (in->sample_width == -64) { + return __doFrameFP<double, T, Arithm_FP, Arithm_FP>(in, out, sample_width); + } else + if (in->sample_width == -32) { + return __doFrameFP<float, T, Arithm_FP, Arithm_FP>(in, out, sample_width); + } else + if (in->sample_width <= 8) { + return __doFrameFP<int8_t, T, Arithm_Int, Arithm_FP>(in, out, sample_width); + } else + if (in->sample_width <= 16) { + return __doFrameFP<int16_t, T, Arithm_Int, Arithm_FP>(in, out, sample_width); + } else + return __doFrameFP<int32_t, T, Arithm_Int, Arithm_FP>(in, out, sample_width); + +} + +template<typename T> +static inline bool _doFrame(AudioFrame* in, AudioFrame* out, int sample_width) +{ + if (in->sample_width == -64) { + return __doFrameFP<double, T, Arithm_FP, Arithm_Int>(in, out, sample_width); + } else + if (in->sample_width == -32) { + return __doFrameFP<float, T, Arithm_FP, Arithm_Int>(in, out, sample_width); + } else + if (in->sample_width <= 8) { + return __doFrame<int8_t, T>(in, out, sample_width); + } else + if (in->sample_width <= 16) { + return __doFrame<int16_t, T>(in, out, sample_width); + } else + return __doFrame<int32_t, T>(in, out, sample_width); +} + +bool Converter::doFrame(AudioFrame* in, AudioFrame* out) +{ + if (m_sample_width == 0) return false; + if (!out && in->sample_width == m_sample_width) return true; + + if (m_sample_width < 0) { + if (m_sample_width == -64) + return _doFrameFP<double>(in, out, m_sample_width); + else + return _doFrameFP<float>(in, out, m_sample_width); + } else + if (m_sample_width <= 8) { + return _doFrame<int8_t>(in, out, m_sample_width); + } else + if (m_sample_width <= 16) { + return _doFrame<int16_t>(in, out, m_sample_width); + } else + return _doFrame<int32_t>(in, out, m_sample_width); + + return false; +} + +void Converter::setSampleWidth(int sample_width) +{ + m_sample_width = sample_width; +} + + +} // namespace |