Skip to content

Commit

Permalink
Allow non-static LongLivedObjectCollections in TurboModuleBinding (fa…
Browse files Browse the repository at this point in the history
…cebook#42194)

Summary:

Apps that have multiple concurrently running React instances may suffer from issues where tearing down one instance affects the bindings / LongLivedObjectCollection instance of another due to the use of static getter for LongLivedObjectCollection. This should allow host platforms, e.g., react-native-windows (which still forks the TurboModuleBinding C++ files [here](https://github.com/microsoft/react-native-windows/tree/main/vnext/ReactCommon/TEMP_UntilReactCommonUpdate/react/nativemodule/core/ReactCommon) for the reasons already mentioned) to manage per instance LongLivedObjectCollections.

## Changelog

[Internal]

Differential Revision: D52581170
  • Loading branch information
rozele authored and facebook-github-bot committed Jan 9, 2024
1 parent 05ca2c6 commit 7a76394
Show file tree
Hide file tree
Showing 4 changed files with 81 additions and 10 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,25 @@ class CallbackWrapper : public LongLivedObject {
jsi::Function&& callback,
jsi::Runtime& runtime,
std::shared_ptr<CallInvoker> jsInvoker)
: callback_(std::move(callback)),
: longLivedObjectCollection_(),
callback_(std::move(callback)),
runtime_(runtime),
jsInvoker_(std::move(jsInvoker)) {}

CallbackWrapper(
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection,
jsi::Function&& callback,
jsi::Runtime& runtime,
std::shared_ptr<CallInvoker> jsInvoker)
: longLivedObjectCollection_(longLivedObjectCollection),
callback_(std::move(callback)),
runtime_(runtime),
jsInvoker_(std::move(jsInvoker)) {}

// Use a weak_ptr to avoid a retain cycle: LongLivedObjectCollection owns all
// CallbackWrappers. So, CallbackWrapper cannot own its
// LongLivedObjectCollection.
std::weak_ptr<LongLivedObjectCollection> longLivedObjectCollection_;
jsi::Function callback_;
jsi::Runtime& runtime_;
std::shared_ptr<CallInvoker> jsInvoker_;
Expand All @@ -43,6 +58,17 @@ class CallbackWrapper : public LongLivedObject {
return wrapper;
}

static std::weak_ptr<CallbackWrapper> createWeak(
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection,
jsi::Function&& callback,
jsi::Runtime& runtime,
std::shared_ptr<CallInvoker> jsInvoker) {
auto wrapper = std::shared_ptr<CallbackWrapper>(new CallbackWrapper(
longLivedObjectCollection, std::move(callback), runtime, jsInvoker));
longLivedObjectCollection->add(wrapper);
return wrapper;
}

// Delete the enclosed jsi::Function
void destroy() {
allowRelease();
Expand All @@ -63,6 +89,16 @@ class CallbackWrapper : public LongLivedObject {
std::shared_ptr<CallInvoker> jsInvokerPtr() noexcept {
return jsInvoker_;
}

void allowRelease() override {
if (auto longLivedObjectCollection = longLivedObjectCollection_.lock()) {
if (longLivedObjectCollection != nullptr) {
longLivedObjectCollection->remove(this);
return;
}
}
LongLivedObject::allowRelease();
}
};

} // namespace facebook::react
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ namespace facebook::react {
*/
class LongLivedObject {
public:
void allowRelease();
virtual void allowRelease();

protected:
LongLivedObject() = default;
Expand All @@ -42,6 +42,7 @@ class LongLivedObjectCollection {
public:
static LongLivedObjectCollection& get();

LongLivedObjectCollection() = default;
LongLivedObjectCollection(const LongLivedObjectCollection&) = delete;
void operator=(const LongLivedObjectCollection&) = delete;

Expand All @@ -51,8 +52,6 @@ class LongLivedObjectCollection {
size_t size() const;

private:
LongLivedObjectCollection() = default;

std::unordered_set<std::shared_ptr<LongLivedObject>> collection_;
mutable std::mutex collectionMutex_;
};
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -95,12 +95,26 @@ static void defineReadOnlyGlobal(
*/

TurboModuleBinding::TurboModuleBinding(
TurboModuleProviderFunctionType&& moduleProvider)
: moduleProvider_(std::move(moduleProvider)) {}
TurboModuleProviderFunctionType&& moduleProvider,
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection)
: moduleProvider_(std::move(moduleProvider)),
longLivedObjectCollection_(std::move(longLivedObjectCollection)) {}

void TurboModuleBinding::install(
jsi::Runtime& runtime,
TurboModuleProviderFunctionType&& moduleProvider,
TurboModuleProviderFunctionType&& legacyModuleProvider) {
install(
runtime,
std::move(moduleProvider),
nullptr,
std::move(legacyModuleProvider));
}

void TurboModuleBinding::install(
jsi::Runtime& runtime,
TurboModuleProviderFunctionType&& moduleProvider,
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection,
TurboModuleProviderFunctionType&& legacyModuleProvider) {
runtime.global().setProperty(
runtime,
Expand All @@ -109,7 +123,8 @@ void TurboModuleBinding::install(
runtime,
jsi::PropNameID::forAscii(runtime, "__turboModuleProxy"),
1,
[binding = TurboModuleBinding(std::move(moduleProvider))](
[binding = TurboModuleBinding(
std::move(moduleProvider), longLivedObjectCollection)](
jsi::Runtime& rt,
const jsi::Value& thisVal,
const jsi::Value* args,
Expand All @@ -132,7 +147,8 @@ void TurboModuleBinding::install(
runtime,
std::make_shared<BridgelessNativeModuleProxy>(
std::make_unique<TurboModuleBinding>(
std::move(legacyModuleProvider)))));
std::move(legacyModuleProvider),
longLivedObjectCollection))));
} else {
defineReadOnlyGlobal(
runtime,
Expand All @@ -144,7 +160,11 @@ void TurboModuleBinding::install(
}

TurboModuleBinding::~TurboModuleBinding() {
LongLivedObjectCollection::get().clear();
if (longLivedObjectCollection_) {
longLivedObjectCollection_->clear();
} else {
LongLivedObjectCollection::get().clear();
}
}

jsi::Value TurboModuleBinding::getModule(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@

#include <ReactCommon/TurboModule.h>
#include <jsi/jsi.h>
#include <react/bridging/LongLivedObject.h>

namespace facebook::react {

Expand All @@ -30,7 +31,21 @@ class TurboModuleBinding {
TurboModuleProviderFunctionType&& moduleProvider,
TurboModuleProviderFunctionType&& legacyModuleProvider = nullptr);

TurboModuleBinding(TurboModuleProviderFunctionType&& moduleProvider);
/*
* Installs TurboModuleBinding into JavaScript runtime.
* Thread synchronization must be enforced externally.
* Uses a host managed LongLivedObjectCollection.
*/
static void install(
jsi::Runtime& runtime,
TurboModuleProviderFunctionType&& moduleProvider,
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection,
TurboModuleProviderFunctionType&& legacyModuleProvider = nullptr);

TurboModuleBinding(
TurboModuleProviderFunctionType&& moduleProvider,
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection);

virtual ~TurboModuleBinding();

private:
Expand All @@ -44,6 +59,7 @@ class TurboModuleBinding {
const;

TurboModuleProviderFunctionType moduleProvider_;
std::shared_ptr<LongLivedObjectCollection> longLivedObjectCollection_;
};

} // namespace facebook::react

0 comments on commit 7a76394

Please sign in to comment.