1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
|
/*
* Copyright (c) 2008 MUSIC TECHNOLOGY GROUP (MTG)
* UNIVERSITAT POMPEU FABRA
*
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
/*! \file filters.c
* \brief various filters
*/
#include "sms.h"
/*! \brief coefficient for pre_emphasis filter */
#define SMS_EMPH_COEF .9
/* pre-emphasis filter function, it returns the filtered value
*
* sfloat fInput; sound sample
*/
sfloat sms_preEmphasis (sfloat fInput, SMS_AnalParams *pAnalParams)
{
sfloat fOutput = 0;
fOutput = fInput - SMS_EMPH_COEF * pAnalParams->preEmphasisLastValue;
pAnalParams->preEmphasisLastValue = fOutput;
return (fOutput);
}
/* de-emphasis filter function, it returns the filtered value
*
* sfloat fInput; sound input
*/
sfloat sms_deEmphasis(sfloat fInput, SMS_SynthParams *pSynthParams)
{
sfloat fOutput = 0;
fOutput = fInput + SMS_EMPH_COEF * pSynthParams->deemphasisLastValue;
pSynthParams->deemphasisLastValue = fInput;
return(fOutput);
}
/*! \brief function to implement a zero-pole filter
*
* \todo will forgetting to reset pD to zero at the beginning of a new analysis
* (when there are multiple analyses within the life of one program)
* cause problems?
*
* \param pFa pointer to numerator coefficients
* \param pFb pointer to denominator coefficients
* \param nCoeff number of coefficients
* \param fInput input sample
* \return value is the filtered sample
*/
static sfloat ZeroPoleFilter (sfloat *pFa, sfloat *pFb, int nCoeff, sfloat fInput )
{
double fOut = 0;
int iSection;
/* static sfloat *pD = NULL; */
/* if (pD == NULL) */
/* pD = (sfloat *) calloc(5, sizeof(sfloat)); */
static sfloat pD[5] = {0, 0, 0, 0, 0};
pD[0] = fInput;
for (iSection = nCoeff-1; iSection > 0; iSection--)
{
fOut = fOut + pFa[iSection] * pD[iSection];
pD[0] = pD[0] - pFb[iSection] * pD[iSection];
pD[iSection] = pD [iSection-1];
}
fOut = fOut + pFa[0] * pD[0];
return((sfloat) fOut);
}
/* cutoff 1500 Hz */
/* static sfloat pFCoeff32k[10] = {0.679459, -2.71784, 4.07676, -2.71784,
0.679459, 1, -3.23168, 3.97664, -2.20137, 0.461665};
static sfloat pFCoeff36k[10] = {0.709489, -2.83796, 4.25694, -2.83796,
0.709489, 1, -3.31681, 4.17425, -2.3574, 0.503375};
static sfloat pFCoeff40k[10] = {0.734408, -2.93763, 4.40645, -2.93763,
0.734408, 1, -3.38497, 4.33706, -2.48914, 0.539355};
static sfloat pFCoeff441k[10] = {0.755893, -3.02357, 4.53536, -3.02357,
0.755893, 1, -3.44205, 4.47657, -2.6043, 0.571374};
static sfloat pFCoeff48k[10] = {0.773347, -3.09339, 4.64008, -3.09339,
0.773347, 1, -3.48731, 4.58929, -2.69888, 0.598065};
*/
/*! \brief function to filter a waveform with a high-pass filter
*
* cutoff =1500 Hz
*
* \todo this filter only works on sample rates up to 48k?
*
* \param sizeResidual size of signal
* \param pResidual pointer to residual signal
* \param iSamplingRate sampling rate of signal
*/
void sms_filterHighPass ( int sizeResidual, sfloat *pResidual, int iSamplingRate)
{
/* cutoff 800Hz */
static sfloat pFCoeff32k[10] = {0.814255, -3.25702, 4.88553, -3.25702,
0.814255, 1, -3.58973, 4.85128, -2.92405, 0.66301};
static sfloat pFCoeff36k[10] = {0.833098, -3.33239, 4.99859, -3.33239,
0.833098, 1, -3.63528, 4.97089, -3.02934,0.694052};
static sfloat pFCoeff40k[10] = {0.848475, -3.3939, 5.09085, -3.3939,
0.848475, 1, -3.67173, 5.068, -3.11597, 0.71991};
static sfloat pFCoeff441k[10] = {0.861554, -3.44622, 5.16932, -3.44622,
0.861554, 1, -3.70223, 5.15023, -3.19013, 0.742275};
static sfloat pFCoeff48k[10] = {0.872061, -3.48824, 5.23236, -3.48824,
0.872061, 1, -3.72641, 5.21605, -3.25002, 0.76049};
sfloat *pFCoeff, fSample = 0;
int i;
if (iSamplingRate <= 32000)
pFCoeff = pFCoeff32k;
else if (iSamplingRate <= 36000)
pFCoeff = pFCoeff36k;
else if (iSamplingRate <= 40000)
pFCoeff = pFCoeff40k;
else if (iSamplingRate <= 44100)
pFCoeff = pFCoeff441k;
else
pFCoeff = pFCoeff48k;
for(i = 0; i < sizeResidual; i++)
{
/* try to avoid underflow when there is nothing to filter */
if (i > 0 && fSample == 0 && pResidual[i] == 0)
return;
fSample = pResidual[i];
pResidual[i] =
ZeroPoleFilter (&pFCoeff[0], &pFCoeff[5], 5, fSample);
}
}
/*! \brief a spectral filter
*
* filter each point of the current array by the surounding
* points using a triangular window
*
* \param pFArray two dimensional input array
* \param size1 vertical size of pFArray
* \param size2 horizontal size of pFArray
* \param pFOutArray output array of size size1
*/
void sms_filterArray (sfloat *pFArray, int size1, int size2, sfloat *pFOutArray)
{
int i, j, iPoint, iFrame, size2_2 = size2-2, size2_1 = size2-1;
sfloat *pFCurrentArray = pFArray + (size2_1) * size1;
sfloat fVal, fWeighting, fTotalWeighting, fTmpVal;
/* find the filtered envelope */
for (i = 0; i < size1; i++)
{
fVal = pFCurrentArray[i];
fTotalWeighting = 1;
/* filter by the surrounding points */
for (j = 1; j < (size2_2); j++)
{
fWeighting = (sfloat) size2 / (1+ j);
/* filter on the vertical dimension */
/* consider the lower points */
iPoint = i - (size2_1) + j;
if (iPoint >= 0)
{
fVal += pFCurrentArray[iPoint] * fWeighting;
fTotalWeighting += fWeighting;
}
/* consider the higher points */
iPoint = i + (size2_1) - j;
if (iPoint < size1)
{
fVal += pFCurrentArray[iPoint] * fWeighting;
fTotalWeighting += fWeighting;
}
/*filter on the horizontal dimension */
/* consider the previous points */
iFrame = j;
fTmpVal = pFArray[iFrame*size1 + i];
if (fTmpVal)
{
fVal += fTmpVal * fWeighting;
fTotalWeighting += fWeighting;
}
}
/* scale value by weighting */
pFOutArray[i] = fVal / fTotalWeighting;
}
}
|