Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for multi-channel plugins #7459

Open
wants to merge 78 commits into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
Show all changes
78 commits
Select commit Hold shift + click to select a range
93884a2
Add PluginPortConfig class
messmerd May 8, 2024
f1f1010
Avoid misleading types and weird casts
messmerd May 8, 2024
c78c17f
Use PluginPortConfig for mono VSTs
messmerd May 8, 2024
4da60af
Refactor; Fix automation clip tooltip
messmerd Jun 16, 2024
798e088
Begin pin connector implementation (WIP)
messmerd Aug 11, 2024
af4d6b3
Loading/saving (WIP)
messmerd Aug 11, 2024
aafe41c
More efficient loading
messmerd Aug 11, 2024
360eca6
Use new routing methods (WIP)
messmerd Aug 11, 2024
7b7f674
Progress (WIP)
messmerd Aug 12, 2024
1568d3a
Merge branch 'master' into pin-connector
messmerd Aug 12, 2024
5f31c04
Fix up post-merge (WIP)
messmerd Aug 12, 2024
853d80c
Fix crash
messmerd Aug 14, 2024
7935718
Begin implementing PluginPinConnectorView
messmerd Aug 14, 2024
b3ad72a
Progress (WIP)
messmerd Aug 14, 2024
b55bdec
Fix pin icon while dragging
messmerd Aug 14, 2024
bda93bc
More progress
messmerd Aug 15, 2024
15f395e
Draw plugin channel text
messmerd Aug 15, 2024
8d2acad
Fix Linux build
messmerd Aug 15, 2024
a00cdba
Some refactoring
messmerd Aug 16, 2024
c98353f
Fix build
messmerd Aug 16, 2024
774e3b6
Introduce MatrixView class
messmerd Aug 17, 2024
e768a58
Use layouts
messmerd Aug 17, 2024
2ee2c08
Make PluginPinConnector::Matrix a struct; Add tooltips
messmerd Aug 17, 2024
8445f2c
Set model display names
messmerd Aug 17, 2024
ecc6318
Fix a display bug
messmerd Aug 17, 2024
a317e16
Remove some unused code
messmerd Aug 17, 2024
2571547
Fix crash when reloading project with pin connector open
messmerd Aug 17, 2024
ada15d3
Fix miscalculation in mousePressEvent
messmerd Aug 18, 2024
a308725
Better pin connector button text; Better naming and comments
messmerd Aug 25, 2024
09eb59f
Use pointing hand cursor when hovering over cells
messmerd Aug 25, 2024
f6910c6
Make routing methods const
messmerd Sep 14, 2024
ba3ffc6
Improve routing normalization
messmerd Sep 14, 2024
af5c16b
Use bitwise OR
messmerd Sep 14, 2024
32557d4
Move some pin connector methods into Matrix
messmerd Sep 14, 2024
1716186
Use `const float*` for process's in buffer
messmerd Sep 14, 2024
23fd48b
Rename `m_trackChannelsUsed` to `m_trackChannelsUpperBound`
messmerd Sep 21, 2024
089ffb4
Merge branch 'master' into pin-connector
messmerd Sep 21, 2024
62efd0e
Various changes
messmerd Sep 23, 2024
b7883f1
Use `MAXIMUM_BUFFER_SIZE`
messmerd Sep 23, 2024
bd404df
Revert strongly typed sample types
messmerd Oct 16, 2024
dcf1fc9
Minor optimization in `mixInputs` lambda
messmerd Oct 16, 2024
f522c17
Merge branch 'master' into pin-connector
messmerd Nov 23, 2024
6e1d921
Pin connector refactor (#2)
messmerd Dec 14, 2024
d69c449
VST effect wet/dry mixing
messmerd Dec 14, 2024
afc24cc
Add pin connector button to EffectView
messmerd Dec 15, 2024
414e7c9
Remove old pin connector button from VstEffectControls
messmerd Dec 15, 2024
a56520f
Update pin connector button's tooltip when channel count changes
messmerd Dec 15, 2024
a43e121
Fix default connections for instruments with sidechain inputs
messmerd Dec 15, 2024
9136b7d
Draw arrows in and out of plugin channels; Simplifications
messmerd Dec 15, 2024
d4b9b43
Add pluginBuffersChange signal; improve error handling
messmerd Dec 17, 2024
5bd1ec3
Merge branch 'master' into pin-connector
messmerd Dec 22, 2024
2a271fd
Fix VST silent output
messmerd Dec 24, 2024
b81546a
Fix isMidiBased
messmerd Dec 30, 2024
36c99d9
Rename classes
messmerd Dec 31, 2024
3661bbd
Address review comments
messmerd Dec 31, 2024
262e540
Fix crash when loading VST instrument as a VST effect
messmerd Jan 8, 2025
b34df28
Use "summing" behavior rather than "averaging"
messmerd Jan 12, 2025
eadeabf
Plugin audio ports (#3)
messmerd Jan 29, 2025
b96b2f9
Merge branch 'master' into pin-connector
messmerd Jan 29, 2025
15b15cf
Add pin connector button for all instruments
messmerd Jan 31, 2025
0107d60
Include memory header
messmerd Jan 31, 2025
50dc34d
Merge branch 'master' into pin-connector
messmerd Feb 5, 2025
11aa3fa
Use std::span
messmerd Feb 5, 2025
b2b8b3b
Fix saving and loading
messmerd Feb 5, 2025
c132756
Use `@tparam` instead of `@param`
messmerd Feb 8, 2025
28aca19
Pass audio port type as template template parameter
messmerd Feb 9, 2025
f3a8b3f
Remove `CoreAudioData` and `CoreAudioDataMut`
messmerd Feb 9, 2025
b4736cc
Merge branch 'master' into pin-connector
messmerd Feb 9, 2025
1b5d0c8
Merge branch 'master' into pin-connector
messmerd Mar 2, 2025
7d68f44
Rename classes
messmerd Mar 6, 2025
488275e
Implement "direct routing" optimization
messmerd Mar 26, 2025
a3fd61b
Remove CoreAudioBus and CoreAudioBusMut type aliases
messmerd Mar 26, 2025
8931027
Add another unit test for direct routing
messmerd Mar 29, 2025
8174059
Test the track channel count
messmerd Mar 29, 2025
1b6040f
Replace some mentions of "plugin" with "processor"
messmerd Mar 31, 2025
89c8453
Use channel index types consistently to remove unnecessary casts
messmerd Mar 31, 2025
751e629
Merge branch 'master' into pin-connector
messmerd Mar 31, 2025
9814b54
Include optional
messmerd Mar 31, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
308 changes: 308 additions & 0 deletions include/AudioBuffer.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,308 @@
/*
* AudioBuffer.h - Customizable audio buffer
*
* Copyright (c) 2025 Dalton Messmer <messmer.dalton/at/gmail.com>
*
* This file is part of LMMS - https://lmms.io
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This program 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
* General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program (see COPYING); if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA.
*
*/

#ifndef LMMS_AUDIO_BUFFER_H
#define LMMS_AUDIO_BUFFER_H

#include <type_traits>
#include <vector>

#include "AudioData.h"
#include "AudioPortsConfig.h"
#include "LmmsTypes.h"
#include "SampleFrame.h"

namespace lmms
{


namespace detail {

/**
* Metafunction to select the appropriate non-owning audio buffer view
* given the layout, sample type, and channel count
*/
template<AudioDataKind kind, bool interleaved, proc_ch_t channels, bool isConst>
struct AudioDataViewTypeHelper
{
static_assert(always_false_v<AudioDataViewTypeHelper<kind, interleaved, channels, isConst>>,
"Unsupported audio data type");
};

//! Non-interleaved specialization
template<AudioDataKind kind, proc_ch_t channels, bool isConst>
struct AudioDataViewTypeHelper<kind, false, channels, isConst>
{
using type = SplitAudioData<
std::conditional_t<isConst, const GetAudioDataType<kind>, GetAudioDataType<kind>>,
channels>;
};

//! SampleFrame specialization
template<proc_ch_t channels, bool isConst>
struct AudioDataViewTypeHelper<AudioDataKind::SampleFrame, true, channels, isConst>
{
static_assert(channels == 0 || channels == 2,
"Plugins using SampleFrame buffers must have exactly 0 or 2 inputs or outputs");
using type = std::conditional_t<isConst, std::span<const SampleFrame>, std::span<SampleFrame>>;
};

} // namespace detail


//! Metafunction to select the appropriate non-owning audio buffer view
template<AudioPortsConfig config, bool isOutput, bool isConst>
using AudioDataViewType = typename detail::AudioDataViewTypeHelper<
config.kind, config.interleaved, (isOutput ? config.outputs : config.inputs), isConst>::type;


namespace detail {

//! Interface for accessing input/output audio buffers
template<AudioPortsConfig config, bool inplace = config.inplace>
class AudioBuffer;

//! Dynamically in-place specialization
template<AudioPortsConfig config>
class AudioBuffer<config, false>
{
public:
virtual ~AudioBuffer() = default;

virtual auto inputBuffer() -> AudioDataViewType<config, false, false> = 0;
virtual auto outputBuffer() -> AudioDataViewType<config, true, false> = 0;
virtual auto frames() const -> fpp_t = 0;
virtual void updateBuffers(proc_ch_t channelsIn, proc_ch_t channelsOut, f_cnt_t frames) = 0;
};

//! Statically in-place specialization
template<AudioPortsConfig config>
class AudioBuffer<config, true>
{
public:
virtual ~AudioBuffer() = default;

virtual auto inputOutputBuffer() -> AudioDataViewType<config, false, false> = 0;
virtual auto frames() const -> fpp_t = 0;
virtual void updateBuffers(proc_ch_t channelsIn, proc_ch_t channelsOut, f_cnt_t frames) = 0;
};


//! Optimization - Choose std::array or std::vector based on whether size is known at compile time
template<AudioPortsConfig config>
using AccessBufferType = std::conditional_t<
config.staticChannelCount(),
std::array<GetAudioDataType<config.kind>*, config.inputs + config.outputs>,
std::vector<GetAudioDataType<config.kind>*>>;


//! Default implementation of `AudioBuffer`
template<AudioPortsConfig config,
AudioDataKind kind = config.kind, bool interleaved = config.interleaved, bool inplace = config.inplace>
class DefaultAudioBuffer;

//! Specialization for dynamically in-place, non-interleaved buffers
template<AudioPortsConfig config, AudioDataKind kind>
class DefaultAudioBuffer<config, kind, false, false>
: public AudioBuffer<config>
{
using SampleT = GetAudioDataType<kind>;

public:
DefaultAudioBuffer() = default;
~DefaultAudioBuffer() override = default;

auto inputBuffer() -> SplitAudioData<SampleT, config.inputs> final
{
return {m_accessBuffer.data(), m_channelsIn, m_frames};
}

auto outputBuffer() -> SplitAudioData<SampleT, config.outputs> final
{
return {m_accessBuffer.data() + m_channelsIn, m_channelsOut, m_frames};
}

auto frames() const -> fpp_t final
{
return m_frames;
}

void updateBuffers(proc_ch_t channelsIn, proc_ch_t channelsOut, f_cnt_t frames) final
{
if (channelsIn == DynamicChannelCount || channelsOut == DynamicChannelCount) { return; }

const auto channels = static_cast<std::size_t>(channelsIn + channelsOut);

m_sourceBuffer.resize(channels * frames);
if constexpr (!config.staticChannelCount())
{
m_accessBuffer.resize(channels);
}
else
{
// If channel counts are known at compile time, they should never change
assert(channelsIn == config.inputs);
assert(channelsOut == config.outputs);
}

m_frames = frames;

SampleT* ptr = m_sourceBuffer.data();
for (std::size_t channel = 0; channel < channels; ++channel)
{
m_accessBuffer[channel] = ptr;
ptr += frames;
}

m_channelsIn = channelsIn;
m_channelsOut = channelsOut;
}

private:
//! All input buffers followed by all output buffers
std::vector<SampleT> m_sourceBuffer;

//! Provides [channel][frame] view into `m_sourceBuffer`
AccessBufferType<config> m_accessBuffer;

proc_ch_t m_channelsIn = config.inputs;
proc_ch_t m_channelsOut = config.outputs;
f_cnt_t m_frames = 0;
};


//! Specialization for statically in-place, non-interleaved buffers
template<AudioPortsConfig config, AudioDataKind kind>
class DefaultAudioBuffer<config, kind, false, true>
: public AudioBuffer<config>
{
static_assert(config.inputs == config.outputs || config.inputs == 0 || config.outputs == 0,
"in-place buffers must have same number of input channels and output channels, "
"or one of the channel counts must be fixed at zero");

using SampleT = GetAudioDataType<kind>;

public:
DefaultAudioBuffer() = default;
~DefaultAudioBuffer() override = default;

auto inputOutputBuffer() -> SplitAudioData<SampleT, config.outputs> final
{
return {m_accessBuffer.data(), m_channels, m_frames};
}

auto frames() const -> fpp_t final
{
return m_frames;
}

void updateBuffers(proc_ch_t channelsIn, proc_ch_t channelsOut, f_cnt_t frames) final
{
assert(channelsIn == channelsOut || channelsIn == 0 || channelsOut == 0);
if (channelsIn == DynamicChannelCount || channelsOut == DynamicChannelCount) { return; }

const auto channels = std::max(channelsIn, channelsOut);

m_sourceBuffer.resize(channels * frames);
if constexpr (!config.staticChannelCount())
{
m_accessBuffer.resize(channels);
}
else
{
// If channel counts are known at compile time, they should never change
assert(channelsIn == config.inputs);
assert(channelsOut == config.outputs);
}

m_frames = frames;

SampleT* ptr = m_sourceBuffer.data();
for (proc_ch_t channel = 0; channel < channels; ++channel)
{
m_accessBuffer[channel] = ptr;
ptr += frames;
}

m_channels = channelsOut;
}

private:
//! Input/output buffers
std::vector<SampleT> m_sourceBuffer;

//! Provides [channel][frame] view into `m_sourceBuffer`
AccessBufferType<config> m_accessBuffer;

proc_ch_t m_channels = config.outputs;
f_cnt_t m_frames = 0;
};


//! Specialization for 2-channel SampleFrame buffers
template<AudioPortsConfig config>
class DefaultAudioBuffer<config, AudioDataKind::SampleFrame, true, true>
: public AudioBuffer<config>
{
public:
DefaultAudioBuffer() = default;
~DefaultAudioBuffer() override = default;

auto inputOutputBuffer() -> std::span<SampleFrame> final
{
return m_buffer;
}

auto frames() const -> fpp_t final
{
return m_buffer.size();
}

void updateBuffers(proc_ch_t channelsIn, proc_ch_t channelsOut, f_cnt_t frames) final
{
(void)channelsIn;
(void)channelsOut;
m_buffer.resize(frames);
}

private:
std::vector<SampleFrame> m_buffer;
};

} // namespace detail


//! Interface for accessing input/output audio buffers
template<AudioPortsConfig config>
using AudioBuffer = detail::AudioBuffer<config>;


//! Default implementation of `AudioBuffer`
template<AudioPortsConfig config>
using DefaultAudioBuffer = detail::DefaultAudioBuffer<config>;


} // namespace lmms

#endif // LMMS_AUDIO_BUFFER_H
5 changes: 3 additions & 2 deletions include/AudioBusHandle.h
Original file line number Diff line number Diff line change
@@ -31,6 +31,7 @@
#include <QMutex>

#include "PlayHandle.h"
#include "SampleFrame.h"

namespace lmms
{
@@ -58,7 +59,7 @@ class AudioBusHandle : public ThreadableJob
BoolModel* mutedModel = nullptr);
virtual ~AudioBusHandle();

SampleFrame* buffer() { return m_buffer; }
std::span<SampleFrame> buffer() { return m_buffer; }

// indicate whether JACK & Co should provide output-buffer at ext. port
bool extOutputEnabled() const { return m_extOutputEnabled; }
@@ -85,7 +86,7 @@ class AudioBusHandle : public ThreadableJob
private:
volatile bool m_bufferUsage;

SampleFrame* const m_buffer;
std::span<SampleFrame> m_buffer; //!< owning view

bool m_extOutputEnabled;
mix_ch_t m_nextMixerChannel;
Loading
Loading