14#if IMPELLER_SUPPORTS_RENDERING
22#include "third_party/skia/include/core/SkString.h"
23#include "third_party/skia/include/effects/SkRuntimeEffect.h"
47Dart_Handle ConvertUniformDescriptionToMap(
50 constexpr int num_entries = 5;
51 Dart_Handle keys = Dart_NewList(num_entries);
53 Dart_Handle values = Dart_NewList(num_entries);
57 Dart_NewStringFromCString(uniform_description.
name.c_str());
59 [[maybe_unused]] Dart_Handle result =
60 Dart_ListSetAt(keys, 0, Dart_NewStringFromCString(
"name"));
62 result = Dart_ListSetAt(values, 0,
name);
67 switch (uniform_description.
type) {
69 type = Dart_NewStringFromCString(
"Float");
72 type = Dart_NewStringFromCString(
"SampledImage");
75 type = Dart_NewStringFromCString(
"Struct");
79 [[maybe_unused]] Dart_Handle result =
80 Dart_ListSetAt(keys, 1, Dart_NewStringFromCString(
"type"));
82 result = Dart_ListSetAt(values, 1,
type);
87 Dart_NewIntegerFromUint64(uniform_description.
GetDartSize());
89 [[maybe_unused]] Dart_Handle result =
90 Dart_ListSetAt(keys, 2, Dart_NewStringFromCString(
"size"));
92 result = Dart_ListSetAt(values, 2,
size);
97 Dart_Handle struct_member_names =
100 for (
const auto& elem : uniform_description.struct_fields) {
101 Dart_ListSetAt(struct_member_names,
i++,
102 Dart_NewStringFromCString(elem.name.c_str()));
104 [[maybe_unused]] Dart_Handle result = Dart_ListSetAt(
105 keys, 3, Dart_NewStringFromCString(
"struct_field_names"));
107 result = Dart_ListSetAt(values, 3, struct_member_names);
112 Dart_Handle struct_member_bytes =
115 for (
const auto& elem : uniform_description.struct_fields) {
116 Dart_ListSetAt(struct_member_bytes,
i++, Dart_NewInteger(elem.byte_size));
118 [[maybe_unused]] Dart_Handle result = Dart_ListSetAt(
119 keys, 4, Dart_NewStringFromCString(
"struct_field_bytes"));
121 result = Dart_ListSetAt(values, 4, struct_member_bytes);
125 Dart_NewMap(Dart_TypeString(), keys, Dart_TypeObject(), values);
135 std::shared_ptr<AssetManager> asset_manager =
138 std::unique_ptr<fml::Mapping>
data = asset_manager->GetAsMapping(asset_name);
139 if (
data ==
nullptr) {
140 return std::string(
"Asset '") + asset_name + std::string(
"' not found");
143 auto runtime_stages =
146 if (!runtime_stages.ok()) {
147 return std::string(
"Asset '") + asset_name +
148 std::string(
"' manifest could not be decoded: ") +
149 runtime_stages.status().ToString();
152 if (runtime_stages->empty()) {
153 return std::string(
"Asset '") + asset_name +
154 std::string(
"' does not contain any shader data.");
159 std::shared_ptr<impeller::RuntimeStage> runtime_stage =
160 (*runtime_stages)[backend];
161 if (!runtime_stage) {
162 std::ostringstream stream;
163 stream <<
"Asset '" << asset_name
164 <<
"' does not contain appropriate runtime stage data for current "
168 for (
const auto& kvp : *runtime_stages) {
176 int sampled_image_count = 0;
177 size_t other_uniforms_bytes = 0;
178 const std::vector<impeller::RuntimeUniformDescription>& uniforms =
179 runtime_stage->GetUniforms();
180 Dart_Handle uniform_info = Dart_NewList(uniforms.size());
182 for (
size_t i = 0;
i < uniforms.size(); ++
i) {
186 Dart_Handle map = ConvertUniformDescriptionToMap(uniform_description);
187 [[maybe_unused]] Dart_Handle dart_result =
188 Dart_ListSetAt(uniform_info,
i, map);
191 if (uniform_description.
type ==
193 sampled_image_count++;
195 other_uniforms_bytes += uniform_description.
GetDartSize();
204 [runtime_stage, snapshot_controller]() {
205 if (!snapshot_controller) {
208 snapshot_controller->CacheRuntimeStage(runtime_stage);
210#if IMPELLER_SUPPORTS_RENDERING
214 const auto& code_mapping = runtime_stage->GetCodeMapping();
215 auto code_size = code_mapping->GetSize();
217 reinterpret_cast<const char*
>(code_mapping->GetMapping());
219 SkRuntimeEffect::Result result =
220 SkRuntimeEffect::MakeForShader(SkString(sksl, code_size));
221 if (result.effect ==
nullptr) {
222 return std::string(
"Invalid SkSL:\n") + sksl +
223 std::string(
"\nSkSL Error:\n") + result.errorText.c_str();
228 Dart_Handle ths = Dart_HandleFromWeakPersistent(
dart_wrapper());
229 if (Dart_IsError(ths)) {
230 Dart_PropagateError(ths);
233 Dart_Handle result = Dart_SetField(ths,
tonic::ToDart(
"_samplerCount"),
234 Dart_NewInteger(sampled_image_count));
235 if (Dart_IsError(result)) {
236 return "Failed to set sampler count for fragment program.";
239 size_t rounded_uniform_bytes =
240 (other_uniforms_bytes +
sizeof(float) - 1) & ~(
sizeof(float) - 1);
241 size_t float_count = rounded_uniform_bytes /
sizeof(float);
243 result = Dart_SetField(ths,
tonic::ToDart(
"_uniformFloatCount"),
244 Dart_NewInteger(float_count));
245 if (Dart_IsError(result)) {
246 return "Failed to set uniform float count for fragment program.";
249 result = Dart_SetField(ths,
tonic::ToDart(
"_uniformInfo"), uniform_info);
250 if (Dart_IsError(result)) {
251 FML_DLOG(ERROR) << Dart_GetError(result);
252 return "Failed to set uniform info for fragment program.";
259 std::shared_ptr<std::vector<uint8_t>> float_uniforms,
260 const std::vector<std::shared_ptr<DlColorSource>>& children) {
262 std::move(float_uniforms));
266 std::shared_ptr<std::vector<uint8_t>> float_uniforms,
267 const std::vector<std::shared_ptr<DlColorSource>>& children) {
269 std::move(float_uniforms));
273 auto res = fml::MakeRefCounted<FragmentProgram>();
274 res->AssociateWithDartWrapper(wrapper);
277FragmentProgram::FragmentProgram() =
default;
static std::shared_ptr< DlColorSource > MakeRuntimeEffect(sk_sp< DlRuntimeEffect > runtime_effect, std::vector< std::shared_ptr< DlColorSource > > samplers, std::shared_ptr< std::vector< uint8_t > > uniform_data)
static std::shared_ptr< DlImageFilter > MakeRuntimeEffect(sk_sp< DlRuntimeEffect > runtime_effect, std::vector< std::shared_ptr< DlColorSource > > samplers, std::shared_ptr< std::vector< uint8_t > > uniform_data)
static sk_sp< DlRuntimeEffect > Make(std::shared_ptr< impeller::RuntimeStage > runtime_stage)
static sk_sp< DlRuntimeEffect > Make(const sk_sp< SkRuntimeEffect > &runtime_effect)
std::shared_ptr< DlImageFilter > MakeDlImageFilter(std::shared_ptr< std::vector< uint8_t > > float_uniforms, const std::vector< std::shared_ptr< DlColorSource > > &children)
std::shared_ptr< DlColorSource > MakeDlColorSource(std::shared_ptr< std::vector< uint8_t > > float_uniforms, const std::vector< std::shared_ptr< DlColorSource > > &children)
~FragmentProgram() override
std::string initFromAsset(const std::string &asset_name)
static void Create(Dart_Handle wrapper)
fml::RefPtr< fml::TaskRunner > GetRasterTaskRunner() const
PlatformConfiguration * platform_configuration() const
const TaskRunners & GetTaskRunners() const
fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > GetSnapshotDelegate() const
impeller::RuntimeStageBackend GetRuntimeStageBackend() const
The runtime stage to use for fragment shaders.
static UIDartState * Current()
virtual void PostTask(const fml::closure &task) override
static absl::StatusOr< Map > DecodeRuntimeStages(const std::shared_ptr< fml::Mapping > &payload)
Dart_WeakPersistentHandle dart_wrapper() const
#define IMPLEMENT_WRAPPERTYPEINFO(LibraryName, ClassName)
#define FML_DLOG(severity)
#define FML_DCHECK(condition)
it will be possible to load the file into Perfetto s trace viewer use test Running tests that layout and measure text will not yield consistent results across various platforms Enabling this option will make font resolution default to the Ahem test font on all disable asset Prevents usage of any non test fonts unless they were explicitly Loaded via prefetched default font Indicates whether the embedding started a prefetch of the default font manager before creating the engine run In non interactive keep the shell running after the Dart script has completed enable serial On low power devices with low core running concurrent GC tasks on threads can cause them to contend with the UI thread which could potentially lead to jank This option turns off all concurrent GC activities domain network JSON encoded network policy per domain This overrides the DisallowInsecureConnections switch Embedder can specify whether to allow or disallow insecure connections at a domain level old gen heap size
DEF_SWITCHES_START aot vmservice shared library name
DEF_SWITCHES_START aot vmservice shared library Name of the *so containing AOT compiled Dart assets for launching the service isolate vm snapshot data
static std::string RuntimeStageBackendToString(impeller::RuntimeStageBackend backend)
Dart_Handle ToDart(const T &object)
#define FML_TRACE_EVENT(category_group, name,...)