Flutter Engine
The Flutter Engine
Loading...
Searching...
No Matches
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
46
48 double height,
49 LayerTreeBuilder builder) {
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
154void ShellTest::SetViewportMetrics(Shell* shell, double width, double height) {
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) {
240 SkMatrix identity;
241 identity.setIdentity();
242 auto root_layer = std::make_shared<TransformLayer>(identity);
243 auto layer_tree = std::make_unique<LayerTree>(
244 LayerTree::Config{.root_layer = 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) {
262 auto packet = std::make_unique<PointerDataPacket>(1);
263 DispatchPointerData(shell, std::move(packet));
264}
265
266void ShellTest::DispatchPointerData(Shell* shell,
267 std::unique_ptr<PointerDataPacket> packet) {
269 shell->GetTaskRunners().GetPlatformTaskRunner()->PostTask(
270 [&latch, shell, &packet]() {
271 // Goes through PlatformView to ensure packet is corrected converted.
272 shell->GetPlatformView()->DispatchPointerDataPacket(std::move(packet));
273 latch.Signal();
274 });
275 latch.Wait();
276}
277
278int ShellTest::UnreportedTimingsCount(Shell* shell) {
279 return shell->unreported_timings_.size();
280}
281
282void ShellTest::SetNeedsReportTimings(Shell* shell, bool value) {
283 shell->SetNeedsReportTimings(value);
284}
285
286bool ShellTest::GetNeedsReportTimings(Shell* shell) {
287 return shell->needs_report_timings_;
288}
289
290void ShellTest::StorePersistentCache(PersistentCache* cache,
291 const SkData& key,
292 const SkData& value) {
293 cache->store(key, value);
294}
295
296void ShellTest::OnServiceProtocol(
297 Shell* shell,
298 ServiceProtocolEnum some_protocol,
299 const fml::RefPtr<fml::TaskRunner>& task_runner,
301 rapidjson::Document* response) {
302 std::promise<bool> finished;
303 fml::TaskRunner::RunNowOrPostTask(task_runner, [shell, some_protocol, params,
304 response, &finished]() {
305 switch (some_protocol) {
306 case ServiceProtocolEnum::kGetSkSLs:
307 shell->OnServiceProtocolGetSkSLs(params, response);
308 break;
309 case ServiceProtocolEnum::kEstimateRasterCacheMemory:
310 shell->OnServiceProtocolEstimateRasterCacheMemory(params, response);
311 break;
312 case ServiceProtocolEnum::kSetAssetBundlePath:
313 shell->OnServiceProtocolSetAssetBundlePath(params, response);
314 break;
315 case ServiceProtocolEnum::kRunInView:
316 shell->OnServiceProtocolRunInView(params, response);
317 break;
318 case ServiceProtocolEnum::kRenderFrameWithRasterStats:
319 shell->OnServiceProtocolRenderFrameWithRasterStats(params, response);
320 break;
321 }
322 finished.set_value(true);
323 });
324 finished.get_future().wait();
325}
326
327std::shared_ptr<txt::FontCollection> ShellTest::GetFontCollection(
328 Shell* shell) {
329 return shell->weak_engine_->GetFontCollection().GetFontCollection();
330}
331
332Settings ShellTest::CreateSettingsForFixture() {
333 Settings settings;
334 settings.leak_vm = false;
335 settings.task_observer_add = [](intptr_t key, const fml::closure& handler) {
337 };
338 settings.task_observer_remove = [](intptr_t key) {
340 };
341 settings.isolate_create_callback = [this]() {
342 native_resolver_->SetNativeResolverForIsolate();
343 };
344#if OS_FUCHSIA
345 settings.verbose_logging = true;
346#endif
347 SetSnapshotsAndAssets(settings);
348 return settings;
349}
350
351TaskRunners ShellTest::GetTaskRunnersForFixture() {
352 return {
353 "test",
354 thread_host_.platform_thread->GetTaskRunner(), // platform
355 thread_host_.raster_thread->GetTaskRunner(), // raster
356 thread_host_.ui_thread->GetTaskRunner(), // ui
357 thread_host_.io_thread->GetTaskRunner() // io
358 };
359}
360
361fml::TimePoint ShellTest::GetLatestFrameTargetTime(Shell* shell) const {
362 return shell->GetLatestFrameTargetTime();
363}
364
365std::unique_ptr<Shell> ShellTest::CreateShell(
366 const Settings& settings,
367 std::optional<TaskRunners> task_runners) {
368 return CreateShell({
369 .settings = settings,
370 .task_runners = std::move(task_runners),
371 });
372}
373
374std::unique_ptr<Shell> ShellTest::CreateShell(const Config& config) {
375 TaskRunners task_runners = config.task_runners.has_value()
376 ? config.task_runners.value()
378 Shell::CreateCallback<PlatformView> platform_view_create_callback =
380 if (!platform_view_create_callback) {
381 platform_view_create_callback = ShellTestPlatformViewBuilder({});
382 }
383
384 Shell::CreateCallback<Rasterizer> rasterizer_create_callback =
385 [](Shell& shell) { return std::make_unique<Rasterizer>(shell); };
386
387 return Shell::Create(flutter::PlatformData(), //
388 task_runners, //
389 config.settings, //
390 platform_view_create_callback, //
391 rasterizer_create_callback, //
392 config.is_gpu_disabled //
393 );
394}
395
396void ShellTest::DestroyShell(std::unique_ptr<Shell> shell) {
397 DestroyShell(std::move(shell), GetTaskRunnersForFixture());
398}
399
400void ShellTest::DestroyShell(std::unique_ptr<Shell> shell,
401 const TaskRunners& task_runners) {
404 [&shell, &latch]() mutable {
405 shell.reset();
406 latch.Signal();
407 });
408 latch.Wait();
409}
410
411size_t ShellTest::GetLiveTrackedPathCount(
412 const std::shared_ptr<VolatilePathTracker>& tracker) {
413 return std::count_if(
414 tracker->paths_.begin(), tracker->paths_.end(),
415 [](const std::weak_ptr<VolatilePathTracker::TrackedPath>& path) {
416 return path.lock();
417 });
418}
419
420void ShellTest::TurnOffGPU(Shell* shell, bool value) {
421 shell->is_gpu_disabled_sync_switch_->SetSwitch(value);
422}
423
424} // namespace testing
425} // namespace flutter
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:119
fml::RefPtr< fml::TaskRunner > GetPlatformTaskRunner() const
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)
void AddTaskObserver(intptr_t key, const fml::closure &callback)
static FML_EMBEDDER_ONLY MessageLoop & GetCurrent()
static void RunNowOrPostTask(const fml::RefPtr< fml::TaskRunner > &runner, const fml::closure &task)
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
@ kRaster
Suitable for thread which raster data.
Definition embedder.h:264
const EmbeddedViewParams * params
ThreadHost thread_host_
FlutterEngine engine
Definition main.cc:68
uint8_t value
GAsyncResult * result
Win32Message message
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
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
internal::CopyableLambda< T > MakeCopyable(T lambda)
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
std::shared_ptr< Layer > root_layer
Definition layer_tree.h:25
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
flutter::ViewportMetrics viewport_metrics
Definition shell_test.h:40
static FrameContent NoViews()
Definition shell_test.cc:25