Ardour  9.0-pre0-1051-g8dd89616e1
dsp_filter.h
Go to the documentation of this file.
1 /*
2  * Copyright (C) 2016-2017 Robin Gareus <robin@gareus.org>
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License along
15  * with this program; if not, write to the Free Software Foundation, Inc.,
16  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
17  */
18 #ifndef _dsp_filter_h_
19 #define _dsp_filter_h_
20 
21 #include <atomic>
22 #include <cstdint>
23 #include <cstring>
24 #include <assert.h>
25 #include <glib.h>
26 #include <glibmm.h>
27 #include <fftw3.h>
28 
29 #include "pbd/enum_convert.h"
30 #include "pbd/malign.h"
31 
32 #include "ardour/buffer_set.h"
33 #include "ardour/chan_mapping.h"
35 #include "ardour/types.h"
36 
37 namespace ARDOUR { namespace DSP {
38 
56  class DspShm {
57  public:
58  DspShm (size_t s = 0)
59  : _data (0)
60  , _size (0)
61  {
62  assert (sizeof(float) == sizeof (int32_t));
63  assert (sizeof(float) == sizeof (int));
64  allocate (s);
65  }
66 
67  ~DspShm () {
69  }
70 
75  void allocate (size_t s) {
76  if (s == _size) { return; }
78  cache_aligned_malloc ((void**) &_data, sizeof (float) * s);
79  if (_data) { _size = s; }
80  }
81 
83  void clear () {
84  memset (_data, 0, sizeof(float) * _size);
85  }
86 
92  float* to_float (size_t off) {
93  if (off >= _size) { return 0; }
94  return &(((float*)_data)[off]);
95  }
96 
102  int32_t* to_int (size_t off) {
103  if (off >= _size) { return 0; }
104  return &(((int32_t*)_data)[off]);
105  }
106 
116  void atomic_set_int (size_t off, int32_t val) {
117  /* no read or write that precedes the write we are
118  about to do can be reordered past this fence.
119  */
120  std::atomic_thread_fence (std::memory_order_release);
121  ((int32_t*)_data)[off] = val;
122  }
123 
133  int32_t atomic_get_int (size_t off) {
134  /* no read that precedes the read we are
135  about to do can be reordered past this fence.
136  */
137  std::atomic_thread_fence (std::memory_order_acquire);
138  return (((int32_t*)_data)[off]);
139  }
140 
141  private:
142  void* _data;
143  size_t _size;
144  };
145 
147  void memset (float *data, const float val, const uint32_t n_samples);
155  void mmult (float *data, float *mult, const uint32_t n_samples);
163  void peaks (const float *data, float &min, float &max, uint32_t n_samples);
164 
170  float log_meter (float power);
176  float log_meter_coeff (float coeff);
177 
178  void process_map (BufferSet* bufs,
179  const ChanCount& n_out,
180  const ChanMapping& in_map,
181  const ChanMapping& out_map,
182  pframes_t nframes, samplecnt_t offset);
183 
186  public:
192  LowPass (double samplerate, float freq);
198  void proc (float *data, const uint32_t n_samples);
207  void ctrl (float *data, const float val, const uint32_t n_samples);
212  void set_cutoff (float freq);
214  void reset () { _z = 0.f; }
215  private:
216  float _rate;
217  float _z;
218  float _a;
219  };
220 
223  public:
224  enum Type {
237  MatchedPeaking
238  };
239 
244  Biquad (double samplerate);
245  Biquad (const Biquad &other);
246 
252  void run (float *data, const uint32_t n_samples);
260  void compute (Type t, double freq, double Q, double gain);
261 
263  void configure (double a1, double a2, double b0, double b1, double b2);
264 
265  /* copy coefficients from other instance, retain state */
266  void configure (Biquad const&);
267 
272  float dB_at_freq (float freq) const;
273 
275  void reset () { _z1 = _z2 = 0.0; }
276 
277  void coefficients (double& a1, double& a2, double& b0, double& b1, double& b2) const;
278  private:
279  void set_vicanek_poles (const double W0, const double Q, const double A = 1.0);
280  void calc_vicanek (const double W0, double& A0, double& A1, double& A2, double& phi0, double& phi1, double& phi2);
281 
282  double _rate;
283  float _z1, _z2;
284  double _a1, _a2;
285  double _b0, _b1, _b2;
286  };
287 
289  public:
290  virtual ~SpectrumAnalyzer () {}
291  virtual float power_at_bin (const uint32_t bin, const float gain, bool pink) const = 0;
292  virtual float freq_at_bin (const uint32_t bin) const = 0;
293  };
294 
296  public:
297  FFTSpectrum (uint32_t window_size, double rate);
299 
307  void set_data_hann (float const * const data, const uint32_t n_samples, const uint32_t offset = 0);
308 
310  void execute ();
311 
317  float power_at_bin (const uint32_t bin, const float gain = 1.f, bool pink = false) const;
318 
319  float freq_at_bin (const uint32_t bin) const {
320  return bin * _fft_freq_per_bin;
321  }
322 
323  private:
324  float* hann_window;
325 
326  void init (uint32_t window_size, double rate);
327  void reset ();
328 
330  uint32_t _fft_data_size;
332 
333  float* _fft_data_in;
335  float* _fft_power;
336 
337  fftwf_plan _fftplan;
338  };
339 
341  public:
342  PerceptualAnalyzer (double rate, int ipsize = 4096);
345 
346  class Trace {
347  public:
348  Trace (int size);
349  ~Trace ();
350 
351  bool _valid;
352  int32_t _count;
353  float *_data;
354  };
355 
356  enum ProcessMode {
359  MM_AVER
360  };
361 
362  enum Speed {
367  Noise
368  };
369 
370  enum Warp {
373  High
374  };
375 
376  void set_wfact (float wfact);
377  void set_speed (float speed);
378 
379  void set_wfact (enum Warp);
380  void set_speed (enum Speed);
381 
382  void reset ();
383 
384  int fftlen () const { return _fftlen; }
385  float* ipdata () const { return _ipdata; }
386  Trace* power () const { return _power; }
387  Trace* peakp () const { return _peakp; }
388  float pmax () const { return _pmax; }
389 
391  void process (int iplen, ProcessMode mode = MM_NONE);
392 
393  static double warp_freq (double w, double f);
394 
395  float freq_at_bin (const uint32_t bin) const;
396  float power_at_bin (const uint32_t bin, const float gain = 1.f, bool pink = false) const;
397 
398  private:
399  static const int _fftlen = 512;
400 
401  void init ();
402  float conv0 (fftwf_complex*);
403  float conv1 (fftwf_complex*);
404 
405  int _ipsize;
406  int _icount;
407  fftwf_plan _fftplan;
408  float *_ipdata;
409  float *_warped;
410  fftwf_complex *_trdata;
413  float _fsamp;
414  float _wfact;
415  float _speed;
416  float _pmax;
417  float _fscale[513];
418  float _bwcorr[513];
419  };
420 
422  public:
424 
425  enum Type {
429  };
430 
431  void run (float *data, const uint32_t n_samples);
432  void set_type (Type t);
433 
434  private:
435  uint32_t randi ();
436  float randf () { return (randi () / 1073741824.f) - 1.f; }
437  float grandf ();
438 
440  uint32_t _rseed;
441  /* pink-noise */
442  float _b0, _b1, _b2, _b3, _b4, _b5, _b6;
443  /* gaussian white */
444  bool _pass;
445  float _rn;
446 
447  };
448 
449 } } /* namespace */
450 
451 namespace PBD {
454 }
455 #endif
void set_vicanek_poles(const double W0, const double Q, const double A=1.0)
void calc_vicanek(const double W0, double &A0, double &A1, double &A2, double &phi0, double &phi1, double &phi2)
Biquad(double samplerate)
void compute(Type t, double freq, double Q, double gain)
void run(float *data, const uint32_t n_samples)
void coefficients(double &a1, double &a2, double &b0, double &b1, double &b2) const
Biquad(const Biquad &other)
void configure(Biquad const &)
void configure(double a1, double a2, double b0, double b1, double b2)
float dB_at_freq(float freq) const
void allocate(size_t s)
Definition: dsp_filter.h:75
DspShm(size_t s=0)
Definition: dsp_filter.h:58
int32_t atomic_get_int(size_t off)
Definition: dsp_filter.h:133
void atomic_set_int(size_t off, int32_t val)
Definition: dsp_filter.h:116
float * to_float(size_t off)
Definition: dsp_filter.h:92
int32_t * to_int(size_t off)
Definition: dsp_filter.h:102
void set_data_hann(float const *const data, const uint32_t n_samples, const uint32_t offset=0)
FFTSpectrum(uint32_t window_size, double rate)
void init(uint32_t window_size, double rate)
float power_at_bin(const uint32_t bin, const float gain=1.f, bool pink=false) const
float freq_at_bin(const uint32_t bin) const
Definition: dsp_filter.h:319
void run(float *data, const uint32_t n_samples)
void set_cutoff(float freq)
LowPass(double samplerate, float freq)
void ctrl(float *data, const float val, const uint32_t n_samples)
void proc(float *data, const uint32_t n_samples)
float power_at_bin(const uint32_t bin, const float gain=1.f, bool pink=false) const
void process(int iplen, ProcessMode mode=MM_NONE)
PerceptualAnalyzer(PerceptualAnalyzer const &)=delete
PerceptualAnalyzer(double rate, int ipsize=4096)
static double warp_freq(double w, double f)
float freq_at_bin(const uint32_t bin) const
float conv1(fftwf_complex *)
float conv0(fftwf_complex *)
virtual float power_at_bin(const uint32_t bin, const float gain, bool pink) const =0
virtual float freq_at_bin(const uint32_t bin) const =0
#define LIBARDOUR_API
int cache_aligned_malloc(void **memptr, size_t size)
void cache_aligned_free(void *memptr)
float log_meter_coeff(float coeff)
void mmult(float *data, float *mult, const uint32_t n_samples)
float log_meter(float power)
void process_map(BufferSet *bufs, const ChanCount &n_out, const ChanMapping &in_map, const ChanMapping &out_map, pframes_t nframes, samplecnt_t offset)
void peaks(const float *data, float &min, float &max, uint32_t n_samples)
void memset(float *data, const float val, const uint32_t n_samples)
PBD::PropertyDescriptor< gain_t > gain
uint32_t pframes_t
Temporal::samplecnt_t samplecnt_t
Definition: axis_view.h:42
DEFINE_ENUM_CONVERT(ARDOUR::DSP::PerceptualAnalyzer::Speed)