// Copyright (C) 2010-2011, Gabriel Dos Reis. // 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 The Numerical Algorithms Group Ltd. 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 OWNER // 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. // --% Author: Gabriel Dos Reis #include <string.h> #include <open-axiom/string-pool> namespace OpenAxiom { // ---------------- // -- StringItem -- // ---------------- bool StringItem::equal(const Byte* str, size_t sz) const { if (length != sz) return false; for (size_t i = 0; i < sz; ++i) if (text[i] != str[i]) return false; return true; } // ---------------- // -- StringPool -- // ---------------- StringPool::StringPool() : BasicHashTable<StringItem>(109), strings(2 * Memory::page_size()) { } // Return a hash for the string starting from `str' // of length `sz'. static size_t hash(const Byte* str, size_t sz) { size_t h = 0; for(size_t i = 0; i < sz; ++i) h = str[i] + (h << 6) + (h << 16) - h; return h; } const Byte* StringPool::make_copy(const Byte* f, size_t sz) { Byte* s = strings.allocate(sz + 1); memcpy(s, f, sz); s[sz] = '\0'; return s; } StringPool::EntryType* StringPool::intern(const Byte* src, size_t sz) { const size_t h = hash(src, sz); EntryType* e = hash_chain(h); if (sz == 0) return e; for (; e->text != 0; e = e->chain) { if (e->hash == h and e->equal(src, sz)) return e; // If this is the last entry in this hash chain, allocate // a new bucket to hold the information we want to store. if (e->chain == 0) e->chain = new_bucket(); } e->text = make_copy(src, sz); e->length = sz; e->hash = h; return e; } StringPool::EntryType* StringPool::intern(const char* s) { return intern(reinterpret_cast<const Byte*>(s), strlen(s)); } }