1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227
|
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/compositor/test/direct_layer_tree_frame_sink.h"
#include <memory>
#include "base/compiler_specific.h"
#include "base/functional/bind.h"
#include "base/task/single_thread_task_runner.h"
#include "base/trace_event/trace_event.h"
#include "build/build_config.h"
#include "cc/trees/layer_tree_frame_sink_client.h"
#include "components/viz/common/hit_test/hit_test_region_list.h"
#include "components/viz/common/quads/compositor_frame.h"
#include "components/viz/common/surfaces/frame_sink_id.h"
#include "components/viz/common/surfaces/parent_local_surface_id_allocator.h"
#include "components/viz/service/display/display.h"
#include "components/viz/service/frame_sinks/frame_sink_manager_impl.h"
#include "components/viz/service/surfaces/surface.h"
#include "gpu/ipc/client/client_shared_image_interface.h"
#if BUILDFLAG(IS_APPLE)
#include "ui/accelerated_widget_mac/ca_layer_frame_sink.h"
#endif
namespace ui {
DirectLayerTreeFrameSink::DirectLayerTreeFrameSink(
const viz::FrameSinkId& frame_sink_id,
viz::FrameSinkManagerImpl* frame_sink_manager,
viz::Display* display,
scoped_refptr<viz::RasterContextProvider> context_provider,
scoped_refptr<viz::RasterContextProvider> worker_context_provider,
scoped_refptr<base::SingleThreadTaskRunner> compositor_task_runner,
gfx::AcceleratedWidget widget)
: LayerTreeFrameSink(std::move(context_provider),
std::move(worker_context_provider),
std::move(compositor_task_runner),
/*shared_image_interface=*/nullptr),
frame_sink_id_(frame_sink_id),
frame_sink_manager_(frame_sink_manager),
display_(display),
widget_(widget) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
}
DirectLayerTreeFrameSink::~DirectLayerTreeFrameSink() {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
// Reset `client_` in Display to avoid accessing `client_` after this is
// destructed. This is to resolve the circular dependency between Display and
// DirectLayerTreeFrameSink.
display_->ResetDisplayClientForTesting(/*old_client=*/this);
}
bool DirectLayerTreeFrameSink::BindToClient(
cc::LayerTreeFrameSinkClient* client) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
if (!cc::LayerTreeFrameSink::BindToClient(client))
return false;
support_ = std::make_unique<viz::CompositorFrameSinkSupport>(
this, frame_sink_manager_, frame_sink_id_, /*is_root=*/true);
begin_frame_source_ = std::make_unique<viz::ExternalBeginFrameSource>(this);
client_->SetBeginFrameSource(begin_frame_source_.get());
// Avoid initializing GL context here, as this should be sharing the
// Display's context.
display_->Initialize(this, frame_sink_manager_->surface_manager());
support_->SetUpHitTest(display_);
return true;
}
void DirectLayerTreeFrameSink::DetachFromClient() {
client_->SetBeginFrameSource(nullptr);
begin_frame_source_.reset();
// Unregister the SurfaceFactoryClient here instead of the dtor so that only
// one client is alive for this namespace at any given time.
support_.reset();
cc::LayerTreeFrameSink::DetachFromClient();
}
void DirectLayerTreeFrameSink::SubmitCompositorFrame(
viz::CompositorFrame frame,
bool hit_test_data_changed) {
DCHECK(frame.metadata.begin_frame_ack.has_damage);
DCHECK(frame.metadata.begin_frame_ack.frame_id.IsSequenceValid());
if (frame.size_in_pixels() != last_swap_frame_size_ ||
frame.device_scale_factor() != device_scale_factor_ ||
!parent_local_surface_id_allocator_.HasValidLocalSurfaceId()) {
parent_local_surface_id_allocator_.GenerateId();
last_swap_frame_size_ = frame.size_in_pixels();
device_scale_factor_ = frame.device_scale_factor();
display_->SetLocalSurfaceId(
parent_local_surface_id_allocator_.GetCurrentLocalSurfaceId(),
device_scale_factor_);
}
std::optional<viz::HitTestRegionList> hit_test_region_list =
client_->BuildHitTestData();
if (!hit_test_region_list) {
last_hit_test_data_ = viz::HitTestRegionList();
} else if (!hit_test_data_changed) {
// Do not send duplicate hit-test data.
if (viz::HitTestRegionList::IsEqual(*hit_test_region_list,
last_hit_test_data_)) {
DCHECK(!viz::HitTestRegionList::IsEqual(*hit_test_region_list,
viz::HitTestRegionList()));
hit_test_region_list = std::nullopt;
} else {
last_hit_test_data_ = *hit_test_region_list;
}
} else {
last_hit_test_data_ = *hit_test_region_list;
}
support_->SubmitCompositorFrame(
parent_local_surface_id_allocator_.GetCurrentLocalSurfaceId(),
std::move(frame), std::move(hit_test_region_list));
}
void DirectLayerTreeFrameSink::DidNotProduceFrame(
const viz::BeginFrameAck& ack,
cc::FrameSkippedReason reason) {
DCHECK(!ack.has_damage);
DCHECK(ack.frame_id.IsSequenceValid());
support_->DidNotProduceFrame(ack);
}
void DirectLayerTreeFrameSink::DisplayOutputSurfaceLost() {
is_lost_ = true;
client_->DidLoseLayerTreeFrameSink();
}
void DirectLayerTreeFrameSink::DisplayWillDrawAndSwap(
bool will_draw_and_swap,
viz::AggregatedRenderPassList* render_passes) {
if (support_->GetHitTestAggregator()) {
support_->GetHitTestAggregator()->Aggregate(display_->CurrentSurfaceId());
}
}
void DirectLayerTreeFrameSink::DisplayDidReceiveCALayerParams(
const gfx::CALayerParams& ca_layer_params) {
#if BUILDFLAG(IS_APPLE)
ui::CALayerFrameSink* ca_layer_frame_sink =
ui::CALayerFrameSink::FromAcceleratedWidget(widget_);
if (ca_layer_frame_sink) {
ca_layer_frame_sink->UpdateCALayerTree(ca_layer_params);
} else {
DLOG(WARNING) << "Received frame for non-existent widget.";
}
#else
// Suppress -Wunused-private-field warning.
(void)widget_;
NOTREACHED();
#endif
}
base::TimeDelta
DirectLayerTreeFrameSink::GetPreferredFrameIntervalForFrameSinkId(
const viz::FrameSinkId& id,
viz::mojom::CompositorFrameSinkType* type) {
return frame_sink_manager_->GetPreferredFrameIntervalForFrameSinkId(id, type);
}
void DirectLayerTreeFrameSink::DidReceiveCompositorFrameAck(
std::vector<viz::ReturnedResource> resources) {
// Submitting a CompositorFrame can synchronously draw and dispatch a frame
// ack. PostTask to ensure the client is notified on a new stack frame.
compositor_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&DirectLayerTreeFrameSink::DidReceiveCompositorFrameAckInternal,
weak_factory_.GetWeakPtr(), std::move(resources)));
}
void DirectLayerTreeFrameSink::DidReceiveCompositorFrameAckInternal(
std::vector<viz::ReturnedResource> resources) {
client_->ReclaimResources(std::move(resources));
client_->DidReceiveCompositorFrameAck();
}
void DirectLayerTreeFrameSink::OnBeginFrame(
const viz::BeginFrameArgs& args,
const viz::FrameTimingDetailsMap& timing_details,
std::vector<viz::ReturnedResource> resources) {
if (!resources.empty()) {
ReclaimResources(std::move(resources));
}
for (const auto& pair : timing_details)
client_->DidPresentCompositorFrame(pair.first, pair.second);
if (!needs_begin_frames_) {
// OnBeginFrame() can be called just to deliver presentation feedback, so
// report that we didn't use this BeginFrame.
DidNotProduceFrame(viz::BeginFrameAck(args, false),
cc::FrameSkippedReason::kNoDamage);
return;
}
begin_frame_source_->OnBeginFrame(args);
}
void DirectLayerTreeFrameSink::ReclaimResources(
std::vector<viz::ReturnedResource> resources) {
client_->ReclaimResources(std::move(resources));
}
void DirectLayerTreeFrameSink::OnBeginFramePausedChanged(bool paused) {
begin_frame_source_->OnSetBeginFrameSourcePaused(paused);
}
void DirectLayerTreeFrameSink::OnNeedsBeginFrames(bool needs_begin_frames) {
needs_begin_frames_ = needs_begin_frames;
support_->SetNeedsBeginFrame(needs_begin_frames);
}
} // namespace ui
|