query_conditions.hpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083
  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_QUERY_CONDITIONS_HPP
  19. #define REALM_QUERY_CONDITIONS_HPP
  20. #include <cstdint>
  21. #include <string>
  22. #include <realm/unicode.hpp>
  23. #include <realm/binary_data.hpp>
  24. #include <realm/query_value.hpp>
  25. #include <realm/mixed.hpp>
  26. #include <realm/utilities.hpp>
  27. namespace realm {
  28. enum Action {
  29. act_ReturnFirst,
  30. act_Sum,
  31. act_Max,
  32. act_Min,
  33. act_Count,
  34. act_FindAll,
  35. act_CallbackIdx,
  36. act_Average
  37. };
  38. class ClusterKeyArray;
  39. class QueryStateBase {
  40. public:
  41. int64_t m_minmax_key; // used only for min/max, to save index of current min/max value
  42. uint64_t m_key_offset;
  43. const ClusterKeyArray* m_key_values;
  44. QueryStateBase(size_t limit)
  45. : m_minmax_key(-1)
  46. , m_key_offset(0)
  47. , m_key_values(nullptr)
  48. , m_match_count(0)
  49. , m_limit(limit)
  50. {
  51. }
  52. virtual ~QueryStateBase()
  53. {
  54. }
  55. // Called when we have a match.
  56. // The return value indicates if the query should continue.
  57. virtual bool match(size_t, Mixed) noexcept = 0;
  58. virtual bool match_pattern(size_t, uint64_t)
  59. {
  60. return false;
  61. }
  62. inline size_t match_count() const noexcept
  63. {
  64. return m_match_count;
  65. }
  66. inline size_t limit() const noexcept
  67. {
  68. return m_limit;
  69. }
  70. protected:
  71. size_t m_match_count;
  72. size_t m_limit;
  73. private:
  74. virtual void dyncast();
  75. };
  76. template <class>
  77. class QueryStateMin;
  78. template <class>
  79. class QueryStateMax;
  80. class QueryStateCount : public QueryStateBase {
  81. public:
  82. QueryStateCount(size_t limit = -1)
  83. : QueryStateBase(limit)
  84. {
  85. }
  86. bool match(size_t, Mixed) noexcept final
  87. {
  88. ++m_match_count;
  89. return (m_limit > m_match_count);
  90. }
  91. size_t get_count() const noexcept
  92. {
  93. return m_match_count;
  94. }
  95. };
  96. // Array::VTable only uses the first 4 conditions (enums) in an array of function pointers
  97. enum { cond_Equal, cond_NotEqual, cond_Greater, cond_Less, cond_VTABLE_FINDER_COUNT, cond_None, cond_LeftNotNull };
  98. // Quick hack to make "Queries with Integer null columns" able to compile in Visual Studio 2015 which doesn't full
  99. // support sfinae
  100. // (real cause hasn't been investigated yet, cannot exclude that we don't obey c++11 standard)
  101. struct HackClass {
  102. template <class A, class B, class C>
  103. bool can_match(A, B, C)
  104. {
  105. REALM_ASSERT(false);
  106. return false;
  107. }
  108. template <class A, class B, class C>
  109. bool will_match(A, B, C)
  110. {
  111. REALM_ASSERT(false);
  112. return false;
  113. }
  114. };
  115. // Does v2 contain v1?
  116. struct Contains : public HackClass {
  117. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  118. {
  119. return v2.contains(v1);
  120. }
  121. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  122. {
  123. return v2.contains(v1);
  124. }
  125. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  126. {
  127. return v2.contains(v1);
  128. }
  129. bool operator()(StringData v1, const std::array<uint8_t, 256> &charmap, StringData v2) const
  130. {
  131. return v2.contains(v1, charmap);
  132. }
  133. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  134. {
  135. if (m1.is_null())
  136. return !m2.is_null();
  137. if (Mixed::types_are_comparable(m1, m2)) {
  138. BinaryData b1 = m1.get_binary();
  139. BinaryData b2 = m2.get_binary();
  140. return operator()(b1, b2, false, false);
  141. }
  142. return false;
  143. }
  144. template <class A, class B>
  145. bool operator()(A, B) const
  146. {
  147. REALM_ASSERT(false);
  148. return false;
  149. }
  150. template <class A, class B, class C, class D>
  151. bool operator()(A, B, C, D) const
  152. {
  153. REALM_ASSERT(false);
  154. return false;
  155. }
  156. bool operator()(int64_t, int64_t, bool, bool) const
  157. {
  158. REALM_ASSERT(false);
  159. return false;
  160. }
  161. static std::string description()
  162. {
  163. return "CONTAINS";
  164. }
  165. static const int condition = -1;
  166. };
  167. // Does v2 contain something like v1 (wildcard matching)?
  168. struct Like : public HackClass {
  169. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  170. {
  171. return v2.like(v1);
  172. }
  173. bool operator()(BinaryData b1, const char*, const char*, BinaryData b2, bool = false, bool = false) const
  174. {
  175. StringData s1(b1.data(), b1.size());
  176. StringData s2(b2.data(), b2.size());
  177. return s2.like(s1);
  178. }
  179. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  180. {
  181. return v2.like(v1);
  182. }
  183. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  184. {
  185. StringData s1(b1.data(), b1.size());
  186. StringData s2(b2.data(), b2.size());
  187. return s2.like(s1);
  188. }
  189. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  190. {
  191. if (m1.is_null() && m2.is_null())
  192. return true;
  193. if (Mixed::types_are_comparable(m1, m2)) {
  194. BinaryData b1 = m1.get_binary();
  195. BinaryData b2 = m2.get_binary();
  196. return operator()(b1, b2, false, false);
  197. }
  198. return false;
  199. }
  200. template <class A, class B>
  201. bool operator()(A, B) const
  202. {
  203. REALM_ASSERT(false);
  204. return false;
  205. }
  206. template <class A, class B, class C, class D>
  207. bool operator()(A, B, C, D) const
  208. {
  209. REALM_ASSERT(false);
  210. return false;
  211. }
  212. bool operator()(int64_t, int64_t, bool, bool) const
  213. {
  214. REALM_ASSERT(false);
  215. return false;
  216. }
  217. static std::string description()
  218. {
  219. return "LIKE";
  220. }
  221. static const int condition = -1;
  222. };
  223. // Does v2 begin with v1?
  224. struct BeginsWith : public HackClass {
  225. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  226. {
  227. return v2.begins_with(v1);
  228. }
  229. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  230. {
  231. return v2.begins_with(v1);
  232. }
  233. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  234. {
  235. return v2.begins_with(v1);
  236. }
  237. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  238. {
  239. if (Mixed::types_are_comparable(m1, m2)) {
  240. BinaryData b1 = m1.get_binary();
  241. BinaryData b2 = m2.get_binary();
  242. return b2.begins_with(b1);
  243. }
  244. return false;
  245. }
  246. template <class A, class B, class C, class D>
  247. bool operator()(A, B, C, D) const
  248. {
  249. REALM_ASSERT(false);
  250. return false;
  251. }
  252. template <class A, class B>
  253. bool operator()(A, B) const
  254. {
  255. REALM_ASSERT(false);
  256. return false;
  257. }
  258. static std::string description()
  259. {
  260. return "BEGINSWITH";
  261. }
  262. static const int condition = -1;
  263. };
  264. // Does v2 end with v1?
  265. struct EndsWith : public HackClass {
  266. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  267. {
  268. return v2.ends_with(v1);
  269. }
  270. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  271. {
  272. return v2.ends_with(v1);
  273. }
  274. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  275. {
  276. return v2.ends_with(v1);
  277. }
  278. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  279. {
  280. if (Mixed::types_are_comparable(m1, m2)) {
  281. BinaryData b1 = m1.get_binary();
  282. BinaryData b2 = m2.get_binary();
  283. return operator()(b1, b2, false, false);
  284. }
  285. return false;
  286. }
  287. template <class A, class B>
  288. bool operator()(A, B) const
  289. {
  290. REALM_ASSERT(false);
  291. return false;
  292. }
  293. template <class A, class B, class C, class D>
  294. bool operator()(A, B, C, D) const
  295. {
  296. REALM_ASSERT(false);
  297. return false;
  298. }
  299. static std::string description()
  300. {
  301. return "ENDSWITH";
  302. }
  303. static const int condition = -1;
  304. };
  305. struct Equal {
  306. static const int avx = 0x00; // _CMP_EQ_OQ
  307. // bool operator()(const bool v1, const bool v2, bool v1null = false, bool v2null = false) const { return v1 ==
  308. // v2; }
  309. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  310. {
  311. return v1 == v2;
  312. }
  313. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  314. {
  315. return v1 == v2;
  316. }
  317. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  318. {
  319. return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 == m2));
  320. }
  321. template <class T>
  322. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  323. {
  324. return (v1null && v2null) || (!v1null && !v2null && v1 == v2);
  325. }
  326. static const int condition = cond_Equal;
  327. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  328. {
  329. return (v >= lbound && v <= ubound);
  330. }
  331. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  332. {
  333. return (v == 0 && ubound == 0 && lbound == 0);
  334. }
  335. static std::string description()
  336. {
  337. return "==";
  338. }
  339. };
  340. struct NotEqual {
  341. static const int avx = 0x0B; // _CMP_FALSE_OQ
  342. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  343. {
  344. return v1 != v2;
  345. }
  346. // bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const { return v1 != v2; }
  347. template <class T>
  348. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  349. {
  350. if (!v1null && !v2null)
  351. return v1 != v2;
  352. if (v1null && v2null)
  353. return false;
  354. return true;
  355. }
  356. bool operator()(const QueryValue& m1, const Mixed& m2) const
  357. {
  358. return !Equal()(m1, m2);
  359. }
  360. static const int condition = cond_NotEqual;
  361. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  362. {
  363. return !(v == 0 && ubound == 0 && lbound == 0);
  364. }
  365. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  366. {
  367. return (v > ubound || v < lbound);
  368. }
  369. template <class A, class B, class C, class D>
  370. bool operator()(A, B, C, D) const = delete;
  371. static std::string description()
  372. {
  373. return "!=";
  374. }
  375. };
  376. // Does v2 contain v1?
  377. struct ContainsIns : public HackClass {
  378. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  379. bool = false) const
  380. {
  381. if (v2.is_null() && !v1.is_null())
  382. return false;
  383. if (v1.size() == 0 && !v2.is_null())
  384. return true;
  385. return search_case_fold(v2, v1_upper, v1_lower, v1.size()) != v2.size();
  386. }
  387. // Slow version, used if caller hasn't stored an upper and lower case version
  388. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  389. {
  390. if (v2.is_null() && !v1.is_null())
  391. return false;
  392. if (v1.size() == 0 && !v2.is_null())
  393. return true;
  394. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  395. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  396. return search_case_fold(v2, v1_upper.c_str(), v1_lower.c_str(), v1.size()) != v2.size();
  397. }
  398. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  399. {
  400. StringData s1(b1.data(), b1.size());
  401. StringData s2(b2.data(), b2.size());
  402. return this->operator()(s1, s2, false, false);
  403. }
  404. // Case insensitive Boyer-Moore version
  405. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, const std::array<uint8_t, 256> &charmap, StringData v2) const
  406. {
  407. if (v2.is_null() && !v1.is_null())
  408. return false;
  409. if (v1.size() == 0 && !v2.is_null())
  410. return true;
  411. return contains_ins(v2, v1_upper, v1_lower, v1.size(), charmap);
  412. }
  413. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  414. {
  415. if (m1.is_null())
  416. return !m2.is_null();
  417. if (Mixed::types_are_comparable(m1, m2)) {
  418. BinaryData b1 = m1.get_binary();
  419. BinaryData b2 = m2.get_binary();
  420. return operator()(b1, b2, false, false);
  421. }
  422. return false;
  423. }
  424. template <class A, class B>
  425. bool operator()(A, B) const
  426. {
  427. REALM_ASSERT(false);
  428. return false;
  429. }
  430. template <class A, class B, class C, class D>
  431. bool operator()(A, B, C, D) const
  432. {
  433. REALM_ASSERT(false);
  434. return false;
  435. }
  436. bool operator()(int64_t, int64_t, bool, bool) const
  437. {
  438. REALM_ASSERT(false);
  439. return false;
  440. }
  441. static std::string description()
  442. {
  443. return "CONTAINS[c]";
  444. }
  445. static const int condition = -1;
  446. };
  447. // Does v2 contain something like v1 (wildcard matching)?
  448. struct LikeIns : public HackClass {
  449. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  450. bool = false) const
  451. {
  452. if (v2.is_null() || v1.is_null()) {
  453. return (v2.is_null() && v1.is_null());
  454. }
  455. return string_like_ins(v2, v1_lower, v1_upper);
  456. }
  457. bool operator()(BinaryData b1, const char* b1_upper, const char* b1_lower, BinaryData b2, bool = false,
  458. bool = false) const
  459. {
  460. if (b2.is_null() || b1.is_null()) {
  461. return (b2.is_null() && b1.is_null());
  462. }
  463. StringData s2(b2.data(), b2.size());
  464. return string_like_ins(s2, b1_lower, b1_upper);
  465. }
  466. // Slow version, used if caller hasn't stored an upper and lower case version
  467. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  468. {
  469. if (v2.is_null() || v1.is_null()) {
  470. return (v2.is_null() && v1.is_null());
  471. }
  472. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  473. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  474. return string_like_ins(v2, v1_lower, v1_upper);
  475. }
  476. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  477. {
  478. if (b2.is_null() || b1.is_null()) {
  479. return (b2.is_null() && b1.is_null());
  480. }
  481. StringData s1(b1.data(), b1.size());
  482. StringData s2(b2.data(), b2.size());
  483. std::string s1_upper = case_map(s1, true, IgnoreErrors);
  484. std::string s1_lower = case_map(s1, false, IgnoreErrors);
  485. return string_like_ins(s2, s1_lower, s1_upper);
  486. }
  487. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  488. {
  489. if (m1.is_null() && m2.is_null())
  490. return true;
  491. if (Mixed::types_are_comparable(m1, m2)) {
  492. BinaryData b1 = m1.get_binary();
  493. BinaryData b2 = m2.get_binary();
  494. return operator()(b1, b2, false, false);
  495. }
  496. return false;
  497. }
  498. template <class A, class B>
  499. bool operator()(A, B) const
  500. {
  501. REALM_ASSERT(false);
  502. return false;
  503. }
  504. template <class A, class B, class C, class D>
  505. bool operator()(A, B, C, D) const
  506. {
  507. REALM_ASSERT(false);
  508. return false;
  509. }
  510. bool operator()(int64_t, int64_t, bool, bool) const
  511. {
  512. REALM_ASSERT(false);
  513. return false;
  514. }
  515. static std::string description()
  516. {
  517. return "LIKE[c]";
  518. }
  519. static const int condition = -1;
  520. };
  521. // Does v2 begin with v1?
  522. struct BeginsWithIns : public HackClass {
  523. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  524. bool = false) const
  525. {
  526. if (v2.is_null() && !v1.is_null())
  527. return false;
  528. return v1.size() <= v2.size() && equal_case_fold(v2.prefix(v1.size()), v1_upper, v1_lower);
  529. }
  530. // Slow version, used if caller hasn't stored an upper and lower case version
  531. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  532. {
  533. if (v2.is_null() && !v1.is_null())
  534. return false;
  535. if (v1.size() > v2.size())
  536. return false;
  537. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  538. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  539. return equal_case_fold(v2.prefix(v1.size()), v1_upper.c_str(), v1_lower.c_str());
  540. }
  541. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  542. {
  543. StringData s1(b1.data(), b1.size());
  544. StringData s2(b2.data(), b2.size());
  545. return this->operator()(s1, s2, false, false);
  546. }
  547. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  548. {
  549. if (Mixed::types_are_comparable(m1, m2)) {
  550. BinaryData b1 = m1.get_binary();
  551. BinaryData b2 = m2.get_binary();
  552. return operator()(b1, b2, false, false);
  553. }
  554. return false;
  555. }
  556. template <class A, class B>
  557. bool operator()(A, B) const
  558. {
  559. REALM_ASSERT(false);
  560. return false;
  561. }
  562. template <class A, class B, class C, class D>
  563. bool operator()(A, B, C, D) const
  564. {
  565. REALM_ASSERT(false);
  566. return false;
  567. }
  568. bool operator()(int64_t, int64_t, bool, bool) const
  569. {
  570. REALM_ASSERT(false);
  571. return false;
  572. }
  573. static std::string description()
  574. {
  575. return "BEGINSWITH[c]";
  576. }
  577. static const int condition = -1;
  578. };
  579. // Does v2 end with v1?
  580. struct EndsWithIns : public HackClass {
  581. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  582. bool = false) const
  583. {
  584. if (v2.is_null() && !v1.is_null())
  585. return false;
  586. return v1.size() <= v2.size() && equal_case_fold(v2.suffix(v1.size()), v1_upper, v1_lower);
  587. }
  588. // Slow version, used if caller hasn't stored an upper and lower case version
  589. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  590. {
  591. if (v2.is_null() && !v1.is_null())
  592. return false;
  593. if (v1.size() > v2.size())
  594. return false;
  595. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  596. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  597. return equal_case_fold(v2.suffix(v1.size()), v1_upper.c_str(), v1_lower.c_str());
  598. }
  599. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  600. {
  601. StringData s1(b1.data(), b1.size());
  602. StringData s2(b2.data(), b2.size());
  603. return this->operator()(s1, s2, false, false);
  604. }
  605. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  606. {
  607. if (Mixed::types_are_comparable(m1, m2)) {
  608. BinaryData b1 = m1.get_binary();
  609. BinaryData b2 = m2.get_binary();
  610. return operator()(b1, b2, false, false);
  611. }
  612. return false;
  613. }
  614. template <class A, class B>
  615. bool operator()(A, B) const
  616. {
  617. REALM_ASSERT(false);
  618. return false;
  619. }
  620. template <class A, class B, class C, class D>
  621. bool operator()(A, B, C, D) const
  622. {
  623. REALM_ASSERT(false);
  624. return false;
  625. }
  626. bool operator()(int64_t, int64_t, bool, bool) const
  627. {
  628. REALM_ASSERT(false);
  629. return false;
  630. }
  631. static std::string description()
  632. {
  633. return "ENDSWITH[c]";
  634. }
  635. static const int condition = -1;
  636. };
  637. struct EqualIns : public HackClass {
  638. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  639. bool = false) const
  640. {
  641. if (v1.is_null() != v2.is_null())
  642. return false;
  643. return v1.size() == v2.size() && equal_case_fold(v2, v1_upper, v1_lower);
  644. }
  645. // Slow version, used if caller hasn't stored an upper and lower case version
  646. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  647. {
  648. if (v1.is_null() != v2.is_null())
  649. return false;
  650. if (v1.size() != v2.size())
  651. return false;
  652. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  653. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  654. return equal_case_fold(v2, v1_upper.c_str(), v1_lower.c_str());
  655. }
  656. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  657. {
  658. StringData s1(b1.data(), b1.size());
  659. StringData s2(b2.data(), b2.size());
  660. return this->operator()(s1, s2, false, false);
  661. }
  662. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  663. {
  664. return (m1.is_null() && m2.is_null()) ||
  665. (Mixed::types_are_comparable(m1, m2) && operator()(m1.get_binary(), m2.get_binary(), false, false));
  666. }
  667. template <class A, class B>
  668. bool operator()(A, B) const
  669. {
  670. REALM_ASSERT(false);
  671. return false;
  672. }
  673. template <class A, class B, class C, class D>
  674. bool operator()(A, B, C, D) const
  675. {
  676. REALM_ASSERT(false);
  677. return false;
  678. }
  679. bool operator()(int64_t, int64_t, bool, bool) const
  680. {
  681. REALM_ASSERT(false);
  682. return false;
  683. }
  684. static std::string description()
  685. {
  686. return "==[c]";
  687. }
  688. static const int condition = -1;
  689. };
  690. struct NotEqualIns : public HackClass {
  691. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  692. bool = false) const
  693. {
  694. if (v1.is_null() != v2.is_null())
  695. return true;
  696. return v1.size() != v2.size() || !equal_case_fold(v2, v1_upper, v1_lower);
  697. }
  698. // Slow version, used if caller hasn't stored an upper and lower case version
  699. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  700. {
  701. if (v1.is_null() != v2.is_null())
  702. return true;
  703. if (v1.size() != v2.size())
  704. return true;
  705. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  706. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  707. return !equal_case_fold(v2, v1_upper.c_str(), v1_lower.c_str());
  708. }
  709. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  710. {
  711. StringData s1(b1.data(), b1.size());
  712. StringData s2(b2.data(), b2.size());
  713. return this->operator()(s1, s2, false, false);
  714. }
  715. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  716. {
  717. return !EqualIns()(m1, m2);
  718. }
  719. template <class A, class B>
  720. bool operator()(A, B) const
  721. {
  722. REALM_ASSERT(false);
  723. return false;
  724. }
  725. template <class A, class B, class C, class D>
  726. bool operator()(A, B, C, D) const
  727. {
  728. REALM_ASSERT(false);
  729. return false;
  730. }
  731. static std::string description()
  732. {
  733. return "!=[c]";
  734. }
  735. static const int condition = -1;
  736. };
  737. struct Greater {
  738. static const int avx = 0x1E; // _CMP_GT_OQ
  739. template <class T>
  740. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  741. {
  742. if (v1null || v2null)
  743. return false;
  744. return v1 > v2;
  745. }
  746. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  747. {
  748. return Mixed::types_are_comparable(m1, m2) && (m1 > m2);
  749. }
  750. static const int condition = cond_Greater;
  751. template <class A, class B, class C, class D>
  752. bool operator()(A, B, C, D) const
  753. {
  754. REALM_ASSERT(false);
  755. return false;
  756. }
  757. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  758. {
  759. static_cast<void>(lbound);
  760. return ubound > v;
  761. }
  762. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  763. {
  764. static_cast<void>(ubound);
  765. return lbound > v;
  766. }
  767. static std::string description()
  768. {
  769. return ">";
  770. }
  771. };
  772. struct None {
  773. template <class T>
  774. bool operator()(const T&, const T&, bool = false, bool = false) const
  775. {
  776. return true;
  777. }
  778. static const int condition = cond_None;
  779. template <class A, class B, class C, class D>
  780. bool operator()(A, B, C, D) const
  781. {
  782. REALM_ASSERT(false);
  783. return false;
  784. }
  785. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  786. {
  787. static_cast<void>(lbound);
  788. static_cast<void>(ubound);
  789. static_cast<void>(v);
  790. return true;
  791. }
  792. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  793. {
  794. static_cast<void>(lbound);
  795. static_cast<void>(ubound);
  796. static_cast<void>(v);
  797. return true;
  798. }
  799. static std::string description()
  800. {
  801. return "none";
  802. }
  803. };
  804. struct NotNull {
  805. template <class T>
  806. bool operator()(const T&, const T&, bool v = false, bool = false) const
  807. {
  808. return !v;
  809. }
  810. static const int condition = cond_LeftNotNull;
  811. template <class A, class B, class C, class D>
  812. bool operator()(A, B, C, D) const
  813. {
  814. REALM_ASSERT(false);
  815. return false;
  816. }
  817. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  818. {
  819. static_cast<void>(lbound);
  820. static_cast<void>(ubound);
  821. static_cast<void>(v);
  822. return true;
  823. }
  824. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  825. {
  826. static_cast<void>(lbound);
  827. static_cast<void>(ubound);
  828. static_cast<void>(v);
  829. return true;
  830. }
  831. static std::string description()
  832. {
  833. return "!= NULL";
  834. }
  835. };
  836. struct Less {
  837. static const int avx = 0x11; // _CMP_LT_OQ
  838. template <class T>
  839. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  840. {
  841. if (v1null || v2null)
  842. return false;
  843. return v1 < v2;
  844. }
  845. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  846. {
  847. return Mixed::types_are_comparable(m1, m2) && (m1 < m2);
  848. }
  849. template <class A, class B, class C, class D>
  850. bool operator()(A, B, C, D) const
  851. {
  852. REALM_ASSERT(false);
  853. return false;
  854. }
  855. static const int condition = cond_Less;
  856. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  857. {
  858. static_cast<void>(ubound);
  859. return lbound < v;
  860. }
  861. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  862. {
  863. static_cast<void>(lbound);
  864. return ubound < v;
  865. }
  866. static std::string description()
  867. {
  868. return "<";
  869. }
  870. };
  871. struct LessEqual : public HackClass {
  872. static const int avx = 0x12; // _CMP_LE_OQ
  873. template <class T>
  874. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  875. {
  876. if (v1null && v2null)
  877. return true;
  878. return (!v1null && !v2null && v1 <= v2);
  879. }
  880. bool operator()(const util::Optional<bool>& v1, const util::Optional<bool>& v2, bool v1null, bool v2null) const
  881. {
  882. if (v1null && v2null)
  883. return false;
  884. return (!v1null && !v2null && v1.value() <= v2.value());
  885. }
  886. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  887. {
  888. return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 <= m2));
  889. }
  890. template <class A, class B, class C, class D>
  891. bool operator()(A, B, C, D) const
  892. {
  893. REALM_ASSERT(false);
  894. return false;
  895. }
  896. static std::string description()
  897. {
  898. return "<=";
  899. }
  900. static const int condition = -1;
  901. };
  902. struct GreaterEqual : public HackClass {
  903. static const int avx = 0x1D; // _CMP_GE_OQ
  904. template <class T>
  905. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  906. {
  907. if (v1null && v2null)
  908. return true;
  909. return (!v1null && !v2null && v1 >= v2);
  910. }
  911. bool operator()(const util::Optional<bool>& v1, const util::Optional<bool>& v2, bool v1null, bool v2null) const
  912. {
  913. if (v1null && v2null)
  914. return false;
  915. return (!v1null && !v2null && v1.value() >= v2.value());
  916. }
  917. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  918. {
  919. return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 >= m2));
  920. }
  921. template <class A, class B, class C, class D>
  922. bool operator()(A, B, C, D) const
  923. {
  924. REALM_ASSERT(false);
  925. return false;
  926. }
  927. static std::string description()
  928. {
  929. return ">=";
  930. }
  931. static const int condition = -1;
  932. };
  933. // CompareLess is a temporary hack to have a generalized way to compare any realm types. Todo, enable correct <
  934. // operator of StringData (currently gives circular header dependency with utf8.hpp)
  935. template <class T>
  936. struct CompareLess {
  937. static bool compare(T v1, T v2, bool = false, bool = false)
  938. {
  939. return v1 < v2;
  940. }
  941. };
  942. template <>
  943. struct CompareLess<StringData> {
  944. static bool compare(StringData v1, StringData v2, bool = false, bool = false)
  945. {
  946. bool ret = utf8_compare(v1.data(), v2.data());
  947. return ret;
  948. }
  949. };
  950. } // namespace realm
  951. #endif // REALM_QUERY_CONDITIONS_HPP