Skip to content

Commit

Permalink
Add local time sync ref (for AbletonLink etc.) selection to SoundManager
Browse files Browse the repository at this point in the history
Moved pNewMainClockRef selection out of main loop and added proper handling of setups where only headphones or booth are configured.
Simplified fallback mechanisms and removed redundant code.
Added unit tests for new function selectLocalTimeSyncRef
  • Loading branch information
JoergAtGithub committed Jan 12, 2025
1 parent 1803e02 commit ee7b067
Show file tree
Hide file tree
Showing 4 changed files with 272 additions and 45 deletions.
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -2484,6 +2484,7 @@ add_executable(
src/test/signalpathtest.cpp
src/test/skincontext_test.cpp
src/test/softtakeover_test.cpp
src/test/soundmanager_test.cpp
src/test/soundproxy_test.cpp
src/test/soundsourceproviderregistrytest.cpp
src/test/sqliteliketest.cpp
Expand Down
161 changes: 116 additions & 45 deletions src/soundio/soundmanager.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -321,6 +321,40 @@ void SoundManager::queryDevicesMixxx() {
m_devices.append(currentDevice);
}

SoundDevicePointer SoundManager::selectLocalTimeSyncRef(
const QHash<SoundDevicePointer, QList<AudioOutput>>& deviceOutputs,
const QList<SoundDevicePointer>& devices) {
const std::array<AudioPathType, 5> priorityOrder = {
AudioPathType::Main,
AudioPathType::Deck,
AudioPathType::Bus,
AudioPathType::Headphones,
AudioPathType::Booth};

SoundDevicePointer pNewLocalTimeSyncRef = nullptr;
for (const auto& pDevice : std::as_const(devices)) {
if (pDevice->getDeviceId().name == kNetworkDeviceInternalName) {
continue;
}
QList<AudioOutput> outputs = deviceOutputs[pDevice];
for (const auto& type : priorityOrder) {
auto it = std::find_if(outputs.begin(),
outputs.end(),
[type](const AudioOutput& out) {
return out.getType() == type;
});
if (it != outputs.end()) {
pNewLocalTimeSyncRef = pDevice;
break;
}
}
if (pNewLocalTimeSyncRef) {
break;
}
}
return pNewLocalTimeSyncRef;
}

SoundDeviceStatus SoundManager::setupDevices() {
// NOTE(rryan): Big warning: This function is concurrent with calls to
// pushBuffer and onDeviceOutputCallback until closeDevices() below.
Expand All @@ -333,7 +367,7 @@ SoundDeviceStatus SoundManager::setupDevices() {
// it was. Clearing it causes the engine to stop being processed which
// results in a stuttering noise (sometimes a loud buzz noise at low
// latencies) when changing devices.
//m_pClkRefDevice = NULL;
// m_pClkRefDevice = NULL;
m_pErrorDevice.clear();
int outputDevicesOpened = 0;
int inputDevicesOpened = 0;
Expand All @@ -351,10 +385,6 @@ SoundDeviceStatus SoundManager::setupDevices() {
// callback from the logic in SoundDevicePortAudio. They should communicate
// via message passing over a request/response FIFO.

// Instead of clearing m_pClkRefDevice and then assigning it directly,
// compute the new one then atomically hand off below.
SoundDevicePointer pNewMainClockRef;

m_audioLatencyOverloadCount.set(0);

// load with all configured devices.
Expand All @@ -364,6 +394,77 @@ SoundDeviceStatus SoundManager::setupDevices() {
// pair is isInput, isOutput
QVector<DeviceMode> toOpen;
bool haveOutput = false;

Check failure on line 396 in src/soundio/soundmanager.cpp

View workflow job for this annotation

GitHub Actions / Ubuntu 24.04

variable ‘haveOutput’ set but not used [-Werror=unused-but-set-variable]

Check failure on line 396 in src/soundio/soundmanager.cpp

View workflow job for this annotation

GitHub Actions / clazy

variable 'haveOutput' set but not used [-Werror,-Wunused-but-set-variable]

Check failure on line 396 in src/soundio/soundmanager.cpp

View workflow job for this annotation

GitHub Actions / macOS 13 x64

variable 'haveOutput' set but not used [-Werror,-Wunused-but-set-variable]

Check warning on line 396 in src/soundio/soundmanager.cpp

View workflow job for this annotation

GitHub Actions / coverage

variable ‘haveOutput’ set but not used [-Wunused-but-set-variable]

Check failure on line 396 in src/soundio/soundmanager.cpp

View workflow job for this annotation

GitHub Actions / macOS 13 arm64

variable 'haveOutput' set but not used [-Werror,-Wunused-but-set-variable]

// Get all outputs for each device
QHash<SoundDevicePointer, QList<AudioOutput>> deviceOutputs;
for (const auto& pDevice : std::as_const(m_devices)) {
deviceOutputs[pDevice] = m_config.getOutputs().values(pDevice->getDeviceId());
// Statically connect the Network Device to the Sidechain
if (pDevice->getDeviceId().name == kNetworkDeviceInternalName) {
AudioOutput out(AudioPathType::RecordBroadcast,
0,
mixxx::audio::ChannelCount::stereo(),
0);
deviceOutputs[pDevice].append(out);
}
}

// Select pNewLocalTimeSyncRef
// The local time sync reference is the device that is used for the
// syncronization of processes outside the audio proccessing engine
// to the DAC timing of the local sounddevice the DJ hears.
// This sync reference shall be used for:
// 1.) VSync of the waveforms
// 2.) Sync of external audio device (e.g. drum machine) with feedback
// to an auxilary input of the mixer in a external mixing setup
// 3.) Sync of lighting (DMX) or video (VJ)
SoundDevicePointer pNewLocalTimeSyncRef = selectLocalTimeSyncRef(deviceOutputs, m_devices);

// Select pNewMainClockRef
// The main clock reference is the device that is used for the
// audio processing in the engine. It can be either a local
// PortAudio device or the Network clock in case of broadcasting.
// There are four use cases:
// 1.) No broadcasting->Always Soundcard Clock
// 2.) JACK API used->Always Soundcard Clock
// 3.) Broadcasting of internal mixed Main signal->Always Network Clock
// 4.) Broadcasting of Record/Broadcast input SoundDevice->Always Soundcard Clock
SoundDevicePointer pNewMainClockRef;
if (m_config.getForceNetworkClock() && !jackApiUsed()) {
for (const auto& pDevice : std::as_const(m_devices)) {
if (pDevice->getDeviceId().name == kNetworkDeviceInternalName) {
pNewMainClockRef = pDevice;
break;
}
}
} else {
pNewMainClockRef = pNewLocalTimeSyncRef;
}

// Fallback to keep waveforms running if no local SoundDevice is configured
// If pNewLocalTimeSyncRef or pNewMainClockRef is still nullptr,
// set it to the first network clock device.
if (!pNewLocalTimeSyncRef || !pNewMainClockRef) {
for (const auto& pDevice : std::as_const(m_devices)) {
if (pDevice->getDeviceId().name == kNetworkDeviceInternalName) {
if (!pNewLocalTimeSyncRef) {
qWarning() << "No local sound device configured, local "
"sync reference not set! Using"
<< pDevice->getDisplayName();
pNewLocalTimeSyncRef = pDevice;
}
if (!pNewMainClockRef) {
qWarning() << "Output sound device clock reference not set! Using"
<< pDevice->getDisplayName();
pNewMainClockRef = pDevice;
}
if (pNewLocalTimeSyncRef && pNewMainClockRef) {
break;
}
}
}
}

// loop over all available devices
for (const auto& pDevice : std::as_const(m_devices)) {
DeviceMode mode = {pDevice, false, false};
Expand Down Expand Up @@ -393,22 +494,9 @@ SoundDeviceStatus SoundManager::setupDevices() {
m_pEngineMixer->onInputConnected(in);
}
}
QList<AudioOutput> outputs =
m_config.getOutputs().values(pDevice->getDeviceId());

// Statically connect the Network Device to the Sidechain
if (pDevice->getDeviceId().name == kNetworkDeviceInternalName) {
AudioOutput out(AudioPathType::RecordBroadcast,
0,
mixxx::audio::ChannelCount::stereo(),
0);
outputs.append(out);
if (m_config.getForceNetworkClock() && !jackApiUsed()) {
pNewMainClockRef = pDevice;
}
}

for (const auto& out : std::as_const(outputs)) {
// Iterate over all outputs for the current device
for (const auto& out : std::as_const(deviceOutputs[pDevice])) {
mode.isOutput = true;
if (pDevice->getDeviceId().name != kNetworkDeviceInternalName) {
haveOutput = true;
Expand All @@ -427,16 +515,6 @@ SoundDeviceStatus SoundManager::setupDevices() {
goto closeAndError;
}

if (!m_config.getForceNetworkClock() || jackApiUsed()) {
if (out.getType() == AudioPathType::Main) {
pNewMainClockRef = pDevice;
} else if ((out.getType() == AudioPathType::Deck ||
out.getType() == AudioPathType::Bus) &&
!pNewMainClockRef) {
pNewMainClockRef = pDevice;
}
}

// Check if any AudioSource is registered for this AudioOutput and
// call the onOutputConnected method.
for (auto it = m_registeredSources.find(out);
Expand All @@ -453,19 +531,10 @@ SoundDeviceStatus SoundManager::setupDevices() {
}
}

for (const auto& mode: toOpen) {
for (const auto& mode : toOpen) {
SoundDevicePointer pDevice = mode.pDevice;
m_pErrorDevice = pDevice;

// If we have not yet set a clock source then we use the first
// output pDevice
if (pNewMainClockRef.isNull() &&
(!haveOutput || mode.isOutput)) {
pNewMainClockRef = pDevice;
qWarning() << "Output sound device clock reference not set! Using"
<< pDevice->getDisplayName();
}

int syncBuffers = m_config.getSyncBuffers();
// If we are in safe mode and using experimental polling support, use
// the default of 2 sync buffers instead.
Expand All @@ -485,11 +554,14 @@ SoundDeviceStatus SoundManager::setupDevices() {
}
}

if (pNewMainClockRef) {
VERIFY_OR_DEBUG_ASSERT(pNewMainClockRef) {
// This should never happen, because the user can't delete the last
// broadcast device in the preferences.
qWarning() << "No output devices opened, no clock reference device set";
}
else {
qDebug() << "Using" << pNewMainClockRef->getDisplayName()
<< "as output sound device clock reference";
} else {
qWarning() << "No output devices opened, no clock reference device set";
}

qDebug() << outputDevicesOpened << "output sound devices opened";
Expand All @@ -499,8 +571,7 @@ SoundDeviceStatus SoundManager::setupDevices() {
}

m_pControlObjectSoundStatusCO->set(
outputDevicesOpened > 0 ?
SOUNDMANAGER_CONNECTED : SOUNDMANAGER_DISCONNECTED);
outputDevicesOpened > 0 ? SOUNDMANAGER_CONNECTED : SOUNDMANAGER_DISCONNECTED);

// returns OK if we were able to open all the devices the user wanted
if (devicesNotFound.isEmpty()) {
Expand Down
4 changes: 4 additions & 0 deletions src/soundio/soundmanager.h
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,10 @@ class SoundManager : public QObject {
void queryDevicesPortaudio();
void queryDevicesMixxx();

static SoundDevicePointer selectLocalTimeSyncRef(
const QHash<SoundDevicePointer, QList<AudioOutput>>& deviceOutputs,
const QList<SoundDevicePointer>& devices);

// Opens all the devices chosen by the user in the preferences dialog, and
// establishes the proper connections between them and the mixing engine.
SoundDeviceStatus setupDevices();
Expand Down
Loading

0 comments on commit ee7b067

Please sign in to comment.