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 | 2 | * \file |
3 | 3 | * Holds requests ready for processing. |
4 | - * | |
4 | + * | |
5 | 5 | * \todo change this to a real queue. |
6 | 6 | * |
7 | 7 | * \author Georg Hopp |
... | ... | @@ -32,34 +32,23 @@ |
32 | 32 | |
33 | 33 | |
34 | 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 | 39 | int free_msgs; |
50 | 40 | }; |
51 | 41 | TR_INSTANCE_INIT(TR_Queue); |
52 | 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 | 53 | #endif // __TR_QUEUE_H__ |
65 | 54 | ... | ... |
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 | void * |
27 | 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 | 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 | 41 | // vim: set ts=4 sw=4: | ... | ... |
... | ... | @@ -23,22 +23,49 @@ |
23 | 23 | #include "trbase.h" |
24 | 24 | #include "tr/queue.h" |
25 | 25 | |
26 | +static | |
27 | +inline | |
26 | 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 | 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 | 31 | { |
32 | 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 | 37 | this->free_msgs = 1; |
35 | 38 | |
36 | 39 | return 0; |
... | ... | @@ -41,8 +44,20 @@ void |
41 | 44 | queueDtor(void * _this) |
42 | 45 | { |
43 | 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 | 63 | TR_INIT_IFACE(TR_Class, queueCtor, queueDtor, NULL); | ... | ... |
Please
register
or
login
to post a comment