123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383 |
- /*************************************************************************
- *
- * 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_ARRAY_INTEGER_HPP
- #define REALM_ARRAY_INTEGER_HPP
- #include <realm/array.hpp>
- #include <realm/util/safe_int_ops.hpp>
- #include <realm/util/optional.hpp>
- #include <realm/array_key.hpp>
- namespace realm {
- class ArrayInteger : public Array, public ArrayPayload {
- public:
- using value_type = int64_t;
- using Array::add;
- using Array::get;
- using Array::insert;
- using Array::move;
- using Array::set;
- explicit ArrayInteger(Allocator&) noexcept;
- ~ArrayInteger() noexcept override {}
- static value_type default_value(bool)
- {
- return 0;
- }
- void init_from_ref(ref_type ref) noexcept override
- {
- Array::init_from_ref(ref);
- }
- void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
- {
- Array::set_parent(parent, ndx_in_parent);
- }
- // Disable copying, this is not allowed.
- ArrayInteger& operator=(const ArrayInteger&) = delete;
- ArrayInteger(const ArrayInteger&) = delete;
- void create()
- {
- Array::create(type_Normal, false, 0, 0);
- }
- Mixed get_any(size_t ndx) const override;
- bool is_null(size_t) const
- {
- return false;
- }
- };
- class ArrayIntNull : public Array, public ArrayPayload {
- public:
- using value_type = util::Optional<int64_t>;
- explicit ArrayIntNull(Allocator&) noexcept;
- ~ArrayIntNull() noexcept override;
- static value_type default_value(bool nullable)
- {
- return nullable ? util::none : util::Optional<int64_t>(0);
- }
- /// Construct an array of the specified type and size, and return just the
- /// reference to the underlying memory. All elements will be initialized to
- /// the specified value.
- static MemRef create_array(Type, bool context_flag, size_t size, Allocator&);
- void create()
- {
- MemRef r = create_array(type_Normal, false, 0, m_alloc);
- init_from_mem(r);
- }
- void init_from_ref(ref_type) noexcept override;
- void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
- {
- Array::set_parent(parent, ndx_in_parent);
- }
- void init_from_mem(MemRef) noexcept;
- void init_from_parent() noexcept;
- size_t size() const noexcept;
- bool is_empty() const noexcept;
- void insert(size_t ndx, value_type value);
- void add(value_type value);
- void set(size_t ndx, value_type value);
- value_type get(size_t ndx) const noexcept;
- Mixed get_any(size_t ndx) const override;
- static value_type get(const char* header, size_t ndx) noexcept;
- void get_chunk(size_t ndx, value_type res[8]) const noexcept;
- void set_null(size_t ndx);
- bool is_null(size_t ndx) const noexcept;
- int64_t null_value() const noexcept;
- void erase(size_t ndx);
- void erase(size_t begin, size_t end);
- void move(ArrayIntNull& dst, size_t ndx);
- void clear();
- void move(size_t begin, size_t end, size_t dest_begin);
- bool find(int cond, value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state) const;
- // This is the one installed into the m_finder slots.
- template <class cond, size_t bitwidth>
- bool find(int64_t value, size_t start, size_t end, size_t baseindex, QueryStateBase* state) const;
- template <class cond, class Callback>
- bool find(value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
- Callback callback) const;
- // Wrappers for backwards compatibility and for simple use without
- // setting up state initialization etc
- template <class cond>
- size_t find_first(value_type value, size_t start = 0, size_t end = npos) const;
- void find_all(IntegerColumn* result, value_type value, size_t col_offset = 0, size_t begin = 0,
- size_t end = npos) const;
- size_t find_first(value_type value, size_t begin = 0, size_t end = npos) const;
- protected:
- void avoid_null_collision(int64_t value);
- private:
- int_fast64_t choose_random_null(int64_t incoming) const;
- void replace_nulls_with(int64_t new_null);
- bool can_use_as_null(int64_t value) const;
- template <class Callback>
- bool find_impl(int cond, value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
- Callback callback) const;
- template <class cond, class Callback>
- bool find_impl(value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
- Callback callback) const;
- };
- // Implementation:
- inline ArrayInteger::ArrayInteger(Allocator& allocator) noexcept
- : Array(allocator)
- {
- m_is_inner_bptree_node = false;
- }
- inline ArrayIntNull::ArrayIntNull(Allocator& allocator) noexcept
- : Array(allocator)
- {
- }
- inline ArrayIntNull::~ArrayIntNull() noexcept {}
- inline size_t ArrayIntNull::size() const noexcept
- {
- return Array::size() - 1;
- }
- inline bool ArrayIntNull::is_empty() const noexcept
- {
- return size() == 0;
- }
- inline void ArrayIntNull::insert(size_t ndx, value_type value)
- {
- if (value) {
- avoid_null_collision(*value);
- Array::insert(ndx + 1, *value);
- }
- else {
- Array::insert(ndx + 1, null_value());
- }
- }
- inline void ArrayIntNull::add(value_type value)
- {
- if (value) {
- avoid_null_collision(*value);
- Array::add(*value);
- }
- else {
- Array::add(null_value());
- }
- }
- inline void ArrayIntNull::set(size_t ndx, value_type value)
- {
- if (value) {
- avoid_null_collision(*value);
- Array::set(ndx + 1, *value);
- }
- else {
- Array::set(ndx + 1, null_value());
- }
- }
- inline void ArrayIntNull::set_null(size_t ndx)
- {
- Array::set(ndx + 1, null_value());
- }
- inline ArrayIntNull::value_type ArrayIntNull::get(size_t ndx) const noexcept
- {
- int64_t value = Array::get(ndx + 1);
- if (value == null_value()) {
- return util::none;
- }
- return util::some<int64_t>(value);
- }
- inline ArrayIntNull::value_type ArrayIntNull::get(const char* header, size_t ndx) noexcept
- {
- int64_t null_value = Array::get(header, 0);
- int64_t value = Array::get(header, ndx + 1);
- if (value == null_value) {
- return util::none;
- }
- else {
- return util::some<int64_t>(value);
- }
- }
- inline bool ArrayIntNull::is_null(size_t ndx) const noexcept
- {
- return !get(ndx);
- }
- inline int64_t ArrayIntNull::null_value() const noexcept
- {
- return Array::get(0);
- }
- inline void ArrayIntNull::erase(size_t ndx)
- {
- Array::erase(ndx + 1);
- }
- inline void ArrayIntNull::erase(size_t begin, size_t end)
- {
- Array::erase(begin + 1, end + 1);
- }
- inline void ArrayIntNull::clear()
- {
- Array::truncate(0);
- Array::add(0);
- }
- inline void ArrayIntNull::move(size_t begin, size_t end, size_t dest_begin)
- {
- Array::move(begin + 1, end + 1, dest_begin + 1);
- }
- inline bool ArrayIntNull::find(int cond, value_type value, size_t start, size_t end, size_t baseindex,
- QueryStateBase* state) const
- {
- return find_impl(cond, value, start, end, baseindex, state, nullptr);
- }
- template <class cond, class Callback>
- inline bool ArrayIntNull::find(value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
- Callback callback) const
- {
- return find_impl<cond, Callback>(value, start, end, baseindex, state, callback);
- }
- template <class Callback>
- inline bool ArrayIntNull::find_impl(int cond, value_type value, size_t start, size_t end, size_t baseindex,
- QueryStateBase* state, Callback callback) const
- {
- switch (cond) {
- case cond_Equal:
- return find_impl<Equal>(value, start, end, baseindex, state, callback);
- case cond_NotEqual:
- return find_impl<NotEqual>(value, start, end, baseindex, state, callback);
- case cond_Greater:
- return find_impl<Greater>(value, start, end, baseindex, state, callback);
- case cond_Less:
- return find_impl<Less>(value, start, end, baseindex, state, callback);
- case cond_None:
- return find_impl<None>(value, start, end, baseindex, state, callback);
- case cond_LeftNotNull:
- return find_impl<NotNull>(value, start, end, baseindex, state, callback);
- }
- REALM_ASSERT_DEBUG(false);
- return false;
- }
- template <class cond, class Callback>
- bool ArrayIntNull::find_impl(value_type opt_value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
- Callback callback) const
- {
- int64_t null_value = Array::get(0);
- bool find_null = !bool(opt_value);
- int64_t value;
- size_t end2 = (end == npos ? size() : end) + 1;
- size_t start2 = start + 1;
- size_t baseindex2 = baseindex - 1;
- if constexpr (std::is_same_v<cond, Equal>) {
- if (find_null) {
- value = null_value;
- }
- else {
- if (*opt_value == null_value) {
- // If the value to search for is equal to the null value, the value cannot be in the array
- return true;
- }
- else {
- value = *opt_value;
- }
- }
- // Fall back to plain Array find.
- return Array::find<cond>(value, start2, end2, baseindex2, state, callback);
- }
- else {
- cond c;
- if (opt_value) {
- value = *opt_value;
- }
- else {
- value = null_value;
- }
- for (size_t i = start2; i < end2; ++i) {
- int64_t v = Array::get(i);
- bool value_is_null = (v == null_value);
- if (c(v, value, value_is_null, find_null)) {
- util::Optional<int64_t> v2 = value_is_null ? util::none : util::make_optional(v);
- if (!Array::find_action(i + baseindex2, v2, state, callback)) {
- return false; // tell caller to stop aggregating/search
- }
- }
- }
- return true; // tell caller to continue aggregating/search (on next array leafs)
- }
- }
- template <class cond>
- size_t ArrayIntNull::find_first(value_type value, size_t start, size_t end) const
- {
- QueryStateFindFirst state;
- find_impl<cond>(value, start, end, 0, &state, nullptr);
- if (state.match_count() > 0)
- return to_size_t(state.m_state);
- else
- return not_found;
- }
- inline size_t ArrayIntNull::find_first(value_type value, size_t begin, size_t end) const
- {
- return find_first<Equal>(value, begin, end);
- }
- } // namespace realm
- #endif // REALM_ARRAY_INTEGER_HPP
|