/* * SampleFormatCopy.h * ------------------ * Purpose: Functions for copying sample data. * Notes : (currently none) * Authors: OpenMPT Devs * The OpenMPT source code is released under the BSD license. Read LICENSE for more details. */ #pragma once #include "BuildSettings.h" #include "../common/Endianness.h" #include "SampleFormatConverters.h" OPENMPT_NAMESPACE_BEGIN ////////////////////////////////////////////////////// // Actual sample conversion functions // Copy a sample data buffer. // targetBuffer: Buffer in which the sample should be copied into. // numSamples: Number of samples of size T that should be copied. targetBuffer is expected to be able to hold "numSamples * incTarget" samples. // incTarget: Number of samples by which the target data pointer is increased each time. // sourceBuffer: Buffer from which the samples should be read. // sourceSize: Size of source buffer, in bytes. // incSource: Number of samples by which the source data pointer is increased each time. // // Template arguments: // SampleConversion: Functor of type SampleConversionFunctor to apply sample conversion (see above for existing functors). template size_t CopySample(typename SampleConversion::output_t * MPT_RESTRICT outBuf, size_t numSamples, size_t incTarget, const typename SampleConversion::input_t * MPT_RESTRICT inBuf, size_t sourceSize, size_t incSource, SampleConversion conv = SampleConversion()) { const size_t sampleSize = incSource * SampleConversion::input_inc * sizeof(typename SampleConversion::input_t); LimitMax(numSamples, sourceSize / sampleSize); const size_t copySize = numSamples * sampleSize; SampleConversion sampleConv(conv); while(numSamples--) { *outBuf = sampleConv(inBuf); outBuf += incTarget; inBuf += incSource * SampleConversion::input_inc; } return copySize; } // Copy numChannels interleaved sample streams. template void CopyInterleavedSampleStreams(typename SampleConversion::output_t * MPT_RESTRICT outBuf, const typename SampleConversion::input_t * MPT_RESTRICT inBuf, size_t numFrames, size_t numChannels, SampleConversion *conv) { while(numFrames--) { for(size_t channel = 0; channel < numChannels; ++channel) { *outBuf = conv[channel](*inBuf); inBuf++; outBuf++; } } } // Copy numChannels interleaved sample streams. template void CopyInterleavedSampleStreams(typename SampleConversion::output_t * MPT_RESTRICT outBuf, const typename SampleConversion::input_t * MPT_RESTRICT inBuf, size_t numFrames, size_t numChannels, std::vector &conv) { MPT_ASSERT(conv.size() >= numChannels); CopyInterleavedSampleStreams(outBuf, inBuf, numFrames, numChannels, &(conv[0])); } template void ConvertInterleavedFixedPointToInterleaved(Tsample * MPT_RESTRICT p, const Tfixed * MPT_RESTRICT mixbuffer, std::size_t channels, std::size_t count) { SC::ConvertFixedPoint conv; count *= channels; for(std::size_t i = 0; i < count; ++i) { p[i] = conv(mixbuffer[i]); } } template void ConvertInterleavedFixedPointToNonInterleaved(Tsample * const * const MPT_RESTRICT buffers, const Tfixed * MPT_RESTRICT mixbuffer, std::size_t channels, std::size_t count) { SC::ConvertFixedPoint conv; for(std::size_t i = 0; i < count; ++i) { for(std::size_t channel = 0; channel < channels; ++channel) { buffers[channel][i] = conv(*mixbuffer); mixbuffer++; } } } // Copy from an interleaed buffer of #channels. template void CopyInterleavedToChannel(typename SampleConversion::output_t * MPT_RESTRICT dst, const typename SampleConversion::input_t * MPT_RESTRICT src, std::size_t channels, std::size_t countChunk, std::size_t channel, SampleConversion conv = SampleConversion()) { SampleConversion sampleConv(conv); src += channel; for(std::size_t i = 0; i < countChunk; ++i) { *dst = sampleConv(*src); src += channels; dst++; } } // Copy buffer to an interleaed buffer of #channels. template void CopyChannelToInterleaved(typename SampleConversion::output_t * MPT_RESTRICT dst, const typename SampleConversion::input_t * MPT_RESTRICT src, std::size_t channels, std::size_t countChunk, std::size_t channel, SampleConversion conv = SampleConversion()) { SampleConversion sampleConv(conv); dst += channel; for(std::size_t i = 0; i < countChunk; ++i) { *dst = sampleConv(*src); src++; dst += channels; } } OPENMPT_NAMESPACE_END