binary_data.hpp 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  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_BINARY_DATA_HPP
  19. #define REALM_BINARY_DATA_HPP
  20. #include <realm/owned_data.hpp>
  21. #include <realm/util/features.h>
  22. #include <realm/utilities.hpp>
  23. #include <algorithm>
  24. #include <cstddef>
  25. #include <ostream>
  26. #include <string>
  27. namespace realm {
  28. /// A reference to a chunk of binary data.
  29. ///
  30. /// This class does not own the referenced memory, nor does it in any other way
  31. /// attempt to manage the lifetime of it.
  32. ///
  33. /// \sa StringData
  34. class BinaryData {
  35. public:
  36. BinaryData() noexcept
  37. : m_data(nullptr)
  38. , m_size(0)
  39. {
  40. }
  41. BinaryData(const char* external_data, size_t data_size) noexcept
  42. : m_data(external_data)
  43. , m_size(data_size)
  44. {
  45. }
  46. // Note! This version includes a trailing null character when using in place constant strings
  47. template <size_t N>
  48. explicit BinaryData(const char (&external_data)[N])
  49. : m_data(external_data)
  50. , m_size(N)
  51. {
  52. }
  53. template <class T, class A>
  54. explicit BinaryData(const std::basic_string<char, T, A>&);
  55. // BinaryData does not store data, callers must manage their own strings.
  56. template <class T, class A>
  57. BinaryData(const std::basic_string<char, T, A>&&) = delete;
  58. template <class T, class A>
  59. explicit operator std::basic_string<char, T, A>() const;
  60. char operator[](size_t i) const noexcept
  61. {
  62. return m_data[i];
  63. }
  64. const char* data() const noexcept
  65. {
  66. return m_data;
  67. }
  68. size_t size() const noexcept
  69. {
  70. return m_size;
  71. }
  72. /// Is this a null reference?
  73. ///
  74. /// An instance of BinaryData is a null reference when, and only when the
  75. /// stored size is zero (size()) and the stored pointer is the null pointer
  76. /// (data()).
  77. ///
  78. /// In the case of the empty byte sequence, the stored size is still zero,
  79. /// but the stored pointer is **not** the null pointer. Note that the actual
  80. /// value of the pointer is immaterial in this case (as long as it is not
  81. /// zero), because when the size is zero, it is an error to dereference the
  82. /// pointer.
  83. ///
  84. /// Conversion of a BinaryData object to `bool` yields the logical negation
  85. /// of the result of calling this function. In other words, a BinaryData
  86. /// object is converted to true if it is not the null reference, otherwise
  87. /// it is converted to false.
  88. ///
  89. /// It is important to understand that all of the functions and operators in
  90. /// this class, and most of the functions in the Realm API in general
  91. /// makes no distinction between a null reference and a reference to the
  92. /// empty byte sequence. These functions and operators never look at the
  93. /// stored pointer if the stored size is zero.
  94. bool is_null() const noexcept;
  95. friend bool operator==(const BinaryData&, const BinaryData&) noexcept;
  96. friend bool operator!=(const BinaryData&, const BinaryData&) noexcept;
  97. //@{
  98. /// Trivial bytewise lexicographical comparison.
  99. friend bool operator<(const BinaryData&, const BinaryData&) noexcept;
  100. friend bool operator>(const BinaryData&, const BinaryData&) noexcept;
  101. friend bool operator<=(const BinaryData&, const BinaryData&) noexcept;
  102. friend bool operator>=(const BinaryData&, const BinaryData&) noexcept;
  103. //@}
  104. bool begins_with(BinaryData) const noexcept;
  105. bool ends_with(BinaryData) const noexcept;
  106. bool contains(BinaryData) const noexcept;
  107. template <class C, class T>
  108. friend std::basic_ostream<C, T>& operator<<(std::basic_ostream<C, T>&, const BinaryData&);
  109. explicit operator bool() const noexcept;
  110. private:
  111. const char* m_data;
  112. size_t m_size;
  113. };
  114. /// A read-only chunk of binary data.
  115. class OwnedBinaryData : public OwnedData {
  116. public:
  117. using OwnedData::OwnedData;
  118. OwnedBinaryData() = default;
  119. OwnedBinaryData(const BinaryData& binary_data)
  120. : OwnedData(binary_data.data(), binary_data.size())
  121. {
  122. }
  123. BinaryData get() const
  124. {
  125. return {data(), size()};
  126. }
  127. };
  128. // Implementation:
  129. template <class T, class A>
  130. inline BinaryData::BinaryData(const std::basic_string<char, T, A>& s)
  131. : m_data(s.data())
  132. , m_size(s.size())
  133. {
  134. }
  135. template <class T, class A>
  136. inline BinaryData::operator std::basic_string<char, T, A>() const
  137. {
  138. return std::basic_string<char, T, A>(m_data, m_size);
  139. }
  140. inline bool BinaryData::is_null() const noexcept
  141. {
  142. return !m_data;
  143. }
  144. inline bool operator==(const BinaryData& a, const BinaryData& b) noexcept
  145. {
  146. return a.m_size == b.m_size && a.is_null() == b.is_null() && safe_equal(a.m_data, a.m_data + a.m_size, b.m_data);
  147. }
  148. inline bool operator!=(const BinaryData& a, const BinaryData& b) noexcept
  149. {
  150. return !(a == b);
  151. }
  152. inline bool operator<(const BinaryData& a, const BinaryData& b) noexcept
  153. {
  154. if (a.is_null() || b.is_null())
  155. return !a.is_null() < !b.is_null();
  156. return std::lexicographical_compare(a.m_data, a.m_data + a.m_size, b.m_data, b.m_data + b.m_size);
  157. }
  158. inline bool operator>(const BinaryData& a, const BinaryData& b) noexcept
  159. {
  160. return b < a;
  161. }
  162. inline bool operator<=(const BinaryData& a, const BinaryData& b) noexcept
  163. {
  164. return !(b < a);
  165. }
  166. inline bool operator>=(const BinaryData& a, const BinaryData& b) noexcept
  167. {
  168. return !(a < b);
  169. }
  170. inline bool BinaryData::begins_with(BinaryData d) const noexcept
  171. {
  172. if (is_null() && !d.is_null())
  173. return false;
  174. return d.m_size <= m_size && safe_equal(m_data, m_data + d.m_size, d.m_data);
  175. }
  176. inline bool BinaryData::ends_with(BinaryData d) const noexcept
  177. {
  178. if (is_null() && !d.is_null())
  179. return false;
  180. return d.m_size <= m_size && safe_equal(m_data + m_size - d.m_size, m_data + m_size, d.m_data);
  181. }
  182. inline bool BinaryData::contains(BinaryData d) const noexcept
  183. {
  184. if (is_null() && !d.is_null())
  185. return false;
  186. return d.m_size == 0 || std::search(m_data, m_data + m_size, d.m_data, d.m_data + d.m_size) != m_data + m_size;
  187. }
  188. template <class C, class T>
  189. inline std::basic_ostream<C, T>& operator<<(std::basic_ostream<C, T>& out, const BinaryData& d)
  190. {
  191. if (d.is_null()) {
  192. out << "null";
  193. }
  194. else {
  195. out << "BinaryData(" << static_cast<const void*>(d.m_data) << ", " << d.m_size << ")";
  196. }
  197. return out;
  198. }
  199. inline BinaryData::operator bool() const noexcept
  200. {
  201. return !is_null();
  202. }
  203. } // namespace realm
  204. #endif // REALM_BINARY_DATA_HPP