utilities.hpp 11 KB

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