#pragma once
#include <initializer_list>
#include <memory>
#include <type_traits>
#include "rocksdb/rocksdb_namespace.h"
namespace ROCKSDB_NAMESPACE {
template <class DestClass, class SrcClass>
inline DestClass* static_cast_with_check(SrcClass* x) {
DestClass* ret = static_cast<DestClass*>(x);
#ifdef ROCKSDB_USE_RTTI
assert(ret == dynamic_cast<DestClass*>(x));
#endif
return ret;
}
template <class DestClass, class SrcClass>
inline std::shared_ptr<DestClass> static_cast_with_check(
std::shared_ptr<SrcClass>&& x) {
#if defined(ROCKSDB_USE_RTTI) && !defined(NDEBUG)
auto orig_raw = x.get();
#endif
auto ret = std::static_pointer_cast<DestClass>(std::move(x));
#if defined(ROCKSDB_USE_RTTI) && !defined(NDEBUG)
assert(ret.get() == dynamic_cast<DestClass*>(orig_raw));
#endif
return ret;
}
template <typename To, typename From>
inline To lossless_cast(From x) {
using FromValue = typename std::remove_reference_t<From>;
if constexpr (std::is_pointer_v<FromValue>) {
static_assert(std::is_pointer_v<To>);
using FromDeref = typename std::remove_pointer_t<FromValue>;
using ToDeref = typename std::remove_pointer_t<To>;
static_assert(std::is_integral_v<FromDeref> || std::is_enum_v<FromDeref>,
"Only works on integral types");
static_assert(std::is_integral_v<ToDeref> || std::is_enum_v<To>,
"Only works on integral types");
static_assert(sizeof(ToDeref) == sizeof(FromDeref), "Must be lossless");
return reinterpret_cast<To>(x);
} else {
static_assert(std::is_integral_v<FromValue> || std::is_enum_v<FromValue>,
"Only works on integral types");
static_assert(std::is_integral_v<To> || std::is_enum_v<To>,
"Only works on integral types");
static_assert(sizeof(To) >= sizeof(FromValue), "Must be lossless");
return static_cast<To>(x);
}
}
template <typename T>
inline const std::initializer_list<T>& List(
const std::initializer_list<T>& list) {
return list;
}
template <typename T>
class UnownedPtr {
public:
UnownedPtr() = default;
UnownedPtr(std::nullptr_t) {}
UnownedPtr(T* ptr) : ptr_(ptr) {}
UnownedPtr(const UnownedPtr&) = default;
UnownedPtr(UnownedPtr&&) = default;
UnownedPtr& operator=(const UnownedPtr&) = default;
UnownedPtr& operator=(UnownedPtr&&) = default;
T* get() const { return ptr_; }
T* operator->() const { return ptr_; }
T& operator*() const { return *ptr_; }
operator bool() const { return ptr_ != nullptr; }
private:
T* ptr_ = nullptr;
};
}