/* Copyright (C) 2021-2023 Free Software Foundation, Inc. Contributed by Oracle. This file is part of GNU Binutils. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. This program 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */ /* * Interval Map implementation. * * Interval Map makes the following assumptions: * - if duplicate keys, the last one will be stored * - */ #ifndef _DBE_INTERVALMAP_H #define _DBE_INTERVALMAP_H #include #include #include template class IntervalMap : public Map { public: IntervalMap (); ~IntervalMap (); void put (Key_t key, Value_t val); Value_t get (Key_t key); Value_t get (Key_t key, typename Map::Relation rel); Value_t remove (Key_t key); private: struct Entry { Key_t key; Value_t val; }; static const int CHUNK_SIZE; int entries; int nchunks; Entry **chunks; Vector *index; }; template const int IntervalMap::CHUNK_SIZE = 16384; template IntervalMap::IntervalMap () { entries = 0; nchunks = 0; chunks = NULL; index = new Vector; } template IntervalMap::~IntervalMap () { for (int i = 0; i < nchunks; i++) delete[] chunks[i]; delete[] chunks; delete index; } template void IntervalMap::put (Key_t key, Value_t val) { int lo = 0; int hi = entries - 1; while (lo <= hi) { int md = (lo + hi) / 2; Entry *entry = index->fetch (md); int cmp = entry->key < key ? -1 : entry->key > key ? 1 : 0; if (cmp < 0) lo = md + 1; else if (cmp > 0) hi = md - 1; else { entry->val = val; return; } } if (entries >= nchunks * CHUNK_SIZE) { nchunks++; // Reallocate Entry chunk array Entry **new_chunks = new Entry*[nchunks]; for (int i = 0; i < nchunks - 1; i++) new_chunks[i] = chunks[i]; delete chunks; chunks = new_chunks; // Allocate new chunk for entries. chunks[nchunks - 1] = new Entry[CHUNK_SIZE]; } Entry *entry = &chunks[entries / CHUNK_SIZE][entries % CHUNK_SIZE]; entry->key = key; entry->val = val; index->insert (lo, entry); entries++; } template Value_t IntervalMap::get (Key_t key) { return get (key, Map::REL_EQ); } template Value_t IntervalMap::get (Key_t key, typename Map::Relation rel) { int lo = 0; int hi = entries - 1; while (lo <= hi) { int md = (lo + hi) / 2; Entry *entry = index->fetch (md); int cmp = entry->key < key ? -1 : entry->key > key ? 1 : 0; switch (rel) { case Map::REL_LT: if (cmp < 0) lo = md + 1; else hi = md - 1; break; case Map::REL_GT: if (cmp <= 0) lo = md + 1; else hi = md - 1; break; case Map::REL_LE: case Map::REL_GE: case Map::REL_EQ: if (cmp < 0) lo = md + 1; else if (cmp > 0) hi = md - 1; else return entry->val; break; } } switch (rel) { case Map::REL_LT: case Map::REL_LE: return hi >= 0 ? index->fetch (hi)->val : (Value_t) 0; case Map::REL_GT: case Map::REL_GE: return lo < entries ? index->fetch (lo)->val : (Value_t) 0; case Map::REL_EQ: break; } return (Value_t) 0; } template Value_t IntervalMap::remove (Key_t) { // Not implemented if (1) assert (0); return (Value_t) 0; } #endif