| |
| /* |
| * Copyright 2011 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| |
| #ifndef GrBinHashKey_DEFINED |
| #define GrBinHashKey_DEFINED |
| |
| #include "GrTypes.h" |
| |
| /** |
| * Hash function class that can take a data chunk of any predetermined length. The hash function |
| * used is the One-at-a-Time Hash (http://burtleburtle.net/bob/hash/doobs.html). |
| * |
| * Keys are computed from ENTRY objects. ENTRY must be fully ordered by a member: |
| * int compare(const GrTBinHashKey<ENTRY, ..>& k); |
| * which returns negative if the ENTRY < k, 0 if it equals k, and positive if k < the ENTRY. |
| * Additionally, ENTRY must be flattenable into the key using setKeyData. |
| * |
| * This class satisfies the requirements to be a key for a GrTHashTable. |
| */ |
| template<typename ENTRY, size_t KEY_SIZE> |
| class GrTBinHashKey { |
| public: |
| enum { kKeySize = KEY_SIZE }; |
| |
| GrTBinHashKey() { |
| this->reset(); |
| } |
| |
| GrTBinHashKey(const GrTBinHashKey<ENTRY, KEY_SIZE>& other) { |
| *this = other; |
| } |
| |
| GrTBinHashKey<ENTRY, KEY_SIZE>& operator=(const GrTBinHashKey<ENTRY, KEY_SIZE>& other) { |
| memcpy(this, &other, sizeof(*this)); |
| return *this; |
| } |
| |
| ~GrTBinHashKey() { |
| } |
| |
| void reset() { |
| fHash = 0; |
| #if GR_DEBUG |
| fIsValid = false; |
| #endif |
| } |
| |
| void setKeyData(const uint32_t* SK_RESTRICT data) { |
| SkASSERT(GrIsALIGN4(KEY_SIZE)); |
| memcpy(&fData, data, KEY_SIZE); |
| |
| uint32_t hash = 0; |
| size_t len = KEY_SIZE; |
| while (len >= 4) { |
| hash += *data++; |
| hash += (fHash << 10); |
| hash ^= (hash >> 6); |
| len -= 4; |
| } |
| hash += (fHash << 3); |
| hash ^= (fHash >> 11); |
| hash += (fHash << 15); |
| #if GR_DEBUG |
| fIsValid = true; |
| #endif |
| fHash = hash; |
| } |
| |
| int compare(const GrTBinHashKey<ENTRY, KEY_SIZE>& key) const { |
| SkASSERT(fIsValid && key.fIsValid); |
| return memcmp(fData, key.fData, KEY_SIZE); |
| } |
| |
| static bool EQ(const ENTRY& entry, const GrTBinHashKey<ENTRY, KEY_SIZE>& key) { |
| SkASSERT(key.fIsValid); |
| return 0 == entry.compare(key); |
| } |
| |
| static bool LT(const ENTRY& entry, const GrTBinHashKey<ENTRY, KEY_SIZE>& key) { |
| SkASSERT(key.fIsValid); |
| return entry.compare(key) < 0; |
| } |
| |
| uint32_t getHash() const { |
| SkASSERT(fIsValid); |
| return fHash; |
| } |
| |
| const uint8_t* getData() const { |
| SkASSERT(fIsValid); |
| return fData; |
| } |
| |
| private: |
| uint32_t fHash; |
| uint8_t fData[KEY_SIZE]; // Buffer for key storage |
| |
| #if GR_DEBUG |
| public: |
| bool fIsValid; |
| #endif |
| }; |
| |
| #endif |