// Copyright 2011 Google Inc. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef NINJA_MAP_H_ #define NINJA_MAP_H_ #include "string_piece.h" static const unsigned int kSeed = 0xDECAFBAD; // MurmurHash2, by Austin Appleby static inline unsigned int MurmurHash2(const void* key, int len, unsigned int seed) { const unsigned int m = 0x5bd1e995; const int r = 24; unsigned int h = seed ^ len; const unsigned char * data = (const unsigned char *)key; while(len >= 4) { unsigned int k = *(unsigned int *)data; k *= m; k ^= k >> r; k *= m; h *= m; h ^= k; data += 4; len -= 4; } switch(len) { case 3: h ^= data[2] << 16; case 2: h ^= data[1] << 8; case 1: h ^= data[0]; h *= m; }; h ^= h >> 13; h *= m; h ^= h >> 15; return h; } static size_t StlHash(StringPiece str) { const char* p = str.str_; int len = str.len_; size_t hash = 0; while (len--) hash = 5 * hash + *p++; return hash; } #ifdef _MSC_VER #include using stdext::hash_map; using stdext::hash_compare; struct ExternalStringCmp { bool operator()(const char* a, const char* b) const { return strcmp(a, b) < 0; } }; #else #include using __gnu_cxx::hash_map; namespace __gnu_cxx { template<> struct hash { size_t operator()(const std::string& s) const { return hash()(s.c_str()); } }; } /// Hash functor for StringPiece. struct ExternalStringHash { size_t operator()(StringPiece key) const { return StlHash(key); } }; #endif /// A template for hash_maps keyed by a StringPiece whose string is /// owned externally (typically by the values). Use like: /// ExternalStringHash::Type foos; to make foos into a hash /// mapping StringPiece => Foo*. template struct ExternalStringHashMap { #ifdef _MSC_VER typedef hash_map > Type; #else typedef hash_map Type; #endif }; #endif // NINJA_MAP_H_