null.hpp 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 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_NULL_HPP
  19. #define REALM_NULL_HPP
  20. #include <cmath>
  21. #include <cstring>
  22. #include <realm/util/features.h>
  23. #include <realm/util/optional.hpp>
  24. #include <realm/utilities.hpp>
  25. #include <realm/exceptions.hpp>
  26. namespace realm {
  27. /*
  28. Represents null in Query, find(), get(), set(), etc.
  29. Float/Double: Realm can both store user-given NaNs and null. Any user-given signaling NaN is converted to
  30. 0x7fa00000 (if float) or 0x7ff4000000000000 (if double). Any user-given quiet NaN is converted to
  31. 0x7fc00000 (if float) or 0x7ff8000000000000 (if double). So Realm does not preserve the optional bits in
  32. user-given NaNs.
  33. However, since both clang and gcc on x64 and ARM, and also Java on x64, return these bit patterns when
  34. requesting NaNs, these will actually seem to roundtrip bit-exact for the end-user in most cases.
  35. If set_null() is called, a null is stored in form of the bit pattern 0xffffffff (if float) or
  36. 0xffffffffffffffff (if double). These are quiet NaNs.
  37. Executing a query that involves a float/double column that contains NaNs gives an undefined result. If
  38. it contains signaling NaNs, it may throw an exception.
  39. Notes on IEEE:
  40. A NaN float is any bit pattern `s 11111111 S xxxxxxxxxxxxxxxxxxxxxx` where `s` and `x` are arbitrary, but at
  41. least 1 `x` must be 1. If `S` is 1, it's a quiet NaN, else it's a signaling NaN.
  42. A NaN doubule is the same as above, but for `s eeeeeeeeeee S xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx`
  43. The `S` bit is at position 22 (float) or 51 (double).
  44. */
  45. struct null {
  46. constexpr null() = default;
  47. operator int64_t()
  48. {
  49. throw(LogicError::type_mismatch);
  50. }
  51. template <class T>
  52. operator util::Optional<T>()
  53. {
  54. return util::none;
  55. }
  56. template <class T>
  57. bool operator==(const T&) const
  58. {
  59. REALM_ASSERT(false);
  60. return false;
  61. }
  62. template <class T>
  63. bool operator!=(const T&) const
  64. {
  65. REALM_ASSERT(false);
  66. return false;
  67. }
  68. template <class T>
  69. bool operator>(const T&) const
  70. {
  71. REALM_ASSERT(false);
  72. return false;
  73. }
  74. template <class T>
  75. bool operator>=(const T&) const
  76. {
  77. REALM_ASSERT(false);
  78. return false;
  79. }
  80. template <class T>
  81. bool operator<=(const T&) const
  82. {
  83. REALM_ASSERT(false);
  84. return false;
  85. }
  86. template <class T>
  87. bool operator<(const T&) const
  88. {
  89. REALM_ASSERT(false);
  90. return false;
  91. }
  92. /// Returns whether `v` bitwise equals the null bit-pattern
  93. template <class T>
  94. static bool is_null_float(T v)
  95. {
  96. T i = null::get_null_float<T>();
  97. return std::memcmp(&i, &v, sizeof(T)) == 0;
  98. }
  99. /// Returns the quiet NaNs that represent null for floats/doubles in Realm in stored payload.
  100. template <class T>
  101. static T get_null_float()
  102. {
  103. typename std::conditional<std::is_same<T, float>::value, uint32_t, uint64_t>::type i;
  104. int64_t double_nan = 0x7ff80000000000aa;
  105. i = std::is_same<T, float>::value ? 0x7fc000aa : static_cast<decltype(i)>(double_nan);
  106. T d = type_punning<T, decltype(i)>(i);
  107. REALM_ASSERT_DEBUG(std::isnan(d));
  108. REALM_ASSERT_DEBUG(!is_signaling(d));
  109. return d;
  110. }
  111. /// Takes a NaN as argument and returns whether or not it's signaling
  112. template <class T>
  113. static bool is_signaling(T v)
  114. {
  115. REALM_ASSERT(std::isnan(static_cast<double>(v)));
  116. typename std::conditional<std::is_same<T, float>::value, uint32_t, uint64_t>::type i;
  117. size_t signal_bit = std::is_same<T, float>::value ? 22 : 51; // If this bit is set, it's quiet
  118. i = type_punning<decltype(i), T>(v);
  119. return !(i & (1ull << signal_bit));
  120. }
  121. /// Converts any signaling or quiet NaN to their their respective bit patterns that are used on x64 gcc+clang,
  122. /// ARM clang and x64 Java.
  123. template <class T>
  124. static T to_realm(T v)
  125. {
  126. if (std::isnan(static_cast<double>(v))) {
  127. typename std::conditional<std::is_same<T, float>::value, uint32_t, uint64_t>::type i;
  128. if (std::is_same<T, float>::value) {
  129. i = is_signaling(v) ? 0x7fa00000 : 0x7fc00000;
  130. }
  131. else {
  132. i = static_cast<decltype(i)>(is_signaling(v) ? 0x7ff4000000000000 : 0x7ff8000000000000);
  133. }
  134. return type_punning<T, decltype(i)>(i);
  135. }
  136. else {
  137. return v;
  138. }
  139. }
  140. };
  141. template <class OS>
  142. OS& operator<<(OS& os, const null&)
  143. {
  144. os << "(null)";
  145. return os;
  146. }
  147. } // namespace realm
  148. #endif // REALM_NULL_HPP