Mercurial > hg > beaglert
diff projects/measure_noisefloor/render.cpp @ 91:241d4d5df929
Double buffered the noise floor project; works without glitches now
author | andrewm |
---|---|
date | Sun, 19 Jul 2015 23:19:27 +0100 |
parents | d41631e0fe0e |
children |
line wrap: on
line diff
--- a/projects/measure_noisefloor/render.cpp Sun Jul 19 16:42:51 2015 +0100 +++ b/projects/measure_noisefloor/render.cpp Sun Jul 19 23:19:27 2015 +0100 @@ -12,19 +12,16 @@ int gBufferSize = 8192; -// Buffers to hold samples for noise analysis -float* gBuffers[10]; -int gBufferPointers[10]; +// Double buffers to hold samples for noise analysis +float *gReadBuffers[10], *gWriteBuffers[10]; +float *gBuffers0[10], *gBuffers1[10]; -// Outputs to display -float gDCLevels[10]; -float gNoiseLevels[10]; -float gNumSamplesAnalysed[10]; +int gWriteBufferPointers[10], gReadBufferPointers[10]; -// Task to print results which would otherwise be too slow for render() -AuxiliaryTask gPrintTask; +// Task to analyse and print results which would otherwise be too slow for render() +AuxiliaryTask gAnalysisTask; -void printResults(); +void analyseResults(); // setup() is called once before the audio rendering starts. // Use it to perform any initialisation and allocation which is dependent @@ -39,15 +36,18 @@ { // Clear the filter data structures for(int i = 0; i < 10; i++) { - gBufferPointers[i] = 0; - gBuffers[i] = new float[gBufferSize]; - if(gBuffers[i] == 0) { + gReadBufferPointers[i] = gWriteBufferPointers[i] = 0; + gBuffers0[i] = new float[gBufferSize]; + gBuffers1[i] = new float[gBufferSize]; + gWriteBuffers[i] = gBuffers0[i]; + gReadBuffers[i] = gBuffers1[i]; + if(gBuffers0[i] == 0 || gBuffers1[i] == 0) { rt_printf("Error allocating buffer %d\n", i); return false; } } - gPrintTask = BeagleRT_createAuxiliaryTask(printResults, 50, "beaglert-print-results"); + gAnalysisTask = BeagleRT_createAuxiliaryTask(analyseResults, 50, "beaglert-analyse-results"); return true; } @@ -64,11 +64,11 @@ for(unsigned int n = 0; n < context->audioFrames; n++) { // Store audio inputs in buffer for(unsigned int ch = 0; ch < context->audioChannels; ch++) { - if(gBufferPointers[ch] < gBufferSize) { - gBuffers[ch][gBufferPointers[ch]] = + if(gWriteBufferPointers[ch] < gBufferSize) { + gWriteBuffers[ch][gWriteBufferPointers[ch]] = context->audioIn[n * context->audioChannels + ch]; - gBufferPointers[ch]++; - if(gBufferPointers[ch] >= gBufferSize) + gWriteBufferPointers[ch]++; + if(gWriteBufferPointers[ch] >= gBufferSize) bufferIsFull = true; } } @@ -78,58 +78,64 @@ for(unsigned int n = 0; n < context->analogFrames; n++) { // Store analog inputs in buffer, starting at channel 2 for(unsigned int ch = 0; ch < context->analogChannels; ch++) { - if(gBufferPointers[ch + 2] < gBufferSize) { - gBuffers[ch + 2][gBufferPointers[ch + 2]] = + if(gWriteBufferPointers[ch + 2] < gBufferSize) { + gWriteBuffers[ch + 2][gWriteBufferPointers[ch + 2]] = context->analogIn[n * context->analogChannels + ch]; - gBufferPointers[ch + 2]++; - if(gBufferPointers[ch + 2] >= gBufferSize) + gWriteBufferPointers[ch + 2]++; + if(gWriteBufferPointers[ch + 2] >= gBufferSize) bufferIsFull = true; } + + // Set all analog outputs to halfway point so they can be more + // easily measured for noise + context->analogOut[n * context->analogChannels + ch] = 0.5; } } } + if(bufferIsFull) { - // Analyse all active channels at once + // Swap buffers and reset write pointers for(int ch = 0; ch < 10; ch++) { - // gBufferPointers[ch] tells us how many samples were stored in the buffer - gNumSamplesAnalysed[ch] = gBufferPointers[ch]; + gReadBufferPointers[ch] = gWriteBufferPointers[ch]; + gWriteBufferPointers[ch] = 0; - if(gBufferPointers[ch] != 0) { - float mean = 0; - for(int n = 0; n < gBufferPointers[ch]; n++) { - mean += gBuffers[ch][n]; - } - mean /= (float)gBufferPointers[ch]; - - float rms = 0; - for(int n = 0; n < gBufferPointers[ch]; n++) { - rms += (gBuffers[ch][n] - mean) * (gBuffers[ch][n] - mean); - } - rms = sqrtf(rms / (float)gBufferPointers[ch]); - - gDCLevels[ch] = mean; - gNoiseLevels[ch] = rms; + if(gReadBuffers[ch] == gBuffers0[ch]) { + gReadBuffers[ch] = gBuffers1[ch]; + gWriteBuffers[ch] = gBuffers0[ch]; } - - // Reset pointer to 0 for next time - gBufferPointers[ch] = 0; + else { + gReadBuffers[ch] = gBuffers0[ch]; + gWriteBuffers[ch] = gBuffers1[ch]; + } } - BeagleRT_scheduleAuxiliaryTask(gPrintTask); + BeagleRT_scheduleAuxiliaryTask(gAnalysisTask); } } -void printResults() +void analyseResults() { rt_printf("\e[1;1H\e[2J"); // Command to clear the screen - + // Print the analysis results. channels 0-1 are audio, channels 2-9 are analog for(int ch = 0; ch < 10; ch++) { - int samples = gNumSamplesAnalysed[ch]; - if(samples == 0) + // Skip unused channels + if(gReadBufferPointers[ch] == 0) continue; + float mean = 0; + for(int n = 0; n < gReadBufferPointers[ch]; n++) { + mean += gReadBuffers[ch][n]; + } + mean /= (float)gReadBufferPointers[ch]; + + float rms = 0; + for(int n = 0; n < gReadBufferPointers[ch]; n++) { + rms += (gReadBuffers[ch][n] - mean) * (gReadBuffers[ch][n] - mean); + } + rms = sqrtf(rms / (float)gReadBufferPointers[ch]); + if(ch == 0) rt_printf("Audio In L: "); else if(ch == 1) @@ -138,10 +144,10 @@ rt_printf("Analog In %d: ", ch - 2); rt_printf("Noise %6.1fdB DC offset %6.4f (%6.1fdB) window size: %d\n", - 20.0f * log10f(gNoiseLevels[ch]), - gDCLevels[ch], - 20.0f * log10f(fabsf(gDCLevels[ch])), - samples); + 20.0f * log10f(rms), + mean, + 20.0f * log10f(fabsf(mean)), + gReadBufferPointers[ch]); } } @@ -150,6 +156,8 @@ void cleanup(BeagleRTContext *context, void *userData) { - for(int i = 0; i < 10; i++) - delete gBuffers[i]; + for(int i = 0; i < 10; i++) { + delete gBuffers0[i]; + delete gBuffers1[i]; + } }