chris@163
|
1 /**
|
chris@163
|
2 * Copyright (c) 2014, 2015, Enzien Audio Ltd.
|
chris@163
|
3 *
|
chris@163
|
4 * Permission to use, copy, modify, and/or distribute this software for any
|
chris@163
|
5 * purpose with or without fee is hereby granted, provided that the above
|
chris@163
|
6 * copyright notice and this permission notice appear in all copies.
|
chris@163
|
7 *
|
chris@163
|
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
chris@163
|
9 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
chris@163
|
10 * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
chris@163
|
11 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
chris@163
|
12 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
chris@163
|
13 * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
chris@163
|
14 * PERFORMANCE OF THIS SOFTWARE.
|
chris@163
|
15 */
|
chris@163
|
16
|
chris@163
|
17 #include "ControlDelay.h"
|
chris@163
|
18
|
chris@163
|
19 hv_size_t cDelay_init(HvBase *_c, ControlDelay *o, float delayMs) {
|
chris@163
|
20 o->delay = ctx_millisecondsToSamples(_c, delayMs);
|
chris@163
|
21 hv_memset(o->msgs, __HV_DELAY_MAX_MESSAGES*sizeof(HvMessage *));
|
chris@163
|
22 return 0;
|
chris@163
|
23 }
|
chris@163
|
24
|
chris@163
|
25 void cDelay_onMessage(HvBase *_c, ControlDelay *o, int letIn, const HvMessage *const m,
|
chris@163
|
26 void (*sendMessage)(HvBase *, int, const HvMessage *const)) {
|
chris@163
|
27 switch (letIn) {
|
chris@163
|
28 case 0: {
|
chris@163
|
29 if (msg_compareSymbol(m, 0, "flush")) {
|
chris@163
|
30 // send all messages immediately
|
chris@163
|
31 for (int i = 0; i < __HV_DELAY_MAX_MESSAGES; i++) {
|
chris@163
|
32 HvMessage *n = o->msgs[i];
|
chris@163
|
33 if (n != NULL) {
|
chris@163
|
34 msg_setTimestamp(n, msg_getTimestamp(m)); // update the timestamp to now
|
chris@163
|
35 sendMessage(_c, 0, n); // send the message
|
chris@163
|
36 ctx_cancelMessage(_c, n, sendMessage); // then clear it
|
chris@163
|
37 // NOTE(mhroth): there may be a problem here if a flushed message causes a clear message to return
|
chris@163
|
38 // to this object in the same step
|
chris@163
|
39 }
|
chris@163
|
40 }
|
chris@163
|
41 hv_memset(o->msgs, __HV_DELAY_MAX_MESSAGES*sizeof(HvMessage *));
|
chris@163
|
42 } else if (msg_compareSymbol(m, 0, "clear")) {
|
chris@163
|
43 // cancel (clear) all (pending) messages
|
chris@163
|
44 for (int i = 0; i < __HV_DELAY_MAX_MESSAGES; i++) {
|
chris@163
|
45 HvMessage *n = o->msgs[i];
|
chris@163
|
46 if (n != NULL) {
|
chris@163
|
47 ctx_cancelMessage(_c, n, sendMessage);
|
chris@163
|
48 }
|
chris@163
|
49 }
|
chris@163
|
50 hv_memset(o->msgs, __HV_DELAY_MAX_MESSAGES*sizeof(HvMessage *));
|
chris@163
|
51 } else {
|
chris@163
|
52 hv_uint32_t ts = msg_getTimestamp(m);
|
chris@163
|
53 msg_setTimestamp((HvMessage *) m, ts+o->delay); // update the timestamp to set the delay
|
chris@163
|
54 int i;
|
chris@163
|
55 for (i = 0; i < __HV_DELAY_MAX_MESSAGES; i++) {
|
chris@163
|
56 if (o->msgs[i] == NULL) {
|
chris@163
|
57 o->msgs[i] = ctx_scheduleMessage(_c, m, sendMessage, 0);
|
chris@163
|
58 break;
|
chris@163
|
59 }
|
chris@163
|
60 }
|
chris@163
|
61 hv_assert(i < __HV_DELAY_MAX_MESSAGES); // scheduled message limit reached
|
chris@163
|
62 msg_setTimestamp((HvMessage *) m, ts); // return to the original timestamp
|
chris@163
|
63 }
|
chris@163
|
64 break;
|
chris@163
|
65 }
|
chris@163
|
66 case 1: {
|
chris@163
|
67 if (msg_isFloat(m,0)) {
|
chris@163
|
68 // set delay in milliseconds
|
chris@163
|
69 o->delay = ctx_millisecondsToSamples(_c,msg_getFloat(m,0));
|
chris@163
|
70 }
|
chris@163
|
71 break;
|
chris@163
|
72 }
|
chris@163
|
73 case 2: {
|
chris@163
|
74 if (msg_isFloat(m,0)) {
|
chris@163
|
75 // set delay in samples
|
chris@163
|
76 o->delay = (hv_uint32_t) msg_getFloat(m,0);
|
chris@163
|
77 }
|
chris@163
|
78 break;
|
chris@163
|
79 }
|
chris@163
|
80 default: break;
|
chris@163
|
81 }
|
chris@163
|
82 }
|
chris@163
|
83
|
chris@163
|
84 void cDelay_clearExecutingMessage(ControlDelay *o, const HvMessage *const m) {
|
chris@163
|
85 for (int i = 0; i < __HV_DELAY_MAX_MESSAGES; ++i) {
|
chris@163
|
86 if (o->msgs[i] == m) {
|
chris@163
|
87 o->msgs[i] = NULL;
|
chris@163
|
88 break;
|
chris@163
|
89 }
|
chris@163
|
90 }
|
chris@163
|
91 }
|