OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 #include "vm/class_finalizer.h" | 5 #include "vm/class_finalizer.h" |
6 | 6 |
7 #include "vm/flags.h" | 7 #include "vm/flags.h" |
8 #include "vm/heap.h" | 8 #include "vm/heap.h" |
9 #include "vm/isolate.h" | 9 #include "vm/isolate.h" |
10 #include "vm/longjump.h" | 10 #include "vm/longjump.h" |
(...skipping 614 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
625 if (!cls.is_finalized()) { | 625 if (!cls.is_finalized()) { |
626 FinalizeTypeParameters(cls); | 626 FinalizeTypeParameters(cls); |
627 } | 627 } |
628 Type& super_type = Type::Handle(cls.super_type()); | 628 Type& super_type = Type::Handle(cls.super_type()); |
629 if (!super_type.IsNull()) { | 629 if (!super_type.IsNull()) { |
630 const Class& super_class = Class::Handle(super_type.type_class()); | 630 const Class& super_class = Class::Handle(super_type.type_class()); |
631 AbstractTypeArguments& super_type_args = AbstractTypeArguments::Handle(); | 631 AbstractTypeArguments& super_type_args = AbstractTypeArguments::Handle(); |
632 if (super_type.IsBeingFinalized()) { | 632 if (super_type.IsBeingFinalized()) { |
633 // This type references itself via its type arguments. This is legal, but | 633 // This type references itself via its type arguments. This is legal, but |
634 // we must avoid endless recursion. We therefore map the innermost | 634 // we must avoid endless recursion. We therefore map the innermost |
635 // super type to Dynamic. | 635 // super type to dynamic. |
636 // Note that a direct self-reference via the super class chain is illegal | 636 // Note that a direct self-reference via the super class chain is illegal |
637 // and reported as an error earlier. | 637 // and reported as an error earlier. |
638 // Such legal self-references occur with F-bounded quantification. | 638 // Such legal self-references occur with F-bounded quantification. |
639 // Example 1: class Derived extends Base<Derived>. | 639 // Example 1: class Derived extends Base<Derived>. |
640 // The type 'Derived' forms a cycle by pointing to itself via its | 640 // The type 'Derived' forms a cycle by pointing to itself via its |
641 // flattened type argument vector: Derived[Base[Derived[Base[...]]]] | 641 // flattened type argument vector: Derived[Base[Derived[Base[...]]]] |
642 // We break the cycle as follows: Derived[Base[Derived[Dynamic]]] | 642 // We break the cycle as follows: Derived[Base[Derived[dynamic]]] |
643 // Example 2: class Derived extends Base<Middle<Derived>> results in | 643 // Example 2: class Derived extends Base<Middle<Derived>> results in |
644 // Derived[Base[Middle[Derived[Dynamic]]]] | 644 // Derived[Base[Middle[Derived[dynamic]]]] |
645 // Example 3: class Derived<T> extends Base<Derived<T>> results in | 645 // Example 3: class Derived<T> extends Base<Derived<T>> results in |
646 // Derived[Base[Derived[Dynamic]], T]. | 646 // Derived[Base[Derived[dynamic]], T]. |
647 ASSERT(super_type_args.IsNull()); // Same as a vector of Dynamic. | 647 ASSERT(super_type_args.IsNull()); // Same as a vector of dynamic. |
648 } else { | 648 } else { |
649 super_type ^= FinalizeType(cls, super_type, finalization); | 649 super_type ^= FinalizeType(cls, super_type, finalization); |
650 cls.set_super_type(super_type); | 650 cls.set_super_type(super_type); |
651 super_type_args = super_type.arguments(); | 651 super_type_args = super_type.arguments(); |
652 } | 652 } |
653 const intptr_t num_super_type_params = super_class.NumTypeParameters(); | 653 const intptr_t num_super_type_params = super_class.NumTypeParameters(); |
654 const intptr_t offset = super_class.NumTypeArguments(); | 654 const intptr_t offset = super_class.NumTypeArguments(); |
655 const intptr_t super_offset = offset - num_super_type_params; | 655 const intptr_t super_offset = offset - num_super_type_params; |
656 ASSERT(offset == (cls.NumTypeArguments() - cls.NumTypeParameters())); | 656 ASSERT(offset == (cls.NumTypeArguments() - cls.NumTypeParameters())); |
657 AbstractType& super_type_arg = AbstractType::Handle(Type::DynamicType()); | 657 AbstractType& super_type_arg = AbstractType::Handle(Type::DynamicType()); |
(...skipping 89 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
747 // Malformed type arguments to a constructor of a generic type are | 747 // Malformed type arguments to a constructor of a generic type are |
748 // reported as a compile-time error. | 748 // reported as a compile-time error. |
749 if (finalization >= kCanonicalizeForCreation) { | 749 if (finalization >= kCanonicalizeForCreation) { |
750 const Script& script = Script::Handle(cls.script()); | 750 const Script& script = Script::Handle(cls.script()); |
751 const String& type_name = | 751 const String& type_name = |
752 String::Handle(parameterized_type.UserVisibleName()); | 752 String::Handle(parameterized_type.UserVisibleName()); |
753 ReportError(script, parameterized_type.token_pos(), | 753 ReportError(script, parameterized_type.token_pos(), |
754 "type '%s' has malformed type argument", | 754 "type '%s' has malformed type argument", |
755 type_name.ToCString()); | 755 type_name.ToCString()); |
756 } | 756 } |
757 // In production mode, malformed type arguments are mapped to Dynamic. | 757 // In production mode, malformed type arguments are mapped to dynamic. |
758 // In checked mode, a type with malformed type arguments is malformed. | 758 // In checked mode, a type with malformed type arguments is malformed. |
759 if (FLAG_enable_type_checks || FLAG_error_on_malformed_type) { | 759 if (FLAG_enable_type_checks || FLAG_error_on_malformed_type) { |
760 const Error& error = Error::Handle(type_argument.malformed_error()); | 760 const Error& error = Error::Handle(type_argument.malformed_error()); |
761 const String& type_name = | 761 const String& type_name = |
762 String::Handle(parameterized_type.UserVisibleName()); | 762 String::Handle(parameterized_type.UserVisibleName()); |
763 FinalizeMalformedType(error, cls, parameterized_type, finalization, | 763 FinalizeMalformedType(error, cls, parameterized_type, finalization, |
764 "type '%s' has malformed type argument", | 764 "type '%s' has malformed type argument", |
765 type_name.ToCString()); | 765 type_name.ToCString()); |
766 return parameterized_type.raw(); | 766 return parameterized_type.raw(); |
767 } else { | 767 } else { |
(...skipping 150 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
918 void ClassFinalizer::ResolveAndFinalizeSignature(const Class& cls, | 918 void ClassFinalizer::ResolveAndFinalizeSignature(const Class& cls, |
919 const Function& function) { | 919 const Function& function) { |
920 // Resolve result type. | 920 // Resolve result type. |
921 AbstractType& type = AbstractType::Handle(function.result_type()); | 921 AbstractType& type = AbstractType::Handle(function.result_type()); |
922 // In case of a factory, the parser sets the factory result type to a type | 922 // In case of a factory, the parser sets the factory result type to a type |
923 // with an unresolved class whose name matches the factory name. | 923 // with an unresolved class whose name matches the factory name. |
924 // It is not a compile time error if this name does not resolve to a class or | 924 // It is not a compile time error if this name does not resolve to a class or |
925 // interface. | 925 // interface. |
926 ResolveType(cls, type, kCanonicalize); | 926 ResolveType(cls, type, kCanonicalize); |
927 type = FinalizeType(cls, type, kCanonicalize); | 927 type = FinalizeType(cls, type, kCanonicalize); |
928 // In production mode, a malformed result type is mapped to Dynamic. | 928 // In production mode, a malformed result type is mapped to dynamic. |
929 if (!FLAG_enable_type_checks && type.IsMalformed()) { | 929 if (!FLAG_enable_type_checks && type.IsMalformed()) { |
930 type = Type::DynamicType(); | 930 type = Type::DynamicType(); |
931 } | 931 } |
932 function.set_result_type(type); | 932 function.set_result_type(type); |
933 // Resolve formal parameter types. | 933 // Resolve formal parameter types. |
934 const intptr_t num_parameters = function.NumParameters(); | 934 const intptr_t num_parameters = function.NumParameters(); |
935 for (intptr_t i = 0; i < num_parameters; i++) { | 935 for (intptr_t i = 0; i < num_parameters; i++) { |
936 type = function.ParameterTypeAt(i); | 936 type = function.ParameterTypeAt(i); |
937 ResolveType(cls, type, kCanonicalize); | 937 ResolveType(cls, type, kCanonicalize); |
938 type = FinalizeType(cls, type, kCanonicalize); | 938 type = FinalizeType(cls, type, kCanonicalize); |
939 // In production mode, a malformed parameter type is mapped to Dynamic. | 939 // In production mode, a malformed parameter type is mapped to dynamic. |
940 if (!FLAG_enable_type_checks && type.IsMalformed()) { | 940 if (!FLAG_enable_type_checks && type.IsMalformed()) { |
941 type = Type::DynamicType(); | 941 type = Type::DynamicType(); |
942 } | 942 } |
943 function.SetParameterTypeAt(i, type); | 943 function.SetParameterTypeAt(i, type); |
944 } | 944 } |
945 } | 945 } |
946 | 946 |
947 | 947 |
948 static RawClass* FindSuperOwnerOfInstanceMember(const Class& cls, | 948 static RawClass* FindSuperOwnerOfInstanceMember(const Class& cls, |
949 const String& name) { | 949 const String& name) { |
(...skipping 427 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1377 String::Handle(interface.Name()).ToCString()); | 1377 String::Handle(interface.Name()).ToCString()); |
1378 } | 1378 } |
1379 interface_class = interface.type_class(); | 1379 interface_class = interface.type_class(); |
1380 if (interface_class.IsSignatureClass()) { | 1380 if (interface_class.IsSignatureClass()) { |
1381 const Script& script = Script::Handle(cls.script()); | 1381 const Script& script = Script::Handle(cls.script()); |
1382 ReportError(script, cls.token_pos(), | 1382 ReportError(script, cls.token_pos(), |
1383 "'%s' is used where an interface or class name is expected", | 1383 "'%s' is used where an interface or class name is expected", |
1384 String::Handle(interface_class.Name()).ToCString()); | 1384 String::Handle(interface_class.Name()).ToCString()); |
1385 } | 1385 } |
1386 // Verify that unless cls belongs to core lib, it cannot extend or implement | 1386 // Verify that unless cls belongs to core lib, it cannot extend or implement |
1387 // any of bool, num, int, double, String, Function, Dynamic. | 1387 // any of bool, num, int, double, String, Function, dynamic. |
1388 // The exception is signature classes, which are compiler generated and | 1388 // The exception is signature classes, which are compiler generated and |
1389 // represent a function type, therefore implementing the Function interface. | 1389 // represent a function type, therefore implementing the Function interface. |
1390 if (!cls_belongs_to_core_lib) { | 1390 if (!cls_belongs_to_core_lib) { |
1391 if (interface.IsBoolType() || | 1391 if (interface.IsBoolType() || |
1392 interface.IsNumberType() || | 1392 interface.IsNumberType() || |
1393 interface.IsIntType() || | 1393 interface.IsIntType() || |
1394 interface.IsDoubleType() || | 1394 interface.IsDoubleType() || |
1395 interface.IsStringInterface() || | 1395 interface.IsStringInterface() || |
1396 (interface.IsFunctionType() && !cls.IsSignatureClass()) || | 1396 (interface.IsFunctionType() && !cls.IsSignatureClass()) || |
1397 interface.IsDynamicType()) { | 1397 interface.IsDynamicType()) { |
(...skipping 187 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1585 void ClassFinalizer::ReportError(const char* format, ...) { | 1585 void ClassFinalizer::ReportError(const char* format, ...) { |
1586 va_list args; | 1586 va_list args; |
1587 va_start(args, format); | 1587 va_start(args, format); |
1588 const Error& error = Error::Handle( | 1588 const Error& error = Error::Handle( |
1589 Parser::FormatError(Script::Handle(), -1, "Error", format, args)); | 1589 Parser::FormatError(Script::Handle(), -1, "Error", format, args)); |
1590 va_end(args); | 1590 va_end(args); |
1591 ReportError(error); | 1591 ReportError(error); |
1592 } | 1592 } |
1593 | 1593 |
1594 } // namespace dart | 1594 } // namespace dart |
OLD | NEW |