Flutter Engine
 
Loading...
Searching...
No Matches
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
6
7#include <memory>
8#include <utility>
9
14#if IMPELLER_SUPPORTS_RENDERING
16#endif // IMPELLER_SUPPORTS_RENDERING
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
42Dart_Handle Picture::toImage(uint32_t width,
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
58static sk_sp<DlImage> CreateDeferredImage(
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 const fml::RefPtr<SkiaUnrefQueue>& unref_queue) {
66#if IMPELLER_SUPPORTS_RENDERING
67 if (impeller) {
69 std::move(display_list), DlISize(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(
79 width, height, kRGBA_8888_SkColorType, kPremul_SkAlphaType);
80 return DlDeferredImageGPUSkia::Make(image_info, std::move(display_list),
81 std::move(snapshot_delegate),
82 raster_task_runner, unref_queue);
83#endif // !SLIMPELLER
84}
85
86// static
87void Picture::RasterizeToImageSync(sk_sp<DisplayList> display_list,
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), unref_queue);
103 image->set_image(dl_image);
104 image->AssociateWithDartWrapper(raw_image_handle);
105}
106
108 display_list_.reset();
110}
111
113 if (display_list_) {
114 return display_list_->bytes() + sizeof(Picture);
115 } else {
116 return sizeof(Picture);
117 }
118}
119
120Dart_Handle Picture::RasterizeToImage(const sk_sp<DisplayList>& display_list,
121 uint32_t width,
122 uint32_t height,
123 Dart_Handle raw_image_callback) {
125 raw_image_callback);
126}
127
129 std::unique_ptr<LayerTree> layer_tree,
130 Dart_Handle raw_image_callback) {
131 FML_DCHECK(layer_tree != nullptr);
132 auto frame_size = layer_tree->frame_size();
133 return DoRasterizeToImage(nullptr, std::move(layer_tree), frame_size.width,
134 frame_size.height, raw_image_callback);
135}
136
137Dart_Handle Picture::DoRasterizeToImage(const sk_sp<DisplayList>& display_list,
138 std::unique_ptr<LayerTree> layer_tree,
139 uint32_t width,
140 uint32_t height,
141 Dart_Handle raw_image_callback) {
142 // Either display_list or layer_tree should be provided.
143 FML_DCHECK((display_list == nullptr) != (layer_tree == nullptr));
144
145 if (Dart_IsNull(raw_image_callback) || !Dart_IsClosure(raw_image_callback)) {
146 return tonic::ToDart("Image callback was invalid");
147 }
148
149 if (width == 0 || height == 0) {
150 return tonic::ToDart("Image dimensions for scene were invalid.");
151 }
152
153 auto* dart_state = UIDartState::Current();
154 auto image_callback = std::make_unique<tonic::DartPersistentValue>(
155 dart_state, raw_image_callback);
156 auto unref_queue = dart_state->GetSkiaUnrefQueue();
157 auto ui_task_runner = dart_state->GetTaskRunners().GetUITaskRunner();
158 auto raster_task_runner = dart_state->GetTaskRunners().GetRasterTaskRunner();
159 auto snapshot_delegate = dart_state->GetSnapshotDelegate();
160
161 // We can't create an image on this task runner because we don't have a
162 // graphics context. Even if we did, it would be slow anyway. Also, this
163 // thread owns the sole reference to the layer tree. So we do it in the
164 // raster thread.
165
166 auto ui_task =
167 // The static leak checker gets confused by the use of fml::MakeCopyable.
168 // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
169 fml::MakeCopyable([image_callback = std::move(image_callback),
170 unref_queue](sk_sp<DlImage> image) mutable {
171 auto dart_state = image_callback->dart_state().lock();
172 if (!dart_state) {
173 // The root isolate could have died in the meantime.
174 return;
175 }
176 tonic::DartState::Scope scope(dart_state);
177
178 if (!image) {
179 tonic::DartInvoke(image_callback->Get(), {Dart_Null()});
180 return;
181 }
182
183 if (!image->isUIThreadSafe()) {
184 // All images with impeller textures should already be safe.
185 FML_DCHECK(image->impeller_texture() == nullptr);
186 image =
187 DlImageGPU::Make({image->skia_image(), std::move(unref_queue)});
188 }
189
190 auto dart_image = CanvasImage::Create();
191 dart_image->set_image(image);
192 auto* raw_dart_image = tonic::ToDart(dart_image);
193
194 // All done!
195 tonic::DartInvoke(image_callback->Get(), {raw_dart_image});
196
197 // image_callback is associated with the Dart isolate and must be
198 // deleted on the UI thread.
199 image_callback.reset();
200 // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
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 = DlISize(width, height);
210 sk_sp<DisplayList> snapshot_display_list = display_list;
211 if (layer_tree) {
212 FML_DCHECK(picture_bounds == layer_tree->frame_size());
213 snapshot_display_list =
214 layer_tree->Flatten(DlRect::MakeWH(width, height),
215 snapshot_delegate->GetTextureRegistry(),
216 snapshot_delegate->GetGrContext());
217 }
218 snapshot_delegate->MakeRasterSnapshot(
219 snapshot_display_list, picture_bounds,
220 [ui_task_runner, ui_task](const sk_sp<DlImage>& image) {
222 ui_task_runner, [ui_task, image]() { ui_task(image); });
223 });
224 }));
225
226 return Dart_Null();
227 // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
228}
229
230} // namespace flutter
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
static Dart_Handle RasterizeLayerTreeToImage(std::unique_ptr< LayerTree > layer_tree, Dart_Handle raw_image_callback)
Definition picture.cc:128
size_t GetAllocationSize() const
Definition picture.cc:112
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:137
static Dart_Handle RasterizeToImage(const sk_sp< DisplayList > &display_list, uint32_t width, uint32_t height, Dart_Handle raw_image_callback)
Definition picture.cc:120
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)
#define IMPLEMENT_WRAPPERTYPEINFO(LibraryName, ClassName)
FlutterVulkanImage * image
#define FML_LOG(severity)
Definition logging.h:101
#define FML_DCHECK(condition)
Definition logging.h:122
impeller::ISize32 DlISize
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, const fml::RefPtr< SkiaUnrefQueue > &unref_queue)
Definition scene.cc:74
internal::CopyableLambda< T > MakeCopyable(T lambda)
Definition ref_ptr.h:261
Dart_Handle ToDart(const T &object)
Dart_Handle DartInvoke(Dart_Handle closure, std::initializer_list< Dart_Handle > args)
int32_t height
int32_t width
static constexpr TRect MakeWH(Type width, Type height)
Definition rect.h:140