* Copyright (c) 2025 Huawei Technologies Co., Ltd.
* This program is free software, you can redistribute it and/or modify it under the terms and conditions of
* CANN Open Software License Agreement Version 2.0 (the "License").
* Please refer to the License for details. You may not use this file except in compliance with the License.
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED,
* INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY, OR FITNESS FOR A PARTICULAR PURPOSE.
* See LICENSE in the root of the software repository for the full text of the License.
*/
#ifndef STORE_OBJ_REF_H
#define STORE_OBJ_REF_H
#include <cstdint>
namespace shm {
namespace store {
class SmReferable {
public:
SmReferable() = default;
virtual ~SmReferable() = default;
inline void IncreaseRef()
{
__sync_fetch_and_add(&mRefCount, 1);
}
inline void DecreaseRef()
{
if (__sync_sub_and_fetch(&mRefCount, 1) == 0) {
delete this;
}
}
protected:
int32_t mRefCount = 0;
};
template <typename T> class SmRef {
public:
SmRef() noexcept = default;
SmRef(T *newObj) noexcept
{
if (newObj != nullptr) {
newObj->IncreaseRef();
mObj = newObj;
}
}
SmRef(const SmRef<T> &other) noexcept
{
if (other.mObj != nullptr) {
other.mObj->IncreaseRef();
mObj = other.mObj;
}
}
SmRef(SmRef<T> &&other) noexcept : mObj(std::__exchange(other.mObj, nullptr))
{
}
~SmRef()
{
if (mObj != nullptr) {
mObj->DecreaseRef();
}
}
inline SmRef<T> &operator = (T *newObj)
{
this->Set(newObj);
return *this;
}
inline SmRef<T> &operator = (const SmRef<T> &other)
{
if (this != &other) {
this->Set(other.mObj);
}
return *this;
}
SmRef<T> &operator = (SmRef<T> &&other) noexcept
{
if (this != &other) {
auto tmp = mObj;
mObj = std::__exchange(other.mObj, nullptr);
if (tmp != nullptr) {
tmp->DecreaseRef();
}
}
return *this;
}
inline bool operator == (const SmRef<T> &other) const
{
return mObj == other.mObj;
}
inline bool operator == (T *other) const
{
return mObj == other;
}
inline bool operator != (const SmRef<T> &other) const
{
return mObj != other.mObj;
}
inline bool operator != (T *other) const
{
return mObj != other;
}
inline T *operator->() const
{
return mObj;
}
inline T *Get() const
{
return mObj;
}
inline void Set(T *newObj)
{
if (newObj == mObj) {
return;
}
if (newObj != nullptr) {
newObj->IncreaseRef();
}
if (mObj != nullptr) {
mObj->DecreaseRef();
}
mObj = newObj;
}
private:
T *mObj = nullptr;
};
template <class Src, class Des> SmRef<Des> inline Convert(const SmRef<Src> &child)
{
Des *converted = dynamic_cast<Des *>(child.Get());
if (converted) {
return SmRef<Des>(converted);
}
return nullptr;
}
template <typename C, typename... ARGS> inline SmRef<C> SmMakeRef(ARGS... args)
{
return new (std::nothrow) C(args...);
}
}
}
#endif