/* * Copyright (c) 2021-2022, 2024-2025 Arm Limited. * * SPDX-License-Identifier: MIT * * 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. */ #pragma once #include #include #include "custom_allocator.hpp" #include "helpers.hpp" namespace util { /** * @brief This utility class has the same purpose as std::unordered_map, but * ensures that the operations that could result in out of memory * exceptions don't throw them and also ensures that the memory can * only be allocated by an custom_allocator. */ template , typename Comparator = std::equal_to, typename Allocator = util::custom_allocator>> class unordered_map : public std::unordered_map, private noncopyable { using base = std::unordered_map; using size_type = typename base::size_type; using iterator = typename base::iterator; public: /** Must be constructed with a custom_allocator. */ unordered_map() = delete; /** * Delete all member functions that can cause allocation failure by throwing std::bad_alloc. */ Value &operator[](const Key &key) = delete; Value &operator[](Key &&key) = delete; void insert() = delete; void emplace() = delete; void emplace_hint() = delete; void reserve() = delete; void rehash() = delete; /** * @brief Construct a new unordered map object with a custom allocator. * * @param allocator The allocator that will be used. */ explicit unordered_map(util::custom_allocator> allocator) : base(allocator) { } /** * @brief Like std::unordered_map.insert but doesn't throw on out of memory errors. * * @param value The value to insert in the map. * @return std::optional> If successful, the optional will * contain the same return value as from std::unordered_map.insert, otherwise * if out of memory, the function returns std::nullopt. */ std::optional> try_insert(const std::pair &value) { try { return { base::insert(value) }; } catch (std::bad_alloc &e) { return std::nullopt; } } /** * @brief Like std::unordered_map.try_emplace but doesn't throw on out of memory errors. * * @param args The arguments to be forwarded to the constructor of the value type. * @return std::pair If successful, the pair will contain * the same return value as from std::unordered_map.emplace, otherwise * if out of memory, the function returns { base::end(), false }. */ template std::pair try_emplace(Args &&...args) { try { return base::try_emplace(std::forward(args)...); } catch (std::bad_alloc &e) { return { base::end(), false }; } } /** * @brief Like std::unordered_map.reserve but doesn't throw on out of memory errors. * * @param size The new capacity of the container. Same as std::unordered_map.reserve. * @return true If the container was resized successfuly. * @return false If the host has run out of memory */ bool try_reserve(size_type size) { try { base::reserve(size); return true; } catch (std::bad_alloc &e) { return false; } } /** * @brief Like std::unordered_map.rehash but doesn't throw on out of memory errors. * * @param count Number of buckets. Same as std::unordered_map.rehash. * @return true If the container was rehashed successfuly. * @return false If the host has run out of memory */ bool try_rehash(size_type count) { try { base::rehash(count); return true; } catch (std::bad_alloc &e) { return false; } } }; } /* namespace util */