blob: 9eec30412034cae794799175486969e89b3c0e1e [file] [log] [blame]
/*
* 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 <log4cxx/logstring.h>
#if !defined(LOG4CXX)
#define LOG4CXX 1
#endif
#include <log4cxx/helpers/aprinitializer.h>
#include <apr_pools.h>
#include <apr_atomic.h>
#include <assert.h>
#include <log4cxx/helpers/threadspecificdata.h>
#include <apr_thread_mutex.h>
#include <apr_thread_proc.h>
#include <log4cxx/helpers/filewatchdog.h>
#include <log4cxx/helpers/date.h>
using namespace log4cxx::helpers;
using namespace log4cxx;
bool APRInitializer::isDestructed = false;
struct APRInitializer::APRInitializerPrivate{
APRInitializerPrivate() :
p(0),
startTime(0),
tlsKey(0){
}
apr_pool_t* p;
std::mutex mutex;
std::list<FileWatchdog*> watchdogs;
log4cxx_time_t startTime;
apr_threadkey_t* tlsKey;
std::map<size_t, ObjectPtr> objects;
};
namespace
{
extern "C" void tlsDestruct(void* ptr)
{
delete ((ThreadSpecificData*) ptr);
}
// The first object created and the last object destroyed
struct apr_environment
{
apr_environment()
{
apr_initialize();
}
~apr_environment()
{
apr_terminate();
}
};
}
APRInitializer::APRInitializer() :
m_priv(std::make_unique<APRInitializerPrivate>())
{
apr_pool_create(&m_priv->p, NULL);
apr_atomic_init(m_priv->p);
m_priv->startTime = Date::currentTime();
#if APR_HAS_THREADS
apr_status_t stat = apr_threadkey_private_create(&m_priv->tlsKey, tlsDestruct, m_priv->p);
assert(stat == APR_SUCCESS);
assert(stat == APR_SUCCESS);
#endif
}
APRInitializer::~APRInitializer()
{
stopWatchDogs();
isDestructed = true;
#if APR_HAS_THREADS
std::unique_lock<std::mutex> lock(m_priv->mutex);
apr_threadkey_private_delete(m_priv->tlsKey);
#endif
}
void APRInitializer::stopWatchDogs()
{
#if APR_HAS_THREADS
std::unique_lock<std::mutex> lock(m_priv->mutex);
#endif
while (!m_priv->watchdogs.empty())
{
delete m_priv->watchdogs.back();
m_priv->watchdogs.pop_back();
}
}
void APRInitializer::unregisterAll()
{
getInstance().stopWatchDogs();
}
APRInitializer& APRInitializer::getInstance()
{
static apr_environment env;
static APRInitializer init;
return init;
}
log4cxx_time_t APRInitializer::initialize()
{
return getInstance().m_priv->startTime;
}
apr_pool_t* APRInitializer::getRootPool()
{
return getInstance().m_priv->p;
}
apr_threadkey_t* APRInitializer::getTlsKey()
{
return getInstance().m_priv->tlsKey;
}
void APRInitializer::registerCleanup(FileWatchdog* watchdog)
{
APRInitializer& instance(getInstance());
#if APR_HAS_THREADS
std::unique_lock<std::mutex> lock(instance.m_priv->mutex);
#endif
instance.m_priv->watchdogs.push_back(watchdog);
}
void APRInitializer::unregisterCleanup(FileWatchdog* watchdog)
{
APRInitializer& instance(getInstance());
#if APR_HAS_THREADS
std::unique_lock<std::mutex> lock(instance.m_priv->mutex);
#endif
for (std::list<FileWatchdog*>::iterator iter = instance.m_priv->watchdogs.begin();
iter != instance.m_priv->watchdogs.end();
iter++)
{
if (*iter == watchdog)
{
instance.m_priv->watchdogs.erase(iter);
return;
}
}
}
void APRInitializer::addObject(size_t key, const ObjectPtr& pObject)
{
#if APR_HAS_THREADS
std::unique_lock<std::mutex> lock(m_priv->mutex);
#endif
m_priv->objects[key] = pObject;
}
const ObjectPtr& APRInitializer::findOrAddObject(size_t key, std::function<ObjectPtr()> creator)
{
#if APR_HAS_THREADS
std::unique_lock<std::mutex> lock(m_priv->mutex);
#endif
auto pItem = m_priv->objects.find(key);
if (m_priv->objects.end() == pItem)
pItem = m_priv->objects.emplace(key, creator()).first;
return pItem->second;
}