dart_isolate_unittests.cc 15.2 KB
Newer Older
M
Michael Goderbauer 已提交
1
// Copyright 2013 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5
#include "flutter/fml/make_copyable.h"
6 7
#include "flutter/fml/mapping.h"
#include "flutter/fml/paths.h"
8
#include "flutter/fml/synchronization/count_down_latch.h"
9
#include "flutter/fml/synchronization/waitable_event.h"
10 11 12
#include "flutter/fml/thread.h"
#include "flutter/runtime/dart_isolate.h"
#include "flutter/runtime/dart_vm.h"
13
#include "flutter/runtime/dart_vm_lifecycle.h"
14
#include "flutter/runtime/runtime_test.h"
15 16
#include "flutter/testing/testing.h"
#include "flutter/testing/thread_test.h"
17
#include "third_party/tonic/converter/dart_converter.h"
18
#include "third_party/tonic/scopes/dart_isolate_scope.h"
19

20
namespace flutter {
21
namespace testing {
22

23
using DartIsolateTest = RuntimeTest;
24 25

TEST_F(DartIsolateTest, RootIsolateCreationAndShutdown) {
26
  ASSERT_FALSE(DartVMRef::IsInstanceRunning());
27
  auto settings = CreateSettingsForFixture();
28 29 30 31
  auto vm_ref = DartVMRef::Create(settings);
  ASSERT_TRUE(vm_ref);
  auto vm_data = vm_ref.GetVMData();
  ASSERT_TRUE(vm_data);
32 33 34 35 36
  TaskRunners task_runners(GetCurrentTestName(),    //
                           GetCurrentTaskRunner(),  //
                           GetCurrentTaskRunner(),  //
                           GetCurrentTaskRunner(),  //
                           GetCurrentTaskRunner()   //
37
  );
38
  auto weak_isolate = DartIsolate::CreateRootIsolate(
39 40 41 42
      vm_data->GetSettings(),             // settings
      vm_data->GetIsolateSnapshot(),      // isolate snapshot
      std::move(task_runners),            // task runners
      nullptr,                            // window
43
      {},                                 // snapshot delegate
44
      {},                                 // io manager
45
      {},                                 // unref queue
46
      {},                                 // image decoder
47 48 49 50 51
      "main.dart",                        // advisory uri
      "main",                             // advisory entrypoint,
      nullptr,                            // flags
      settings.isolate_create_callback,   // isolate create callback
      settings.isolate_shutdown_callback  // isolate shutdown callback
52
  );
53
  auto root_isolate = weak_isolate.lock();
54 55 56 57 58
  ASSERT_TRUE(root_isolate);
  ASSERT_EQ(root_isolate->GetPhase(), DartIsolate::Phase::LibrariesSetup);
  ASSERT_TRUE(root_isolate->Shutdown());
}

59
TEST_F(DartIsolateTest, IsolateShutdownCallbackIsInIsolateScope) {
60
  ASSERT_FALSE(DartVMRef::IsInstanceRunning());
61
  auto settings = CreateSettingsForFixture();
62 63 64 65
  auto vm_ref = DartVMRef::Create(settings);
  ASSERT_TRUE(vm_ref);
  auto vm_data = vm_ref.GetVMData();
  ASSERT_TRUE(vm_data);
66 67 68 69 70
  TaskRunners task_runners(GetCurrentTestName(),    //
                           GetCurrentTaskRunner(),  //
                           GetCurrentTaskRunner(),  //
                           GetCurrentTaskRunner(),  //
                           GetCurrentTaskRunner()   //
71 72
  );
  auto weak_isolate = DartIsolate::CreateRootIsolate(
73 74 75 76
      vm_data->GetSettings(),             // settings
      vm_data->GetIsolateSnapshot(),      // isolate snapshot
      std::move(task_runners),            // task runners
      nullptr,                            // window
77
      {},                                 // snapshot delegate
78
      {},                                 // io manager
79
      {},                                 // unref queue
80
      {},                                 // image decoder
81 82 83 84 85
      "main.dart",                        // advisory uri
      "main",                             // advisory entrypoint
      nullptr,                            // flags
      settings.isolate_create_callback,   // isolate create callback
      settings.isolate_shutdown_callback  // isolate shutdown callback
86 87 88 89 90 91 92 93 94 95 96 97 98
  );
  auto root_isolate = weak_isolate.lock();
  ASSERT_TRUE(root_isolate);
  ASSERT_EQ(root_isolate->GetPhase(), DartIsolate::Phase::LibrariesSetup);
  size_t destruction_callback_count = 0;
  root_isolate->AddIsolateShutdownCallback([&destruction_callback_count]() {
    ASSERT_NE(Dart_CurrentIsolate(), nullptr);
    destruction_callback_count++;
  });
  ASSERT_TRUE(root_isolate->Shutdown());
  ASSERT_EQ(destruction_callback_count, 1u);
}

99 100
class AutoIsolateShutdown {
 public:
101 102
  AutoIsolateShutdown() = default;

103
  AutoIsolateShutdown(std::shared_ptr<DartIsolate> isolate,
104 105
                      fml::RefPtr<fml::TaskRunner> runner)
      : isolate_(std::move(isolate)), runner_(std::move(runner)) {}
106 107

  ~AutoIsolateShutdown() {
108 109 110 111
    if (!IsValid()) {
      return;
    }
    fml::AutoResetWaitableEvent latch;
112 113 114 115 116 117 118 119
    fml::TaskRunner::RunNowOrPostTask(
        runner_, [isolate = std::move(isolate_), &latch]() {
          if (!isolate->Shutdown()) {
            FML_LOG(ERROR) << "Could not shutdown isolate.";
            FML_CHECK(false);
          }
          latch.Signal();
        });
120
    latch.Wait();
121 122
  }

123
  bool IsValid() const { return isolate_ != nullptr && runner_; }
124

125 126
  FML_WARN_UNUSED_RESULT
  bool RunInIsolateScope(std::function<bool(void)> closure) {
127
    if (!IsValid()) {
128 129
      return false;
    }
130 131 132 133 134 135 136 137 138 139 140 141 142

    bool result = false;
    fml::AutoResetWaitableEvent latch;
    fml::TaskRunner::RunNowOrPostTask(
        runner_, [this, &result, &latch, closure]() {
          tonic::DartIsolateScope scope(isolate_->isolate());
          tonic::DartApiScope api_scope;
          if (closure) {
            result = closure();
          }
          latch.Signal();
        });
    latch.Wait();
143 144 145
    return true;
  }

146
  DartIsolate* get() {
147 148 149 150 151
    FML_CHECK(isolate_);
    return isolate_.get();
  }

 private:
152
  std::shared_ptr<DartIsolate> isolate_;
153
  fml::RefPtr<fml::TaskRunner> runner_;
154 155 156 157

  FML_DISALLOW_COPY_AND_ASSIGN(AutoIsolateShutdown);
};

158 159
static void RunDartCodeInIsolate(DartVMRef& vm_ref,
                                 std::unique_ptr<AutoIsolateShutdown>& result,
160 161
                                 const Settings& settings,
                                 fml::RefPtr<fml::TaskRunner> task_runner,
162
                                 std::string entrypoint,
163
                                 const std::vector<std::string>& args) {
164
  FML_CHECK(task_runner->RunsTasksOnCurrentThread());
165

166
  if (!vm_ref) {
167
    return;
168
  }
169

170 171 172 173 174
  TaskRunners task_runners(GetCurrentTestName(),  //
                           task_runner,           //
                           task_runner,           //
                           task_runner,           //
                           task_runner            //
175 176
  );

177 178 179 180 181 182
  auto vm_data = vm_ref.GetVMData();

  if (!vm_data) {
    return;
  }

183
  auto weak_isolate = DartIsolate::CreateRootIsolate(
184 185 186 187
      vm_data->GetSettings(),             // settings
      vm_data->GetIsolateSnapshot(),      // isolate snapshot
      std::move(task_runners),            // task runners
      nullptr,                            // window
188
      {},                                 // snapshot delegate
189
      {},                                 // io manager
190
      {},                                 // unref queue
191
      {},                                 // image decoder
192 193 194 195 196
      "main.dart",                        // advisory uri
      "main",                             // advisory entrypoint
      nullptr,                            // flags
      settings.isolate_create_callback,   // isolate create callback
      settings.isolate_shutdown_callback  // isolate shutdown callback
197 198
  );

199 200
  auto root_isolate =
      std::make_unique<AutoIsolateShutdown>(weak_isolate.lock(), task_runner);
201 202 203

  if (!root_isolate->IsValid()) {
    FML_LOG(ERROR) << "Could not create isolate.";
204
    return;
205 206 207 208
  }

  if (root_isolate->get()->GetPhase() != DartIsolate::Phase::LibrariesSetup) {
    FML_LOG(ERROR) << "Created isolate is in unexpected phase.";
209
    return;
210 211
  }

212
  if (!DartVM::IsRunningPrecompiledCode()) {
213 214
    auto kernel_file_path =
        fml::paths::JoinPaths({GetFixturesPath(), "kernel_blob.bin"});
215

216 217
    if (!fml::IsFile(kernel_file_path)) {
      FML_LOG(ERROR) << "Could not locate kernel file.";
218
      return;
219
    }
220

221 222
    auto kernel_file = fml::OpenFile(kernel_file_path.c_str(), false,
                                     fml::FilePermission::kRead);
223

224 225
    if (!kernel_file.is_valid()) {
      FML_LOG(ERROR) << "Kernel file descriptor was invalid.";
226
      return;
227
    }
228

229
    auto kernel_mapping = std::make_unique<fml::FileMapping>(kernel_file);
230

231 232
    if (kernel_mapping->GetMapping() == nullptr) {
      FML_LOG(ERROR) << "Could not setup kernel mapping.";
233
      return;
234
    }
235

236 237 238 239
    if (!root_isolate->get()->PrepareForRunningFromKernel(
            std::move(kernel_mapping))) {
      FML_LOG(ERROR)
          << "Could not prepare to run the isolate from the kernel file.";
240
      return;
241 242 243 244 245
    }
  } else {
    if (!root_isolate->get()->PrepareForRunningFromPrecompiledCode()) {
      FML_LOG(ERROR)
          << "Could not prepare to run the isolate from precompiled code.";
246
      return;
247
    }
248 249 250 251
  }

  if (root_isolate->get()->GetPhase() != DartIsolate::Phase::Ready) {
    FML_LOG(ERROR) << "Isolate is in unexpected phase.";
252
    return;
253 254
  }

255
  if (!root_isolate->get()->Run(entrypoint, args,
256
                                settings.root_isolate_create_callback)) {
257 258
    FML_LOG(ERROR) << "Could not run the method \"" << entrypoint
                   << "\" in the isolate.";
259
    return;
260 261
  }

262 263 264 265 266 267 268
  root_isolate->get()->AddIsolateShutdownCallback(
      settings.root_isolate_shutdown_callback);

  result = std::move(root_isolate);
}

static std::unique_ptr<AutoIsolateShutdown> RunDartCodeInIsolate(
269
    DartVMRef& vm_ref,
270 271
    const Settings& settings,
    fml::RefPtr<fml::TaskRunner> task_runner,
272
    std::string entrypoint,
273
    const std::vector<std::string>& args) {
274 275 276 277
  std::unique_ptr<AutoIsolateShutdown> result;
  fml::AutoResetWaitableEvent latch;
  fml::TaskRunner::RunNowOrPostTask(
      task_runner, fml::MakeCopyable([&]() mutable {
278
        RunDartCodeInIsolate(vm_ref, result, settings, task_runner, entrypoint,
279
                             args);
280 281 282 283
        latch.Signal();
      }));
  latch.Wait();
  return result;
284 285 286
}

TEST_F(DartIsolateTest, IsolateCanLoadAndRunDartCode) {
287 288 289
  ASSERT_FALSE(DartVMRef::IsInstanceRunning());
  const auto settings = CreateSettingsForFixture();
  auto vm_ref = DartVMRef::Create(settings);
290
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, GetCurrentTaskRunner(),
291
                                      "main", {});
292 293 294 295 296
  ASSERT_TRUE(isolate);
  ASSERT_EQ(isolate->get()->GetPhase(), DartIsolate::Phase::Running);
}

TEST_F(DartIsolateTest, IsolateCannotLoadAndRunUnknownDartEntrypoint) {
297 298 299 300
  ASSERT_FALSE(DartVMRef::IsInstanceRunning());
  const auto settings = CreateSettingsForFixture();
  auto vm_ref = DartVMRef::Create(settings);
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, GetCurrentTaskRunner(),
301
                                      "thisShouldNotExist", {});
302 303 304
  ASSERT_FALSE(isolate);
}

305
TEST_F(DartIsolateTest, CanRunDartCodeCodeSynchronously) {
306 307 308
  ASSERT_FALSE(DartVMRef::IsInstanceRunning());
  const auto settings = CreateSettingsForFixture();
  auto vm_ref = DartVMRef::Create(settings);
309
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, GetCurrentTaskRunner(),
310
                                      "main", {});
311 312

  ASSERT_TRUE(isolate);
313
  ASSERT_EQ(isolate->get()->GetPhase(), DartIsolate::Phase::Running);
314 315 316 317 318 319 320 321 322
  ASSERT_TRUE(isolate->RunInIsolateScope([]() -> bool {
    if (tonic::LogIfError(::Dart_Invoke(Dart_RootLibrary(),
                                        tonic::ToDart("sayHi"), 0, nullptr))) {
      return false;
    }
    return true;
  }));
}

323
TEST_F(DartIsolateTest, CanRegisterNativeCallback) {
324
  ASSERT_FALSE(DartVMRef::IsInstanceRunning());
325 326 327 328 329 330
  fml::AutoResetWaitableEvent latch;
  AddNativeCallback("NotifyNative",
                    CREATE_NATIVE_ENTRY(([&latch](Dart_NativeArguments args) {
                      FML_LOG(ERROR) << "Hello from Dart!";
                      latch.Signal();
                    })));
331 332
  const auto settings = CreateSettingsForFixture();
  auto vm_ref = DartVMRef::Create(settings);
333
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, CreateNewThread(),
334
                                      "canRegisterNativeCallback", {});
335 336 337 338 339
  ASSERT_TRUE(isolate);
  ASSERT_EQ(isolate->get()->GetPhase(), DartIsolate::Phase::Running);
  latch.Wait();
}

340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355
TEST_F(DartIsolateTest, CanSaveCompilationTrace) {
  if (DartVM::IsRunningPrecompiledCode()) {
    // Can only save compilation traces in JIT modes.
    GTEST_SKIP();
    return;
  }
  fml::AutoResetWaitableEvent latch;
  AddNativeCallback("NotifyNative",
                    CREATE_NATIVE_ENTRY(([&latch](Dart_NativeArguments args) {
                      ASSERT_TRUE(tonic::DartConverter<bool>::FromDart(
                          Dart_GetNativeArgument(args, 0)));
                      latch.Signal();
                    })));

  const auto settings = CreateSettingsForFixture();
  auto vm_ref = DartVMRef::Create(settings);
356
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, CreateNewThread(),
357
                                      "testCanSaveCompilationTrace", {});
358 359 360 361 362 363
  ASSERT_TRUE(isolate);
  ASSERT_EQ(isolate->get()->GetPhase(), DartIsolate::Phase::Running);

  latch.Wait();
}

364 365
TEST_F(DartIsolateTest, CanLaunchSecondaryIsolates) {
  fml::CountDownLatch latch(3);
366
  fml::AutoResetWaitableEvent child_shutdown_latch;
367
  fml::AutoResetWaitableEvent root_isolate_shutdown_latch;
368 369 370 371 372 373 374 375 376 377 378
  AddNativeCallback("NotifyNative",
                    CREATE_NATIVE_ENTRY(([&latch](Dart_NativeArguments args) {
                      latch.CountDown();
                    })));
  AddNativeCallback(
      "PassMessage", CREATE_NATIVE_ENTRY(([&latch](Dart_NativeArguments args) {
        auto message = tonic::DartConverter<std::string>::FromDart(
            Dart_GetNativeArgument(args, 0));
        ASSERT_EQ("Hello from code is secondary isolate.", message);
        latch.CountDown();
      })));
379
  auto settings = CreateSettingsForFixture();
380 381 382
  settings.root_isolate_shutdown_callback = [&root_isolate_shutdown_latch]() {
    root_isolate_shutdown_latch.Signal();
  };
383 384 385
  settings.isolate_shutdown_callback = [&child_shutdown_latch]() {
    child_shutdown_latch.Signal();
  };
386
  auto vm_ref = DartVMRef::Create(settings);
387 388
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, CreateNewThread(),
                                      "testCanLaunchSecondaryIsolate", {});
389 390
  ASSERT_TRUE(isolate);
  ASSERT_EQ(isolate->get()->GetPhase(), DartIsolate::Phase::Running);
391
  child_shutdown_latch.Wait();  // wait for child isolate to shutdown first
392
  ASSERT_FALSE(root_isolate_shutdown_latch.IsSignaledForTest());
393 394
  latch.Wait();  // wait for last NotifyNative called by main isolate
  // root isolate will be auto-shutdown
395 396 397 398 399 400 401 402 403 404 405 406 407
}

TEST_F(DartIsolateTest, CanRecieveArguments) {
  fml::AutoResetWaitableEvent latch;
  AddNativeCallback("NotifyNative",
                    CREATE_NATIVE_ENTRY(([&latch](Dart_NativeArguments args) {
                      ASSERT_TRUE(tonic::DartConverter<bool>::FromDart(
                          Dart_GetNativeArgument(args, 0)));
                      latch.Signal();
                    })));

  const auto settings = CreateSettingsForFixture();
  auto vm_ref = DartVMRef::Create(settings);
408 409
  auto isolate = RunDartCodeInIsolate(vm_ref, settings, CreateNewThread(),
                                      "testCanRecieveArguments", {"arg1"});
410 411 412 413 414 415
  ASSERT_TRUE(isolate);
  ASSERT_EQ(isolate->get()->GetPhase(), DartIsolate::Phase::Running);

  latch.Wait();
}

416
}  // namespace testing
417
}  // namespace flutter