forked from 51Degrees/common-cxx
-
Notifications
You must be signed in to change notification settings - Fork 0
/
pool.c
149 lines (134 loc) · 4.5 KB
/
pool.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
/* *********************************************************************
* This Original Work is copyright of 51 Degrees Mobile Experts Limited.
* Copyright 2019 51 Degrees Mobile Experts Limited, 5 Charlotte Close,
* Caversham, Reading, Berkshire, United Kingdom RG4 7BY.
*
* This Original Work is licensed under the European Union Public Licence (EUPL)
* v.1.2 and is subject to its terms as set out below.
*
* If a copy of the EUPL was not distributed with this file, You can obtain
* one at https://opensource.org/licenses/EUPL-1.2.
*
* The 'Compatible Licences' set out in the Appendix to the EUPL (as may be
* amended by the European Commission) shall be deemed incompatible for
* the purposes of the Work and the provisions of the compatibility
* clause in Article 5 of the EUPL shall not apply.
*
* If using the Work as, or as part of, a network application, by
* including the attribution notice(s) required under Article 5 of the EUPL
* in the end user terms of the application under an appropriate heading,
* such notice(s) shall fulfill the requirements of that article.
* ********************************************************************* */
#include "pool.h"
#include "fiftyone.h"
fiftyoneDegreesPool* fiftyoneDegreesPoolInit(
fiftyoneDegreesPool *pool,
uint16_t concurrency,
void *state,
fiftyoneDegreesPoolResourceCreate resourceCreate,
fiftyoneDegreesPoolResourceFree resourceFree,
fiftyoneDegreesException *exception) {
uint16_t i = 1;
PoolItem *item;
// Add one to the concurrency value so that a NULL marker can be
// written as the last item in the linked list that if returned
// indicates that the concurrency has been exceeded.
uint16_t listItems = concurrency + 1;
// Set the stack and head of the linked list.
pool->count = 0;
pool->head.exchange = 0;
pool->resourceFree = resourceFree;
// Allocate memory for the stack.
pool->stack = (PoolItem*)Malloc(sizeof(PoolItem) * listItems);
if (pool->stack != NULL) {
// The entry at index 0 in the stack is the null item which if ever
// retrieved indicates that the pool is exhausted. It does not contain
// a value resource.
item = &pool->stack[0];
item->pool = pool;
item->resource = NULL;
// Initialise all the resources in the pool after the null terminator.
while (i < listItems && EXCEPTION_OKAY) {
item = &pool->stack[i];
item->pool = pool;
item->resource = resourceCreate(pool, state, exception);
pool->count++;
item->next = pool->head.values.index;
pool->head.values.index = i;
i++;
}
}
else {
EXCEPTION_SET(INSUFFICIENT_MEMORY);
}
return pool;
}
fiftyoneDegreesPoolItem* fiftyoneDegreesPoolItemGet(
fiftyoneDegreesPool *pool,
fiftyoneDegreesException *exception) {
PoolHead orig;
#ifndef FIFTYONE_DEGREES_NO_THREADING
PoolHead next;
do {
#endif
orig = pool->head;
// Check that the head of the list is not the null resource which
// would indicate that there are more active concurrent operations than
// the pool has been configured for.
if (pool->stack[orig.values.index].resource == NULL) {
EXCEPTION_SET(INSUFFICIENT_HANDLES)
return NULL;
}
#ifndef FIFTYONE_DEGREES_NO_THREADING
next.values.aba = orig.values.aba + 1;
next.values.index = pool->stack[orig.values.index].next;
} while (INTERLOCK_EXCHANGE(
pool->head.exchange,
next.exchange,
orig.exchange) != orig.exchange);
#else
pool->head.values.index = pool->stack[orig.values.index].next;
#endif
return &pool->stack[orig.values.index];
}
void fiftyoneDegreesPoolItemRelease(fiftyoneDegreesPoolItem *item) {
#ifndef FIFTYONE_DEGREES_NO_THREADING
PoolHead orig, next;
do {
orig = item->pool->head;
item->next = orig.values.index;
next.values.aba = orig.values.aba + 1;
next.values.index = (uint16_t)(item - item->pool->stack);
} while (INTERLOCK_EXCHANGE(
item->pool->head.exchange,
next.exchange,
orig.exchange) != orig.exchange);
#else
item->next = item->pool->head.values.index;
item->pool->head.values.index =
(uint16_t)(item - item->pool->stack);
#endif
}
void fiftyoneDegreesPoolReset(fiftyoneDegreesPool *pool) {
pool->head.values.index = 0;
pool->head.values.aba = 0;
pool->stack = NULL;
pool->count = 0;
pool->resourceFree = NULL;
}
void fiftyoneDegreesPoolFree(fiftyoneDegreesPool *pool) {
void *resource;
uint16_t i;
if (pool->stack != NULL) {
if (pool->resourceFree != NULL) {
for (i = 0; i <= pool->count; i++) {
resource = pool->stack[i].resource;
if (resource != NULL) {
pool->resourceFree(pool, resource);
}
}
}
Free(pool->stack);
}
PoolReset(pool);
}