mirror of
https://github.com/LMMS/lmms.git
synced 2026-05-13 17:27:28 -04:00
Spectrum analyzer update (#5160)
* advanced config: expose hidden constants to user screen * advanced config: add support for FFT window overlapping * waterfall: display at native resolution on high-DPI screens * waterfall: add cursor and improve label density * FFT: fix normalization so that 0 dBFS matches full-scale sinewave * FFT: decouple data acquisition from processing and display * FFT: separate lock for reallocation (to avoid some needless waiting) * moved ranges and other constants to a separate file * debug: better performance measurements * minor fixes * build the ringbuffer library as part of LMMS core
This commit is contained in:
committed by
Johannes Lorenz
parent
2f0010270e
commit
da73ddd242
@@ -27,7 +27,13 @@
|
||||
|
||||
#include "Analyzer.h"
|
||||
|
||||
#ifdef SA_DEBUG
|
||||
#include <chrono>
|
||||
#include <iostream>
|
||||
#endif
|
||||
|
||||
#include "embed.h"
|
||||
#include "lmms_basics.h"
|
||||
#include "plugin_export.h"
|
||||
|
||||
|
||||
@@ -38,7 +44,7 @@ extern "C" {
|
||||
"Spectrum Analyzer",
|
||||
QT_TRANSLATE_NOOP("pluginBrowser", "A graphical spectrum analyzer."),
|
||||
"Martin Pavelek <he29/dot/HS/at/gmail/dot/com>",
|
||||
0x0100,
|
||||
0x0112,
|
||||
Plugin::Effect,
|
||||
new PluginPixmapLoader("logo"),
|
||||
NULL,
|
||||
@@ -50,17 +56,54 @@ extern "C" {
|
||||
Analyzer::Analyzer(Model *parent, const Plugin::Descriptor::SubPluginFeatures::Key *key) :
|
||||
Effect(&analyzer_plugin_descriptor, parent, key),
|
||||
m_processor(&m_controls),
|
||||
m_controls(this)
|
||||
m_controls(this),
|
||||
m_processorThread(m_processor, m_inputBuffer),
|
||||
// Buffer is sized to cover 4* the current maximum LMMS audio buffer size,
|
||||
// so that it has some reserve space in case data processor is busy.
|
||||
m_inputBuffer(4 * m_maxBufferSize)
|
||||
{
|
||||
m_processorThread.start();
|
||||
}
|
||||
|
||||
|
||||
Analyzer::~Analyzer()
|
||||
{
|
||||
m_processor.terminate();
|
||||
m_inputBuffer.wakeAll();
|
||||
m_processorThread.wait();
|
||||
}
|
||||
|
||||
// Take audio data and pass them to the spectrum processor.
|
||||
// Skip processing if the controls dialog isn't visible, it would only waste CPU cycles.
|
||||
bool Analyzer::processAudioBuffer(sampleFrame *buffer, const fpp_t frame_count)
|
||||
{
|
||||
// Measure time spent in audio thread; both average and peak should be well under 1 ms.
|
||||
#ifdef SA_DEBUG
|
||||
unsigned int audio_time = std::chrono::high_resolution_clock::now().time_since_epoch().count();
|
||||
if (audio_time - m_last_dump_time > 5000000000) // print every 5 seconds
|
||||
{
|
||||
std::cout << "Analyzer audio thread: " << m_sum_execution / m_dump_count << " ms avg / "
|
||||
<< m_max_execution << " ms peak." << std::endl;
|
||||
m_last_dump_time = audio_time;
|
||||
m_sum_execution = m_max_execution = m_dump_count = 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!isEnabled() || !isRunning ()) {return false;}
|
||||
if (m_controls.isViewVisible()) {m_processor.analyse(buffer, frame_count);}
|
||||
|
||||
// Skip processing if the controls dialog isn't visible, it would only waste CPU cycles.
|
||||
if (m_controls.isViewVisible())
|
||||
{
|
||||
// To avoid processing spikes on audio thread, data are stored in
|
||||
// a lockless ringbuffer and processed in a separate thread.
|
||||
m_inputBuffer.write(buffer, frame_count, true);
|
||||
}
|
||||
#ifdef SA_DEBUG
|
||||
audio_time = std::chrono::high_resolution_clock::now().time_since_epoch().count() - audio_time;
|
||||
m_dump_count++;
|
||||
m_sum_execution += audio_time / 1000000.0;
|
||||
if (audio_time / 1000000.0 > m_max_execution) {m_max_execution = audio_time / 1000000.0;}
|
||||
#endif
|
||||
|
||||
return isRunning();
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user