blob: 2eebd39ee38bd21eab6cfba5574b1eb90202f984 [file] [log] [blame] [edit]
// Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
#include "lib/stacktrace.h"
#include "platform/assert.h"
#include "vm/bootstrap_natives.h"
#include "vm/debugger.h"
#include "vm/exceptions.h"
#include "vm/flags.h"
#include "vm/microtask_mirror_queues.h"
#include "vm/native_entry.h"
#include "vm/object_store.h"
#include "vm/runtime_entry.h"
namespace dart {
// This flag is defined in "runtime/vm/microtask_mirror_queues.cc".
DECLARE_FLAG(bool, profile_microtasks);
DEFINE_NATIVE_ENTRY(AsyncStarMoveNext_debuggerStepCheck, 0, 1) {
#if !defined(PRODUCT)
GET_NON_NULL_NATIVE_ARGUMENT(Closure, generator, arguments->NativeArgAt(0));
Debugger* debugger = isolate->debugger();
if (debugger != nullptr && debugger->IsSingleStepping()) {
debugger->AsyncStepInto(generator);
}
#endif
return Object::null();
}
// Instantiate generic [closure] using the type argument T
// corresponding to Future<T> in the given [future] instance
// (which may extend or implement Future).
DEFINE_NATIVE_ENTRY(SuspendState_instantiateClosureWithFutureTypeArgument,
0,
2) {
GET_NON_NULL_NATIVE_ARGUMENT(Closure, closure, arguments->NativeArgAt(0));
GET_NON_NULL_NATIVE_ARGUMENT(Instance, future, arguments->NativeArgAt(1));
IsolateGroup* isolate_group = thread->isolate_group();
const auto& future_class =
Class::Handle(zone, isolate_group->object_store()->future_class());
ASSERT(future_class.NumTypeArguments() == 1);
const auto& cls = Class::Handle(zone, future.clazz());
auto& type = Type::Handle(zone, cls.GetInstantiationOf(zone, future_class));
ASSERT(!type.IsNull());
if (!type.IsInstantiated()) {
const auto& instance_type_args =
TypeArguments::Handle(zone, future.GetTypeArguments());
type ^=
type.InstantiateFrom(instance_type_args, Object::null_type_arguments(),
kNoneFree, Heap::kOld);
}
auto& type_args = TypeArguments::Handle(zone, type.arguments());
ASSERT(type_args.IsNull() || type_args.Length() == 1);
type_args = type_args.Canonicalize(thread);
ASSERT(closure.delayed_type_arguments() ==
Object::empty_type_arguments().ptr());
closure.set_delayed_type_arguments(type_args);
return closure.ptr();
}
DEFINE_NATIVE_ENTRY(MicrotaskMirrorQueue_onScheduleAsyncCallback, 0, 0) {
// There is logic in `sdk/lib/async/schedule_microtask.dart` that ensures that
// this function can only ever be called when the `--profile-microtasks` CLI
// flag is set in non-PRODUCT modes.
#if !defined(PRODUCT)
ASSERT(FLAG_profile_microtasks);
const StackTrace& stack_trace = GetCurrentStackTrace(
// We pass a `skip_frames` argument of 1 to skip the
// `_MicrotaskMirrorQueue._onScheduleAsyncCallback` frame.
1);
MicrotaskMirrorQueues::GetQueue(static_cast<int64_t>(isolate->main_port()))
->OnScheduleAsyncCallback(stack_trace);
return Object::null();
#else
UNREACHABLE();
#endif // !defined(PRODUCT)
}
DEFINE_NATIVE_ENTRY(MicrotaskMirrorQueue_onSchedulePriorityAsyncCallback,
0,
0) {
// There is logic in `sdk/lib/async/schedule_microtask.dart` that ensures that
// this function can only ever be called when the `--profile-microtasks` CLI
// flag is set in non-PRODUCT modes.
#if !defined(PRODUCT)
ASSERT(FLAG_profile_microtasks);
MicrotaskMirrorQueues::GetQueue(static_cast<int64_t>(isolate->main_port()))
->OnSchedulePriorityAsyncCallback();
return Object::null();
#else
UNREACHABLE();
#endif // !defined(PRODUCT)
}
DEFINE_NATIVE_ENTRY(MicrotaskMirrorQueue_onAsyncCallbackComplete, 0, 2) {
// There is logic in `sdk/lib/async/schedule_microtask.dart` that ensures that
// this function can only ever be called when the `--profile-microtasks` CLI
// flag is set in non-PRODUCT modes.
#if !defined(PRODUCT)
ASSERT(FLAG_profile_microtasks);
GET_NON_NULL_NATIVE_ARGUMENT(Integer, start_time, arguments->NativeArgAt(0));
GET_NON_NULL_NATIVE_ARGUMENT(Integer, end_time, arguments->NativeArgAt(1));
MicrotaskMirrorQueues::GetQueue(static_cast<int64_t>(isolate->main_port()))
->OnAsyncCallbackComplete(start_time.Value(), end_time.Value());
return Object::null();
#else
UNREACHABLE();
#endif // !defined(PRODUCT)
}
} // namespace dart