chris@160
|
1 /*
|
chris@160
|
2 * render.cpp
|
chris@160
|
3 *
|
chris@160
|
4 * Template render.cpp file for on-board heavy compiling
|
chris@160
|
5 *
|
chris@160
|
6 * N.B. this is currently *not* compatible with foleyDesigner source files!
|
chris@160
|
7 *
|
chris@160
|
8 * Created on: November 5, 2015
|
chris@160
|
9 *
|
chris@160
|
10 * Christian Heinrichs
|
chris@160
|
11 *
|
chris@160
|
12 */
|
chris@160
|
13
|
chris@160
|
14 #include <BeagleRT.h>
|
chris@160
|
15 #include <cmath>
|
chris@160
|
16 #include "../include/Utilities.h"
|
chris@160
|
17 #include "Heavy_bbb.h"
|
chris@190
|
18 #include <string.h>
|
chris@190
|
19 #include <stdlib.h>
|
chris@190
|
20 #include <string.h>
|
chris@160
|
21
|
chris@160
|
22 /*
|
chris@160
|
23 * HEAVY CONTEXT & BUFFERS
|
chris@160
|
24 */
|
chris@160
|
25
|
chris@160
|
26 Hv_bbb *gHeavyContext;
|
chris@160
|
27 float *gHvInputBuffers = NULL, *gHvOutputBuffers = NULL;
|
chris@160
|
28 int gHvInputChannels = 0, gHvOutputChannels = 0;
|
chris@160
|
29
|
chris@160
|
30 float gInverseSampleRate;
|
chris@160
|
31
|
chris@160
|
32 /*
|
chris@160
|
33 * HEAVY FUNCTIONS
|
chris@160
|
34 */
|
chris@160
|
35
|
chris@160
|
36 void printHook(double timestampSecs, const char *printLabel, const char *msgString, void *userData) {
|
chris@160
|
37 printf("Message from Heavy patch: [@ %.3f] %s: %s\n", timestampSecs, printLabel, msgString);
|
chris@160
|
38 }
|
chris@160
|
39
|
chris@160
|
40 static void sendHook(
|
chris@160
|
41 double timestamp, // in milliseconds
|
chris@160
|
42 const char *receiverName,
|
chris@160
|
43 const HvMessage *const m,
|
chris@160
|
44 void *userData) {
|
chris@160
|
45
|
chris@160
|
46 // only react to messages sent to receivers named "hello"
|
chris@160
|
47 if (!strncmp(receiverName, "hello", 5)) {
|
chris@160
|
48 }
|
chris@160
|
49
|
chris@160
|
50 }
|
chris@160
|
51
|
chris@160
|
52 /*
|
chris@166
|
53 * SETUP, RENDER LOOP & CLEANUP
|
chris@160
|
54 */
|
chris@160
|
55
|
chris@160
|
56 bool setup(BeagleRTContext *context, void *userData) {
|
chris@160
|
57
|
chris@160
|
58 /* HEAVY */
|
chris@160
|
59
|
chris@160
|
60 gHeavyContext = hv_bbb_new(context->audioSampleRate);
|
chris@160
|
61
|
chris@160
|
62 gHvInputChannels = hv_getNumInputChannels(gHeavyContext);
|
chris@160
|
63 gHvOutputChannels = hv_getNumOutputChannels(gHeavyContext);
|
chris@160
|
64
|
chris@160
|
65 rt_printf("Starting Heavy context with %d input channels and %d output channels\n",
|
chris@160
|
66 gHvInputChannels, gHvOutputChannels);
|
chris@160
|
67
|
chris@160
|
68 if(gHvInputChannels != 0) {
|
chris@160
|
69 gHvInputBuffers = (float *)calloc(gHvInputChannels * context->audioFrames,sizeof(float));
|
chris@160
|
70 }
|
chris@160
|
71 if(gHvOutputChannels != 0) {
|
chris@160
|
72 gHvOutputBuffers = (float *)calloc(gHvOutputChannels * context->audioFrames,sizeof(float));
|
chris@160
|
73 }
|
chris@160
|
74
|
chris@160
|
75 gInverseSampleRate = 1.0 / context->audioSampleRate;
|
chris@160
|
76
|
chris@160
|
77 // Set heavy print hook
|
chris@160
|
78 hv_setPrintHook(gHeavyContext, &printHook);
|
chris@160
|
79 // Set heavy send hook
|
chris@160
|
80 hv_setSendHook(gHeavyContext, sendHook);
|
chris@160
|
81
|
chris@160
|
82 return true;
|
chris@160
|
83 }
|
chris@160
|
84
|
chris@160
|
85
|
chris@160
|
86 void render(BeagleRTContext *context, void *userData)
|
chris@160
|
87 {
|
chris@160
|
88
|
chris@160
|
89 // De-interleave the data
|
chris@160
|
90 if(gHvInputBuffers != NULL) {
|
chris@160
|
91 for(int n = 0; n < context->audioFrames; n++) {
|
chris@160
|
92 for(int ch = 0; ch < gHvInputChannels; ch++) {
|
chris@160
|
93 if(ch >= context->audioChannels+context->analogChannels) {
|
chris@160
|
94 // THESE ARE PARAMETER INPUT 'CHANNELS' USED FOR ROUTING
|
chris@160
|
95 // 'sensor' outputs from routing channels of dac~ are passed through here
|
chris@160
|
96 break;
|
chris@160
|
97 } else {
|
chris@160
|
98 // If more than 2 ADC inputs are used in the pd patch, route the analog inputs
|
chris@160
|
99 // i.e. ADC3->analogIn0 etc. (first two are always audio inputs)
|
chris@160
|
100 if(ch >= context->audioChannels) {
|
chris@160
|
101 int m = n/2;
|
chris@160
|
102 float mIn = context->analogIn[m*context->analogChannels + (ch-context->audioChannels)];
|
chris@160
|
103 gHvInputBuffers[ch * context->audioFrames + n] = mIn;
|
chris@160
|
104 } else {
|
chris@160
|
105 gHvInputBuffers[ch * context->audioFrames + n] = context->audioIn[n * context->audioChannels + ch];
|
chris@160
|
106 }
|
chris@160
|
107 }
|
chris@160
|
108 }
|
chris@160
|
109 }
|
chris@160
|
110 }
|
chris@160
|
111
|
chris@160
|
112 // replacement for bang~ object
|
chris@160
|
113 //hv_vscheduleMessageForReceiver(gHeavyContext, "bbb_bang", 0.0f, "b");
|
chris@160
|
114
|
chris@160
|
115 hv_bbb_process_inline(gHeavyContext, gHvInputBuffers, gHvOutputBuffers, context->audioFrames);
|
chris@160
|
116
|
chris@160
|
117 // Interleave the output data
|
chris@160
|
118 if(gHvOutputBuffers != NULL) {
|
chris@160
|
119 for(int n = 0; n < context->audioFrames; n++) {
|
chris@160
|
120
|
chris@160
|
121 for(int ch = 0; ch < gHvOutputChannels; ch++) {
|
chris@160
|
122 if(ch >= context->audioChannels+context->analogChannels) {
|
chris@160
|
123 // THESE ARE SENSOR OUTPUT 'CHANNELS' USED FOR ROUTING
|
chris@160
|
124 // they are the content of the 'sensor output' dac~ channels
|
chris@160
|
125 } else {
|
chris@160
|
126 if(ch >= context->audioChannels) {
|
chris@160
|
127 int m = n/2;
|
chris@160
|
128 context->analogOut[m * context->analogFrames + (ch-context->audioChannels)] = constrain(gHvOutputBuffers[ch*context->audioFrames + n],0.0,1.0);
|
chris@160
|
129 } else {
|
chris@160
|
130 context->audioOut[n * context->audioChannels + ch] = gHvOutputBuffers[ch * context->audioFrames + n];
|
chris@160
|
131 }
|
chris@160
|
132 }
|
chris@160
|
133 }
|
chris@160
|
134 }
|
chris@160
|
135 }
|
chris@160
|
136
|
chris@160
|
137 }
|
chris@160
|
138
|
chris@160
|
139
|
chris@160
|
140 void cleanup(BeagleRTContext *context, void *userData)
|
chris@160
|
141 {
|
chris@160
|
142
|
chris@160
|
143 hv_bbb_free(gHeavyContext);
|
chris@160
|
144 if(gHvInputBuffers != NULL)
|
chris@160
|
145 free(gHvInputBuffers);
|
chris@160
|
146 if(gHvOutputBuffers != NULL)
|
chris@160
|
147 free(gHvOutputBuffers);
|
chris@160
|
148
|
chris@160
|
149 }
|