Containers (STL-compatible) StateMachines MessageBus and more for Embedded Systems. See www.etlcpp.com
Diff: reference_flat_set.h
- Revision:
- 0:b47c2a7920c2
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/reference_flat_set.h Fri Mar 16 16:34:18 2018 +0000 @@ -0,0 +1,853 @@ +///\file + +/****************************************************************************** +The MIT License(MIT) + +Embedded Template Library. +https://github.com/ETLCPP/etl +http://www.etlcpp.com + +Copyright(c) 2017 jwellbelove + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files(the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and / or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions : + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. +******************************************************************************/ + +#ifndef __ETL_REFERENCE_FLAT_SET__ +#define __ETL_REFERENCE_FLAT_SET__ + +#include <iterator> +#include <algorithm> +#include <functional> +#include <utility> +#include <stddef.h> + +#include "platform.h" +#include "type_traits.h" +#include "pool.h" +#include "error_handler.h" +#include "exception.h" +#include "vector.h" + +#undef ETL_FILE +#define ETL_FILE "32" + +namespace etl +{ + //*************************************************************************** + ///\ingroup reference_flat_set + /// Exception base for flat_sets + //*************************************************************************** + class flat_set_exception : public exception + { + public: + + flat_set_exception(string_type reason_, string_type file_name_, numeric_type line_number_) + : exception(reason_, file_name_, line_number_) + { + } + }; + + //*************************************************************************** + ///\ingroup reference_flat_set + /// Vector full exception. + //*************************************************************************** + class flat_set_full : public flat_set_exception + { + public: + + flat_set_full(string_type file_name_, numeric_type line_number_) + : flat_set_exception(ETL_ERROR_TEXT("flat_set:full", ETL_FILE"A"), file_name_, line_number_) + { + } + }; + + //*************************************************************************** + ///\ingroup reference_flat_set + /// Vector iterator exception. + //*************************************************************************** + class flat_set_iterator : public flat_set_exception + { + public: + + flat_set_iterator(string_type file_name_, numeric_type line_number_) + : flat_set_exception(ETL_ERROR_TEXT("flat_set:iterator", ETL_FILE"C"), file_name_, line_number_) + { + } + }; + + //*************************************************************************** + /// The base class for specifically sized reference_flat_sets. + /// Can be used as a reference type for all reference_flat_sets containing a specific type. + ///\ingroup reference_flat_set + //*************************************************************************** + template <typename T, typename TKeyCompare = std::less<T> > + class ireference_flat_set + { + public: + + typedef T key_type; + typedef T value_type; + typedef TKeyCompare key_compare; + typedef value_type& reference; + typedef const value_type& const_reference; + typedef value_type* pointer; + typedef const value_type* const_pointer; + typedef size_t size_type; + + protected: + + typedef etl::ivector<value_type*> lookup_t; + + public: + + //************************************************************************* + class iterator : public std::iterator<std::bidirectional_iterator_tag, value_type> + { + public: + + friend class ireference_flat_set; + + iterator() + { + } + + iterator(typename lookup_t::iterator ilookup_) + : ilookup(ilookup_) + { + } + + iterator(const iterator& other) + : ilookup(other.ilookup) + { + } + + iterator& operator =(const iterator& other) + { + ilookup = other.ilookup; + return *this; + } + + iterator& operator ++() + { + ++ilookup; + return *this; + } + + iterator operator ++(int) + { + iterator temp(*this); + ++ilookup; + return temp; + } + + iterator& operator --() + { + --ilookup; + return *this; + } + + iterator operator --(int) + { + iterator temp(*this); + --ilookup; + return temp; + } + + reference operator *() + { + return *(*ilookup); + } + + const_reference operator *() const + { + return *(*ilookup); + } + + pointer operator &() + { + return etl::addressof(*(*ilookup)); + } + + const_pointer operator &() const + { + return &(*(*ilookup)); + } + + pointer operator ->() + { + return etl::addressof(*(*ilookup)); + } + + const_pointer operator ->() const + { + return etl::addressof(*(*ilookup)); + } + + friend bool operator == (const iterator& lhs, const iterator& rhs) + { + return lhs.ilookup == rhs.ilookup; + } + + friend bool operator != (const iterator& lhs, const iterator& rhs) + { + return !(lhs == rhs); + } + + private: + + typename lookup_t::iterator ilookup; + }; + + //************************************************************************* + class const_iterator : public std::iterator<std::bidirectional_iterator_tag, const value_type> + { + public: + + friend class ireference_flat_set; + + const_iterator() + { + } + + const_iterator(typename lookup_t::const_iterator ilookup_) + : ilookup(ilookup_) + { + } + + const_iterator(const iterator& other) + : ilookup(other.ilookup) + { + } + + const_iterator(const const_iterator& other) + : ilookup(other.ilookup) + { + } + + const_iterator& operator =(const iterator& other) + { + ilookup = other.ilookup; + return *this; + } + + const_iterator& operator =(const const_iterator& other) + { + ilookup = other.ilookup; + return *this; + } + + const_iterator& operator ++() + { + ++ilookup; + return *this; + } + + const_iterator operator ++(int) + { + const_iterator temp(*this); + ++ilookup; + return temp; + } + + const_iterator& operator --() + { + --ilookup; + return *this; + } + + const_iterator operator --(int) + { + const_iterator temp(*this); + --ilookup; + return temp; + } + + const_reference operator *() const + { + return *(*ilookup); + } + + const_pointer operator &() const + { + return etl::addressof(*(*ilookup)); + } + + const_pointer operator ->() const + { + return etl::addressof(*(*ilookup)); + } + + friend bool operator == (const const_iterator& lhs, const const_iterator& rhs) + { + return lhs.ilookup == rhs.ilookup; + } + + friend bool operator != (const const_iterator& lhs, const const_iterator& rhs) + { + return !(lhs == rhs); + } + + private: + + typename lookup_t::const_iterator ilookup; + }; + + protected: + + typedef typename etl::parameter_type<T>::type parameter_t; + + public: + + typedef std::reverse_iterator<iterator> reverse_iterator; + typedef std::reverse_iterator<const_iterator> const_reverse_iterator; + typedef typename std::iterator_traits<iterator>::difference_type difference_type; + + //********************************************************************* + /// Returns an iterator to the beginning of the reference_flat_set. + ///\return An iterator to the beginning of the reference_flat_set. + //********************************************************************* + iterator begin() + { + return iterator(lookup.begin()); + } + + //********************************************************************* + /// Returns a const_iterator to the beginning of the reference_flat_set. + ///\return A const iterator to the beginning of the reference_flat_set. + //********************************************************************* + const_iterator begin() const + { + return const_iterator(lookup.begin()); + } + + //********************************************************************* + /// Returns an iterator to the end of the reference_flat_set. + ///\return An iterator to the end of the reference_flat_set. + //********************************************************************* + iterator end() + { + return iterator(lookup.end()); + } + + //********************************************************************* + /// Returns a const_iterator to the end of the reference_flat_set. + ///\return A const iterator to the end of the reference_flat_set. + //********************************************************************* + const_iterator end() const + { + return const_iterator(lookup.end()); + } + + //********************************************************************* + /// Returns a const_iterator to the beginning of the reference_flat_set. + ///\return A const iterator to the beginning of the reference_flat_set. + //********************************************************************* + const_iterator cbegin() const + { + return const_iterator(lookup.cbegin()); + } + + //********************************************************************* + /// Returns a const_iterator to the end of the reference_flat_set. + ///\return A const iterator to the end of the reference_flat_set. + //********************************************************************* + const_iterator cend() const + { + return const_iterator(lookup.cend()); + } + + //********************************************************************* + /// Returns an reverse iterator to the reverse beginning of the reference_flat_set. + ///\return Iterator to the reverse beginning of the reference_flat_set. + //********************************************************************* + reverse_iterator rbegin() + { + return reverse_iterator(lookup.rbegin()); + } + + //********************************************************************* + /// Returns a const reverse iterator to the reverse beginning of the reference_flat_set. + ///\return Const iterator to the reverse beginning of the reference_flat_set. + //********************************************************************* + const_reverse_iterator rbegin() const + { + return const_reverse_iterator(lookup.rbegin()); + } + + //********************************************************************* + /// Returns a reverse iterator to the end + 1 of the reference_flat_set. + ///\return Reverse iterator to the end + 1 of the reference_flat_set. + //********************************************************************* + reverse_iterator rend() + { + return reverse_iterator(lookup.rend()); + } + + //********************************************************************* + /// Returns a const reverse iterator to the end + 1 of the reference_flat_set. + ///\return Const reverse iterator to the end + 1 of the reference_flat_set. + //********************************************************************* + const_reverse_iterator rend() const + { + return const_reverse_iterator(lookup.rend()); + } + + //********************************************************************* + /// Returns a const reverse iterator to the reverse beginning of the reference_flat_set. + ///\return Const reverse iterator to the reverse beginning of the reference_flat_set. + //********************************************************************* + const_reverse_iterator crbegin() const + { + return const_reverse_iterator(lookup.crbegin()); + } + + //********************************************************************* + /// Returns a const reverse iterator to the end + 1 of the reference_flat_set. + ///\return Const reverse iterator to the end + 1 of the reference_flat_set. + //********************************************************************* + const_reverse_iterator crend() const + { + return const_reverse_iterator(lookup.crend()); + } + + //********************************************************************* + /// Assigns values to the reference_flat_set. + /// If asserts or exceptions are enabled, emits reference_flat_set_full if the reference_flat_set does not have enough free space. + /// If asserts or exceptions are enabled, emits reference_flat_set_iterator if the iterators are reversed. + ///\param first The iterator to the first element. + ///\param last The iterator to the last element + 1. + //********************************************************************* + template <typename TIterator> + void assign(TIterator first, TIterator last) + { +#if defined(ETL_DEBUG) + difference_type d = std::distance(first, last); + ETL_ASSERT(d <= difference_type(capacity()), ETL_ERROR(flat_set_full)); +#endif + + clear(); + + while (first != last) + { + insert(*first++); + } + } + + //********************************************************************* + /// Inserts a value to the reference_flat_set. + /// If asserts or exceptions are enabled, emits reference_flat_set_full if the reference_flat_set is already full. + ///\param value The value to insert. + //********************************************************************* + std::pair<iterator, bool> insert(reference value) + { + iterator i_element = lower_bound(value); + + return insert_at(i_element, value); + } + + //********************************************************************* + /// Inserts a value to the reference_flat_set. + /// If asserts or exceptions are enabled, emits reference_flat_set_full if the reference_flat_set is already full. + ///\param position The position to insert at. + ///\param value The value to insert. + //********************************************************************* + iterator insert(iterator position, reference value) + { + return insert(value).first; + } + + //********************************************************************* + /// Inserts a range of values to the reference_flat_set. + /// If asserts or exceptions are enabled, emits reference_flat_set_full if the reference_flat_set does not have enough free space. + ///\param position The position to insert at. + ///\param first The first element to add. + ///\param last The last + 1 element to add. + //********************************************************************* + template <class TIterator> + void insert(TIterator first, TIterator last) + { + while (first != last) + { + insert(*first++); + } + } + + //********************************************************************* + /// Erases an element. + ///\param key The key to erase. + ///\return The number of elements erased. 0 or 1. + //********************************************************************* + size_t erase(parameter_t key) + { + iterator i_element = find(key); + + if (i_element == end()) + { + return 0; + } + else + { + lookup.erase(i_element.ilookup); + return 1; + } + } + + //********************************************************************* + /// Erases an element. + ///\param i_element Iterator to the element. + //********************************************************************* + void erase(iterator i_element) + { + lookup.erase(i_element.ilookup); + } + + //********************************************************************* + /// Erases a range of elements. + /// The range includes all the elements between first and last, including the + /// element pointed by first, but not the one pointed by last. + ///\param first Iterator to the first element. + ///\param last Iterator to the last element. + //********************************************************************* + void erase(iterator first, iterator last) + { + lookup.erase(first.ilookup, last.ilookup);; + } + + //************************************************************************* + /// Clears the reference_flat_set. + //************************************************************************* + void clear() + { + erase(begin(), end()); + } + + //********************************************************************* + /// Finds an element. + ///\param key The key to search for. + ///\return An iterator pointing to the element or end() if not found. + //********************************************************************* + iterator find(parameter_t key) + { + iterator itr = std::lower_bound(begin(), end(), key, TKeyCompare()); + + if (itr != end()) + { + if (!key_compare()(*itr, key) && !key_compare()(key, *itr)) + { + return itr; + } + else + { + return end(); + } + } + + return end(); + } + + //********************************************************************* + /// Finds an element. + ///\param key The key to search for. + ///\return An iterator pointing to the element or end() if not found. + //********************************************************************* + const_iterator find(parameter_t key) const + { + const_iterator itr = std::lower_bound(begin(), end(), key, TKeyCompare()); + + if (itr != end()) + { + if (!key_compare()(*itr, key) && !key_compare()(key, *itr)) + { + return itr; + } + else + { + return end(); + } + } + + return end(); + } + + //********************************************************************* + /// Counts an element. + ///\param key The key to search for. + ///\return 1 if the key exists, otherwise 0. + //********************************************************************* + size_t count(parameter_t key) const + { + return (find(key) == end()) ? 0 : 1; + } + + //********************************************************************* + /// Finds the lower bound of a key + ///\param key The key to search for. + ///\return An iterator. + //********************************************************************* + iterator lower_bound(parameter_t key) + { + return std::lower_bound(begin(), end(), key, TKeyCompare()); + } + + //********************************************************************* + /// Finds the lower bound of a key + ///\param key The key to search for. + ///\return An iterator. + //********************************************************************* + const_iterator lower_bound(parameter_t key) const + { + return std::lower_bound(cbegin(), cend(), key, TKeyCompare()); + } + + //********************************************************************* + /// Finds the upper bound of a key + ///\param key The key to search for. + ///\return An iterator. + //********************************************************************* + iterator upper_bound(parameter_t key) + { + return std::upper_bound(begin(), end(), key, TKeyCompare()); + } + + //********************************************************************* + /// Finds the upper bound of a key + ///\param key The key to search for. + ///\return An iterator. + //********************************************************************* + const_iterator upper_bound(parameter_t key) const + { + return std::upper_bound(cbegin(), cend(), key, TKeyCompare()); + } + + //********************************************************************* + /// Finds the range of equal elements of a key + ///\param key The key to search for. + ///\return An iterator pair. + //********************************************************************* + std::pair<iterator, iterator> equal_range(parameter_t key) + { + return std::equal_range(begin(), end(), key, TKeyCompare()); + } + + //********************************************************************* + /// Finds the range of equal elements of a key + ///\param key The key to search for. + ///\return An iterator pair. + //********************************************************************* + std::pair<const_iterator, const_iterator> equal_range(parameter_t key) const + { + return std::upper_bound(cbegin(), cend(), key, TKeyCompare()); + } + + //************************************************************************* + /// Gets the current size of the reference_flat_set. + ///\return The current size of the reference_flat_set. + //************************************************************************* + size_type size() const + { + return lookup.size(); + } + + //************************************************************************* + /// Checks the 'empty' state of the reference_flat_set. + ///\return <b>true</b> if empty. + //************************************************************************* + bool empty() const + { + return lookup.empty(); + } + + //************************************************************************* + /// Checks the 'full' state of the reference_flat_set. + ///\return <b>true</b> if full. + //************************************************************************* + bool full() const + { + return lookup.full(); + } + + //************************************************************************* + /// Returns the capacity of the reference_flat_set. + ///\return The capacity of the reference_flat_set. + //************************************************************************* + size_type capacity() const + { + return lookup.capacity(); + } + + //************************************************************************* + /// Returns the maximum possible size of the reference_flat_set. + ///\return The maximum size of the reference_flat_set. + //************************************************************************* + size_type max_size() const + { + return lookup.max_size(); + } + + //************************************************************************* + /// Returns the remaining capacity. + ///\return The remaining capacity. + //************************************************************************* + size_t available() const + { + return lookup.available(); + } + + protected: + + //********************************************************************* + /// Constructor. + //********************************************************************* + ireference_flat_set(lookup_t& lookup_) + : lookup(lookup_) + { + } + + //********************************************************************* + /// Inserts a value to the reference_flat_set. + ///\param i_element The place to insert. + ///\param value The value to insert. + //********************************************************************* + std::pair<iterator, bool> insert_at(iterator i_element, reference value) + { + std::pair<iterator, bool> result(end(), false); + + if (i_element == end()) + { + // At the end. + ETL_ASSERT(!lookup.full(), ETL_ERROR(flat_set_full)); + + lookup.push_back(&value); + result.first = --end(); + result.second = true; + } + else + { + // Not at the end. + result.first = i_element; + + // Existing element? + if (value != *i_element) + { + // A new one. + ETL_ASSERT(!lookup.full(), ETL_ERROR(flat_set_full)); + lookup.insert(i_element.ilookup, &value); + result.second = true; + } + } + + return result; + } + + private: + + // Disable copy construction. + ireference_flat_set(const ireference_flat_set&); + ireference_flat_set& operator =(const ireference_flat_set&); + + lookup_t& lookup; + }; + + //*************************************************************************** + /// An reference flat set + ///\ingroup reference_flat_set + //*************************************************************************** + template <typename TKey, const size_t MAX_SIZE_, typename TKeyCompare = std::less<TKey> > + class reference_flat_set : public ireference_flat_set<TKey, TKeyCompare> + { + public: + + static const size_t MAX_SIZE = MAX_SIZE_; + + //************************************************************************* + /// Constructor. + //************************************************************************* + reference_flat_set() + : ireference_flat_set<TKey, TKeyCompare>(lookup) + { + } + + //************************************************************************* + /// Copy constructor. + //************************************************************************* + reference_flat_set(const reference_flat_set& other) + : ireference_flat_set<TKey, TKeyCompare>(lookup) + { + ireference_flat_set<TKey, TKeyCompare>::assign(other.cbegin(), other.cend()); + } + + //************************************************************************* + /// Constructor, from an iterator range. + ///\tparam TIterator The iterator type. + ///\param first The iterator to the first element. + ///\param last The iterator to the last element + 1. + //************************************************************************* + template <typename TIterator> + reference_flat_set(TIterator first, TIterator last) + : ireference_flat_set<TKey, TKeyCompare>(lookup) + { + ireference_flat_set<TKey, TKeyCompare>::assign(first, last); + } + + //************************************************************************* + /// Destructor. + //************************************************************************* + ~reference_flat_set() + { + ireference_flat_set<TKey, TKeyCompare>::clear(); + } + + private: + + typedef TKey value_type; + + // The vector that stores pointers to the nodes. + etl::vector<value_type*, MAX_SIZE> lookup; + }; + + //*************************************************************************** + /// Equal operator. + ///\param lhs Reference to the first reference_flat_set. + ///\param rhs Reference to the second reference_flat_set. + ///\return <b>true</b> if the arrays are equal, otherwise <b>false</b> + ///\ingroup reference_flat_set + //*************************************************************************** + template <typename T, typename TKeyCompare> + bool operator ==(const etl::ireference_flat_set<T, TKeyCompare>& lhs, const etl::ireference_flat_set<T, TKeyCompare>& rhs) + { + return (lhs.size() == rhs.size()) && std::equal(lhs.begin(), lhs.end(), rhs.begin()); + } + + //*************************************************************************** + /// Not equal operator. + ///\param lhs Reference to the first reference_flat_set. + ///\param rhs Reference to the second reference_flat_set. + ///\return <b>true</b> if the arrays are not equal, otherwise <b>false</b> + ///\ingroup reference_flat_set + //*************************************************************************** + template <typename T, typename TKeyCompare> + bool operator !=(const etl::ireference_flat_set<T, TKeyCompare>& lhs, const etl::ireference_flat_set<T, TKeyCompare>& rhs) + { + return !(lhs == rhs); + } +} + +#undef ETL_FILE +#endif +