libcruft-util/alloc/raw/linear.cpp
Danny Robson 7af076e2de alloc: prefer std::byte representations for iterators
this allows the users to more easily walk the byte ranges (or perform
simply pointer arithmetic), without as much danger of using the values
in an expression inadvertantly.
2018-02-28 17:55:56 +11:00

167 lines
3.6 KiB
C++

/*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* Copyright 2015 Danny Robson <danny@nerdcruft.net>
*/
#include "linear.hpp"
#include "../../pointer.hpp"
#include "../../debug.hpp"
using util::alloc::raw::linear;
///////////////////////////////////////////////////////////////////////////////
linear::linear (void *begin, void *end):
m_begin (reinterpret_cast<std::byte*> (begin)),
m_end (reinterpret_cast<std::byte*> (end)),
m_cursor (reinterpret_cast<std::byte*> (begin))
{
CHECK_NEZ (begin);
CHECK_NEZ (end);
CHECK_LE (begin, end);
}
///////////////////////////////////////////////////////////////////////////////
void*
linear::allocate (size_t bytes)
{
return allocate (bytes, alignof (std::max_align_t));
}
//-----------------------------------------------------------------------------
void*
linear::allocate (size_t bytes, size_t alignment)
{
auto ptr = align (m_cursor, alignment);
if (ptr + bytes > m_end)
throw std::bad_alloc ();
m_cursor = ptr + bytes;
CHECK_NEZ (ptr);
return ptr;
}
//-----------------------------------------------------------------------------
void
linear::deallocate (void *ptr, size_t bytes)
{
deallocate (ptr, bytes, alignof (std::max_align_t));
}
//-----------------------------------------------------------------------------
void
linear::deallocate (void *ptr, size_t bytes, size_t alignment)
{
(void)ptr;
(void)bytes;
(void)alignment;
}
//-----------------------------------------------------------------------------
std::byte*
linear::data (void)
{
return m_begin;
}
//-----------------------------------------------------------------------------
const std::byte*
linear::data (void) const
{
return m_begin;
}
//-----------------------------------------------------------------------------
std::byte*
linear::begin (void)
{
return m_begin;
}
//-----------------------------------------------------------------------------
const std::byte*
linear::begin (void) const
{
return m_begin;
}
//-----------------------------------------------------------------------------
std::byte*
linear::end (void)
{
return m_end;
}
//-----------------------------------------------------------------------------
const std::byte*
linear::end (void) const
{
return m_end;
}
//-----------------------------------------------------------------------------
size_t
linear::offset (const void *_ptr) const
{
auto ptr = reinterpret_cast<const std::byte*> (_ptr);
CHECK_GE (ptr, m_begin);
return ptr - m_begin;
}
///////////////////////////////////////////////////////////////////////////////
void
linear::reset (void)
{
m_cursor = m_begin;
}
///////////////////////////////////////////////////////////////////////////////
size_t
linear::capacity (void) const
{
return m_end - m_begin;
}
//-----------------------------------------------------------------------------
size_t
linear::used (void) const
{
return m_cursor - m_begin;
}
//-----------------------------------------------------------------------------
size_t
linear::remain (void) const
{
return capacity () - used ();
}