diff options
author | Pablo Neira Ayuso <pablo@netfilter.org> | 2009-01-15 23:19:58 +0100 |
---|---|---|
committer | Pablo Neira Ayuso <pablo@netfilter.org> | 2009-01-15 23:19:58 +0100 |
commit | e2af183ea7e5ea35a1582f40a01a7c49e83b31be (patch) | |
tree | 5d5c5fabca580aa2851fb39c3e343b5bc324342e /src/queue.c | |
parent | 2cacd3a802510bde43e23cf4c7d39f51a2eaf460 (diff) | |
download | conntrack-tools-e2af183ea7e5ea35a1582f40a01a7c49e83b31be.tar.gz conntrack-tools-e2af183ea7e5ea35a1582f40a01a7c49e83b31be.zip |
sync: unify tx_list and tx_queue into one single tx_queue
This patch unifies the tx_list and the tx_queue to have only one
transmission queue. Since the tx_list hold state objects and
tx_queue control messages, I have introduced a queue node type
that can be used to differenciate the kind of information that
the node stores: object or control message.
This patch also reworks the existing queue class to include a
file descriptor that can be used to know if there are new data
added to the queue (see QUEUE_F_EVFD flag).
In this change, I have also modified the current evfd to make the
file descriptor to make read operations non-blocking. Moreover,
it keeps a counter that is used to know how many messages are
inserted in the queue.
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
Diffstat (limited to 'src/queue.c')
-rw-r--r-- | src/queue.c | 124 |
1 files changed, 68 insertions, 56 deletions
diff --git a/src/queue.c b/src/queue.c index cdd70ae..cffcc93 100644 --- a/src/queue.c +++ b/src/queue.c @@ -1,5 +1,5 @@ /* - * (C) 2006-2008 by Pablo Neira Ayuso <pablo@netfilter.org> + * (C) 2006-2009 by Pablo Neira Ayuso <pablo@netfilter.org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -17,110 +17,122 @@ */ #include "queue.h" +#include "event.h" #include <errno.h> #include <stdlib.h> #include <string.h> -struct queue *queue_create(size_t max_size) +struct queue *queue_create(int max_objects, unsigned int flags) { struct queue *b; - b = malloc(sizeof(struct queue)); + b = calloc(sizeof(struct queue), 1); if (b == NULL) return NULL; - memset(b, 0, sizeof(struct queue)); - b->max_size = max_size; + b->max_elems = max_objects; INIT_LIST_HEAD(&b->head); + b->flags = flags; + + if (flags & QUEUE_F_EVFD) { + b->evfd = create_evfd(); + if (b->evfd == NULL) { + free(b); + return NULL; + } + } return b; } void queue_destroy(struct queue *b) { - struct list_head *i, *tmp; - struct queue_node *node; - - /* XXX: set cur_size and num_elems */ - list_for_each_safe(i, tmp, &b->head) { - node = (struct queue_node *) i; - list_del(i); - free(node); - } + if (b->flags & QUEUE_F_EVFD) + destroy_evfd(b->evfd); free(b); } -static struct queue_node *queue_node_create(const void *data, size_t size) +void queue_node_init(struct queue_node *n, int type) { - struct queue_node *n; + INIT_LIST_HEAD(&n->head); + n->type = type; +} - n = malloc(sizeof(struct queue_node) + size); - if (n == NULL) +void *queue_node_data(struct queue_node *n) +{ + return ((char *)n) + sizeof(struct queue_node); +} + +struct queue_object *queue_object_new(int type, size_t size) +{ + struct queue_object *obj; + + obj = calloc(sizeof(struct queue_object) + size, 1); + if (obj == NULL) return NULL; - n->size = size; - memcpy(n->data, data, size); + obj->qnode.size = size; + queue_node_init(&obj->qnode, type); - return n; + return obj; } -int queue_add(struct queue *b, const void *data, size_t size) +void queue_object_free(struct queue_object *obj) { - int ret = 0; - struct queue_node *n; - - /* does it fit this queue? */ - if (size > b->max_size) { - errno = ENOSPC; - ret = -1; - goto err; - } + free(obj); +} -retry: - /* queue is full: kill the oldest entry */ - if (b->cur_size + size > b->max_size) { - n = (struct queue_node *) b->head.prev; - list_del(b->head.prev); - b->cur_size -= n->size; - free(n); - goto retry; - } +int queue_add(struct queue *b, struct queue_node *n) +{ + if (!list_empty(&n->head)) + return 0; - n = queue_node_create(data, size); - if (n == NULL) { - ret = -1; - goto err; + if (b->num_elems >= b->max_elems) { + errno = ENOSPC; + return -1; } - + n->owner = b; list_add_tail(&n->head, &b->head); - b->cur_size += size; b->num_elems++; + if (b->evfd) + write_evfd(b->evfd); + return 1; +} -err: - return ret; +int queue_del(struct queue_node *n) +{ + if (list_empty(&n->head)) + return 0; + + list_del_init(&n->head); + n->owner->num_elems--; + if (n->owner->evfd) + read_evfd(n->owner->evfd); + n->owner = NULL; + return 1; } -void queue_del(struct queue *b, void *data) +int queue_in(struct queue *b, struct queue_node *n) { - struct queue_node *n = container_of(data, struct queue_node, data); + return b == n->owner; +} - list_del(&n->head); - b->cur_size -= n->size; - b->num_elems--; - free(n); +int queue_get_eventfd(struct queue *b) +{ + return get_read_evfd(b->evfd); } void queue_iterate(struct queue *b, const void *data, - int (*iterate)(void *data1, const void *data2)) + int (*iterate)(struct queue_node *n, const void *data2)) { struct list_head *i, *tmp; struct queue_node *n; list_for_each_safe(i, tmp, &b->head) { n = (struct queue_node *) i; - if (iterate(n->data, data)) + if (iterate(n, data)) break; } } |