Memory.h#
Fully qualified name: carb/cpp/Memory.h
File members: carb/cpp/Memory.h
// Copyright (c) 2023-2025, NVIDIA CORPORATION. All rights reserved.
//
// NVIDIA CORPORATION and its licensors retain all intellectual property
// and proprietary rights in and to this software, related documentation
// and any modifications thereto. Any use, reproduction, disclosure or
// distribution of this software and related documentation without an express
// license agreement from NVIDIA CORPORATION is strictly prohibited.
//
#pragma once
#include "../Defines.h"
#include <memory>
#include <type_traits>
namespace carb
{
namespace cpp
{
namespace detail
{
template <class T, class = void>
struct IsMSVCIterator : std::false_type
{
};
// OVCC-1775: MSVC iterators assert in debug if you try to take the address of them.
#if CARB_COMPILER_MSC
template <class T>
struct IsMSVCIterator<T, std::void_t<decltype(std::declval<T>()._Unwrapped())>>
: public std::is_pointer<decltype(std::declval<T>()._Unwrapped())>
{
};
#endif
template <class T>
inline constexpr bool IsMSVCIterator_v = IsMSVCIterator<T>::value;
} // namespace detail
template <typename T>
constexpr std::enable_if_t<!std::is_array<T>::value> destroy_at(T* const p) noexcept(std::is_nothrow_destructible<T>::value)
{
p->~T();
}
template <typename T, std::size_t N>
constexpr void destroy_at(T (*array)[N]) noexcept
{
static_assert(noexcept(carb::cpp::destroy_at(array[0])),
"destroy_at for array requires elements to have noexcept destructor");
for (T& elem : *array)
{
carb::cpp::destroy_at(std::addressof(elem));
}
}
#if !CARB_HAS_CPP20 || defined DOXYGEN_BUILD
template <typename T, typename... TArgs>
T* construct_at(T* place, TArgs&&... args) noexcept
CARB_NO_DOC((noexcept(::new (static_cast<void*>(place)) T(std::forward<TArgs>(args)...))))
{
return ::new (static_cast<void*>(place)) T(std::forward<TArgs>(args)...);
}
template <class T>
constexpr T* to_address(T* p) noexcept
{
static_assert(!std::is_function_v<T>, "Ill-formed if T is a function type");
return p;
}
# ifndef DOXYGEN_BUILD
template <class Ptr, std::enable_if_t<detail::IsMSVCIterator_v<Ptr>, bool> = false>
constexpr auto to_address(const Ptr& p) noexcept
{
return p._Unwrapped();
}
# endif
CARB_NO_DOC(
template <
class Ptr,
std::enable_if_t<!detail::IsMSVCIterator_v<Ptr> && std::is_pointer_v<decltype(std::declval<Ptr>().operator->())>, bool> = false>
)
constexpr auto to_address(const Ptr& p) noexcept
{
return to_address(p.operator->());
}
#else
using std::construct_at;
using std::to_address;
#endif
} // namespace cpp
} // namespace carb