chris@160
|
1 /**
|
chris@160
|
2 * Copyright (c) 2014, 2015, Enzien Audio Ltd.
|
chris@160
|
3 *
|
chris@160
|
4 * Permission to use, copy, modify, and/or distribute this software for any
|
chris@160
|
5 * purpose with or without fee is hereby granted, provided that the above
|
chris@160
|
6 * copyright notice and this permission notice appear in all copies.
|
chris@160
|
7 *
|
chris@160
|
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
chris@160
|
9 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
chris@160
|
10 * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
chris@160
|
11 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
chris@160
|
12 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
chris@160
|
13 * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
chris@160
|
14 * PERFORMANCE OF THIS SOFTWARE.
|
chris@160
|
15 */
|
chris@160
|
16
|
chris@160
|
17 #ifndef _HEAVY_MESSAGE_H_
|
chris@160
|
18 #define _HEAVY_MESSAGE_H_
|
chris@160
|
19
|
chris@160
|
20 #include "Utils.h"
|
chris@160
|
21
|
chris@160
|
22 typedef enum ElementType {
|
chris@160
|
23 BANG,
|
chris@160
|
24 FLOAT,
|
chris@160
|
25 SYMBOL,
|
chris@160
|
26 HASH
|
chris@160
|
27 } ElementType;
|
chris@160
|
28
|
chris@160
|
29 typedef struct Element {
|
chris@160
|
30 ElementType type;
|
chris@160
|
31 union {
|
chris@160
|
32 float f; // float
|
chris@160
|
33 char *s; // symbol
|
chris@160
|
34 hv_uint32_t h; // hash
|
chris@160
|
35 } data;
|
chris@160
|
36 } Element;
|
chris@160
|
37
|
chris@160
|
38 typedef struct HvMessage {
|
chris@160
|
39 hv_uint32_t timestamp; // the sample at which this message should be processed
|
chris@160
|
40 hv_uint16_t numElements;
|
chris@160
|
41 hv_uint16_t numBytes; // the number of bytes that this message occupies in memory
|
chris@160
|
42 Element elem;
|
chris@160
|
43 } HvMessage;
|
chris@160
|
44
|
chris@160
|
45 #define HV_MESSAGE_ON_STACK(_x) (HvMessage *) hv_alloca(msg_getByteSize(_x))
|
chris@160
|
46
|
chris@160
|
47 /** Returns the total length in bytes of this message for a given number of elements. */
|
chris@160
|
48 static inline hv_size_t msg_getByteSize(hv_size_t numElements) {
|
chris@160
|
49 hv_assert(numElements > 0);
|
chris@160
|
50 return sizeof(HvMessage) + ((numElements-1) * sizeof(Element));
|
chris@160
|
51 }
|
chris@160
|
52
|
chris@160
|
53 HvMessage *msg_copy(const HvMessage *m);
|
chris@160
|
54
|
chris@160
|
55 /** Returns the number of bytes that this message would occupy on the heap. */
|
chris@160
|
56 hv_size_t msg_getNumHeapBytes(const HvMessage *m);
|
chris@160
|
57
|
chris@160
|
58 /** Copies the message into the given buffer. The buffer must be at least as large as msg_getNumHeapBytes(). */
|
chris@160
|
59 void msg_copyToBuffer(const HvMessage *m, char *buffer, hv_size_t len);
|
chris@160
|
60
|
chris@160
|
61 void msg_setElementToFrom(HvMessage *n, int indexN, const HvMessage *const m, int indexM);
|
chris@160
|
62
|
chris@160
|
63 /** Frees a message on the heap. Does nothing if argument is NULL. */
|
chris@160
|
64 void msg_free(HvMessage *m);
|
chris@160
|
65
|
chris@160
|
66 HvMessage *msg_init(HvMessage *m, hv_size_t numElements, hv_uint32_t timestamp);
|
chris@160
|
67
|
chris@160
|
68 HvMessage *msg_initWithFloat(HvMessage *m, hv_uint32_t timestamp, float f);
|
chris@160
|
69
|
chris@160
|
70 HvMessage *msg_initWithBang(HvMessage *m, hv_uint32_t timestamp);
|
chris@160
|
71
|
chris@160
|
72 HvMessage *msg_initWithSymbol(HvMessage *m, hv_uint32_t timestamp, char *s);
|
chris@160
|
73
|
chris@160
|
74 HvMessage *msg_initWithHash(HvMessage *m, hv_uint32_t timestamp, hv_uint32_t h);
|
chris@160
|
75
|
chris@160
|
76 HvMessage *msg_initV(HvMessage *const m, const hv_uint32_t timestamp, const char *format, ...);
|
chris@160
|
77
|
chris@160
|
78 static inline hv_uint32_t msg_getTimestamp(const HvMessage *m) {
|
chris@160
|
79 return m->timestamp;
|
chris@160
|
80 }
|
chris@160
|
81
|
chris@160
|
82 static inline void msg_setTimestamp(HvMessage *m, hv_uint32_t timestamp) {
|
chris@160
|
83 m->timestamp = timestamp;
|
chris@160
|
84 }
|
chris@160
|
85
|
chris@160
|
86 static inline int msg_getNumElements(const HvMessage *m) {
|
chris@160
|
87 return (int) m->numElements;
|
chris@160
|
88 }
|
chris@160
|
89
|
chris@160
|
90 /** Returns the number of bytes this message in memory. */
|
chris@160
|
91 static inline hv_size_t msg_getNumBytes(const HvMessage *m) {
|
chris@160
|
92 return m->numBytes;
|
chris@160
|
93 }
|
chris@160
|
94
|
chris@160
|
95 static inline ElementType msg_getType(const HvMessage *m, int index) {
|
chris@160
|
96 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
97 return (&(m->elem)+index)->type;
|
chris@160
|
98 }
|
chris@160
|
99
|
chris@160
|
100 static inline void msg_setBang(HvMessage *m, int index) {
|
chris@160
|
101 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
102 (&(m->elem)+index)->type = BANG;
|
chris@160
|
103 (&(m->elem)+index)->data.s = NULL;
|
chris@160
|
104 }
|
chris@160
|
105
|
chris@160
|
106 static inline bool msg_isBang(const HvMessage *m, int index) {
|
chris@160
|
107 return (index < msg_getNumElements(m)) ? (msg_getType(m,index) == BANG) : false;
|
chris@160
|
108 }
|
chris@160
|
109
|
chris@160
|
110 static inline void msg_setFloat(HvMessage *m, int index, float f) {
|
chris@160
|
111 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
112 (&(m->elem)+index)->type = FLOAT;
|
chris@160
|
113 (&(m->elem)+index)->data.f = f;
|
chris@160
|
114 }
|
chris@160
|
115
|
chris@160
|
116 static inline float msg_getFloat(const HvMessage *const m, int index) {
|
chris@160
|
117 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
118 return (&(m->elem)+index)->data.f;
|
chris@160
|
119 }
|
chris@160
|
120
|
chris@160
|
121 static inline bool msg_isFloat(const HvMessage *const m, int index) {
|
chris@160
|
122 return (index < msg_getNumElements(m)) ? (msg_getType(m,index) == FLOAT) : false;
|
chris@160
|
123 }
|
chris@160
|
124
|
chris@160
|
125 static inline void msg_setHash(HvMessage *m, int index, hv_uint32_t h) {
|
chris@160
|
126 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
127 (&(m->elem)+index)->type = HASH;
|
chris@160
|
128 (&(m->elem)+index)->data.h = h;
|
chris@160
|
129 }
|
chris@160
|
130
|
chris@160
|
131 static inline bool msg_isHash(const HvMessage *m, int index) {
|
chris@160
|
132 return (index < msg_getNumElements(m)) ? (msg_getType(m, index) == HASH) : false;
|
chris@160
|
133 }
|
chris@160
|
134
|
chris@160
|
135 /** Returns true if the element is a hash or symbol. False otherwise. */
|
chris@160
|
136 static inline bool msg_isHashLike(const HvMessage *m, int index) {
|
chris@160
|
137 return (index < msg_getNumElements(m)) ? ((msg_getType(m, index) == HASH) || (msg_getType(m, index) == SYMBOL)) : false;
|
chris@160
|
138 }
|
chris@160
|
139
|
chris@160
|
140 /** Returns a 32-bit hash of the given string. */
|
chris@160
|
141 hv_uint32_t msg_symbolToHash(const char *s);
|
chris@160
|
142
|
chris@160
|
143 /** Returns a 32-bit hash of the given element. */
|
chris@160
|
144 hv_uint32_t msg_getHash(const HvMessage *const m, int i);
|
chris@160
|
145
|
chris@160
|
146 static inline void msg_setSymbol(HvMessage *m, int index, char *s) {
|
chris@160
|
147 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
148 (&(m->elem)+index)->type = SYMBOL;
|
chris@160
|
149 (&(m->elem)+index)->data.s = s;
|
chris@160
|
150 }
|
chris@160
|
151
|
chris@160
|
152 static inline char *msg_getSymbol(const HvMessage *m, int index) {
|
chris@160
|
153 hv_assert(index < msg_getNumElements(m)); // invalid index
|
chris@160
|
154 return (&(m->elem)+index)->data.s;
|
chris@160
|
155 }
|
chris@160
|
156
|
chris@160
|
157 static inline bool msg_isSymbol(const HvMessage *m, int index) {
|
chris@160
|
158 return (index < msg_getNumElements(m)) ? (msg_getType(m, index) == SYMBOL) : false;
|
chris@160
|
159 }
|
chris@160
|
160
|
chris@160
|
161 bool msg_compareSymbol(const HvMessage *m, int i, const char *s);
|
chris@160
|
162
|
chris@160
|
163 /** Returns 1 if the element i_m of message m is equal to element i_n of message n. */
|
chris@160
|
164 bool msg_equalsElement(const HvMessage *m, int i_m, const HvMessage *n, int i_n);
|
chris@160
|
165
|
chris@160
|
166 bool msg_hasFormat(const HvMessage *m, const char *fmt);
|
chris@160
|
167
|
chris@160
|
168 /**
|
chris@160
|
169 * Create a string representation of the message. Suitable for use by the print object.
|
chris@160
|
170 * The resulting string must be freed by the caller.
|
chris@160
|
171 */
|
chris@160
|
172 char *msg_toString(const HvMessage *msg);
|
chris@160
|
173
|
chris@160
|
174 /**
|
chris@160
|
175 * Resolves any number of dollar arguments and generates a string based on the arguments.
|
chris@160
|
176 * @param m The message from which to take values
|
chris@160
|
177 * @param n The message to fill in
|
chris@160
|
178 * @param z The element index to resolve
|
chris@160
|
179 * @param buf The scratch (i.e. resolution) buffer
|
chris@160
|
180 * @param len The length of the scratch buffer
|
chris@160
|
181 * @param args A string of 'i' and 's' chars indicating the type of the arguments, either indicies or strings
|
chris@160
|
182 * @param varargs The components to resolve, either dollar indicies or strings.
|
chris@160
|
183 * If the index is negative, the graph id is used
|
chris@160
|
184 * @return true if the resolution buffer is needed, false otherwise.
|
chris@160
|
185 */
|
chris@160
|
186 // bool msg_resolveDollarArguments(HvMessage *m, HvMessage *n, int z, char *buf, hv_size_t len, const char *args, ...);
|
chris@160
|
187
|
chris@160
|
188 #endif // _HEAVY_MESSAGE_H_
|