2015-11-23 04:35:40 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
|
|
|
|
#include "utils/crtp.hpp"
|
|
|
|
#include "threading/sync/lockable.hpp"
|
|
|
|
|
|
|
|
template <class Derived>
|
2015-12-06 23:42:47 +08:00
|
|
|
class LazyGC : Crtp<Derived>
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
void add_ref()
|
|
|
|
{
|
|
|
|
ref_count.fetch_add(1, std::memory_order_relaxed);
|
|
|
|
}
|
|
|
|
|
|
|
|
void release_ref()
|
|
|
|
{
|
|
|
|
// get refcount and subtract atomically
|
|
|
|
auto count = ref_count.fetch_sub(1, std::memory_order_acq_rel);
|
|
|
|
|
|
|
|
// fetch_sub first returns and then subtrarcts so the refcount is
|
|
|
|
// zero when fetch_sub returns 1
|
|
|
|
if(count != 1)
|
|
|
|
return;
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
if(!dirty.load(std::memory_order_acquire))
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto guard = this->derived().gc_lock_acquire();
|
|
|
|
|
|
|
|
if(!dirty.load(std::memory_order_acquire))
|
|
|
|
return;
|
2015-11-23 04:35:40 +08:00
|
|
|
|
|
|
|
this->derived().vacuum();
|
|
|
|
}
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
protected:
|
2015-11-23 04:35:40 +08:00
|
|
|
std::atomic<int> ref_count {0};
|
2015-12-06 23:42:47 +08:00
|
|
|
std::atomic<bool> dirty {false};
|
2015-11-23 04:35:40 +08:00
|
|
|
};
|