Flutter Engine
The Flutter Engine
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
picture.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "flutter/lib/ui/painting/picture.h"
6
7#include <memory>
8#include <utility>
9
10#include "flutter/fml/make_copyable.h"
11#include "flutter/lib/ui/painting/canvas.h"
12#include "flutter/lib/ui/painting/display_list_deferred_image_gpu_skia.h"
13#include "flutter/lib/ui/ui_dart_state.h"
14#if IMPELLER_SUPPORTS_RENDERING
15#include "flutter/lib/ui/painting/display_list_deferred_image_gpu_impeller.h"
16#endif // IMPELLER_SUPPORTS_RENDERING
17#include "flutter/lib/ui/painting/display_list_image_gpu.h"
24
25namespace flutter {
26
28
30 Dart_Handle dart_handle,
31 sk_sp<DisplayList> display_list) {
32 FML_DCHECK(display_list->isUIThreadSafe());
33 auto canvas_picture = fml::MakeRefCounted<Picture>(std::move(display_list));
34 canvas_picture->AssociateWithDartWrapper(dart_handle);
35}
36
37Picture::Picture(sk_sp<DisplayList> display_list)
38 : display_list_(std::move(display_list)) {}
39
40Picture::~Picture() = default;
41
43 uint32_t height,
44 Dart_Handle raw_image_callback) {
45 if (!display_list_) {
46 return tonic::ToDart("Picture is null");
47 }
48 return RasterizeToImage(display_list_, width, height, raw_image_callback);
49}
50
52 uint32_t height,
53 Dart_Handle raw_image_handle) {
54 FML_DCHECK(display_list_);
55 RasterizeToImageSync(display_list_, width, height, raw_image_handle);
56}
57
59 bool impeller,
60 sk_sp<DisplayList> display_list,
61 uint32_t width,
62 uint32_t height,
64 fml::RefPtr<fml::TaskRunner> raster_task_runner,
65 fml::RefPtr<SkiaUnrefQueue> unref_queue) {
66#if IMPELLER_SUPPORTS_RENDERING
67 if (impeller) {
69 std::move(display_list), SkISize::Make(width, height),
70 std::move(snapshot_delegate), std::move(raster_task_runner));
71 }
72#endif // IMPELLER_SUPPORTS_RENDERING
73
74#if SLIMPELLER
75 FML_LOG(FATAL) << "Impeller opt-out unavailable.";
76 return nullptr;
77#else // SLIMPELLER
78 const SkImageInfo image_info = SkImageInfo::Make(
81 image_info, std::move(display_list), std::move(snapshot_delegate),
82 raster_task_runner, std::move(unref_queue));
83#endif // !SLIMPELLER
84}
85
86// static
88 uint32_t width,
89 uint32_t height,
90 Dart_Handle raw_image_handle) {
91 auto* dart_state = UIDartState::Current();
92 if (!dart_state) {
93 return;
94 }
95 auto unref_queue = dart_state->GetSkiaUnrefQueue();
96 auto snapshot_delegate = dart_state->GetSnapshotDelegate();
97 auto raster_task_runner = dart_state->GetTaskRunners().GetRasterTaskRunner();
98
100 auto dl_image = CreateDeferredImage(
101 dart_state->IsImpellerEnabled(), std::move(display_list), width, height,
102 std::move(snapshot_delegate), std::move(raster_task_runner),
103 std::move(unref_queue));
104 image->set_image(dl_image);
105 image->AssociateWithDartWrapper(raw_image_handle);
106}
107
109 display_list_.reset();
111}
112
114 if (display_list_) {
115 return display_list_->bytes() + sizeof(Picture);
116 } else {
117 return sizeof(Picture);
118 }
119}
120
122 uint32_t width,
123 uint32_t height,
124 Dart_Handle raw_image_callback) {
126 raw_image_callback);
127}
128
130 std::unique_ptr<LayerTree> layer_tree,
131 Dart_Handle raw_image_callback) {
132 FML_DCHECK(layer_tree != nullptr);
133 auto frame_size = layer_tree->frame_size();
134 return DoRasterizeToImage(nullptr, std::move(layer_tree), frame_size.width(),
135 frame_size.height(), raw_image_callback);
136}
137
139 std::unique_ptr<LayerTree> layer_tree,
140 uint32_t width,
141 uint32_t height,
142 Dart_Handle raw_image_callback) {
143 // Either display_list or layer_tree should be provided.
144 FML_DCHECK((display_list == nullptr) != (layer_tree == nullptr));
145
146 if (Dart_IsNull(raw_image_callback) || !Dart_IsClosure(raw_image_callback)) {
147 return tonic::ToDart("Image callback was invalid");
148 }
149
150 if (width == 0 || height == 0) {
151 return tonic::ToDart("Image dimensions for scene were invalid.");
152 }
153
154 auto* dart_state = UIDartState::Current();
155 auto image_callback = std::make_unique<tonic::DartPersistentValue>(
156 dart_state, raw_image_callback);
157 auto unref_queue = dart_state->GetSkiaUnrefQueue();
158 auto ui_task_runner = dart_state->GetTaskRunners().GetUITaskRunner();
159 auto raster_task_runner = dart_state->GetTaskRunners().GetRasterTaskRunner();
160 auto snapshot_delegate = dart_state->GetSnapshotDelegate();
161
162 // We can't create an image on this task runner because we don't have a
163 // graphics context. Even if we did, it would be slow anyway. Also, this
164 // thread owns the sole reference to the layer tree. So we do it in the
165 // raster thread.
166
167 auto ui_task =
168 // The static leak checker gets confused by the use of fml::MakeCopyable.
169 // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
170 fml::MakeCopyable([image_callback = std::move(image_callback),
171 unref_queue](sk_sp<DlImage> image) mutable {
172 auto dart_state = image_callback->dart_state().lock();
173 if (!dart_state) {
174 // The root isolate could have died in the meantime.
175 return;
176 }
177 tonic::DartState::Scope scope(dart_state);
178
179 if (!image) {
180 tonic::DartInvoke(image_callback->Get(), {Dart_Null()});
181 return;
182 }
183
184 if (!image->isUIThreadSafe()) {
185 // All images with impeller textures should already be safe.
186 FML_DCHECK(image->impeller_texture() == nullptr);
187 image =
188 DlImageGPU::Make({image->skia_image(), std::move(unref_queue)});
189 }
190
191 auto dart_image = CanvasImage::Create();
192 dart_image->set_image(image);
193 auto* raw_dart_image = tonic::ToDart(dart_image);
194
195 // All done!
196 tonic::DartInvoke(image_callback->Get(), {raw_dart_image});
197
198 // image_callback is associated with the Dart isolate and must be
199 // deleted on the UI thread.
200 image_callback.reset();
201 });
202
203 // Kick things off on the raster rask runner.
205 raster_task_runner,
206 fml::MakeCopyable([ui_task_runner, snapshot_delegate, display_list, width,
207 height, ui_task,
208 layer_tree = std::move(layer_tree)]() mutable {
209 auto picture_bounds = SkISize::Make(width, height);
211 sk_sp<DisplayList> snapshot_display_list = display_list;
212 if (layer_tree) {
213 FML_DCHECK(picture_bounds == layer_tree->frame_size());
214 snapshot_display_list =
215 layer_tree->Flatten(SkRect::MakeWH(width, height),
216 snapshot_delegate->GetTextureRegistry(),
217 snapshot_delegate->GetGrContext());
218 }
219 snapshot_delegate->MakeRasterSnapshot(
220 snapshot_display_list, picture_bounds,
221 [ui_task_runner, ui_task](const sk_sp<DlImage>& image) {
223 ui_task_runner, [ui_task, image]() { ui_task(image); });
224 });
225 }));
226
227 return Dart_Null();
228}
229
230} // namespace flutter
@ kPremul_SkAlphaType
pixel components are premultiplied by alpha
Definition: SkAlphaType.h:29
@ kRGBA_8888_SkColorType
pixel with 8 bits for red, green, blue, alpha; in 32-bit word
Definition: SkColorType.h:24
static fml::RefPtr< CanvasImage > Create()
Definition: image.h:28
static sk_sp< DlDeferredImageGPUImpeller > Make(std::unique_ptr< LayerTree > layer_tree, fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > snapshot_delegate, fml::RefPtr< fml::TaskRunner > raster_task_runner)
static sk_sp< DlDeferredImageGPUSkia > Make(const SkImageInfo &image_info, sk_sp< DisplayList > display_list, fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > snapshot_delegate, const fml::RefPtr< fml::TaskRunner > &raster_task_runner, fml::RefPtr< SkiaUnrefQueue > unref_queue)
static sk_sp< DlImageGPU > Make(SkiaGPUObject< SkImage > image)
~Picture() override
sk_sp< DisplayList > display_list() const
Definition: picture.h:27
static void CreateAndAssociateWithDartWrapper(Dart_Handle dart_handle, sk_sp< DisplayList > display_list)
Definition: picture.cc:29
Dart_Handle toImage(uint32_t width, uint32_t height, Dart_Handle raw_image_callback)
Definition: picture.cc:42
void dispose()
Definition: picture.cc:108
static Dart_Handle RasterizeLayerTreeToImage(std::unique_ptr< LayerTree > layer_tree, Dart_Handle raw_image_callback)
Definition: picture.cc:129
size_t GetAllocationSize() const
Definition: picture.cc:113
void toImageSync(uint32_t width, uint32_t height, Dart_Handle raw_image_handle)
Definition: picture.cc:51
static Dart_Handle DoRasterizeToImage(const sk_sp< DisplayList > &display_list, std::unique_ptr< LayerTree > layer_tree, uint32_t width, uint32_t height, Dart_Handle raw_image_callback)
Definition: picture.cc:138
static Dart_Handle RasterizeToImage(const sk_sp< DisplayList > &display_list, uint32_t width, uint32_t height, Dart_Handle raw_image_callback)
Definition: picture.cc:121
static void RasterizeToImageSync(sk_sp< DisplayList > display_list, uint32_t width, uint32_t height, Dart_Handle raw_image_handle)
Definition: picture.cc:87
static UIDartState * Current()
static void RunNowOrPostTask(const fml::RefPtr< fml::TaskRunner > &runner, const fml::closure &task)
Definition: task_runner.cc:55
struct _Dart_Handle * Dart_Handle
Definition: dart_api.h:258
DART_EXPORT Dart_Handle Dart_Null(void)
DART_EXPORT bool Dart_IsNull(Dart_Handle object)
DART_EXPORT bool Dart_IsClosure(Dart_Handle object)
#define FATAL(error)
#define FML_LOG(severity)
Definition: logging.h:82
#define FML_DCHECK(condition)
Definition: logging.h:103
sk_sp< const SkImage > image
Definition: SkRecords.h:269
IMPLEMENT_WRAPPERTYPEINFO(flutter_gpu, FlutterGpuTestClass)
static sk_sp< DlImage > CreateDeferredImage(bool impeller, std::unique_ptr< LayerTree > layer_tree, fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > snapshot_delegate, fml::RefPtr< fml::TaskRunner > raster_task_runner, fml::RefPtr< SkiaUnrefQueue > unref_queue)
Definition: scene.cc:74
internal::CopyableLambda< T > MakeCopyable(T lambda)
Definition: make_copyable.h:57
Definition: ref_ptr.h:256
Dart_Handle ToDart(const T &object)
Dart_Handle DartInvoke(Dart_Handle closure, std::initializer_list< Dart_Handle > args)
Definition: dart_invoke.cc:20
int32_t height
int32_t width
static constexpr SkISize Make(int32_t w, int32_t h)
Definition: SkSize.h:20
static SkImageInfo Make(int width, int height, SkColorType ct, SkAlphaType at)
static constexpr SkRect MakeWH(float w, float h)
Definition: SkRect.h:609