123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011 |
- /*************************************************************************
- *
- * Copyright 2016 Realm Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- **************************************************************************/
- #ifndef REALM_QUERY_CONDITIONS_HPP
- #define REALM_QUERY_CONDITIONS_HPP
- #include <cstdint>
- #include <string>
- #include <realm/query_state.hpp>
- #include <realm/unicode.hpp>
- #include <realm/binary_data.hpp>
- #include <realm/query_value.hpp>
- #include <realm/mixed.hpp>
- #include <realm/utilities.hpp>
- namespace realm {
- // Quick hack to make "Queries with Integer null columns" able to compile in Visual Studio 2015 which doesn't full
- // support sfinae
- // (real cause hasn't been investigated yet, cannot exclude that we don't obey c++11 standard)
- struct HackClass {
- template <class A, class B, class C>
- bool can_match(A, B, C)
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C>
- bool will_match(A, B, C)
- {
- REALM_ASSERT(false);
- return false;
- }
- };
- // Does v2 contain v1?
- struct Contains : public HackClass {
- bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
- {
- return v2.contains(v1);
- }
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- return v2.contains(v1);
- }
- bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
- {
- return v2.contains(v1);
- }
- bool operator()(StringData v1, const std::array<uint8_t, 256> &charmap, StringData v2) const
- {
- return v2.contains(v1, charmap);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_null())
- return !m2.is_null();
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "CONTAINS";
- }
- static const int condition = -1;
- };
- // Does v2 contain something like v1 (wildcard matching)?
- struct Like : public HackClass {
- bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
- {
- return v2.like(v1);
- }
- bool operator()(BinaryData b1, const char*, const char*, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return s2.like(s1);
- }
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- return v2.like(v1);
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return s2.like(s1);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_null() && m2.is_null())
- return true;
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "LIKE";
- }
- static const int condition = -1;
- };
- // Does v2 begin with v1?
- struct BeginsWith : public HackClass {
- bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
- {
- return v2.begins_with(v1);
- }
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- return v2.begins_with(v1);
- }
- bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
- {
- return v2.begins_with(v1);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return b2.begins_with(b1);
- }
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "BEGINSWITH";
- }
- static const int condition = -1;
- };
- // Does v2 end with v1?
- struct EndsWith : public HackClass {
- bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
- {
- return v2.ends_with(v1);
- }
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- return v2.ends_with(v1);
- }
- bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
- {
- return v2.ends_with(v1);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "ENDSWITH";
- }
- static const int condition = -1;
- };
- struct Equal {
- static const int avx = 0x00; // _CMP_EQ_OQ
- // bool operator()(const bool v1, const bool v2, bool v1null = false, bool v2null = false) const { return v1 ==
- // v2; }
- bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
- {
- return v1 == v2;
- }
- bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
- {
- return v1 == v2;
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 == m2));
- }
- template <class T>
- bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
- {
- return (v1null && v2null) || (!v1null && !v2null && v1 == v2);
- }
- static const int condition = cond_Equal;
- bool can_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- return (v >= lbound && v <= ubound);
- }
- bool will_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- return (v == 0 && ubound == 0 && lbound == 0);
- }
- static std::string description()
- {
- return "==";
- }
- };
- struct NotEqual {
- static const int avx = 0x0B; // _CMP_FALSE_OQ
- bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
- {
- return v1 != v2;
- }
- // bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const { return v1 != v2; }
- template <class T>
- bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
- {
- if (!v1null && !v2null)
- return v1 != v2;
- if (v1null && v2null)
- return false;
- return true;
- }
- bool operator()(const QueryValue& m1, const Mixed& m2) const
- {
- return !Equal()(m1, m2);
- }
- static const int condition = cond_NotEqual;
- bool can_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- return !(v == 0 && ubound == 0 && lbound == 0);
- }
- bool will_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- return (v > ubound || v < lbound);
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const = delete;
- static std::string description()
- {
- return "!=";
- }
- };
- // Does v2 contain v1?
- struct ContainsIns : public HackClass {
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
- bool = false) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
- if (v1.size() == 0 && !v2.is_null())
- return true;
- return search_case_fold(v2, v1_upper, v1_lower, v1.size()) != v2.size();
- }
- // Slow version, used if caller hasn't stored an upper and lower case version
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
- if (v1.size() == 0 && !v2.is_null())
- return true;
- std::string v1_upper = case_map(v1, true, IgnoreErrors);
- std::string v1_lower = case_map(v1, false, IgnoreErrors);
- return search_case_fold(v2, v1_upper.c_str(), v1_lower.c_str(), v1.size()) != v2.size();
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return this->operator()(s1, s2, false, false);
- }
- // Case insensitive Boyer-Moore version
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, const std::array<uint8_t, 256> &charmap, StringData v2) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
-
- if (v1.size() == 0 && !v2.is_null())
- return true;
-
- return contains_ins(v2, v1_upper, v1_lower, v1.size(), charmap);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_null())
- return !m2.is_null();
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "CONTAINS[c]";
- }
- static const int condition = -1;
- };
- // Does v2 contain something like v1 (wildcard matching)?
- struct LikeIns : public HackClass {
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
- bool = false) const
- {
- if (v2.is_null() || v1.is_null()) {
- return (v2.is_null() && v1.is_null());
- }
- return string_like_ins(v2, v1_lower, v1_upper);
- }
- bool operator()(BinaryData b1, const char* b1_upper, const char* b1_lower, BinaryData b2, bool = false,
- bool = false) const
- {
- if (b2.is_null() || b1.is_null()) {
- return (b2.is_null() && b1.is_null());
- }
- StringData s2(b2.data(), b2.size());
- return string_like_ins(s2, b1_lower, b1_upper);
- }
- // Slow version, used if caller hasn't stored an upper and lower case version
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- if (v2.is_null() || v1.is_null()) {
- return (v2.is_null() && v1.is_null());
- }
- std::string v1_upper = case_map(v1, true, IgnoreErrors);
- std::string v1_lower = case_map(v1, false, IgnoreErrors);
- return string_like_ins(v2, v1_lower, v1_upper);
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- if (b2.is_null() || b1.is_null()) {
- return (b2.is_null() && b1.is_null());
- }
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- std::string s1_upper = case_map(s1, true, IgnoreErrors);
- std::string s1_lower = case_map(s1, false, IgnoreErrors);
- return string_like_ins(s2, s1_lower, s1_upper);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_null() && m2.is_null())
- return true;
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "LIKE[c]";
- }
- static const int condition = -1;
- };
- // Does v2 begin with v1?
- struct BeginsWithIns : public HackClass {
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
- bool = false) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
- return v1.size() <= v2.size() && equal_case_fold(v2.prefix(v1.size()), v1_upper, v1_lower);
- }
- // Slow version, used if caller hasn't stored an upper and lower case version
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
- if (v1.size() > v2.size())
- return false;
- std::string v1_upper = case_map(v1, true, IgnoreErrors);
- std::string v1_lower = case_map(v1, false, IgnoreErrors);
- return equal_case_fold(v2.prefix(v1.size()), v1_upper.c_str(), v1_lower.c_str());
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return this->operator()(s1, s2, false, false);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "BEGINSWITH[c]";
- }
- static const int condition = -1;
- };
- // Does v2 end with v1?
- struct EndsWithIns : public HackClass {
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
- bool = false) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
- return v1.size() <= v2.size() && equal_case_fold(v2.suffix(v1.size()), v1_upper, v1_lower);
- }
- // Slow version, used if caller hasn't stored an upper and lower case version
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- if (v2.is_null() && !v1.is_null())
- return false;
- if (v1.size() > v2.size())
- return false;
- std::string v1_upper = case_map(v1, true, IgnoreErrors);
- std::string v1_lower = case_map(v1, false, IgnoreErrors);
- return equal_case_fold(v2.suffix(v1.size()), v1_upper.c_str(), v1_lower.c_str());
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return this->operator()(s1, s2, false, false);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
- BinaryData b1 = m1.get_binary();
- BinaryData b2 = m2.get_binary();
- return operator()(b1, b2, false, false);
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "ENDSWITH[c]";
- }
- static const int condition = -1;
- };
- struct EqualIns : public HackClass {
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
- bool = false) const
- {
- if (v1.is_null() != v2.is_null())
- return false;
- return v1.size() == v2.size() && equal_case_fold(v2, v1_upper, v1_lower);
- }
- // Slow version, used if caller hasn't stored an upper and lower case version
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- if (v1.is_null() != v2.is_null())
- return false;
- if (v1.size() != v2.size())
- return false;
- std::string v1_upper = case_map(v1, true, IgnoreErrors);
- std::string v1_lower = case_map(v1, false, IgnoreErrors);
- return equal_case_fold(v2, v1_upper.c_str(), v1_lower.c_str());
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return this->operator()(s1, s2, false, false);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- if (m1.is_null() && m2.is_null()) {
- return true;
- }
- else if (Mixed::types_are_comparable(m1, m2)) {
- if (m1.is_type(type_String, type_Binary)) {
- return operator()(m1.get_binary(), m2.get_binary(), false, false);
- }
- else {
- return m1 == m2;
- }
- }
- return false;
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool operator()(int64_t, int64_t, bool, bool) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "==[c]";
- }
- static const int condition = -1;
- };
- struct NotEqualIns : public HackClass {
- bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
- bool = false) const
- {
- if (v1.is_null() != v2.is_null())
- return true;
- return v1.size() != v2.size() || !equal_case_fold(v2, v1_upper, v1_lower);
- }
- // Slow version, used if caller hasn't stored an upper and lower case version
- bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
- {
- if (v1.is_null() != v2.is_null())
- return true;
- if (v1.size() != v2.size())
- return true;
- std::string v1_upper = case_map(v1, true, IgnoreErrors);
- std::string v1_lower = case_map(v1, false, IgnoreErrors);
- return !equal_case_fold(v2, v1_upper.c_str(), v1_lower.c_str());
- }
- bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
- {
- StringData s1(b1.data(), b1.size());
- StringData s2(b2.data(), b2.size());
- return this->operator()(s1, s2, false, false);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- return !EqualIns()(m1, m2);
- }
- template <class A, class B>
- bool operator()(A, B) const
- {
- REALM_ASSERT(false);
- return false;
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "!=[c]";
- }
- static const int condition = -1;
- };
- struct Greater {
- static const int avx = 0x1E; // _CMP_GT_OQ
- template <class T>
- bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
- {
- if (v1null || v2null)
- return false;
- return v1 > v2;
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- return Mixed::types_are_comparable(m1, m2) && (m1 > m2);
- }
- static const int condition = cond_Greater;
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool can_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(lbound);
- return ubound > v;
- }
- bool will_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(ubound);
- return lbound > v;
- }
- static std::string description()
- {
- return ">";
- }
- };
- struct None {
- template <class T>
- bool operator()(const T&, const T&, bool = false, bool = false) const
- {
- return true;
- }
- static const int condition = cond_None;
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool can_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(lbound);
- static_cast<void>(ubound);
- static_cast<void>(v);
- return true;
- }
- bool will_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(lbound);
- static_cast<void>(ubound);
- static_cast<void>(v);
- return true;
- }
- static std::string description()
- {
- return "none";
- }
- };
- struct NotNull {
- template <class T>
- bool operator()(const T&, const T&, bool v = false, bool = false) const
- {
- return !v;
- }
- static const int condition = cond_LeftNotNull;
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- bool can_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(lbound);
- static_cast<void>(ubound);
- static_cast<void>(v);
- return true;
- }
- bool will_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(lbound);
- static_cast<void>(ubound);
- static_cast<void>(v);
- return true;
- }
- static std::string description()
- {
- return "!= NULL";
- }
- };
- struct Less {
- static const int avx = 0x11; // _CMP_LT_OQ
- template <class T>
- bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
- {
- if (v1null || v2null)
- return false;
- return v1 < v2;
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- return Mixed::types_are_comparable(m1, m2) && (m1 < m2);
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static const int condition = cond_Less;
- bool can_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(ubound);
- return lbound < v;
- }
- bool will_match(int64_t v, int64_t lbound, int64_t ubound)
- {
- static_cast<void>(lbound);
- return ubound < v;
- }
- static std::string description()
- {
- return "<";
- }
- };
- struct LessEqual : public HackClass {
- static const int avx = 0x12; // _CMP_LE_OQ
- template <class T>
- bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
- {
- if (v1null && v2null)
- return true;
- return (!v1null && !v2null && v1 <= v2);
- }
- bool operator()(const util::Optional<bool>& v1, const util::Optional<bool>& v2, bool v1null, bool v2null) const
- {
- if (v1null && v2null)
- return false;
- return (!v1null && !v2null && *v1 <= *v2);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 <= m2));
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return "<=";
- }
- static const int condition = -1;
- };
- struct GreaterEqual : public HackClass {
- static const int avx = 0x1D; // _CMP_GE_OQ
- template <class T>
- bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
- {
- if (v1null && v2null)
- return true;
- return (!v1null && !v2null && v1 >= v2);
- }
- bool operator()(const util::Optional<bool>& v1, const util::Optional<bool>& v2, bool v1null, bool v2null) const
- {
- if (v1null && v2null)
- return false;
- return (!v1null && !v2null && *v1 >= *v2);
- }
- bool operator()(const QueryValue& m1, const QueryValue& m2) const
- {
- return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 >= m2));
- }
- template <class A, class B, class C, class D>
- bool operator()(A, B, C, D) const
- {
- REALM_ASSERT(false);
- return false;
- }
- static std::string description()
- {
- return ">=";
- }
- static const int condition = -1;
- };
- // CompareLess is a temporary hack to have a generalized way to compare any realm types. Todo, enable correct <
- // operator of StringData (currently gives circular header dependency with utf8.hpp)
- template <class T>
- struct CompareLess {
- static bool compare(T v1, T v2, bool = false, bool = false)
- {
- return v1 < v2;
- }
- };
- template <>
- struct CompareLess<StringData> {
- static bool compare(StringData v1, StringData v2, bool = false, bool = false)
- {
- bool ret = utf8_compare(v1.data(), v2.data());
- return ret;
- }
- };
- } // namespace realm
- #endif // REALM_QUERY_CONDITIONS_HPP
|