Line | Branch | Exec | Source |
---|---|---|---|
1 | #include "gwion_util.h" | ||
2 | #include "gwion_ast.h" | ||
3 | #include "gwion_env.h" | ||
4 | #include "vm.h" | ||
5 | #include "gwion.h" | ||
6 | #include "object.h" | ||
7 | #include "instr.h" | ||
8 | #include "operator.h" | ||
9 | #include "import.h" | ||
10 | #include "gwi.h" | ||
11 | #include "ugen.h" | ||
12 | #include "driver.h" | ||
13 | #include "gwi.h" | ||
14 | #include "array.h" | ||
15 | |||
16 | 3287132 | static TICK(dac_tick) { | |
17 | 3287132 | m_float *out = ((VM *)u->module.gen.data)->bbq->out; | |
18 | 3287132 | uint i = 0; | |
19 | 3287132 | m_float sum = 0; | |
20 | do { | ||
21 | 6574264 | sum += out[i] = UGEN(u->connect.multi->channel[i])->in; | |
22 |
2/2✓ Branch 0 taken 3287132 times.
✓ Branch 1 taken 3287132 times.
|
6574264 | } while (++i < u->connect.multi->n_out); |
23 | 3287132 | u->out = sum / u->connect.multi->n_out; | |
24 | 3287132 | } | |
25 | |||
26 | 528001 | static TICK(adc_tick) { | |
27 | 528001 | const m_float *in = ((VM *)u->module.gen.data)->bbq->in; | |
28 | 528001 | uint i = 0; | |
29 | 528001 | m_float sum = 0; | |
30 | 1056002 | do sum += UGEN(u->connect.multi->channel[i])->out = in[i]; | |
31 |
2/2✓ Branch 0 taken 528001 times.
✓ Branch 1 taken 528001 times.
|
1056002 | while (++i < u->connect.multi->n_out); |
32 | 528001 | u->out = sum / u->connect.multi->n_out; | |
33 | 528001 | } | |
34 | |||
35 | #define COMPUTE(a) \ | ||
36 | if (!(a)->done) (a)->compute((a)); | ||
37 | |||
38 | 19393336 | __attribute__((hot)) ANN void compute_mono(const UGen u) { | |
39 | 19393336 | u->done = 1; | |
40 | 19393336 | const uint size = u->connect.net->size; | |
41 |
2/2✓ Branch 0 taken 8110302 times.
✓ Branch 1 taken 11283034 times.
|
19393336 | if (size) { |
42 | 8110302 | const Vector vec = &u->connect.net->from; | |
43 | 8110302 | const UGen v = (UGen)vector_front(vec); | |
44 |
2/2✓ Branch 0 taken 4295155 times.
✓ Branch 1 taken 3815147 times.
|
8110302 | COMPUTE(v) |
45 | 8110302 | u->in = v->out; | |
46 |
2/2✓ Branch 0 taken 7054274 times.
✓ Branch 1 taken 8110302 times.
|
15164576 | for (uint i = 1; i < size; ++i) { |
47 | 7054274 | const UGen w = (UGen)vector_at(vec, i); | |
48 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 7054269 times.
|
7054274 | COMPUTE(w) |
49 | static const void *dispatch[] = {&&add, &&sub, &&mul, &&div}; | ||
50 | do { | ||
51 | 7054274 | goto *dispatch[u->op]; | |
52 | 6766274 | add: | |
53 | 6766274 | u->in += w->out; | |
54 | 6766274 | break; | |
55 | 96000 | sub: | |
56 | 96000 | u->in -= w->out; | |
57 | 96000 | break; | |
58 | 96000 | mul: | |
59 | 96000 | u->in *= w->out; | |
60 | 96000 | break; | |
61 | 96000 | div: | |
62 | 96000 | u->in *= w->out; | |
63 | 96000 | break; | |
64 | } while (0); | ||
65 | } | ||
66 | } | ||
67 | 19393336 | } | |
68 | 3815133 | __attribute__((hot)) ANN void compute_multi(const UGen u) { | |
69 | 3815133 | u->done = 1; | |
70 | 3815133 | uint i = 0; | |
71 | 7630266 | do compute_mono(UGEN(u->connect.multi->channel[i])); | |
72 |
2/2✓ Branch 0 taken 3815133 times.
✓ Branch 1 taken 3815133 times.
|
7630266 | while (++i < u->connect.multi->n_chan); |
73 | 3815133 | } | |
74 | |||
75 | 3815133 | __attribute__((hot)) ANN void compute_chan(const UGen u) { | |
76 |
1/2✓ Branch 0 taken 3815133 times.
✗ Branch 1 not taken.
|
3815133 | COMPUTE(u->module.ref); |
77 | 3815133 | } | |
78 | |||
79 | #define describe_compute(func, opt, aux) \ | ||
80 | __attribute__((hot)) ANN void gen_compute_##func##opt(const UGen u) { \ | ||
81 | compute_##func(u); \ | ||
82 | aux u->module.gen.tick(u); \ | ||
83 | } | ||
84 | 20329880 | describe_compute(mono, , ) describe_compute(multi, , ) | |
85 | ✗ | describe_compute(mono, trig, | |
86 | { u->module.gen.trig->compute(u->module.gen.trig); }) | ||
87 | ✗ | describe_compute(multi, trig, | |
88 | { u->module.gen.trig->compute(u->module.gen.trig); }) | ||
89 | |||
90 | 4483 | ANEW UGen new_UGen(MemPool p) { | |
91 | 4483 | const UGen u = mp_calloc(p, UGen); | |
92 | 4483 | u->op = 0; | |
93 | 4483 | u->compute = gen_compute_mono; | |
94 | 4483 | return u; | |
95 | } | ||
96 | |||
97 | 4471 | ANEW M_Object new_M_UGen(const struct Gwion_ *gwion) { | |
98 | 4471 | const M_Object o = new_object(gwion->mp, gwion->type[et_ugen]); | |
99 | 4471 | UGEN(o) = new_UGen(gwion->mp); | |
100 | 4471 | return o; | |
101 | } | ||
102 | |||
103 | 1276 | ANN static void assign_channel(const struct Gwion_ *gwion, const UGen u) { | |
104 | 1276 | u->multi = 1; | |
105 | 1276 | u->compute = gen_compute_multi; | |
106 | 2552 | u->connect.multi->channel = | |
107 | 1276 | (M_Object *)xmalloc(u->connect.multi->n_chan * SZ_INT); | |
108 |
2/2✓ Branch 0 taken 2552 times.
✓ Branch 1 taken 1276 times.
|
3828 | for (uint i = u->connect.multi->n_chan + 1; --i;) { |
109 | 2552 | const uint j = i - 1; | |
110 | 2552 | const M_Object chan = new_M_UGen(gwion); | |
111 | 2552 | ugen_ini(gwion, UGEN(chan), u->connect.multi->n_in > j, | |
112 | 2552 | u->connect.multi->n_out > j); | |
113 | 2552 | UGEN(chan)->module.ref = u; | |
114 | 2552 | UGEN(chan)->compute = compute_chan; | |
115 | 2552 | u->connect.multi->channel[j] = chan; | |
116 | } | ||
117 | 1276 | } | |
118 | |||
119 | 1931 | ANN void ugen_gen(const struct Gwion_ *gwion, const UGen u, const f_tick tick, | |
120 | void *data, const m_bool trig) { | ||
121 | 1931 | u->module.gen.tick = tick; | |
122 | 1931 | u->module.gen.data = data; | |
123 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1931 times.
|
1931 | if (trig) { |
124 | ✗ | u->module.gen.trig = new_UGen(gwion->mp); | |
125 | ✗ | u->module.gen.trig->compute = compute_mono; | |
126 | ✗ | ugen_ini(gwion, u->module.gen.trig, 1, 1); | |
127 | ✗ | u->compute = (u->compute == gen_compute_mono ? gen_compute_monotrig | |
128 | ✗ | : gen_compute_multitrig); | |
129 | } | ||
130 | 1931 | } | |
131 | |||
132 | 3845 | ANN void ugen_ini(const struct Gwion_ *gwion, const UGen u, const uint in, | |
133 | const uint out) { | ||
134 | 3845 | const uint chan = in > out ? in : out; | |
135 |
2/2✓ Branch 0 taken 3207 times.
✓ Branch 1 taken 638 times.
|
3845 | if (chan == 1) { |
136 | 3207 | u->connect.net = mp_calloc(gwion->mp, ugen_net); | |
137 | 3207 | vector_init(&u->connect.net->from); | |
138 | 3207 | vector_init(&u->connect.net->to); | |
139 | } else { | ||
140 | 638 | u->connect.multi = mp_calloc(gwion->mp, ugen_multi); | |
141 | 638 | u->connect.multi->n_in = in; | |
142 | 638 | u->connect.multi->n_out = out; | |
143 | 638 | u->connect.multi->n_chan = chan; | |
144 | 638 | assign_channel(gwion, u); | |
145 | } | ||
146 | 3845 | } | |
147 | |||
148 | 16 | ANN void connect_init(const VM_Shred shred, restrict M_Object *lhs, | |
149 | restrict M_Object *rhs) { | ||
150 | 16 | POP_REG(shred, SZ_INT * 2); | |
151 | 16 | *rhs = *(M_Object *)REG(SZ_INT); | |
152 | 16 | *lhs = *(M_Object *)REG(0); | |
153 | 16 | } | |
154 | |||
155 | #define describe_connect(name, func) \ | ||
156 | ANN static inline void name##onnect(const restrict UGen lhs, \ | ||
157 | const restrict UGen rhs) { \ | ||
158 | func(&rhs->connect.net->from, (vtype)lhs); \ | ||
159 | func(&lhs->connect.net->to, (vtype)rhs); \ | ||
160 | rhs->connect.net->size = (uint)vector_size(&rhs->connect.net->from); \ | ||
161 | } | ||
162 | 1302 | describe_connect(C, vector_add) | |
163 | 2 | describe_connect(Disc, vector_rem2) | |
164 | |||
165 | 16 | ANN static void release_connect(const VM_Shred shred) { | |
166 | 16 | *(M_Object *)REG(0) = *(M_Object *)REG(SZ_INT); | |
167 | 16 | PUSH_REG(shred, SZ_INT); | |
168 | 16 | } | |
169 | |||
170 | typedef ANN void (*f_connect)(const UGen lhs, const UGen rhs); | ||
171 | 654 | ANN /* static */ void _do_(const f_connect f, const UGen lhs, const UGen rhs) { | |
172 | 654 | const m_bool l_multi = lhs->multi; | |
173 | 654 | const m_bool r_multi = rhs->multi; | |
174 |
2/2✓ Branch 0 taken 642 times.
✓ Branch 1 taken 12 times.
|
654 | const uint l_max = l_multi ? lhs->connect.multi->n_out : 1; |
175 |
2/2✓ Branch 0 taken 11 times.
✓ Branch 1 taken 643 times.
|
654 | const uint r_max = r_multi ? rhs->connect.multi->n_in : 1; |
176 | 654 | const uint max = l_max > r_max ? l_max : r_max; | |
177 | 654 | uint i = 0; | |
178 | assert(r_max > 0); | ||
179 | do { | ||
180 |
2/2✓ Branch 0 taken 1284 times.
✓ Branch 1 taken 20 times.
|
1304 | const UGen l = l_multi ? UGEN(lhs->connect.multi->channel[i % l_max]) : lhs; |
181 |
2/2✓ Branch 0 taken 22 times.
✓ Branch 1 taken 1282 times.
|
1304 | const UGen r = r_multi ? UGEN(rhs->connect.multi->channel[i % r_max]) : rhs; |
182 | 1304 | f(l, r); | |
183 |
2/2✓ Branch 0 taken 650 times.
✓ Branch 1 taken 654 times.
|
1304 | } while (++i < max); |
184 | 654 | } | |
185 | |||
186 | 638 | ANN void ugen_connect(const restrict UGen lhs, const restrict UGen rhs) { | |
187 | 638 | _do_(Connect, lhs, rhs); | |
188 | 638 | } | |
189 | |||
190 | ✗ | ANN void ugen_disconnect(const restrict UGen lhs, const restrict UGen rhs) { | |
191 | ✗ | _do_(Disconnect, lhs, rhs); | |
192 | } | ||
193 | |||
194 | #define TRIG_EX \ | ||
195 | if (!UGEN(rhs)->module.gen.trig) { \ | ||
196 | release_connect(shred); \ | ||
197 | handle(shred, "NonTriggerException"); \ | ||
198 | return; \ | ||
199 | } | ||
200 | #define describe_connect_instr(name, func, tgt, opt) \ | ||
201 | static INSTR(name##func) { \ | ||
202 | M_Object lhs, rhs; \ | ||
203 | connect_init(shred, &lhs, &rhs); \ | ||
204 | opt _do_(func, UGEN(lhs), tgt); \ | ||
205 | release_connect(shred); \ | ||
206 | } | ||
207 | 15 | describe_connect_instr(Ugen, Connect, UGEN(rhs), ); | |
208 | 1 | describe_connect_instr(Ugen, Disconnect, UGEN(rhs), ); | |
209 | ✗ | describe_connect_instr(Trig, Connect, UGEN(rhs)->module.gen.trig, TRIG_EX); | |
210 | ✗ | describe_connect_instr(Trig, Disconnect, UGEN(rhs)->module.gen.trig, TRIG_EX); | |
211 | |||
212 | #define describe_connectaa_instr(name, func, tgt, opt) \ | ||
213 | static INSTR(name##func) { \ | ||
214 | M_Object lhs, rhs; \ | ||
215 | connect_init(shred, &lhs, &rhs); \ | ||
216 | const m_uint lsz = m_vector_size(ARRAY(lhs)); \ | ||
217 | const m_uint rsz = m_vector_size(ARRAY(rhs)); \ | ||
218 | const m_uint min = lsz < rsz ? lsz : rsz; \ | ||
219 | for (m_uint i = 0; i < min; ++i) { \ | ||
220 | M_Object ulhs, urhs; \ | ||
221 | m_vector_get(ARRAY(lhs), i, &ulhs); \ | ||
222 | m_vector_get(ARRAY(rhs), i, &urhs); \ | ||
223 | opt if (ulhs && urhs) _do_(func, UGEN(ulhs), tgt); \ | ||
224 | } \ | ||
225 | release_connect(shred); \ | ||
226 | } | ||
227 | ✗ | describe_connectaa_instr( | |
228 | UgenAA, Connect, | ||
229 | ✗ | UGEN(urhs), ) describe_connectaa_instr(UgenAA, Disconnect, | |
230 | UGEN(urhs), ) | ||
231 | ✗ | describe_connectaa_instr( | |
232 | TrigAA, Connect, UGEN(urhs)->module.gen.trig, | ||
233 | ✗ | TRIG_EX) describe_connectaa_instr(TrigAA, Disconnect, | |
234 | UGEN(urhs)->module.gen.trig, | ||
235 | TRIG_EX) | ||
236 | |||
237 | #define describe_connectua_instr(name, func, tgt, opt) \ | ||
238 | static INSTR(name##func) { \ | ||
239 | M_Object lhs, rhs; \ | ||
240 | connect_init(shred, &lhs, &rhs); \ | ||
241 | const m_uint rsz = m_vector_size(ARRAY(rhs)); \ | ||
242 | for (m_uint i = 0; i < rsz; ++i) { \ | ||
243 | M_Object urhs; \ | ||
244 | m_vector_get(ARRAY(rhs), i, &urhs); \ | ||
245 | opt if (urhs) _do_(func, UGEN(lhs), tgt); \ | ||
246 | } \ | ||
247 | release_connect(shred); \ | ||
248 | } | ||
249 | ✗ | describe_connectua_instr(UgenUA, Connect, UGEN(urhs), ) | |
250 | ✗ | describe_connectua_instr(UgenUA, Disconnect, UGEN(urhs), ) | |
251 | ✗ | describe_connectua_instr(TrigUA, Connect, | |
252 | UGEN(urhs)->module.gen.trig, | ||
253 | TRIG_EX) | ||
254 | ✗ | describe_connectua_instr( | |
255 | TrigUA, Disconnect, UGEN(urhs)->module.gen.trig, | ||
256 | TRIG_EX) | ||
257 | |||
258 | #define describe_connectau_instr(name, func, tgt, opt) \ | ||
259 | static INSTR(name##func) { \ | ||
260 | M_Object lhs, rhs; \ | ||
261 | connect_init(shred, &lhs, &rhs); \ | ||
262 | const m_uint lsz = m_vector_size(ARRAY(lhs)); \ | ||
263 | for (m_uint i = 0; i < lsz; ++i) { \ | ||
264 | M_Object ulhs; \ | ||
265 | m_vector_get(ARRAY(lhs), i, &ulhs); \ | ||
266 | opt if (ulhs) _do_(func, UGEN(ulhs), tgt); \ | ||
267 | } \ | ||
268 | release_connect(shred); \ | ||
269 | } | ||
270 | ✗ | describe_connectau_instr(UgenAU, Connect, | |
271 | UGEN(rhs), ) | ||
272 | ✗ | describe_connectau_instr(UgenAU, Disconnect, | |
273 | UGEN(rhs), ) | ||
274 | ✗ | describe_connectau_instr( | |
275 | TrigAU, Connect, | ||
276 | UGEN(rhs)->module.gen.trig, TRIG_EX) | ||
277 | ✗ | describe_connectau_instr( | |
278 | TrigAU, Disconnect, | ||
279 | UGEN(rhs)->module.gen.trig, | ||
280 | TRIG_EX) | ||
281 | |||
282 | 12 | static CTOR(ugen_ctor) { | |
283 | 12 | UGEN(o) = new_UGen(shred->info->mp); | |
284 | 12 | vector_add(&shred->info->vm->ugen, (vtype)UGEN(o)); | |
285 | 12 | } | |
286 | |||
287 | #define describe_release_func(src, tgt, opt) \ | ||
288 | ANN static void release_##src(const UGen ug) { \ | ||
289 | for (uint i = (uint)vector_size(&ug->connect.net->src) + 1; --i;) { \ | ||
290 | const UGen u = (UGen)vector_at(&ug->connect.net->src, i - 1); \ | ||
291 | vector_rem2(&u->connect.net->tgt, (vtype)ug); \ | ||
292 | opt \ | ||
293 | } \ | ||
294 | vector_release(&ug->connect.net->src); \ | ||
295 | } | ||
296 |
2/2✓ Branch 3 taken 5 times.
✓ Branch 4 taken 3203 times.
|
3208 | describe_release_func(from, to, ) |
297 |
2/2✓ Branch 3 taken 1295 times.
✓ Branch 4 taken 3203 times.
|
4498 | describe_release_func(to, from, --u->connect.net->size;) |
298 | |||
299 | 3203 | ANN static void release_mono(MemPool p, const UGen ug) { | |
300 | 3203 | release_to(ug); | |
301 | 3203 | release_from(ug); | |
302 | 3203 | mp_free(p, ugen_net, ug->connect.net); | |
303 | 3203 | } | |
304 | |||
305 | 1276 | ANN static void release_multi(const UGen ug, const VM_Shred shred) { | |
306 |
2/2✓ Branch 0 taken 2552 times.
✓ Branch 1 taken 1276 times.
|
3828 | for (uint i = ug->connect.multi->n_chan + 1; --i;) |
307 | 2552 | release(ug->connect.multi->channel[i - 1], shred); | |
308 | 1276 | xfree(ug->connect.multi->channel); | |
309 | 1276 | } | |
310 | |||
311 | 4479 | static DTOR(ugen_dtor) { | |
312 | 4479 | const UGen ug = UGEN(o); | |
313 | 4479 | MemPool p = shred->info->mp; | |
314 | 4479 | vector_rem2(&shred->info->vm->ugen, (vtype)ug); | |
315 |
2/2✓ Branch 0 taken 3203 times.
✓ Branch 1 taken 1276 times.
|
4479 | if (!ug->multi) |
316 | 3203 | release_mono(p, ug); | |
317 | else | ||
318 | 1276 | release_multi(ug, shred); | |
319 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 4479 times.
|
4479 | if (ug->module.gen.trig) { |
320 | ✗ | release_mono(p, ug->module.gen.trig); | |
321 | ✗ | mp_free(shred->info->mp, UGen, ug->module.gen.trig); | |
322 | } | ||
323 | 4479 | mp_free(shred->info->mp, UGen, ug); | |
324 | 4479 | } | |
325 | |||
326 | 2 | static MFUN(ugen_channel) { | |
327 | 2 | const m_int i = *(m_int *)MEM(SZ_INT); | |
328 |
3/4✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
|
2 | if (!UGEN(o)->multi && !i) |
329 | 1 | *(M_Object *)RETURN = o; | |
330 |
1/4✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
|
1 | else if (i >= 0 && (m_uint)i < UGEN(o)->connect.multi->n_chan) |
331 | ✗ | *(M_Object *)RETURN = UGEN(o)->connect.multi->channel[i]; | |
332 | else | ||
333 | 1 | xfun_handle(shred, "InvalidChannelRequest"); | |
334 | 2 | } | |
335 | |||
336 | 10 | static MFUN(ugen_get_op) { *(m_uint *)RETURN = UGEN(o)->op + 1; } | |
337 | |||
338 | 5 | ANN static void set_op(const UGen u, const uint f) { | |
339 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 4 times.
|
5 | if (u->multi) { |
340 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 1 times.
|
3 | for (uint i = u->connect.multi->n_chan + 1; --i;) |
341 | 2 | UGEN(u->connect.multi->channel[i - 1])->op = f; | |
342 | } | ||
343 | 5 | u->op = f; | |
344 | 5 | } | |
345 | |||
346 | 6 | static MFUN(ugen_set_op) { | |
347 | 6 | const m_int i = *(m_int *)MEM(SZ_INT); | |
348 |
3/4✓ Branch 0 taken 5 times.
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 5 times.
✗ Branch 3 not taken.
|
6 | if (i >= 1 && i <= 4) set_op(UGEN(o), i - 1); |
349 | 6 | *(m_int *)RETURN = UGEN(o)->op + 1; | |
350 | 6 | } | |
351 | |||
352 | 12 | static MFUN(ugen_get_last) { *(m_float *)RETURN = UGEN(o)->out; } | |
353 | |||
354 | struct ugen_importer { | ||
355 | const VM * vm; | ||
356 | const f_tick tick; | ||
357 | const m_str name; | ||
358 | const uint nchan; | ||
359 | }; | ||
360 | |||
361 | 1276 | ANN static UGen add_ugen(const Gwi gwi, struct ugen_importer *imp) { | |
362 | 1276 | VM * vm = gwi_vm(gwi); | |
363 | 1276 | const M_Object o = new_M_UGen(gwi->gwion); | |
364 | 1276 | const UGen u = UGEN(o); | |
365 | 1276 | ugen_ini(vm->gwion, u, imp->nchan, imp->nchan); | |
366 | // ugen_gen(vm->gwion, u, imp->tick, (void *)imp->vm, 0); | ||
367 | 1276 | ugen_gen(vm->gwion, u, imp->tick, o, 0); | |
368 | 1276 | vector_add(&vm->ugen, (vtype)u); | |
369 | 1276 | gwi_item_ini(gwi, "UGen", imp->name); | |
370 | 1276 | gwi_item_end(gwi, ae_flag_const, obj, o); | |
371 | 1276 | return u; | |
372 | } | ||
373 | |||
374 | 638 | static GWION_IMPORT(global_ugens) { | |
375 | 638 | VM *const vm = gwi_vm(gwi); | |
376 | 638 | struct ugen_importer imp_hole = {vm, compute_mono, "blackhole", 1}; | |
377 | 638 | const UGen hole = add_ugen(gwi, &imp_hole); | |
378 | // dac needs to have *multi* | ||
379 | 638 | const M_Object dac = new_M_UGen(gwi->gwion); | |
380 | 638 | const UGen u = UGEN(dac); | |
381 | 638 | u->connect.multi = mp_calloc(gwi->gwion->mp, ugen_multi); | |
382 | 638 | u->connect.multi->n_in = vm->bbq->si->out; | |
383 | 638 | u->connect.multi->n_out = vm->bbq->si->out; | |
384 | 638 | u->connect.multi->n_chan = vm->bbq->si->out; | |
385 | 638 | assign_channel(gwi->gwion, u); | |
386 | 638 | ugen_gen(vm->gwion, u, dac_tick, (void *)vm, 0); | |
387 | 638 | vector_add(&vm->ugen, (vtype)u); | |
388 | 638 | gwi_item_ini(gwi, "UGen", "dac"); | |
389 | 638 | gwi_item_end(gwi, ae_flag_const, obj, dac); | |
390 | 638 | ugen_connect(u, hole); | |
391 | |||
392 | |||
393 | 638 | struct ugen_importer imp_adc = {vm, adc_tick, "adc", vm->bbq->si->in}; | |
394 | 638 | (void)add_ugen(gwi, &imp_adc); | |
395 | 638 | SET_FLAG(gwi->gwion->type[et_ugen], abstract); | |
396 | 638 | return GW_OK; | |
397 | } | ||
398 | |||
399 | 22 | static OP_CHECK(opck_chuck_ugen) { | |
400 | 22 | const Exp_Binary *bin = (Exp_Binary *)data; | |
401 | 22 | return bin->rhs->type; | |
402 | } | ||
403 | |||
404 | 638 | GWION_IMPORT(ugen) { | |
405 | 638 | const Type t_ugen = gwi_class_ini(gwi, "UGen", NULL); | |
406 | 638 | gwi_class_xtor(gwi, ugen_ctor, ugen_dtor); | |
407 | 638 | gwi->gwion->type[et_ugen] = t_ugen; // use func | |
408 | 638 | t_ugen->nspc->offset += SZ_INT; | |
409 | |||
410 | 638 | GWI_BB(gwi_func_ini(gwi, "UGen", "chan")) | |
411 | 638 | GWI_BB(gwi_func_arg(gwi, "int", "arg0")) | |
412 | 638 | GWI_BB(gwi_func_end(gwi, ugen_channel, ae_flag_none)) | |
413 | |||
414 | 638 | GWI_BB(gwi_func_ini(gwi, "int", "op")) | |
415 | 638 | GWI_BB(gwi_func_end(gwi, ugen_get_op, ae_flag_none)) | |
416 | |||
417 | 638 | GWI_BB(gwi_func_ini(gwi, "int", "op")) | |
418 | 638 | GWI_BB(gwi_func_arg(gwi, "int", "arg0")) | |
419 | 638 | GWI_BB(gwi_func_end(gwi, ugen_set_op, ae_flag_none)) | |
420 | |||
421 | 638 | GWI_BB(gwi_func_ini(gwi, "float", "last")) | |
422 | 638 | GWI_BB(gwi_func_end(gwi, ugen_get_last, ae_flag_none)) | |
423 | 638 | GWI_BB(gwi_class_end(gwi)) | |
424 | |||
425 | 638 | GWI_BB(gwi_oper_ini(gwi, "UGen", "UGen", "UGen")) | |
426 | 638 | _CHECK_OP("~>", chuck_ugen, UgenConnect) | |
427 | 638 | _CHECK_OP("~<", chuck_ugen, UgenDisconnect) | |
428 | 638 | _CHECK_OP(":~>", chuck_ugen, TrigConnect) | |
429 | 638 | _CHECK_OP(":~<", chuck_ugen, TrigDisconnect) | |
430 | |||
431 | 638 | GWI_BB(gwi_oper_ini(gwi, "UGen[]", "UGen[]", "UGen[]")) | |
432 | 638 | _CHECK_OP("~>", chuck_ugen, UgenAAConnect) | |
433 | 638 | _CHECK_OP("~<", chuck_ugen, UgenAADisconnect) | |
434 | 638 | _CHECK_OP(":~>", chuck_ugen, TrigAAConnect) | |
435 | 638 | _CHECK_OP(":~<", chuck_ugen, TrigAADisconnect) | |
436 | |||
437 | 638 | GWI_BB(gwi_oper_ini(gwi, "UGen", "UGen[]", "UGen[]")) | |
438 | 638 | _CHECK_OP("~>", chuck_ugen, UgenUAConnect) | |
439 | 638 | _CHECK_OP("~<", chuck_ugen, UgenUADisconnect) | |
440 | 638 | _CHECK_OP(":~>", chuck_ugen, TrigUAConnect) | |
441 | 638 | _CHECK_OP(":~<", chuck_ugen, TrigUADisconnect) | |
442 | |||
443 | 638 | GWI_BB(gwi_oper_ini(gwi, "UGen[]", "UGen", "UGen")) | |
444 | 638 | _CHECK_OP("~>", chuck_ugen, UgenAUConnect) | |
445 | 638 | _CHECK_OP("~<", chuck_ugen, UgenAUDisconnect) | |
446 | 638 | _CHECK_OP(":~>", chuck_ugen, TrigAUConnect) | |
447 | 638 | _CHECK_OP(":~<", chuck_ugen, TrigAUDisconnect) | |
448 | |||
449 | 638 | return gwimport_global_ugens(gwi); | |
450 | } | ||
451 |