chris@163: /** chris@163: * Copyright (c) 2014, 2015, Enzien Audio Ltd. chris@163: * chris@163: * Permission to use, copy, modify, and/or distribute this software for any chris@163: * purpose with or without fee is hereby granted, provided that the above chris@163: * copyright notice and this permission notice appear in all copies. chris@163: * chris@163: * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH chris@163: * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY chris@163: * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, chris@163: * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM chris@163: * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR chris@163: * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR chris@163: * PERFORMANCE OF THIS SOFTWARE. chris@163: */ chris@163: chris@163: #include "HvMessage.h" chris@163: chris@163: HvMessage *msg_init(HvMessage *m, hv_size_t numElements, hv_uint32_t timestamp) { chris@163: m->timestamp = timestamp; chris@163: m->numElements = (hv_uint16_t) numElements; chris@163: m->numBytes = (hv_uint16_t) msg_getByteSize(numElements); chris@163: return m; chris@163: } chris@163: chris@163: HvMessage *msg_initWithFloat(HvMessage *m, hv_uint32_t timestamp, float f) { chris@163: m->timestamp = timestamp; chris@163: m->numElements = 1; chris@163: m->numBytes = sizeof(HvMessage); chris@163: msg_setFloat(m, 0, f); chris@163: return m; chris@163: } chris@163: chris@163: HvMessage *msg_initWithBang(HvMessage *m, hv_uint32_t timestamp) { chris@163: m->timestamp = timestamp; chris@163: m->numElements = 1; chris@163: m->numBytes = sizeof(HvMessage); chris@163: msg_setBang(m, 0); chris@163: return m; chris@163: } chris@163: chris@163: HvMessage *msg_initWithSymbol(HvMessage *m, hv_uint32_t timestamp, char *s) { chris@163: m->timestamp = timestamp; chris@163: m->numElements = 1; chris@163: m->numBytes = sizeof(HvMessage); chris@163: msg_setSymbol(m, 0, s); chris@163: return m; chris@163: } chris@163: chris@163: HvMessage *msg_initWithHash(HvMessage *m, hv_uint32_t timestamp, hv_uint32_t h) { chris@163: m->timestamp = timestamp; chris@163: m->numElements = 1; chris@163: m->numBytes = sizeof(HvMessage); chris@163: msg_setHash(m, 0, h); chris@163: return m; chris@163: } chris@163: chris@163: HvMessage *msg_initV(HvMessage *const m, const hv_uint32_t timestamp, const char *format, ...) { chris@163: va_list ap; chris@163: va_start(ap, format); chris@163: chris@163: const int numElem = (int) hv_strlen(format); chris@163: msg_init(m, numElem, timestamp); chris@163: for (int i = 0; i < numElem; i++) { chris@163: switch (format[i]) { chris@163: case 'b': msg_setBang(m,i); break; chris@163: case 'f': msg_setFloat(m, i, (float) va_arg(ap, double)); break; chris@163: case 's': msg_setSymbol(m, i, (char *) va_arg(ap, char *)); break; chris@163: case 'h': // hash not supported chris@163: default: break; chris@163: } chris@163: } chris@163: va_end(ap); chris@163: chris@163: return m; chris@163: } chris@163: chris@163: hv_size_t msg_getNumHeapBytes(const HvMessage *m) { chris@163: // get the size of all symbol elements chris@163: hv_size_t rsizeofsym = 0; chris@163: for (int i = 0; i < msg_getNumElements(m); ++i) { chris@163: if (msg_isSymbol(m,i)) { chris@163: rsizeofsym += (hv_size_t) hv_strlen(msg_getSymbol(m,i)) + 1; // +1 to allow for trailing '\0' chris@163: } chris@163: } chris@163: chris@163: // the total byte size on the heap chris@163: return (msg_getByteSize(msg_getNumElements(m)) + rsizeofsym); chris@163: } chris@163: chris@163: void msg_copyToBuffer(const HvMessage *m, char *buffer, hv_size_t len) { chris@163: HvMessage *r = (HvMessage *) buffer; chris@163: chris@163: // assert that the message is not already larger than the length of the buffer chris@163: hv_assert(msg_getNumBytes(m) <= len); chris@163: chris@163: // copy the basic message to the buffer chris@163: hv_memcpy(r, m, msg_getNumBytes(m)); chris@163: chris@163: hv_size_t len_r = msg_getNumBytes(m); chris@163: chris@163: char *p = buffer + msg_getByteSize(msg_getNumElements(m)); // points to the end of the base message chris@163: for (int i = 0; i < msg_getNumElements(m); ++i) { chris@163: if (msg_isSymbol(m,i)) { chris@163: const hv_size_t symLen = (hv_size_t) hv_strlen(msg_getSymbol(m,i)) + 1; // include the trailing null char chris@163: hv_assert(len_r + symLen <= len); // stay safe! chris@163: hv_strncpy(p, msg_getSymbol(m,i), symLen); chris@163: msg_setSymbol(r, i, p); chris@163: p += symLen; chris@163: len_r += symLen; chris@163: } chris@163: } chris@163: chris@163: r->numBytes = (hv_uint16_t) len_r; // update the message size in memory chris@163: } chris@163: chris@163: // the message is serialised such that all symbol elements are placed in order at the end of the buffer chris@163: HvMessage *msg_copy(const HvMessage *m) { chris@163: const hv_size_t heapSize = msg_getNumHeapBytes(m); chris@163: char *r = (char *) hv_malloc(heapSize); chris@163: msg_copyToBuffer(m, r, heapSize); chris@163: return (HvMessage *) r; chris@163: } chris@163: chris@163: void msg_free(HvMessage *m) { chris@163: hv_free(m); // because heap messages are serialised in memory, a simple call to free releases the message chris@163: } chris@163: chris@163: bool msg_hasFormat(const HvMessage *m, const char *fmt) { chris@163: if (fmt == NULL) return false; chris@163: if (msg_getNumElements(m) != hv_strlen(fmt)) return false; chris@163: for (int i = 0; i < msg_getNumElements(m); i++) { chris@163: switch (fmt[i]) { chris@163: case 'b': if (!msg_isBang(m, i)) return false; break; chris@163: case 'f': if (!msg_isFloat(m, i)) return false; break; chris@163: case 's': if (!msg_isSymbol(m, i)) return false; break; chris@163: case 'h': if (!msg_isHash(m, i)) return false; break; chris@163: default: return false; chris@163: } chris@163: } chris@163: return true; chris@163: } chris@163: chris@163: bool msg_compareSymbol(const HvMessage *m, int i, const char *s) { chris@163: switch (msg_getType(m,i)) { chris@163: case SYMBOL: return !hv_strcmp(msg_getSymbol(m, i), s); chris@163: case HASH: return (msg_getHash(m,i) == msg_symbolToHash(s)); chris@163: default: return false; chris@163: } chris@163: } chris@163: chris@163: bool msg_equalsElement(const HvMessage *m, int i_m, const HvMessage *n, int i_n) { chris@163: if (i_m < msg_getNumElements(m) && i_n < msg_getNumElements(n)) { chris@163: if (msg_getType(m, i_m) == msg_getType(n, i_n)) { chris@163: switch (msg_getType(m, i_m)) { chris@163: case BANG: return true; chris@163: case FLOAT: return (msg_getFloat(m, i_m) == msg_getFloat(n, i_n)); chris@163: case SYMBOL: return msg_compareSymbol(m, i_m, msg_getSymbol(n, i_n)); chris@163: case HASH: return msg_getHash(m,i_m) == msg_getHash(n,i_n); chris@163: default: break; chris@163: } chris@163: } chris@163: } chris@163: return false; chris@163: } chris@163: chris@163: void msg_setElementToFrom(HvMessage *n, int i_n, const HvMessage *const m, int i_m) { chris@163: switch (msg_getType(m, i_m)) { chris@163: case BANG: msg_setBang(n, i_n); break; chris@163: case FLOAT: msg_setFloat(n, i_n, msg_getFloat(m, i_m)); break; chris@163: case SYMBOL: msg_setSymbol(n, i_n, msg_getSymbol(m, i_m)); break; chris@163: case HASH: msg_setHash(n, i_n, msg_getHash(m, i_m)); chris@163: default: break; chris@163: } chris@163: } chris@163: chris@163: hv_uint32_t msg_symbolToHash(const char *s) { chris@163: // this hash is based MurmurHash2 chris@163: // http://en.wikipedia.org/wiki/MurmurHash chris@163: // https://sites.google.com/site/murmurhash/ chris@163: static const unsigned int n = 0x5bd1e995; chris@163: static const int r = 24; chris@163: chris@163: int len = (int) hv_strlen(s); chris@163: hv_uint32_t x = (hv_uint32_t) (len); // seed (0) ^ len chris@163: chris@163: while (len >= 4) { chris@163: hv_uint32_t k = *((hv_uint32_t *)s); chris@163: k *= n; chris@163: k ^= k >> r; chris@163: k *= n; chris@163: x *= n; chris@163: x ^= k; chris@163: s += 4; len -= 4; chris@163: } chris@163: chris@163: switch(len) { chris@163: case 3: x ^= s[2] << 16; chris@163: case 2: x ^= s[1] << 8; chris@163: case 1: x ^= s[0]; x *= n; chris@163: default: break; chris@163: } chris@163: chris@163: x ^= x >> 13; chris@163: x *= n; chris@163: x ^= x >> 15; chris@163: chris@163: return x; chris@163: } chris@163: chris@163: hv_uint32_t msg_getHash(const HvMessage *const m, int i) { chris@163: hv_assert(i < msg_getNumElements(m)); // invalid index chris@163: switch (msg_getType(m,i)) { chris@163: case BANG: return 0xFFFFFFFF; chris@163: case FLOAT: { chris@163: float f = msg_getFloat(m,i); chris@163: return *((hv_uint32_t *) &f); chris@163: } chris@163: case SYMBOL: return msg_symbolToHash(msg_getSymbol(m,i)); chris@163: case HASH: return (&(m->elem)+i)->data.h; chris@163: default: return 0; chris@163: } chris@163: } chris@163: chris@163: char *msg_toString(const HvMessage *m) { chris@163: hv_assert(msg_getNumElements(m) > 0); chris@163: int *len = (int *) hv_alloca(msg_getNumElements(m)*sizeof(int)); chris@163: int size = 0; // the total length of our final buffer chris@163: chris@163: // loop through every element in our list of atoms chris@163: // first loop figures out how long our buffer should be chris@163: for (int i = 0; i < msg_getNumElements(m); i++) { chris@163: // length of our string is each atom plus a space, or \0 on the end chris@163: switch (msg_getType(m, i)) { chris@163: case BANG: len[i] = hv_snprintf(NULL, 0, "%s", "bang") + 1; break; chris@163: case FLOAT: len[i] = hv_snprintf(NULL, 0, "%g", msg_getFloat(m, i)) + 1; break; chris@163: case SYMBOL: len[i] = hv_snprintf(NULL, 0, "%s", msg_getSymbol(m, i)) + 1; break; chris@163: case HASH: len[i] = hv_snprintf(NULL, 0, "0x%X", msg_getHash(m, i)) + 1; break; chris@163: default: break; chris@163: } chris@163: size += len[i]; chris@163: } chris@163: chris@163: hv_assert(size > 0); chris@163: chris@163: // now we do the piecewise concatenation into our final string chris@163: // the final buffer we will pass back after concatenating all strings - user should free it chris@163: char *finalString = (char *) hv_malloc(size*sizeof(char)); chris@163: int pos = 0; chris@163: for (int i = 0; i < msg_getNumElements(m); i++) { chris@163: // put a string representation of each atom into the final string chris@163: switch (msg_getType(m, i)) { chris@163: case BANG: hv_snprintf(finalString+pos, len[i], "%s", "bang"); break; chris@163: case FLOAT: hv_snprintf(finalString+pos, len[i], "%g", msg_getFloat(m, i)); break; chris@163: case SYMBOL: hv_snprintf(finalString+pos, len[i], "%s", msg_getSymbol(m, i)); break; chris@163: case HASH: hv_snprintf(finalString+pos, len[i], "0x%X", msg_getHash(m, i)); break; chris@163: default: break; chris@163: } chris@163: pos += len[i]; chris@163: finalString[pos-1] = 32; // ASCII space chris@163: } chris@163: finalString[size-1] = '\0'; // ensure that the string is null terminated chris@163: return finalString; chris@163: } chris@163: chris@163: /* chris@163: * TODO(mhroth): unnecessary for now chris@163: bool msg_resolveDollarArguments(HvMessage *m, HvMessage *n, int z, char *buf, hv_size_t len, const char *args, ...) { chris@163: va_list ap; chris@163: va_start(ap, args); chris@163: chris@163: hv_memset(buf, 0, len); // clear the buffer chris@163: hv_size_t j = 0; // position in buffer chris@163: const hv_size_t numArgs = hv_strlen(args); // the number of arguments chris@163: chris@163: // if there is only one argument then the result has the chance of being a number, otherwise no chris@163: bool isNumber = (numArgs == 1); chris@163: chris@163: for (hv_size_t i = 0; i < numArgs; ++i) { chris@163: switch (args[i]) { chris@163: case 'i': { // a message index chris@163: const int index = (int) va_arg(ap, int); chris@163: if (index < 0) { chris@163: // $0 always resolve to "0" chris@163: const hv_size_t x = 1; chris@163: if (x < len-j) { // always < in order to allow for trailing \0 chris@163: j += snprintf(buf+j, len-j, "0"); chris@163: } chris@163: } else { chris@163: switch (msg_getType(m, index)) { chris@163: default: chris@163: case BANG: break; // this case should never happen chris@163: case FLOAT: { chris@163: const hv_size_t x = snprintf(NULL, 0, "%g", msg_getFloat(m,index)); chris@163: if (x < len-j) { // ensure that the buffer is big enough chris@163: j += snprintf(buf+j, len-j, "%g", msg_getFloat(m,index)); chris@163: } chris@163: break; chris@163: } chris@163: case SYMBOL: { chris@163: const hv_size_t x = snprintf(NULL, 0, "%s", msg_getSymbol(m,index)); chris@163: if (x < len-j) { chris@163: j += snprintf(buf+j, len-j, "%s", msg_getSymbol(m,index)); chris@163: isNumber = false; chris@163: } chris@163: break; chris@163: } chris@163: } chris@163: } chris@163: break; chris@163: } chris@163: case 's': { // a string chris@163: const char *s = (char *) va_arg(ap, char *); chris@163: const hv_size_t x = snprintf(NULL, 0, "%s", s); chris@163: if (x <= len-j) { chris@163: j += snprintf(buf+j, len-j, "%s", s); chris@163: isNumber = false; chris@163: } chris@163: break; chris@163: } chris@163: default: break; chris@163: } chris@163: } chris@163: chris@163: if (isNumber) { chris@163: msg_setFloat(n,z,(float) atof(buf)); chris@163: } else { chris@163: msg_setSymbol(n,z,buf); chris@163: } chris@163: chris@163: va_end(ap); chris@163: chris@163: return !isNumber; chris@163: } chris@163: */