365 lines
12 KiB
C++
365 lines
12 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 "Instance.h"
|
||
|
|
||
|
#include "ErrorUtils.h"
|
||
|
#include "JSBigString.h"
|
||
|
#include "JSBundleType.h"
|
||
|
#include "JSExecutor.h"
|
||
|
#include "MessageQueueThread.h"
|
||
|
#include "MethodCall.h"
|
||
|
#include "NativeToJsBridge.h"
|
||
|
#include "RAMBundleRegistry.h"
|
||
|
#include "RecoverableError.h"
|
||
|
#include "SystraceSection.h"
|
||
|
|
||
|
#include <cxxreact/JSIndexedRAMBundle.h>
|
||
|
#include <folly/json.h>
|
||
|
#include <react/debug/react_native_assert.h>
|
||
|
|
||
|
#include <glog/logging.h>
|
||
|
|
||
|
#include <condition_variable>
|
||
|
#include <exception>
|
||
|
#include <memory>
|
||
|
#include <mutex>
|
||
|
#include <string>
|
||
|
|
||
|
namespace facebook::react {
|
||
|
|
||
|
Instance::~Instance() {
|
||
|
if (nativeToJsBridge_) {
|
||
|
nativeToJsBridge_->destroy();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Instance::unregisterFromInspector() {
|
||
|
if (inspectorTarget_) {
|
||
|
assert(runtimeInspectorTarget_);
|
||
|
inspectorTarget_->unregisterRuntime(*runtimeInspectorTarget_);
|
||
|
assert(parentInspectorTarget_);
|
||
|
parentInspectorTarget_->unregisterInstance(*inspectorTarget_);
|
||
|
parentInspectorTarget_ = nullptr;
|
||
|
inspectorTarget_ = nullptr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Instance::initializeBridge(
|
||
|
std::unique_ptr<InstanceCallback> callback,
|
||
|
std::shared_ptr<JSExecutorFactory> jsef,
|
||
|
std::shared_ptr<MessageQueueThread> jsQueue,
|
||
|
std::shared_ptr<ModuleRegistry> moduleRegistry,
|
||
|
jsinspector_modern::HostTarget* parentInspectorTarget) {
|
||
|
callback_ = std::move(callback);
|
||
|
moduleRegistry_ = std::move(moduleRegistry);
|
||
|
parentInspectorTarget_ = parentInspectorTarget;
|
||
|
|
||
|
jsQueue->runOnQueueSync([this, &jsef, jsQueue]() mutable {
|
||
|
nativeToJsBridge_ = std::make_shared<NativeToJsBridge>(
|
||
|
jsef.get(), moduleRegistry_, jsQueue, callback_);
|
||
|
|
||
|
// If a parent inspector HostTarget is provided, perform inspector
|
||
|
// initialization synchronously.
|
||
|
if (parentInspectorTarget_ != nullptr) {
|
||
|
auto inspectorExecutor = parentInspectorTarget_->executorFromThis();
|
||
|
std::mutex inspectorInitializedMutex;
|
||
|
std::condition_variable inspectorInitializedCv;
|
||
|
bool inspectorInitialized = false;
|
||
|
|
||
|
// Schedule work on the inspector thread. NOTE: We expect this callback
|
||
|
// to always execute, given the invariant that `initializeBridge` (this
|
||
|
// method) completes before `unregisterFromInspector` is called.
|
||
|
// - On iOS, instance creation and invalidation both run on the main
|
||
|
// queue (`RCTCxxBridge::start,invalidate` use `RCTAssertMainQueue`).
|
||
|
// - On Android, `ReactContext` must be initialized with a constructed
|
||
|
// `CatalystInstance` (in which `Instance::initializeBridge` has
|
||
|
// completed) before `destroy` can be called.
|
||
|
inspectorExecutor([this,
|
||
|
&inspectorInitialized,
|
||
|
&inspectorInitializedMutex,
|
||
|
&inspectorInitializedCv](
|
||
|
jsinspector_modern::HostTarget& hostTarget) {
|
||
|
// NOTE: By passing *this, we strongly assume the Instance will still
|
||
|
// be alive by the time this executes.
|
||
|
// - On iOS, instance creation is done synchronously
|
||
|
// (`RCTCxxBridge::_initializeBridgeLocked`).
|
||
|
// - On Android, we explicitly wait for instance creation before
|
||
|
// destruction (`ReactInstanceManager::mReactContextLock`).
|
||
|
inspectorTarget_ = &hostTarget.registerInstance(*this);
|
||
|
RuntimeExecutor runtimeExecutorIfJsi = getRuntimeExecutor();
|
||
|
runtimeInspectorTarget_ = &inspectorTarget_->registerRuntime(
|
||
|
nativeToJsBridge_->getInspectorTargetDelegate(),
|
||
|
runtimeExecutorIfJsi ? runtimeExecutorIfJsi : [](auto) {});
|
||
|
|
||
|
// Signal that initialization is complete
|
||
|
{
|
||
|
std::lock_guard lock(inspectorInitializedMutex);
|
||
|
inspectorInitialized = true;
|
||
|
}
|
||
|
inspectorInitializedCv.notify_one();
|
||
|
});
|
||
|
|
||
|
// Wait for the initialization work to complete
|
||
|
{
|
||
|
std::unique_lock lock(inspectorInitializedMutex);
|
||
|
inspectorInitializedCv.wait(
|
||
|
lock, [&inspectorInitialized] { return inspectorInitialized; });
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Initialize the JavaScript runtime after we've initialized the inspector
|
||
|
nativeToJsBridge_->initializeRuntime();
|
||
|
|
||
|
// After NativeToJsBridge is created, the jsi::Runtime should exist. Also,
|
||
|
// the JS message queue thread exists. So, it's safe to schedule all queued
|
||
|
// up JS calls.
|
||
|
jsCallInvoker_->setNativeToJsBridgeAndFlushCalls(nativeToJsBridge_);
|
||
|
|
||
|
std::scoped_lock lock(m_syncMutex);
|
||
|
m_syncReady = true;
|
||
|
m_syncCV.notify_all();
|
||
|
});
|
||
|
|
||
|
CHECK(nativeToJsBridge_);
|
||
|
}
|
||
|
|
||
|
void Instance::loadBundle(
|
||
|
std::unique_ptr<RAMBundleRegistry> bundleRegistry,
|
||
|
std::unique_ptr<const JSBigString> string,
|
||
|
std::string sourceURL) {
|
||
|
callback_->incrementPendingJSCalls();
|
||
|
SystraceSection s("Instance::loadBundle", "sourceURL", sourceURL);
|
||
|
nativeToJsBridge_->loadBundle(
|
||
|
std::move(bundleRegistry), std::move(string), std::move(sourceURL));
|
||
|
}
|
||
|
|
||
|
void Instance::loadBundleSync(
|
||
|
std::unique_ptr<RAMBundleRegistry> bundleRegistry,
|
||
|
std::unique_ptr<const JSBigString> string,
|
||
|
std::string sourceURL) {
|
||
|
std::unique_lock<std::mutex> lock(m_syncMutex);
|
||
|
m_syncCV.wait(lock, [this] { return m_syncReady; });
|
||
|
|
||
|
SystraceSection s("Instance::loadBundleSync", "sourceURL", sourceURL);
|
||
|
nativeToJsBridge_->loadBundleSync(
|
||
|
std::move(bundleRegistry), std::move(string), std::move(sourceURL));
|
||
|
}
|
||
|
|
||
|
void Instance::setSourceURL(std::string sourceURL) {
|
||
|
callback_->incrementPendingJSCalls();
|
||
|
SystraceSection s("Instance::setSourceURL", "sourceURL", sourceURL);
|
||
|
|
||
|
nativeToJsBridge_->loadBundle(nullptr, nullptr, std::move(sourceURL));
|
||
|
}
|
||
|
|
||
|
void Instance::loadScriptFromString(
|
||
|
std::unique_ptr<const JSBigString> string,
|
||
|
std::string sourceURL,
|
||
|
bool loadSynchronously) {
|
||
|
SystraceSection s("Instance::loadScriptFromString", "sourceURL", sourceURL);
|
||
|
if (loadSynchronously) {
|
||
|
loadBundleSync(nullptr, std::move(string), std::move(sourceURL));
|
||
|
} else {
|
||
|
loadBundle(nullptr, std::move(string), std::move(sourceURL));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Instance::loadRAMBundleFromString(
|
||
|
std::unique_ptr<const JSBigString> script,
|
||
|
const std::string& sourceURL) {
|
||
|
auto bundle = std::make_unique<JSIndexedRAMBundle>(std::move(script));
|
||
|
auto startupScript = bundle->getStartupCode();
|
||
|
auto registry = RAMBundleRegistry::singleBundleRegistry(std::move(bundle));
|
||
|
loadRAMBundle(std::move(registry), std::move(startupScript), sourceURL, true);
|
||
|
}
|
||
|
|
||
|
void Instance::loadRAMBundleFromFile(
|
||
|
const std::string& sourcePath,
|
||
|
const std::string& sourceURL,
|
||
|
bool loadSynchronously) {
|
||
|
auto bundle = std::make_unique<JSIndexedRAMBundle>(sourcePath.c_str());
|
||
|
auto startupScript = bundle->getStartupCode();
|
||
|
auto registry = RAMBundleRegistry::multipleBundlesRegistry(
|
||
|
std::move(bundle), JSIndexedRAMBundle::buildFactory());
|
||
|
loadRAMBundle(
|
||
|
std::move(registry),
|
||
|
std::move(startupScript),
|
||
|
sourceURL,
|
||
|
loadSynchronously);
|
||
|
}
|
||
|
|
||
|
void Instance::loadRAMBundle(
|
||
|
std::unique_ptr<RAMBundleRegistry> bundleRegistry,
|
||
|
std::unique_ptr<const JSBigString> startupScript,
|
||
|
std::string startupScriptSourceURL,
|
||
|
bool loadSynchronously) {
|
||
|
if (loadSynchronously) {
|
||
|
loadBundleSync(
|
||
|
std::move(bundleRegistry),
|
||
|
std::move(startupScript),
|
||
|
std::move(startupScriptSourceURL));
|
||
|
} else {
|
||
|
loadBundle(
|
||
|
std::move(bundleRegistry),
|
||
|
std::move(startupScript),
|
||
|
std::move(startupScriptSourceURL));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Instance::setGlobalVariable(
|
||
|
std::string propName,
|
||
|
std::unique_ptr<const JSBigString> jsonValue) {
|
||
|
nativeToJsBridge_->setGlobalVariable(
|
||
|
std::move(propName), std::move(jsonValue));
|
||
|
}
|
||
|
|
||
|
void* Instance::getJavaScriptContext() {
|
||
|
return nativeToJsBridge_ ? nativeToJsBridge_->getJavaScriptContext()
|
||
|
: nullptr;
|
||
|
}
|
||
|
|
||
|
bool Instance::isInspectable() {
|
||
|
return nativeToJsBridge_ ? nativeToJsBridge_->isInspectable() : false;
|
||
|
}
|
||
|
|
||
|
bool Instance::isBatchActive() {
|
||
|
return nativeToJsBridge_ ? nativeToJsBridge_->isBatchActive() : false;
|
||
|
}
|
||
|
|
||
|
void Instance::callJSFunction(
|
||
|
std::string&& module,
|
||
|
std::string&& method,
|
||
|
folly::dynamic&& params) {
|
||
|
callback_->incrementPendingJSCalls();
|
||
|
nativeToJsBridge_->callFunction(
|
||
|
std::move(module), std::move(method), std::move(params));
|
||
|
}
|
||
|
|
||
|
void Instance::callJSCallback(uint64_t callbackId, folly::dynamic&& params) {
|
||
|
SystraceSection s("Instance::callJSCallback");
|
||
|
callback_->incrementPendingJSCalls();
|
||
|
nativeToJsBridge_->invokeCallback((double)callbackId, std::move(params));
|
||
|
}
|
||
|
|
||
|
void Instance::registerBundle(
|
||
|
uint32_t bundleId,
|
||
|
const std::string& bundlePath) {
|
||
|
nativeToJsBridge_->registerBundle(bundleId, bundlePath);
|
||
|
}
|
||
|
|
||
|
const ModuleRegistry& Instance::getModuleRegistry() const {
|
||
|
return *moduleRegistry_;
|
||
|
}
|
||
|
|
||
|
ModuleRegistry& Instance::getModuleRegistry() {
|
||
|
return *moduleRegistry_;
|
||
|
}
|
||
|
|
||
|
void Instance::handleMemoryPressure(int pressureLevel) {
|
||
|
if (nativeToJsBridge_) {
|
||
|
// This class resets `nativeToJsBridge_` only in the destructor,
|
||
|
// hence a race is not possible there.
|
||
|
nativeToJsBridge_->handleMemoryPressure(pressureLevel);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
std::shared_ptr<CallInvoker> Instance::getJSCallInvoker() {
|
||
|
return std::static_pointer_cast<CallInvoker>(jsCallInvoker_);
|
||
|
}
|
||
|
|
||
|
RuntimeExecutor Instance::getRuntimeExecutor() {
|
||
|
// HACK: RuntimeExecutor is not compatible with non-JSIExecutor, we return
|
||
|
// a null callback, which the caller should handle.
|
||
|
if (!getJavaScriptContext()) {
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
std::weak_ptr<NativeToJsBridge> weakNativeToJsBridge = nativeToJsBridge_;
|
||
|
return [weakNativeToJsBridge](
|
||
|
std::function<void(jsi::Runtime & runtime)>&& callback) {
|
||
|
if (auto strongNativeToJsBridge = weakNativeToJsBridge.lock()) {
|
||
|
strongNativeToJsBridge->runOnExecutorQueue(
|
||
|
[callback = std::move(callback)](JSExecutor* executor) {
|
||
|
// Assumes the underlying executor is a JSIExecutor
|
||
|
jsi::Runtime* runtime =
|
||
|
(jsi::Runtime*)executor->getJavaScriptContext();
|
||
|
try {
|
||
|
react_native_assert(runtime != nullptr);
|
||
|
callback(*runtime);
|
||
|
executor->flush();
|
||
|
} catch (jsi::JSError& originalError) {
|
||
|
handleJSError(*runtime, originalError, true);
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
std::shared_ptr<NativeMethodCallInvoker>
|
||
|
Instance::getDecoratedNativeMethodCallInvoker(
|
||
|
std::shared_ptr<NativeMethodCallInvoker> nativeMethodCallInvoker) {
|
||
|
return nativeToJsBridge_->getDecoratedNativeMethodCallInvoker(
|
||
|
nativeMethodCallInvoker);
|
||
|
}
|
||
|
|
||
|
void Instance::JSCallInvoker::setNativeToJsBridgeAndFlushCalls(
|
||
|
std::weak_ptr<NativeToJsBridge> nativeToJsBridge) {
|
||
|
std::scoped_lock guard(m_mutex);
|
||
|
|
||
|
m_shouldBuffer = false;
|
||
|
m_nativeToJsBridge = nativeToJsBridge;
|
||
|
while (m_workBuffer.size() > 0) {
|
||
|
scheduleAsync(std::move(m_workBuffer.front()));
|
||
|
m_workBuffer.pop_front();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Instance::JSCallInvoker::invokeSync(CallFunc&& /*work*/) {
|
||
|
// TODO: Replace JS Callinvoker with RuntimeExecutor.
|
||
|
throw std::runtime_error(
|
||
|
"Synchronous native -> JS calls are currently not supported.");
|
||
|
}
|
||
|
|
||
|
void Instance::JSCallInvoker::invokeAsync(CallFunc&& work) noexcept {
|
||
|
std::scoped_lock guard(m_mutex);
|
||
|
|
||
|
/**
|
||
|
* Why is is necessary to queue up async work?
|
||
|
*
|
||
|
* 1. TurboModuleManager must be created synchronously after the Instance,
|
||
|
* before we load the source code. This is when the NativeModule system
|
||
|
* is initialized. RCTDevLoadingView shows bundle download progress.
|
||
|
* 2. TurboModuleManager requires a JS CallInvoker.
|
||
|
* 3. The JS CallInvoker requires the NativeToJsBridge, which is created on
|
||
|
* the JS thread in Instance::initializeBridge.
|
||
|
*
|
||
|
* Therefore, although we don't call invokeAsync before the JS bundle is
|
||
|
* executed, this buffering is implemented anyways to ensure that work
|
||
|
* isn't discarded.
|
||
|
*/
|
||
|
if (m_shouldBuffer) {
|
||
|
m_workBuffer.push_back(std::move(work));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
scheduleAsync(std::move(work));
|
||
|
}
|
||
|
|
||
|
void Instance::JSCallInvoker::scheduleAsync(CallFunc&& work) noexcept {
|
||
|
if (auto strongNativeToJsBridge = m_nativeToJsBridge.lock()) {
|
||
|
strongNativeToJsBridge->runOnExecutorQueue(
|
||
|
[work = std::move(work)](JSExecutor* executor) {
|
||
|
auto* runtime = (jsi::Runtime*)executor->getJavaScriptContext();
|
||
|
work(*runtime);
|
||
|
executor->flush();
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
} // namespace facebook::react
|