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
|
#include "guttersynth.h"
#include "plugin.h"
#include "handling.h"
static const char* badHandle = "guttersynth handle is not valid";
static csnd::Csound* cs = NULL;
void* allocate(size_t size) {
return cs->malloc(size);
}
void deallocate(void* item) {
cs->free(item);
}
/*
from max patch
sliders:
* singlegain = gain
* c = damp + (matrix out) , clipped
* dt = rate
* gamma = q
*/
class GutterSynth {
gutter_state* g;
csnd::Table* tGain;
csnd::Table* tFreq;
csnd::Table* tQ;
int dataSize;
void update_tables() {
int changed = 0;
for (int b = 0; b < g->bankCount; b++) {
if (memcmp(g->gains+(g->filterCount*b), tGain[b].data(), dataSize) != 0) {
memcpy(g->gains+(g->filterCount*b), tGain[b].data(), dataSize);
changed = 1;
}
if (memcmp(g->filterFreqs+(g->filterCount*b), tFreq[b].data(), dataSize) != 0) {
memcpy(g->filterFreqs+(g->filterCount*b), tFreq[b].data(), dataSize);
changed = 1;
}
if (memcmp(g->Q+(g->filterCount*b), tQ[b].data(), dataSize) != 0) {
memcpy(g->Q+(g->filterCount*b), tQ[b].data(), dataSize);
changed = 1;
}
}
if (changed) {
gutter_calccoeffs(g);
}
}
};
// ibankcount, ifiltercount, igaintbls[], ifreqtbls[], iqtables[], kgamma, komega, kc, kdt, ksinglegain, kfiltersOn, ksmoothing, kdistmethod
struct gutter : csnd::Plugin<1, 13> {
static constexpr char const *otypes = "a";
static constexpr char const *itypes = "iii[]i[]i[]kkkkkkkk";
gutter_state* g;
csnd::Table* tGain;
csnd::Table* tFreq;
csnd::Table* tQ;
int dataSize;
int init() {
if (cs == NULL) cs = csound;
csound->plugin_deinit(this);
g = gutter_init_ca(
(int)inargs[0], (int)inargs[1], csound->sr(),
&allocate,
&deallocate
);
csnd::Vector<MYFLT> &ftsGain = inargs.vector_data<MYFLT>(2);
csnd::Vector<MYFLT> &ftsFreq = inargs.vector_data<MYFLT>(3);
csnd::Vector<MYFLT> &ftsQ = inargs.vector_data<MYFLT>(4);
if (ftsGain.len() != g->bankCount) return csound->init_error("Gain tables do not match bank count");
if (ftsFreq.len() != g->bankCount) return csound->init_error("Frequency tables do not match bank count");
if (ftsQ.len() != g->bankCount) return csound->init_error("Q tables do not match bank count");
tGain = (csnd::Table*) csound->malloc(sizeof(csnd::Table) * g->bankCount);
tFreq = (csnd::Table*) csound->malloc(sizeof(csnd::Table) * g->bankCount);
tQ = (csnd::Table*) csound->malloc(sizeof(csnd::Table) * g->bankCount);
for (int b = 0; b < g->bankCount; b++) {
tGain[b].init(csound, &ftsGain[b]);
tFreq[b].init(csound, &ftsFreq[b]);
tQ[b].init(csound, &ftsQ[b]);
}
update_tables();
g->enableAudioInput = 0;
dataSize = sizeof(MYFLT) * g->filterCount;
return OK;
}
void update_tables() {
int changed = 0;
for (int b = 0; b < g->bankCount; b++) {
if (memcmp(g->gains+(g->filterCount*b), tGain[b].data(), dataSize) != 0) {
memcpy(g->gains+(g->filterCount*b), tGain[b].data(), dataSize);
changed = 1;
}
if (memcmp(g->filterFreqs+(g->filterCount*b), tFreq[b].data(), dataSize) != 0) {
memcpy(g->filterFreqs+(g->filterCount*b), tFreq[b].data(), dataSize);
changed = 1;
}
if (memcmp(g->Q+(g->filterCount*b), tQ[b].data(), dataSize) != 0) {
memcpy(g->Q+(g->filterCount*b), tQ[b].data(), dataSize);
changed = 1;
}
}
if (changed) {
gutter_calccoeffs(g);
}
}
int deinit() {
csound->free(tGain);
csound->free(tFreq);
csound->free(tQ);
gutter_cleanup(g);
return OK;
}
int aperf() {
update_tables(); // on trig?
g->gamma = inargs[5];
g->omega = inargs[6];
g->c = inargs[7];
g->dt = inargs[8];
g->singleGain = inargs[9];
g->filtersOn = (int) inargs[10];
g->smoothing = inargs[11];
g->distortionMethod = inargs[12];
gutter_process_samples(g, outargs(0), nsmps);
return OK;
}
};
#include <modload.h>
void csnd::on_load(csnd::Csound *csound) {
csnd::plugin<gutter>(csound, "gutter", csnd::thread::ia);
}
|