#pragma once #include namespace glcr { template class SharedPtr { public: SharedPtr() : init_(false), ptr_(0), ref_cnt_(0) {} // Takes ownership. SharedPtr(T* ptr) { ptr_ = ptr; ref_cnt_ = new uint64_t(1); } SharedPtr(const SharedPtr& other) : init_(other.init_), ptr_(other.ptr_), ref_cnt_(other.ref_cnt_) { (*ref_cnt_)++; } SharedPtr& operator=(const SharedPtr& other) { Cleanup(); init_ = other.init_; ptr_ = other.ptr_; ref_cnt_ = other.ref_cnt_; (*ref_cnt_)++; return *this; } ~SharedPtr() { Cleanup(); } T& operator*() { return *ptr_; } const T& operator*() const { return *ptr_; } T* operator->() { return ptr_; } const T* operator->() const { return ptr_; } T* ptr() { return ptr_; } bool operator==(const SharedPtr& other) { return ptr_ == other.ptr_; } bool empty() { return !init_; } private: bool init_ = true; T* ptr_; uint64_t* ref_cnt_; void Cleanup() { if (!init_) { return; } if (--(*ref_cnt_) == 0) { delete ptr_; delete ref_cnt_; } } }; template SharedPtr MakeShared(A... args) { return {new T(args...)}; } } // namespace glcr