18#if IMPELLER_SUPPORTS_RENDERING
22#include "third_party/skia/include/core/SkImage.h"
23#include "third_party/skia/include/core/SkSurface.h"
24#include "third_party/skia/include/encode/SkPngEncoder.h"
39void FinalizeSkData(
void* isolate_callback_data,
void* peer) {
40 SkData*
buffer =
reinterpret_cast<SkData*
>(peer);
44void InvokeDataCallback(std::unique_ptr<DartPersistentValue>
callback,
46 std::shared_ptr<tonic::DartState> dart_state =
callback->dart_state().lock();
52 std::string error_copy(
buffer.status().message());
53 Dart_Handle dart_string =
ToDart(error_copy);
60 void* bytes =
const_cast<void*
>(
buffer.value()->data());
62 void* peer =
reinterpret_cast<void*
>(
buffer.value().release());
63 Dart_Handle dart_data = Dart_NewExternalTypedDataWithFinalizer(
64 Dart_TypedData_kUint8, bytes,
length, peer,
length, FinalizeSkData);
69 const sk_sp<SkImage>& raster_image,
76 if (!raster_image->peekPixels(&pixmap)) {
78 "Could not copy pixels from the raster image.");
83 return SkData::MakeWithCopy(pixmap.addr(), pixmap.computeByteSize());
87 auto surface = SkSurfaces::Raster(
88 SkImageInfo::Make(raster_image->width(), raster_image->height(),
93 "Could not set up the surface for swizzle.");
96 surface->writePixels(pixmap, 0, 0);
98 if (!
surface->peekPixels(&pixmap)) {
100 "Pixel address is not available.");
103 return SkData::MakeWithCopy(pixmap.addr(), pixmap.computeByteSize());
106void EncodeImageAndInvokeDataCallback(
107 const sk_sp<DlImage>&
image,
108 std::unique_ptr<DartPersistentValue>
callback,
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) {
121 InvokeDataCallback(std::move(
callback), std::move(encoded));
127 [callback_task = std::move(callback_task),
format,
129 if (raster_image.ok()) {
132 ui_task_runner->PostTask([callback_task = callback_task,
133 encoded = std::move(encoded)]()
mutable {
134 callback_task(std::move(encoded));
137 ui_task_runner->PostTask([callback_task = callback_task,
138 raster_image = raster_image]()
mutable {
139 callback_task(raster_image.status());
145#if IMPELLER_SUPPORTS_RENDERING
146 if (is_impeller_enabled) {
148 image, encode_task, raster_task_runner, io_task_runner,
149 snapshot_delegate, is_gpu_disabled_sync_switch, impeller_context);
155 io_task_runner, resource_context, snapshot_delegate,
156 is_gpu_disabled_sync_switch);
159 FML_LOG(FATAL) <<
"Unsupported renderer.";
167 Dart_Handle callback_handle) {
169 return ToDart(
"encode called with non-genuine Image.");
172 if (!Dart_IsClosure(callback_handle)) {
173 return ToDart(
"Callback must be a function.");
178 auto callback = std::make_unique<DartPersistentValue>(
187 image_format, ui_task_runner = task_runners.GetUITaskRunner(),
188 raster_task_runner = task_runners.GetRasterTaskRunner(),
189 io_task_runner = task_runners.GetIOTaskRunner(),
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);
215 auto png_image = SkPngEncoder::Encode(
nullptr, raster_image.get(), {});
217 if (png_image ==
nullptr) {
219 "Could not convert raster image to PNG.");
224 return CopyImageByteData(raster_image, kRGBA_8888_SkColorType,
225 kPremul_SkAlphaType);
228 return CopyImageByteData(raster_image, kRGBA_8888_SkColorType,
229 kUnpremul_SkAlphaType);
232 return CopyImageByteData(raster_image, raster_image->colorType(),
233 raster_image->alphaType());
235 return CopyImageByteData(raster_image, kRGBA_F32_SkColorType,
236 kUnpremul_SkAlphaType);
240 "Unknown error encoding image.");
sk_sp< DlImage > image() const
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 fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > &snapshot_delegate, 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
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
static DartState * Current()
FlutterVulkanImage * image
uint32_t uint32_t * format
FlutterDesktopBinaryReply callback
#define FML_LOG(severity)
#define FML_DCHECK(condition)
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)
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 disable vm Disable the Dart VM Service The Dart VM Service is never available in release mode Bind to the IPv6 localhost address for the Dart VM Service Ignored if vm service host is set profile Make the profiler discard new samples once the profiler sample buffer is full When this flag is not the profiler sample buffer is used as a ring buffer
internal::CopyableLambda< T > MakeCopyable(T lambda)
Dart_Handle ToDart(const T &object)
Dart_Handle DartInvoke(Dart_Handle closure, std::initializer_list< Dart_Handle > args)
#define TRACE_EVENT0(category_group, name)