You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdemultimedia/arts/modules/effects/synth_stereo_fir_equalizer_...

222 lines
5.2 KiB

/*
Copyright (C) 2001-2002 Stefan Westerfeld
stefan@space.twc.de
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include <math.h>
#include <arts/debug.h>
#include <arts/fft.h>
#include <arts/stdsynthmodule.h>
#include <arts/connect.h>
#include "artsmoduleseffects.h"
#include <stdio.h>
#include <stdlib.h>
#include <kglobal.h>
#include <klocale.h>
using namespace std;
using namespace Arts;
static inline bool odd(int x) { return ((x & 1) == 1); }
/* returns a blackman window: x is supposed to be in the interval [0..1] */
static inline float blackmanWindow(float x)
{
if(x < 0) return 0;
if(x > 1) return 0;
return 0.42-0.5*cos(M_PI*x*2)+0.08*cos(4*M_PI*x);
}
void firapprox(double *filter, int filtersize, vector<GraphPoint>& points)
{
assert((filtersize >= 3) && odd(filtersize));
int fft_size = 8;
while(fft_size/2 < filtersize)
fft_size *= 2;
vector<GraphPoint>::iterator pi = points.begin();
float lfreq=-2, lval=1.0, rfreq=-1, rval=1.0;
float *re = (float*) malloc(fft_size * sizeof(float));
for(int i=0;i<fft_size/2;i++)
{
float freq = float(i)/float(fft_size/2);
while(freq > rfreq && pi != points.end())
{
lfreq = rfreq; rfreq = pi->x;
lval = rval; rval = pi->y;
pi++;
}
float pos = (freq-lfreq)/(rfreq-lfreq);
float val = lval*(1.0-pos) + rval*pos;
//printf("%f %f\n",freq,val);
re[i] = re[fft_size-1-i] = val;
}
float *filter_re = (float*) malloc(fft_size * sizeof(float));
float *filter_im = (float*) malloc(fft_size * sizeof(float));
arts_fft_float (fft_size, 1, re, 0, filter_re, filter_im);
for(int i=0;i<filtersize;i++)
{
filter[i] = filter_re[(i+fft_size-filtersize/2) & (fft_size-1)]
* blackmanWindow(float(i+1)/float(filtersize+1));
}
free(re);
free(filter_re);
free(filter_im);
}
namespace Arts {
class Synth_STEREO_FIR_EQUALIZER_impl
: virtual public Synth_STEREO_FIR_EQUALIZER_skel,
virtual public StdSynthModule
{
vector<GraphPoint> _frequencies;
long _taps;
unsigned long bpos;
double filter[256];
float lbuffer[256];
float rbuffer[256];
public:
Synth_STEREO_FIR_EQUALIZER_impl()
{
_frequencies.push_back(GraphPoint(0.0,1.0));
_frequencies.push_back(GraphPoint(1.0,1.0));
_taps = 3;
for(bpos = 0; bpos < 256; bpos++)
lbuffer[bpos] = rbuffer[bpos] = 0.0;
calcFilter();
}
vector<GraphPoint> *frequencies() {
return new vector<GraphPoint>(_frequencies);
}
void frequencies(const vector<GraphPoint>& newFrequencies)
{
_frequencies = newFrequencies;
calcFilter();
}
long taps()
{
return _taps;
}
void taps(long newTaps)
{
arts_return_if_fail(newTaps >= 3 && newTaps <= 255);
if(!odd(newTaps))
newTaps++;
_taps = newTaps;
calcFilter();
}
void calcFilter()
{
firapprox(filter, _taps, _frequencies);
}
void calculateBlock(unsigned long samples)
{
for(unsigned i=0;i<samples;i++)
{
double lval = 0.0;
double rval = 0.0;
lbuffer[bpos & 255] = inleft[i];
rbuffer[bpos & 255] = inright[i];
for(int j=0;j<_taps;j++)
{
lval += lbuffer[(bpos-j) & 255] * filter[j];
rval += rbuffer[(bpos-j) & 255] * filter[j];
}
outleft[i] = lval;
outright[i] = rval;
bpos++;
}
}
};
REGISTER_IMPLEMENTATION(Synth_STEREO_FIR_EQUALIZER_impl);
class StereoFirEqualizerGuiFactory_impl : public StereoFirEqualizerGuiFactory_skel
{
public:
Widget createGui(Object equalizer);
};
REGISTER_IMPLEMENTATION(StereoFirEqualizerGuiFactory_impl);
}
Widget StereoFirEqualizerGuiFactory_impl::createGui(Object object)
{
KGlobal::locale()->insertCatalogue( "artsmodules" );
arts_return_val_if_fail(!object.isNull(), Arts::Widget::null());
Synth_STEREO_FIR_EQUALIZER equalizer = DynamicCast(object);
arts_return_val_if_fail(!equalizer.isNull(), Arts::Widget::null());
VBox vbox;
vbox.show();
Graph g;
g.parent(vbox);
g.width(400);
g.height(230);
g.caption(i18n("a graph").utf8().data());
g.minx(0.0);
g.maxx(1.0);
g.miny(0.0);
g.maxy(1.0);
g.show();
GraphLine gline;
gline.graph(g);
vector<GraphPoint> *points = equalizer.frequencies();
gline.points(*points);
delete points;
gline.color("red");
gline.editable(true);
connect(gline,"points_changed", equalizer, "frequencies");
g._addChild(gline,"gline");
SpinBox spinbox;
spinbox.caption(i18n("channels").utf8().data());
spinbox.min(3); spinbox.max(255);
spinbox.value(equalizer.taps());
spinbox.parent(vbox);
spinbox.show();
connect(spinbox,"value_changed", equalizer, "taps");
vbox._addChild(spinbox,"spinbox");
vbox._addChild(g,"g");
return vbox;
}