123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193 |
- /*
- * Copyright (C) 2005 iptelorg GmbH
- *
- * This file is part of ser, a free SIP server.
- *
- * ser is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version
- *
- * For a license to use the ser software under conditions
- * other than those described here, or to purchase support for this
- * software, please contact iptel.org by e-mail at the following addresses:
- * [email protected]
- *
- * ser is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- */
- #ifndef __MSG_QUEUE_H
- #define __MSG_QUEUE_H
- #include <cds/sync.h>
- #include <cds/ref_cntr.h>
- #ifdef __cplusplus
- extern "C" {
- #endif
- /** \ingroup cds
- * @{
- *
- * \defgroup cds_msg_queue Message Queue
- *
- * Message queue is a structure useful for sending data between processes.
- * It can be synchronized via its own mutex or the synchronization can
- * be left on caller. It can use reference counter which is useful
- * when accessing dynamicaly allocated queue destroyed by its last user.
- *
- * \todo To meaningfully use reference counters it is needed to add
- * function for adding new reference to message queue.
- *
- * \todo Introduce message types because it is often needed.
- * @{
- * */
- typedef void (*destroy_function_f)(void *);
- /** Structure holding message which can be put
- * into message queue.
- *
- * There is a need to allow destroing the message without knowing its
- * internals (destroying message queue with non-processed messages) and thus
- * the destroy_function able to fully destroy whole data hold by message must
- * be given. It is mostly needed to choose the function manually only for
- * complex data with pointers which content need to be freed too.
- * For simple structures it is set automaticaly during the message creation.
- */
- typedef struct _mq_message_t {
- /** pointer to data hold by the message */
- void *data;
- /** length of data hold by message */
- int data_len;
- /** pointer to next message in the queue - is used only when
- * message is in the queue */
- struct _mq_message_t *next;
- /** pointer to destroy function */
- destroy_function_f destroy_function;
- enum {
- message_allocated_with_data, /**< data are allocated together with message structure */
- message_holding_data_ptr /**< message holds only pointer to data */
- } allocation_style; /**< member describing the manner of data storage */
- char data_buf[1]; /**< data buffer used when data are allocated together with message */
- } mq_message_t;
- /** Message queue flag meaning that internal queue mutex is used for
- * synchronization. It is set during message queue initialization via \ref
- * msg_queue_init or \ref msg_queue_init_ex. */
- #define MQ_USE_MUTEX 1
- /** Message queue flag meaning that reference counters are used.
- * To set this flag is needed to call \ref msg_queue_init_ref_cnt with
- * non-NULL group parameter. */
- #define MQ_USE_REF_CNTR 2
- /** Message queue structure.
- * Never access its members directly (they may change), always
- * use interface functions!
- */
- typedef struct msg_queue {
- /** Reference counter. Need not to be used.
- * If you want to use reference counter to message queue you have
- * to call \ref msg_queue_init_ref_cnt function with not-NULL reference
- * counter group.*/
- reference_counter_data_t ref;
- /** Pointer to the first message in the queue. Messages are hold in
- * one way linked list, each message contains pointer to next one. */
- mq_message_t *first;
- /** Pointer to last message in the queue to speed up appending messages
- * into the queue.*/
- mq_message_t *last;
- /** Queue mutex - might not be initialized, depends on initialization
- * parameters */
- cds_mutex_t q_mutex;
- /** flags - see MQ_xxx constants */
- unsigned int flags;
- } msg_queue_t;
- /** Macro for accessing message data.
- * It is better to use this macro than accessing internal members of
- * the structure. */
- #define get_message_data(msg) (msg ? msg->data: NULL)
- /** Macro for determining message data length.
- * It is better to use this macro than accessing internal members of
- * the structure. */
- #define get_message_data_len(msg) (msg ? msg->data_len: 0)
- /** The space for data is allocated in messages data
- * (they are automaticaly freed!)! Pointer to allocated
- * data bytes is in data variable in the message structure. */
- mq_message_t *create_message_ex(int data_len);
- /** Creates message holding data allocated using cds_malloc.
- * Data must be allocated using cds_malloc or there must be
- * set destroy function via \ref set_data_destroy_function
- * because they are automaticaly freed by free_message! */
- mq_message_t *create_message(void *data, int data_len);
- /** Sets function which will be called by free_message to destroy data.
- *
- * This function may be useful when a complex structure with pointers is added
- * as data parameter. */
- void set_data_destroy_function(mq_message_t *msg, destroy_function_f func);
- /** Initializes message.
- * If auto_free set, data must be allocated using cds_malloc and are
- * automaticaly freed by free_message (and if msg_queue_destroy called) */
- void init_message_ex(mq_message_t *m, void *data, int data_len, destroy_function_f func);
- /** Frees the message and data hold by the message. */
- void free_message(mq_message_t *msg);
- /** Put message into queue. */
- int push_message(msg_queue_t *q, mq_message_t *m);
- /** Remove message from queue. */
- mq_message_t *pop_message(msg_queue_t *q);
- /** Tests if message queue holds a message.
- * \retval 1 if empty
- * \retval 0 if NOT empty. */
- int is_msg_queue_empty(msg_queue_t *q);
- /** Initializes message queue with a mutex guarding queue operations. */
- int msg_queue_init(msg_queue_t *q);
- /** Initializes message queue. If synchronize is set it initializes
- * a mutex guarding queue operations otherwise the message queue remains
- * unsynchronized. */
- int msg_queue_init_ex(msg_queue_t *q, int synchronize);
- /** Initializes reference counter for given message queue
- * \param grp specifies group of reference counters to use. The message
- * queue will stop using the reference counter if NULL.
- * \param q specifies the message queue */
- void msg_queue_init_ref_cnt(msg_queue_t *q, reference_counter_group_t *grp);
- /** Destroys message queue if no more references exist.
- * This function destroys all message queue internal data but doesn't free
- * the message queue itself. It can be useful for staticaly allocated queues
- * or when allocated not using cds_malloc. */
- void msg_queue_destroy(msg_queue_t *q);
- /** Destroys and frees the message queue if no more references exist.
- * It uses cds_free for freeing the memory. */
- void msg_queue_free(msg_queue_t *q);
- /** @}
- @} */
- #ifdef __cplusplus
- }
- #endif
- #endif
|