Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(402)

Unified Diff: runtime/vm/object.cc

Issue 8391007: Complete generic closure type checking. (Closed) Base URL: http://dart.googlecode.com/svn/branches/bleeding_edge/dart/
Patch Set: '' Created 9 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « runtime/vm/object.h ('k') | tests/co19/co19-runtime.status » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: runtime/vm/object.cc
===================================================================
--- runtime/vm/object.cc (revision 712)
+++ runtime/vm/object.cc (working copy)
@@ -1285,9 +1285,11 @@
if (IsSignatureClass() && other.IsSignatureClass()) {
const Function& fun = Function::Handle(signature_function());
const Function& other_fun = Function::Handle(other.signature_function());
- // TODO(regis): We need to consider the type arguments.
- return fun.IsSubtypeOf(other_fun);
+ return fun.IsSubtypeOf(type_arguments,
+ other_fun,
+ other_type_arguments);
}
+
if (is_interface()) {
// We already checked the case where 'other' is an interface. Now, 'this',
// an interface, cannot be more specific than a class, except class Object,
@@ -1319,16 +1321,19 @@
const Class& other,
const TypeArguments& other_type_arguments) const {
if (test == kIsAssignableTo) {
- // TODO(regis): We do not follow the guide that says that "a type T is
- // assignable to a type S if T is a subtype of S or S is a subtype of T",
- // since this would lead to heap pollution. We only apply that rule to
- // parameter types when checking assignability of function types.
- // Revisit if necessary.
+ // The spec states that "a type T is assignable to a type S if T is a
+ // subtype of S or S is a subtype of T". This is from the perspective of a
+ // static checker, which does not know the actual type of the assigned
+ // value. However, this type information is available at run time in checked
+ // mode. We therefore apply a more restrictive subtype check, which prevents
+ // heap pollution. We only keep the assignability check when assigning
+ // values of a function type.
if (IsSignatureClass() && other.IsSignatureClass()) {
const Function& src_fun = Function::Handle(signature_function());
const Function& dst_fun = Function::Handle(other.signature_function());
- // TODO(regis): We need to consider the type arguments.
- return src_fun.IsAssignableTo(dst_fun);
+ return src_fun.IsAssignableTo(type_arguments,
+ dst_fun,
+ other_type_arguments);
}
// Continue with a subtype test.
test = kIsSubtypeOf;
@@ -1780,11 +1785,8 @@
ASSERT(other.IsFinalized());
// Type parameters cannot be handled by Class::IsMoreSpecificThan().
if (IsTypeParameter() || other.IsTypeParameter()) {
- // TODO(regis): Revisit this temporary workaround. See issue 5474672.
- return
- (!IsTypeParameter() && other.IsTypeParameter()) ||
- (IsTypeParameter() && other.IsTypeParameter() &&
- (Index() == other.Index()));
+ return IsTypeParameter() && other.IsTypeParameter() &&
+ (Index() == other.Index());
}
const Class& cls = Class::Handle(type_class());
return cls.IsMoreSpecificThan(TypeArguments::Handle(arguments()),
@@ -1798,11 +1800,8 @@
ASSERT(other.IsFinalized());
// Type parameters cannot be handled by Class::TestType().
if (IsTypeParameter() || other.IsTypeParameter()) {
- // TODO(regis): Revisit this temporary workaround. See issue 5474672.
- return
- (!IsTypeParameter() && other.IsTypeParameter()) ||
- (IsTypeParameter() && other.IsTypeParameter() &&
- (Index() == other.Index()));
+ return IsTypeParameter() && other.IsTypeParameter() &&
+ (Index() == other.Index());
}
const Class& cls = Class::Handle(type_class());
if (test == kIsSubtypeOf) {
@@ -2719,7 +2718,10 @@
}
-bool Function::TestType(TypeTestKind test, const Function& other) const {
+bool Function::TestType(TypeTestKind test,
+ const TypeArguments& type_arguments,
+ const Function& other,
+ const TypeArguments& other_type_arguments) const {
const intptr_t num_fixed_params = num_fixed_parameters();
const intptr_t num_opt_params = num_optional_parameters();
const intptr_t other_num_fixed_params = other.num_fixed_parameters();
@@ -2729,23 +2731,18 @@
(num_opt_params < other_num_opt_params))) {
return false;
}
- // TODO(regis): We currently ignore type parameters. We need to consider the
- // parameter type upper bound, if any.
- // Note that unless we use this code to check function overrides at compile
- // time, all parameter types should be instantiated and we can remove code
- // checking for type parameters.
-
// Check the result type.
- const Type& other_res_type = Type::Handle(other.result_type());
- if (!other_res_type.IsTypeParameter() &&
- !other_res_type.IsVarType() &&
- !other_res_type.IsVoidType()) {
- const Type& res_type = Type::Handle(result_type());
- if (!res_type.IsTypeParameter() &&
- !res_type.IsVarType() &&
- (res_type.IsVoidType() || !res_type.IsSubtypeOf(other_res_type)) &&
- ((test == Type::kIsSubtypeOf) ||
- (!other_res_type.IsSubtypeOf(res_type)))) {
+ Type& other_res_type = Type::Handle(other.result_type());
+ if (!other_res_type.IsInstantiated()) {
+ other_res_type = other_res_type.InstantiateFrom(other_type_arguments, 0);
+ }
+ if (!other_res_type.IsVarType() && !other_res_type.IsVoidType()) {
+ Type& res_type = Type::Handle(result_type());
+ if (!res_type.IsInstantiated()) {
+ res_type = res_type.InstantiateFrom(type_arguments, 0);
+ }
+ if (!res_type.IsVarType() &&
+ (res_type.IsVoidType() || !res_type.IsAssignableTo(other_res_type))) {
return false;
}
}
@@ -2754,11 +2751,18 @@
Type& other_param_type = Type::Handle();
for (intptr_t i = 0; i < num_fixed_params; i++) {
param_type = ParameterTypeAt(i);
- if (param_type.IsTypeParameter() || param_type.IsVarType()) {
+ if (!param_type.IsInstantiated()) {
+ param_type = param_type.InstantiateFrom(type_arguments, 0);
+ }
+ if (param_type.IsVarType()) {
continue;
}
other_param_type = other.ParameterTypeAt(i);
- if (other_param_type.IsTypeParameter() || other_param_type.IsVarType()) {
+ if (!other_param_type.IsInstantiated()) {
+ other_param_type =
+ other_param_type.InstantiateFrom(other_type_arguments, 0);
+ }
+ if (other_param_type.IsVarType()) {
continue;
}
// Subtyping and assignability rules are identical when applied to parameter
@@ -2788,12 +2792,18 @@
if (ParameterNameAt(j) == other_param_name.raw()) {
found_param_name = true;
param_type = ParameterTypeAt(j);
- if (param_type.IsTypeParameter() || param_type.IsVarType()) {
+ if (!param_type.IsInstantiated()) {
+ param_type = param_type.InstantiateFrom(type_arguments, 0);
+ }
+ if (param_type.IsVarType()) {
break;
}
other_param_type = other.ParameterTypeAt(i);
- if (other_param_type.IsTypeParameter() ||
- other_param_type.IsVarType()) {
+ if (!other_param_type.IsInstantiated()) {
+ other_param_type =
+ other_param_type.InstantiateFrom(other_type_arguments, 0);
+ }
+ if (other_param_type.IsVarType()) {
break;
}
if (!param_type.IsSubtypeOf(other_param_type) &&
@@ -2830,12 +2840,18 @@
if (other.ParameterNameAt(j) == param_name.raw()) {
found_param_name = true;
other_param_type = other.ParameterTypeAt(j);
- if (other_param_type.IsTypeParameter() ||
- other_param_type.IsVarType()) {
+ if (!other_param_type.IsInstantiated()) {
+ other_param_type =
+ other_param_type.InstantiateFrom(other_type_arguments, 0);
+ }
+ if (other_param_type.IsVarType()) {
break;
}
param_type = ParameterTypeAt(i);
- if (param_type.IsTypeParameter() || param_type.IsVarType()) {
+ if (!param_type.IsInstantiated()) {
+ param_type = param_type.InstantiateFrom(type_arguments, 0);
+ }
+ if (param_type.IsVarType()) {
break;
}
if (!other_param_type.IsSubtypeOf(param_type) &&
« no previous file with comments | « runtime/vm/object.h ('k') | tests/co19/co19-runtime.status » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698