2019-03-28 14:27:34 +11:00
|
|
|
/*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*
|
|
|
|
* Copyright 2019 Danny Robson <danny@nerdcruft.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2019-04-01 09:07:10 +11:00
|
|
|
#include "../debug.hpp"
|
2019-03-28 14:27:34 +11:00
|
|
|
|
|
|
|
#include <array>
|
|
|
|
#include <utility>
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
#include <cstddef>
|
|
|
|
|
|
|
|
|
|
|
|
namespace cruft::map {
|
|
|
|
/// A flat map structure with a static capacity store and dynamic size.
|
|
|
|
template <std::size_t SizeV, typename KeyT, typename ValueT>
|
|
|
|
class fixed {
|
|
|
|
public:
|
|
|
|
static constexpr auto elements = SizeV;
|
|
|
|
|
|
|
|
using key_type = KeyT;
|
|
|
|
using mapped_type = ValueT;
|
|
|
|
using value_type = std::pair<KeyT,ValueT>;
|
|
|
|
|
|
|
|
using iterator = value_type*;
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
fixed () = default;
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
fixed (std::initializer_list<value_type> keyvals)
|
|
|
|
{
|
|
|
|
if (keyvals.size () > capacity ())
|
|
|
|
throw std::bad_alloc ();
|
|
|
|
|
|
|
|
for (auto const &kv: keyvals) {
|
|
|
|
auto const &[pos,success] = insert (kv);
|
|
|
|
CHECK (success);
|
|
|
|
(void)pos;
|
|
|
|
(void)success;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fixed (fixed&&) noexcept;
|
|
|
|
fixed& operator= (fixed&&) noexcept;
|
|
|
|
|
|
|
|
|
|
|
|
fixed (fixed const&);
|
|
|
|
fixed& operator= (fixed const&);
|
|
|
|
|
|
|
|
|
|
|
|
~fixed () { clear (); }
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
mapped_type& at (KeyT const &key) &
|
|
|
|
{
|
|
|
|
for (auto &i: *this)
|
|
|
|
if (i.first == key)
|
|
|
|
return i.second;
|
|
|
|
|
|
|
|
throw std::out_of_range ("Element out of range");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
mapped_type const& at (KeyT const &key) const&
|
|
|
|
{
|
|
|
|
for (auto &i: *this)
|
|
|
|
if (i.first == key)
|
|
|
|
return i.second;
|
|
|
|
|
|
|
|
throw std::out_of_range ("Element out of range");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
std::pair<iterator,bool> insert (value_type const &keyval)
|
|
|
|
{
|
|
|
|
for (auto &i: *this) {
|
|
|
|
if (i.first == keyval.first) {
|
|
|
|
return { &i, false };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_size >= capacity ())
|
|
|
|
throw std::bad_alloc ();
|
|
|
|
|
|
|
|
auto ptr = m_store.data + m_size;
|
|
|
|
new (ptr) value_type (keyval);
|
|
|
|
++m_size;
|
|
|
|
|
|
|
|
return { ptr, true };
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
mapped_type& operator[] (KeyT const &key)
|
|
|
|
{
|
|
|
|
for (auto &i: *this)
|
|
|
|
if (i.first == key)
|
|
|
|
return i.second;
|
|
|
|
|
|
|
|
if (m_size >= capacity ())
|
|
|
|
throw std::bad_alloc ();
|
|
|
|
|
|
|
|
auto ptr = m_store.data + m_size;
|
|
|
|
new (ptr) value_type ({ key, {} });
|
|
|
|
++m_size;
|
|
|
|
|
|
|
|
return ptr->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
void clear (void)
|
|
|
|
{
|
|
|
|
for (auto i: *this)
|
|
|
|
i.~value_type ();
|
|
|
|
m_size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
auto begin (void)& { return std::begin (m_store.data); }
|
|
|
|
auto end (void)& { return begin () + m_size; }
|
|
|
|
|
|
|
|
auto begin (void) const& { return std::begin (m_store.data); }
|
|
|
|
auto end (void) const& { return begin () + m_size; }
|
|
|
|
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------
|
|
|
|
auto size (void) const { return m_size; }
|
|
|
|
static auto capacity (void) { return elements; }
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::size_t m_size = 0;
|
|
|
|
|
|
|
|
union storage {
|
|
|
|
storage () {};
|
|
|
|
~storage () {};
|
|
|
|
|
|
|
|
char defer;
|
|
|
|
value_type data[elements];
|
|
|
|
} m_store;
|
|
|
|
};
|
|
|
|
}
|