Commit object pool
This commit is contained in:
parent
666fd78172
commit
9dc12f8e50
6
CMakeLists.txt
Normal file
6
CMakeLists.txt
Normal file
@ -0,0 +1,6 @@
|
||||
cmake_minimum_required(VERSION 2.8)
|
||||
|
||||
project(ObjectPool)
|
||||
add_executable(${PROJECT_NAME} "main.cpp")
|
||||
|
||||
target_link_libraries(${PROJECT_NAME} -lpthread)
|
170
ObjectPool.h
Normal file
170
ObjectPool.h
Normal file
@ -0,0 +1,170 @@
|
||||
#ifndef OBJECTPOOL_H
|
||||
#define OBJECTPOOL_H
|
||||
|
||||
#include <boost/lockfree/queue.hpp>
|
||||
#include <functional>
|
||||
#include <mutex>
|
||||
#include <thread>
|
||||
#include <vector>
|
||||
|
||||
using AvailableObjectsQueueType =
|
||||
boost::lockfree::queue<std::size_t, boost::lockfree::fixed_sized<true>>;
|
||||
|
||||
template <typename T>
|
||||
class ObjectPool;
|
||||
|
||||
template <typename T>
|
||||
class BorrowedObject
|
||||
{
|
||||
T* obj;
|
||||
ObjectPool<T>* poolPtr;
|
||||
const std::size_t objectIndex;
|
||||
bool returned = false;
|
||||
|
||||
BorrowedObject() = delete;
|
||||
explicit BorrowedObject(T* Borrowed, ObjectPool<T>* PoolPtr, std::size_t ObjIndex)
|
||||
: obj(Borrowed), poolPtr(PoolPtr), objectIndex(ObjIndex)
|
||||
{
|
||||
}
|
||||
|
||||
public:
|
||||
friend ObjectPool<T>;
|
||||
void returnObj()
|
||||
{
|
||||
if (!returned && obj != nullptr) {
|
||||
poolPtr->availableObjectsQueue.push(objectIndex);
|
||||
poolPtr->availableObjectsCount++;
|
||||
returned = true;
|
||||
}
|
||||
}
|
||||
T* getObj() { return obj; }
|
||||
~BorrowedObject() { returnObj(); }
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
class ObjectPool
|
||||
{
|
||||
AvailableObjectsQueueType availableObjectsQueue;
|
||||
std::size_t objectCount;
|
||||
std::vector<T> objects;
|
||||
std::atomic_uint64_t availableObjectsCount;
|
||||
std::atomic_bool shutdownFlag;
|
||||
|
||||
public:
|
||||
friend BorrowedObject<T>;
|
||||
ObjectPool(std::size_t Size, std::function<T(std::size_t)> objectInitializers);
|
||||
ObjectPool() = delete;
|
||||
ObjectPool(const ObjectPool&) = delete;
|
||||
ObjectPool(ObjectPool&&) = delete;
|
||||
ObjectPool& operator=(const ObjectPool&) = delete;
|
||||
ObjectPool& operator=(ObjectPool&&) = delete;
|
||||
|
||||
BorrowedObject<T> borrowObj(uint64_t sleepTime_ms = 0);
|
||||
BorrowedObject<T> try_borrowObj(bool& success);
|
||||
std::size_t size() const;
|
||||
uint64_t getAvailableObjectsCount() const;
|
||||
std::vector<T>& getInternalObjects_unsafe();
|
||||
void shutdown();
|
||||
~ObjectPool();
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
uint64_t ObjectPool<T>::getAvailableObjectsCount() const
|
||||
{
|
||||
return availableObjectsCount.load();
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
ObjectPool<T>::ObjectPool(std::size_t Size, std::function<T(std::size_t)> objectInitializers)
|
||||
: availableObjectsQueue(Size)
|
||||
{
|
||||
shutdownFlag.store(false);
|
||||
for (std::size_t i = 0; i < Size; i++) {
|
||||
availableObjectsQueue.push(i);
|
||||
}
|
||||
objects.reserve(Size);
|
||||
for (std::size_t i = 0; i < Size; i++) {
|
||||
objects.push_back(objectInitializers(i));
|
||||
}
|
||||
objectCount = Size;
|
||||
availableObjectsCount.store(Size);
|
||||
}
|
||||
|
||||
/// Obj can be null only when shutting down
|
||||
template <typename T>
|
||||
BorrowedObject<T> ObjectPool<T>::borrowObj(uint64_t sleepTime_ms)
|
||||
{
|
||||
std::size_t currIndex = -1;
|
||||
bool isShuttingDown = false;
|
||||
while (!(isShuttingDown = shutdownFlag.load(std::memory_order_relaxed)) &&
|
||||
!availableObjectsQueue.pop(currIndex)) {
|
||||
if (sleepTime_ms) {
|
||||
std::this_thread::sleep_for(std::chrono::milliseconds(sleepTime_ms));
|
||||
}
|
||||
}
|
||||
if (!isShuttingDown) {
|
||||
availableObjectsCount--;
|
||||
BorrowedObject<T> res(&objects[currIndex], this, currIndex);
|
||||
assert(availableObjectsCount.load() >= 0);
|
||||
return res;
|
||||
} else {
|
||||
BorrowedObject<T> res(nullptr, this, currIndex);
|
||||
assert(availableObjectsCount.load() >= 0);
|
||||
return res;
|
||||
}
|
||||
}
|
||||
|
||||
/// Obj can be null when shutting down or if the object is not available
|
||||
template <typename T>
|
||||
BorrowedObject<T> ObjectPool<T>::try_borrowObj(bool& success)
|
||||
{
|
||||
std::size_t currIndex = -1;
|
||||
bool isShuttingDown = shutdownFlag.load(std::memory_order_relaxed);
|
||||
if (!isShuttingDown && availableObjectsQueue.pop(currIndex)) {
|
||||
success = true;
|
||||
} else {
|
||||
success = false;
|
||||
}
|
||||
if (!isShuttingDown && success) {
|
||||
availableObjectsCount--;
|
||||
BorrowedObject<T> res(&objects[currIndex], this, currIndex);
|
||||
assert(availableObjectsCount.load() >= 0);
|
||||
return res;
|
||||
} else {
|
||||
BorrowedObject<T> res(nullptr, this, currIndex);
|
||||
assert(availableObjectsCount.load() >= 0);
|
||||
return res;
|
||||
}
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
std::size_t ObjectPool<T>::size() const
|
||||
{
|
||||
return objectCount;
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
std::vector<T>& ObjectPool<T>::getInternalObjects_unsafe()
|
||||
{
|
||||
return objects;
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void ObjectPool<T>::shutdown()
|
||||
{
|
||||
// this can be called more than once
|
||||
shutdownFlag.store(true);
|
||||
while (availableObjectsCount != objectCount) {
|
||||
std::this_thread::sleep_for(std::chrono::milliseconds(10));
|
||||
}
|
||||
availableObjectsQueue.consume_all([](std::size_t&) {});
|
||||
objects.clear();
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
ObjectPool<T>::~ObjectPool()
|
||||
{
|
||||
shutdown();
|
||||
}
|
||||
|
||||
#endif // OBJECTPOOL_H
|
83
main.cpp
Normal file
83
main.cpp
Normal file
@ -0,0 +1,83 @@
|
||||
#include "ObjectPool.h"
|
||||
#include <cassert>
|
||||
|
||||
// this is a simple wrapper for gtest, you can move these tests to your gtests
|
||||
#define TEST(a,b)
|
||||
#define EXPECT_EQ(a__,b__) assert(a__ == b__);
|
||||
#define EXPECT_NE(a__,b__) assert(a__ != b__);
|
||||
#define EXPECT_TRUE(a__) assert(a__);
|
||||
#define EXPECT_TRUE(a__) assert(a__);
|
||||
#define EXPECT_FALSE(a__) assert(!a__);
|
||||
|
||||
int main() {
|
||||
TEST(Util, ObjectPool_stress_test)
|
||||
{
|
||||
std::size_t poolSize = 100;
|
||||
std::size_t num_threads = 1000;
|
||||
|
||||
std::vector<std::unique_ptr<std::thread>> threads(num_threads);
|
||||
|
||||
const int factor = 10;
|
||||
ObjectPool<int> p(poolSize, [](std::size_t z) { return factor * z; });
|
||||
for (std::unique_ptr<std::thread>& t : threads) {
|
||||
t.reset(new std::thread([&p]() {
|
||||
auto o = p.borrowObj();
|
||||
*o.getObj() = *o.getObj() + 1;
|
||||
}));
|
||||
}
|
||||
|
||||
for (std::unique_ptr<std::thread>& t : threads) {
|
||||
t->join();
|
||||
}
|
||||
|
||||
int expectedSum = num_threads; // every thread adds 1 to the sum
|
||||
int sum = 0;
|
||||
for (std::size_t i = 0; i < poolSize; i++) {
|
||||
expectedSum += factor * i; // initial value in pool member initializer
|
||||
sum += *p.borrowObj().getObj();
|
||||
}
|
||||
|
||||
EXPECT_EQ(sum, expectedSum);
|
||||
}
|
||||
|
||||
TEST(Util, ObjectPool_try_borrow)
|
||||
{
|
||||
std::size_t poolSize = 4;
|
||||
|
||||
const int factor = 10;
|
||||
ObjectPool<int> p(poolSize, [](std::size_t z) { return factor * z; });
|
||||
bool success = false;
|
||||
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 4);
|
||||
|
||||
BorrowedObject<int> o1 = p.try_borrowObj(success);
|
||||
EXPECT_TRUE(success);
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 3);
|
||||
EXPECT_NE(o1.getObj(), nullptr);
|
||||
|
||||
BorrowedObject<int> o2 = p.try_borrowObj(success);
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 2);
|
||||
EXPECT_TRUE(success);
|
||||
EXPECT_NE(o2.getObj(), nullptr);
|
||||
|
||||
BorrowedObject<int> o3 = p.try_borrowObj(success);
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 1);
|
||||
EXPECT_TRUE(success);
|
||||
EXPECT_NE(o3.getObj(), nullptr);
|
||||
|
||||
BorrowedObject<int> o4 = p.try_borrowObj(success);
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 0);
|
||||
EXPECT_TRUE(success);
|
||||
EXPECT_NE(o4.getObj(), nullptr);
|
||||
|
||||
BorrowedObject<int> o5 = p.try_borrowObj(success);
|
||||
EXPECT_FALSE(success);
|
||||
EXPECT_EQ(o5.getObj(), nullptr);
|
||||
|
||||
o5.returnObj();
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 0);
|
||||
|
||||
o4.returnObj();
|
||||
EXPECT_EQ(p.getAvailableObjectsCount(), 1);
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user