node_header.hpp 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. /*************************************************************************
  2. *
  3. * Copyright 2018 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_NODE_HEADER_HPP
  19. #define REALM_NODE_HEADER_HPP
  20. #include <realm/util/assert.hpp>
  21. namespace realm {
  22. const size_t max_array_size = 0x00ffffffL; // Maximum number of elements in an array
  23. const size_t max_array_payload_aligned = 0x07ffffc0L; // Maximum number of bytes that the payload of an array can be
  24. // Even though the encoding supports arrays with size up to max_array_payload_aligned,
  25. // the maximum allocation size is smaller as it must fit within a memory section
  26. // (a contiguous virtual address range). This limitation is enforced in SlabAlloc::do_alloc().
  27. class NodeHeader {
  28. public:
  29. enum Type {
  30. type_Normal,
  31. /// This array is the main array of an innner node of a B+-tree as used
  32. /// in table columns.
  33. type_InnerBptreeNode,
  34. /// This array may contain refs to subarrays. An element whose least
  35. /// significant bit is zero, is a ref pointing to a subarray. An element
  36. /// whose least significant bit is one, is just a value. It is the
  37. /// responsibility of the application to ensure that non-ref values have
  38. /// their least significant bit set. This will generally be done by
  39. /// shifting the desired vlue to the left by one bit position, and then
  40. /// setting the vacated bit to one.
  41. type_HasRefs
  42. };
  43. enum WidthType {
  44. wtype_Bits = 0, // width indicates how many bits every element occupies
  45. wtype_Multiply = 1, // width indicates how many bytes every element occupies
  46. wtype_Ignore = 2, // each element is 1 byte
  47. };
  48. static const int header_size = 8; // Number of bytes used by header
  49. // The encryption layer relies on headers always fitting within a single page.
  50. static_assert(header_size == 8, "Header must always fit in entirely on a page");
  51. static char* get_data_from_header(char* header) noexcept
  52. {
  53. return header + header_size;
  54. }
  55. static char* get_header_from_data(char* data) noexcept
  56. {
  57. return data - header_size;
  58. }
  59. static const char* get_data_from_header(const char* header) noexcept
  60. {
  61. return get_data_from_header(const_cast<char*>(header));
  62. }
  63. static bool get_is_inner_bptree_node_from_header(const char* header) noexcept
  64. {
  65. typedef unsigned char uchar;
  66. const uchar* h = reinterpret_cast<const uchar*>(header);
  67. return (int(h[4]) & 0x80) != 0;
  68. }
  69. static bool get_hasrefs_from_header(const char* header) noexcept
  70. {
  71. typedef unsigned char uchar;
  72. const uchar* h = reinterpret_cast<const uchar*>(header);
  73. return (int(h[4]) & 0x40) != 0;
  74. }
  75. static bool get_context_flag_from_header(const char* header) noexcept
  76. {
  77. typedef unsigned char uchar;
  78. const uchar* h = reinterpret_cast<const uchar*>(header);
  79. return (int(h[4]) & 0x20) != 0;
  80. }
  81. static WidthType get_wtype_from_header(const char* header) noexcept
  82. {
  83. typedef unsigned char uchar;
  84. const uchar* h = reinterpret_cast<const uchar*>(header);
  85. return WidthType((int(h[4]) & 0x18) >> 3);
  86. }
  87. static uint_least8_t get_width_from_header(const char* header) noexcept
  88. {
  89. typedef unsigned char uchar;
  90. const uchar* h = reinterpret_cast<const uchar*>(header);
  91. return uint_least8_t((1 << (int(h[4]) & 0x07)) >> 1);
  92. }
  93. static size_t get_size_from_header(const char* header) noexcept
  94. {
  95. typedef unsigned char uchar;
  96. const uchar* h = reinterpret_cast<const uchar*>(header);
  97. return (size_t(h[5]) << 16) + (size_t(h[6]) << 8) + h[7];
  98. }
  99. static size_t get_capacity_from_header(const char* header) noexcept
  100. {
  101. typedef unsigned char uchar;
  102. const uchar* h = reinterpret_cast<const uchar*>(header);
  103. return (size_t(h[0]) << 19) + (size_t(h[1]) << 11) + (h[2] << 3);
  104. }
  105. static Type get_type_from_header(const char* header) noexcept
  106. {
  107. if (get_is_inner_bptree_node_from_header(header))
  108. return type_InnerBptreeNode;
  109. if (get_hasrefs_from_header(header))
  110. return type_HasRefs;
  111. return type_Normal;
  112. }
  113. static void set_is_inner_bptree_node_in_header(bool value, char* header) noexcept
  114. {
  115. typedef unsigned char uchar;
  116. uchar* h = reinterpret_cast<uchar*>(header);
  117. h[4] = uchar((int(h[4]) & ~0x80) | int(value) << 7);
  118. }
  119. static void set_hasrefs_in_header(bool value, char* header) noexcept
  120. {
  121. typedef unsigned char uchar;
  122. uchar* h = reinterpret_cast<uchar*>(header);
  123. h[4] = uchar((int(h[4]) & ~0x40) | int(value) << 6);
  124. }
  125. static void set_context_flag_in_header(bool value, char* header) noexcept
  126. {
  127. typedef unsigned char uchar;
  128. uchar* h = reinterpret_cast<uchar*>(header);
  129. h[4] = uchar((int(h[4]) & ~0x20) | int(value) << 5);
  130. }
  131. static void set_wtype_in_header(WidthType value, char* header) noexcept
  132. {
  133. // Indicates how to calculate size in bytes based on width
  134. // 0: bits (width/8) * size
  135. // 1: multiply width * size
  136. // 2: ignore 1 * size
  137. typedef unsigned char uchar;
  138. uchar* h = reinterpret_cast<uchar*>(header);
  139. h[4] = uchar((int(h[4]) & ~0x18) | int(value) << 3);
  140. }
  141. static void set_width_in_header(int value, char* header) noexcept
  142. {
  143. // Pack width in 3 bits (log2)
  144. int w = 0;
  145. while (value) {
  146. ++w;
  147. value >>= 1;
  148. }
  149. REALM_ASSERT_3(w, <, 8);
  150. typedef unsigned char uchar;
  151. uchar* h = reinterpret_cast<uchar*>(header);
  152. h[4] = uchar((int(h[4]) & ~0x7) | w);
  153. }
  154. static void set_size_in_header(size_t value, char* header) noexcept
  155. {
  156. REALM_ASSERT_3(value, <=, max_array_size);
  157. typedef unsigned char uchar;
  158. uchar* h = reinterpret_cast<uchar*>(header);
  159. h[5] = uchar((value >> 16) & 0x000000FF);
  160. h[6] = uchar((value >> 8) & 0x000000FF);
  161. h[7] = uchar(value & 0x000000FF);
  162. }
  163. // Note: There is a copy of this function is test_alloc.cpp
  164. static void set_capacity_in_header(size_t value, char* header) noexcept
  165. {
  166. REALM_ASSERT_3(value, <=, (0xffffff << 3));
  167. typedef unsigned char uchar;
  168. uchar* h = reinterpret_cast<uchar*>(header);
  169. h[0] = uchar((value >> 19) & 0x000000FF);
  170. h[1] = uchar((value >> 11) & 0x000000FF);
  171. h[2] = uchar(value >> 3 & 0x000000FF);
  172. }
  173. static size_t get_byte_size_from_header(const char* header) noexcept
  174. {
  175. size_t size = get_size_from_header(header);
  176. uint_least8_t width = get_width_from_header(header);
  177. WidthType wtype = get_wtype_from_header(header);
  178. size_t num_bytes = calc_byte_size(wtype, size, width);
  179. return num_bytes;
  180. }
  181. static size_t calc_byte_size(WidthType wtype, size_t size, uint_least8_t width) noexcept
  182. {
  183. size_t num_bytes = 0;
  184. switch (wtype) {
  185. case wtype_Bits: {
  186. // Current assumption is that size is at most 2^24 and that width is at most 64.
  187. // In that case the following will never overflow. (Assuming that size_t is at least 32 bits)
  188. REALM_ASSERT_3(size, <, 0x1000000);
  189. size_t num_bits = size * width;
  190. num_bytes = (num_bits + 7) >> 3;
  191. break;
  192. }
  193. case wtype_Multiply: {
  194. num_bytes = size * width;
  195. break;
  196. }
  197. case wtype_Ignore:
  198. num_bytes = size;
  199. break;
  200. }
  201. // Ensure 8-byte alignment
  202. num_bytes = (num_bytes + 7) & ~size_t(7);
  203. num_bytes += header_size;
  204. return num_bytes;
  205. }
  206. };
  207. }
  208. #endif /* REALM_NODE_HEADER_HPP */