From 96f1072c5ffbf68334d80190801cf3da96eca813 Mon Sep 17 00:00:00 2001 From: Boris Kolpackov Date: Thu, 18 Jun 2015 14:47:24 +0200 Subject: Move string-table from build2 to libbutl --- butl/string-table | 86 +++++++++++++++++++++++++++++++++++++++++++++++++++ butl/string-table.txx | 33 ++++++++++++++++++++ 2 files changed, 119 insertions(+) create mode 100644 butl/string-table create mode 100644 butl/string-table.txx (limited to 'butl') diff --git a/butl/string-table b/butl/string-table new file mode 100644 index 0000000..e36ba08 --- /dev/null +++ b/butl/string-table @@ -0,0 +1,86 @@ +// file : butl/string-table -*- C++ -*- +// copyright : Copyright (c) 2014-2015 Code Synthesis Ltd +// license : MIT; see accompanying LICENSE file + +#ifndef BUTL_STRING_TABLE +#define BUTL_STRING_TABLE + +#include +#include +#include + +#include + +namespace butl +{ + // A pool of strings and, optionally, other accompanying data in which + // each entry is assigned an individual index (or id) of type I (e.g., + // uint8_t, uint16_t, etc., depending on how many entries are expected). + // Index value 0 is reserved to indicate the "no entry" condition. + // + template + struct string_table_element + { + const I i; + const D d; + }; + + template + struct string_table_element + { + const I i; + const std::string d; + }; + + template + struct string_table_traits; + + template <> + struct string_table_traits + { + static const std::string& + key (const std::string& d) {return d;} + }; + + template + struct string_table + { + // Insert new entry unless one already exists. + // + I + insert (const D&); + + // Find existing. + // + I + find (const std::string& k) const + { + auto i (map_.find (key_type (&k))); + return i != map_.end () ? i->second.i : 0; + } + + // Reverse lookup. + // + const D& + operator[] (I i) const {assert (i > 0); return vec_[i - 1]->second.d;} + + I + size () const {return static_cast (vec_.size ());} + + bool + empty () const {return vec_.empty ();} + + private: + using key_type = butl::map_key; + using value_type = string_table_element; + using map_type = std::unordered_map; + using traits = string_table_traits; + + map_type map_; + std::vector vec_; + }; +} + +#include + +#endif // BUTL_STRING_TABLE diff --git a/butl/string-table.txx b/butl/string-table.txx new file mode 100644 index 0000000..3e74997 --- /dev/null +++ b/butl/string-table.txx @@ -0,0 +1,33 @@ +// file : butl/string-table.txx -*- C++ -*- +// copyright : Copyright (c) 2014-2015 Code Synthesis Ltd +// license : MIT; see accompanying LICENSE file + +#include // numeric_limits +#include // size_t +#include + +namespace butl +{ + template + I string_table:: + insert (const D& d) + { + std::size_t i (vec_.size () + 1); + + // Note: move(d) would be tricky since the key still points to it. + // + auto r (map_.emplace ( + key_type (&traits::key (d)), + value_type {static_cast (i), d})); + + if (r.second) + { + assert (i <= std::numeric_limits::max ()); + + r.first->first.p = &traits::key (r.first->second.d); // Update key. + vec_.push_back (r.first); + } + + return r.first->second.i; + } +} -- cgit v1.1