utilities.hpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  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_UTILITIES_HPP
  19. #define REALM_UTILITIES_HPP
  20. #include <cstdint>
  21. #include <cstdlib>
  22. #include <cstdlib> // size_t
  23. #include <cstdio>
  24. #include <algorithm>
  25. #include <functional>
  26. #include <time.h>
  27. #ifdef _WIN32
  28. #include <WinSock2.h>
  29. #include <intrin.h>
  30. #include <BaseTsd.h>
  31. #if !defined(_SSIZE_T_) && !defined(_SSIZE_T_DEFINED)
  32. typedef SSIZE_T ssize_t;
  33. #define _SSIZE_T_
  34. #define _SSIZE_T_DEFINED
  35. #endif
  36. #endif // _WIN32
  37. #include <realm/util/features.h>
  38. #include <realm/util/assert.hpp>
  39. #include <realm/util/functional.hpp>
  40. #include <realm/util/safe_int_ops.hpp>
  41. // GCC defines __i386__ and __x86_64__
  42. #if (defined(__X86__) || defined(__i386__) || defined(i386) || defined(_M_IX86) || defined(__386__) || \
  43. defined(__x86_64__) || defined(_M_X64))
  44. #define REALM_X86_OR_X64
  45. #define REALM_X86_OR_X64_TRUE true
  46. #else
  47. #define REALM_X86_OR_X64_TRUE false
  48. #endif
  49. // GCC defines __arm__
  50. #ifdef __arm__
  51. #define REALM_ARCH_ARM
  52. #endif
  53. #if defined _LP64 || defined __LP64__ || defined __64BIT__ || defined _ADDR64 || defined _WIN64 || \
  54. defined __arch64__ || (defined(__WORDSIZE) && __WORDSIZE == 64) || (defined __sparc && defined __sparcv9) || \
  55. defined __x86_64 || defined __amd64 || defined __x86_64__ || defined _M_X64 || defined _M_IA64 || \
  56. defined __ia64 || defined __IA64__
  57. #define REALM_PTR_64
  58. #endif
  59. #if defined(REALM_PTR_64) && defined(REALM_X86_OR_X64) && !REALM_WATCHOS
  60. #define REALM_COMPILER_SSE // Compiler supports SSE 4.2 through __builtin_ accessors or back-end assembler
  61. #define REALM_COMPILER_AVX
  62. #endif
  63. namespace realm {
  64. using StringCompareCallback = util::UniqueFunction<bool(const char* string1, const char* string2)>;
  65. extern signed char sse_support;
  66. extern signed char avx_support;
  67. template <int version>
  68. REALM_FORCEINLINE bool sseavx()
  69. {
  70. /*
  71. Return whether or not SSE 3.0 (if version = 30) or 4.2 (for version = 42) is supported. Return value
  72. is based on the CPUID instruction.
  73. sse_support = -1: No SSE support
  74. sse_support = 0: SSE3
  75. sse_support = 1: SSE42
  76. avx_support = -1: No AVX support
  77. avx_support = 0: AVX1 supported
  78. sse_support = 1: AVX2 supported (not yet implemented for detection in our cpuid_init(), todo)
  79. This lets us test very rapidly at runtime because we just need 1 compare instruction (with 0) to test both for
  80. SSE 3 and 4.2 by caller (compiler optimizes if calls are concecutive), and can decide branch with ja/jl/je because
  81. sse_support is signed type. Also, 0 requires no immediate operand. Same for AVX.
  82. We runtime-initialize sse_support in a constructor of a static variable which is not guaranteed to be called
  83. prior to cpu_sse(). So we compile-time initialize sse_support to -2 as fallback.
  84. */
  85. static_assert(version == 1 || version == 2 || version == 30 || version == 42,
  86. "Only version == 1 (AVX), 2 (AVX2), 30 (SSE 3) and 42 (SSE 4.2) are supported for detection");
  87. #ifdef REALM_COMPILER_SSE
  88. if (version == 30)
  89. return (sse_support >= 0);
  90. else if (version == 42)
  91. return (sse_support > 0); // faster than == 1 (0 requres no immediate operand)
  92. else if (version == 1) // avx
  93. return (avx_support >= 0);
  94. else if (version == 2) // avx2
  95. return (avx_support > 0);
  96. else
  97. return false;
  98. #else
  99. return false;
  100. #endif
  101. }
  102. void cpuid_init();
  103. void* round_up(void* p, size_t align);
  104. void* round_down(void* p, size_t align);
  105. constexpr size_t round_up(size_t p, size_t align);
  106. constexpr size_t round_down(size_t p, size_t align);
  107. void millisleep(unsigned long milliseconds);
  108. #ifdef _WIN32
  109. int gettimeofday(struct timeval* tp, struct timezone* tzp);
  110. #endif
  111. int64_t platform_timegm(tm time);
  112. #ifdef REALM_SLAB_ALLOC_TUNE
  113. void process_mem_usage(double& vm_usage, double& resident_set);
  114. #endif
  115. // popcount
  116. int fast_popcount32(int32_t x);
  117. int fast_popcount64(int64_t x);
  118. uint64_t fastrand(uint64_t max = 0xffffffffffffffffULL, bool is_seed = false);
  119. // Class to be used when a private generator is wanted.
  120. // Object of this class should not be shared between threads.
  121. class FastRand {
  122. public:
  123. FastRand(uint64_t seed = 1)
  124. : m_state(seed)
  125. {
  126. }
  127. uint64_t operator()(uint64_t max = uint64_t(-1));
  128. private:
  129. uint64_t m_state;
  130. };
  131. // log2 - returns -1 if x==0, otherwise log2(x)
  132. inline int log2(size_t x)
  133. {
  134. if (x == 0)
  135. return -1;
  136. #if defined(__GNUC__)
  137. #ifdef REALM_PTR_64
  138. return 63 - __builtin_clzll(x); // returns int
  139. #else
  140. return 31 - __builtin_clz(x); // returns int
  141. #endif
  142. #elif defined(_WIN32)
  143. unsigned long index = 0;
  144. #ifdef REALM_PTR_64
  145. unsigned char c = _BitScanReverse64(&index, x); // outputs unsigned long
  146. #else
  147. unsigned char c = _BitScanReverse(&index, x); // outputs unsigned long
  148. #endif
  149. return static_cast<int>(index);
  150. #else // not __GNUC__ and not _WIN32
  151. int r = 0;
  152. while (x >>= 1) {
  153. r++;
  154. }
  155. return r;
  156. #endif
  157. }
  158. // count trailing zeros (from least-significant bit)
  159. inline int ctz(size_t x)
  160. {
  161. if (x == 0)
  162. return sizeof(x) * 8;
  163. #if defined(__GNUC__)
  164. #ifdef REALM_PTR_64
  165. return __builtin_ctzll(x); // returns int
  166. #else
  167. return __builtin_ctz(x); // returns int
  168. #endif
  169. #elif defined(_WIN32)
  170. unsigned long index = 0;
  171. #ifdef REALM_PTR_64
  172. unsigned char c = _BitScanForward64(&index, x); // outputs unsigned long
  173. #else
  174. unsigned char c = _BitScanForward(&index, x); // outputs unsigned long
  175. #endif
  176. return static_cast<int>(index);
  177. #else // not __GNUC__ and not _WIN32
  178. int r = 0;
  179. while (r < sizeof(size_t) * 8 && (x & 1) == 0) {
  180. x >>= 1;
  181. r++;
  182. }
  183. return r;
  184. #endif
  185. }
  186. // Implementation:
  187. // Safe cast from 64 to 32 bits on 32 bit architecture. Differs from to_ref() by not testing alignment and
  188. // REF-bitflag.
  189. inline size_t to_size_t(int_fast64_t v) noexcept
  190. {
  191. REALM_ASSERT_DEBUG(!util::int_cast_has_overflow<size_t>(v));
  192. return size_t(v);
  193. }
  194. template <typename ReturnType, typename OriginalType>
  195. ReturnType type_punning(OriginalType variable) noexcept
  196. {
  197. union Both {
  198. OriginalType in;
  199. ReturnType out;
  200. };
  201. Both both;
  202. both.out = ReturnType(); // Clear all bits in case ReturnType is larger than OriginalType
  203. both.in = variable;
  204. return both.out;
  205. }
  206. // Also see the comments in Array::index_string()
  207. enum FindRes {
  208. // Indicate that no results were found in the search
  209. FindRes_not_found,
  210. // Indicates a single result is found
  211. FindRes_single,
  212. // Indicates more than one result is found and they are stored in a column
  213. FindRes_column
  214. };
  215. enum IndexMethod {
  216. index_FindFirst,
  217. index_FindAll_nocopy,
  218. index_Count,
  219. };
  220. // Combined result of the index_FindAll_nocopy operation. The column returned
  221. // can contain results that are not matches but all matches are within the
  222. // returned start_ndx and end_ndx.
  223. struct InternalFindResult {
  224. // Reference to a IntegerColumn containing result rows, or a single row
  225. // value if the result is FindRes_single.
  226. int64_t payload;
  227. // Offset into the result column to start at.
  228. size_t start_ndx;
  229. // Offset index in the result column to end at.
  230. size_t end_ndx;
  231. };
  232. // realm::is_any<T, U1, U2, U3, ...> ==
  233. // std::is_same<T, U1>::value || std::is_same<T, U2>::value || std::is_same<T, U3>::value ...
  234. template <typename... T>
  235. struct is_any : std::false_type {
  236. };
  237. template <typename T, typename... Ts>
  238. struct is_any<T, T, Ts...> : std::true_type {
  239. };
  240. template <typename T, typename U, typename... Ts>
  241. struct is_any<T, U, Ts...> : is_any<T, Ts...> {
  242. };
  243. template <typename... Ts>
  244. inline constexpr bool is_any_v = is_any<Ts...>::value;
  245. // Use realm::safe_equal() instead of std::equal() if one of the parameters can be a null pointer.
  246. template <class InputIterator1, class InputIterator2>
  247. bool safe_equal(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2)
  248. {
  249. #if defined(_MSC_VER)
  250. // VS has a special check in debug mode against passing a null pointer std::equal(); it will give a warning
  251. // at runtime if this is observed.
  252. // It's uncertain if this is allowed by the C++ standard. For details, see
  253. // http://stackoverflow.com/questions/19120779/is-char-p-0-stdequalp-p-p-well-defined-according-to-the-c-standard.
  254. // So we use a safe C++14 method instead that takes two range pairs.
  255. size_t len = last1 - first1;
  256. return std::equal(first1, last1, first2, first2 + len);
  257. #else
  258. return std::equal(first1, last1, first2);
  259. #endif
  260. }
  261. // Use realm::safe_copy_n() instead of std::copy_n() if one of the parameters can be a null pointer. See the
  262. // explanation of safe_equal() above; same things apply.
  263. template <class InputIt, class Size, class OutputIt>
  264. OutputIt safe_copy_n(InputIt first, Size count, OutputIt result)
  265. {
  266. #if defined(_MSC_VER)
  267. // This loop and the method prototype is copy pasted
  268. // from "Possible implementation" on http://en.cppreference.com/w/cpp/algorithm/copy_n
  269. if (count > 0) {
  270. *result++ = *first;
  271. for (Size i = 1; i < count; ++i) {
  272. *result++ = *++first;
  273. }
  274. }
  275. return result;
  276. #else
  277. return std::copy_n(first, count, result);
  278. #endif
  279. }
  280. // Converts ascii c-locale uppercase characters to lower case,
  281. // leaves other char values unchanged.
  282. inline char toLowerAscii(char c)
  283. {
  284. if (isascii(c) && isupper(c)) {
  285. #if REALM_ANDROID
  286. return tolower(c); // _tolower is not supported on all ABI levels
  287. #else
  288. return _tolower(c);
  289. #endif
  290. }
  291. return c;
  292. }
  293. inline void* round_up(void* p, size_t align)
  294. {
  295. size_t r = size_t(p) % align == 0 ? 0 : align - size_t(p) % align;
  296. return static_cast<char*>(p) + r;
  297. }
  298. inline void* round_down(void* p, size_t align)
  299. {
  300. size_t r = size_t(p);
  301. return reinterpret_cast<void*>(r & ~(align - 1));
  302. }
  303. constexpr inline size_t round_up(size_t p, size_t align)
  304. {
  305. size_t r = p % align == 0 ? 0 : align - p % align;
  306. return p + r;
  307. }
  308. constexpr inline size_t round_down(size_t p, size_t align)
  309. {
  310. size_t r = p;
  311. return r & (~(align - 1));
  312. }
  313. template <class T>
  314. struct Wrap {
  315. Wrap(const T& v)
  316. : m_value(v)
  317. {
  318. }
  319. operator T() const
  320. {
  321. return m_value;
  322. }
  323. private:
  324. T m_value;
  325. };
  326. // PlacementDelete is intended for use with std::unique_ptr when it holds an object allocated with
  327. // placement new. It simply calls the object's destructor without freeing the memory.
  328. struct PlacementDelete {
  329. template <class T>
  330. void operator()(T* v) const
  331. {
  332. v->~T();
  333. }
  334. };
  335. #ifdef _WIN32
  336. typedef HANDLE FileDesc;
  337. #else
  338. typedef int FileDesc;
  339. #endif
  340. enum class IteratorControl { AdvanceToNext, Stop };
  341. } // namespace realm
  342. #endif // REALM_UTILITIES_HPP