File: PV_ThirdParty.cpp

package info (click to toggle)
supercollider 1%3A3.13.0%2Brepack-1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm
  • size: 80,292 kB
  • sloc: cpp: 476,363; lisp: 84,680; ansic: 77,685; sh: 25,509; python: 7,909; makefile: 3,440; perl: 1,964; javascript: 974; xml: 826; java: 677; yacc: 314; lex: 175; objc: 152; ruby: 136
file content (79 lines) | stat: -rw-r--r-- 2,412 bytes parent folder | download | duplicates (3)
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
/*
    SuperCollider real time audio synthesis system
    Copyright (c) 2002 James McCartney. All rights reserved.
    http://www.audiosynth.com

    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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301  USA
*/

// third party Phase Vocoder UGens


#include "FFT_UGens.h"


extern "C" {
void PV_ConformalMap_Ctor(PV_Unit* unit);
void PV_ConformalMap_next(PV_Unit* unit, int inNumSamples);
}


void PV_ConformalMap_Ctor(PV_Unit* unit) {
    SETCALC(PV_ConformalMap_next);
    ZOUT0(0) = ZIN0(0);
}


void PV_ConformalMap_next(PV_Unit* unit, int inNumSamples) {
    PV_GET_BUF

    SCComplexBuf* p = ToComplexApx(buf);

    float real2 = ZIN0(1);
    float imag2 = ZIN0(2);

    for (int i = 0; i < numbins; ++i) {
        float real1 = p->bin[i].real;
        float imag1 = p->bin[i].imag;

        // apply conformal map z-> z-a/(1-za*) where z is the existing complex number in the bin and a is defined by
        // inputs 1 and 2
        float numr = real1 - real2;
        float numi = imag1 - imag2;
        float denomr = 1.f - (real1 * real2 + imag1 * imag2);
        float denomi = (real1 * imag2 - real2 * imag1);

        numr = numr * denomr + numi * denomi;
        numi = numi * denomr - numr * denomi;

        // squared modulus
        denomr = denomr * denomr + denomi * denomi;

        // avoid possible divide by zero
        if (denomr < 0.001f)
            denomr = 0.001f;
        denomr = 1.f / denomr;

        p->bin[i].real = numr * denomr;
        p->bin[i].imag = numi * denomr;
    }
}


#define DefinePVUnit(name) (*ft->fDefineUnit)(#name, sizeof(PV_Unit), (UnitCtorFunc)&name##_Ctor, 0, 0);


// void initPV_ThirdParty(InterfaceTable *it);
void initPV_ThirdParty(InterfaceTable* it) { DefinePVUnit(PV_ConformalMap); }