205 lines
6.2 KiB
C++
205 lines
6.2 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 "RuntimeSchedulerBinding.h"
|
|
#include <ReactCommon/SchedulerPriority.h>
|
|
#include "RuntimeScheduler.h"
|
|
#include "SchedulerPriorityUtils.h"
|
|
#include "primitives.h"
|
|
|
|
#include <chrono>
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
namespace facebook::react {
|
|
|
|
std::shared_ptr<RuntimeSchedulerBinding>
|
|
RuntimeSchedulerBinding::createAndInstallIfNeeded(
|
|
jsi::Runtime &runtime,
|
|
std::shared_ptr<RuntimeScheduler> const &runtimeScheduler) {
|
|
auto runtimeSchedulerModuleName = "nativeRuntimeScheduler";
|
|
|
|
auto runtimeSchedulerValue =
|
|
runtime.global().getProperty(runtime, runtimeSchedulerModuleName);
|
|
if (runtimeSchedulerValue.isUndefined()) {
|
|
// The global namespace does not have an instance of the binding;
|
|
// we need to create, install and return it.
|
|
auto runtimeSchedulerBinding =
|
|
std::make_shared<RuntimeSchedulerBinding>(runtimeScheduler);
|
|
auto object =
|
|
jsi::Object::createFromHostObject(runtime, runtimeSchedulerBinding);
|
|
runtime.global().setProperty(
|
|
runtime, runtimeSchedulerModuleName, std::move(object));
|
|
return runtimeSchedulerBinding;
|
|
}
|
|
|
|
// The global namespace already has an instance of the binding;
|
|
// we need to return that.
|
|
auto runtimeSchedulerObject = runtimeSchedulerValue.asObject(runtime);
|
|
return runtimeSchedulerObject.getHostObject<RuntimeSchedulerBinding>(runtime);
|
|
}
|
|
|
|
std::shared_ptr<RuntimeSchedulerBinding> RuntimeSchedulerBinding::getBinding(
|
|
jsi::Runtime &runtime) {
|
|
auto runtimeSchedulerModuleName = "nativeRuntimeScheduler";
|
|
|
|
auto runtimeSchedulerValue =
|
|
runtime.global().getProperty(runtime, runtimeSchedulerModuleName);
|
|
if (runtimeSchedulerValue.isUndefined()) {
|
|
return nullptr;
|
|
}
|
|
|
|
auto runtimeSchedulerObject = runtimeSchedulerValue.asObject(runtime);
|
|
return runtimeSchedulerObject.getHostObject<RuntimeSchedulerBinding>(runtime);
|
|
}
|
|
|
|
RuntimeSchedulerBinding::RuntimeSchedulerBinding(
|
|
std::shared_ptr<RuntimeScheduler> runtimeScheduler)
|
|
: runtimeScheduler_(std::move(runtimeScheduler)) {}
|
|
|
|
bool RuntimeSchedulerBinding::getIsSynchronous() const {
|
|
return runtimeScheduler_->getIsSynchronous();
|
|
}
|
|
|
|
jsi::Value RuntimeSchedulerBinding::get(
|
|
jsi::Runtime &runtime,
|
|
jsi::PropNameID const &name) {
|
|
auto propertyName = name.utf8(runtime);
|
|
|
|
if (propertyName == "unstable_scheduleCallback") {
|
|
return jsi::Function::createFromHostFunction(
|
|
runtime,
|
|
name,
|
|
3,
|
|
[this](
|
|
jsi::Runtime &runtime,
|
|
jsi::Value const &,
|
|
jsi::Value const *arguments,
|
|
size_t count) noexcept -> jsi::Value {
|
|
SchedulerPriority priority = fromRawValue(arguments[0].getNumber());
|
|
auto callback = arguments[1].getObject(runtime).getFunction(runtime);
|
|
|
|
auto task =
|
|
runtimeScheduler_->scheduleTask(priority, std::move(callback));
|
|
|
|
return valueFromTask(runtime, task);
|
|
});
|
|
}
|
|
|
|
if (propertyName == "unstable_cancelCallback") {
|
|
return jsi::Function::createFromHostFunction(
|
|
runtime,
|
|
name,
|
|
1,
|
|
[this](
|
|
jsi::Runtime &runtime,
|
|
jsi::Value const &,
|
|
jsi::Value const *arguments,
|
|
size_t) noexcept -> jsi::Value {
|
|
runtimeScheduler_->cancelTask(*taskFromValue(runtime, arguments[0]));
|
|
return jsi::Value::undefined();
|
|
});
|
|
}
|
|
|
|
if (propertyName == "unstable_shouldYield") {
|
|
return jsi::Function::createFromHostFunction(
|
|
runtime,
|
|
name,
|
|
0,
|
|
[this](
|
|
jsi::Runtime &,
|
|
jsi::Value const &,
|
|
jsi::Value const *,
|
|
size_t) noexcept -> jsi::Value {
|
|
auto shouldYield = runtimeScheduler_->getShouldYield();
|
|
return {shouldYield};
|
|
});
|
|
}
|
|
|
|
if (propertyName == "unstable_requestPaint") {
|
|
return jsi::Function::createFromHostFunction(
|
|
runtime,
|
|
name,
|
|
0,
|
|
[](jsi::Runtime &,
|
|
jsi::Value const &,
|
|
jsi::Value const *,
|
|
size_t) noexcept -> jsi::Value {
|
|
// RequestPaint is left empty by design.
|
|
return jsi::Value::undefined();
|
|
});
|
|
}
|
|
|
|
if (propertyName == "unstable_now") {
|
|
return jsi::Function::createFromHostFunction(
|
|
runtime,
|
|
name,
|
|
0,
|
|
[this](
|
|
jsi::Runtime &,
|
|
jsi::Value const &,
|
|
jsi::Value const *,
|
|
size_t) noexcept -> jsi::Value {
|
|
auto now = runtimeScheduler_->now();
|
|
auto asDouble =
|
|
std::chrono::duration<double, std::milli>(now.time_since_epoch())
|
|
.count();
|
|
return {asDouble};
|
|
});
|
|
}
|
|
|
|
// TODO: remove this, as it's deprecated in the JS scheduler
|
|
if (propertyName == "unstable_getCurrentPriorityLevel") {
|
|
return jsi::Function::createFromHostFunction(
|
|
runtime,
|
|
name,
|
|
0,
|
|
[this](
|
|
jsi::Runtime &runtime,
|
|
jsi::Value const &,
|
|
jsi::Value const *,
|
|
size_t) noexcept -> jsi::Value {
|
|
auto currentPriorityLevel =
|
|
runtimeScheduler_->getCurrentPriorityLevel();
|
|
return jsi::Value(runtime, serialize(currentPriorityLevel));
|
|
});
|
|
}
|
|
|
|
if (propertyName == "unstable_ImmediatePriority") {
|
|
return jsi::Value(runtime, serialize(SchedulerPriority::ImmediatePriority));
|
|
}
|
|
|
|
if (propertyName == "unstable_UserBlockingPriority") {
|
|
return jsi::Value(
|
|
runtime, serialize(SchedulerPriority::UserBlockingPriority));
|
|
}
|
|
|
|
if (propertyName == "unstable_NormalPriority") {
|
|
return jsi::Value(runtime, serialize(SchedulerPriority::NormalPriority));
|
|
}
|
|
|
|
if (propertyName == "unstable_LowPriority") {
|
|
return jsi::Value(runtime, serialize(SchedulerPriority::LowPriority));
|
|
}
|
|
|
|
if (propertyName == "unstable_IdlePriority") {
|
|
return jsi::Value(runtime, serialize(SchedulerPriority::IdlePriority));
|
|
}
|
|
|
|
if (propertyName == "$$typeof") {
|
|
return jsi::Value::undefined();
|
|
}
|
|
|
|
#ifdef REACT_NATIVE_DEBUG
|
|
throw std::runtime_error("undefined property");
|
|
#else
|
|
return jsi::Value::undefined();
|
|
#endif
|
|
}
|
|
|
|
} // namespace facebook::react
|