| /* |
| * Copyright (C) 2019 The Android Open Source Project |
| * |
| * 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 SRC_TRACE_PROCESSOR_CONTAINERS_NULLABLE_VECTOR_H_ |
| #define SRC_TRACE_PROCESSOR_CONTAINERS_NULLABLE_VECTOR_H_ |
| |
| #include <stdint.h> |
| |
| #include <deque> |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/ext/base/optional.h" |
| #include "src/trace_processor/containers/row_map.h" |
| |
| namespace perfetto { |
| namespace trace_processor { |
| |
| // Base class for NullableVector which allows type erasure to be implemented |
| // (e.g. allows for std::unique_ptr<NullableVectorBase>). |
| class NullableVectorBase { |
| public: |
| NullableVectorBase() = default; |
| virtual ~NullableVectorBase(); |
| |
| NullableVectorBase(const NullableVectorBase&) = delete; |
| NullableVectorBase& operator=(const NullableVectorBase&) = delete; |
| |
| NullableVectorBase(NullableVectorBase&&) = default; |
| NullableVectorBase& operator=(NullableVectorBase&&) noexcept = default; |
| }; |
| |
| // A data structure which compactly stores a list of possibly nullable data. |
| // |
| // Internally, this class is implemented using a combination of a std::deque |
| // with a BitVector used to store whether each index is null or not. |
| // By default, for each null value, it only uses a single bit inside the |
| // BitVector at a slight cost (searching the BitVector to find the index into |
| // the std::deque) when looking up the data. |
| template <typename T> |
| class NullableVector : public NullableVectorBase { |
| private: |
| enum class Mode { |
| // Sparse mode is the default mode and ensures that nulls are stored using |
| // only |
| // a single bit (at the cost of making setting null entries to non-null |
| // O(n)). |
| kSparse, |
| |
| // Dense mode forces the reservation of space for null entries which |
| // increases |
| // memory usage but allows for O(1) set operations. |
| kDense, |
| }; |
| |
| public: |
| // Creates an empty NullableVector. |
| NullableVector() : NullableVector<T>(Mode::kSparse) {} |
| ~NullableVector() override = default; |
| |
| explicit NullableVector(const NullableVector&) = delete; |
| NullableVector& operator=(const NullableVector&) = delete; |
| |
| NullableVector(NullableVector&&) = default; |
| NullableVector& operator=(NullableVector&&) noexcept = default; |
| |
| // Creates a sparse nullable vector |
| static NullableVector<T> Sparse() { return NullableVector<T>(Mode::kSparse); } |
| |
| // Creates a dense nullable vector |
| static NullableVector<T> Dense() { return NullableVector<T>(Mode::kDense); } |
| |
| // Returns the optional value at |idx| or base::nullopt if the value is null. |
| base::Optional<T> Get(uint32_t idx) const { |
| if (mode_ == Mode::kDense) { |
| bool contains = valid_.Contains(idx); |
| return contains ? base::make_optional(data_[idx]) : base::nullopt; |
| } else { |
| auto opt_row = valid_.RowOf(idx); |
| return opt_row ? base::make_optional(data_[*opt_row]) : base::nullopt; |
| } |
| } |
| |
| // Returns the non-null value at |non_null_idx| where |non_null_idx| gives the |
| // index of the entry in-terms of non-null entries only. |
| // |
| // For example: |
| // this = [0, null, 2, null, 4] |
| // |
| // GetNonNull(0) = 0 |
| // GetNonNull(1) = 2 |
| // GetNonNull(2) = 4 |
| // ... |
| T GetNonNull(uint32_t non_null_idx) const { |
| if (mode_ == Mode::kDense) { |
| return data_[valid_.Get(non_null_idx)]; |
| } else { |
| PERFETTO_DCHECK(non_null_idx < data_.size()); |
| return data_[non_null_idx]; |
| } |
| } |
| |
| // Adds the given value to the NullableVector. |
| void Append(T val) { |
| data_.emplace_back(val); |
| valid_.Insert(size_++); |
| } |
| |
| // Adds the given optional value to the NullableVector. |
| void Append(base::Optional<T> val) { |
| if (val) { |
| Append(*val); |
| } else { |
| AppendNull(); |
| } |
| } |
| |
| // Sets the value at |idx| to the given |val|. |
| void Set(uint32_t idx, T val) { |
| if (mode_ == Mode::kDense) { |
| if (!valid_.Contains(idx)) { |
| valid_.Insert(idx); |
| } |
| data_[idx] = val; |
| } else { |
| auto opt_row = valid_.RowOf(idx); |
| |
| // Generally, we will be setting a null row to non-null so optimize for |
| // that path. |
| if (PERFETTO_UNLIKELY(opt_row)) { |
| data_[*opt_row] = val; |
| } else { |
| valid_.Insert(idx); |
| |
| uint32_t inserted_row = *valid_.RowOf(idx); |
| data_.insert(data_.begin() + static_cast<ptrdiff_t>(inserted_row), val); |
| } |
| } |
| } |
| |
| // Returns the size of the NullableVector; this includes any null values. |
| uint32_t size() const { return size_; } |
| |
| // Returns whether data in this NullableVector is stored densely. |
| bool IsDense() const { return mode_ == Mode::kDense; } |
| |
| private: |
| explicit NullableVector(Mode mode) : mode_(mode) {} |
| |
| void AppendNull() { |
| if (mode_ == Mode::kDense) { |
| data_.emplace_back(); |
| } |
| size_++; |
| } |
| |
| Mode mode_ = Mode::kSparse; |
| |
| std::deque<T> data_; |
| RowMap valid_; |
| uint32_t size_ = 0; |
| }; |
| |
| } // namespace trace_processor |
| } // namespace perfetto |
| |
| #endif // SRC_TRACE_PROCESSOR_CONTAINERS_NULLABLE_VECTOR_H_ |