From e5d92ba4009cb42b76f974c98f608bf1e66a987f Mon Sep 17 00:00:00 2001 From: Florent Pollet Date: Mon, 24 Jun 2024 11:19:04 -0400 Subject: [PATCH 01/11] fix: std calc in lfp viewer --- Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp | 23 +++++++++++++++++---- 1 file changed, 19 insertions(+), 4 deletions(-) diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp index 18087a32e..c0895ed68 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp @@ -1554,9 +1554,17 @@ float LfpDisplaySplitter::getMean(int chan) float numPts = 0; float sample = 0.0f; - for (int samp = 0; samp < (lfpDisplay->getWidth() - leftmargin); samp += 10) + + // use 0.1s of sample to compute Mean and Std + float totalPoints = 0.1 * sampleRate; + + // avoid crash if no signal + if ((displayBufferIndex[chan] - totalPoints) <= 0) + return 0; + + for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - sample = *screenBufferMean->getReadPointer(chan, samp); + sample = *displayBuffer->getReadPointer(chan, samp); total += sample; numPts++; } @@ -1573,9 +1581,16 @@ float LfpDisplaySplitter::getStd(int chan) float mean = getMean(chan); float numPts = 1; - for (int samp = 0; samp < (lfpDisplay->getWidth() - leftmargin); samp += 10) + // use 0.1s of sample to compute Mean and Std + float totalPoints = 0.1 * sampleRate; + + // avoid crash if no signal + if ((displayBufferIndex[chan] - totalPoints) <= 0) + return 0; + + for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - std += pow((*screenBufferMean->getReadPointer(chan, samp) - mean),2); + std += pow((*displayBuffer->getReadPointer(chan, samp) - mean),2); numPts++; } From 34822e537c629f5f0b744e999228bf97d1512e3d Mon Sep 17 00:00:00 2001 From: Florent Pollet Date: Wed, 26 Jun 2024 16:13:09 -0400 Subject: [PATCH 02/11] fix: debug display buffer index --- Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp | 19 ++++++++++++++++--- 1 file changed, 16 insertions(+), 3 deletions(-) diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp index c0895ed68..9a1ad9c5b 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp @@ -1584,15 +1584,28 @@ float LfpDisplaySplitter::getStd(int chan) // use 0.1s of sample to compute Mean and Std float totalPoints = 0.1 * sampleRate; + //LOGD("DisplayBufferIndex ", displayBufferIndex[chan], " - ", totalPoints, " points required"); + // avoid crash if no signal - if ((displayBufferIndex[chan] - totalPoints) <= 0) - return 0; + //if ((displayBufferIndex[chan] - totalPoints) <= 0) { + // LOGD("\tReturning ", 0); + // return 0; + //} for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - std += pow((*displayBuffer->getReadPointer(chan, samp) - mean),2); + float value = 0; + + if (samp >= 0) + value = *displayBuffer->getReadPointer(chan, samp); + else + value = *displayBuffer->getReadPointer(chan, displayBuffer->getNumSamples() - samp); + + std += pow((value - mean),2); + //LOGD("\tval ", *displayBuffer->getReadPointer(chan, samp)); numPts++; } + LOGD("\tReturning ", sqrt(std / numPts)); return sqrt(std / numPts); From 1646588da2c46451661ba1194c55796ab45094be Mon Sep 17 00:00:00 2001 From: Florent Pollet Date: Wed, 26 Jun 2024 16:13:58 -0400 Subject: [PATCH 03/11] feat: negative index buffer reading --- Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp | 13 ++----------- 1 file changed, 2 insertions(+), 11 deletions(-) diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp index 9a1ad9c5b..558f0dbec 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp @@ -1584,25 +1584,16 @@ float LfpDisplaySplitter::getStd(int chan) // use 0.1s of sample to compute Mean and Std float totalPoints = 0.1 * sampleRate; - //LOGD("DisplayBufferIndex ", displayBufferIndex[chan], " - ", totalPoints, " points required"); - - // avoid crash if no signal - //if ((displayBufferIndex[chan] - totalPoints) <= 0) { - // LOGD("\tReturning ", 0); - // return 0; - //} - for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { float value = 0; - if (samp >= 0) + if (samp >= 0) // read the beginning of the buffer value = *displayBuffer->getReadPointer(chan, samp); - else + else // read the end of the buffer if negative index value = *displayBuffer->getReadPointer(chan, displayBuffer->getNumSamples() - samp); std += pow((value - mean),2); - //LOGD("\tval ", *displayBuffer->getReadPointer(chan, samp)); numPts++; } LOGD("\tReturning ", sqrt(std / numPts)); From 36365abf03025df5ed054f4d3e2e3711aa895590 Mon Sep 17 00:00:00 2001 From: Florent Pollet Date: Wed, 26 Jun 2024 16:17:39 -0400 Subject: [PATCH 04/11] fix: getMean too displayBuffer --- Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp | 20 +++++++++++--------- 1 file changed, 11 insertions(+), 9 deletions(-) diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp index 558f0dbec..8852ee430 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp @@ -1557,14 +1557,16 @@ float LfpDisplaySplitter::getMean(int chan) // use 0.1s of sample to compute Mean and Std float totalPoints = 0.1 * sampleRate; - - // avoid crash if no signal - if ((displayBufferIndex[chan] - totalPoints) <= 0) - return 0; for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - sample = *displayBuffer->getReadPointer(chan, samp); + float sample = 0; + + if (samp >= 0) // read the beginning of the buffer + sample = *displayBuffer->getReadPointer(chan, samp); + else // read the end of the buffer if negative index + sample = *displayBuffer->getReadPointer(chan, displayBuffer->getNumSamples() - samp); + total += sample; numPts++; } @@ -1586,14 +1588,14 @@ float LfpDisplaySplitter::getStd(int chan) for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - float value = 0; + float sample = 0; if (samp >= 0) // read the beginning of the buffer - value = *displayBuffer->getReadPointer(chan, samp); + sample = *displayBuffer->getReadPointer(chan, samp); else // read the end of the buffer if negative index - value = *displayBuffer->getReadPointer(chan, displayBuffer->getNumSamples() - samp); + sample = *displayBuffer->getReadPointer(chan, displayBuffer->getNumSamples() - samp); - std += pow((value - mean),2); + std += pow((sample - mean),2); numPts++; } LOGD("\tReturning ", sqrt(std / numPts)); From 24e4db02d7d4987e30220af89e94139e2d4c4830 Mon Sep 17 00:00:00 2001 From: Florent Pollet Date: Wed, 26 Jun 2024 16:21:01 -0400 Subject: [PATCH 05/11] chore: cleaning variables --- Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp index 8852ee430..4dc29ba4f 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp @@ -1560,8 +1560,6 @@ float LfpDisplaySplitter::getMean(int chan) for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - float sample = 0; - if (samp >= 0) // read the beginning of the buffer sample = *displayBuffer->getReadPointer(chan, samp); else // read the end of the buffer if negative index @@ -1579,6 +1577,7 @@ float LfpDisplaySplitter::getMean(int chan) float LfpDisplaySplitter::getStd(int chan) { float std = 0.0f; + float sample = 0.0f; float mean = getMean(chan); float numPts = 1; @@ -1588,8 +1587,6 @@ float LfpDisplaySplitter::getStd(int chan) for (int samp = displayBufferIndex[chan] - totalPoints; samp < displayBufferIndex[chan]; samp += 1) { - float sample = 0; - if (samp >= 0) // read the beginning of the buffer sample = *displayBuffer->getReadPointer(chan, samp); else // read the end of the buffer if negative index @@ -1598,7 +1595,6 @@ float LfpDisplaySplitter::getStd(int chan) std += pow((sample - mean),2); numPts++; } - LOGD("\tReturning ", sqrt(std / numPts)); return sqrt(std / numPts); From daf7b81aa881761b2c0d02ade5066e497f5af9a9 Mon Sep 17 00:00:00 2001 From: Florent Pollet Date: Thu, 11 Jul 2024 22:07:57 -0400 Subject: [PATCH 06/11] fix: separate screen and display buffer for substract offset feature lfp viewer --- Plugins/LfpDisplayNode/LfpChannelDisplay.cpp | 4 ++-- .../LfpDisplayNode/LfpChannelDisplayInfo.cpp | 2 +- Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp | 22 +++++++++++++++++-- Plugins/LfpDisplayNode/LfpDisplayCanvas.h | 7 ++++-- 4 files changed, 28 insertions(+), 7 deletions(-) diff --git a/Plugins/LfpDisplayNode/LfpChannelDisplay.cpp b/Plugins/LfpDisplayNode/LfpChannelDisplay.cpp index 532231a73..faf7313f0 100644 --- a/Plugins/LfpDisplayNode/LfpChannelDisplay.cpp +++ b/Plugins/LfpDisplayNode/LfpChannelDisplay.cpp @@ -243,7 +243,7 @@ void LfpChannelDisplay::pxPaint() double a = (canvasSplit->getYCoordMax(chan, index)/range*channelHeightFloat); double b = (canvasSplit->getYCoordMin(chan, index)/range*channelHeightFloat); - double mean = (canvasSplit->getMean(chan)/range*channelHeightFloat); + double mean = (canvasSplit->getScreenBufferMean(chan)/range*channelHeightFloat); if (drawWithOffsetCorrection) { @@ -508,7 +508,7 @@ void LfpChannelDisplay::pxPaintHistory(int playhead, int rightEdge, int maxScree double a = (canvasSplit->getYCoordMax(chan, index) / range * channelHeightFloat); double b = (canvasSplit->getYCoordMin(chan, index) / range * channelHeightFloat); - double mean = (canvasSplit->getMean(chan) / range * channelHeightFloat); + double mean = (canvasSplit->getScreenBufferMean(chan) / range * channelHeightFloat); if (drawWithOffsetCorrection) { diff --git a/Plugins/LfpDisplayNode/LfpChannelDisplayInfo.cpp b/Plugins/LfpDisplayNode/LfpChannelDisplayInfo.cpp index f65a08f92..e91af3586 100644 --- a/Plugins/LfpDisplayNode/LfpChannelDisplayInfo.cpp +++ b/Plugins/LfpDisplayNode/LfpChannelDisplayInfo.cpp @@ -247,7 +247,7 @@ void LfpChannelDisplayInfo::paint(Graphics& g) g.setColour(Colours::grey); g.drawText(String(canvasSplit->getStd(chan)), 5, center+110,41,10,Justification::centred,false); - g.drawText(String(canvasSplit->getMean(chan)), 5, center+60,41,10,Justification::centred,false); + g.drawText(String(canvasSplit->getDisplayBufferMean(chan)), 5, center+60,41,10,Justification::centred,false); if (x > 0) { diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp index 4dc29ba4f..5b5aa6c0b 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.cpp @@ -1548,7 +1548,25 @@ uint16 LfpDisplaySplitter::getChannelStreamId(int channel) return processor->getContinuousChannel(channel)->getStreamId(); } -float LfpDisplaySplitter::getMean(int chan) +float LfpDisplaySplitter::getScreenBufferMean(int chan) +{ + float total = 0.0f; + float numPts = 0; + + float sample = 0.0f; + for (int samp = 0; samp < (lfpDisplay->getWidth() - leftmargin); samp += 10) + { + sample = *screenBufferMean->getReadPointer(chan, samp); + total += sample; + numPts++; + } + + //std::cout << sample << std::endl; + + return total / numPts; +} + +float LfpDisplaySplitter::getDisplayBufferMean(int chan) { float total = 0.0f; float numPts = 0; @@ -1579,7 +1597,7 @@ float LfpDisplaySplitter::getStd(int chan) float std = 0.0f; float sample = 0.0f; - float mean = getMean(chan); + float mean = getDisplayBufferMean(chan); float numPts = 1; // use 0.1s of sample to compute Mean and Std diff --git a/Plugins/LfpDisplayNode/LfpDisplayCanvas.h b/Plugins/LfpDisplayNode/LfpDisplayCanvas.h index 884a81edc..83d3b76e0 100644 --- a/Plugins/LfpDisplayNode/LfpDisplayCanvas.h +++ b/Plugins/LfpDisplayNode/LfpDisplayCanvas.h @@ -263,8 +263,11 @@ class LfpDisplaySplitter : public Component, /** Gets the event channel state for a particular sample */ const float getEventState(int samp); - /** Returns the mean of a given channel */ - float getMean(int chan); + /** Returns the mean of a given channel, computed from the screen buffer */ + float getScreenBufferMean(int chan); + + /** Returns the mean of a given channel, computed from the display buffer */ + float getDisplayBufferMean(int chan); /** Returns the standard deviation of a given channnel*/ float getStd(int chan); From b4138b40e59ab47620164155e34b1e3a49720811 Mon Sep 17 00:00:00 2001 From: Anjal Doshi Date: Fri, 19 Jul 2024 11:21:48 -0700 Subject: [PATCH 07/11] Prevent adding spikes to queue when recording is inactive --- Source/Processors/RecordNode/RecordNode.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Source/Processors/RecordNode/RecordNode.cpp b/Source/Processors/RecordNode/RecordNode.cpp index 0bfcdc0a5..f07aac095 100755 --- a/Source/Processors/RecordNode/RecordNode.cpp +++ b/Source/Processors/RecordNode/RecordNode.cpp @@ -790,7 +790,7 @@ void RecordNode::handleSpike(SpikePtr spike) eventMonitor->receivedSpikes++; - if (recordSpikes) + if (recordSpikes && isRecording) { spike->setTimestampInSeconds(synchronizer.convertSampleNumberToTimestamp(spike->getStreamId(), spike->getSampleNumber())); From 1075b3fc9b2c62c34139015694af6ed80a349a8e Mon Sep 17 00:00:00 2001 From: Anjal Doshi Date: Fri, 19 Jul 2024 11:37:17 -0700 Subject: [PATCH 08/11] Update selected channels parameter value when channel count changes --- Source/Processors/Parameter/Parameter.cpp | 34 +++++++++++++++++++++-- 1 file changed, 31 insertions(+), 3 deletions(-) diff --git a/Source/Processors/Parameter/Parameter.cpp b/Source/Processors/Parameter/Parameter.cpp index b52b26de7..e71219441 100755 --- a/Source/Processors/Parameter/Parameter.cpp +++ b/Source/Processors/Parameter/Parameter.cpp @@ -361,10 +361,24 @@ SelectedChannelsParameter::SelectedChannelsParameter(GenericProcessor* processor void SelectedChannelsParameter::setNextValue(var newValue_) { - if (newValue_.getArray()->size() <= maxSelectableChannels) + if (newValue_ == currentValue) + return; + + int arraySize = newValue_.getArray()->size(); + if (arraySize <= maxSelectableChannels && arraySize <= channelCount) { + for (int i = 0; i < arraySize; i++) + { + if ((int) newValue_[i] < 0 || (int) newValue_[i] >= channelCount) + return; + } + newValue = newValue_; } + else + { + return; + } processor->parameterChangeRequest(this); } @@ -446,9 +460,23 @@ Array SelectedChannelsParameter::parseSelectedString(const String& input) return selectedChannels; } -void SelectedChannelsParameter::setChannelCount(int count) +void SelectedChannelsParameter::setChannelCount(int newCount) { - channelCount = count; + if (channelCount > newCount && getScope() != ParameterScope::SPIKE_CHANNEL_SCOPE) + { + Array values; + for (int i = 0; i < currentValue.getArray()->size(); i++) + { + if ((int) currentValue[i] < newCount) + { + values.add (currentValue[i]); + } + } + + currentValue = values; + } + + channelCount = newCount; // std::cout << "Setting selected channels channels count to " << count << " at " << this << std::endl; } From d63ff0256b24ca87219d7c59f7771d995f125a48 Mon Sep 17 00:00:00 2001 From: Anjal Doshi Date: Fri, 19 Jul 2024 12:19:53 -0700 Subject: [PATCH 09/11] Ensure processor nodes are deleted when removed from ProccesorGraph --- Source/Processors/ProcessorGraph/ProcessorGraph.cpp | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/Source/Processors/ProcessorGraph/ProcessorGraph.cpp b/Source/Processors/ProcessorGraph/ProcessorGraph.cpp index 24abae466..72d5f75fc 100644 --- a/Source/Processors/ProcessorGraph/ProcessorGraph.cpp +++ b/Source/Processors/ProcessorGraph/ProcessorGraph.cpp @@ -664,7 +664,14 @@ void ProcessorGraph::clearSignalChain() std::unique_ptr editor; editor.swap(processor->editor); editor.reset(); + Node::Ptr node = removeNode(nodeId); + + // Decrement reference count for the node + // Ensures processor destructor gets called + while (node->getReferenceCount() > 1) + node->decReferenceCount(); + node.reset(); } @@ -1356,6 +1363,12 @@ void ProcessorGraph::removeProcessor(GenericProcessor* processor) editor.reset(); Node::Ptr node = removeNode(nodeId); + + // Decrement reference count for the node + // Ensures processor destructor gets called + while (node->getReferenceCount() > 1) + node->decReferenceCount(); + node.reset(); } From 36a6f7ee81e036eb6de354125868edc63556040c Mon Sep 17 00:00:00 2001 From: Josh Siegle Date: Thu, 29 Aug 2024 11:54:02 -0700 Subject: [PATCH 10/11] Start audio callbacks after starting acquisition --- Source/UI/ControlPanel.cpp | 31 +++++++++++++++---------------- 1 file changed, 15 insertions(+), 16 deletions(-) diff --git a/Source/UI/ControlPanel.cpp b/Source/UI/ControlPanel.cpp index a08acc542..b536ff8db 100755 --- a/Source/UI/ControlPanel.cpp +++ b/Source/UI/ControlPanel.cpp @@ -528,27 +528,26 @@ void ControlPanel::startAcquisition(bool recordingShouldAlsoStart) graph->updateConnections(); - if (audio->beginCallbacks()) // starts acquisition callbacks + graph->startAcquisition(); // inform processors acquisition is starting + + if (recordingShouldAlsoStart) { - if (recordingShouldAlsoStart) - { - startRecording(); - playButton->setToggleState(true, dontSendNotification); - } + startRecording(); + playButton->setToggleState(true, dontSendNotification); + } - playButton->getNormalImage()->replaceColour(defaultButtonColour, Colours::yellow); + playButton->getNormalImage()->replaceColour(defaultButtonColour, Colours::yellow); - clock->start(); // starts the clock - audioEditor->disable(); + clock->start(); // starts the clock + audioEditor->disable(); - stopTimer(); - startTimer(250); // refresh every 250 ms + stopTimer(); + startTimer(250); // refresh every 250 ms - recordSelector->setEnabled(false); // why is this outside the "if" statement? - recordOptionsButton->setEnabled(false); - - graph->startAcquisition(); // start data flow - } + recordSelector->setEnabled(false); // why is this outside the "if" statement? + recordOptionsButton->setEnabled(false); + + audio->beginCallbacks(); } } From 3756bfaa06ff66477e317105c74560275796af3f Mon Sep 17 00:00:00 2001 From: Anjal Doshi Date: Tue, 24 Sep 2024 18:00:22 -0700 Subject: [PATCH 11/11] Audio Monitor: refactor and remove monitor channel limit --- .../SpikeDisplayNode/SpikeDisplayNode.cpp | 2 +- .../Processors/AudioMonitor/AudioMonitor.cpp | 317 ++++++++---------- Source/Processors/AudioMonitor/AudioMonitor.h | 71 ++-- .../AudioMonitor/AudioMonitorEditor.cpp | 11 +- 4 files changed, 200 insertions(+), 201 deletions(-) diff --git a/Plugins/BasicSpikeDisplay/SpikeDisplayNode/SpikeDisplayNode.cpp b/Plugins/BasicSpikeDisplay/SpikeDisplayNode/SpikeDisplayNode.cpp index e56a1146d..15f39bbdf 100644 --- a/Plugins/BasicSpikeDisplay/SpikeDisplayNode/SpikeDisplayNode.cpp +++ b/Plugins/BasicSpikeDisplay/SpikeDisplayNode/SpikeDisplayNode.cpp @@ -106,7 +106,7 @@ void SpikeDisplayNode::setParameter(int param, float val) for (auto ch : chan->localChannelIndexes) { - msg += String(ch) + " "; + msg += String(ch + 1) + " "; } //std::cout << "MESSAGE: " << msg << std::endl; diff --git a/Source/Processors/AudioMonitor/AudioMonitor.cpp b/Source/Processors/AudioMonitor/AudioMonitor.cpp index 86002e738..8edc1c8fe 100644 --- a/Source/Processors/AudioMonitor/AudioMonitor.cpp +++ b/Source/Processors/AudioMonitor/AudioMonitor.cpp @@ -26,12 +26,92 @@ #include -AudioMonitor::AudioMonitor() - : GenericProcessor ("Audio Monitor"), +AudioMonitorSettings::AudioMonitorSettings() + : numChannels(0), + sampleRate(0), destBufferSampleRate(44100.0f), estimatedSamples(1024) { +} + +void AudioMonitorSettings::createFilters (int nChans, float sampleRate_) +{ + numChannels = nChans; + sampleRate = sampleRate_; + + bandpassfilters.clear(); + antialiasingfilters.clear(); + + Dsp::Params bandpassParams; + bandpassParams[0] = sampleRate; // sample rate + bandpassParams[1] = 2; // order + bandpassParams[2] = (7000 + 100) / 2; // center frequency + bandpassParams[3] = 7000 - 100; // bandwidth + + for (int n = 0; n < numChannels; ++n) + { + bandpassfilters.add(new Dsp::SmoothedFilterDesign + , // order + 1, // number of channels (must be const) + Dsp::DirectFormII>(1)); // realization + + bandpassfilters.getLast()->setParams (bandpassParams); + + antialiasingfilters.add(new Dsp::SmoothedFilterDesign(1024)); + } + + updateAntiAliasingFilterParameters(); + +} + +void AudioMonitorSettings::updateAntiAliasingFilterParameters () +{ + Dsp::Params antialiasingParams; + antialiasingParams[0] = destBufferSampleRate; // sample rate + antialiasingParams[1] = destBufferSampleRate / 2; // cutoff frequency + antialiasingParams[2] = 1.25; //Q // + + for (int i = 0; i < numChannels; ++i) + { + antialiasingfilters[i]->setParams(antialiasingParams); + } +} + +void AudioMonitorSettings::setOutputSampleRate(double outputSampleRate, double estimatedSamplesPerBlock) +{ + destBufferSampleRate = outputSampleRate; + estimatedSamples = estimatedSamplesPerBlock; + + numSamplesExpected = (sampleRate / destBufferSampleRate) * estimatedSamples; + ratio = sampleRate/destBufferSampleRate; + + samplesInBackupBuffer.clear(); + samplesInOverflowBuffer.clear(); + + bufferA.clear(); + bufferB.clear(); + bufferSwap.clear(); + + for (int i = 0; i < numChannels; i++) + { + samplesInBackupBuffer.emplace(i, 0.0f); + samplesInOverflowBuffer.emplace(i, 0.0f); + + bufferA.emplace(i, std::make_unique>(1, (int)destBufferSampleRate)); + bufferB.emplace(i, std::make_unique>(1, (int)destBufferSampleRate)); + bufferSwap.emplace(i, false); + } + + updateAntiAliasingFilterParameters(); +} + + +AudioMonitor::AudioMonitor() + : GenericProcessor ("Audio Monitor") +{ + tempBuffer = std::make_unique(); addBooleanParameter(Parameter::GLOBAL_SCOPE, @@ -47,21 +127,9 @@ AudioMonitor::AudioMonitor() addSelectedChannelsParameter(Parameter::STREAM_SCOPE, String("Channels"), - "Channels to monitor", - 4); - - for (int i = 0; i < MAX_CHANNELS; i++) - { - - bandpassfilters.add (new Dsp::SmoothedFilterDesign - , // order - 1, // number of channels (must be const) - Dsp::DirectFormII> (1)); // realization - - antialiasingfilters.add (new Dsp::SmoothedFilterDesign (1024)); - } + "Channels to monitor"); + tempBuffer->setSize(1, 4096); } @@ -76,21 +144,13 @@ AudioProcessorEditor* AudioMonitor::createEditor() void AudioMonitor::updateSettings() { updatePlaybackBuffer(); - + + settings.update (getDataStreams()); + for (auto stream : dataStreams) { - - Array* activeChannels = stream->getParameter("Channels")->getValue().getArray(); - - if (activeChannels->size() > 0) - { - selectedStream = stream->getStreamId(); - - for (int i = 0; i < activeChannels->size(); i++) - { - updateFilter(i, selectedStream); - } - } + settings[stream->getStreamId()]->createFilters(stream->getChannelCount(), + stream->getSampleRate()); } } @@ -111,132 +171,30 @@ void AudioMonitor::updatePlaybackBuffer() void AudioMonitor::prepareToPlay(double sampleRate_, int estimatedSamplesPerBlock) { - - destBufferSampleRate = sampleRate_; - estimatedSamples = estimatedSamplesPerBlock; - recreateBuffers(); - -} - - -void AudioMonitor::recreateBuffers() -{ - numSamplesExpected.clear(); - sourceBufferSampleRate.clear(); - - ratio.clear(); - - for (int i = 0; i < getNumInputs(); i++) + for (auto stream : dataStreams) { - numSamplesExpected.emplace(i, - continuousChannels[i]->getSampleRate() - / destBufferSampleRate - * estimatedSamples); - - sourceBufferSampleRate.emplace(i, continuousChannels[i]->getSampleRate()); - - ratio.emplace(i, sourceBufferSampleRate[i]/destBufferSampleRate); - + settings[stream->getStreamId()]->setOutputSampleRate(sampleRate_, estimatedSamplesPerBlock); } - - samplesInBackupBuffer.clear(); - samplesInOverflowBuffer.clear(); - - bufferA.clear(); - bufferB.clear(); - bufferSwap.clear(); - - for (int i = 0; i < MAX_CHANNELS; i++) - { - - samplesInBackupBuffer.emplace(i, 0.0f); - samplesInOverflowBuffer.emplace(i, 0.0f); - - bufferA.emplace(i, std::make_unique>(1,44100)); - bufferB.emplace(i, std::make_unique>(1,44100)); - bufferSwap.emplace(i, false); - - } - - tempBuffer->setSize(1, 4096); } void AudioMonitor::parameterValueChanged(Parameter* param) { - LOGD("Audio Monitor: Value changed for ", param->getName(), ": ", (int)param->getValue()); + LOGD("Audio Monitor: Value changed for ", param->getName()); if (param->getName().equalsIgnoreCase("Channels")) { - - selectedStream = param->getStreamId(); - Array* activeChannels = param->getValue().getArray(); - - if (activeChannels->size() == 0) - LOGA("No channels selected."); LOGD("Num selected channels: ", activeChannels->size()); - - for (int i = 0; i < activeChannels->size(); i++) - { - - int localIndex =(int) activeChannels->getReference(i); - - LOGA("Selected channel ", localIndex); - - auto continuousChannels = getDataStream(selectedStream)->getContinuousChannels(); - - if (continuousChannels.size() > localIndex) - { - int globalIndex = continuousChannels[localIndex]->getGlobalIndex(); - - updateFilter(i, selectedStream); - } - - } - - // clear monitored channels on all other streams - //for (auto stream : dataStreams) - //{ - // if (stream->getStreamId() != selectedStream) - // { - // stream->getParameter("Channels")->currentValue = Array(); - // } - //} } } - -void AudioMonitor::updateFilter(int i, uint16 streamId) -{ - - Dsp::Params params1; - params1[0] = getDataStream(streamId)->getSampleRate(); // sample rate - params1[1] = 2; // order - params1[2] = (7000 + 100) / 2; // center frequency - params1[3] = 7000 - 100; // bandwidth - - bandpassfilters[i]->setParams (params1); - - double cutoffFreq = destBufferSampleRate / 2; // upsample - - double sampleFreq = destBufferSampleRate; // upsample - - Dsp::Params params2; - params2[0] = sampleFreq; // sample rate - params2[1] = cutoffFreq; // cutoff frequency - params2[2] = 1.25; //Q // - - antialiasingfilters[i]->setParams(params2); - -} - void AudioMonitor::handleBroadcastMessage(String msg) { - LOGD("Audio Monitor received message: ", msg); + LOGC("Audio Monitor received message: ", msg); StringArray parts = StringArray::fromTokens(msg, " ", ""); @@ -248,23 +206,45 @@ void AudioMonitor::handleBroadcastMessage(String msg) if (command.equalsIgnoreCase("SELECT")) { - if (parts.size() >= 4) + if(parts.size() == 3) + { + String command = parts[2]; + + if (command.equalsIgnoreCase ("NONE")) + { + Array ch; + + for (auto stream : getDataStreams()) + stream->getParameter ("channels")->setNextValue (ch); + + } + } + else if (parts.size() >= 4) { uint16 streamId = parts[2].getIntValue(); - - DataStream* stream = getDataStream(streamId); - + + DataStream* stream = getDataStream (streamId); + if (stream != nullptr) { - - int localChannel = parts[3].getIntValue() - 1; - - if (localChannel >= 0 && localChannel < stream->getContinuousChannels().size()) + Array ch; + int channelCount = 0; + + while (channelCount < (parts.size() - 3)) { - Array ch; - ch.add(localChannel); - - stream->getParameter("Channels")->setNextValue(ch); + int localChannel = parts[channelCount + 3].getIntValue() - 1; + + if (localChannel >= 0 && localChannel < stream->getContinuousChannels().size()) + { + ch.add (localChannel); + } + + channelCount++; + } + + if (ch.size() > 0) + { + stream->getParameter ("Channels")->setNextValue (ch); } } } @@ -293,7 +273,7 @@ void AudioMonitor::process (AudioBuffer& buffer) if (stream->getStreamId() == selectedStream && (*stream)["enable_stream"]) { - + auto streamSettings = settings[selectedStream]; AudioSampleBuffer* overflowBuffer; AudioSampleBuffer* backupBuffer; @@ -308,34 +288,34 @@ void AudioMonitor::process (AudioBuffer& buffer) tempBuffer->clear(); - if (!bufferSwap[i]) + if (!streamSettings->bufferSwap[localIndex]) { - overflowBuffer = bufferA[i].get(); - backupBuffer = bufferB[i].get(); + overflowBuffer = streamSettings->bufferA[localIndex].get(); + backupBuffer = streamSettings->bufferB[localIndex].get(); - bufferSwap[i] = true; + streamSettings->bufferSwap[localIndex] = true; } else { - overflowBuffer = bufferB[i].get(); - backupBuffer = bufferA[i].get(); + overflowBuffer = streamSettings->bufferB[localIndex].get(); + backupBuffer = streamSettings->bufferA[localIndex].get(); - bufferSwap[i] = false; + streamSettings->bufferSwap[localIndex] = false; } backupBuffer->clear(); - samplesInOverflowBuffer[i] = samplesInBackupBuffer[i]; // size of buffer after last round - samplesInBackupBuffer[i] = 0; + streamSettings->samplesInOverflowBuffer[localIndex] = streamSettings->samplesInBackupBuffer[localIndex]; // size of buffer after last round + streamSettings->samplesInBackupBuffer[localIndex] = 0; double orphanedSamples = 0; // 1. copy overflow buffer double samplesToCopyFromOverflowBuffer = - ((samplesInOverflowBuffer[i] <= numSamplesExpected[globalIndex]) ? - samplesInOverflowBuffer[i] : - numSamplesExpected[globalIndex]); + ((streamSettings->samplesInOverflowBuffer[localIndex] <= streamSettings->numSamplesExpected) ? + streamSettings->samplesInOverflowBuffer[localIndex] : + streamSettings->numSamplesExpected); // LOGD("Number of samples to copy: ", samplesToCopyFromOverflowBuffer); @@ -353,7 +333,7 @@ void AudioMonitor::process (AudioBuffer& buffer) 1.0f // gain to apply ); - double leftoverSamples = samplesInOverflowBuffer[i] - samplesToCopyFromOverflowBuffer; + double leftoverSamples = streamSettings->samplesInOverflowBuffer[localIndex] - samplesToCopyFromOverflowBuffer; //std::cout << "Copying to backup buffer: " << leftoverSamples << std::endl; @@ -370,10 +350,10 @@ void AudioMonitor::process (AudioBuffer& buffer) ); } - samplesInBackupBuffer[i] = leftoverSamples; + streamSettings->samplesInBackupBuffer[localIndex] = leftoverSamples; } - double remainingSamples = double(numSamplesExpected[globalIndex]) - samplesToCopyFromOverflowBuffer; + double remainingSamples = double(streamSettings->numSamplesExpected) - samplesToCopyFromOverflowBuffer; double samplesAvailable = double(getNumSamplesInBlock(selectedStream)); @@ -404,11 +384,12 @@ void AudioMonitor::process (AudioBuffer& buffer) //std::cout << "Orphaned samples: " << orphanedSamples << std::endl; - if (orphanedSamples > 0 && (samplesInBackupBuffer[i] + orphanedSamples < backupBuffer->getNumSamples())) + if (orphanedSamples > 0 && + (streamSettings->samplesInBackupBuffer[localIndex] + orphanedSamples < backupBuffer->getNumSamples())) { backupBuffer->addFrom(0, // destination channel - samplesInBackupBuffer[i], // destination start sample + streamSettings->samplesInBackupBuffer[localIndex], // destination start sample buffer, // source globalIndex, // source channel (int) remainingSamples, // source start sample @@ -416,7 +397,7 @@ void AudioMonitor::process (AudioBuffer& buffer) 1.0f // gain to apply ); - samplesInBackupBuffer[i] = samplesInBackupBuffer[i] + orphanedSamples; + streamSettings->samplesInBackupBuffer[localIndex] += orphanedSamples; } @@ -431,7 +412,7 @@ void AudioMonitor::process (AudioBuffer& buffer) if (totalCopied == 0) continue; - bandpassfilters[i]->process(totalCopied, &ptr); + streamSettings->bandpassfilters[localIndex]->process(totalCopied, &ptr); /*if (i == 0) { @@ -484,7 +465,7 @@ void AudioMonitor::process (AudioBuffer& buffer) 1, // number of samples alpha); // gain to apply to source - subSampleOffset += ratio[globalIndex]; + subSampleOffset += streamSettings->ratio; while (subSampleOffset >= 1.0) { diff --git a/Source/Processors/AudioMonitor/AudioMonitor.h b/Source/Processors/AudioMonitor/AudioMonitor.h index bf3c2d1e3..38f217cc1 100644 --- a/Source/Processors/AudioMonitor/AudioMonitor.h +++ b/Source/Processors/AudioMonitor/AudioMonitor.h @@ -31,7 +31,47 @@ #include "../GenericProcessor/GenericProcessor.h" #include "../Dsp/Dsp.h" -#define MAX_CHANNELS 4 +/** + Creates and holds the filter settings for one input stream in the AudioMonitor. +*/ +class AudioMonitorSettings +{ +public: + + AudioMonitorSettings(); + + int numChannels; + float sampleRate; + double destBufferSampleRate; + double estimatedSamples; + + /** Bandpass filters (1 per channel)*/ + OwnedArray bandpassfilters; + + /** Antialiasing filters (1 per channel)*/ + OwnedArray antialiasingfilters; + + std::map>> bufferA; + std::map>> bufferB; + + /** Per-channel buffer state information*/ + std::map samplesInBackupBuffer; + std::map samplesInOverflowBuffer; + std::map bufferSwap; + double numSamplesExpected; + double ratio; + + /** Creates new filters when input settings change*/ + void createFilters(int numChannels, float sampleRate); + + /** Updates filters when parameters change*/ + void updateAntiAliasingFilterParameters(); + + /** Re-sets the copy buffers prior to starting acquisition*/ + void setOutputSampleRate(double outputSampleRate, double estimatedSamplesPerBlock); + +}; + /** Reads data from a file. @@ -68,39 +108,16 @@ class AudioMonitor : public GenericProcessor /** Resets the connections prior to a new round of data acquisition. */ void resetConnections() override; - - /** Updates the bandpass filter parameters, given the currently monitored stream*/ - void updateFilter(int i, uint16 streamId); /** Allows other processors to configure the Audio Monitor during acquisition*/ void handleBroadcastMessage(String message) override; -private: - - /** Re-sets the copy buffers prior to acquisition*/ - void recreateBuffers(); - - std::map>> bufferA; - std::map>> bufferB; + void setSelectedStream(uint16 streamId) { selectedStream = streamId; } - /** Per-channel buffer state information*/ - std::map samplesInBackupBuffer; - std::map samplesInOverflowBuffer; - std::map sourceBufferSampleRate; - std::map bufferSwap; - - std::map numSamplesExpected; - std::map ratio; - - double destBufferSampleRate; - double estimatedSamples; +private: - /** 4 bandpass filters (1 per selected channel)*/ - OwnedArray bandpassfilters; + StreamSettings settings; - /** 4 antialiasing filters (1 per selected channel)*/ - OwnedArray antialiasingfilters; - /** Holds the data for one channel, before it's copied to the output*/ std::unique_ptr> tempBuffer; diff --git a/Source/Processors/AudioMonitor/AudioMonitorEditor.cpp b/Source/Processors/AudioMonitor/AudioMonitorEditor.cpp index da8e8bf9c..9df78a60a 100644 --- a/Source/Processors/AudioMonitor/AudioMonitorEditor.cpp +++ b/Source/Processors/AudioMonitor/AudioMonitorEditor.cpp @@ -176,16 +176,16 @@ AudioMonitorEditor::AudioMonitorEditor (GenericProcessor* parentNode) { audioMonitor = static_cast(parentNode); - addSelectedChannelsParameterEditor("Channels", 10, 35); + addSelectedChannelsParameterEditor("Channels", 15, 35); Parameter* muteParam = parentNode->getParameter("mute_audio"); - addCustomParameterEditor(new MonitorMuteButton(muteParam), 135, 35); + addCustomParameterEditor(new MonitorMuteButton(muteParam), 130, 35); Parameter* outputParam = parentNode->getParameter("audio_output"); - addCustomParameterEditor(new AudioOutputSelector(outputParam), 20, 65); + addCustomParameterEditor(new AudioOutputSelector(outputParam), 15, 65); spikeChannelSelector = std::make_unique("Spike Channels"); - spikeChannelSelector->setBounds(20, 100, 140, 20); + spikeChannelSelector->setBounds(15, 100, 140, 20); for (int i = 0; i < audioMonitor->getTotalSpikeChannels() ; i++) { @@ -198,11 +198,12 @@ AudioMonitorEditor::AudioMonitorEditor (GenericProcessor* parentNode) spikeChannelSelector->addListener(this); addAndMakeVisible(spikeChannelSelector.get()); - desiredWidth = 180; + desiredWidth = 170; } void AudioMonitorEditor::selectedStreamHasChanged() { + audioMonitor->setSelectedStream(selectedStream); if (selectedStream == 0) {