| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441 |
- // Protocol Buffers - Google's data interchange format
- // Copyright 2008 Google Inc. All rights reserved.
- // https://developers.google.com/protocol-buffers/
- //
- // 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 Google Inc. 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: kenton@google.com (Kenton Varda)
- //
- // Deals with the fact that hash_map is not defined everywhere.
- #ifndef GOOGLE_PROTOBUF_STUBS_HASH_H__
- #define GOOGLE_PROTOBUF_STUBS_HASH_H__
- #include <string.h>
- #include <google/protobuf/stubs/common.h>
- #define GOOGLE_PROTOBUF_HAVE_HASH_MAP 1
- #define GOOGLE_PROTOBUF_HAVE_HASH_SET 1
- // Use C++11 unordered_{map|set} if available.
- #if ((defined(_LIBCPP_STD_VER) && _LIBCPP_STD_VER >= 11) || \
- (((__cplusplus >= 201103L) || defined(__GXX_EXPERIMENTAL_CXX0X)) && \
- (__GLIBCXX__ > 20090421)))
- # define GOOGLE_PROTOBUF_HAS_CXX11_HASH
- // For XCode >= 4.6: the compiler is clang with libc++.
- // For earlier XCode version: the compiler is gcc-4.2.1 with libstdc++.
- // libc++ provides <unordered_map> and friends even in non C++11 mode,
- // and it does not provide the tr1 library. Therefore the following macro
- // checks against this special case.
- // Note that we should not test the __APPLE_CC__ version number or the
- // __clang__ macro, since the new compiler can still use -stdlib=libstdc++, in
- // which case <unordered_map> is not compilable without -std=c++11
- #elif defined(__APPLE_CC__)
- # if __GNUC__ >= 4
- # define GOOGLE_PROTOBUF_HAS_TR1
- # else
- // Not tested for gcc < 4... These setting can compile under 4.2.1 though.
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE __gnu_cxx
- # include <ext/hash_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS hash_map
- # include <ext/hash_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS hash_set
- # endif
- // Version checks for gcc.
- #elif defined(__GNUC__)
- // For GCC 4.x+, use tr1::unordered_map/set; otherwise, follow the
- // instructions from:
- // https://gcc.gnu.org/onlinedocs/libstdc++/manual/backwards.html
- # if __GNUC__ >= 4
- # define GOOGLE_PROTOBUF_HAS_TR1
- # elif __GNUC__ >= 3
- # include <backward/hash_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS hash_map
- # include <backward/hash_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS hash_set
- # if __GNUC__ == 3 && __GNUC_MINOR__ == 0
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE std // GCC 3.0
- # else
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE __gnu_cxx // GCC 3.1 and later
- # endif
- # else
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE
- # include <hash_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS hash_map
- # include <hash_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS hash_set
- # endif
- // GCC <= 4.1 does not define std::tr1::hash for `long long int` or `long long unsigned int`
- # if __GNUC__ == 4 && defined(__GNUC_MINOR__) && __GNUC_MINOR__ <= 1
- # undef GOOGLE_PROTOBUF_HAS_TR1
- # undef GOOGLE_PROTOBUF_HAVE_HASH_MAP
- # undef GOOGLE_PROTOBUF_HAVE_HASH_SET
- # endif
- // Version checks for MSC.
- // Apparently Microsoft decided to move hash_map *back* to the std namespace in
- // MSVC 2010:
- // http://blogs.msdn.com/vcblog/archive/2009/05/25/stl-breaking-changes-in-visual-studio-2010-beta-1.aspx
- // And.. they are moved back to stdext in MSVC 2013 (haven't checked 2012). That
- // said, use unordered_map for MSVC 2010 and beyond is our safest bet.
- #elif defined(_MSC_VER)
- # if _MSC_VER >= 1600 // Since Visual Studio 2010
- # define GOOGLE_PROTOBUF_HAS_CXX11_HASH
- # define GOOGLE_PROTOBUF_HASH_COMPARE std::hash_compare
- # elif _MSC_VER >= 1500 // Since Visual Studio 2008
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE stdext
- # include <hash_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS hash_map
- # include <hash_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS hash_set
- # define GOOGLE_PROTOBUF_HASH_COMPARE stdext::hash_compare
- # define GOOGLE_PROTOBUF_CONTAINERS_NEED_HASH_COMPARE
- # elif _MSC_VER >= 1310
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE stdext
- # include <hash_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS hash_map
- # include <hash_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS hash_set
- # define GOOGLE_PROTOBUF_HASH_COMPARE stdext::hash_compare
- # else
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE std
- # include <hash_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS hash_map
- # include <hash_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS hash_set
- # define GOOGLE_PROTOBUF_HASH_COMPARE stdext::hash_compare
- # endif
- // **ADD NEW COMPILERS SUPPORT HERE.**
- // For other compilers, undefine the macro and fallback to use std::map, in
- // google/protobuf/stubs/hash.h
- #else
- # undef GOOGLE_PROTOBUF_HAVE_HASH_MAP
- # undef GOOGLE_PROTOBUF_HAVE_HASH_SET
- #endif
- #if defined(GOOGLE_PROTOBUF_HAS_CXX11_HASH)
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE std
- # include <unordered_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS unordered_map
- # include <unordered_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS unordered_set
- #elif defined(GOOGLE_PROTOBUF_HAS_TR1)
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE std::tr1
- # include <tr1/unordered_map>
- # define GOOGLE_PROTOBUF_HASH_MAP_CLASS unordered_map
- # include <tr1/unordered_set>
- # define GOOGLE_PROTOBUF_HASH_SET_CLASS unordered_set
- #endif
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE_DECLARATION_START \
- namespace google { \
- namespace protobuf {
- # define GOOGLE_PROTOBUF_HASH_NAMESPACE_DECLARATION_END }}
- #undef GOOGLE_PROTOBUF_HAS_CXX11_HASH
- #undef GOOGLE_PROTOBUF_HAS_TR1
- #if defined(GOOGLE_PROTOBUF_HAVE_HASH_MAP) && \
- defined(GOOGLE_PROTOBUF_HAVE_HASH_SET)
- #else
- #define GOOGLE_PROTOBUF_MISSING_HASH
- #include <map>
- #include <set>
- #endif
- namespace google {
- namespace protobuf {
- #ifdef GOOGLE_PROTOBUF_MISSING_HASH
- #undef GOOGLE_PROTOBUF_MISSING_HASH
- // This system doesn't have hash_map or hash_set. Emulate them using map and
- // set.
- // Make hash<T> be the same as less<T>. Note that everywhere where custom
- // hash functions are defined in the protobuf code, they are also defined such
- // that they can be used as "less" functions, which is required by MSVC anyway.
- template <typename Key>
- struct hash {
- // Dummy, just to make derivative hash functions compile.
- int operator()(const Key& key) {
- GOOGLE_LOG(FATAL) << "Should never be called.";
- return 0;
- }
- inline bool operator()(const Key& a, const Key& b) const {
- return a < b;
- }
- };
- // Make sure char* is compared by value.
- template <>
- struct hash<const char*> {
- // Dummy, just to make derivative hash functions compile.
- int operator()(const char* key) {
- GOOGLE_LOG(FATAL) << "Should never be called.";
- return 0;
- }
- inline bool operator()(const char* a, const char* b) const {
- return strcmp(a, b) < 0;
- }
- };
- template <typename Key, typename Data,
- typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key>,
- typename Alloc = std::allocator< std::pair<const Key, Data> > >
- class hash_map : public std::map<Key, Data, HashFcn, Alloc> {
- typedef std::map<Key, Data, HashFcn, Alloc> BaseClass;
- public:
- hash_map(int a = 0, const HashFcn& b = HashFcn(),
- const EqualKey& c = EqualKey(),
- const Alloc& d = Alloc()) : BaseClass(b, d) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- template <typename Key,
- typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key> >
- class hash_set : public std::set<Key, HashFcn> {
- public:
- hash_set(int = 0) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- #elif defined(_MSC_VER) && !defined(_STLPORT_VERSION) && \
- !(defined(_LIBCPP_STD_VER) && _LIBCPP_STD_VER >= 11)
- template <typename Key>
- struct hash : public GOOGLE_PROTOBUF_HASH_COMPARE<Key> {
- };
- // MSVC's hash_compare<const char*> hashes based on the string contents but
- // compares based on the string pointer. WTF?
- class CstringLess {
- public:
- inline bool operator()(const char* a, const char* b) const {
- return strcmp(a, b) < 0;
- }
- };
- template <>
- struct hash<const char*>
- : public GOOGLE_PROTOBUF_HASH_COMPARE<const char*, CstringLess> {};
- #ifdef GOOGLE_PROTOBUF_CONTAINERS_NEED_HASH_COMPARE
- template <typename Key, typename HashFcn, typename EqualKey>
- struct InternalHashCompare : public GOOGLE_PROTOBUF_HASH_COMPARE<Key> {
- InternalHashCompare() {}
- InternalHashCompare(HashFcn hashfcn, EqualKey equalkey)
- : hashfcn_(hashfcn), equalkey_(equalkey) {}
- size_t operator()(const Key& key) const { return hashfcn_(key); }
- bool operator()(const Key& key1, const Key& key2) const {
- return !equalkey_(key1, key2);
- }
- HashFcn hashfcn_;
- EqualKey equalkey_;
- };
- template <typename Key, typename Data,
- typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key>,
- typename Alloc = std::allocator< std::pair<const Key, Data> > >
- class hash_map
- : public GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_MAP_CLASS<
- Key, Data, InternalHashCompare<Key, HashFcn, EqualKey>, Alloc> {
- typedef GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_MAP_CLASS<
- Key, Data, InternalHashCompare<Key, HashFcn, EqualKey>, Alloc> BaseClass;
- public:
- hash_map(int a = 0, const HashFcn& b = HashFcn(),
- const EqualKey& c = EqualKey(), const Alloc& d = Alloc())
- : BaseClass(InternalHashCompare<Key, HashFcn, EqualKey>(b, c), d) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- template <typename Key, typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key> >
- class hash_set
- : public GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_SET_CLASS<
- Key, InternalHashCompare<Key, HashFcn, EqualKey> > {
- public:
- hash_set(int = 0) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- #else // GOOGLE_PROTOBUF_CONTAINERS_NEED_HASH_COMPARE
- template <typename Key, typename Data,
- typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key>,
- typename Alloc = std::allocator< std::pair<const Key, Data> > >
- class hash_map
- : public GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_MAP_CLASS<
- Key, Data, HashFcn, EqualKey, Alloc> {
- typedef GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_MAP_CLASS<
- Key, Data, HashFcn, EqualKey, Alloc> BaseClass;
- public:
- hash_map(int a = 0, const HashFcn& b = HashFcn(),
- const EqualKey& c = EqualKey(),
- const Alloc& d = Alloc()) : BaseClass(a, b, c, d) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- template <typename Key, typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key> >
- class hash_set
- : public GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_SET_CLASS<
- Key, HashFcn, EqualKey> {
- public:
- hash_set(int = 0) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- #endif // GOOGLE_PROTOBUF_CONTAINERS_NEED_HASH_COMPARE
- #else // defined(_MSC_VER) && !defined(_STLPORT_VERSION)
- template <typename Key>
- struct hash : public GOOGLE_PROTOBUF_HASH_NAMESPACE::hash<Key> {
- };
- template <typename Key>
- struct hash<const Key*> {
- inline size_t operator()(const Key* key) const {
- return reinterpret_cast<size_t>(key);
- }
- };
- // Unlike the old SGI version, the TR1 "hash" does not special-case char*. So,
- // we go ahead and provide our own implementation.
- template <>
- struct hash<const char*> {
- inline size_t operator()(const char* str) const {
- size_t result = 0;
- for (; *str != '\0'; str++) {
- result = 5 * result + static_cast<size_t>(*str);
- }
- return result;
- }
- };
- template<>
- struct hash<bool> {
- size_t operator()(bool x) const {
- return static_cast<size_t>(x);
- }
- };
- template <typename Key, typename Data,
- typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key>,
- typename Alloc = std::allocator< std::pair<const Key, Data> > >
- class hash_map
- : public GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_MAP_CLASS<
- Key, Data, HashFcn, EqualKey, Alloc> {
- typedef GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_MAP_CLASS<
- Key, Data, HashFcn, EqualKey, Alloc> BaseClass;
- public:
- hash_map(int a = 0, const HashFcn& b = HashFcn(),
- const EqualKey& c = EqualKey(),
- const Alloc& d = Alloc()) : BaseClass(a, b, c, d) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- template <typename Key, typename HashFcn = hash<Key>,
- typename EqualKey = std::equal_to<Key> >
- class hash_set
- : public GOOGLE_PROTOBUF_HASH_NAMESPACE::GOOGLE_PROTOBUF_HASH_SET_CLASS<
- Key, HashFcn, EqualKey> {
- public:
- hash_set(int = 0) {}
- HashFcn hash_function() const { return HashFcn(); }
- };
- #endif // !GOOGLE_PROTOBUF_MISSING_HASH
- template <>
- struct hash<string> {
- inline size_t operator()(const string& key) const {
- return hash<const char*>()(key.c_str());
- }
- static const size_t bucket_size = 4;
- static const size_t min_buckets = 8;
- inline bool operator()(const string& a, const string& b) const {
- return a < b;
- }
- };
- template <typename First, typename Second>
- struct hash<std::pair<First, Second> > {
- inline size_t operator()(const std::pair<First, Second>& key) const {
- size_t first_hash = hash<First>()(key.first);
- size_t second_hash = hash<Second>()(key.second);
- // FIXME(kenton): What is the best way to compute this hash? I have
- // no idea! This seems a bit better than an XOR.
- return first_hash * ((1 << 16) - 1) + second_hash;
- }
- static const size_t bucket_size = 4;
- static const size_t min_buckets = 8;
- inline bool operator()(const std::pair<First, Second>& a,
- const std::pair<First, Second>& b) const {
- return a < b;
- }
- };
- // Used by GCC/SGI STL only. (Why isn't this provided by the standard
- // library? :( )
- struct streq {
- inline bool operator()(const char* a, const char* b) const {
- return strcmp(a, b) == 0;
- }
- };
- } // namespace protobuf
- } // namespace google
- #endif // GOOGLE_PROTOBUF_STUBS_HASH_H__
|