Files
react-native/ReactCommon/react/renderer/runtimescheduler/RuntimeScheduler.cpp
T
Samuel Susla dbda1917cc Make RuntimeScheduler module compile in C++ 14
Summary:
changelog: [internal]

Don't use C++ 17 features in RuntimeScheduler module as it needs to be imported into C++14 module.

Also removes redundant dependency.

Reviewed By: ShikaSD

Differential Revision: D30485642

fbshipit-source-id: 0a20f85c596eebe193affc815c8ca851fc72e46d
2021-09-01 11:26:27 -07:00

119 lines
3.2 KiB
C++

/*
* 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 "RuntimeScheduler.h"
namespace facebook {
namespace react {
#pragma mark - Public
RuntimeScheduler::RuntimeScheduler(
RuntimeExecutor const &runtimeExecutor,
std::function<RuntimeSchedulerTimePoint()> now)
: runtimeExecutor_(runtimeExecutor), now_(now) {}
void RuntimeScheduler::scheduleWork(
std::function<void(jsi::Runtime &)> callback) const {
if (enableYielding_) {
shouldYield_ = true;
runtimeExecutor_(
[this, callback = std::move(callback)](jsi::Runtime &runtime) {
shouldYield_ = false;
callback(runtime);
startWorkLoop(runtime);
});
} else {
runtimeExecutor_([callback = std::move(callback)](jsi::Runtime &runtime) {
callback(runtime);
});
}
}
std::shared_ptr<Task> RuntimeScheduler::scheduleTask(
SchedulerPriority priority,
jsi::Function callback) {
auto expirationTime = now() + timeoutForSchedulerPriority(priority);
auto task =
std::make_shared<Task>(priority, std::move(callback), expirationTime);
taskQueue_.push(task);
if (!isCallbackScheduled_ && !isPerformingWork_) {
isCallbackScheduled_ = true;
runtimeExecutor_([this](jsi::Runtime &runtime) {
isCallbackScheduled_ = false;
startWorkLoop(runtime);
});
}
return task;
}
bool RuntimeScheduler::getShouldYield() const noexcept {
return shouldYield_;
}
void RuntimeScheduler::cancelTask(const std::shared_ptr<Task> &task) noexcept {
task->callback.reset();
}
SchedulerPriority RuntimeScheduler::getCurrentPriorityLevel() const noexcept {
return currentPriority_;
}
RuntimeSchedulerTimePoint RuntimeScheduler::now() const noexcept {
return now_();
}
void RuntimeScheduler::setEnableYielding(bool enableYielding) {
enableYielding_ = enableYielding;
}
void RuntimeScheduler::executeNowOnTheSameThread(
std::function<void(jsi::Runtime &runtime)> callback) const {
shouldYield_ = true;
executeSynchronouslyOnSameThread_CAN_DEADLOCK(
runtimeExecutor_,
[callback = std::move(callback)](jsi::Runtime &runtime) {
callback(runtime);
});
shouldYield_ = false;
}
#pragma mark - Private
void RuntimeScheduler::startWorkLoop(jsi::Runtime &runtime) const {
auto previousPriority = currentPriority_;
isPerformingWork_ = true;
while (!taskQueue_.empty()) {
auto topPriorityTask = taskQueue_.top();
auto now = now_();
auto didUserCallbackTimeout = topPriorityTask->expirationTime <= now;
if (!didUserCallbackTimeout && shouldYield_) {
// This task hasn't expired and we need to yield.
break;
}
currentPriority_ = topPriorityTask->priority;
auto result = topPriorityTask->execute(runtime);
if (result.isObject() && result.getObject(runtime).isFunction(runtime)) {
topPriorityTask->callback =
result.getObject(runtime).getFunction(runtime);
} else {
if (taskQueue_.top() == topPriorityTask) {
taskQueue_.pop();
}
}
}
currentPriority_ = previousPriority;
isPerformingWork_ = false;
}
} // namespace react
} // namespace facebook