| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 
 | //
// Compile-time string hashing function.
//
// Example of usage:
//   printf( "Hash test: %u", StringHash("test").GetHash() );
//   printf( "Hash test: %u", StringHash("aVeryLongTestWhichStillWorks").GetHash() );
//
#ifndef STRING_HASH_H
#define STRING_HASH_H
// Try and use an inline that will force it to happen.
// icc -inline-level=1 -inline-forceinline, and with Apple's gcc you can use gcc -obey-inline.
#if defined(_MSC_VER)
    // MSVC is __forceinline.
    #define FORCE_INLINE __forceinline
#elif defined(__GNUC__)
    // gcc is an attribute.
    #define FORCE_INLINE __attribute__((always_inline))
#else
    // default to inline.
    #define FORCE_INLINE inline
#endif
// Hashing function. If the FNV is not working well, we can try Murmur Hash or City Hash
template <unsigned int N, unsigned int I>
struct FnvHash
{
    FORCE_INLINE static unsigned int Hash(const char (&str)[N])
    {
        return( (FnvHash<N, I-1>::Hash(str) ^ str[I-1]) * 16777619u );
    }
};
 
template <unsigned int N>
struct FnvHash<N, 1>
{
    FORCE_INLINE static unsigned int Hash(const char (&str)[N])
    {
        return( (2166136261u ^ str[0]) * 16777619u );
    }
};
class StringHash
{
    public:
        template <unsigned int N>
        FORCE_INLINE StringHash(const char (&str)[N])
        : m_hash(FnvHash<N, N>::Hash(str))
        {
        }
        FORCE_INLINE unsigned int GetHash() { return( m_hash ); }
    private:
        unsigned int m_hash;
};
#endif  // STRING_HASH_H
 |