blob: d9fb920d95cf04a911398714fdcc637e54a7e9c8 [file] [log] [blame]
// Copyright 2014 The Cobalt Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include <memory>
#include "cobalt/renderer/pipeline.h"
#include "base/bind.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/path_service.h"
#include "base/trace_event/trace_event.h"
#include "cobalt/base/address_sanitizer.h"
#include "cobalt/base/cobalt_paths.h"
#include "cobalt/base/polymorphic_downcast.h"
#include "cobalt/extension/graphics.h"
#include "cobalt/math/rect_f.h"
#include "cobalt/render_tree/brush.h"
#include "cobalt/render_tree/composition_node.h"
#include "cobalt/render_tree/dump_render_tree_to_string.h"
#include "cobalt/render_tree/rect_node.h"
#include "nb/memory_scope.h"
#include "starboard/system.h"
using cobalt::render_tree::Node;
using cobalt::render_tree::animations::AnimateNode;
namespace cobalt {
namespace renderer {
namespace {
#if !defined(COBALT_MINIMUM_FRAME_TIME_IN_MILLISECONDS)
// This default value has been moved from cobalt/build/cobalt_configuration.gypi
// in favor of the usage of
// CobaltExtensionGraphicsApi::GetMinimumFrameIntervalInMilliseconds API.
const float kCobaltMinimumFrameTimeInMilliseconds = 16.0f;
#endif
// How quickly the renderer time adjusts to changing submission times.
// 500ms is chosen as a default because it is fast enough that the user will not
// usually notice input lag from a slow timeline renderer, but slow enough that
// quick updates while a quick animation is playing should not jank.
const double kTimeToConvergeInMS = 500.0;
// The stack size to be used for the renderer thread. This is must be large
// enough to support recursing on the render tree.
#if defined(COBALT_BUILD_TYPE_DEBUG)
const int kRendererThreadStackSize =
256 * 1024 + base::kAsanAdditionalStackSize;
#else
const int kRendererThreadStackSize =
128 * 1024 + base::kAsanAdditionalStackSize;
#endif
// How many entries the rasterize periodic timer will contain before updating.
const size_t kRasterizePeriodicTimerEntriesPerUpdate = 60;
// The maximum numer of entries that the rasterize animations timer can contain
// before automatically updating. In the typical use case, the update will
// occur manually when the animations expire.
const size_t kRasterizeAnimationsTimerMaxEntries = 60;
void DestructSubmissionOnMessageLoop(base::MessageLoop* message_loop,
std::unique_ptr<Submission> submission) {
TRACE_EVENT0("cobalt::renderer", "DestructSubmissionOnMessageLoop()");
if (base::MessageLoop::current() != message_loop) {
message_loop->task_runner()->DeleteSoon(FROM_HERE, submission.release());
}
}
bool ShouldClearFrameOnShutdown(render_tree::ColorRGBA* out_clear_color) {
const CobaltExtensionGraphicsApi* graphics_extension =
static_cast<const CobaltExtensionGraphicsApi*>(
SbSystemGetExtension(kCobaltExtensionGraphicsName));
if (graphics_extension &&
strcmp(graphics_extension->name, kCobaltExtensionGraphicsName) == 0 &&
graphics_extension->version >= 4) {
float r, g, b, a;
if (graphics_extension->ShouldClearFrameOnShutdown(&r, &g, &b, &a)) {
out_clear_color->set_r(r);
out_clear_color->set_g(g);
out_clear_color->set_b(b);
out_clear_color->set_a(a);
return true;
}
return false;
}
// Default is to clear to opaque black.
out_clear_color->set_r(0.0f);
out_clear_color->set_g(0.0f);
out_clear_color->set_b(0.0f);
out_clear_color->set_a(1.0f);
return true;
}
} // namespace
Pipeline::Pipeline(const CreateRasterizerFunction& create_rasterizer_function,
const scoped_refptr<backend::RenderTarget>& render_target,
backend::GraphicsContext* graphics_context,
bool submit_even_if_render_tree_is_unchanged,
ShutdownClearMode clear_on_shutdown_mode,
const Options& options)
: rasterizer_created_event_(
base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED),
render_target_(render_target),
graphics_context_(graphics_context),
rasterizer_thread_("Rasterizer"),
submission_disposal_thread_("RasterzrSubDisp"),
submit_even_if_render_tree_is_unchanged_(
submit_even_if_render_tree_is_unchanged),
last_did_rasterize_(false),
last_animations_expired_(true),
last_stat_tracked_animations_expired_(true),
rasterize_animations_timer_("Renderer.Rasterize.Animations",
kRasterizeAnimationsTimerMaxEntries,
true /*enable_entry_list_c_val*/),
ALLOW_THIS_IN_INITIALIZER_LIST(rasterize_periodic_interval_timer_(
"Renderer.Rasterize.DurationInterval",
kRasterizeAnimationsTimerMaxEntries, true /*enable_entry_list_c_val*/,
base::Bind(&Pipeline::FrameStatsOnFlushCallback,
base::Unretained(this)))),
rasterize_animations_interval_timer_(
"Renderer.Rasterize.AnimationsInterval",
kRasterizeAnimationsTimerMaxEntries,
true /*enable_entry_list_c_val*/),
new_render_tree_rasterize_count_(
"Count.Renderer.Rasterize.NewRenderTree", 0,
"Total number of new render trees rasterized."),
new_render_tree_rasterize_time_(
"Time.Renderer.Rasterize.NewRenderTree", 0,
"The last time a new render tree was rasterized."),
has_active_animations_c_val_(
"Renderer.HasActiveAnimations", false,
"Is non-zero if the current render tree has active animations."),
animations_start_time_(
"Time.Renderer.Rasterize.Animations.Start", 0,
"The most recent time animations started playing."),
animations_end_time_("Time.Renderer.Rasterize.Animations.End", 0,
"The most recent time animations ended playing."),
fallback_rasterize_count_(
"Count.Renderer.Rasterize.FallbackRasterize", 0,
"Total number of times Skia was used to render a "
"non-text render tree node."),
#if defined(ENABLE_DEBUGGER)
ALLOW_THIS_IN_INITIALIZER_LIST(dump_current_render_tree_command_handler_(
"dump_render_tree",
base::Bind(&Pipeline::OnDumpCurrentRenderTree,
base::Unretained(this)),
"Dumps the current render tree to text.",
"Dumps the current render tree either to the console if no parameter "
"is specified, or to a file with the specified filename relative to "
"the debug output folder.")),
ALLOW_THIS_IN_INITIALIZER_LIST(toggle_fps_stdout_command_handler_(
"toggle_fps_stdout",
base::Bind(&Pipeline::OnToggleFpsStdout, base::Unretained(this)),
"Toggles printing framerate stats to stdout.",
"When enabled, at the end of each animation (or every time a maximum "
"number of frames are rendered), framerate statistics are printed "
"to stdout.")),
ALLOW_THIS_IN_INITIALIZER_LIST(toggle_fps_overlay_command_handler_(
"toggle_fps_overlay",
base::Bind(&Pipeline::OnToggleFpsOverlay, base::Unretained(this)),
"Toggles rendering framerate stats to an overlay on the display.",
"Framerate statistics are rendered to a display overlay. The "
"numbers are updated at the end of each animation (or every time a "
"maximum number of frames are rendered), framerate statistics are "
"printed to stdout.")),
#endif // defined(ENABLE_DEBUGGER)
clear_on_shutdown_mode_(clear_on_shutdown_mode),
enable_fps_stdout_(options.enable_fps_stdout),
enable_fps_overlay_(options.enable_fps_overlay),
fps_overlay_update_pending_(false) {
TRACE_EVENT0("cobalt::renderer", "Pipeline::Pipeline()");
// The actual Pipeline can be constructed from any thread, but we want
// rasterizer_thread_checker_ to be associated with the rasterizer thread,
// so we detach it here and let it reattach itself to the rasterizer thread
// when CalledOnValidThread() is called on rasterizer_thread_checker_ below.
DETACH_FROM_THREAD(rasterizer_thread_checker_);
base::Thread::Options thread_options(base::MessageLoop::TYPE_DEFAULT,
kRendererThreadStackSize);
thread_options.priority = base::ThreadPriority::HIGHEST;
rasterizer_thread_.StartWithOptions(thread_options);
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE,
base::Bind(&Pipeline::InitializeRasterizerThread, base::Unretained(this),
create_rasterizer_function));
}
Pipeline::~Pipeline() {
TRACE_EVENT0("cobalt::renderer", "Pipeline::~Pipeline()");
// First we shutdown the submission queue. We do this as a separate step from
// rasterizer shutdown because it may post messages back to the rasterizer
// thread as it clears itself out (e.g. it may ask the rasterizer thread to
// delete textures). We wait for this shutdown to complete before proceeding
// to shutdown the rasterizer thread.
rasterizer_thread_.message_loop()->task_runner()->PostBlockingTask(
FROM_HERE,
base::Bind(&Pipeline::ShutdownSubmissionQueue, base::Unretained(this)));
// Submit a shutdown task to the rasterizer thread so that it can shutdown
// anything that must be shutdown from that thread.
rasterizer_thread_.message_loop()->task_runner()->PostBlockingTask(
FROM_HERE,
base::Bind(&Pipeline::ShutdownRasterizerThread, base::Unretained(this)));
// Finally shutdown the rasterizer. Do this after ShutdownRasterizerThread()
// has run, just in case it posted more tasks to the rasterizer thread. This
// will free the rasterizer after the posted tasks have executed (unless they
// were delayed tasks).
rasterizer_thread_.message_loop()->task_runner()->PostBlockingTask(
FROM_HERE,
base::Bind(&Pipeline::ShutdownRasterizer, base::Unretained(this)));
rasterizer_thread_.Stop();
}
render_tree::ResourceProvider* Pipeline::GetResourceProvider() {
rasterizer_created_event_.Wait();
return rasterizer_->GetResourceProvider();
}
bool Pipeline::IsMapToMeshEnabled(const Pipeline* pipeline) {
backend::GraphicsContext* graphics_context =
pipeline ? pipeline->graphics_context_ : nullptr;
return backend::GraphicsContext::IsMapToMeshEnabled(graphics_context);
}
void Pipeline::Submit(const Submission& render_tree_submission) {
TRACE_EVENT0("cobalt::renderer", "Pipeline::Submit()");
// Execute the actual set of the new render tree on the rasterizer tree.
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE, base::Bind(&Pipeline::SetNewRenderTree, base::Unretained(this),
CollectAnimations(render_tree_submission)));
}
void Pipeline::Clear() {
TRACE_EVENT0("cobalt::renderer", "Pipeline::Clear()");
rasterizer_thread_.message_loop()->task_runner()->PostBlockingTask(
FROM_HERE,
base::Bind(&Pipeline::ClearCurrentRenderTree, base::Unretained(this)));
}
void Pipeline::RasterizeToRGBAPixels(
const scoped_refptr<render_tree::Node>& render_tree_root,
const base::Optional<math::Rect>& clip_rect,
const RasterizationCompleteCallback& complete) {
TRACK_MEMORY_SCOPE("Renderer");
TRACE_EVENT0("cobalt::renderer", "Pipeline::RasterizeToRGBAPixels()");
if (base::MessageLoop::current() != rasterizer_thread_.message_loop()) {
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE,
base::Bind(&Pipeline::RasterizeToRGBAPixels, base::Unretained(this),
render_tree_root, clip_rect, complete));
return;
}
// The default render_rect is the the whole display target.
const math::Rect& render_rect =
clip_rect ? clip_rect.value() : math::Rect(render_target_->GetSize());
// Create a new target that is the same dimensions as the render_rect.
scoped_refptr<backend::RenderTarget> offscreen_target =
graphics_context_->CreateDownloadableOffscreenRenderTarget(
render_rect.size());
// Offset the whole render tree to put the render_rect at the origin.
scoped_refptr<render_tree::Node> offset_tree_root =
new render_tree::CompositionNode(
render_tree_root.get(),
math::Vector2dF(-render_rect.x(), -render_rect.y()));
scoped_refptr<render_tree::Node> animate_node =
new render_tree::animations::AnimateNode(offset_tree_root);
Submission submission = Submission(animate_node);
// Rasterize this submission into the newly created target.
RasterizeSubmissionToRenderTarget(submission, offscreen_target, true);
// Load the texture's pixel data into a CPU memory buffer and return it.
complete.Run(graphics_context_->DownloadPixelDataAsRGBA(offscreen_target),
render_rect.size());
}
void Pipeline::TimeFence(base::TimeDelta time_fence) {
TRACK_MEMORY_SCOPE("Renderer");
TRACE_EVENT0("cobalt::renderer", "Pipeline::TimeFence()");
if (base::MessageLoop::current() != rasterizer_thread_.message_loop()) {
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE,
base::Bind(&Pipeline::TimeFence, base::Unretained(this), time_fence));
return;
}
if (!time_fence_) {
time_fence_ = time_fence;
} else {
LOG(ERROR) << "Attempting to set a time fence while one was already set.";
}
}
void Pipeline::SetNewRenderTree(const Submission& render_tree_submission) {
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
DCHECK(render_tree_submission.render_tree.get());
TRACE_EVENT0("cobalt::renderer", "Pipeline::SetNewRenderTree()");
// If a time fence is active, save the submission to be queued only after
// we pass the time fence. Overwrite any existing waiting submission in this
// case.
if (time_fence_) {
post_fence_submission_ = render_tree_submission;
post_fence_receipt_time_ = base::TimeTicks::Now();
return;
}
QueueSubmission(render_tree_submission, base::TimeTicks::Now());
// Start the rasterization timer if it is not yet started.
if (!rasterize_timer_) {
// Artificially limit the period between submissions. This is useful for
// platforms which do not rate limit themselves during swaps. Be careful
// to use a non-zero interval time even if throttling occurs during frame
// swaps. It is possible that a submission is not rendered (this can
// happen if the render tree has not changed between submissions), so no
// frame swap occurs, and the minimum frame time is the only throttle.
float minimum_frame_interval_milliseconds =
graphics_context_
? graphics_context_->GetMinimumFrameIntervalInMilliseconds()
: -1.0f;
#if SB_API_VERSION >= 12 && defined(COBALT_MINIMUM_FRAME_TIME_IN_MILLISECONDS)
#error \
"'cobalt_minimum_frame_time_in_milliseconds' was replaced by" \
"CobaltExtensionGraphicsApi::GetMinimumFrameIntervalInMilliseconds."
#elif SB_API_VERSION < 12 && defined(COBALT_MINIMUM_FRAME_TIME_IN_MILLISECONDS)
COMPILE_ASSERT(COBALT_MINIMUM_FRAME_TIME_IN_MILLISECONDS > 0,
frame_time_must_be_positive);
if (minimum_frame_interval_milliseconds < 0.0f) {
minimum_frame_interval_milliseconds =
COBALT_MINIMUM_FRAME_TIME_IN_MILLISECONDS;
} else {
DLOG(ERROR)
<< "COBALT_MINIMUM_FRAME_TIME_IN_MILLISECONDS and "
"CobaltExtensionGraphicsApi::GetMinimumFrameIntervalInMilliseconds"
"are both defined."
"Remove the 'cobalt_minimum_frame_time_in_milliseconds' ";
"from ../gyp_configuration.gypi in favor of the usage of "
"CobaltExtensionGraphicsApi::GetMinimumFrameIntervalInMilliseconds."
}
#else
if (minimum_frame_interval_milliseconds < 0.0f) {
minimum_frame_interval_milliseconds =
kCobaltMinimumFrameTimeInMilliseconds;
}
#endif
DCHECK(minimum_frame_interval_milliseconds > 0.0f);
rasterize_timer_.emplace(
FROM_HERE,
base::TimeDelta::FromMillisecondsD(minimum_frame_interval_milliseconds),
base::BindRepeating(&Pipeline::RasterizeCurrentTree,
base::Unretained(this)));
rasterize_timer_->Reset();
}
}
void Pipeline::ClearCurrentRenderTree() {
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
TRACE_EVENT0("cobalt::renderer", "Pipeline::ClearCurrentRenderTree()");
ResetSubmissionQueue();
rasterize_timer_ = base::nullopt;
}
void Pipeline::RasterizeCurrentTree() {
TRACK_MEMORY_SCOPE("Renderer");
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
TRACE_EVENT0("cobalt::renderer", "Pipeline::RasterizeCurrentTree()");
base::TimeTicks start_rasterize_time = base::TimeTicks::Now();
Submission submission =
submission_queue_->GetCurrentSubmission(start_rasterize_time);
bool is_new_render_tree = submission.render_tree != last_render_tree_;
bool has_render_tree_changed =
!last_animations_expired_ || is_new_render_tree;
bool force_rasterize =
submit_even_if_render_tree_is_unchanged_ || fps_overlay_update_pending_;
float maximum_frame_interval_milliseconds =
graphics_context_
? graphics_context_->GetMaximumFrameIntervalInMilliseconds()
: -1.0f;
if (maximum_frame_interval_milliseconds >= 0.0f) {
base::TimeDelta max_time_between_rasterize =
base::TimeDelta::FromMillisecondsD(maximum_frame_interval_milliseconds);
if (start_rasterize_time - last_rasterize_time_ >
max_time_between_rasterize) {
force_rasterize = true;
}
}
// If our render tree hasn't changed from the one that was previously
// rendered and it's okay on this system to not flip the display buffer
// frequently, then we can just not do anything here.
if (force_rasterize || has_render_tree_changed) {
// Check whether the animations in the render tree that is being rasterized
// are active.
render_tree::animations::AnimateNode* animate_node =
base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
submission.render_tree.get());
// Rasterize the last submitted render tree.
bool did_rasterize = RasterizeSubmissionToRenderTarget(
submission, render_target_, force_rasterize);
if (did_rasterize) {
last_rasterize_time_ = start_rasterize_time;
}
bool animations_expired = animate_node->expiry() <= submission.time_offset;
bool stat_tracked_animations_expired =
animate_node->depends_on_time_expiry() <= submission.time_offset;
UpdateRasterizeStats(did_rasterize, stat_tracked_animations_expired,
is_new_render_tree, start_rasterize_time,
base::TimeTicks::Now());
last_did_rasterize_ = did_rasterize;
last_animations_expired_ = animations_expired;
last_stat_tracked_animations_expired_ = stat_tracked_animations_expired;
}
if (time_fence_ && submission_queue_->submission_time(
base::TimeTicks::Now()) >= *time_fence_) {
// A time fence was active and we just crossed it, so reset it.
time_fence_ = base::nullopt;
if (post_fence_submission_) {
// A submission was waiting to be queued once we passed the time fence,
// so go ahead and queue it now.
QueueSubmission(*post_fence_submission_, *post_fence_receipt_time_);
post_fence_submission_ = base::nullopt;
post_fence_receipt_time_ = base::nullopt;
}
}
}
void Pipeline::UpdateRasterizeStats(bool did_rasterize,
bool are_stat_tracked_animations_expired,
bool is_new_render_tree,
base::TimeTicks start_time,
base::TimeTicks end_time) {
bool last_animations_active =
!last_stat_tracked_animations_expired_ && last_did_rasterize_;
bool animations_active =
!are_stat_tracked_animations_expired && did_rasterize;
// Rasterizations are only tracked by the timers when there are animations.
// This applies when animations were active during either the last
// rasterization or the current one. The reason for including the last one is
// that if animations have just expired, then this rasterization produces the
// final state of the animated tree.
if (last_animations_active || animations_active) {
if (did_rasterize) {
rasterize_animations_timer_.Start(start_time);
rasterize_animations_timer_.Stop(end_time);
// The interval timers require animations to have been active the previous
// frame so that there is a full interval to record.
if (last_animations_active) {
rasterize_periodic_interval_timer_.Stop(end_time);
rasterize_animations_interval_timer_.Stop(end_time);
}
} else {
// If we didn't actually rasterize anything, then don't count this sample.
rasterize_periodic_interval_timer_.Cancel();
rasterize_animations_interval_timer_.Cancel();
}
// If animations are active, then they are guaranteed at least one more
// interval. Start the timer to record its duration.
if (animations_active) {
rasterize_periodic_interval_timer_.Start(end_time);
rasterize_animations_interval_timer_.Start(end_time);
}
// Check for if the animations are starting or ending.
if (!last_animations_active && animations_active) {
animations_start_time_ = end_time.ToInternalValue();
has_active_animations_c_val_ = true;
} else if (last_animations_active && !animations_active) {
animations_end_time_ = end_time.ToInternalValue();
rasterize_animations_interval_timer_.Flush();
rasterize_animations_timer_.Flush();
has_active_animations_c_val_ = false;
}
}
if (is_new_render_tree) {
++new_render_tree_rasterize_count_;
new_render_tree_rasterize_time_ = end_time.ToInternalValue();
}
fallback_rasterize_count_ = rasterizer_->GetFallbackRasterizeCount();
}
bool Pipeline::RasterizeSubmissionToRenderTarget(
const Submission& submission,
const scoped_refptr<backend::RenderTarget>& render_target,
bool force_rasterize) {
TRACE_EVENT0("cobalt::renderer",
"Pipeline::RasterizeSubmissionToRenderTarget()");
// Keep track of the last render tree that we rendered so that we can watch
// if it changes, in which case we should reset our tracked
// |previous_animated_area_|.
if (submission.render_tree != last_render_tree_) {
last_render_tree_ = submission.render_tree;
last_animated_render_tree_ = NULL;
previous_animated_area_ = base::nullopt;
last_render_time_ = base::nullopt;
}
// Animate the render tree using the submitted animations.
render_tree::animations::AnimateNode* animate_node =
last_animated_render_tree_
? last_animated_render_tree_.get()
: base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
submission.render_tree.get());
// Some animations require a GL graphics context to be current. Specifically,
// a call to SbPlayerGetCurrentFrame() may be made to get the current video
// frame to drive a video-as-an-animated-image.
rasterizer::Rasterizer::ScopedMakeCurrent scoped_make_current(
rasterizer_.get());
render_tree::animations::AnimateNode::AnimateResults results =
animate_node->Apply(submission.time_offset);
if (results.animated == last_animated_render_tree_ && !force_rasterize) {
return false;
}
last_animated_render_tree_ = results.animated;
// Calculate a bounding box around the active animations. Union it with the
// bounding box around active animations from the previous frame, and we get
// a scissor rectangle marking the dirty regions of the screen.
math::RectF animated_bounds = results.get_animation_bounds_since.Run(
last_render_time_ ? *last_render_time_ : base::TimeDelta());
math::Rect rounded_bounds = math::RoundOut(animated_bounds);
base::Optional<math::Rect> redraw_area;
if (previous_animated_area_) {
redraw_area = math::UnionRects(rounded_bounds, *previous_animated_area_);
}
previous_animated_area_ = rounded_bounds;
scoped_refptr<render_tree::Node> submit_tree = results.animated->source();
if (enable_fps_overlay_ && fps_overlay_) {
submit_tree = fps_overlay_->AnnotateRenderTreeWithOverlay(
results.animated->source().get());
fps_overlay_update_pending_ = false;
}
// Rasterize the animated render tree.
rasterizer::Rasterizer::Options rasterizer_options;
rasterizer_options.dirty = redraw_area;
rasterizer_->Submit(submit_tree, render_target, rasterizer_options);
// Run all of this submission's callbacks.
for (const auto& callback : submission.on_rasterized_callbacks) {
callback.Run();
}
last_render_time_ = submission.time_offset;
return true;
}
void Pipeline::InitializeRasterizerThread(
const CreateRasterizerFunction& create_rasterizer_function) {
TRACE_EVENT0("cobalt::renderer", "Pipeline::InitializeRasterizerThread");
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
rasterizer_ = create_rasterizer_function.Run();
rasterizer_created_event_.Signal();
// Note that this is setup as high priority, but lower than the rasterizer
// thread's priority (ThreadPriority::HIGHEST). This is to ensure that
// we never interrupt the rasterizer in order to dispose render trees, but
// at the same time we do want to prioritize cleaning them up to avoid
// large queues of pending render tree disposals.
base::Thread::Options options(base::MessageLoop::TYPE_DEFAULT,
kRendererThreadStackSize);
options.priority = base::ThreadPriority::HIGHEST;
submission_disposal_thread_.StartWithOptions(options);
ResetSubmissionQueue();
}
void Pipeline::ShutdownSubmissionQueue() {
TRACE_EVENT0("cobalt::renderer", "Pipeline::ShutdownSubmissionQueue()");
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
// Clear out our time fence data, especially |post_fence_submission_| which
// may refer to a render tree.
time_fence_ = base::nullopt;
post_fence_submission_ = base::nullopt;
post_fence_receipt_time_ = base::nullopt;
// Stop and shutdown the rasterizer timer. If we won't have a submission
// queue anymore, we won't be able to rasterize anymore.
rasterize_timer_ = base::nullopt;
// Do not retain any more references to the current render tree (which
// may refer to rasterizer resources) or animations which may refer to
// render trees.
submission_queue_ = base::nullopt;
// Shut down our submission disposer thread. This needs to happen now to
// ensure that any pending "dispose" messages are processed. Each disposal
// may result in new messages being posted to this rasterizer thread's message
// loop, and so we want to make sure these are all queued up before
// proceeding.
submission_disposal_thread_.Stop();
}
void Pipeline::ShutdownRasterizerThread() {
TRACE_EVENT0("cobalt::renderer", "Pipeline::ShutdownRasterizerThread()");
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
// Shutdown the FPS overlay which may reference render trees.
fps_overlay_ = base::nullopt;
// Submit a fullscreen rect node to clear the display before shutting
// down. This can be helpful if we quit while playing a video via
// punch-through, which may result in unexpected images/colors appearing for
// a flicker behind the display.
render_tree::ColorRGBA clear_color;
if (render_target_ && clear_on_shutdown_mode_ == kClearAccordingToPlatform &&
ShouldClearFrameOnShutdown(&clear_color)) {
rasterizer_->Submit(new render_tree::RectNode(
math::RectF(render_target_->GetSize()),
std::unique_ptr<render_tree::Brush>(
new render_tree::SolidColorBrush(clear_color))),
render_target_);
}
// This potential reference to a render tree whose animations may have ended
// must be destroyed before we shutdown the rasterizer thread since it may
// contain references to render tree nodes and resources.
last_render_tree_ = NULL;
last_animated_render_tree_ = NULL;
}
#if defined(ENABLE_DEBUGGER)
void Pipeline::OnDumpCurrentRenderTree(const std::string& message) {
if (base::MessageLoop::current() != rasterizer_thread_.message_loop()) {
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE, base::Bind(&Pipeline::OnDumpCurrentRenderTree,
base::Unretained(this), message));
return;
}
if (!rasterize_timer_) {
LOG(INFO) << "No render tree available yet.";
return;
}
// Grab the most recent submission, animate it, and then dump the results to
// text.
Submission submission =
submission_queue_->GetCurrentSubmission(base::TimeTicks::Now());
render_tree::animations::AnimateNode* animate_node =
base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
submission.render_tree.get());
render_tree::animations::AnimateNode::AnimateResults results =
animate_node->Apply(submission.time_offset);
std::string tree_dump =
render_tree::DumpRenderTreeToString(results.animated->source().get());
if (message.empty() || message == "undefined") {
// If no filename was specified, send output to the console.
LOG(INFO) << tree_dump.c_str();
} else {
// If a filename was specified, dump the output to that file.
base::FilePath out_dir;
base::PathService::Get(paths::DIR_COBALT_DEBUG_OUT, &out_dir);
base::WriteFile(out_dir.Append(message), tree_dump.c_str(),
tree_dump.length());
}
}
void Pipeline::OnToggleFpsStdout(const std::string& message) {
if (base::MessageLoop::current() != rasterizer_thread_.message_loop()) {
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE, base::Bind(&Pipeline::OnToggleFpsStdout,
base::Unretained(this), message));
return;
}
enable_fps_stdout_ = !enable_fps_stdout_;
}
void Pipeline::OnToggleFpsOverlay(const std::string& message) {
if (base::MessageLoop::current() != rasterizer_thread_.message_loop()) {
rasterizer_thread_.message_loop()->task_runner()->PostTask(
FROM_HERE, base::Bind(&Pipeline::OnToggleFpsOverlay,
base::Unretained(this), message));
return;
}
enable_fps_overlay_ = !enable_fps_overlay_;
fps_overlay_update_pending_ = enable_fps_overlay_;
}
#endif // #if defined(ENABLE_DEBUGGER)
Submission Pipeline::CollectAnimations(
const Submission& render_tree_submission) {
// Constructing an AnimateNode will result in the tree being traversed to
// collect all sub-AnimateNodes into the new one, in order to maintain the
// invariant that a sub-tree of an AnimateNode has no AnimateNodes.
Submission collected_submission = render_tree_submission;
collected_submission.render_tree = new render_tree::animations::AnimateNode(
render_tree_submission.render_tree);
return collected_submission;
}
namespace {
void PrintFPS(const base::CValCollectionTimerStatsFlushResults& results) {
SbLogRaw(base::StringPrintf("FPS => # samples: %d, avg: %.1fms, "
"[min, max]: [%.1fms, %.1fms]\n"
" 25th : 50th : 75th : 95th pct - "
"%.1fms : %.1fms : %.1fms : %.1fms\n",
static_cast<unsigned int>(results.sample_count),
results.average.InMillisecondsF(),
results.minimum.InMillisecondsF(),
results.maximum.InMillisecondsF(),
results.percentile_25th.InMillisecondsF(),
results.percentile_50th.InMillisecondsF(),
results.percentile_75th.InMillisecondsF(),
results.percentile_95th.InMillisecondsF())
.c_str());
}
} // namespace
void Pipeline::FrameStatsOnFlushCallback(
const base::CValCollectionTimerStatsFlushResults& flush_results) {
DCHECK_CALLED_ON_VALID_THREAD(rasterizer_thread_checker_);
if (enable_fps_overlay_) {
if (!fps_overlay_) {
fps_overlay_.emplace(rasterizer_->GetResourceProvider());
}
fps_overlay_->UpdateOverlay(flush_results);
fps_overlay_update_pending_ = true;
}
if (enable_fps_stdout_) {
PrintFPS(flush_results);
}
}
void Pipeline::ResetSubmissionQueue() {
TRACK_MEMORY_SCOPE("Renderer");
TRACE_EVENT0("cobalt::renderer", "Pipeline::ResetSubmissionQueue()");
submission_queue_ = base::nullopt;
submission_queue_.emplace(
current_timeline_info_.max_submission_queue_size,
base::TimeDelta::FromMillisecondsD(kTimeToConvergeInMS),
current_timeline_info_.allow_latency_reduction,
base::Bind(&DestructSubmissionOnMessageLoop,
submission_disposal_thread_.message_loop()));
}
void Pipeline::QueueSubmission(const Submission& submission,
base::TimeTicks receipt_time) {
TRACK_MEMORY_SCOPE("Renderer");
TRACE_EVENT0("cobalt::renderer", "Pipeline::QueueSubmission()");
// Upon each submission, check if the timeline has changed. If it has,
// reset our submission queue (possibly with a new configuration specified
// within |timeline_info|.
if (submission.timeline_info.id != current_timeline_info_.id) {
current_timeline_info_ = submission.timeline_info;
ResetSubmissionQueue();
}
submission_queue_->PushSubmission(submission, receipt_time);
}
} // namespace renderer
} // namespace cobalt