123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359 |
- /*************************************************************************
- *
- * 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_KEYS_HPP
- #define REALM_KEYS_HPP
- #include <realm/util/to_string.hpp>
- #include <realm/column_type.hpp>
- #include <ostream>
- #include <vector>
- namespace realm {
- class Obj;
- struct TableKey {
- static constexpr uint32_t null_value = uint32_t(-1) >> 1; // free top bit
- constexpr TableKey() noexcept
- : value(null_value)
- {
- }
- constexpr explicit TableKey(uint32_t val) noexcept
- : value(val)
- {
- }
- constexpr bool operator==(const TableKey& rhs) const noexcept
- {
- return value == rhs.value;
- }
- constexpr bool operator!=(const TableKey& rhs) const noexcept
- {
- return value != rhs.value;
- }
- constexpr bool operator<(const TableKey& rhs) const noexcept
- {
- return value < rhs.value;
- }
- constexpr bool operator>(const TableKey& rhs) const noexcept
- {
- return value > rhs.value;
- }
- constexpr explicit operator bool() const noexcept
- {
- return value != null_value;
- }
- uint32_t value;
- };
- inline std::ostream& operator<<(std::ostream& os, TableKey tk)
- {
- os << "TableKey(" << tk.value << ")";
- return os;
- }
- namespace util {
- inline std::string to_string(TableKey tk)
- {
- return to_string(tk.value);
- }
- } // namespace util
- class TableVersions : public std::vector<std::pair<TableKey, uint64_t>> {
- public:
- TableVersions() {}
- TableVersions(TableKey key, uint64_t version)
- {
- emplace_back(key, version);
- }
- bool operator==(const TableVersions& other) const;
- };
- struct ColKey {
- static constexpr int64_t null_value = int64_t(uint64_t(-1) >> 1); // free top bit
- struct Idx {
- unsigned val;
- };
- constexpr ColKey() noexcept = default;
- constexpr explicit ColKey(int64_t val) noexcept
- : value(val)
- {
- }
- constexpr ColKey(Idx index, ColumnType type, ColumnAttrMask attrs, uint64_t tag) noexcept
- : ColKey((index.val & 0xFFFFUL) | ((int(type) & 0x3FUL) << 16) | ((attrs.m_value & 0xFFUL) << 22) |
- ((tag & 0xFFFFFFFFUL) << 30))
- {
- }
- bool is_nullable() const
- {
- return get_attrs().test(col_attr_Nullable);
- }
- bool is_list() const
- {
- return get_attrs().test(col_attr_List);
- }
- bool is_set() const
- {
- return get_attrs().test(col_attr_Set);
- }
- bool is_dictionary() const
- {
- return get_attrs().test(col_attr_Dictionary);
- }
- bool is_collection() const
- {
- return get_attrs().test(col_attr_Collection);
- }
- bool operator==(const ColKey& rhs) const noexcept
- {
- return value == rhs.value;
- }
- bool operator!=(const ColKey& rhs) const noexcept
- {
- return value != rhs.value;
- }
- bool operator<(const ColKey& rhs) const noexcept
- {
- return value < rhs.value;
- }
- bool operator>(const ColKey& rhs) const noexcept
- {
- return value > rhs.value;
- }
- explicit operator bool() const noexcept
- {
- return value != null_value;
- }
- Idx get_index() const noexcept
- {
- return Idx{static_cast<unsigned>(value) & 0xFFFFU};
- }
- ColumnType get_type() const noexcept
- {
- return ColumnType(ColumnType::Type((static_cast<unsigned>(value) >> 16) & 0x3F));
- }
- ColumnAttrMask get_attrs() const noexcept
- {
- return ColumnAttrMask((static_cast<unsigned>(value) >> 22) & 0xFF);
- }
- unsigned get_tag() const noexcept
- {
- return (value >> 30) & 0xFFFFFFFFUL;
- }
- int64_t value = null_value;
- };
- static_assert(ColKey::null_value == 0x7fffffffffffffff, "Fix this");
- inline std::ostream& operator<<(std::ostream& os, ColKey ck)
- {
- os << "ColKey(" << ck.value << ")";
- return os;
- }
- struct ObjKey {
- constexpr ObjKey() noexcept
- : value(-1)
- {
- }
- explicit constexpr ObjKey(int64_t val) noexcept
- : value(val)
- {
- }
- bool is_unresolved() const
- {
- return value <= -2;
- }
- ObjKey get_unresolved() const
- {
- return ObjKey(-2 - value);
- }
- bool operator==(const ObjKey& rhs) const noexcept
- {
- return value == rhs.value;
- }
- bool operator!=(const ObjKey& rhs) const noexcept
- {
- return value != rhs.value;
- }
- bool operator<(const ObjKey& rhs) const noexcept
- {
- return value < rhs.value;
- }
- bool operator<=(const ObjKey& rhs) const noexcept
- {
- return value <= rhs.value;
- }
- bool operator>(const ObjKey& rhs) const noexcept
- {
- return value > rhs.value;
- }
- bool operator>=(const ObjKey& rhs) const noexcept
- {
- return value >= rhs.value;
- }
- explicit operator bool() const noexcept
- {
- return value != -1;
- }
- int64_t value;
- private:
- // operator bool will enable casting to integer. Prevent this.
- operator int64_t() const = delete;
- };
- inline std::ostream& operator<<(std::ostream& ostr, ObjKey key)
- {
- ostr << "ObjKey(" << key.value << ")";
- return ostr;
- }
- class ObjKeys : public std::vector<ObjKey> {
- public:
- explicit ObjKeys(const std::vector<int64_t>& init)
- {
- reserve(init.size());
- for (auto i : init) {
- emplace_back(i);
- }
- }
- ObjKeys(const std::initializer_list<int64_t>& list)
- {
- reserve(list.size());
- for (auto i : list) {
- emplace_back(i);
- }
- }
- ObjKeys() {}
- };
- struct ObjLink {
- public:
- constexpr ObjLink() = default;
- constexpr ObjLink(TableKey table_key, ObjKey obj_key)
- : m_obj_key(obj_key)
- , m_table_key(table_key)
- {
- }
- explicit operator bool() const
- {
- return bool(m_table_key) && bool(m_obj_key);
- }
- bool is_null() const
- {
- return !bool(*this);
- }
- bool is_unresolved() const
- {
- return m_obj_key.is_unresolved();
- }
- bool operator==(const ObjLink& other) const
- {
- return m_obj_key == other.m_obj_key && m_table_key == other.m_table_key;
- }
- bool operator!=(const ObjLink& other) const
- {
- return m_obj_key != other.m_obj_key || m_table_key != other.m_table_key;
- }
- bool operator<(const ObjLink& rhs) const
- {
- if (m_table_key < rhs.m_table_key) {
- return true;
- }
- else if (m_table_key == rhs.m_table_key) {
- return m_obj_key < rhs.m_obj_key;
- }
- else {
- // m_table_key >= rhs.m_table_key
- return false;
- }
- }
- bool operator>(const ObjLink& rhs) const
- {
- return (*this != rhs) && !(*this < rhs);
- }
- TableKey get_table_key() const
- {
- return m_table_key;
- }
- ObjKey get_obj_key() const
- {
- return m_obj_key;
- }
- private:
- // Having ObjKey first ensures that there will be no uninitialized space
- // in the first 12 bytes. This is important when generating a hash
- ObjKey m_obj_key;
- TableKey m_table_key;
- };
- inline std::ostream& operator<<(std::ostream& os, ObjLink link)
- {
- os << '{' << link.get_table_key() << ',' << link.get_obj_key() << '}';
- return os;
- }
- constexpr ObjKey null_key;
- namespace util {
- inline std::string to_string(ColKey ck)
- {
- return to_string(ck.value);
- }
- } // namespace util
- } // namespace realm
- namespace std {
- template <>
- struct hash<realm::ObjKey> {
- size_t operator()(realm::ObjKey key) const
- {
- return std::hash<uint64_t>{}(key.value);
- }
- };
- template <>
- struct hash<realm::ColKey> {
- size_t operator()(realm::ColKey key) const
- {
- return std::hash<int64_t>{}(key.value);
- }
- };
- template <>
- struct hash<realm::TableKey> {
- size_t operator()(realm::TableKey key) const
- {
- return std::hash<uint32_t>{}(key.value);
- }
- };
- } // namespace std
- #endif
|