#include "audiographer/general/sample_format_converter.h" #include "audiographer/exception.h" #include "audiographer/type_utils.h" #include "private/gdither/gdither.h" #include namespace AudioGrapher { template SampleFormatConverter::SampleFormatConverter (ChannelCount channels) : channels (channels), dither (0), data_out_size (0), data_out (0), clip_floats (false) { } template <> void SampleFormatConverter::init (nframes_t max_frames, int type, int data_width) { if (throw_level (ThrowObject) && data_width != 32) { throw Exception (*this, "Unsupported data width"); } init_common (max_frames); dither = gdither_new (GDitherNone, channels, GDitherFloat, data_width); } template <> void SampleFormatConverter::init (nframes_t max_frames, int type, int data_width) { if(throw_level (ThrowObject) && data_width < 24) { throw Exception (*this, "Trying to use SampleFormatConverter for data widths < 24"); } init_common (max_frames); if (data_width == 24) { dither = gdither_new ((GDitherType) type, channels, GDither32bit, data_width); } else if (data_width == 32) { dither = gdither_new (GDitherNone, channels, GDitherFloat, data_width); } else if (throw_level (ThrowObject)) { throw Exception (*this, "Unsupported data width"); } } template <> void SampleFormatConverter::init (nframes_t max_frames, int type, int data_width) { if (throw_level (ThrowObject) && data_width != 16) { throw Exception (*this, "Unsupported data width"); } init_common (max_frames); dither = gdither_new ((GDitherType) type, channels, GDither16bit, data_width); } template <> void SampleFormatConverter::init (nframes_t max_frames, int type, int data_width) { if (throw_level (ThrowObject) && data_width != 8) { throw Exception (*this, "Unsupported data width"); } init_common (max_frames); dither = gdither_new ((GDitherType) type, channels, GDither8bit, data_width); } template void SampleFormatConverter::init_common (nframes_t max_frames ) { reset(); if (max_frames > data_out_size) { delete[] data_out; data_out = new TOut[max_frames]; data_out_size = max_frames; } } template SampleFormatConverter::~SampleFormatConverter () { reset(); } template void SampleFormatConverter::reset() { if (dither) { gdither_free (dither); dither = 0; } delete[] data_out; data_out_size = 0; data_out = 0; clip_floats = false; } /* Basic const version of process() */ template void SampleFormatConverter::process (ProcessContext const & c_in) { float const * const data = c_in.data(); check_frame_and_channel_count (c_in.frames (), c_in.channels ()); /* Do conversion */ for (uint32_t chn = 0; chn < c_in.channels(); ++chn) { gdither_runf (dither, chn, c_in.frames_per_channel (), data, data_out); } /* Write forward */ ProcessContext c_out(c_in, data_out); output (c_out); } /* Basic non-const version of process(), calls the const one */ template void SampleFormatConverter::process (ProcessContext & c_in) { process (static_cast const &> (c_in)); } /* template specialization for float, in-place processing (non-const) */ template<> void SampleFormatConverter::process (ProcessContext & c_in) { nframes_t frames = c_in.frames(); float * data = c_in.data(); if (clip_floats) { for (nframes_t x = 0; x < frames; ++x) { if (data[x] > 1.0f) { data[x] = 1.0f; } else if (data[x] < -1.0f) { data[x] = -1.0f; } } } output (c_in); } /* template specialized const version, copies the data, and calls the non-const version */ template<> void SampleFormatConverter::process (ProcessContext const & c_in) { // Make copy of data and pass it to non-const version check_frame_and_channel_count (c_in.frames(), c_in.channels()); TypeUtils::copy (c_in.data(), data_out, c_in.frames()); ProcessContext c (c_in, data_out); process (c); } template void SampleFormatConverter::check_frame_and_channel_count(nframes_t frames, ChannelCount channels_) { if (throw_level (ThrowStrict) && channels_ != channels) { throw Exception (*this, boost::str (boost::format ("Wrong channel count given to process(), %1% instead of %2%") % channels_ % channels)); } if (throw_level (ThrowProcess) && frames > data_out_size) { throw Exception (*this, boost::str (boost::format ("Too many frames given to process(), %1% instad of %2%") % frames % data_out_size)); } } template class SampleFormatConverter; template class SampleFormatConverter; template class SampleFormatConverter; template class SampleFormatConverter; } // namespace