1 /* Licensed to the Apache Software Foundation (ASF) under one or more
2 * contributor license agreements. See the NOTICE file distributed with
3 * this work for additional information regarding copyright ownership.
4 * The ASF licenses this file to You under the Apache License, Version 2.0
5 * (the "License"); you may not use this file except in compliance with
6 * the License. You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
20 #include "apr_general.h"
22 #include "apr_reslist.h"
23 #include "apr_thread_pool.h"
27 #endif /* APR_HAVE_TIME_H */
35 #define RESLIST_SMAX 10
36 #define RESLIST_HMAX 20
37 #define RESLIST_TTL APR_TIME_C(35000) /* 35 ms */
38 #define CONSUMER_THREADS 25
39 #define CONSUMER_ITERATIONS 250
40 #define CONSTRUCT_SLEEP_TIME APR_TIME_C(25000) /* 25 ms */
41 #define DESTRUCT_SLEEP_TIME APR_TIME_C(10000) /* 10 ms */
42 #define WORK_DELAY_SLEEP_TIME APR_TIME_C(15000) /* 15 ms */
45 apr_interval_time_t sleep_upon_construct
;
46 apr_interval_time_t sleep_upon_destruct
;
55 /* Linear congruential generator */
56 static apr_uint32_t
lgc(apr_uint32_t a
)
60 z
%= APR_UINT64_C(4294967291);
61 return (apr_uint32_t
)z
;
64 static apr_status_t
my_constructor(void **resource
, void *params
,
68 my_parameters_t
*my_params
= params
;
70 /* Create some resource */
71 res
= apr_palloc(pool
, sizeof(*res
));
72 res
->id
= my_params
->c_count
++;
74 /* Sleep for awhile, to simulate construction overhead. */
75 apr_sleep(my_params
->sleep_upon_construct
);
77 /* Set the resource so it can be managed by the reslist */
82 static apr_status_t
my_destructor(void *resource
, void *params
,
85 my_resource_t
*res
= resource
;
86 my_parameters_t
*my_params
= params
;
87 res
->id
= my_params
->d_count
++;
89 apr_sleep(my_params
->sleep_upon_destruct
);
97 apr_reslist_t
*reslist
;
98 apr_interval_time_t work_delay_sleep
;
101 /* MAX_UINT * .95 = 2**32 * .95 = 4080218931u */
102 #define PERCENT95th 4080218931u
104 static void * APR_THREAD_FUNC
resource_consuming_thread(apr_thread_t
*thd
,
112 my_thread_info_t
*thread_info
= data
;
113 apr_reslist_t
*rl
= thread_info
->reslist
;
116 apr_generate_random_bytes((void*)&chance
, sizeof(chance
));
118 chance
= (apr_uint32_t
)(apr_time_now() % APR_TIME_C(4294967291));
121 for (i
= 0; i
< CONSUMER_ITERATIONS
; i
++) {
122 rv
= apr_reslist_acquire(rl
, &vp
);
123 ABTS_INT_EQUAL(thread_info
->tc
, APR_SUCCESS
, rv
);
125 apr_sleep(thread_info
->work_delay_sleep
);
127 /* simulate a 5% chance of the resource being bad */
128 chance
= lgc(chance
);
129 if ( chance
< PERCENT95th
) {
130 rv
= apr_reslist_release(rl
, res
);
131 ABTS_INT_EQUAL(thread_info
->tc
, APR_SUCCESS
, rv
);
133 rv
= apr_reslist_invalidate(rl
, res
);
134 ABTS_INT_EQUAL(thread_info
->tc
, APR_SUCCESS
, rv
);
141 static void test_timeout(abts_case
*tc
, apr_reslist_t
*rl
)
144 my_resource_t
*resources
[RESLIST_HMAX
];
149 apr_reslist_timeout_set(rl
, 1000);
151 /* deplete all possible resources from the resource list
152 * so that the next call will block until timeout is reached
153 * (since there are no other threads to make a resource
157 for (i
= 0; i
< RESLIST_HMAX
; i
++) {
158 rv
= apr_reslist_acquire(rl
, (void**)&resources
[i
]);
159 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
162 /* next call will block until timeout is reached */
163 rv
= apr_reslist_acquire(rl
, &vp
);
164 ABTS_TRUE(tc
, APR_STATUS_IS_TIMEUP(rv
));
168 /* release the resources; otherwise the destroy operation
171 for (i
= 0; i
< RESLIST_HMAX
; i
++) {
172 rv
= apr_reslist_release(rl
, resources
[i
]);
173 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
177 static void test_shrinking(abts_case
*tc
, apr_reslist_t
*rl
)
180 my_resource_t
*resources
[RESLIST_HMAX
];
184 int sleep_time
= RESLIST_TTL
/ RESLIST_HMAX
;
186 /* deplete all possible resources from the resource list */
187 for (i
= 0; i
< RESLIST_HMAX
; i
++) {
188 rv
= apr_reslist_acquire(rl
, (void**)&resources
[i
]);
189 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
192 /* Free all resources above RESLIST_SMAX - 1 */
193 for (i
= RESLIST_SMAX
- 1; i
< RESLIST_HMAX
; i
++) {
194 rv
= apr_reslist_release(rl
, resources
[i
]);
195 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
198 for (i
= 0; i
< RESLIST_HMAX
; i
++) {
199 rv
= apr_reslist_acquire(rl
, &vp
);
200 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
202 apr_sleep(sleep_time
);
203 rv
= apr_reslist_release(rl
, res
);
204 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
206 apr_sleep(sleep_time
);
209 * Now free the remaining elements. This should trigger the shrinking of
212 for (i
= 0; i
< RESLIST_SMAX
- 1; i
++) {
213 rv
= apr_reslist_release(rl
, resources
[i
]);
214 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
218 static void test_reslist(abts_case
*tc
, void *data
)
223 my_parameters_t
*params
;
224 apr_thread_pool_t
*thrp
;
225 my_thread_info_t thread_info
[CONSUMER_THREADS
];
227 rv
= apr_thread_pool_create(&thrp
, CONSUMER_THREADS
/2, CONSUMER_THREADS
, p
);
228 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
230 /* Create some parameters that will be passed into each
231 * constructor and destructor call. */
232 params
= apr_pcalloc(p
, sizeof(*params
));
233 params
->sleep_upon_construct
= CONSTRUCT_SLEEP_TIME
;
234 params
->sleep_upon_destruct
= DESTRUCT_SLEEP_TIME
;
236 /* We're going to want 10 blocks of data from our target rmm. */
237 rv
= apr_reslist_create(&rl
, RESLIST_MIN
, RESLIST_SMAX
, RESLIST_HMAX
,
238 RESLIST_TTL
, my_constructor
, my_destructor
,
240 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
242 for (i
= 0; i
< CONSUMER_THREADS
; i
++) {
243 thread_info
[i
].tid
= i
;
244 thread_info
[i
].tc
= tc
;
245 thread_info
[i
].reslist
= rl
;
246 thread_info
[i
].work_delay_sleep
= WORK_DELAY_SLEEP_TIME
;
247 rv
= apr_thread_pool_push(thrp
, resource_consuming_thread
,
248 &thread_info
[i
], 0, NULL
);
249 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
252 rv
= apr_thread_pool_destroy(thrp
);
253 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
255 test_timeout(tc
, rl
);
257 test_shrinking(tc
, rl
);
258 ABTS_INT_EQUAL(tc
, RESLIST_SMAX
, params
->c_count
- params
->d_count
);
260 rv
= apr_reslist_destroy(rl
);
261 ABTS_INT_EQUAL(tc
, APR_SUCCESS
, rv
);
264 #endif /* APR_HAS_THREADS */
266 abts_suite
*testreslist(abts_suite
*suite
)
268 suite
= ADD_SUITE(suite
);
271 abts_run_test(suite
, test_reslist
, NULL
);