aboutsummaryrefslogtreecommitdiff
path: root/src/adlmidi_bankmap.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/adlmidi_bankmap.h')
-rw-r--r--src/adlmidi_bankmap.h114
1 files changed, 114 insertions, 0 deletions
diff --git a/src/adlmidi_bankmap.h b/src/adlmidi_bankmap.h
new file mode 100644
index 0000000..881b9b5
--- /dev/null
+++ b/src/adlmidi_bankmap.h
@@ -0,0 +1,114 @@
+/*
+ * libADLMIDI is a free MIDI to WAV conversion library with OPL3 emulation
+ *
+ * Original ADLMIDI code: Copyright (c) 2010-2014 Joel Yliluoma <bisqwit@iki.fi>
+ * ADLMIDI Library API: Copyright (c) 2015-2018 Vitaly Novichkov <admin@wohlnet.ru>
+ *
+ * Library is based on the ADLMIDI, a MIDI player for Linux and Windows with OPL3 emulation:
+ * http://iki.fi/bisqwit/source/adlmidi.html
+ *
+ * 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 of the License, or
+ * 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef ADLMIDI_BANKMAP_H
+#define ADLMIDI_BANKMAP_H
+
+#include <list>
+#include <memory>
+#include <utility>
+#include <cstdint>
+
+/**
+ * A simple hash map which accepts bank numbers as keys, can be reserved to a
+ * fixed size, offers O(1) search and insertion, has a hash function to
+ * optimize for the worst case, and has some good cache locality properties.
+ */
+template <class T>
+class BasicBankMap
+{
+public:
+ typedef uint16_t key_type; /* the bank identifier */
+ typedef T mapped_type;
+ typedef std::pair<key_type, T> value_type;
+
+ BasicBankMap();
+ void reserve(size_t capacity);
+
+ size_t size() const
+ { return m_size; }
+ size_t capacity() const
+ { return m_capacity; }
+
+ class iterator;
+ iterator begin() const;
+ iterator end() const;
+
+ struct do_not_expand_t {};
+
+ iterator find(key_type key);
+ void erase(iterator it);
+ std::pair<iterator, bool> insert(const value_type &value);
+ std::pair<iterator, bool> insert(const value_type &value, do_not_expand_t);
+ void clear();
+
+ T &operator[](key_type key);
+
+private:
+ struct Slot;
+ enum { minimum_allocation = 4 };
+ enum
+ {
+ hash_bits = 8, /* worst case # of collisions: 128^2/2^hash_bits */
+ hash_buckets = 1 << hash_bits,
+ };
+
+public:
+ class iterator
+ {
+ public:
+ iterator() {}
+ value_type &operator*() const { return slot->value; }
+ value_type *operator->() const { return &slot->value; }
+ iterator &operator++();
+ bool operator==(const iterator &o) const;
+ bool operator!=(const iterator &o) const;
+ private:
+ Slot **buckets = nullptr, *slot = nullptr;
+ size_t index = 0;
+ iterator(Slot **buckets, Slot *slot, size_t index);
+ friend BasicBankMap;
+ };
+
+private:
+ struct Slot {
+ Slot *next = nullptr, *prev = nullptr;
+ value_type value;
+ };
+ std::unique_ptr<Slot *[]> m_buckets;
+ std::list<std::unique_ptr<Slot[]>> m_allocations;
+ Slot *m_freeslots = nullptr;
+ size_t m_size = 0;
+ size_t m_capacity = 0;
+ static size_t hash(key_type key);
+ Slot *allocate_slot();
+ Slot *ensure_allocate_slot();
+ void free_slot(Slot *slot);
+ Slot *bucket_find(size_t index, key_type key);
+ void bucket_add(size_t index, Slot *slot);
+ void bucket_remove(size_t index, Slot *slot);
+};
+
+#include "adlmidi_bankmap.tcc"
+
+#endif // ADLMIDI_BANKMAP_H