global_key.hpp 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. /*************************************************************************
  2. *
  3. * Copyright 2019 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_GLOBAL_KEY_HPP
  19. #define REALM_GLOBAL_KEY_HPP
  20. #include <realm/keys.hpp>
  21. #include <realm/util/optional.hpp>
  22. #include <limits>
  23. #include <cstdint>
  24. namespace realm {
  25. class StringData;
  26. class Mixed;
  27. /// GlobalKeys are globally unique for a given class (table), and up to 128 bits
  28. /// wide. They are represented as two 64-bit integers, each of which may
  29. /// frequently be small, for best on-wire compressibility.
  30. ///
  31. /// We define a way to map from 128-bit on-write GlobalKeys to local 64-bit ObjKeys.
  32. ///
  33. /// The three object ID types are:
  34. /// a. Global Keys for objects in tables without primary keys.
  35. /// b. Global Keys for objects in tables with integer primary keys.
  36. /// c. Global Keys for objects in tables with other primary key types.
  37. ///
  38. /// For objects without primary keys (a), a "squeezed" tuple of the
  39. /// client_file_ident and a peer-local sequence number is used as the local
  40. /// ObjKey. The on-write Object ID is the "unsqueezed" format.
  41. ///
  42. /// For integer primary keys (b), the GlobalKey just the integer value as the low
  43. /// part.
  44. ///
  45. /// For objects with other types of primary keys (c), the GlobalKey is a 128-bit
  46. /// hash of the primary key value. However, the local object ID must be a 63-bit
  47. /// integer, because that is the maximum size integer that can be used in an ObjKey.
  48. /// The solution is to optimistically use the lower 62 bits of the on-wire GlobalKey.
  49. /// If this results in a ObjKey which is already in use, a new local ObjKey is
  50. /// generated with the 63th bit set and using a locally generated sequence number for
  51. /// the lower bits. The mapping between GlobalKey and ObjKey is stored in the Table
  52. /// structure.
  53. struct GlobalKey {
  54. constexpr GlobalKey(uint64_t h, uint64_t l)
  55. : m_lo(l)
  56. , m_hi(h)
  57. {
  58. }
  59. static GlobalKey from_string(StringData);
  60. constexpr GlobalKey(realm::util::None = realm::util::none)
  61. : m_lo(-1)
  62. , m_hi(-1)
  63. {
  64. }
  65. // Construct an GlobalKey from either a string, an integer or a GlobalId
  66. GlobalKey(Mixed pk);
  67. // Construct an object id from the local squeezed ObjKey
  68. GlobalKey(ObjKey squeezed, uint64_t sync_file_id)
  69. {
  70. uint64_t u = uint64_t(squeezed.value);
  71. m_lo = (u & 0xff) | ((u & 0xffffff0000) >> 8);
  72. m_hi = ((u & 0xff00) >> 8) | ((u & 0xffffff0000000000) >> 32);
  73. if (m_hi == 0)
  74. m_hi = sync_file_id;
  75. }
  76. constexpr GlobalKey(const GlobalKey&) noexcept = default;
  77. GlobalKey& operator=(const GlobalKey&) noexcept = default;
  78. constexpr uint64_t lo() const
  79. {
  80. return m_lo;
  81. }
  82. constexpr uint64_t hi() const
  83. {
  84. return m_hi;
  85. }
  86. std::string to_string() const;
  87. constexpr bool operator<(const GlobalKey& other) const
  88. {
  89. return (m_hi == other.m_hi) ? (m_lo < other.m_lo) : (m_hi < other.m_hi);
  90. }
  91. constexpr bool operator==(const GlobalKey& other) const
  92. {
  93. return m_hi == other.m_hi && m_lo == other.m_lo;
  94. }
  95. constexpr bool operator!=(const GlobalKey& other) const
  96. {
  97. return !(*this == other);
  98. }
  99. explicit constexpr operator bool() const noexcept
  100. {
  101. return (*this != GlobalKey{});
  102. }
  103. // Generate a local ObjKey from the GlobalKey. If the object is created
  104. // in this realm (sync_file_id == hi) then 0 is used for hi. In this
  105. // way we achieves that objects created before first contact with the
  106. // server does not need to change key.
  107. ObjKey get_local_key(uint64_t sync_file_id)
  108. {
  109. REALM_ASSERT(m_hi <= 0x3fffffff);
  110. auto high = m_hi;
  111. if (high == sync_file_id)
  112. high = 0;
  113. uint64_t a = m_lo & 0xff;
  114. uint64_t b = (high & 0xff) << 8;
  115. uint64_t c = (m_lo & 0xffffff00) << 8;
  116. uint64_t d = (high & 0x3fffff00) << 32;
  117. return ObjKey(int64_t(a | b | c | d));
  118. }
  119. private:
  120. uint64_t m_lo;
  121. uint64_t m_hi;
  122. };
  123. std::ostream& operator<<(std::ostream&, const GlobalKey&);
  124. std::istream& operator>>(std::istream&, GlobalKey&);
  125. } // namespace realm
  126. namespace std {
  127. template <>
  128. struct hash<realm::GlobalKey> {
  129. size_t operator()(realm::GlobalKey oid) const
  130. {
  131. return std::hash<uint64_t>{}(oid.lo()) ^ std::hash<uint64_t>{}(oid.hi());
  132. }
  133. };
  134. } // namespace std
  135. #endif /* REALM_OBJECT_ID_HPP */