react-native/React/Fabric/Utils/RuntimeEventBeat.mm
Valentin Shergin df4521e6c4 Fabric: Making jsi::Runtime a part of EventBeat and EventPipe
Summary:
We double down on JSI in Fabric. So, practically, JSI is now a hard dependency for Fabric. I hope it's for good.
Now `jsi::Runtime` is coupled with scheduling via `EventExecuter`, so we have to make `jsi::Runtime` a part of `EventBeat` to proxy runtime reference to bindgings.

Reviewed By: sahrens

Differential Revision: D12837225

fbshipit-source-id: 98edc33d6a3358e6c2905f2f03ce0004a9ca0503
2018-11-06 11:03:50 -08:00

72 lines
2.0 KiB
Plaintext

/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
#include "RuntimeEventBeat.h"
namespace facebook {
namespace react {
RuntimeEventBeat::RuntimeEventBeat(RuntimeExecutor runtimeExecutor):
runtimeExecutor_(std::move(runtimeExecutor)) {
mainRunLoopObserver_ =
CFRunLoopObserverCreateWithHandler(
NULL /* allocator */,
kCFRunLoopBeforeWaiting /* activities */,
true /* repeats */,
0 /* order */,
^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
// Note: We only `induce` beat here; actual beat will be performed on
// a different thread.
this->induce();
}
);
assert(mainRunLoopObserver_);
CFRunLoopAddObserver(CFRunLoopGetMain(), mainRunLoopObserver_, kCFRunLoopCommonModes);
}
RuntimeEventBeat::~RuntimeEventBeat() {
CFRunLoopRemoveObserver(CFRunLoopGetMain(), mainRunLoopObserver_, kCFRunLoopCommonModes);
CFRelease(mainRunLoopObserver_);
}
void RuntimeEventBeat::induce() const {
if (!isRequested_ || isBusy_) {
return;
}
#ifndef NDEBUG
// We do a trick here.
// If `wasExecuted` was destroyed before set to `true`,
// it means that the execution block was deallocated not being executed.
// This indicates that `messageQueueThread_` is being deallocated.
// This trick is quite expensive due to deallocation and messing with atomic
// counters. Seems we need this only for making hot-reloading mechanism
// thread-safe. Hence, let's leave it to be DEBUG-only for now.
auto wasExecuted = std::shared_ptr<bool>(new bool {false}, [this](bool *wasExecuted) {
if (!*wasExecuted && failCallback_) {
failCallback_();
}
delete wasExecuted;
});
#endif
isBusy_ = true;
runtimeExecutor_([=](jsi::Runtime &runtime) mutable {
this->beat(runtime);
isBusy_ = false;
#ifndef NDEBUG
*wasExecuted = true;
#endif
});
}
} // namespace react
} // namespace facebook