/** * Provides a map template which doesn't require heap allocations for lookups. */ #pragma once #include "types.h" #include #include #include #include #include namespace detail { struct transparent_string_hash { using is_transparent = void; std::size_t operator()(const std::string_view& v) const { return std::hash{}(v); } std::size_t operator()(const std::string& s) const { return std::hash{}(s); } std::size_t operator()(const char* s) const { return operator()(std::string_view(s)); } }; struct transparent_string_equal { using is_transparent = void; bool operator()(const std::string& lhs, const std::string_view& rhs) const { return lhs == rhs; } bool operator()(const std::string& lhs, const std::string& rhs) const { return lhs == rhs; } bool operator()(const std::string& lhs, const char* rhs) const { return lhs == rhs; } bool operator()(const std::string_view& lhs, const std::string& rhs) const { return lhs == rhs; } bool operator()(const char* lhs, const std::string& rhs) const { return lhs == rhs; } }; struct transparent_string_less { using is_transparent = void; bool operator()(const std::string& lhs, const std::string_view& rhs) const { return lhs < rhs; } bool operator()(const std::string& lhs, const std::string& rhs) const { return lhs < rhs; } bool operator()(const std::string& lhs, const char* rhs) const { return lhs < rhs; } bool operator()(const std::string_view& lhs, const std::string& rhs) const { return lhs < rhs; } bool operator()(const char* lhs, const std::string& rhs) const { return lhs < rhs; } }; } // namespace detail // This requires C++20, so fallback to ugly heap allocations if we don't have it. #if __cplusplus >= 202002L template using UnorderedStringMap = std::unordered_map; template using UnorderedStringMultimap = std::unordered_multimap; using UnorderedStringSet = std::unordered_set; using UnorderedStringMultiSet = std::unordered_multiset; template ALWAYS_INLINE typename UnorderedStringMap::const_iterator UnorderedStringMapFind(const UnorderedStringMap& map, const KeyType& key) { return map.find(key); } template ALWAYS_INLINE typename UnorderedStringMap::iterator UnorderedStringMapFind(UnorderedStringMap& map, const KeyType& key) { return map.find(key); } #else template using UnorderedStringMap = std::unordered_map; template using UnorderedStringMultimap = std::unordered_multimap; using UnorderedStringSet = std::unordered_set; using UnorderedStringMultiSet = std::unordered_multiset; template ALWAYS_INLINE typename UnorderedStringMap::const_iterator UnorderedStringMapFind(const UnorderedStringMap& map, const KeyType& key) { return map.find(std::string(key)); } template ALWAYS_INLINE typename UnorderedStringMap::iterator UnorderedStringMapFind(UnorderedStringMap& map, const KeyType& key) { return map.find(std::string(key)); } #endif template using StringMap = std::map; template using StringMultiMap = std::multimap; using StringSet = std::set; using StringMultiSet = std::multiset;