blob: 12096d9f4a5c743c36c11bffa7a0d67dc7283267 [file] [log] [blame]
/*
* root_pools.c : Implement svn_root_pools__* API
*
* ====================================================================
* 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 "svn_pools.h"
#include "private/svn_subr_private.h"
#include "private/svn_mutex.h"
struct svn_root_pools__t
{
/* unused pools.
* Use MUTEX to serialize access to this collection.
*/
apr_array_header_t *unused_pools;
/* Mutex to serialize access to UNUSED_POOLS */
svn_mutex__t *mutex;
};
svn_error_t *
svn_root_pools__create(svn_root_pools__t **pools)
{
/* the collection of root pools must be managed independently from
any other pool */
apr_pool_t *pool
= apr_allocator_owner_get(svn_pool_create_allocator(FALSE));
/* construct result object */
svn_root_pools__t *result = apr_pcalloc(pool, sizeof(*result));
SVN_ERR(svn_mutex__init(&result->mutex, TRUE, pool));
result->unused_pools = apr_array_make(pool, 16, sizeof(apr_pool_t *));
/* done */
*pools = result;
return SVN_NO_ERROR;
}
/* Return a currently unused connection pool in *POOL. If no such pool
* exists, create a new root pool and return that in *POOL.
*/
static svn_error_t *
acquire_pool_internal(apr_pool_t **pool,
svn_root_pools__t *pools)
{
SVN_ERR(svn_mutex__lock(pools->mutex));
*pool = pools->unused_pools->nelts
? *(apr_pool_t **)apr_array_pop(pools->unused_pools)
: apr_allocator_owner_get(svn_pool_create_allocator(FALSE));
SVN_ERR(svn_mutex__unlock(pools->mutex, SVN_NO_ERROR));
return SVN_NO_ERROR;
}
apr_pool_t *
svn_root_pools__acquire_pool(svn_root_pools__t *pools)
{
apr_pool_t *pool;
svn_error_t *err = acquire_pool_internal(&pool, pools);
if (err)
{
/* Mutex failure?! Well, try to continue with unrecycled data. */
svn_error_clear(err);
pool = apr_allocator_owner_get(svn_pool_create_allocator(FALSE));
}
return pool;
}
void
svn_root_pools__release_pool(apr_pool_t *pool,
svn_root_pools__t *pools)
{
svn_error_t *err;
svn_pool_clear(pool);
err = svn_mutex__lock(pools->mutex);
if (err)
{
svn_error_clear(err);
svn_pool_destroy(pool);
}
else
{
APR_ARRAY_PUSH(pools->unused_pools, apr_pool_t *) = pool;
svn_error_clear(svn_mutex__unlock(pools->mutex, SVN_NO_ERROR));
}
}