// vim: set sts=8 ts=2 sw=2 tw=99 et: // // Copyright (C) 2013, David Anderson and AlliedModders LLC // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are met: // // * Redistributions of source code must retain the above copyright notice, this // list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright notice, // this list of conditions and the following disclaimer in the documentation // and/or other materials provided with the distribution. // * Neither the name of AlliedModders LLC nor the names of its contributors // may be used to endorse or promote products derived from this software // without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE // POSSIBILITY OF SUCH DAMAGE. #ifndef _include_amtl_hashset_h_ #define _include_amtl_hashset_h_ #include namespace ke { // Template parameters: // // K - Key type. // HashPolicy - A struct with a hash and comparator function for each lookup type: // static uint32_t hash(const Type &value); // static bool matches(const Type &value, const K &key); // // Like HashMap and HashTable, init() must be called to construct the set. template class HashSet : public AllocPolicy { struct Policy { typedef K Payload; template static uint32_t hash(const Lookup &key) { return HashPolicy::hash(key); } template static bool matches(const Lookup &key, const Payload &payload) { return HashPolicy::matches(key, payload); } }; typedef HashTable Internal; public: HashSet(AllocPolicy ap = AllocPolicy()) : table_(ap) { } // capacity must be a power of two. bool init(size_t capacity = 16) { return table_.init(capacity); } typedef typename Internal::Result Result; typedef typename Internal::Insert Insert; typedef typename Internal::iterator iterator; template Result find(const Lookup &key) { return table_.find(key); } template Insert findForAdd(const Lookup &key) { return table_.findForAdd(key); } template void removeIfExists(const Lookup &key) { return table_.remove(key); } void remove(Result &r) { table_.remove(r); } // The map must not have been mutated in between findForAdd() and add(). // The Insert object is still valid after add() returns, however. template bool add(Insert &i, UK &&key) { return table_.add(i, ke::Forward(key)); } // This can be used to avoid compiler constructed temporaries, since AMTL // does not yet support move semantics. If you use this, the key and value // must be set after. bool add(Insert &i) { return table_.add(i); } iterator iter() { return iterator(&table_); } void clear() { table_.clear(); } size_t estimateMemoryUse() const { return table_.estimateMemoryUse(); } private: Internal table_; }; } #endif // _include_amtl_hashset_h_