ScratchBuffer.h#
Fully qualified name: omni/extras/ScratchBuffer.h
File members: omni/extras/ScratchBuffer.h
// SPDX-FileCopyrightText: Copyright (c) 2021-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
// SPDX-License-Identifier: LicenseRef-NvidiaProprietary
//
// NVIDIA CORPORATION, its affiliates and licensors retain all intellectual
// property and proprietary rights in and to this material, related
// documentation and any modifications thereto. Any use, reproduction,
// disclosure or distribution of this material and related documentation
// without an express license agreement from NVIDIA CORPORATION or
// its affiliates is strictly prohibited.
#pragma once
#include "../../carb/Defines.h"
#include <algorithm>
#include <iterator>
namespace omni
{
namespace extras
{
#ifdef DOXYGEN_BUILD // Sphinx does not like the computation and reports errors, so special case for documentation.
template <typename T, size_t BaseSize_ = 16, size_t ShrinkThreshold_ = 100>
#else
template <typename T, size_t BaseSize_ = CARB_MAX(512ull / sizeof(T), 16ull), size_t ShrinkThreshold_ = 100>
#endif
class ScratchBuffer
{
public:
    using DataType = T;
    static constexpr size_t BaseSize = BaseSize_;
    ScratchBuffer()
    {
        m_dynamicArray = m_localArray;
        m_size = BaseSize;
        m_capacity = BaseSize;
    }
    ScratchBuffer(const ScratchBuffer& right) : ScratchBuffer()
    {
        *this = right;
    }
    ScratchBuffer(ScratchBuffer&&) = delete;
    ~ScratchBuffer()
    {
        if (m_dynamicArray != nullptr && m_dynamicArray != m_localArray)
            delete[] m_dynamicArray;
    }
    ScratchBuffer& operator=(const ScratchBuffer& right)
    {
        if (&right == this)
            return *this;
        if (!resize(right.m_size))
            return *this;
        std::copy_n(right.m_dynamicArray, m_size, m_dynamicArray);
        return *this;
    }
    ScratchBuffer& operator=(ScratchBuffer&&) = delete;
    T& operator[](size_t index)
    {
        return m_dynamicArray[index];
    }
    T* data()
    {
        return m_dynamicArray;
    }
    const T* data() const
    {
        return m_dynamicArray;
    }
    size_t size() const
    {
        return m_size;
    }
    bool resize(size_t count)
    {
        T* tmp = m_localArray;
        size_t copyCount;
        if (count == 0)
            return true;
        // the buffer is already the requested size -> nothing to do => succeed.
        if (count == m_size)
            return true;
        // the buffer is already large enough => don't resize unless the change is drastic.
        if (count <= m_capacity)
        {
            if ((count * 100) >= ShrinkThreshold_ * m_size)
            {
                m_size = count;
                return true;
            }
        }
        if (count > BaseSize)
        {
            tmp = new (std::nothrow) T[count];
            if (tmp == nullptr)
                return false;
        }
        // the buffer didn't change -> nothing to copy => update parameters and succeed.
        if (tmp == m_dynamicArray)
        {
            m_size = count;
            m_capacity = BaseSize;
            return true;
        }
        copyCount = m_size;
        if (m_size > count)
            copyCount = count;
#ifdef __CUDACC__ // Silence CUDA warning #128-D: loop is not reachable
        for (size_t i = 0; i != copyCount; ++i)
            tmp[i] = std::move(m_dynamicArray[i]);
#else
        std::copy_n(std::make_move_iterator(m_dynamicArray), copyCount, tmp);
#endif
        if (m_dynamicArray != nullptr && m_dynamicArray != m_localArray)
            delete[] m_dynamicArray;
        m_dynamicArray = tmp;
        m_size = count;
        m_capacity = count;
        return true;
    }
private:
    T m_localArray[BaseSize];
    T* m_dynamicArray;
    size_t m_size = BaseSize;
    size_t m_capacity = BaseSize;
};
} // namespace extras
} // namespace omni