mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
ce50c43986
Summary: changelog: [internal] Add more clang tidy rules to prevent common class of bugs. Reviewed By: javache Differential Revision: D39245194 fbshipit-source-id: 5521c5c4653d7005b96ebba494e810ba5075afbc
191 lines
5.5 KiB
C++
191 lines
5.5 KiB
C++
/*
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
#include "MountingCoordinator.h"
|
|
|
|
#ifdef RN_SHADOW_TREE_INTROSPECTION
|
|
#include <glog/logging.h>
|
|
#include <sstream>
|
|
#endif
|
|
|
|
#include <condition_variable>
|
|
|
|
#include <react/debug/react_native_assert.h>
|
|
#include <react/renderer/mounting/ShadowViewMutation.h>
|
|
|
|
namespace facebook {
|
|
namespace react {
|
|
|
|
MountingCoordinator::MountingCoordinator(const ShadowTreeRevision &baseRevision)
|
|
: surfaceId_(baseRevision.rootShadowNode->getSurfaceId()),
|
|
baseRevision_(baseRevision),
|
|
telemetryController_(*this) {
|
|
#ifdef RN_SHADOW_TREE_INTROSPECTION
|
|
stubViewTree_ = buildStubViewTreeWithoutUsingDifferentiator(
|
|
*baseRevision_.rootShadowNode);
|
|
#endif
|
|
}
|
|
|
|
SurfaceId MountingCoordinator::getSurfaceId() const {
|
|
return surfaceId_;
|
|
}
|
|
|
|
void MountingCoordinator::push(ShadowTreeRevision const &revision) const {
|
|
{
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
|
|
|
react_native_assert(
|
|
!lastRevision_.has_value() || revision.number != lastRevision_->number);
|
|
|
|
if (!lastRevision_.has_value() || lastRevision_->number < revision.number) {
|
|
lastRevision_ = revision;
|
|
}
|
|
}
|
|
|
|
signal_.notify_all();
|
|
}
|
|
|
|
void MountingCoordinator::revoke() const {
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
|
// We have two goals here.
|
|
// 1. We need to stop retaining `ShadowNode`s to not prolong their lifetime
|
|
// to prevent them from overliving `ComponentDescriptor`s.
|
|
// 2. A possible call to `pullTransaction()` should return empty optional.
|
|
baseRevision_.rootShadowNode.reset();
|
|
lastRevision_.reset();
|
|
}
|
|
|
|
bool MountingCoordinator::waitForTransaction(
|
|
std::chrono::duration<double> timeout) const {
|
|
std::unique_lock<std::mutex> lock(mutex_);
|
|
return signal_.wait_for(
|
|
lock, timeout, [this]() { return lastRevision_.has_value(); });
|
|
}
|
|
|
|
void MountingCoordinator::updateBaseRevision(
|
|
ShadowTreeRevision const &baseRevision) const {
|
|
baseRevision_ = baseRevision;
|
|
}
|
|
|
|
void MountingCoordinator::resetLatestRevision() const {
|
|
lastRevision_.reset();
|
|
}
|
|
|
|
std::optional<MountingTransaction> MountingCoordinator::pullTransaction()
|
|
const {
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
|
|
|
auto transaction = std::optional<MountingTransaction>{};
|
|
|
|
// Base case
|
|
if (lastRevision_.has_value()) {
|
|
number_++;
|
|
|
|
auto telemetry = lastRevision_->telemetry;
|
|
|
|
telemetry.willDiff();
|
|
|
|
auto mutations = calculateShadowViewMutations(
|
|
*baseRevision_.rootShadowNode, *lastRevision_->rootShadowNode);
|
|
|
|
telemetry.didDiff();
|
|
|
|
transaction = MountingTransaction{
|
|
surfaceId_, number_, std::move(mutations), telemetry};
|
|
}
|
|
|
|
// Override case
|
|
auto mountingOverrideDelegate = mountingOverrideDelegate_.lock();
|
|
auto shouldOverridePullTransaction = mountingOverrideDelegate &&
|
|
mountingOverrideDelegate->shouldOverridePullTransaction();
|
|
|
|
if (shouldOverridePullTransaction) {
|
|
auto mutations = ShadowViewMutation::List{};
|
|
auto telemetry = TransactionTelemetry{};
|
|
|
|
if (transaction.has_value()) {
|
|
mutations = transaction->getMutations();
|
|
telemetry = transaction->getTelemetry();
|
|
} else {
|
|
number_++;
|
|
telemetry.willLayout();
|
|
telemetry.didLayout();
|
|
telemetry.willCommit();
|
|
telemetry.didCommit();
|
|
telemetry.willDiff();
|
|
telemetry.didDiff();
|
|
}
|
|
|
|
transaction = mountingOverrideDelegate->pullTransaction(
|
|
surfaceId_, number_, telemetry, std::move(mutations));
|
|
}
|
|
|
|
#ifdef RN_SHADOW_TREE_INTROSPECTION
|
|
if (transaction.has_value()) {
|
|
// We have something to validate.
|
|
auto mutations = transaction->getMutations();
|
|
|
|
// No matter what the source of the transaction is, it must be able to
|
|
// mutate the existing stub view tree.
|
|
stubViewTree_.mutate(mutations);
|
|
|
|
// If the transaction was overridden, we don't have a model of the shadow
|
|
// tree therefore we cannot validate the validity of the mutation
|
|
// instructions.
|
|
if (!shouldOverridePullTransaction && lastRevision_.has_value()) {
|
|
auto stubViewTree = buildStubViewTreeWithoutUsingDifferentiator(
|
|
*lastRevision_->rootShadowNode);
|
|
|
|
bool treesEqual = stubViewTree_ == stubViewTree;
|
|
|
|
if (!treesEqual) {
|
|
// Display debug info
|
|
auto line = std::string{};
|
|
std::stringstream ssOldTree(
|
|
baseRevision_.rootShadowNode->getDebugDescription());
|
|
while (std::getline(ssOldTree, line, '\n')) {
|
|
LOG(ERROR) << "Old tree:" << line;
|
|
}
|
|
|
|
std::stringstream ssMutations(getDebugDescription(mutations, {}));
|
|
while (std::getline(ssMutations, line, '\n')) {
|
|
LOG(ERROR) << "Mutations:" << line;
|
|
}
|
|
|
|
std::stringstream ssNewTree(
|
|
lastRevision_->rootShadowNode->getDebugDescription());
|
|
while (std::getline(ssNewTree, line, '\n')) {
|
|
LOG(ERROR) << "New tree:" << line;
|
|
}
|
|
}
|
|
|
|
react_native_assert(
|
|
(treesEqual) && "Incorrect set of mutations detected.");
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (lastRevision_.has_value()) {
|
|
baseRevision_ = std::move(*lastRevision_);
|
|
lastRevision_.reset();
|
|
}
|
|
return transaction;
|
|
}
|
|
|
|
TelemetryController const &MountingCoordinator::getTelemetryController() const {
|
|
return telemetryController_;
|
|
}
|
|
|
|
void MountingCoordinator::setMountingOverrideDelegate(
|
|
std::weak_ptr<MountingOverrideDelegate const> delegate) const {
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
|
mountingOverrideDelegate_ = std::move(delegate);
|
|
}
|
|
|
|
} // namespace react
|
|
} // namespace facebook
|