/* 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 #include #include "apr_general.h" #include "apu.h" #include "apr_reslist.h" #include "apr_thread_pool.h" #if APR_HAVE_TIME_H #include #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(3500) /* 3.5 ms */ #define CONSUMER_THREADS 25 #define CONSUMER_ITERATIONS 100 #define CONSTRUCT_SLEEP_TIME APR_TIME_C(2500) /* 2.5 ms */ #define DESTRUCT_SLEEP_TIME APR_TIME_C(1000) /* 1.0 ms */ #define WORK_DELAY_SLEEP_TIME APR_TIME_C(1500) /* 1.5 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; int acquire_flags; } 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_ex(rl, &vp, thread_info->acquire_flags); 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, int acquire_flags) { 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_ex(rl, (void**)&resources[i], acquire_flags); ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); } /* next call will block until timeout is reached */ rv = apr_reslist_acquire_ex(rl, &vp, acquire_flags); 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, int acquire_flags) { 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_ex(rl, (void**)&resources[i], acquire_flags); 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_invalidate(rl, resources[i]); ABTS_INT_EQUAL(tc, APR_SUCCESS, rv); } for (i = 0; i < RESLIST_HMAX; i++) { rv = apr_reslist_acquire_ex(rl, &vp, acquire_flags); 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]; int acquire_flags = (int)(apr_uintptr_t)data; 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; thread_info[i].acquire_flags = acquire_flags; 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, acquire_flags); test_shrinking(tc, rl, acquire_flags); 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, (void*)(apr_uintptr_t)APR_RESLIST_ACQUIRE_LIFO); abts_run_test(suite, test_reslist, (void*)(apr_uintptr_t)APR_RESLIST_ACQUIRE_FIFO); abts_run_test(suite, test_reslist_no_ttl, NULL); #endif return suite; }