Commit 886d391288d124a845ee1a32fa82571cc8d3d04a
1 parent
26d6138c
Created new queue implementation based on dynamically growing arrays
Showing
10 changed files
with
70 additions
and
263 deletions
1 | /** | 1 | /** |
2 | * \file | 2 | * \file |
3 | * Holds requests ready for processing. | 3 | * Holds requests ready for processing. |
4 | - * | 4 | + * |
5 | * \todo change this to a real queue. | 5 | * \todo change this to a real queue. |
6 | * | 6 | * |
7 | * \author Georg Hopp | 7 | * \author Georg Hopp |
@@ -32,34 +32,23 @@ | @@ -32,34 +32,23 @@ | ||
32 | 32 | ||
33 | 33 | ||
34 | TR_CLASS(TR_Queue) { | 34 | TR_CLASS(TR_Queue) { |
35 | - void * msg; | ||
36 | - TR_Queue next; | ||
37 | - | ||
38 | - /** | ||
39 | - * first and last are only available in the initial queue | ||
40 | - * element (the root). This elelment does not contain any message | ||
41 | - * and exists only for organizational purpose. | ||
42 | - * | ||
43 | - * \todo next and first always have to be the same...so get rid | ||
44 | - * of first. | ||
45 | - */ | ||
46 | - TR_Queue first; | ||
47 | - TR_Queue last; | ||
48 | - size_t nmsg; | 35 | + size_t size; |
36 | + size_t start; | ||
37 | + size_t end; | ||
38 | + void ** data; | ||
49 | int free_msgs; | 39 | int free_msgs; |
50 | }; | 40 | }; |
51 | TR_INSTANCE_INIT(TR_Queue); | 41 | TR_INSTANCE_INIT(TR_Queue); |
52 | TR_CLASSVARS_DECL(TR_Queue) {}; | 42 | TR_CLASSVARS_DECL(TR_Queue) {}; |
53 | 43 | ||
54 | -void TR_queuePut(TR_Queue, void *); | ||
55 | -void TR_queuePutFirst(TR_Queue, void *); | ||
56 | -void * TR_queueGet(TR_Queue); | ||
57 | -TR_Queue TR_queueFind(TR_Queue, void *); | ||
58 | -TR_Queue TR_queueFindParent(TR_Queue, void *); | ||
59 | -void TR_queueDelete(TR_Queue, void *); | ||
60 | -void TR_queueDestroy(TR_Queue); | 44 | +void TR_queuePut(TR_Queue, void *); |
45 | +void TR_queuePutFirst(TR_Queue, void *); | ||
46 | +void * TR_queueGet(TR_Queue); | ||
61 | 47 | ||
62 | -#define TR_queueEmpty(this) (0 >= (this)->nmsg) | 48 | +#define TR_queueEmpty(this) (this->start == this->end) |
49 | +#define TR_queueFirst(this) ((this)->start) | ||
50 | +#define TR_queueLast(this) ((this)->end - 1) | ||
51 | +#define TR_queueSize(this) ((this)->size) | ||
63 | 52 | ||
64 | #endif // __TR_QUEUE_H__ | 53 | #endif // __TR_QUEUE_H__ |
65 | 54 |
@@ -6,12 +6,7 @@ AM_LDFLAGS += | @@ -6,12 +6,7 @@ AM_LDFLAGS += | ||
6 | 6 | ||
7 | QUEUE = queue.c \ | 7 | QUEUE = queue.c \ |
8 | get.c \ | 8 | get.c \ |
9 | - put.c \ | ||
10 | - put_first.c \ | ||
11 | - find.c \ | ||
12 | - find_parent.c \ | ||
13 | - delete.c \ | ||
14 | - destroy.c | 9 | + put.c |
15 | 10 | ||
16 | noinst_LTLIBRARIES = libqueue.la | 11 | noinst_LTLIBRARIES = libqueue.la |
17 | 12 |
src/queue/delete.c
deleted
100644 → 0
1 | -/** | ||
2 | - * \file | ||
3 | - * | ||
4 | - * \author Georg Hopp | ||
5 | - * | ||
6 | - * \copyright | ||
7 | - * Copyright © 2014 Georg Hopp | ||
8 | - * | ||
9 | - * This program is free software: you can redistribute it and/or modify | ||
10 | - * it under the terms of the GNU General Public License as published by | ||
11 | - * the Free Software Foundation, either version 3 of the License, or | ||
12 | - * (at your option) any later version. | ||
13 | - * | ||
14 | - * This program is distributed in the hope that it will be useful, | ||
15 | - * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
16 | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
17 | - * GNU General Public License for more details. | ||
18 | - * | ||
19 | - * You should have received a copy of the GNU General Public License | ||
20 | - * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
21 | - */ | ||
22 | - | ||
23 | -#include "trbase.h" | ||
24 | -#include "tr/queue.h" | ||
25 | - | ||
26 | -void | ||
27 | -TR_queueDelete(TR_Queue this, void * msg) | ||
28 | -{ | ||
29 | - TR_Queue node, parent; | ||
30 | - | ||
31 | - parent = TR_queueFindParent(this, msg); | ||
32 | - | ||
33 | - if (! parent) { | ||
34 | - return; | ||
35 | - } | ||
36 | - | ||
37 | - node = parent->next; | ||
38 | - parent->next = node->next; | ||
39 | - if (node == this->last) { | ||
40 | - this->last = parent; | ||
41 | - } | ||
42 | - if (node == this->first) { | ||
43 | - this->first = node->next; | ||
44 | - } | ||
45 | - TR_delete(node); | ||
46 | - this->nmsg--; | ||
47 | -} | ||
48 | - | ||
49 | -// vim: set ts=4 sw=4: |
src/queue/destroy.c
deleted
100644 → 0
1 | -/** | ||
2 | - * \file | ||
3 | - * | ||
4 | - * \author Georg Hopp | ||
5 | - * | ||
6 | - * \copyright | ||
7 | - * Copyright © 2014 Georg Hopp | ||
8 | - * | ||
9 | - * This program is free software: you can redistribute it and/or modify | ||
10 | - * it under the terms of the GNU General Public License as published by | ||
11 | - * the Free Software Foundation, either version 3 of the License, or | ||
12 | - * (at your option) any later version. | ||
13 | - * | ||
14 | - * This program is distributed in the hope that it will be useful, | ||
15 | - * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
16 | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
17 | - * GNU General Public License for more details. | ||
18 | - * | ||
19 | - * You should have received a copy of the GNU General Public License | ||
20 | - * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
21 | - */ | ||
22 | - | ||
23 | -#include <stdarg.h> | ||
24 | - | ||
25 | -#include "trbase.h" | ||
26 | -#include "tr/queue.h" | ||
27 | - | ||
28 | -void | ||
29 | -TR_queueDestroy(TR_Queue this) | ||
30 | -{ | ||
31 | - TR_Queue node; | ||
32 | - | ||
33 | - node = this->first; | ||
34 | - | ||
35 | - while (NULL != node) { | ||
36 | - TR_Queue next = node->next; | ||
37 | - if (this->free_msgs) { | ||
38 | - TR_delete(node->msg); | ||
39 | - } | ||
40 | - TR_delete(node); | ||
41 | - node = next; | ||
42 | - } | ||
43 | - | ||
44 | - this->first = this->next = this->last = NULL; | ||
45 | - this->nmsg = 0; | ||
46 | -} | ||
47 | - | ||
48 | -// vim: set ts=4 sw=4: |
src/queue/find.c
deleted
100644 → 0
1 | -/** | ||
2 | - * \file | ||
3 | - * | ||
4 | - * \author Georg Hopp | ||
5 | - * | ||
6 | - * \copyright | ||
7 | - * Copyright © 2014 Georg Hopp | ||
8 | - * | ||
9 | - * This program is free software: you can redistribute it and/or modify | ||
10 | - * it under the terms of the GNU General Public License as published by | ||
11 | - * the Free Software Foundation, either version 3 of the License, or | ||
12 | - * (at your option) any later version. | ||
13 | - * | ||
14 | - * This program is distributed in the hope that it will be useful, | ||
15 | - * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
16 | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
17 | - * GNU General Public License for more details. | ||
18 | - * | ||
19 | - * You should have received a copy of the GNU General Public License | ||
20 | - * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
21 | - */ | ||
22 | - | ||
23 | -#include "trbase.h" | ||
24 | -#include "tr/queue.h" | ||
25 | - | ||
26 | -TR_Queue | ||
27 | -TR_queueFind(TR_Queue this, void * msg) | ||
28 | -{ | ||
29 | - TR_Queue node; | ||
30 | - | ||
31 | - for (node = this->first; node && node->msg != msg; node = node->next); | ||
32 | - | ||
33 | - return node; | ||
34 | -} | ||
35 | - | ||
36 | -// vim: set ts=4 sw=4: |
src/queue/find_parent.c
deleted
100644 → 0
1 | -/** | ||
2 | - * \file | ||
3 | - * | ||
4 | - * \author Georg Hopp | ||
5 | - * | ||
6 | - * \copyright | ||
7 | - * Copyright © 2014 Georg Hopp | ||
8 | - * | ||
9 | - * This program is free software: you can redistribute it and/or modify | ||
10 | - * it under the terms of the GNU General Public License as published by | ||
11 | - * the Free Software Foundation, either version 3 of the License, or | ||
12 | - * (at your option) any later version. | ||
13 | - * | ||
14 | - * This program is distributed in the hope that it will be useful, | ||
15 | - * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
16 | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
17 | - * GNU General Public License for more details. | ||
18 | - * | ||
19 | - * You should have received a copy of the GNU General Public License | ||
20 | - * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
21 | - */ | ||
22 | - | ||
23 | -#include "trbase.h" | ||
24 | -#include "tr/queue.h" | ||
25 | - | ||
26 | -TR_Queue | ||
27 | -TR_queueFindParent(TR_Queue this, void * msg) | ||
28 | -{ | ||
29 | - TR_Queue node; | ||
30 | - | ||
31 | - for (node = this; node->next && node->next->msg != msg; node = node->next); | ||
32 | - | ||
33 | - return node->next ? node : NULL; | ||
34 | -} | ||
35 | - | ||
36 | -// vim: set ts=4 sw=4: |
@@ -26,26 +26,16 @@ | @@ -26,26 +26,16 @@ | ||
26 | void * | 26 | void * |
27 | TR_queueGet(TR_Queue this) | 27 | TR_queueGet(TR_Queue this) |
28 | { | 28 | { |
29 | - TR_Queue first; | ||
30 | - void * msg; | 29 | + void * retval; |
31 | 30 | ||
32 | - if (NULL == this->first) { | 31 | + if (TR_queueEmpty(this)) { |
33 | return NULL; | 32 | return NULL; |
34 | } | 33 | } |
35 | 34 | ||
36 | - msg = this->first->msg; | ||
37 | - first = this->first->next; | 35 | + retval = this->data[this->start]; |
36 | + this->start = this->start + 1 == this->size ? 0 : this->start + 1; | ||
38 | 37 | ||
39 | - if (this->first == this->last) { | ||
40 | - this->last = NULL; | ||
41 | - } | ||
42 | - TR_delete(this->first); | ||
43 | - | ||
44 | - this->next = first; | ||
45 | - this->first = first; | ||
46 | - this->nmsg--; | ||
47 | - | ||
48 | - return msg; | 38 | + return retval; |
49 | } | 39 | } |
50 | 40 | ||
51 | // vim: set ts=4 sw=4: | 41 | // vim: set ts=4 sw=4: |
@@ -23,22 +23,49 @@ | @@ -23,22 +23,49 @@ | ||
23 | #include "trbase.h" | 23 | #include "trbase.h" |
24 | #include "tr/queue.h" | 24 | #include "tr/queue.h" |
25 | 25 | ||
26 | +static | ||
27 | +inline | ||
26 | void | 28 | void |
27 | -TR_queuePut(TR_Queue this, void * msg) | 29 | +_TR_queueResize(TR_Queue this, size_t split) |
28 | { | 30 | { |
29 | - TR_Queue node; | 31 | +#define VPSIZE(elem) ((elem) * sizeof(void*)) |
32 | + | ||
33 | + void ** new = TR_malloc(VPSIZE(this->size + 1)); | ||
34 | + size_t new_size = TR_getUsableSize(new) / sizeof(void *); | ||
30 | 35 | ||
31 | - node = (this->last)? this->last : this; | 36 | +#define GROWTH (new_size - this->size) |
37 | + | ||
38 | + memcpy(new, this->data, VPSIZE(split)); | ||
39 | + memcpy( | ||
40 | + new + GROWTH + this->start, | ||
41 | + &(this->data[this->start]), | ||
42 | + VPSIZE(this->size - this->start)); | ||
43 | + this->start = GROWTH + this->start; | ||
44 | + this->size = new_size; | ||
45 | +} | ||
46 | + | ||
47 | +void | ||
48 | +TR_queuePutFirst(TR_Queue this, void * msg) | ||
49 | +{ | ||
50 | + size_t next = this->start == 0 ? this->size - 1: this->start - 1; | ||
32 | 51 | ||
33 | - node->next = TR_new(TR_Queue); | ||
34 | - this->last = node->next; | 52 | + this->data[next] = msg; |
53 | + this->start = next; | ||
35 | 54 | ||
36 | - if (node == this) { | ||
37 | - this->first = node->next; | 55 | + if (next == this->end) { |
56 | + _TR_queueResize(this, this->end); | ||
38 | } | 57 | } |
58 | +} | ||
59 | + | ||
60 | +void | ||
61 | +TR_queuePut(TR_Queue this, void * msg) | ||
62 | +{ | ||
63 | + this->data[this->end] = msg; | ||
64 | + this->end = this->end + 1 == this->size ? 0 : this->end + 1; | ||
39 | 65 | ||
40 | - node->next->msg = msg; | ||
41 | - this->nmsg++; | 66 | + if (this->end == this->start) { |
67 | + _TR_queueResize(this, this->end); | ||
68 | + } | ||
42 | } | 69 | } |
43 | 70 | ||
44 | // vim: set ts=4 sw=4: | 71 | // vim: set ts=4 sw=4: |
src/queue/put_first.c
deleted
100644 → 0
1 | -/** | ||
2 | - * \file | ||
3 | - * | ||
4 | - * \author Georg Hopp | ||
5 | - * | ||
6 | - * \copyright | ||
7 | - * Copyright © 2014 Georg Hopp | ||
8 | - * | ||
9 | - * This program is free software: you can redistribute it and/or modify | ||
10 | - * it under the terms of the GNU General Public License as published by | ||
11 | - * the Free Software Foundation, either version 3 of the License, or | ||
12 | - * (at your option) any later version. | ||
13 | - * | ||
14 | - * This program is distributed in the hope that it will be useful, | ||
15 | - * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
16 | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
17 | - * GNU General Public License for more details. | ||
18 | - * | ||
19 | - * You should have received a copy of the GNU General Public License | ||
20 | - * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
21 | - */ | ||
22 | - | ||
23 | -#include "trbase.h" | ||
24 | -#include "tr/queue.h" | ||
25 | - | ||
26 | -void | ||
27 | -TR_queuePutFirst(TR_Queue this, void * msg) | ||
28 | -{ | ||
29 | - TR_Queue current_first; | ||
30 | - | ||
31 | - current_first = this->first; | ||
32 | - | ||
33 | - this->first = this->next = TR_new(TR_Queue); | ||
34 | - this->first->next = current_first; | ||
35 | - this->first->msg = msg; | ||
36 | - this->last = this->last ? this->last : this->first; | ||
37 | - this->nmsg++; | ||
38 | -} | ||
39 | - | ||
40 | -// vim: set ts=4 sw=4: |
@@ -31,6 +31,9 @@ queueCtor(void * _this, va_list * params) | @@ -31,6 +31,9 @@ queueCtor(void * _this, va_list * params) | ||
31 | { | 31 | { |
32 | TR_Queue this = _this; | 32 | TR_Queue this = _this; |
33 | 33 | ||
34 | + this->data = (void **)TR_malloc(32 * sizeof(void *)); | ||
35 | + this->size = TR_getUsableSize(this->data) / sizeof(void *); | ||
36 | + this->start = this->end = 0; | ||
34 | this->free_msgs = 1; | 37 | this->free_msgs = 1; |
35 | 38 | ||
36 | return 0; | 39 | return 0; |
@@ -41,8 +44,20 @@ void | @@ -41,8 +44,20 @@ void | ||
41 | queueDtor(void * _this) | 44 | queueDtor(void * _this) |
42 | { | 45 | { |
43 | TR_Queue this = _this; | 46 | TR_Queue this = _this; |
47 | + size_t i; | ||
44 | 48 | ||
45 | - TR_queueDestroy(this); | 49 | + if (this->free_msgs) { |
50 | + for ( | ||
51 | + i = this->start; | ||
52 | + i != this->end; | ||
53 | + i = i + 1 == this->size ? 0 : i + 1) { | ||
54 | + if (this->data[i]) { | ||
55 | + TR_delete(this->data[i]); | ||
56 | + } | ||
57 | + } | ||
58 | + } | ||
59 | + | ||
60 | + TR_MEM_FREE(this->data); | ||
46 | } | 61 | } |
47 | 62 | ||
48 | TR_INIT_IFACE(TR_Class, queueCtor, queueDtor, NULL); | 63 | TR_INIT_IFACE(TR_Class, queueCtor, queueDtor, NULL); |
Please
register
or
login
to post a comment