2009-11-13 21:28:09 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2009, Sangoma Technologies
|
|
|
|
* Moises Silva <moy@sangoma.com>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
*
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* * Neither the name of the original author; nor the names of any contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
|
|
|
|
* OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
|
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
2010-04-19 15:39:03 +00:00
|
|
|
#include "private/ftdm_core.h"
|
2010-01-15 19:22:49 +00:00
|
|
|
|
|
|
|
static ftdm_status_t ftdm_std_queue_create(ftdm_queue_t **outqueue, ftdm_size_t capacity);
|
|
|
|
static ftdm_status_t ftdm_std_queue_enqueue(ftdm_queue_t *queue, void *obj);
|
|
|
|
static void *ftdm_std_queue_dequeue(ftdm_queue_t *queue);
|
|
|
|
static ftdm_status_t ftdm_std_queue_wait(ftdm_queue_t *queue, int ms);
|
2010-03-12 18:27:24 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_get_interrupt(ftdm_queue_t *queue, ftdm_interrupt_t **interrupt);
|
2010-01-15 19:22:49 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_destroy(ftdm_queue_t **inqueue);
|
|
|
|
|
|
|
|
struct ftdm_queue {
|
|
|
|
ftdm_mutex_t *mutex;
|
2010-03-12 18:27:24 +00:00
|
|
|
ftdm_interrupt_t *interrupt;
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_size_t capacity;
|
|
|
|
ftdm_size_t size;
|
2009-11-14 02:04:38 +00:00
|
|
|
unsigned rindex;
|
|
|
|
unsigned windex;
|
|
|
|
void **elements;
|
2009-11-30 16:52:40 +00:00
|
|
|
};
|
2009-11-16 15:21:08 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE_DATA ftdm_queue_handler_t g_ftdm_queue_handler =
|
2009-11-16 15:21:08 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
/*.create = */ ftdm_std_queue_create,
|
|
|
|
/*.enqueue = */ ftdm_std_queue_enqueue,
|
|
|
|
/*.dequeue = */ ftdm_std_queue_dequeue,
|
|
|
|
/*.wait = */ ftdm_std_queue_wait,
|
2010-03-12 18:27:24 +00:00
|
|
|
/*.get_interrupt = */ ftdm_std_queue_get_interrupt,
|
2010-01-15 19:22:49 +00:00
|
|
|
/*.destroy = */ ftdm_std_queue_destroy
|
2009-11-16 15:21:08 +00:00
|
|
|
};
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_global_set_queue_handler(ftdm_queue_handler_t *handler)
|
2009-11-16 15:21:08 +00:00
|
|
|
{
|
|
|
|
if (!handler ||
|
|
|
|
!handler->create ||
|
|
|
|
!handler->enqueue ||
|
|
|
|
!handler->dequeue ||
|
|
|
|
!handler->wait ||
|
2010-03-12 18:27:24 +00:00
|
|
|
!handler->get_interrupt ||
|
2009-11-16 15:21:08 +00:00
|
|
|
!handler->destroy) {
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2009-11-16 15:21:08 +00:00
|
|
|
}
|
2010-01-15 19:22:49 +00:00
|
|
|
memcpy(&g_ftdm_queue_handler, handler, sizeof(*handler));
|
|
|
|
return FTDM_SUCCESS;
|
2009-11-16 15:21:08 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_create(ftdm_queue_t **outqueue, ftdm_size_t capacity)
|
2009-11-13 21:28:09 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_queue_t *queue = NULL;
|
|
|
|
ftdm_assert_return(outqueue, FTDM_FAIL, "Queue double pointer is null\n");
|
|
|
|
ftdm_assert_return(capacity > 0, FTDM_FAIL, "Queue capacity is not bigger than 0\n");
|
2009-11-14 02:04:38 +00:00
|
|
|
|
|
|
|
*outqueue = NULL;
|
2010-01-15 19:22:49 +00:00
|
|
|
queue = ftdm_calloc(1, sizeof(*queue));
|
2009-11-14 02:04:38 +00:00
|
|
|
if (!queue) {
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2009-11-14 02:04:38 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
queue->elements = ftdm_calloc(1, (sizeof(void*)*capacity));
|
2009-11-14 02:04:38 +00:00
|
|
|
if (!queue->elements) {
|
|
|
|
goto failed;
|
|
|
|
}
|
2009-11-21 04:45:22 +00:00
|
|
|
queue->capacity = capacity;
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
if (ftdm_mutex_create(&queue->mutex) != FTDM_SUCCESS) {
|
2009-11-14 02:04:38 +00:00
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
|
2011-12-26 01:49:10 +00:00
|
|
|
if (ftdm_interrupt_create(&queue->interrupt, FTDM_INVALID_SOCKET, FTDM_NO_FLAGS) != FTDM_SUCCESS) {
|
2009-11-14 02:04:38 +00:00
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
|
|
|
|
*outqueue = queue;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2009-11-14 02:04:38 +00:00
|
|
|
|
|
|
|
failed:
|
|
|
|
if (queue) {
|
2010-03-12 18:27:24 +00:00
|
|
|
if (queue->interrupt) {
|
|
|
|
ftdm_interrupt_destroy(&queue->interrupt);
|
2009-11-14 02:04:38 +00:00
|
|
|
}
|
|
|
|
if (queue->mutex) {
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_destroy(&queue->mutex);
|
2009-11-14 02:04:38 +00:00
|
|
|
}
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_safe_free(queue->elements);
|
|
|
|
ftdm_safe_free(queue);
|
2009-11-14 02:04:38 +00:00
|
|
|
}
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2009-11-13 21:28:09 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_enqueue(ftdm_queue_t *queue, void *obj)
|
2009-11-13 21:28:09 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_status_t status = FTDM_FAIL;
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_assert_return(queue != NULL, FTDM_FAIL, "Queue is null!");
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_lock(queue->mutex);
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2009-11-21 04:45:22 +00:00
|
|
|
if (queue->windex == queue->capacity) {
|
2009-11-14 02:04:38 +00:00
|
|
|
/* try to see if we can wrap around */
|
|
|
|
queue->windex = 0;
|
|
|
|
}
|
|
|
|
|
2009-11-21 04:45:22 +00:00
|
|
|
if (queue->size != 0 && queue->windex == queue->rindex) {
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_log(FTDM_LOG_ERROR, "Failed to enqueue obj %p in queue %p, no more room! windex == rindex == %d!\n", obj, queue, queue->windex);
|
2009-11-14 02:04:38 +00:00
|
|
|
goto done;
|
|
|
|
}
|
2009-11-21 04:45:22 +00:00
|
|
|
|
2009-11-14 02:04:38 +00:00
|
|
|
queue->elements[queue->windex++] = obj;
|
2009-11-21 04:45:22 +00:00
|
|
|
queue->size++;
|
2010-01-15 19:22:49 +00:00
|
|
|
status = FTDM_SUCCESS;
|
2009-11-21 04:45:22 +00:00
|
|
|
|
2009-11-14 02:04:38 +00:00
|
|
|
/* wake up queue reader */
|
2010-03-12 18:27:24 +00:00
|
|
|
ftdm_interrupt_signal(queue->interrupt);
|
2009-11-14 02:04:38 +00:00
|
|
|
|
|
|
|
done:
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_unlock(queue->mutex);
|
2009-11-14 02:04:38 +00:00
|
|
|
|
|
|
|
return status;
|
2009-11-13 21:28:09 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
static void *ftdm_std_queue_dequeue(ftdm_queue_t *queue)
|
2009-11-13 21:28:09 +00:00
|
|
|
{
|
2009-11-14 02:04:38 +00:00
|
|
|
void *obj = NULL;
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_assert_return(queue != NULL, NULL, "Queue is null!");
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_lock(queue->mutex);
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2009-11-21 04:45:22 +00:00
|
|
|
if (queue->size == 0) {
|
2009-11-14 02:04:38 +00:00
|
|
|
goto done;
|
|
|
|
}
|
2009-11-20 22:57:24 +00:00
|
|
|
|
2009-11-14 02:04:38 +00:00
|
|
|
obj = queue->elements[queue->rindex];
|
2009-11-20 22:57:24 +00:00
|
|
|
queue->elements[queue->rindex++] = NULL;
|
2009-11-21 04:45:22 +00:00
|
|
|
queue->size--;
|
2009-12-18 20:53:17 +00:00
|
|
|
if (queue->rindex == queue->capacity) {
|
2009-11-14 02:04:38 +00:00
|
|
|
queue->rindex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
2009-11-21 04:45:22 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_unlock(queue->mutex);
|
2009-11-21 04:45:22 +00:00
|
|
|
|
2009-11-14 02:04:38 +00:00
|
|
|
return obj;
|
2009-11-13 21:28:09 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_wait(ftdm_queue_t *queue, int ms)
|
2009-11-13 21:28:09 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_status_t ret;
|
|
|
|
ftdm_assert_return(queue != NULL, FTDM_FAIL, "Queue is null!");
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_lock(queue->mutex);
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2009-11-21 04:45:22 +00:00
|
|
|
/* if there is elements in the queue, no need to wait */
|
|
|
|
if (queue->size != 0) {
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_unlock(queue->mutex);
|
|
|
|
return FTDM_SUCCESS;
|
2009-11-14 02:04:38 +00:00
|
|
|
}
|
|
|
|
|
2009-11-21 04:45:22 +00:00
|
|
|
/* no elements on the queue, wait for someone to write an element */
|
2010-03-12 18:27:24 +00:00
|
|
|
ret = ftdm_interrupt_wait(queue->interrupt, ms);
|
2009-11-21 04:45:22 +00:00
|
|
|
|
|
|
|
/* got an element or timeout, bail out */
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_unlock(queue->mutex);
|
2009-11-21 04:45:22 +00:00
|
|
|
|
2009-11-20 22:57:24 +00:00
|
|
|
return ret;
|
2009-11-13 21:28:09 +00:00
|
|
|
}
|
|
|
|
|
2010-03-12 18:27:24 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_get_interrupt(ftdm_queue_t *queue, ftdm_interrupt_t **interrupt)
|
|
|
|
{
|
|
|
|
ftdm_assert_return(queue != NULL, FTDM_FAIL, "Queue is null!\n");
|
|
|
|
ftdm_assert_return(interrupt != NULL, FTDM_FAIL, "Queue is null!\n");
|
|
|
|
*interrupt = queue->interrupt;
|
|
|
|
return FTDM_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
static ftdm_status_t ftdm_std_queue_destroy(ftdm_queue_t **inqueue)
|
2009-11-13 21:28:09 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_queue_t *queue = NULL;
|
2010-03-12 18:27:24 +00:00
|
|
|
ftdm_assert_return(inqueue != NULL, FTDM_FAIL, "Queue is null!\n");
|
|
|
|
ftdm_assert_return(*inqueue != NULL, FTDM_FAIL, "Queue is null!\n");
|
2009-11-13 21:28:09 +00:00
|
|
|
|
2009-11-14 02:04:38 +00:00
|
|
|
queue = *inqueue;
|
2010-03-12 18:27:24 +00:00
|
|
|
ftdm_interrupt_destroy(&queue->interrupt);
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_mutex_destroy(&queue->mutex);
|
|
|
|
ftdm_safe_free(queue->elements);
|
|
|
|
ftdm_safe_free(queue);
|
2009-11-14 02:04:38 +00:00
|
|
|
*inqueue = NULL;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2009-11-13 21:28:09 +00:00
|
|
|
}
|
2009-11-14 02:04:38 +00:00
|
|
|
|
2009-11-13 21:28:09 +00:00
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
|
|
|
* indent-tabs-mode:t
|
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
2013-06-25 16:50:17 +00:00
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
|
2009-11-13 21:28:09 +00:00
|
|
|
*/
|