-
Notifications
You must be signed in to change notification settings - Fork 0
Developer Guide Engine
The mixing engine is the part of Mixxx that is in charge of resampling, amplifying, clipping, and mixing the audio from decks and samplers into a master and headphone output.
Hundreds of times per second, the operating system's audio API requests
a certain number of audio samples from Mixxx. This request is delivered
to the SoundManager
class via an operating system callback (see
SoundManager (OS audio interface)).
SoundManager
in turn requests that Mixxx's engine produce and mix
together the next buffer of audio.
The operating system callback requesting samples from Mixxx is running in what we call the callback thread. This is usually a realtime thread and is performance sensitive. Doing any kind of I/O or locking of mutexes in this thread is highly discouraged. Anything that can block the callback thread is in danger of causing user-audible skips (called xruns or buffer under-runs) in the output audio.
The goal of the callback thread is to fulfill the operating system's request for the next buffer of audio to play out the computer's speakers. The length of this buffer depends on the latency and samplerate settings the user has configured their soundcard at (configurable in the Mixxx Sound Hardware preferences).
At a latency of X
milliseconds and a samplerate of Y
samples per
second per channel, and stereo channels the number of samples that Mixxx
must generate to fill the buffer is given by this simple relationship:
X * Y * 2
.
For example:
double latency = 0.001; // 1 millisecond
int sampleRate = 44100; // 44.1 thousand samples per second (kHz)
int numChannels = 2; // stereo, 2 channels
int samples_per_buffer = sampleRate * latency * numChannels;
At a latency of 1 millisecond, the operating system will request buffers of audio every 1 millisecond or 1000 times per second.
Almost all mixing components in the engine follow the EngineObject interface. This interface is very simple:
typedef float CSAMPLE;
class EngineObject : public QObject {
Q_OBJECT
public:
EngineObject();
virtual ~EngineObject();
virtual void process(const CSAMPLE *pIn, const CSAMPLE *pOut, const int iLen) = 0;
};
As you can see, this interface contains only one interesting method,
process
. process
takes a buffer of CSAMPLE
values as input and a
buffer of CSAMPLE
values to output, and a number of samples iLen
.
The EngineObject
processes the input audio in pIn
, doing whatever
work it is that it is designed to, and writes the resulting output to
pOut
.
Almost all components of the mixing engine implement this interface. The benefit is that the mixing engine is modular and you can mix and match different mixing components together to get the desired chain of audio processing hooked up.
NOTE: By convention if pIn
and pOut
are equal, it is required that
the EngineObject
should do its work in-place.
EngineMaster
is the master class that drives the entire mixing engine.
SoundManager
calls EngineMaster
directly to request that the next
buffer of audio be generated.
EngineMaster
, like most engine classes, is an EngineObject
and all
of its interesting work is done in its process
method.
Terminology Alert: In the engine, there are two different types of
channels. In the context of the EngineMaster
mixing together different
sources of audio, a channel is a source of audio like a deck, a sampler,
a microphone, etc. In the context of buffers of audio, the number of
channels refers to how many different signals are present in the buffer
(e.g. mono, stereo, multi-channel). Mixxx's mixing engine usually only
deals with stereo audio.
EngineMaster
supports mixing multiple streams of audio together. To
add a channel of audio to EngineMaster
you must create an
EngineChannel
class that represents your channel of audio. For
example, decks use the EngineDeck
, samplers use the EngineSampler
class, and microphones use the EngineMicrophone
class. All 3 of these
are children of EngineChannel
. To add a sampler or deck or microphone
to EngineMaster
you call the addChannel
method on EngineMaster
.
src/engine/enginemaster.cpp EngineMaster::addChannel
As you will find in mixxx.cpp
:
EngineMicrophone* pMicrophone = new EngineMicrophone("[Microphone]");
m_pEngine->addChannel(pMicrophone);
This registers an EngineMicrophone
class with the EngineMaster
. When
mixing the master and headphone outputs, EngineMaster
will query the
EngineMicrophone
that is created for whether it is active, and if so,
ask it to process
itself to generate audio. Once EngineMicrophone
generates audio, EngineMaster
will mix that audio into the master
output.
In EngineMaster::process
, the EngineMaster
does many tasks related
to mixing the audio together. First it looks for all active
EngineChannel
s and then calls process
on each one of them so that
they each generate the audio from their channel to be mixed in this
callback. Next, the EngineMaster
applies the volume to each channel
and adds their sample data to the headphone and master outputs,
depending on the results of EngineChannel::isPFL()
and
EngineChannel::isMaster()
.
After mixing the headphone and master outputs, a series of
EngineObject
s are run on them to perform some post-processing:
-
EngineClipping
-- Clips the master audio to within [-32767, 32768] and provides a clipping indicator control. - Balance is applied to the master output based on the
[Master],balance
control. -
EngineVuMeter
-- Measures the spectral audio energy of the signal and updates VU meter controls. - The master output is submitted to the
EngineSidechain
- The master output is added to the headphone output with a gain
proportional to the
[Master],headMix
control. - The
[Master],headVolume
headphone volume gain is applied to the headphone output -
EngineClipping
is applied to the headphone output, clipping the audio to within [-32767, 32768] and provides a clipping indicator control.
Modulo a couple minor details, this is all there is to
EngineMaster::process
. After process
is done, SoundManager
gets
the master, headphone, and possibly individual channel buffers via the
EngineMaster::buffer
, EngineMaster::getDeckBuffer
, and
EngineMaster::getChannelBuffer
methods.
EngineChannel
is the interface that all audio channels must implement
to integrate with EngineMaster
.
The following methods are used by EngineMaster
to determine how to mix
the EngineChannel
:
-
isActive()
-- if this method returns true then theEngineChannel
is asked to produce audio via itsprocess
method. -
isPFL()
-- if this method returns true then the result of theprocess
call will be mixed into the engine PFL (pre-fader listen, headphone) output. -
EngineChannel
's default implementation ofisPFL()
looks at the value of anpfl
control to determine whether the channel should be heard in the headphone output. This allows other parts of Mixxx to control whether a channel is heard in the headphones or not. -
isMaster()
-- if this method returns true then the result of theprocess
call will be mixed into the engine master output. -
EngineChannel
's default implementation ofisMaster()
always returns true. -
getOrientation()
-- the return of this method determines what orientation thisEngineChannel
has. Orientations can be the left-side of the crossfader, the center (not affected by the crossfader), and right side of the crossfader. -
EngineChannel
's default implementation ofgetOrientation()
looks at the value of anorientation
control to determine which mix orientation the channel should have. This allows other parts of Mixxx to control which side of the crossfader a channel is oriented on.
Decks and samplers are fundamentally the same thing to the mixing
engine. They are both represented by the EngineDeck
class, which is a
sub-class of EngineChannel
. If you take a look at the EngineDeck
implementation in src/engine/enginedeck.cpp
you'll see that it is
pretty straightforward and composed of a small list of EngineObject
s
which process the audio for each deck and sampler.
The list of EngineObject
s that are run in-order when
EngineDeck::process
is called are:
-
EngineBuffer
-- (See also Introduction to Mixxx's Deck/Sampler Processing) Contains almost all player logic -- decodes, re-samples audio, processes loops, hotcues, and syncing. -
EngineVinylSoundEmu
-- Emulates the response of a vinyl record to changes in speed by applying a gain proportional to the speed of the player. -
EnginePregain
-- Applies gain and replaygain to the audio. -
EngineFilterBlock
-- Applies EQ filters (low, mid, high) to the audio. -
EngineFlanger
-- Applies the flanger effect, if enabled. (This will be removed in the future in favor of a generic effects framework) -
EngineClipping
-- Clips the audio to within [-32767, 32768] and provides a clipping indicator control. -
EngineVuMeter
-- Measures the spectral audio energy of the signal and updates VU meter controls.
The resulting buffer of audio is mixed into the master and headphone
outputs by EngineMaster
.
- The
isActive
method is implemented byEngineDeck
and is purely dependent on whether a track is loaded in the deck. - The
isPFL
method is implemented byEngineChannel
. - The
isMaster
method is implemented byEngineChannel
and is always true. - The
getOrientation
method is implemented byEngineChannel
.
Mixxx is a free and open-source DJ software.
Manual
Hardware Compatibility
Reporting Bugs
Getting Involved
Contribution Guidelines
Coding Guidelines
Using Git
Developer Guide
Contributing Mappings
Mixxx Controls
MIDI Scripting
Components JS