Flutter Engine
The Flutter Engine
shell_test.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#define FML_USED_ON_EMBEDDER
6
7#include "flutter/shell/common/shell_test.h"
8
9#include "flutter/flow/frame_timings.h"
10#include "flutter/flow/layers/layer_tree.h"
11#include "flutter/flow/layers/transform_layer.h"
12#include "flutter/fml/build_config.h"
13#include "flutter/fml/make_copyable.h"
14#include "flutter/fml/mapping.h"
15#include "flutter/runtime/dart_vm.h"
16#include "flutter/shell/common/shell_test_platform_view.h"
17#include "flutter/shell/common/vsync_waiter_fallback.h"
18#include "flutter/testing/testing.h"
19
20namespace flutter {
21namespace testing {
22
23constexpr int64_t kImplicitViewId = 0;
24
26 return std::map<int64_t, ViewContent>();
27}
28
32 .viewport_metrics = {1.0, width, height, 22, 0},
33 .builder = {},
34 };
35 return result;
36}
37
42 .builder = {},
43 };
44 return result;
45}
46
48 double height,
52 .viewport_metrics = {1.0, width, height, 22, 0},
53 .builder = std::move(builder),
54 };
55 return result;
56}
57
59 : thread_host_("io.flutter.test." + GetCurrentTestName() + ".",
60 ThreadHost::Type::kPlatform | ThreadHost::Type::kIo |
62
64 std::unique_ptr<PlatformMessage> message) {
65 shell->OnPlatformViewDispatchPlatformMessage(std::move(message));
66}
67
69 Shell* shell,
70 std::unique_ptr<PlatformMessage> message) {
73 shell->GetTaskRunners().GetPlatformTaskRunner(),
75 [shell, &latch, message = std::move(message)]() mutable {
76 if (auto engine = shell->weak_engine_) {
77 engine->HandlePlatformMessage(std::move(message));
78 }
79 latch.Signal();
80 }));
81 latch.Wait();
82}
83
84void ShellTest::PlatformViewNotifyCreated(Shell* shell) {
87 shell->GetTaskRunners().GetPlatformTaskRunner(), [shell, &latch]() {
88 shell->GetPlatformView()->NotifyCreated();
89 latch.Signal();
90 });
91 latch.Wait();
92}
93
94void ShellTest::PlatformViewNotifyDestroyed(Shell* shell) {
97 shell->GetTaskRunners().GetPlatformTaskRunner(), [shell, &latch]() {
98 shell->GetPlatformView()->NotifyDestroyed();
99 latch.Signal();
100 });
101 latch.Wait();
102}
103
104void ShellTest::RunEngine(Shell* shell, RunConfiguration configuration) {
107 shell->GetTaskRunners().GetPlatformTaskRunner(),
108 [shell, &latch, &configuration]() {
109 shell->RunEngine(std::move(configuration),
110 [&latch](Engine::RunStatus run_status) {
111 ASSERT_EQ(run_status, Engine::RunStatus::Success);
112 latch.Signal();
113 });
114 });
115 latch.Wait();
116}
117
118void ShellTest::RestartEngine(Shell* shell, RunConfiguration configuration) {
119 std::promise<bool> restarted;
121 shell->GetTaskRunners().GetUITaskRunner(),
122 [shell, &restarted, &configuration]() {
123 restarted.set_value(shell->engine_->Restart(std::move(configuration)));
124 });
125 ASSERT_TRUE(restarted.get_future().get());
126}
127
128void ShellTest::VSyncFlush(Shell* shell, bool* will_draw_new_frame) {
131 shell->GetTaskRunners().GetPlatformTaskRunner(),
132 [shell, will_draw_new_frame, &latch] {
133 // The following UI task ensures that all previous UI tasks are flushed.
134 fml::AutoResetWaitableEvent ui_latch;
135 shell->GetTaskRunners().GetUITaskRunner()->PostTask(
136 [&ui_latch, will_draw_new_frame]() {
137 if (will_draw_new_frame != nullptr) {
138 *will_draw_new_frame = true;
139 }
140 ui_latch.Signal();
141 });
142
143 ShellTestPlatformView* test_platform_view =
144 static_cast<ShellTestPlatformView*>(shell->GetPlatformView().get());
145 do {
146 test_platform_view->SimulateVSync();
147 } while (ui_latch.WaitWithTimeout(fml::TimeDelta::FromMilliseconds(1)));
148
149 latch.Signal();
150 });
151 latch.Wait();
152}
153
155 flutter::ViewportMetrics viewport_metrics = {
156 1, // device pixel ratio
157 width, // physical width
158 height, // physical height
159 0, // padding top
160 0, // padding right
161 0, // padding bottom
162 0, // padding left
163 0, // view inset top
164 0, // view inset right
165 0, // view inset bottom
166 0, // view inset left
167 0, // gesture inset top
168 0, // gesture inset right
169 0, // gesture inset bottom
170 0, // gesture inset left
171 22, // physical touch slop
172 std::vector<double>(), // display features bounds
173 std::vector<int>(), // display features type
174 std::vector<int>(), // display features state
175 0 // Display ID
176 };
177 // Set viewport to nonempty, and call Animator::BeginFrame to make the layer
178 // tree pipeline nonempty. Without either of this, the layer tree below
179 // won't be rasterized.
181 shell->GetTaskRunners().GetUITaskRunner()->PostTask(
182 [&latch, engine = shell->weak_engine_, viewport_metrics]() {
183 if (engine) {
184 engine->SetViewportMetrics(kImplicitViewId, viewport_metrics);
185 const auto frame_begin_time = fml::TimePoint::Now();
186 const auto frame_end_time =
187 frame_begin_time + fml::TimeDelta::FromSecondsF(1.0 / 60.0);
188 std::unique_ptr<FrameTimingsRecorder> recorder =
189 std::make_unique<FrameTimingsRecorder>();
190 recorder->RecordVsync(frame_begin_time, frame_end_time);
191 engine->animator_->BeginFrame(std::move(recorder));
192 engine->animator_->EndFrame();
193 }
194 latch.Signal();
195 });
196 latch.Wait();
197}
198
199void ShellTest::NotifyIdle(Shell* shell, fml::TimeDelta deadline) {
201 shell->GetTaskRunners().GetUITaskRunner()->PostTask(
202 [&latch, engine = shell->weak_engine_, deadline]() {
203 if (engine) {
204 engine->NotifyIdle(deadline);
205 }
206 latch.Signal();
207 });
208 latch.Wait();
209}
210
211void ShellTest::PumpOneFrame(Shell* shell) {
212 PumpOneFrame(shell, ViewContent::DummyView());
213}
214
215void ShellTest::PumpOneFrame(Shell* shell, FrameContent frame_content) {
216 // Set viewport to nonempty, and call Animator::BeginFrame to make the layer
217 // tree pipeline nonempty. Without either of this, the layer tree below
218 // won't be rasterized.
220 fml::WeakPtr<RuntimeDelegate> runtime_delegate = shell->weak_engine_;
221 shell->GetTaskRunners().GetUITaskRunner()->PostTask(
222 [&latch, engine = shell->weak_engine_, &frame_content,
223 runtime_delegate]() {
224 for (auto& [view_id, view_content] : frame_content) {
225 engine->SetViewportMetrics(view_id, view_content.viewport_metrics);
226 }
227 const auto frame_begin_time = fml::TimePoint::Now();
228 const auto frame_end_time =
229 frame_begin_time + fml::TimeDelta::FromSecondsF(1.0 / 60.0);
230 std::unique_ptr<FrameTimingsRecorder> recorder =
231 std::make_unique<FrameTimingsRecorder>();
232 recorder->RecordVsync(frame_begin_time, frame_end_time);
233 engine->animator_->BeginFrame(std::move(recorder));
234
235 // The BeginFrame phase and the EndFrame phase must be performed in a
236 // single task, otherwise a normal vsync might be inserted in between,
237 // causing flaky assertion errors.
238
239 for (auto& [view_id, view_content] : frame_content) {
241 identity.setIdentity();
242 auto root_layer = std::make_shared<TransformLayer>(identity);
243 auto layer_tree = std::make_unique<LayerTree>(
244 root_layer,
245 SkISize::Make(view_content.viewport_metrics.physical_width,
246 view_content.viewport_metrics.physical_height));
247 float device_pixel_ratio = static_cast<float>(
248 view_content.viewport_metrics.device_pixel_ratio);
249 if (view_content.builder) {
250 view_content.builder(root_layer);
251 }
252 runtime_delegate->Render(view_id, std::move(layer_tree),
253 device_pixel_ratio);
254 }
255 engine->animator_->EndFrame();
256 latch.Signal();
257 });
258 latch.Wait();
259}
260
261void ShellTest::DispatchFakePointerData(Shell* shell, double x) {
262 auto packet = std::make_unique<PointerDataPacket>(1);
263 packet->SetPointerData(0, PointerData{
265 .physical_x = x,
266 });
267 DispatchPointerData(shell, std::move(packet));
268}
269
270void ShellTest::DispatchPointerData(Shell* shell,
271 std::unique_ptr<PointerDataPacket> packet) {
273 shell->GetTaskRunners().GetPlatformTaskRunner()->PostTask(
274 [&latch, shell, &packet]() {
275 // Goes through PlatformView to ensure packet is corrected converted.
276 shell->GetPlatformView()->DispatchPointerDataPacket(std::move(packet));
277 latch.Signal();
278 });
279 latch.Wait();
280}
281
282int ShellTest::UnreportedTimingsCount(Shell* shell) {
283 return shell->unreported_timings_.size();
284}
285
286void ShellTest::SetNeedsReportTimings(Shell* shell, bool value) {
287 shell->SetNeedsReportTimings(value);
288}
289
290bool ShellTest::GetNeedsReportTimings(Shell* shell) {
291 return shell->needs_report_timings_;
292}
293
294void ShellTest::StorePersistentCache(PersistentCache* cache,
295 const SkData& key,
296 const SkData& value) {
297 cache->store(key, value);
298}
299
300void ShellTest::OnServiceProtocol(
301 Shell* shell,
302 ServiceProtocolEnum some_protocol,
303 const fml::RefPtr<fml::TaskRunner>& task_runner,
305 rapidjson::Document* response) {
306 std::promise<bool> finished;
307 fml::TaskRunner::RunNowOrPostTask(task_runner, [shell, some_protocol, params,
308 response, &finished]() {
309 switch (some_protocol) {
310 case ServiceProtocolEnum::kGetSkSLs:
311 shell->OnServiceProtocolGetSkSLs(params, response);
312 break;
313 case ServiceProtocolEnum::kEstimateRasterCacheMemory:
314 shell->OnServiceProtocolEstimateRasterCacheMemory(params, response);
315 break;
316 case ServiceProtocolEnum::kSetAssetBundlePath:
317 shell->OnServiceProtocolSetAssetBundlePath(params, response);
318 break;
319 case ServiceProtocolEnum::kRunInView:
320 shell->OnServiceProtocolRunInView(params, response);
321 break;
322 case ServiceProtocolEnum::kRenderFrameWithRasterStats:
323 shell->OnServiceProtocolRenderFrameWithRasterStats(params, response);
324 break;
325 }
326 finished.set_value(true);
327 });
328 finished.get_future().wait();
329}
330
331std::shared_ptr<txt::FontCollection> ShellTest::GetFontCollection(
332 Shell* shell) {
333 return shell->weak_engine_->GetFontCollection().GetFontCollection();
334}
335
336Settings ShellTest::CreateSettingsForFixture() {
338 settings.leak_vm = false;
339 settings.task_observer_add = [](intptr_t key, const fml::closure& handler) {
341 };
342 settings.task_observer_remove = [](intptr_t key) {
344 };
345 settings.isolate_create_callback = [this]() {
346 native_resolver_->SetNativeResolverForIsolate();
347 };
348#if OS_FUCHSIA
349 settings.verbose_logging = true;
350#endif
351 SetSnapshotsAndAssets(settings);
352 return settings;
353}
354
356 return {
357 "test",
358 thread_host_.platform_thread->GetTaskRunner(), // platform
359 thread_host_.raster_thread->GetTaskRunner(), // raster
360 thread_host_.ui_thread->GetTaskRunner(), // ui
361 thread_host_.io_thread->GetTaskRunner() // io
362 };
363}
364
365fml::TimePoint ShellTest::GetLatestFrameTargetTime(Shell* shell) const {
366 return shell->GetLatestFrameTargetTime();
367}
368
369std::unique_ptr<Shell> ShellTest::CreateShell(
370 const Settings& settings,
371 std::optional<TaskRunners> task_runners) {
372 return CreateShell({
373 .settings = settings,
374 .task_runners = std::move(task_runners),
375 });
376}
377
378std::unique_ptr<Shell> ShellTest::CreateShell(const Config& config) {
379 TaskRunners task_runners = config.task_runners.has_value()
380 ? config.task_runners.value()
382 Shell::CreateCallback<PlatformView> platform_view_create_callback =
384 if (!platform_view_create_callback) {
385 platform_view_create_callback = ShellTestPlatformViewBuilder({});
386 }
387
388 Shell::CreateCallback<Rasterizer> rasterizer_create_callback =
389 [](Shell& shell) { return std::make_unique<Rasterizer>(shell); };
390
392 task_runners, //
393 config.settings, //
394 platform_view_create_callback, //
395 rasterizer_create_callback, //
396 config.is_gpu_disabled //
397 );
398}
399
400void ShellTest::DestroyShell(std::unique_ptr<Shell> shell) {
401 DestroyShell(std::move(shell), GetTaskRunnersForFixture());
402}
403
404void ShellTest::DestroyShell(std::unique_ptr<Shell> shell,
405 const TaskRunners& task_runners) {
408 [&shell, &latch]() mutable {
409 shell.reset();
410 latch.Signal();
411 });
412 latch.Wait();
413}
414
415size_t ShellTest::GetLiveTrackedPathCount(
416 const std::shared_ptr<VolatilePathTracker>& tracker) {
417 return std::count_if(
418 tracker->paths_.begin(), tracker->paths_.end(),
419 [](const std::weak_ptr<VolatilePathTracker::TrackedPath>& path) {
420 return path.lock();
421 });
422}
423
424void ShellTest::TurnOffGPU(Shell* shell, bool value) {
425 shell->is_gpu_disabled_sync_switch_->SetSwitch(value);
426}
427
428} // namespace testing
429} // namespace flutter
for(const auto glyph :glyphs)
Definition: FontMgrTest.cpp:52
static sk_sp< Effect > Create()
Definition: RefCntTest.cpp:117
Definition: SkData.h:25
Specifies all the configuration required by the runtime library to launch the root isolate....
std::map< std::string_view, std::string_view > ServiceProtocolMap
std::function< std::unique_ptr< T >(Shell &)> CreateCallback
Definition: shell.h:120
fml::RefPtr< fml::TaskRunner > GetPlatformTaskRunner() const
Definition: task_runners.cc:30
void SendPlatformMessage(Shell *shell, std::unique_ptr< PlatformMessage > message)
Definition: shell_test.cc:63
void SendEnginePlatformMessage(Shell *shell, std::unique_ptr< PlatformMessage > message)
Definition: shell_test.cc:68
void RemoveTaskObserver(intptr_t key)
Definition: message_loop.cc:68
void AddTaskObserver(intptr_t key, const fml::closure &callback)
Definition: message_loop.cc:64
static FML_EMBEDDER_ONLY MessageLoop & GetCurrent()
Definition: message_loop.cc:19
static void RunNowOrPostTask(const fml::RefPtr< fml::TaskRunner > &runner, const fml::closure &task)
Definition: task_runner.cc:55
static constexpr TimeDelta FromSecondsF(double seconds)
Definition: time_delta.h:53
static constexpr TimeDelta FromMilliseconds(int64_t millis)
Definition: time_delta.h:46
static TimePoint Now()
Definition: time_point.cc:49
@ kHover
The pointer moved while up.
Definition: embedder.h:997
@ kRaster
Suitable for thread which raster data.
Definition: embedder.h:266
const EmbeddedViewParams * params
ThreadHost thread_host_
FlutterEngine engine
Definition: main.cc:68
uint8_t value
GAsyncResult * result
Win32Message message
double x
std::string GetCurrentTestName()
Gets the name of the currently running test. This is useful in generating logs or assets based on tes...
Definition: testing.cc:15
std::function< void(std::shared_ptr< ContainerLayer > root)> LayerTreeBuilder
Definition: shell_test.h:34
std::map< int64_t, ViewContent > FrameContent
Definition: shell_test.h:37
constexpr int64_t kImplicitViewId
static void DispatchPointerDataPacket(JNIEnv *env, jobject jcaller, jlong shell_holder, jobject buffer, jint position)
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 cache
Definition: switches.h:191
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
Definition: switches.h:57
static void SetViewportMetrics(JNIEnv *env, jobject jcaller, jlong shell_holder, jfloat devicePixelRatio, jint physicalWidth, jint physicalHeight, jint physicalPaddingTop, jint physicalPaddingRight, jint physicalPaddingBottom, jint physicalPaddingLeft, jint physicalViewInsetTop, jint physicalViewInsetRight, jint physicalViewInsetBottom, jint physicalViewInsetLeft, jint systemGestureInsetTop, jint systemGestureInsetRight, jint systemGestureInsetBottom, jint systemGestureInsetLeft, jint physicalTouchSlop, jintArray javaDisplayFeaturesBounds, jintArray javaDisplayFeaturesType, jintArray javaDisplayFeaturesState)
internal::CopyableLambda< T > MakeCopyable(T lambda)
Definition: make_copyable.h:57
std::function< void()> closure
Definition: closure.h:14
int32_t height
int32_t width
static constexpr SkISize Make(int32_t w, int32_t h)
Definition: SkSize.h:20
The collection of all the threads used by the engine.
Definition: thread_host.h:21
Shell::CreateCallback< PlatformView > platform_view_create_callback
Definition: shell_test.h:74
std::optional< TaskRunners > task_runners
Definition: shell_test.h:70
static FrameContent ImplicitView(double width, double height, LayerTreeBuilder builder)
Definition: shell_test.cc:47
static FrameContent DummyView(double width=1, double height=1)
Definition: shell_test.cc:29
LayerTreeBuilder builder
Definition: shell_test.h:43
flutter::ViewportMetrics viewport_metrics
Definition: shell_test.h:40
static FrameContent NoViews()
Definition: shell_test.cc:25