|  | /* Licensed to the Apache Software Foundation (ASF) under one or more | 
|  | * contributor license agreements.  See the NOTICE file distributed with | 
|  | * this work for additional information regarding copyright ownership. | 
|  | * The ASF licenses this file to You under the Apache License, Version 2.0 | 
|  | * (the "License"); you may not use this file except in compliance with | 
|  | * the License.  You may obtain a copy of the License at | 
|  | * | 
|  | *     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  |  | 
|  | #include "apr_general.h" | 
|  | #include "apu.h" | 
|  | #include "apr_reslist.h" | 
|  | #include "apr_thread_pool.h" | 
|  |  | 
|  | #if APR_HAVE_TIME_H | 
|  | #include <time.h> | 
|  | #endif /* APR_HAVE_TIME_H */ | 
|  |  | 
|  | #include "abts.h" | 
|  | #include "testutil.h" | 
|  |  | 
|  | #if APR_HAS_THREADS | 
|  |  | 
|  | #define RESLIST_MIN   3 | 
|  | #define RESLIST_SMAX 10 | 
|  | #define RESLIST_HMAX 20 | 
|  | #define RESLIST_TTL  APR_TIME_C(35000) /* 35 ms */ | 
|  | #define CONSUMER_THREADS 25 | 
|  | #define CONSUMER_ITERATIONS 100 | 
|  | #define CONSTRUCT_SLEEP_TIME  APR_TIME_C(25000) /* 25 ms */ | 
|  | #define DESTRUCT_SLEEP_TIME   APR_TIME_C(10000) /* 10 ms */ | 
|  | #define WORK_DELAY_SLEEP_TIME APR_TIME_C(15000) /* 15 ms */ | 
|  |  | 
|  | typedef struct { | 
|  | apr_interval_time_t sleep_upon_construct; | 
|  | apr_interval_time_t sleep_upon_destruct; | 
|  | int c_count; | 
|  | int d_count; | 
|  | } my_parameters_t; | 
|  |  | 
|  | typedef struct { | 
|  | int id; | 
|  | } my_resource_t; | 
|  |  | 
|  | /* Linear congruential generator */ | 
|  | static apr_uint32_t lgc(apr_uint32_t a) | 
|  | { | 
|  | apr_uint64_t z = a; | 
|  | z *= 279470273; | 
|  | z %= APR_UINT64_C(4294967291); | 
|  | return (apr_uint32_t)z; | 
|  | } | 
|  |  | 
|  | static apr_status_t my_constructor(void **resource, void *params, | 
|  | apr_pool_t *pool) | 
|  | { | 
|  | my_resource_t *res; | 
|  | my_parameters_t *my_params = params; | 
|  |  | 
|  | /* Create some resource */ | 
|  | res = apr_palloc(pool, sizeof(*res)); | 
|  | res->id = my_params->c_count++; | 
|  |  | 
|  | /* Sleep for awhile, to simulate construction overhead. */ | 
|  | apr_sleep(my_params->sleep_upon_construct); | 
|  |  | 
|  | /* Set the resource so it can be managed by the reslist */ | 
|  | *resource = res; | 
|  | return APR_SUCCESS; | 
|  | } | 
|  |  | 
|  | static apr_status_t my_destructor(void *resource, void *params, | 
|  | apr_pool_t *pool) | 
|  | { | 
|  | my_resource_t *res = resource; | 
|  | my_parameters_t *my_params = params; | 
|  | res->id = my_params->d_count++; | 
|  |  | 
|  | apr_sleep(my_params->sleep_upon_destruct); | 
|  |  | 
|  | return APR_SUCCESS; | 
|  | } | 
|  |  | 
|  | typedef struct { | 
|  | int tid; | 
|  | abts_case *tc; | 
|  | apr_reslist_t *reslist; | 
|  | apr_interval_time_t work_delay_sleep; | 
|  | } my_thread_info_t; | 
|  |  | 
|  | /* MAX_UINT * .95 = 2**32 * .95 = 4080218931u */ | 
|  | #define PERCENT95th 4080218931u | 
|  |  | 
|  | static void * APR_THREAD_FUNC resource_consuming_thread(apr_thread_t *thd, | 
|  | void *data) | 
|  | { | 
|  | int i; | 
|  | apr_uint32_t chance; | 
|  | void *vp; | 
|  | apr_status_t rv; | 
|  | my_resource_t *res; | 
|  | my_thread_info_t *thread_info = data; | 
|  | apr_reslist_t *rl = thread_info->reslist; | 
|  |  | 
|  | #if APR_HAS_RANDOM | 
|  | apr_generate_random_bytes((void*)&chance, sizeof(chance)); | 
|  | #else | 
|  | chance = (apr_uint32_t)(apr_time_now() % APR_TIME_C(4294967291)); | 
|  | #endif | 
|  |  | 
|  | for (i = 0; i < CONSUMER_ITERATIONS; i++) { | 
|  | rv = apr_reslist_acquire(rl, &vp); | 
|  | ABTS_INT_EQUAL(thread_info->tc, APR_SUCCESS, rv); | 
|  | res = vp; | 
|  | apr_sleep(thread_info->work_delay_sleep); | 
|  |  | 
|  | /* simulate a 5% chance of the resource being bad */ | 
|  | chance = lgc(chance); | 
|  | if ( chance < PERCENT95th ) { | 
|  | rv = apr_reslist_release(rl, res); | 
|  | ABTS_INT_EQUAL(thread_info->tc, APR_SUCCESS, rv); | 
|  | } else { | 
|  | rv = apr_reslist_invalidate(rl, res); | 
|  | ABTS_INT_EQUAL(thread_info->tc, APR_SUCCESS, rv); | 
|  | } | 
|  | } | 
|  |  | 
|  | return APR_SUCCESS; | 
|  | } | 
|  |  | 
|  | static void test_timeout(abts_case *tc, apr_reslist_t *rl) | 
|  | { | 
|  | apr_status_t rv; | 
|  | my_resource_t *resources[RESLIST_HMAX]; | 
|  | void *vp; | 
|  | int i; | 
|  |  | 
|  | apr_reslist_timeout_set(rl, 1000); | 
|  |  | 
|  | /* deplete all possible resources from the resource list | 
|  | * so that the next call will block until timeout is reached | 
|  | * (since there are no other threads to make a resource | 
|  | * available) | 
|  | */ | 
|  |  | 
|  | for (i = 0; i < RESLIST_HMAX; i++) { | 
|  | rv = apr_reslist_acquire(rl, (void**)&resources[i]); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  |  | 
|  | /* next call will block until timeout is reached */ | 
|  | rv = apr_reslist_acquire(rl, &vp); | 
|  | ABTS_TRUE(tc, APR_STATUS_IS_TIMEUP(rv)); | 
|  |  | 
|  | /* release the resources; otherwise the destroy operation | 
|  | * will blow | 
|  | */ | 
|  | for (i = 0; i < RESLIST_HMAX; i++) { | 
|  | rv = apr_reslist_release(rl, resources[i]); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void test_shrinking(abts_case *tc, apr_reslist_t *rl) | 
|  | { | 
|  | apr_status_t rv; | 
|  | my_resource_t *resources[RESLIST_HMAX]; | 
|  | my_resource_t *res; | 
|  | void *vp; | 
|  | int i; | 
|  | int sleep_time = RESLIST_TTL / RESLIST_HMAX; | 
|  |  | 
|  | /* deplete all possible resources from the resource list */ | 
|  | for (i = 0; i < RESLIST_HMAX; i++) { | 
|  | rv = apr_reslist_acquire(rl, (void**)&resources[i]); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  |  | 
|  | /* Free all resources above RESLIST_SMAX - 1 */ | 
|  | for (i = RESLIST_SMAX - 1; i < RESLIST_HMAX; i++) { | 
|  | rv = apr_reslist_release(rl, resources[i]); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  |  | 
|  | for (i = 0; i < RESLIST_HMAX; i++) { | 
|  | rv = apr_reslist_acquire(rl, &vp); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | res = vp; | 
|  | apr_sleep(sleep_time); | 
|  | rv = apr_reslist_release(rl, res); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  | apr_sleep(sleep_time); | 
|  |  | 
|  | /* | 
|  | * Now free the remaining elements. This should trigger the shrinking of | 
|  | * the list | 
|  | */ | 
|  | for (i = 0; i < RESLIST_SMAX - 1; i++) { | 
|  | rv = apr_reslist_release(rl, resources[i]); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void test_reslist(abts_case *tc, void *data) | 
|  | { | 
|  | int i; | 
|  | apr_status_t rv; | 
|  | apr_reslist_t *rl; | 
|  | my_parameters_t *params; | 
|  | apr_thread_pool_t *thrp; | 
|  | my_thread_info_t thread_info[CONSUMER_THREADS]; | 
|  |  | 
|  | rv = apr_thread_pool_create(&thrp, CONSUMER_THREADS/2, CONSUMER_THREADS, p); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  |  | 
|  | /* Create some parameters that will be passed into each | 
|  | * constructor and destructor call. */ | 
|  | params = apr_pcalloc(p, sizeof(*params)); | 
|  | params->sleep_upon_construct = CONSTRUCT_SLEEP_TIME; | 
|  | params->sleep_upon_destruct = DESTRUCT_SLEEP_TIME; | 
|  |  | 
|  | /* We're going to want 10 blocks of data from our target rmm. */ | 
|  | rv = apr_reslist_create(&rl, RESLIST_MIN, RESLIST_SMAX, RESLIST_HMAX, | 
|  | RESLIST_TTL, my_constructor, my_destructor, | 
|  | params, p); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  |  | 
|  | for (i = 0; i < CONSUMER_THREADS; i++) { | 
|  | thread_info[i].tid = i; | 
|  | thread_info[i].tc = tc; | 
|  | thread_info[i].reslist = rl; | 
|  | thread_info[i].work_delay_sleep = WORK_DELAY_SLEEP_TIME; | 
|  | rv = apr_thread_pool_push(thrp, resource_consuming_thread, | 
|  | &thread_info[i], 0, NULL); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  |  | 
|  | rv = apr_thread_pool_destroy(thrp); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  |  | 
|  | test_timeout(tc, rl); | 
|  |  | 
|  | test_shrinking(tc, rl); | 
|  | ABTS_INT_EQUAL(tc, RESLIST_SMAX, params->c_count - params->d_count); | 
|  |  | 
|  | rv = apr_reslist_destroy(rl); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | } | 
|  |  | 
|  | static void test_reslist_no_ttl(abts_case *tc, void *data) | 
|  | { | 
|  | apr_status_t rv; | 
|  | apr_reslist_t *rl; | 
|  | my_parameters_t *params; | 
|  | my_resource_t *res; | 
|  |  | 
|  | /* Parameters (sleep not used) */ | 
|  | params = apr_pcalloc(p, sizeof(*params)); | 
|  |  | 
|  | rv = apr_reslist_create(&rl, | 
|  | /*no min*/0, /*no smax*/0, /*max*/1, /*no ttl*/0, | 
|  | my_constructor, my_destructor, params, p); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  |  | 
|  | /* Acquire/contruct one resource */ | 
|  | rv = apr_reslist_acquire(rl, (void **)&res); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | ABTS_INT_EQUAL(tc, 0, res->id); | 
|  |  | 
|  | /* Release it before next check */ | 
|  | rv = apr_reslist_release(rl, res); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  |  | 
|  | /* Re-acquire/release: the resource should be the same */ | 
|  | rv = apr_reslist_acquire(rl, (void **)&res); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | ABTS_INT_EQUAL(tc, 0, res->id); | 
|  |  | 
|  | /* Release it before cleanup */ | 
|  | rv = apr_reslist_release(rl, res); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  |  | 
|  | rv = apr_reslist_destroy(rl); | 
|  | ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); | 
|  | ABTS_INT_EQUAL(tc, params->d_count, 1); | 
|  | } | 
|  |  | 
|  | #endif /* APR_HAS_THREADS */ | 
|  |  | 
|  | abts_suite *testreslist(abts_suite *suite) | 
|  | { | 
|  | suite = ADD_SUITE(suite); | 
|  |  | 
|  | #if APR_HAS_THREADS | 
|  | abts_run_test(suite, test_reslist, NULL); | 
|  | abts_run_test(suite, test_reslist_no_ttl, NULL); | 
|  | #endif | 
|  |  | 
|  | return suite; | 
|  | } |