/* * Frozen * Copyright 2016 QuarksLab * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you 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 FROZEN_LETITGO_UNORDERED_MAP_H #define FROZEN_LETITGO_UNORDERED_MAP_H #include "frozen/bits/basic_types.h" #include "frozen/bits/constexpr_assert.h" #include "frozen/bits/elsa.h" #include "frozen/bits/exceptions.h" #include "frozen/bits/pmh.h" #include "frozen/bits/version.h" #include "frozen/random.h" #include #include #include namespace frozen { namespace bits { struct GetKey { template constexpr auto const &operator()(KV const &kv) const { return kv.first; } }; } // namespace bits template , class KeyEqual = std::equal_to> class unordered_map : private KeyEqual { static constexpr std::size_t storage_size = bits::next_highest_power_of_two(N) * (N < 32 ? 2 : 1); // size adjustment to prevent high collision rate for small sets using container_type = bits::carray, N>; using tables_type = bits::pmh_tables; container_type items_; tables_type tables_; public: /* typedefs */ using Self = unordered_map; using key_type = Key; using mapped_type = Value; using value_type = typename container_type::value_type; using size_type = typename container_type::size_type; using difference_type = typename container_type::difference_type; using hasher = Hash; using key_equal = KeyEqual; using reference = typename container_type::reference; using const_reference = typename container_type::const_reference; using pointer = typename container_type::pointer; using const_pointer = typename container_type::const_pointer; using iterator = typename container_type::iterator; using const_iterator = typename container_type::const_iterator; public: /* constructors */ unordered_map(unordered_map const &) = default; constexpr unordered_map(container_type items, Hash const &hash, KeyEqual const &equal) : KeyEqual{equal} , items_{items} , tables_{ bits::make_pmh_tables( items_, hash, bits::GetKey{}, default_prg_t{})} {} explicit constexpr unordered_map(container_type items) : unordered_map{items, Hash{}, KeyEqual{}} {} constexpr unordered_map(std::initializer_list items, Hash const & hash, KeyEqual const & equal) : unordered_map{container_type{items}, hash, equal} { constexpr_assert(items.size() == N, "Inconsistent initializer_list size and type size argument"); } constexpr unordered_map(std::initializer_list items) : unordered_map{items, Hash{}, KeyEqual{}} {} /* iterators */ constexpr iterator begin() { return items_.begin(); } constexpr iterator end() { return items_.end(); } constexpr const_iterator begin() const { return items_.begin(); } constexpr const_iterator end() const { return items_.end(); } constexpr const_iterator cbegin() const { return items_.begin(); } constexpr const_iterator cend() const { return items_.end(); } /* capacity */ constexpr bool empty() const { return !N; } constexpr size_type size() const { return N; } constexpr size_type max_size() const { return N; } /* lookup */ template constexpr std::size_t count(KeyType const &key) const { return find(key) != end(); } template constexpr Value const &at(KeyType const &key) const { return at_impl(*this, key); } template constexpr Value &at(KeyType const &key) { return at_impl(*this, key); } template constexpr const_iterator find(KeyType const &key) const { return find_impl(*this, key, hash_function(), key_eq()); } template constexpr iterator find(KeyType const &key) { return find_impl(*this, key, hash_function(), key_eq()); } template constexpr bool contains(KeyType const &key) const { return this->find(key) != this->end(); } template constexpr std::pair equal_range(KeyType const &key) const { return equal_range_impl(*this, key); } template constexpr std::pair equal_range(KeyType const &key) { return equal_range_impl(*this, key); } /* bucket interface */ constexpr std::size_t bucket_count() const { return storage_size; } constexpr std::size_t max_bucket_count() const { return storage_size; } /* observers*/ constexpr const hasher& hash_function() const { return tables_.hash_function(); } constexpr const key_equal& key_eq() const { return static_cast(*this); } private: template static inline constexpr auto& at_impl(This&& self, KeyType const &key) { auto it = self.find(key); if (it != self.end()) return it->second; else FROZEN_THROW_OR_ABORT(std::out_of_range("unknown key")); } template static inline constexpr auto find_impl(This&& self, KeyType const &key, Hasher const &hash, Equal const &equal) { auto const pos = self.tables_.lookup(key, hash); auto it = self.items_.begin() + pos; if (it != self.items_.end() && equal(it->first, key)) return it; else return self.items_.end(); } template static inline constexpr auto equal_range_impl(This&& self, KeyType const &key) { auto const it = self.find(key); if (it != self.end()) return std::make_pair(it, it + 1); else return std::make_pair(self.end(), self.end()); } }; template constexpr auto make_unordered_map(std::pair const (&items)[N]) { return unordered_map{items}; } template constexpr auto make_unordered_map( std::pair const (&items)[N], Hasher const &hash = elsa{}, Equal const &equal = std::equal_to{}) { return unordered_map{items, hash, equal}; } template constexpr auto make_unordered_map(std::array, N> const &items) { return unordered_map{items}; } template constexpr auto make_unordered_map( std::array, N> const &items, Hasher const &hash = elsa{}, Equal const &equal = std::equal_to{}) { return unordered_map{items, hash, equal}; } } // namespace frozen #endif