blob: 03a2cefcd497da76d82a5d80575948f2f03fff63 [file] [log] [blame]
// Copyright (c) 2012, 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 "vm/bootstrap_natives.h"
#include "vm/compiler/jit/compiler.h"
#include "vm/dart_entry.h"
#include "vm/exceptions.h"
#include "vm/native_entry.h"
#include "vm/object.h"
#include "vm/symbols.h"
namespace dart {
DEFINE_NATIVE_ENTRY(Function_apply, 0, 2) {
const int kTypeArgsLen = 0; // TODO(regis): Add support for generic function.
const Array& fun_arguments =
Array::CheckedHandle(zone, arguments->NativeArgAt(0));
const Array& fun_arg_names =
Array::CheckedHandle(zone, arguments->NativeArgAt(1));
const Array& fun_args_desc = Array::Handle(
zone, ArgumentsDescriptor::NewBoxed(kTypeArgsLen, fun_arguments.Length(),
fun_arg_names));
const Object& result = Object::Handle(
zone, DartEntry::InvokeClosure(thread, fun_arguments, fun_args_desc));
if (result.IsError()) {
Exceptions::PropagateError(Error::Cast(result));
}
return result.ptr();
}
static bool ClosureEqualsHelper(Zone* zone,
const Closure& receiver,
const Object& other) {
if (receiver.ptr() == other.ptr()) {
return true;
}
if (!other.IsClosure()) {
return false;
}
const auto& other_closure = Closure::Cast(other);
const auto& func_a = Function::Handle(zone, receiver.function());
const auto& func_b = Function::Handle(zone, other_closure.function());
// Check that functions match.
if (func_a.ptr() != func_b.ptr()) {
// Non-implicit closures taken from different functions are not equal.
if (!func_a.IsImplicitClosureFunction() ||
!func_b.IsImplicitClosureFunction()) {
return false;
}
// If the closure functions are not the same, check the function's name and
// owner, as multiple function objects could exist for the same function due
// to hot reload.
if ((func_a.name() != func_b.name() || func_a.Owner() != func_b.Owner() ||
func_a.is_static() != func_b.is_static())) {
return false;
}
}
// Check that the delayed type argument vectors match.
if (receiver.delayed_type_arguments() !=
other_closure.delayed_type_arguments()) {
// Mismatches should only happen when a generic function is involved.
ASSERT(func_a.IsGeneric() || func_b.IsGeneric());
const auto& type_args_a =
TypeArguments::Handle(zone, receiver.delayed_type_arguments());
const auto& type_args_b =
TypeArguments::Handle(zone, other_closure.delayed_type_arguments());
if (type_args_a.IsNull() || type_args_b.IsNull() ||
(type_args_a.Length() != type_args_b.Length()) ||
!type_args_a.IsEquivalent(type_args_b, TypeEquality::kSyntactical)) {
return false;
}
}
if (func_a.IsImplicitClosureFunction()) {
ASSERT(func_b.IsImplicitClosureFunction());
if (!func_a.is_static()) {
ASSERT(!func_b.is_static());
// Check that the both receiver instances are the same.
const Instance& receiver_a =
Instance::Handle(zone, receiver.GetImplicitClosureReceiver());
const Instance& receiver_b =
Instance::Handle(zone, other_closure.GetImplicitClosureReceiver());
return receiver_a.ptr() == receiver_b.ptr();
}
} else {
ASSERT(!func_b.IsImplicitClosureFunction());
if (func_a.IsGeneric()) {
// Additional constraints for closures of generic functions:
// (1) Different instantiations of the same generic closure
// with the same type arguments should be equal.
// This means that instantiated generic closures are not unique
// and equality of instantiated generic closures should not be
// based on identity.
// (2) Instantiations of non-equal generic closures should be non-equal.
// This means that equality of non-instantiated generic closures
// should not be based on identity too as it won't match equality
// after instantiation.
if ((receiver.GetContext() != other_closure.GetContext()) ||
(receiver.instantiator_type_arguments() !=
other_closure.instantiator_type_arguments()) ||
(receiver.function_type_arguments() !=
other_closure.function_type_arguments())) {
return false;
}
} else {
// Closures of non-generic functions are unique.
return false;
}
}
return true;
}
DEFINE_NATIVE_ENTRY(Closure_equals, 0, 2) {
const Closure& receiver =
Closure::CheckedHandle(zone, arguments->NativeArgAt(0));
GET_NATIVE_ARGUMENT(Instance, other, arguments->NativeArgAt(1));
ASSERT(!other.IsNull());
return Bool::Get(ClosureEqualsHelper(zone, receiver, other)).ptr();
}
DEFINE_NATIVE_ENTRY(Closure_computeHash, 0, 1) {
const Closure& receiver =
Closure::CheckedHandle(zone, arguments->NativeArgAt(0));
return Smi::New(receiver.ComputeHash());
}
} // namespace dart