mirror of
https://github.com/JakeHillion/object-introspection.git
synced 2024-11-09 21:24:14 +00:00
89b230395f
Summary: tbv2: remove unnecessary copy in Element `IntrospectionResult::const_iterator` iterates through the `Element`s in an `IntrospectionResult`. `Element` currently copies the `type_path` which is a `std::vector<string_view>` every time the iterator is incremented. This is unnecessary as the data in the vector only changes slightly between iterations. This change changes the `type_path` field in `Element` to a `std::span<const std::string_view>`. Doing this previously caused SEGVs because of the iterator's potential to be copied. To make it possible we do two things: 1. Make all copies explicit using a clone interface as in `ContainerInfo`. This prevents accidental copies of an expensive structure. 2. After calling the copy constructor in `clone()` update the `span` in `next_` to point at the newly copied structure. Moves are fine because the `span` points at the allocation of the `vector`, not the vector itself. Test Plan: - CI - `FILTER='OilIntegration.*' make test` - Ran `OilgenIntegration.std_vector_vector_int_some` which SEGVd with the `span` change before and now doesn't. This now passes cleanly with ASAN enabled on the target, though isn't available in `main` (only works on my machine). Differential Revision: D53472595 Pulled By: JakeHillion
107 lines
3.5 KiB
C++
107 lines
3.5 KiB
C++
/*
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* 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.
|
|
*/
|
|
#if defined(INCLUDED_OI_INTROSPECTIONRESULT_INL_H) || \
|
|
!defined(INCLUDED_OI_INTROSPECTIONRESULT_H)
|
|
static_assert(false,
|
|
"IntrospectionResult-inl.h provides inline declarations for "
|
|
"IntrospectionResult.h and should only "
|
|
"be included by IntrospectionResult.h");
|
|
#endif
|
|
#define INCLUDED_OI_INTROSPECTIONRESULT_INL_H 1
|
|
|
|
#include <cassert>
|
|
|
|
#include "IntrospectionResult.h"
|
|
|
|
namespace oi {
|
|
|
|
inline IntrospectionResult::IntrospectionResult(std::vector<uint8_t> buf,
|
|
exporters::inst::Inst inst)
|
|
: buf_(std::move(buf)), inst_(inst) {
|
|
}
|
|
|
|
inline IntrospectionResult::const_iterator::const_iterator(
|
|
std::vector<uint8_t>::const_iterator data, exporters::inst::Inst type)
|
|
: data_(data), stack_({type}) {
|
|
}
|
|
inline IntrospectionResult::const_iterator::const_iterator(
|
|
std::vector<uint8_t>::const_iterator data)
|
|
: data_(data) {
|
|
}
|
|
inline IntrospectionResult::const_iterator IntrospectionResult::begin() const {
|
|
return cbegin();
|
|
}
|
|
inline IntrospectionResult::const_iterator IntrospectionResult::cbegin() const {
|
|
auto it = const_iterator{buf_.cbegin(), inst_};
|
|
++it;
|
|
return it;
|
|
}
|
|
inline IntrospectionResult::const_iterator IntrospectionResult::end() const {
|
|
return cend();
|
|
}
|
|
inline IntrospectionResult::const_iterator IntrospectionResult::cend() const {
|
|
return {buf_.cend()};
|
|
}
|
|
|
|
inline IntrospectionResult::const_iterator
|
|
IntrospectionResult::const_iterator::operator++(int) {
|
|
auto old = *this;
|
|
operator++();
|
|
return old;
|
|
}
|
|
inline const result::Element& IntrospectionResult::const_iterator::operator*()
|
|
const {
|
|
assert(next_);
|
|
return *next_;
|
|
}
|
|
inline const result::Element* IntrospectionResult::const_iterator::operator->()
|
|
const {
|
|
return &*next_;
|
|
}
|
|
|
|
inline IntrospectionResult::const_iterator
|
|
IntrospectionResult::const_iterator::clone() const {
|
|
auto ret{*this};
|
|
|
|
// Fix the self referential type_path_ field in next_
|
|
if (ret.next_.has_value()) {
|
|
ret.next_->type_path = ret.type_path_;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
inline bool IntrospectionResult::const_iterator::operator==(
|
|
const IntrospectionResult::const_iterator& that) const {
|
|
// Case 1: The next data to read differs, thus the iterators are different.
|
|
if (this->data_ != that.data_)
|
|
return false;
|
|
// Case 2: Both iterators have no next value, thus they are both complete. It
|
|
// is insufficient to check increments as the number of increments is unknown
|
|
// when constructing `end()`.
|
|
if (!this->next_.has_value() && !that.next_.has_value())
|
|
return true;
|
|
// Case 3: The iterators are reading the same data. If they have produced the
|
|
// same number of elements they are equal, else they are not.
|
|
return this->increments_ == that.increments_;
|
|
}
|
|
inline bool IntrospectionResult::const_iterator::operator!=(
|
|
const IntrospectionResult::const_iterator& that) const {
|
|
return !(*this == that);
|
|
}
|
|
|
|
} // namespace oi
|