|  | // Copyright 2013 The Flutter Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "flutter/shell/common/vsync_waiter.h" | 
|  |  | 
|  | #include "flow/frame_timings.h" | 
|  | #include "flutter/fml/task_runner.h" | 
|  | #include "flutter/fml/trace_event.h" | 
|  | #include "fml/message_loop_task_queues.h" | 
|  | #include "fml/task_queue_id.h" | 
|  |  | 
|  | namespace flutter { | 
|  |  | 
|  | static constexpr const char* kVsyncFlowName = "VsyncFlow"; | 
|  |  | 
|  | #if defined(OS_FUCHSIA) | 
|  | //  ________  _________  ________  ________ | 
|  | // |\   ____\|\___   ___\\   __  \|\   __  \ | 
|  | // \ \  \___|\|___ \  \_\ \  \|\  \ \  \|\  \ | 
|  | //  \ \_____  \   \ \  \ \ \  \\\  \ \   ____\ | 
|  | //   \|____|\  \   \ \  \ \ \  \\\  \ \  \___| | 
|  | //     ____\_\  \   \ \__\ \ \_______\ \__\ | 
|  | //    |\_________\   \|__|  \|_______|\|__| | 
|  | //    \|_________| | 
|  | // | 
|  | // Fuchsia benchmarks depend on this trace event's name.  Please do not change | 
|  | // it without checking that the changes are compatible with Fuchsia benchmarks | 
|  | // first! | 
|  | static constexpr const char* kVsyncTraceName = "vsync callback"; | 
|  | #else | 
|  | static constexpr const char* kVsyncTraceName = "VsyncProcessCallback"; | 
|  | #endif | 
|  |  | 
|  | VsyncWaiter::VsyncWaiter(TaskRunners task_runners) | 
|  | : task_runners_(std::move(task_runners)) {} | 
|  |  | 
|  | VsyncWaiter::~VsyncWaiter() = default; | 
|  |  | 
|  | // Public method invoked by the animator. | 
|  | void VsyncWaiter::AsyncWaitForVsync(const Callback& callback) { | 
|  | if (!callback) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | TRACE_EVENT0("flutter", "AsyncWaitForVsync"); | 
|  |  | 
|  | { | 
|  | std::scoped_lock lock(callback_mutex_); | 
|  | if (callback_) { | 
|  | // The animator may request a frame more than once within a frame | 
|  | // interval. Multiple calls to request frame must result in a single | 
|  | // callback per frame interval. | 
|  | TRACE_EVENT_INSTANT0("flutter", "MultipleCallsToVsyncInFrameInterval"); | 
|  | return; | 
|  | } | 
|  | callback_ = std::move(callback); | 
|  | if (!secondary_callbacks_.empty()) { | 
|  | // Return directly as `AwaitVSync` is already called by | 
|  | // `ScheduleSecondaryCallback`. | 
|  | return; | 
|  | } | 
|  | } | 
|  | AwaitVSync(); | 
|  | } | 
|  |  | 
|  | void VsyncWaiter::ScheduleSecondaryCallback(uintptr_t id, | 
|  | const fml::closure& callback) { | 
|  | FML_DCHECK(task_runners_.GetUITaskRunner()->RunsTasksOnCurrentThread()); | 
|  |  | 
|  | if (!callback) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | TRACE_EVENT0("flutter", "ScheduleSecondaryCallback"); | 
|  |  | 
|  | { | 
|  | std::scoped_lock lock(callback_mutex_); | 
|  | auto [_, inserted] = secondary_callbacks_.emplace(id, std::move(callback)); | 
|  | if (!inserted) { | 
|  | // Multiple schedules must result in a single callback per frame interval. | 
|  | TRACE_EVENT_INSTANT0("flutter", | 
|  | "MultipleCallsToSecondaryVsyncInFrameInterval"); | 
|  | return; | 
|  | } | 
|  | if (callback_) { | 
|  | // Return directly as `AwaitVSync` is already called by | 
|  | // `AsyncWaitForVsync`. | 
|  | return; | 
|  | } | 
|  | } | 
|  | AwaitVSyncForSecondaryCallback(); | 
|  | } | 
|  |  | 
|  | void VsyncWaiter::FireCallback(fml::TimePoint frame_start_time, | 
|  | fml::TimePoint frame_target_time, | 
|  | bool pause_secondary_tasks) { | 
|  | Callback callback; | 
|  | std::vector<fml::closure> secondary_callbacks; | 
|  |  | 
|  | { | 
|  | std::scoped_lock lock(callback_mutex_); | 
|  | callback = std::move(callback_); | 
|  | for (auto& pair : secondary_callbacks_) { | 
|  | secondary_callbacks.push_back(std::move(pair.second)); | 
|  | } | 
|  | secondary_callbacks_.clear(); | 
|  | } | 
|  |  | 
|  | if (!callback && secondary_callbacks.empty()) { | 
|  | // This means that the vsync waiter implementation fired a callback for a | 
|  | // request we did not make. This is a paranoid check but we still want to | 
|  | // make sure we catch misbehaving vsync implementations. | 
|  | TRACE_EVENT_INSTANT0("flutter", "MismatchedFrameCallback"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (callback) { | 
|  | auto flow_identifier = fml::tracing::TraceNonce(); | 
|  | if (pause_secondary_tasks) { | 
|  | PauseDartMicroTasks(); | 
|  | } | 
|  |  | 
|  | // The base trace ensures that flows have a root to begin from if one does | 
|  | // not exist. The trace viewer will ignore traces that have no base event | 
|  | // trace. While all our message loops insert a base trace trace | 
|  | // (MessageLoop::RunExpiredTasks), embedders may not. | 
|  | TRACE_EVENT0("flutter", "VsyncFireCallback"); | 
|  |  | 
|  | TRACE_FLOW_BEGIN("flutter", kVsyncFlowName, flow_identifier); | 
|  |  | 
|  | fml::TaskQueueId ui_task_queue_id = fml::_kUnmerged; | 
|  | if (pause_secondary_tasks) { | 
|  | // Guarding `GetTaskQueueId` behind `pause_secondary_tasks` as on Fuchsia | 
|  | // the task runners don't initialize message loop task queues. | 
|  | // Once the migration to embedder API is done, this can be deleted. | 
|  | ui_task_queue_id = task_runners_.GetUITaskRunner()->GetTaskQueueId(); | 
|  | } | 
|  |  | 
|  | task_runners_.GetUITaskRunner()->PostTaskForTime( | 
|  | [ui_task_queue_id, callback, flow_identifier, frame_start_time, | 
|  | frame_target_time, pause_secondary_tasks]() { | 
|  | FML_TRACE_EVENT("flutter", kVsyncTraceName, "StartTime", | 
|  | frame_start_time, "TargetTime", frame_target_time); | 
|  | std::unique_ptr<FrameTimingsRecorder> frame_timings_recorder = | 
|  | std::make_unique<FrameTimingsRecorder>(); | 
|  | frame_timings_recorder->RecordVsync(frame_start_time, | 
|  | frame_target_time); | 
|  | callback(std::move(frame_timings_recorder)); | 
|  | TRACE_FLOW_END("flutter", kVsyncFlowName, flow_identifier); | 
|  | if (pause_secondary_tasks) { | 
|  | ResumeDartMicroTasks(ui_task_queue_id); | 
|  | } | 
|  | }, | 
|  | frame_start_time); | 
|  | } | 
|  |  | 
|  | for (auto& secondary_callback : secondary_callbacks) { | 
|  | task_runners_.GetUITaskRunner()->PostTaskForTime( | 
|  | std::move(secondary_callback), frame_start_time); | 
|  | } | 
|  | } | 
|  |  | 
|  | void VsyncWaiter::PauseDartMicroTasks() { | 
|  | auto ui_task_queue_id = task_runners_.GetUITaskRunner()->GetTaskQueueId(); | 
|  | auto task_queues = fml::MessageLoopTaskQueues::GetInstance(); | 
|  | task_queues->PauseSecondarySource(ui_task_queue_id); | 
|  | } | 
|  |  | 
|  | void VsyncWaiter::ResumeDartMicroTasks(fml::TaskQueueId ui_task_queue_id) { | 
|  | auto task_queues = fml::MessageLoopTaskQueues::GetInstance(); | 
|  | task_queues->ResumeSecondarySource(ui_task_queue_id); | 
|  | } | 
|  |  | 
|  | }  // namespace flutter |