/** * Copyright 2013-2023 Software Radio Systems Limited * * This file is part of srsRAN. * * srsRAN is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * srsRAN is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * A copy of the GNU Affero General Public License can be found in * the LICENSE file in the top-level directory of this distribution * and at http://www.gnu.org/licenses/. * */ #ifndef SRSRAN_ID_MAP_H #define SRSRAN_ID_MAP_H #include "detail/type_storage.h" #include "expected.h" #include "srsran/support/srsran_assert.h" #include namespace srsran { template class static_circular_map { static_assert(std::is_integral::value and std::is_unsigned::value, "Map key must be an unsigned integer"); using obj_t = std::pair; public: using key_type = K; using mapped_type = T; using value_type = std::pair; using difference_type = std::ptrdiff_t; class iterator { public: using iterator_category = std::forward_iterator_tag; using value_type = std::pair; using difference_type = std::ptrdiff_t; using pointer = value_type*; using reference = value_type&; iterator() = default; iterator(static_circular_map* map, size_t idx_) : ptr(map), idx(idx_) { if (idx < ptr->capacity() and not ptr->present[idx]) { ++(*this); } } iterator& operator++() { while (++idx < ptr->capacity() and not ptr->present[idx]) { } return *this; } obj_t& operator*() { srsran_assert(idx < ptr->capacity(), "Iterator out-of-bounds (%zd >= %zd)", idx, ptr->capacity()); return ptr->get_obj_(idx); } obj_t* operator->() { srsran_assert(idx < ptr->capacity(), "Iterator out-of-bounds (%zd >= %zd)", idx, ptr->capacity()); return &ptr->get_obj_(idx); } const obj_t* operator*() const { srsran_assert(idx < ptr->capacity(), "Iterator out-of-bounds (%zd >= %zd)", idx, ptr->capacity()); return &ptr->get_obj_(idx); } const obj_t* operator->() const { srsran_assert(idx < ptr->capacity(), "Iterator out-of-bounds (%zd >= %zd)", idx, ptr->capacity()); return &ptr->get_obj_(idx); } bool operator==(const iterator& other) const { return ptr == other.ptr and idx == other.idx; } bool operator!=(const iterator& other) const { return not(*this == other); } private: friend class static_circular_map; static_circular_map* ptr = nullptr; size_t idx = 0; }; class const_iterator { public: const_iterator() = default; const_iterator(const static_circular_map* map, size_t idx_) : ptr(map), idx(idx_) { if (idx < ptr->capacity() and not ptr->present[idx]) { ++(*this); } } const_iterator& operator++() { while (++idx < ptr->capacity() and not ptr->present[idx]) { } return *this; } const obj_t* operator*() const { return &ptr->buffer[idx].get(); } const obj_t* operator->() const { return &ptr->buffer[idx].get(); } bool operator==(const const_iterator& other) const { return ptr == other.ptr and idx == other.idx; } bool operator!=(const const_iterator& other) const { return not(*this == other); } private: friend class static_circular_map; const static_circular_map* ptr = nullptr; size_t idx = 0; }; static_circular_map() { std::fill(present.begin(), present.end(), false); } static_circular_map(const static_circular_map& other) : present(other.present), count(other.count) { for (size_t idx = 0; idx < other.capacity(); ++idx) { if (present[idx]) { buffer[idx].template emplace(other.get_obj_(idx)); } } } static_circular_map(static_circular_map&& other) noexcept : present(other.present), count(other.count) { for (size_t idx = 0; idx < other.capacity(); ++idx) { if (present[idx]) { buffer[idx].template emplace(std::move(other.get_obj_(idx))); } } other.clear(); } ~static_circular_map() { clear(); } static_circular_map& operator=(const static_circular_map& other) { if (this == &other) { return *this; } for (size_t idx = 0; idx < other.capacity(); ++idx) { copy_if_present_helper(buffer[idx], other.buffer[idx], present[idx], other.present[idx]); } count = other.count; present = other.present; } static_circular_map& operator=(static_circular_map&& other) noexcept { for (size_t idx = 0; idx < other.capacity(); ++idx) { move_if_present_helper(buffer[idx], other.buffer[idx], present[idx], other.present[idx]); } count = other.count; present = other.present; other.clear(); return *this; } bool contains(K id) const { size_t idx = id % N; return present[idx] and get_obj_(idx).first == id; } bool insert(K id, const T& obj) { size_t idx = id % N; if (present[idx]) { return false; } buffer[idx].template emplace(id, obj); present[idx] = true; count++; return true; } srsran::expected insert(K id, T&& obj) { size_t idx = id % N; if (present[idx]) { return srsran::expected(std::move(obj)); } buffer[idx].template emplace(id, std::move(obj)); present[idx] = true; count++; return iterator(this, idx); } template void overwrite(K id, U&& obj) { size_t idx = id % N; if (present[idx]) { erase(buffer[idx].get().first); } insert(id, std::forward(obj)); } bool erase(K id) { if (not contains(id)) { return false; } size_t idx = id % N; get_obj_(idx).~obj_t(); present[idx] = false; --count; return true; } iterator erase(iterator it) { srsran_assert(it.idx < N and it.ptr == this, "Iterator out-of-bounds (%zd >= %zd)", it.idx, N); iterator next = it; ++next; present[it.idx] = false; get_obj_(it.idx).~obj_t(); --count; return next; } void clear() { for (size_t i = 0; i < N; ++i) { if (present[i]) { present[i] = false; get_obj_(i).~obj_t(); } } count = 0; } T& operator[](K id) { srsran_assert(contains(id), "Accessing non-existent ID=%zd", (size_t)id); return get_obj_(id % N).second; } const T& operator[](K id) const { srsran_assert(contains(id), "Accessing non-existent ID=%zd", (size_t)id); return get_obj_(id % N).second; } size_t size() const { return count; } bool empty() const { return count == 0; } bool full() const { return count == N; } bool has_space(K id) { return not present[id % N]; } size_t capacity() const { return N; } iterator begin() { return iterator(this, 0); } iterator end() { return iterator(this, N); } const_iterator begin() const { return const_iterator(this, 0); } const_iterator end() const { return const_iterator(this, N); } iterator find(K id) { if (contains(id)) { return iterator(this, id % N); } return end(); } const_iterator find(K id) const { if (contains(id)) { return const_iterator(this, id % N); } return end(); } private: obj_t& get_obj_(size_t idx) { return buffer[idx].get(); } const obj_t& get_obj_(size_t idx) const { return buffer[idx].get(); } std::array, N> buffer; std::array present; size_t count = 0; }; /** * Operates like a circular map, but automatically assigns the ID/key to inserted objects in a monotonically * increasing way. The assigned IDs are not necessarily contiguous, as they are selected based on the available slots * in the circular map * @tparam K type of ID/key * @tparam T object being inserted * @tparam MAX_N maximum size of pool */ template class static_id_obj_pool : private static_circular_map { using base_t = static_circular_map; public: using iterator = typename base_t::iterator; using const_iterator = typename base_t::const_iterator; using base_t::operator[]; using base_t::begin; using base_t::contains; using base_t::empty; using base_t::end; using base_t::erase; using base_t::find; using base_t::full; using base_t::size; explicit static_id_obj_pool(K first_id = 0) : next_id(first_id) {} template srsran::expected insert(U&& t) { if (full()) { return srsran::default_error_t{}; } while (not base_t::has_space(next_id)) { ++next_id; } base_t::insert(next_id, std::forward(t)); return next_id++; } private: K next_id = 0; }; } // namespace srsran #endif // SRSRAN_ID_MAP_H