165 lines
4.1 KiB
C++
165 lines
4.1 KiB
C++
/*****************************************************************************
|
|
*
|
|
* This file is part of Mapnik (c++ mapping toolkit)
|
|
*
|
|
* Copyright (C) 2014 Artem Pavlenko
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#ifndef MAPNIK_POOL_HPP
|
|
#define MAPNIK_POOL_HPP
|
|
|
|
// mapnik
|
|
#include <mapnik/unique_lock.hpp>
|
|
#include <mapnik/utils.hpp>
|
|
#include <mapnik/util/noncopyable.hpp>
|
|
|
|
// boost
|
|
#include <memory>
|
|
#ifdef MAPNIK_THREADSAFE
|
|
#include <mutex>
|
|
#endif
|
|
|
|
// stl
|
|
#include <algorithm> // std::max
|
|
#include <deque>
|
|
|
|
namespace mapnik
|
|
{
|
|
template <typename T,template <typename> class Creator>
|
|
class Pool : private util::noncopyable
|
|
{
|
|
using HolderType = std::shared_ptr<T>;
|
|
using ContType = std::deque<HolderType>;
|
|
|
|
Creator<T> creator_;
|
|
unsigned initialSize_;
|
|
unsigned maxSize_;
|
|
ContType pool_;
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mutable std::mutex mutex_;
|
|
#endif
|
|
public:
|
|
|
|
Pool(const Creator<T>& creator,unsigned initialSize, unsigned maxSize)
|
|
:creator_(creator),
|
|
initialSize_(initialSize),
|
|
maxSize_(maxSize)
|
|
{
|
|
for (unsigned i=0; i < initialSize_; ++i)
|
|
{
|
|
HolderType conn(creator_());
|
|
if (conn->isOK())
|
|
pool_.push_back(conn);
|
|
}
|
|
}
|
|
|
|
HolderType borrowObject()
|
|
{
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mapnik::scoped_lock lock(mutex_);
|
|
#endif
|
|
|
|
typename ContType::iterator itr=pool_.begin();
|
|
while ( itr!=pool_.end())
|
|
{
|
|
if (!itr->unique())
|
|
{
|
|
++itr;
|
|
}
|
|
else if ((*itr)->isOK())
|
|
{
|
|
return *itr;
|
|
}
|
|
else
|
|
{
|
|
itr=pool_.erase(itr);
|
|
}
|
|
}
|
|
// all connection have been taken, check if we allowed to grow pool
|
|
if (pool_.size() < maxSize_)
|
|
{
|
|
HolderType conn(creator_());
|
|
if (conn->isOK())
|
|
{
|
|
pool_.push_back(conn);
|
|
return conn;
|
|
}
|
|
}
|
|
return HolderType();
|
|
}
|
|
|
|
unsigned size() const
|
|
{
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mapnik::scoped_lock lock(mutex_);
|
|
#endif
|
|
return pool_.size();
|
|
}
|
|
|
|
unsigned max_size() const
|
|
{
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mapnik::scoped_lock lock(mutex_);
|
|
#endif
|
|
return maxSize_;
|
|
}
|
|
|
|
void set_max_size(unsigned size)
|
|
{
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mapnik::scoped_lock lock(mutex_);
|
|
#endif
|
|
maxSize_ = std::max(maxSize_,size);
|
|
}
|
|
|
|
unsigned initial_size() const
|
|
{
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mapnik::scoped_lock lock(mutex_);
|
|
#endif
|
|
return initialSize_;
|
|
}
|
|
|
|
void set_initial_size(unsigned size)
|
|
{
|
|
#ifdef MAPNIK_THREADSAFE
|
|
mapnik::scoped_lock lock(mutex_);
|
|
#endif
|
|
if (size > initialSize_)
|
|
{
|
|
initialSize_ = size;
|
|
unsigned total_size = pool_.size();
|
|
// ensure we don't have ghost obj's in the pool.
|
|
if (total_size < initialSize_)
|
|
{
|
|
unsigned grow_size = initialSize_ - total_size ;
|
|
|
|
for (unsigned i=0; i < grow_size; ++i)
|
|
{
|
|
HolderType conn(creator_());
|
|
if (conn->isOK())
|
|
pool_.push_back(conn);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
#endif // MAPNIK_POOL_HPP
|