include/photon/common/expirecontainer.h (364 lines of code) (raw):
/*
Copyright 2022 The Photon Authors
Licensed 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.
*/
#pragma once
#include <photon/common/object.h>
#include <photon/common/string_view.h>
#include <photon/common/timeout.h>
#include <photon/common/utility.h>
#include <photon/thread/list.h>
#include <photon/thread/thread.h>
#include <photon/thread/timer.h>
#include <algorithm>
#include <tuple>
#include <type_traits>
#include <unordered_set>
#include <utility>
/**
* @brief ExpireContainerBase is basic class for all expire containers
* provides index for items and a release list for item recycle.
*
*/
class ExpireContainerBase : public Object {
protected:
class Item : public intrusive_list_node<Item> {
protected:
Item() : _timeout(0) {}
public:
photon::Timeout _timeout;
virtual ~Item() {}
virtual size_t key_hash() const = 0;
virtual bool key_equal(const Item* rhs) const = 0;
virtual Item* construct() const = 0;
};
template <typename BaseItem, typename KeyType>
class KeyedItem : public BaseItem {
public:
constexpr static bool _is_string_key =
std::is_base_of<std::string, KeyType>::value ||
std::is_base_of<std::string_view, KeyType>::value ||
std::is_same<const char*, const KeyType>::value;
using ItemKey = KeyType;
using InterfaceKey =
typename std::conditional<_is_string_key, std::string_view,
ItemKey>::type;
ItemKey _key;
KeyedItem(const InterfaceKey& key) : _key(key) {}
virtual size_t key_hash() const override {
return std::hash<ItemKey>()(_key);
}
virtual bool key_equal(const Item* rhs) const override {
return _key == static_cast<const KeyedItem*>(rhs)->_key;
}
virtual KeyedItem* construct() const override {
return new KeyedItem(_key);
}
const ItemKey& key() { return _key; }
};
intrusive_list<Item> _list;
uint64_t _lifespan;
photon::Timer _timer;
photon::spinlock _lock; // protect _list/_set operations
using ItemPtr = Item*;
struct ItemHash {
size_t operator()(const ItemPtr& x) const { return x->key_hash(); }
};
struct ItemEqual {
size_t operator()(const ItemPtr& x, const ItemPtr& y) const {
return x->key_equal(y);
}
};
using Set = std::unordered_set<ItemPtr, ItemHash, ItemEqual>;
Set _set;
ExpireContainerBase(uint64_t lifespan, uint64_t timer_cycle);
~ExpireContainerBase() { clear(); }
using iterator = decltype(_set)::iterator;
std::pair<iterator, bool> insert(Item* item);
iterator begin() { return _set.begin(); }
iterator end() { return _set.end(); }
iterator find(const Item& key_item);
iterator __find_prelock(const Item& key_item);
template <typename T>
struct TypedIterator : public iterator {
TypedIterator(const iterator& rhs) : iterator(rhs) {}
using TPtr = T*;
TPtr operator*() const { return (TPtr)iterator::operator*(); }
TPtr* operator->() const { return (TPtr*)iterator::operator->(); }
};
bool keep_alive(const Item& item, bool insert_if_not_exists);
void enqueue(Item* item) {
_list.pop(item);
item->_timeout.timeout(_lifespan);
_list.push_back(item);
}
public:
void clear();
uint64_t expire();
size_t size() { return _set.size(); }
size_t lifespan() { return _lifespan; }
[[deprecated("use lifespan() instead")]]
size_t expiration() { return _lifespan; }
};
template <typename KeyType, typename... Ts>
class ExpireContainer : public ExpireContainerBase {
public:
using Base = ExpireContainerBase;
ExpireContainer(uint64_t expiration) : Base(expiration, expiration / 16) {}
ExpireContainer(uint64_t expiration, uint64_t timer_cycle)
: Base(expiration, timer_cycle) {}
protected:
using KeyedItem = typename Base::KeyedItem<Base::Item, KeyType>;
class Item : public KeyedItem {
public:
std::tuple<Ts...> payload;
using typename KeyedItem::InterfaceKey;
using typename KeyedItem::ItemKey;
using iterator = typename intrusive_list_node<Item>::iterator;
template <typename... Gs>
Item(const InterfaceKey& key, Gs&&... gs)
: KeyedItem(key), payload(std::forward<Gs>(gs)...) {}
template <size_t idx,
typename = typename std::enable_if<(idx > 0)>::type>
decltype(auto) get_payload() {
return std::get<idx - 1>(payload);
}
template <size_t idx,
typename = typename std::enable_if<(idx == 0)>::type>
InterfaceKey get_payload() {
return KeyedItem::_key;
}
};
intrusive_list<Item>& list() { return (intrusive_list<Item>&)_list; }
public:
using ItemKey = typename Item::ItemKey;
using InterfaceKey = typename Item::InterfaceKey;
using iterator = typename ExpireContainerBase::TypedIterator<Item>;
iterator begin() { return Base::begin(); }
iterator end() { return Base::end(); }
iterator find(const InterfaceKey& key) {
return Base::find(KeyedItem(key));
}
template <typename... Gs>
iterator insert(const InterfaceKey& key, Gs&&... xs) {
auto item = new Item(key, std::forward<Gs>(xs)...);
SCOPED_LOCK(_lock);
auto pr = Base::insert(item);
if (!pr.second) {
delete item;
return end();
}
enqueue(item);
return pr.first;
}
void refresh(Item* item) {
DEFER(expire());
SCOPED_LOCK(_lock);
enqueue(item);
}
};
// a set / list like structure
// able to query whether an item not expired in it.
template <typename T>
class ExpireList : public ExpireContainer<T> {
public:
using Base = ExpireContainer<T>;
using Base::Base;
using typename Base::Item;
bool keep_alive(const T &x, bool insert_if_not_exists) {
return Base::keep_alive(Item(x), insert_if_not_exists);
}
};
class ObjectCacheBase : public ExpireContainerBase {
protected:
using Base = ExpireContainerBase;
using Base::Base;
using Base::KeyedItem;
class Item : public Base::Item {
public:
Item() : Base::Item() {
_obj = nullptr;
_refcnt = 0;
_recycle = nullptr;
_failure = 0;
}
void* _obj;
photon::mutex _mtx;
uint32_t _refcnt;
photon::semaphore* _recycle;
uint64_t _failure;
};
photon::condition_variable blocker;
using ItemPtr = Item*;
// in case of missing, ref_acquire() performs a 2 phase construction:
// (1) creating an item with _obj==nullptr, preventing
// concurrent construction of objects with the same key;
// (2) construction of the object itself, and possibly do
// clean-up in case of failure
Item* ref_acquire(const Item& key_item, Delegate<void, void*> ctor,
uint64_t failure_cooldown = 0);
void* ref_release(ItemPtr item, bool recycle, bool destroy);
void* acquire(const Item& key_item, Delegate<void, void*> ctor,
uint64_t failure_cooldown = 0) {
auto ret = ref_acquire(key_item, ctor, failure_cooldown);
return ret ? ret->_obj : nullptr;
}
// the argument `key` plays the roles of (type-erased) key
void* release(const Item& key_item, bool recycle, bool destroy);
public:
template <typename KeyType, typename ValType>
class PtrItem : public KeyedItem<Item, KeyType> {
public:
using KeyedItem<Item, KeyType>::KeyedItem;
using typename KeyedItem<Item, KeyType>::InterfaceKey;
using ValPtr = ValType;
using ValEntity = typename std::remove_pointer<ValPtr>::type;
virtual PtrItem* construct() const override {
auto item = new PtrItem(this->_key);
item->_obj = nullptr;
item->_refcnt = 0;
item->_recycle = nullptr;
return item;
}
~PtrItem() override { delete (ValPtr)this->_obj; }
ValPtr get_ptr() { return (ValPtr)this->_obj; }
ValEntity& get_ref() { return *(ValPtr)this->_obj; }
static ValPtr get_content(PtrItem* item) {
return item ? item->get_ptr() : nullptr;
}
static ValPtr create_default() { return new ValEntity(); }
template <typename Ctor>
static decltype(auto) initialize(const Ctor& ctor) {
return [&ctor](void* arg) { ((PtrItem*)arg)->_obj = ctor(); };
}
};
template <typename KeyType, typename ValType>
class ListItem : public KeyedItem<Item, KeyType> {
public:
using ValPtr = ValType*;
using ValEntity = ValType;
ValEntity _list;
using KeyedItem<Item, KeyType>::KeyedItem;
using typename KeyedItem<Item, KeyType>::InterfaceKey;
virtual ListItem* construct() const override {
auto item = new ListItem(this->_key);
item->_obj = nullptr;
item->_refcnt = 0;
item->_recycle = nullptr;
return item;
}
~ListItem() { _list.delete_all(); }
ValPtr get_ptr() { return &this->_list; }
ValEntity& get_ref() { return this->_list; }
static ValEntity& get_content(ListItem* item) {
return item->get_ref();
}
static ValEntity create_default() { return ValEntity(); }
template <typename Ctor>
static decltype(auto) initialize(const Ctor& ctor) {
return [&ctor](void* arg) {
((ListItem*)arg)->_list = ctor();
((ListItem*)arg)->_obj = arg;
};
}
};
template <typename ObjectCache>
class Borrow {
public:
using Item = typename ObjectCache::Item;
ObjectCache* _oc;
Item* _ref;
bool _recycle = false;
bool _moveout = false;
Borrow(ObjectCache* oc, Item* ref, bool recycle)
: _oc(oc), _ref(ref), _recycle(recycle) {}
~Borrow() {
if (_ref) _oc->ref_release(_ref, _recycle, !_moveout);
}
Borrow() = delete;
Borrow(const Borrow&) = delete;
Borrow(Borrow&& rhs) { move(std::move(rhs)); }
void operator=(const Borrow&) = delete;
void operator=(Borrow&& rhs) { move(rhs); }
operator bool() const { return _ref; }
bool recycle() const { return _recycle; }
bool recycle(bool x) { return _recycle = x; }
typename Item::ValPtr operator->() { return _ref->get_ptr(); }
typename Item::ValEntity& operator*() { return _ref->get_ref(); }
bool moved() const { return _moveout; }
bool moveout(bool x) { return _moveout = x; }
protected:
void move(Borrow&& rhs) {
_oc = rhs._oc;
rhs._oc = nullptr;
_ref = rhs._ref;
rhs._ref = nullptr;
_recycle = rhs._recycle;
}
};
};
// Resource pool based on reference count
// when the pool is fulled, it will try to remove items which can be sure is not
// referenced the base m_list works as gc list when object acquired, construct
// or findout the object, add reference count; when object release, reduce
// refcount. if some resource is not referenced, it will be put back to gc list
// waiting to release.
template <typename KeyType, typename ValType, typename ItemType>
class __ObjectCache : public ObjectCacheBase {
public:
using Base = ObjectCacheBase;
using Item = ItemType;
using KeyedItem = Base::KeyedItem<Base::Item, KeyType>;
using InterfaceKey = typename Item::InterfaceKey;
using ItemPtr = Item*;
using ValEntity = typename Item::ValEntity;
using Borrow = typename Base::Borrow<__ObjectCache>;
__ObjectCache(uint64_t expiration) : Base(expiration, expiration / 16) {}
__ObjectCache(uint64_t expiration, uint64_t timer_cycle)
: Base(expiration, timer_cycle) {}
template <typename Constructor>
ItemPtr ref_acquire(const InterfaceKey& key, const Constructor& ctor,
uint64_t failure_cooldown = 0) {
auto _ctor = Item::initialize(ctor);
return (ItemPtr)Base::ref_acquire(Item(key), _ctor, failure_cooldown);
}
template <typename Constructor>
decltype(auto) acquire(const InterfaceKey& key, const Constructor& ctor,
uint64_t failure_cooldown = 0) {
return Item::get_content(ref_acquire(key, ctor, failure_cooldown));
}
using iterator = typename ExpireContainerBase::TypedIterator<Item>;
iterator begin() { return Base::begin(); }
iterator end() { return Base::end(); }
iterator find(const InterfaceKey& key) {
return Base::find(KeyedItem(key));
}
// Borrow has defined a bool operator to indicate if ref_acquire is succeeded.
// Users should take care of the error handling if (!borrow_result)
template <typename Constructor>
Borrow borrow(const typename Item::InterfaceKey& key,
const Constructor& ctor, uint64_t failure_cooldown = 0) {
return Borrow(
this,
((__ObjectCache*)this)->ref_acquire(key, ctor, failure_cooldown),
false);
}
Borrow borrow(const typename Item::InterfaceKey& key) {
return borrow(key, &Item::create_default);
}
};
template <typename KeyType, typename ValPtr>
class ObjectCache
: public __ObjectCache<KeyType, ValPtr,
ObjectCacheBase::PtrItem<KeyType, ValPtr>> {
public:
using Base = __ObjectCache<KeyType, ValPtr,
ObjectCacheBase::PtrItem<KeyType, ValPtr>>;
using Base::Base;
ValPtr ref_release(typename Base::ItemPtr item, bool recycle = false,
bool destroy = true) {
return reinterpret_cast<ValPtr>(
Base::ref_release(item, recycle, destroy));
}
ValPtr release(const typename Base::InterfaceKey& key, bool recycle = false,
bool destroy = true) {
return reinterpret_cast<ValPtr>(
Base::release(typename Base::Item(key), recycle, destroy));
}
};
template <typename KeyType, typename NodeType>
class ObjectCache<KeyType, intrusive_list<NodeType>>
: public __ObjectCache<
KeyType, intrusive_list<NodeType>,
ObjectCacheBase::ListItem<KeyType, intrusive_list<NodeType>>> {
public:
using Base = __ObjectCache<
KeyType, intrusive_list<NodeType>,
ObjectCacheBase::ListItem<KeyType, intrusive_list<NodeType>>>;
using Base::Base;
NodeType* ref_release(typename Base::ItemPtr item, bool recycle = false,
bool destroy = true) {
return reinterpret_cast<NodeType*>(
Base::ref_release(item, recycle, destroy));
}
NodeType* release(const typename Base::InterfaceKey& key,
bool recycle = false, bool destroy = true) {
return reinterpret_cast<NodeType*>(
Base::release(typename Base::Item(key), recycle, destroy));
}
};