chris@160: /** chris@160: * Copyright (c) 2014, 2015, Enzien Audio Ltd. chris@160: * chris@160: * Permission to use, copy, modify, and/or distribute this software for any chris@160: * purpose with or without fee is hereby granted, provided that the above chris@160: * copyright notice and this permission notice appear in all copies. chris@160: * chris@160: * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH chris@160: * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY chris@160: * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, chris@160: * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM chris@160: * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR chris@160: * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR chris@160: * PERFORMANCE OF THIS SOFTWARE. chris@160: */ chris@160: chris@160: #include "MessageQueue.h" chris@160: #include "Utils.h" chris@160: chris@160: hv_size_t mq_init(MessageQueue *q) { chris@160: q->head = NULL; chris@160: q->tail = NULL; chris@160: q->pool = NULL; chris@160: return mp_init(&q->mp, 1); chris@160: } chris@160: chris@160: void mq_initWithPoolSize(MessageQueue *q, hv_size_t poolSizeKB) { chris@160: q->head = NULL; chris@160: q->tail = NULL; chris@160: q->pool = NULL; chris@160: mp_init(&q->mp, poolSizeKB); chris@160: } chris@160: chris@160: void mq_free(MessageQueue *q) { chris@160: mq_clear(q); chris@160: while (q->pool != NULL) { chris@160: MessageNode *n = q->pool; chris@160: q->pool = q->pool->next; chris@160: hv_free(n); chris@160: } chris@160: mp_free(&q->mp); chris@160: } chris@160: chris@160: static MessageNode *mq_getOrCreateNodeFromPool(MessageQueue *q) { chris@160: if (q->pool == NULL) { chris@160: // if necessary, create a new empty node chris@160: q->pool = (MessageNode *) hv_malloc(sizeof(MessageNode)); chris@160: q->pool->next = NULL; chris@160: } chris@160: MessageNode *node = q->pool; chris@160: q->pool = q->pool->next; chris@160: return node; chris@160: } chris@160: chris@160: int mq_size(MessageQueue *q) { chris@160: int size = 0; chris@160: MessageNode *n = q->head; chris@160: while (n != NULL) { chris@160: ++size; chris@160: n = n->next; chris@160: } chris@160: return size; chris@160: } chris@160: chris@160: HvMessage *mq_addMessage(MessageQueue *q, const HvMessage *m, int let, chris@160: void (*sendMessage)(struct HvBase *, int, const HvMessage *)) { chris@160: MessageNode *node = mq_getOrCreateNodeFromPool(q); chris@160: node->m = mp_addMessage(&q->mp, m); chris@160: node->let = let; chris@160: node->sendMessage = sendMessage; chris@160: node->prev = NULL; chris@160: node->next = NULL; chris@160: chris@160: if (q->tail != NULL) { chris@160: // the list already contains elements chris@160: q->tail->next = node; chris@160: node->prev = q->tail; chris@160: q->tail = node; chris@160: } else { chris@160: // the list is empty chris@160: node->prev = NULL; chris@160: q->head = node; chris@160: q->tail = node; chris@160: } chris@160: return mq_node_getMessage(node); chris@160: } chris@160: chris@160: HvMessage *mq_addMessageByTimestamp(MessageQueue *q, HvMessage *m, int let, chris@160: void (*sendMessage)(struct HvBase *, int, const HvMessage *)) { chris@160: if (mq_hasMessage(q)) { chris@160: MessageNode *n = mq_getOrCreateNodeFromPool(q); chris@160: n->m = mp_addMessage(&q->mp, m); chris@160: n->let = let; chris@160: n->sendMessage = sendMessage; chris@160: chris@160: if (msg_getTimestamp(m) < msg_getTimestamp(q->head->m)) { chris@160: // the message occurs before the current head chris@160: n->next = q->head; chris@160: q->head->prev = n; chris@160: n->prev = NULL; chris@160: q->head = n; chris@160: } else if (msg_getTimestamp(m) >= msg_getTimestamp(q->tail->m)) { chris@160: // the message occurs after the current tail chris@160: n->next = NULL; chris@160: n->prev = q->tail; chris@160: q->tail->next = n; chris@160: q->tail = n; chris@160: } else { chris@160: // the message occurs somewhere between the head and tail chris@160: MessageNode *node = q->head; chris@160: while (node != NULL) { chris@160: if (m->timestamp < msg_getTimestamp(node->next->m)) { chris@160: MessageNode *r = node->next; chris@160: node->next = n; chris@160: n->next = r; chris@160: n->prev = node; chris@160: r->prev = n; chris@160: break; chris@160: } chris@160: node = node->next; chris@160: } chris@160: } chris@160: return n->m; chris@160: } else { chris@160: // add a message to the head chris@160: return mq_addMessage(q, m, let, sendMessage); chris@160: } chris@160: } chris@160: chris@160: void mq_pop(MessageQueue *q) { chris@160: if (mq_hasMessage(q)) { chris@160: MessageNode *n = q->head; chris@160: chris@160: mp_freeMessage(&q->mp, n->m); chris@160: n->m = NULL; chris@160: chris@160: n->let = 0; chris@160: n->sendMessage = NULL; chris@160: chris@160: q->head = n->next; chris@160: if (q->head == NULL) { chris@160: q->tail = NULL; chris@160: } else { chris@160: q->head->prev = NULL; chris@160: } chris@160: n->next = q->pool; chris@160: n->prev = NULL; chris@160: q->pool = n; chris@160: } chris@160: } chris@160: chris@160: void mq_removeMessage(MessageQueue *q, HvMessage *m, void (*sendMessage)(struct HvBase *, int, const HvMessage *)) { chris@160: if (mq_hasMessage(q)) { chris@160: if (mq_node_getMessage(q->head) == m) { // msg in head node chris@160: // only remove the message if sendMessage is the same as the stored one, chris@160: // if the sendMessage argument is NULL, it is not checked and will remove any matching message pointer chris@160: if (sendMessage == NULL || q->head->sendMessage == sendMessage) { chris@160: mq_pop(q); chris@160: } chris@160: } else { chris@160: MessageNode *prevNode = q->head; chris@160: MessageNode *currNode = q->head->next; chris@160: while ((currNode != NULL) && (currNode->m != m)) { chris@160: prevNode = currNode; chris@160: currNode = currNode->next; chris@160: } chris@160: if (currNode != NULL) { chris@160: if (sendMessage == NULL || currNode->sendMessage == sendMessage) { chris@160: mp_freeMessage(&q->mp, m); chris@160: currNode->m = NULL; chris@160: currNode->let = 0; chris@160: currNode->sendMessage = NULL; chris@160: if (currNode == q->tail) { // msg in tail node chris@160: prevNode->next = NULL; chris@160: q->tail = prevNode; chris@160: } else { // msg in middle node chris@160: prevNode->next = currNode->next; chris@160: currNode->next->prev = prevNode; chris@160: } chris@160: currNode->next = (q->pool == NULL) ? NULL : q->pool; chris@160: currNode->prev = NULL; chris@160: q->pool = currNode; chris@160: } chris@160: } chris@160: } chris@160: } chris@160: } chris@160: chris@160: void mq_clear(MessageQueue *q) { chris@160: while (mq_hasMessage(q)) { chris@160: mq_pop(q); chris@160: } chris@160: } chris@160: chris@160: void mq_clearAfter(MessageQueue *q, const double timestamp) { chris@160: MessageNode *n = q->tail; chris@160: while (n != NULL && timestamp <= msg_getTimestamp(n->m)) { chris@160: // free the node's message chris@160: mp_freeMessage(&q->mp, n->m); chris@160: n->m = NULL; chris@160: n->let = 0; chris@160: n->sendMessage = NULL; chris@160: chris@160: // the tail points at the previous node chris@160: q->tail = n->prev; chris@160: chris@160: // put the node back in the pool chris@160: n->next = q->pool; chris@160: n->prev = NULL; chris@160: if (q->pool != NULL) q->pool->prev = n; chris@160: q->pool = n; chris@160: chris@160: // update the tail node chris@160: n = q->tail; chris@160: } chris@160: chris@160: if (q->tail == NULL) q->head = NULL; chris@160: }