Mercurial > hg > beaglert
comparison projects/measure_noisefloor/render.cpp @ 108:3068421c0737 ultra-staging
Merged default into ultra-staging
author | Giulio Moro <giuliomoro@yahoo.it> |
---|---|
date | Tue, 18 Aug 2015 00:35:15 +0100 |
parents | 241d4d5df929 |
children |
comparison
equal
deleted
inserted
replaced
54:d3f869b98147 | 108:3068421c0737 |
---|---|
1 /* | |
2 * render.cpp | |
3 * | |
4 * Created on: Oct 24, 2014 | |
5 * Author: parallels | |
6 */ | |
7 | |
8 | |
9 #include <BeagleRT.h> | |
10 #include <Utilities.h> | |
11 #include <cmath> | |
12 | |
13 int gBufferSize = 8192; | |
14 | |
15 // Double buffers to hold samples for noise analysis | |
16 float *gReadBuffers[10], *gWriteBuffers[10]; | |
17 float *gBuffers0[10], *gBuffers1[10]; | |
18 | |
19 int gWriteBufferPointers[10], gReadBufferPointers[10]; | |
20 | |
21 // Task to analyse and print results which would otherwise be too slow for render() | |
22 AuxiliaryTask gAnalysisTask; | |
23 | |
24 void analyseResults(); | |
25 | |
26 // setup() is called once before the audio rendering starts. | |
27 // Use it to perform any initialisation and allocation which is dependent | |
28 // on the period size or sample rate. | |
29 // | |
30 // userData holds an opaque pointer to a data structure that was passed | |
31 // in from the call to initAudio(). | |
32 // | |
33 // Return true on success; returning false halts the program. | |
34 | |
35 bool setup(BeagleRTContext *context, void *userData) | |
36 { | |
37 // Clear the filter data structures | |
38 for(int i = 0; i < 10; i++) { | |
39 gReadBufferPointers[i] = gWriteBufferPointers[i] = 0; | |
40 gBuffers0[i] = new float[gBufferSize]; | |
41 gBuffers1[i] = new float[gBufferSize]; | |
42 gWriteBuffers[i] = gBuffers0[i]; | |
43 gReadBuffers[i] = gBuffers1[i]; | |
44 if(gBuffers0[i] == 0 || gBuffers1[i] == 0) { | |
45 rt_printf("Error allocating buffer %d\n", i); | |
46 return false; | |
47 } | |
48 } | |
49 | |
50 gAnalysisTask = BeagleRT_createAuxiliaryTask(analyseResults, 50, "beaglert-analyse-results"); | |
51 | |
52 return true; | |
53 } | |
54 | |
55 // render() is called regularly at the highest priority by the audio engine. | |
56 // Input and output are given from the audio hardware and the other | |
57 // ADCs and DACs (if available). If only audio is available, numMatrixFrames | |
58 // will be 0. | |
59 | |
60 void render(BeagleRTContext *context, void *userData) | |
61 { | |
62 bool bufferIsFull = false; // Whether at least one buffer has filled | |
63 | |
64 for(unsigned int n = 0; n < context->audioFrames; n++) { | |
65 // Store audio inputs in buffer | |
66 for(unsigned int ch = 0; ch < context->audioChannels; ch++) { | |
67 if(gWriteBufferPointers[ch] < gBufferSize) { | |
68 gWriteBuffers[ch][gWriteBufferPointers[ch]] = | |
69 context->audioIn[n * context->audioChannels + ch]; | |
70 gWriteBufferPointers[ch]++; | |
71 if(gWriteBufferPointers[ch] >= gBufferSize) | |
72 bufferIsFull = true; | |
73 } | |
74 } | |
75 } | |
76 | |
77 if(context->analogChannels != 0) { | |
78 for(unsigned int n = 0; n < context->analogFrames; n++) { | |
79 // Store analog inputs in buffer, starting at channel 2 | |
80 for(unsigned int ch = 0; ch < context->analogChannels; ch++) { | |
81 if(gWriteBufferPointers[ch + 2] < gBufferSize) { | |
82 gWriteBuffers[ch + 2][gWriteBufferPointers[ch + 2]] = | |
83 context->analogIn[n * context->analogChannels + ch]; | |
84 gWriteBufferPointers[ch + 2]++; | |
85 if(gWriteBufferPointers[ch + 2] >= gBufferSize) | |
86 bufferIsFull = true; | |
87 } | |
88 | |
89 // Set all analog outputs to halfway point so they can be more | |
90 // easily measured for noise | |
91 context->analogOut[n * context->analogChannels + ch] = 0.5; | |
92 } | |
93 } | |
94 } | |
95 | |
96 | |
97 if(bufferIsFull) { | |
98 // Swap buffers and reset write pointers | |
99 for(int ch = 0; ch < 10; ch++) { | |
100 gReadBufferPointers[ch] = gWriteBufferPointers[ch]; | |
101 gWriteBufferPointers[ch] = 0; | |
102 | |
103 if(gReadBuffers[ch] == gBuffers0[ch]) { | |
104 gReadBuffers[ch] = gBuffers1[ch]; | |
105 gWriteBuffers[ch] = gBuffers0[ch]; | |
106 } | |
107 else { | |
108 gReadBuffers[ch] = gBuffers0[ch]; | |
109 gWriteBuffers[ch] = gBuffers1[ch]; | |
110 } | |
111 } | |
112 | |
113 BeagleRT_scheduleAuxiliaryTask(gAnalysisTask); | |
114 } | |
115 } | |
116 | |
117 void analyseResults() | |
118 { | |
119 rt_printf("\e[1;1H\e[2J"); // Command to clear the screen | |
120 | |
121 // Print the analysis results. channels 0-1 are audio, channels 2-9 are analog | |
122 for(int ch = 0; ch < 10; ch++) { | |
123 // Skip unused channels | |
124 if(gReadBufferPointers[ch] == 0) | |
125 continue; | |
126 | |
127 float mean = 0; | |
128 for(int n = 0; n < gReadBufferPointers[ch]; n++) { | |
129 mean += gReadBuffers[ch][n]; | |
130 } | |
131 mean /= (float)gReadBufferPointers[ch]; | |
132 | |
133 float rms = 0; | |
134 for(int n = 0; n < gReadBufferPointers[ch]; n++) { | |
135 rms += (gReadBuffers[ch][n] - mean) * (gReadBuffers[ch][n] - mean); | |
136 } | |
137 rms = sqrtf(rms / (float)gReadBufferPointers[ch]); | |
138 | |
139 if(ch == 0) | |
140 rt_printf("Audio In L: "); | |
141 else if(ch == 1) | |
142 rt_printf("Audio In R: "); | |
143 else | |
144 rt_printf("Analog In %d: ", ch - 2); | |
145 | |
146 rt_printf("Noise %6.1fdB DC offset %6.4f (%6.1fdB) window size: %d\n", | |
147 20.0f * log10f(rms), | |
148 mean, | |
149 20.0f * log10f(fabsf(mean)), | |
150 gReadBufferPointers[ch]); | |
151 } | |
152 } | |
153 | |
154 // cleanup() is called once at the end, after the audio has stopped. | |
155 // Release any resources that were allocated in setup(). | |
156 | |
157 void cleanup(BeagleRTContext *context, void *userData) | |
158 { | |
159 for(int i = 0; i < 10; i++) { | |
160 delete gBuffers0[i]; | |
161 delete gBuffers1[i]; | |
162 } | |
163 } |