cog/Frameworks/OpenMPT.old/OpenMPT/soundlib/MixerInterface.h

110 lines
3.6 KiB
C++

/*
* MixerInterface.h
* ----------------
* Purpose: The basic mixer interface and main mixer loop, completely agnostic of the actual sample input / output formats.
* 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 "Snd_defs.h"
#include "ModChannel.h"
OPENMPT_NAMESPACE_BEGIN
class CResampler;
//////////////////////////////////////////////////////////////////////////
// Sample conversion traits
template<int channelsOut, int channelsIn, typename out, typename in>
struct MixerTraits
{
enum : int { numChannelsIn = channelsIn }; // Number of channels in sample
enum : int { numChannelsOut = channelsOut }; // Number of mixer output channels
typedef out output_t; // Output buffer sample type
typedef in input_t; // Input buffer sample type
typedef out outbuf_t[channelsOut]; // Output buffer sampling point type
// To perform sample conversion, add a function with the following signature to your derived classes:
// static MPT_CONSTEXPR11_FUN output_t Convert(const input_t x)
};
//////////////////////////////////////////////////////////////////////////
// Interpolation templates
template<class Traits>
struct NoInterpolation
{
MPT_FORCEINLINE void Start(const ModChannel &, const CResampler &) { }
MPT_FORCEINLINE void End(const ModChannel &) { }
MPT_FORCEINLINE void operator() (typename Traits::outbuf_t &outSample, const typename Traits::input_t * const inBuffer, const int32)
{
static_assert(static_cast<int>(Traits::numChannelsIn) <= static_cast<int>(Traits::numChannelsOut), "Too many input channels");
for(int i = 0; i < Traits::numChannelsIn; i++)
{
outSample[i] = Traits::Convert(inBuffer[i]);
}
}
};
// Other interpolation algorithms depend on the input format type (integer / float) and can thus be found in FloatMixer.h and IntMixer.h
//////////////////////////////////////////////////////////////////////////
// Main sample render loop template
// Template parameters:
// Traits: A class derived from MixerTraits that defines the number of channels, sample buffer types, etc..
// InterpolationFunc: Functor for reading the sample data and doing the SRC
// FilterFunc: Functor for applying the resonant filter
// MixFunc: Functor for mixing the computed sample data into the output buffer
template<class Traits, class InterpolationFunc, class FilterFunc, class MixFunc>
static void SampleLoop(ModChannel &chn, const CResampler &resampler, typename Traits::output_t * MPT_RESTRICT outBuffer, unsigned int numSamples)
{
ModChannel &c = chn;
const typename Traits::input_t * MPT_RESTRICT inSample = static_cast<const typename Traits::input_t *>(c.pCurrentSample);
InterpolationFunc interpolate;
FilterFunc filter;
MixFunc mix;
// Do initialisation if necessary
interpolate.Start(c, resampler);
filter.Start(c);
mix.Start(c);
unsigned int samples = numSamples;
SamplePosition smpPos = c.position; // Fixed-point sample position
const SamplePosition increment = c.increment; // Fixed-point sample increment
while(samples--)
{
typename Traits::outbuf_t outSample;
interpolate(outSample, inSample + smpPos.GetInt() * Traits::numChannelsIn, smpPos.GetFract());
filter(outSample, c);
mix(outSample, c, outBuffer);
outBuffer += Traits::numChannelsOut;
smpPos += increment;
}
mix.End(c);
filter.End(c);
interpolate.End(c);
c.position = smpPos;
}
// Type of the SampleLoop function above
typedef void (*MixFuncInterface)(ModChannel &, const CResampler &, mixsample_t *, unsigned int);
OPENMPT_NAMESPACE_END