Flutter Engine
The Flutter Engine
image_encoding.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/image_encoding.h"
6#include "flutter/lib/ui/painting/image_encoding_impl.h"
7
8#include <memory>
9#include <utility>
10
11#include "flutter/common/task_runners.h"
12#include "flutter/fml/build_config.h"
13#include "flutter/fml/make_copyable.h"
14#include "flutter/fml/status_or.h"
15#include "flutter/fml/trace_event.h"
16#include "flutter/lib/ui/painting/image.h"
17#include "fml/status.h"
18#if IMPELLER_SUPPORTS_RENDERING
19#include "flutter/lib/ui/painting/image_encoding_impeller.h"
20#endif // IMPELLER_SUPPORTS_RENDERING
21#include "flutter/lib/ui/painting/image_encoding_skia.h"
28
31using tonic::ToDart;
32
33namespace impeller {
34class Context;
35} // namespace impeller
36namespace flutter {
37namespace {
38
39void FinalizeSkData(void* isolate_callback_data, void* peer) {
40 SkData* buffer = reinterpret_cast<SkData*>(peer);
41 buffer->unref();
42}
43
44void InvokeDataCallback(std::unique_ptr<DartPersistentValue> callback,
46 std::shared_ptr<tonic::DartState> dart_state = callback->dart_state().lock();
47 if (!dart_state) {
48 return;
49 }
50 tonic::DartState::Scope scope(dart_state);
51 if (!buffer.ok()) {
52 std::string error_copy(buffer.status().message());
53 Dart_Handle dart_string = ToDart(error_copy);
54 DartInvoke(callback->value(), {Dart_Null(), dart_string});
55 return;
56 }
57 // Skia will not modify the buffer, and it is backed by memory that is
58 // read/write, so Dart can be given direct access to the buffer through an
59 // external Uint8List.
60 void* bytes = const_cast<void*>(buffer.value()->data());
61 const intptr_t length = buffer.value()->size();
62 void* peer = reinterpret_cast<void*>(buffer.value().release());
64 Dart_TypedData_kUint8, bytes, length, peer, length, FinalizeSkData);
65 DartInvoke(callback->value(), {dart_data, Dart_Null()});
66}
67
68fml::StatusOr<sk_sp<SkData>> CopyImageByteData(
69 const sk_sp<SkImage>& raster_image,
72 FML_DCHECK(raster_image);
73
74 SkPixmap pixmap;
75
76 if (!raster_image->peekPixels(&pixmap)) {
78 "Could not copy pixels from the raster image.");
79 }
80
81 // The color types already match. No need to swizzle. Return early.
82 if (pixmap.colorType() == color_type && pixmap.alphaType() == alpha_type) {
83 return SkData::MakeWithCopy(pixmap.addr(), pixmap.computeByteSize());
84 }
85
86 // Perform swizzle if the type doesnt match the specification.
88 SkImageInfo::Make(raster_image->width(), raster_image->height(),
89 color_type, alpha_type, nullptr));
90
91 if (!surface) {
93 "Could not set up the surface for swizzle.");
94 }
95
96 surface->writePixels(pixmap, 0, 0);
97
98 if (!surface->peekPixels(&pixmap)) {
100 "Pixel address is not available.");
101 }
102
103 return SkData::MakeWithCopy(pixmap.addr(), pixmap.computeByteSize());
104}
105
106void EncodeImageAndInvokeDataCallback(
107 const sk_sp<DlImage>& image,
108 std::unique_ptr<DartPersistentValue> callback,
110 const fml::RefPtr<fml::TaskRunner>& ui_task_runner,
111 const fml::RefPtr<fml::TaskRunner>& raster_task_runner,
112 const fml::RefPtr<fml::TaskRunner>& io_task_runner,
113 const fml::WeakPtr<GrDirectContext>& resource_context,
114 const fml::TaskRunnerAffineWeakPtr<SnapshotDelegate>& snapshot_delegate,
115 const std::shared_ptr<const fml::SyncSwitch>& is_gpu_disabled_sync_switch,
116 const std::shared_ptr<impeller::Context>& impeller_context,
117 bool is_impeller_enabled) {
118 auto callback_task =
119 fml::MakeCopyable([callback = std::move(callback)](
120 fml::StatusOr<sk_sp<SkData>>&& encoded) mutable {
121 InvokeDataCallback(std::move(callback), std::move(encoded));
122 });
123 // The static leak checker gets confused by the use of fml::MakeCopyable in
124 // EncodeImage.
125 // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
126 auto encode_task =
127 [callback_task = std::move(callback_task), format,
128 ui_task_runner](const fml::StatusOr<sk_sp<SkImage>>& raster_image) {
129 if (raster_image.ok()) {
131 EncodeImage(raster_image.value(), format);
132 ui_task_runner->PostTask([callback_task = callback_task,
133 encoded = std::move(encoded)]() mutable {
134 callback_task(std::move(encoded));
135 });
136 } else {
137 ui_task_runner->PostTask([callback_task = callback_task,
138 raster_image = raster_image]() mutable {
139 callback_task(raster_image.status());
140 });
141 }
142 };
143
145#if IMPELLER_SUPPORTS_RENDERING
146 if (is_impeller_enabled) {
148 image, encode_task, raster_task_runner, io_task_runner,
149 is_gpu_disabled_sync_switch, impeller_context);
150 return;
151 }
152#endif // IMPELLER_SUPPORTS_RENDERING
153#if !SLIMPELLER
154 ConvertImageToRasterSkia(image, encode_task, raster_task_runner,
155 io_task_runner, resource_context, snapshot_delegate,
156 is_gpu_disabled_sync_switch);
157
158#else // !SLIMPELLER
159 FML_LOG(FATAL) << "Unsupported renderer.";
160#endif // !SLIMPELLER
161}
162
163} // namespace
164
166 int format,
167 Dart_Handle callback_handle) {
168 if (!canvas_image) {
169 return ToDart("encode called with non-genuine Image.");
170 }
171
172 if (!Dart_IsClosure(callback_handle)) {
173 return ToDart("Callback must be a function.");
174 }
175
176 ImageByteFormat image_format = static_cast<ImageByteFormat>(format);
177
178 auto callback = std::make_unique<DartPersistentValue>(
179 tonic::DartState::Current(), callback_handle);
180
181 const auto& task_runners = UIDartState::Current()->GetTaskRunners();
182
183 // The static leak checker gets confused by the use of fml::MakeCopyable.
184 // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
186 [callback = std::move(callback), image = canvas_image->image(),
187 image_format, ui_task_runner = task_runners.GetUITaskRunner(),
188 raster_task_runner = task_runners.GetRasterTaskRunner(),
189 io_task_runner = task_runners.GetIOTaskRunner(),
190 io_manager = UIDartState::Current()->GetIOManager(),
191 snapshot_delegate = UIDartState::Current()->GetSnapshotDelegate(),
192 is_impeller_enabled =
194 EncodeImageAndInvokeDataCallback(
195 image, std::move(callback), image_format, ui_task_runner,
196 raster_task_runner, io_task_runner,
197 io_manager->GetResourceContext(), snapshot_delegate,
198 io_manager->GetIsGpuDisabledSyncSwitch(),
199 io_manager->GetImpellerContext(), is_impeller_enabled);
200 }));
201
202 return Dart_Null();
203}
204
207 TRACE_EVENT0("flutter", __FUNCTION__);
208
209 if (!raster_image) {
210 return fml::Status(fml::StatusCode::kInternal, "Missing raster image.");
211 }
212
213 switch (format) {
214 case kPNG: {
215 auto png_image = SkPngEncoder::Encode(nullptr, raster_image.get(), {});
216
217 if (png_image == nullptr) {
219 "Could not convert raster image to PNG.");
220 };
221 return png_image;
222 }
223 case kRawRGBA:
224 return CopyImageByteData(raster_image, kRGBA_8888_SkColorType,
226
227 case kRawStraightRGBA:
228 return CopyImageByteData(raster_image, kRGBA_8888_SkColorType,
230
231 case kRawUnmodified:
232 return CopyImageByteData(raster_image, raster_image->colorType(),
233 raster_image->alphaType());
235 return CopyImageByteData(raster_image, kRGBA_F32_SkColorType,
237 }
238
240 "Unknown error encoding image.");
241}
242
243} // namespace flutter
kUnpremul_SkAlphaType
SkAlphaType
Definition: SkAlphaType.h:26
@ kPremul_SkAlphaType
pixel components are premultiplied by alpha
Definition: SkAlphaType.h:29
SkColorType
Definition: SkColorType.h:19
@ kRGBA_8888_SkColorType
pixel with 8 bits for red, green, blue, alpha; in 32-bit word
Definition: SkColorType.h:24
@ kRGBA_F32_SkColorType
pixel using C float for red, green, blue, alpha; in 128-bit word
Definition: SkColorType.h:40
Definition: SkData.h:25
static sk_sp< SkData > MakeWithCopy(const void *data, size_t length)
Definition: SkData.cpp:111
SkAlphaType alphaType() const
Definition: SkImage.cpp:154
bool peekPixels(SkPixmap *pixmap) const
Definition: SkImage.cpp:34
int width() const
Definition: SkImage.h:285
SkColorType colorType() const
Definition: SkImage.cpp:152
int height() const
Definition: SkImage.h:291
SkColorType colorType() const
Definition: SkPixmap.h:173
size_t computeByteSize() const
Definition: SkPixmap.h:231
const void * addr() const
Definition: SkPixmap.h:153
SkAlphaType alphaType() const
Definition: SkPixmap.h:175
sk_sp< DlImage > image() const
Definition: image.h:42
static void ConvertImageToRaster(const sk_sp< DlImage > &dl_image, std::function< void(fml::StatusOr< sk_sp< SkImage > >)> encode_task, const fml::RefPtr< fml::TaskRunner > &raster_task_runner, const fml::RefPtr< fml::TaskRunner > &io_task_runner, const std::shared_ptr< const fml::SyncSwitch > &is_gpu_disabled_sync_switch, const std::shared_ptr< impeller::Context > &impeller_context)
fml::RefPtr< fml::TaskRunner > GetIOTaskRunner() const
Definition: task_runners.cc:38
fml::WeakPtr< IOManager > GetIOManager() const
const TaskRunners & GetTaskRunners() const
bool IsImpellerEnabled() const
Whether Impeller is enabled for this application.
fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > GetSnapshotDelegate() const
static UIDartState * Current()
virtual void PostTask(const fml::closure &task) override
Definition: task_runner.cc:24
T * get() const
Definition: SkRefCnt.h:303
static DartState * Current()
Definition: dart_state.cc:56
struct _Dart_Handle * Dart_Handle
Definition: dart_api.h:258
DART_EXPORT Dart_Handle Dart_NewExternalTypedDataWithFinalizer(Dart_TypedData_Type type, void *data, intptr_t length, void *peer, intptr_t external_allocation_size, Dart_HandleFinalizer callback)
@ Dart_TypedData_kUint8
Definition: dart_api.h:2615
DART_EXPORT Dart_Handle Dart_Null(void)
DART_EXPORT bool Dart_IsClosure(Dart_Handle object)
VkSurfaceKHR surface
Definition: main.cc:49
#define FATAL(error)
FlKeyEvent uint64_t FlKeyResponderAsyncCallback callback
#define FML_LOG(severity)
Definition: logging.h:82
#define FML_DCHECK(condition)
Definition: logging.h:103
size_t length
SK_API bool Encode(SkWStream *dst, const SkPixmap &src, const Options &options)
sk_sp< const SkImage > image
Definition: SkRecords.h:269
SK_API sk_sp< SkSurface > Raster(const SkImageInfo &imageInfo, size_t rowBytes, const SkSurfaceProps *surfaceProps)
@ kRawUnmodified
@ kRawStraightRGBA
@ kRawExtendedRgba128
DEF_SWITCHES_START aot vmservice shared library Name of the *so containing AOT compiled Dart assets for launching the service isolate vm snapshot The VM snapshot data that will be memory mapped as read only SnapshotAssetPath must be present isolate snapshot The isolate snapshot data that will be memory mapped as read only SnapshotAssetPath must be present cache dir Path to the cache directory This is different from the persistent_cache_path in embedder which is used for Skia shader cache icu native lib Path to the library file that exports the ICU data vm service The hostname IP address on which the Dart VM Service should be served If not defaults to or::depending on whether ipv6 is specified vm service A custom Dart VM Service port The default is to pick a randomly available open port disable vm Disable the Dart VM Service The Dart VM Service is never available in release mode disable vm service Disable mDNS Dart VM Service publication Bind to the IPv6 localhost address for the Dart VM Service Ignored if vm service host is set endless trace buffer
Definition: switches.h:126
DEF_SWITCHES_START aot vmservice shared library Name of the *so containing AOT compiled Dart assets for launching the service isolate vm snapshot The VM snapshot data that will be memory mapped as read only SnapshotAssetPath must be present isolate snapshot The isolate snapshot data that will be memory mapped as read only SnapshotAssetPath must be present cache dir Path to the cache directory This is different from the persistent_cache_path in embedder which is used for Skia shader cache icu native lib Path to the library file that exports the ICU data vm service The hostname IP address on which the Dart VM Service should be served If not defaults to or::depending on whether ipv6 is specified vm service A custom Dart VM Service port The default is to pick a randomly available open port disable vm Disable the Dart VM Service The Dart VM Service is never available in release mode disable vm service Disable mDNS Dart VM Service publication Bind to the IPv6 localhost address for the Dart VM Service Ignored if vm service host is set endless trace Enable an endless trace buffer The default is a ring buffer This is useful when very old events need to viewed For during application launch Memory usage will continue to grow indefinitely however Start app with an specific route defined on the framework flutter assets Path to the Flutter assets directory enable service port Allow the VM service to fallback to automatic port selection if binding to a specified port fails trace Trace early application lifecycle Automatically switches to an endless trace buffer trace skia Filters out all Skia trace event categories except those that are specified in this comma separated list dump skp on shader Automatically dump the skp that triggers new shader compilations This is useful for writing custom ShaderWarmUp to reduce jank By this is not enabled to reduce the overhead purge persistent Remove all existing persistent cache This is mainly for debugging purposes such as reproducing the shader compilation jank trace to Write the timeline trace to a file at the specified path The file will be in Perfetto s proto format
Definition: switches.h:203
void ConvertImageToRasterSkia(const sk_sp< DlImage > &dl_image, std::function< void(sk_sp< SkImage >)> encode_task, const fml::RefPtr< fml::TaskRunner > &raster_task_runner, const fml::RefPtr< fml::TaskRunner > &io_task_runner, const fml::WeakPtr< GrDirectContext > &resource_context, const fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > &snapshot_delegate, const std::shared_ptr< const fml::SyncSwitch > &is_gpu_disabled_sync_switch)
Dart_Handle EncodeImage(CanvasImage *canvas_image, int format, Dart_Handle callback_handle)
internal::CopyableLambda< T > MakeCopyable(T lambda)
Definition: make_copyable.h:57
Task::Status Status
Definition: TaskList.cpp:15
Dart_Handle ToDart(const T &object)
Dart_Handle DartInvoke(Dart_Handle closure, std::initializer_list< Dart_Handle > args)
Definition: dart_invoke.cc:20
uint32_t color_type
uint32_t alpha_type
static SkImageInfo Make(int width, int height, SkColorType ct, SkAlphaType at)
#define TRACE_EVENT0(category_group, name)
Definition: trace_event.h:131