| /* |
| * Copyright 2021 WebAssembly Community Group participants |
| * |
| * 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. |
| */ |
| |
| #ifndef wasm_support_insert_ordered_h |
| #define wasm_support_insert_ordered_h |
| |
| #include <list> |
| #include <stddef.h> |
| #include <unordered_map> |
| |
| #include "support/utilities.h" |
| |
| namespace wasm { |
| |
| // like std::set, except that begin() -> end() iterates in the |
| // order that elements were added to the set (not in the order |
| // of operator<(T, T)) |
| template<typename T> struct InsertOrderedSet { |
| std::unordered_map<T, typename std::list<T>::iterator> Map; |
| std::list<T> List; |
| |
| using iterator = typename std::list<T>::iterator; |
| iterator begin() { return List.begin(); } |
| iterator end() { return List.end(); } |
| |
| void erase(const T& val) { |
| auto it = Map.find(val); |
| if (it != Map.end()) { |
| List.erase(it->second); |
| Map.erase(it); |
| } |
| } |
| |
| void erase(iterator position) { |
| Map.erase(*position); |
| List.erase(position); |
| } |
| |
| // cheating a bit, not returning the iterator |
| bool insert(const T& val) { |
| auto [it, inserted] = Map.insert({val, List.begin()}); |
| if (inserted) { |
| List.push_back(val); |
| it->second = --List.end(); |
| } |
| return inserted; |
| } |
| |
| template<typename It> void insert(It begin, It end) { |
| for (; begin != end; ++begin) { |
| insert(*begin); |
| } |
| } |
| |
| size_t size() const { return Map.size(); } |
| bool empty() const { return Map.empty(); } |
| |
| void clear() { |
| Map.clear(); |
| List.clear(); |
| } |
| |
| size_t count(const T& val) const { return Map.count(val); } |
| |
| InsertOrderedSet() = default; |
| InsertOrderedSet(const InsertOrderedSet& other) { *this = other; } |
| InsertOrderedSet& operator=(const InsertOrderedSet& other) { |
| clear(); |
| for (auto i : other.List) { |
| insert(i); // inserting manually creates proper iterators |
| } |
| return *this; |
| } |
| }; |
| |
| // like std::map, except that begin() -> end() iterates in the |
| // order that elements were added to the map (not in the order |
| // of operator<(Key, Key)) |
| template<typename Key, typename T> struct InsertOrderedMap { |
| std::unordered_map<Key, typename std::list<std::pair<const Key, T>>::iterator> |
| Map; |
| std::list<std::pair<const Key, T>> List; |
| |
| using iterator = typename std::list<std::pair<const Key, T>>::iterator; |
| iterator begin() { return List.begin(); } |
| iterator end() { return List.end(); } |
| |
| typedef |
| typename std::list<std::pair<const Key, T>>::const_iterator const_iterator; |
| const_iterator begin() const { return List.begin(); } |
| const_iterator end() const { return List.end(); } |
| |
| std::pair<iterator, bool> insert(const std::pair<const Key, T>& kv) { |
| // Try inserting with a placeholder list iterator. |
| auto inserted = Map.insert({kv.first, List.end()}); |
| if (inserted.second) { |
| // This is a new item; insert it in the list and update the iterator. |
| List.push_back(kv); |
| inserted.first->second = std::prev(List.end()); |
| } |
| return {inserted.first->second, inserted.second}; |
| } |
| |
| T& operator[](const Key& k) { |
| std::pair<const Key, T> kv = {k, {}}; |
| return insert(kv).first->second; |
| } |
| |
| T& at(const Key& k) { return Map.at(k)->second; } |
| |
| iterator find(const Key& k) { |
| auto it = Map.find(k); |
| if (it == Map.end()) { |
| return end(); |
| } |
| return it->second; |
| } |
| |
| const_iterator find(const Key& k) const { |
| auto it = Map.find(k); |
| if (it == Map.end()) { |
| return end(); |
| } |
| return it->second; |
| } |
| |
| void erase(const Key& k) { |
| auto it = Map.find(k); |
| if (it != Map.end()) { |
| List.erase(it->second); |
| Map.erase(it); |
| } |
| } |
| |
| void erase(iterator position) { erase(position->first); } |
| |
| void clear() { |
| Map.clear(); |
| List.clear(); |
| } |
| |
| void swap(InsertOrderedMap<Key, T>& Other) { |
| Map.swap(Other.Map); |
| List.swap(Other.List); |
| } |
| |
| size_t size() const { return Map.size(); } |
| bool empty() const { return Map.empty(); } |
| size_t count(const Key& k) const { return Map.count(k); } |
| |
| InsertOrderedMap() = default; |
| InsertOrderedMap(const InsertOrderedMap& other) { |
| for (auto kv : other) { |
| insert(kv); |
| } |
| } |
| InsertOrderedMap& operator=(const InsertOrderedMap& other) { |
| if (this != &other) { |
| this->~InsertOrderedMap(); |
| new (this) InsertOrderedMap<Key, T>(other); |
| } |
| return *this; |
| } |
| |
| bool operator==(const InsertOrderedMap& other) { |
| return Map == other.Map && List == other.List; |
| } |
| bool operator!=(const InsertOrderedMap& other) { return !(*this == other); } |
| }; |
| |
| } // namespace wasm |
| |
| #endif // wasm_support_insert_ordered_h |