]> git.sesse.net Git - vlc/blob - modules/audio_filter/spatializer/revmodel.cpp
Spatializer audio filter, work for Google Summer of Code, by Isaac Biodun Osunkunle...
[vlc] / modules / audio_filter / spatializer / revmodel.cpp
1 // Reverb model implementation
2 // 
3 // 
4 // Google Summer of Code 2007
5 // 
6 // Authors: Biodun Osunkunle <biodun@videolan.org> 
7 // 
8 // Mentor : Jean-Baptiste Kempf <jb@videolan.org>
9 //
10 // Original written by Jezar at Dreampoint, June 2000
11
12 // This code is public domain
13
14 #include "revmodel.hpp"
15 #include <stdlib.h>
16
17 revmodel::revmodel()
18 {
19         // Tie the components to their buffers
20         combL[0].setbuffer(bufcombL1,combtuningL1);
21         combR[0].setbuffer(bufcombR1,combtuningR1);
22         combL[1].setbuffer(bufcombL2,combtuningL2);
23         combR[1].setbuffer(bufcombR2,combtuningR2);
24         combL[2].setbuffer(bufcombL3,combtuningL3);
25         combR[2].setbuffer(bufcombR3,combtuningR3);
26         combL[3].setbuffer(bufcombL4,combtuningL4);
27         combR[3].setbuffer(bufcombR4,combtuningR4);
28         combL[4].setbuffer(bufcombL5,combtuningL5);
29         combR[4].setbuffer(bufcombR5,combtuningR5);
30         combL[5].setbuffer(bufcombL6,combtuningL6);
31         combR[5].setbuffer(bufcombR6,combtuningR6);
32         combL[6].setbuffer(bufcombL7,combtuningL7);
33         combR[6].setbuffer(bufcombR7,combtuningR7);
34         combL[7].setbuffer(bufcombL8,combtuningL8);
35         combR[7].setbuffer(bufcombR8,combtuningR8);
36         allpassL[0].setbuffer(bufallpassL1,allpasstuningL1);
37         allpassR[0].setbuffer(bufallpassR1,allpasstuningR1);
38         allpassL[1].setbuffer(bufallpassL2,allpasstuningL2);
39         allpassR[1].setbuffer(bufallpassR2,allpasstuningR2);
40         allpassL[2].setbuffer(bufallpassL3,allpasstuningL3);
41         allpassR[2].setbuffer(bufallpassR3,allpasstuningR3);
42         allpassL[3].setbuffer(bufallpassL4,allpasstuningL4);
43         allpassR[3].setbuffer(bufallpassR4,allpasstuningR4);
44
45         // Set default values
46         allpassL[0].setfeedback(0.5f);
47         allpassR[0].setfeedback(0.5f);
48         allpassL[1].setfeedback(0.5f);
49         allpassR[1].setfeedback(0.5f);
50         allpassL[2].setfeedback(0.5f);
51         allpassR[2].setfeedback(0.5f);
52         allpassL[3].setfeedback(0.5f);
53         allpassR[3].setfeedback(0.5f);
54         setwet(initialwet);
55         setroomsize(initialroom);
56         setdry(initialdry);
57         setdamp(initialdamp);
58         setwidth(initialwidth);
59         setmode(initialmode);
60
61         // Buffer will be full of rubbish - so we MUST mute them
62         mute();
63 }
64
65 void revmodel::mute()
66 {
67         int i;
68         if (getmode() >= freezemode)
69                 return;
70
71         for (i = 0 ; i < numcombs ; i++)
72         {
73                 combL[i].mute();
74                 combR[i].mute();
75         }
76         for (i=0;i<numallpasses;i++)
77         {
78                 allpassL[i].mute();
79                 allpassR[i].mute();
80         }
81 }
82
83 void revmodel::processreplace(float *inputL, float *outputL, long numsamples, int skip)
84 {
85         float outL,outR,input;
86         float inputR;
87         int i;
88
89         outL = outR = 0;
90         if (skip > 1) 
91            inputR = inputL[1];
92         else 
93            inputR = inputL[0];         
94                 input = (inputL[0] + inputR) * gain;
95                 
96                 // Accumulate comb filters in parallel
97                 for(i=0; i<numcombs; i++)
98                 {
99                         outL += combL[i].process(input);
100                         outR += combR[i].process(input);
101                 }
102
103                 // Feed through allpasses in series
104                 for(i=0; i<numallpasses; i++)
105                 {
106                         outL = allpassL[i].process(outL);
107                         outR = allpassR[i].process(outR);
108                 }
109
110                 // Calculate output REPLACING anything already there
111                 outputL[0] = (outL*wet1 + outR*wet2 + inputR*dry);
112            if (skip > 1)
113                 outputL[1] = (outR*wet1 + outL*wet2 + inputR*dry);
114 }
115
116 void revmodel::processmix(float *inputL, float *outputL, long numsamples, int skip)
117 {
118         float outL,outR,input;
119         float inputR;
120         int i;
121
122         outL = outR = 0;
123         if (skip > 1) 
124            inputR = inputL[1];
125         else 
126            inputR = inputL[0];         
127                 input = (inputL[0] + inputR) * gain;
128                 
129                 // Accumulate comb filters in parallel
130                 for(i=0; i<numcombs; i++)
131                 {
132                         outL += combL[i].process(input);
133                         outR += combR[i].process(input);
134                 }
135
136                 // Feed through allpasses in series
137                 for(i=0; i<numallpasses; i++)
138                 {
139                         outL = allpassL[i].process(outL);
140                         outR = allpassR[i].process(outR);
141                 }
142
143                 // Calculate output REPLACING anything already there
144                 outputL[0] += (outL*wet1 + outR*wet2 + inputR*dry);
145            if (skip > 1)
146                 outputL[1] += (outR*wet1 + outL*wet2 + inputR*dry);
147 }
148
149 void revmodel::update()
150 {
151 // Recalculate internal values after parameter change
152
153         int i;
154
155         wet1 = wet*(width/2 + 0.5f);
156         wet2 = wet*((1-width)/2);
157
158         if (mode >= freezemode)
159         {
160                 roomsize1 = 1;
161                 damp1 = 0;
162                 gain = muted;
163         }
164         else
165         {
166                 roomsize1 = roomsize;
167                 damp1 = damp;
168                 gain = fixedgain;
169         }
170
171         for(i=0; i<numcombs; i++)
172         {
173                 combL[i].setfeedback(roomsize1);
174                 combR[i].setfeedback(roomsize1);
175         }
176
177         for(i=0; i<numcombs; i++)
178         {
179                 combL[i].setdamp(damp1);
180                 combR[i].setdamp(damp1);
181         }
182 }
183
184 // The following get/set functions are not inlined, because
185 // speed is never an issue when calling them, and also
186 // because as you develop the reverb model, you may
187 // wish to take dynamic action when they are called.
188
189 void revmodel::setroomsize(float value)
190 {
191         roomsize = (value*scaleroom) + offsetroom;
192         update();
193 }
194
195 float revmodel::getroomsize()
196 {
197         return (roomsize-offsetroom)/scaleroom;
198 }
199
200 void revmodel::setdamp(float value)
201 {
202         damp = value*scaledamp;
203         update();
204 }
205
206 float revmodel::getdamp()
207 {
208         return damp/scaledamp;
209 }
210
211 void revmodel::setwet(float value)
212 {
213         wet = value*scalewet;
214         update();
215 }
216
217 float revmodel::getwet()
218 {
219         return wet/scalewet;
220 }
221
222 void revmodel::setdry(float value)
223 {
224         dry = value*scaledry;
225 }
226
227 float revmodel::getdry()
228 {
229         return dry/scaledry;
230 }
231
232 void revmodel::setwidth(float value)
233 {
234         width = value;
235         update();
236 }
237
238 float revmodel::getwidth()
239 {
240         return width;
241 }
242
243 void revmodel::setmode(float value)
244 {
245         mode = value;
246         update();
247 }
248
249 float revmodel::getmode()
250 {
251         if (mode >= freezemode)
252                 return 1;
253         else
254                 return 0;
255 }
256
257 //ends