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

Side by Side Diff: lib/src/checker/checker.dart

Issue 1190413005: Fixes #151 (Closed) Base URL: https://github.com/dart-lang/dev_compiler.git@master
Patch Set: Reformat Created 5 years, 6 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 unified diff | Download patch
« no previous file with comments | « no previous file | lib/src/checker/rules.dart » ('j') | test/checker/checker_test.dart » ('J')
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2015, 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 library dev_compiler.src.checker.checker; 5 library dev_compiler.src.checker.checker;
6 6
7 import 'package:analyzer/analyzer.dart'; 7 import 'package:analyzer/analyzer.dart';
8 import 'package:analyzer/src/generated/ast.dart'; 8 import 'package:analyzer/src/generated/ast.dart';
9 import 'package:analyzer/src/generated/element.dart'; 9 import 'package:analyzer/src/generated/element.dart';
10 import 'package:analyzer/src/generated/scanner.dart' show Token, TokenType; 10 import 'package:analyzer/src/generated/scanner.dart' show Token, TokenType;
(...skipping 319 matching lines...) Expand 10 before | Expand all | Expand 10 after
330 if (info.level >= logger.Level.SEVERE) _failure = true; 330 if (info.level >= logger.Level.SEVERE) _failure = true;
331 _reporter.log(info); 331 _reporter.log(info);
332 } 332 }
333 } 333 }
334 334
335 /// Checks the body of functions and properties. 335 /// Checks the body of functions and properties.
336 class CodeChecker extends RecursiveAstVisitor { 336 class CodeChecker extends RecursiveAstVisitor {
337 final TypeRules _rules; 337 final TypeRules _rules;
338 final CheckerReporter _reporter; 338 final CheckerReporter _reporter;
339 final _OverrideChecker _overrideChecker; 339 final _OverrideChecker _overrideChecker;
340 bool _constantContext = false;
341 bool _failure = false; 340 bool _failure = false;
342 bool get failure => _failure || _overrideChecker._failure; 341 bool get failure => _failure || _overrideChecker._failure;
343 342
344 CodeChecker( 343 CodeChecker(
345 TypeRules rules, CheckerReporter reporter, StrongModeOptions options) 344 TypeRules rules, CheckerReporter reporter, StrongModeOptions options)
346 : _rules = rules, 345 : _rules = rules,
347 _reporter = reporter, 346 _reporter = reporter,
348 _overrideChecker = new _OverrideChecker(rules, reporter, options); 347 _overrideChecker = new _OverrideChecker(rules, reporter, options);
349 348
350 @override 349 @override
351 void visitCompilationUnit(CompilationUnit unit) { 350 void visitCompilationUnit(CompilationUnit unit) {
352 void report(Expression expr) { 351 void report(Expression expr) {
353 _reporter.log(new MissingTypeError(expr)); 352 _reporter.log(new MissingTypeError(expr));
354 } 353 }
355 var callback = _rules.reportMissingType; 354 var callback = _rules.reportMissingType;
356 _rules.reportMissingType = report; 355 _rules.reportMissingType = report;
357 unit.visitChildren(this); 356 unit.visitChildren(this);
358 _rules.reportMissingType = callback; 357 _rules.reportMissingType = callback;
359 } 358 }
360 359
361 _visitMaybeConst(AstNode n, visitNode(AstNode n)) {
362 var o = _constantContext;
363 if (!o) {
364 if (n is VariableDeclarationList) {
365 _constantContext = o || n.isConst;
366 } else if (n is VariableDeclaration) {
367 _constantContext = o || n.isConst;
368 } else if (n is DefaultFormalParameter) {
369 _constantContext = true;
370 } else if (n is FormalParameter) {
371 _constantContext = o || n.isConst;
372 } else if (n is InstanceCreationExpression) {
373 _constantContext = o || n.isConst;
374 } else if (n is ConstructorDeclaration) {
375 _constantContext = o || n.element.isConst;
376 }
377 }
378 visitNode(n);
379 _constantContext = o;
380 }
381
382 @override 360 @override
383 void visitComment(Comment node) { 361 void visitComment(Comment node) {
384 // skip, no need to do typechecking inside comments (they may contain 362 // skip, no need to do typechecking inside comments (they may contain
385 // comment references which would require resolution). 363 // comment references which would require resolution).
386 } 364 }
387 365
388 @override 366 @override
389 void visitClassDeclaration(ClassDeclaration node) { 367 void visitClassDeclaration(ClassDeclaration node) {
390 _overrideChecker.check(node); 368 _overrideChecker.check(node);
391 super.visitClassDeclaration(node); 369 super.visitClassDeclaration(node);
392 } 370 }
393 371
394 @override 372 @override
395 void visitAssignmentExpression(AssignmentExpression node) { 373 void visitAssignmentExpression(AssignmentExpression node) {
396 var token = node.operator; 374 var token = node.operator;
397 if (token.type != TokenType.EQ) { 375 if (token.type != TokenType.EQ) {
398 _checkCompoundAssignment(node); 376 _checkCompoundAssignment(node);
399 } else { 377 } else {
400 DartType staticType = _rules.getStaticType(node.leftHandSide); 378 DartType staticType = _rules.getStaticType(node.leftHandSide);
401 checkAssignment(node.rightHandSide, staticType); 379 checkAssignment(node.rightHandSide, staticType);
402 } 380 }
403 node.visitChildren(this); 381 node.visitChildren(this);
404 } 382 }
405 383
406 /// Check constructor declaration to ensure correct super call placement. 384 /// Check constructor declaration to ensure correct super call placement.
407 @override 385 @override
408 void visitConstructorDeclaration(ConstructorDeclaration node) { 386 void visitConstructorDeclaration(ConstructorDeclaration node) {
409 _visitMaybeConst(node, (node) { 387 node.visitChildren(this);
410 node.visitChildren(this);
411 388
412 final init = node.initializers; 389 final init = node.initializers;
413 for (int i = 0, last = init.length - 1; i < last; i++) { 390 for (int i = 0, last = init.length - 1; i < last; i++) {
414 final node = init[i]; 391 final node = init[i];
415 if (node is SuperConstructorInvocation) { 392 if (node is SuperConstructorInvocation) {
416 _recordMessage(new InvalidSuperInvocation(node)); 393 _recordMessage(new InvalidSuperInvocation(node));
417 }
418 } 394 }
419 }); 395 }
420 } 396 }
421 397
422 @override 398 @override
423 void visitConstructorFieldInitializer(ConstructorFieldInitializer node) { 399 void visitConstructorFieldInitializer(ConstructorFieldInitializer node) {
424 var field = node.fieldName; 400 var field = node.fieldName;
425 DartType staticType = _rules.elementType(field.staticElement); 401 DartType staticType = _rules.elementType(field.staticElement);
426 checkAssignment(node.expression, staticType); 402 checkAssignment(node.expression, staticType);
427 node.visitChildren(this); 403 node.visitChildren(this);
428 } 404 }
429 405
(...skipping 166 matching lines...) Expand 10 before | Expand all | Expand 10 after
596 var element = node.staticElement; 572 var element = node.staticElement;
597 if (element == null) { 573 if (element == null) {
598 _recordMessage(new MissingTypeError(node)); 574 _recordMessage(new MissingTypeError(node));
599 } else { 575 } else {
600 var type = node.staticElement.type; 576 var type = node.staticElement.type;
601 checkArgumentList(node.argumentList, type); 577 checkArgumentList(node.argumentList, type);
602 } 578 }
603 node.visitChildren(this); 579 node.visitChildren(this);
604 } 580 }
605 581
606 AstNode _getOwnerFunction(AstNode node) { 582 DartType _getExpectedReturnType(FunctionBody body, bool yieldStar) {
607 var parent = node.parent; 583 FunctionType functionType;
608 while (parent is! FunctionExpression && 584 var parent = body.parent;
609 parent is! MethodDeclaration && 585 if (parent is Declaration) {
610 parent is! ConstructorDeclaration) { 586 functionType = _rules.elementType(parent.element);
611 parent = parent.parent; 587 } else {
588 assert(parent is FunctionExpression);
589 functionType = _rules.getStaticType(parent);
612 } 590 }
613 return parent;
614 }
615 591
616 FunctionType _getFunctionType(AstNode node) { 592 var type = functionType.returnType;
617 if (node is Declaration) { 593 var provider = _rules.provider;
618 return _rules.elementType(node.element); 594
595 InterfaceType expectedType = null;
596 if (body.isAsynchronous) {
597 if (body.isGenerator) {
598 // Stream<T> -> T
599 expectedType = provider.streamType;
600 } else {
601 // Future<T> -> T
602 // TODO(vsm): Revisit with issue #228.
603 expectedType = provider.futureType;
604 }
619 } else { 605 } else {
620 assert(node is FunctionExpression); 606 if (body.isGenerator) {
621 return _rules.getStaticType(node); 607 // Iterable<T> -> T
608 expectedType = provider.iterableType;
609 } else {
610 // T -> T
611 return type;
612 }
613 }
614 if (yieldStar) {
615 if (type.isDynamic) {
616 // Ensure it's at least a Stream / Iterable.
617 return expectedType.substitute4([provider.dynamicType]);
618 } else {
619 // Analyzer will provide a separate error if expected type
620 // is not compatible with type.
621 return type;
622 }
623 }
624 if (type.isDynamic) {
625 return type;
626 } else if (type is InterfaceType && type.element == expectedType.element) {
627 return type.typeArguments[0];
628 } else {
629 // Malformed type - fallback on analyzer error.
630 return null;
622 } 631 }
623 } 632 }
624 633
625 void _checkReturn(Expression expression, AstNode node) { 634 FunctionBody _getFunctionBody(AstNode node) {
626 var type = _getFunctionType(_getOwnerFunction(node)).returnType; 635 while (node is! FunctionBody) {
636 node = node.parent;
637 }
638 return node as FunctionBody;
639 }
640
641 void _checkReturnOrYield(Expression expression, AstNode node,
642 [bool yieldStar = false]) {
643 var body = _getFunctionBody(node);
644 var type = _getExpectedReturnType(body, yieldStar);
645 if (type == null) {
646 // We have a type mismatch: the async/async*/sync* modifier does
647 // not match the return or yield type. We should have already gotten an
648 // analyzer error in this case.
649 return;
650 }
627 // TODO(vsm): Enforce void or dynamic (to void?) when expression is null. 651 // TODO(vsm): Enforce void or dynamic (to void?) when expression is null.
628 if (expression != null) checkAssignment(expression, type); 652 if (expression != null) checkAssignment(expression, type);
629 } 653 }
630 654
631 @override 655 @override
632 void visitExpressionFunctionBody(ExpressionFunctionBody node) { 656 void visitExpressionFunctionBody(ExpressionFunctionBody node) {
633 _checkReturn(node.expression, node); 657 _checkReturnOrYield(node.expression, node);
634 node.visitChildren(this); 658 node.visitChildren(this);
635 } 659 }
636 660
637 @override 661 @override
638 void visitReturnStatement(ReturnStatement node) { 662 void visitReturnStatement(ReturnStatement node) {
639 _checkReturn(node.expression, node); 663 _checkReturnOrYield(node.expression, node);
640 node.visitChildren(this); 664 node.visitChildren(this);
641 } 665 }
642 666
667 @override
668 void visitYieldStatement(YieldStatement node) {
669 _checkReturnOrYield(node.expression, node, node.star != null);
670 node.visitChildren(this);
671 }
672
643 @override 673 @override
644 void visitPropertyAccess(PropertyAccess node) { 674 void visitPropertyAccess(PropertyAccess node) {
645 var target = node.realTarget; 675 var target = node.realTarget;
646 if (_rules.isDynamicTarget(target)) { 676 if (_rules.isDynamicTarget(target)) {
647 _recordDynamicInvoke(node, target); 677 _recordDynamicInvoke(node, target);
648 } 678 }
649 node.visitChildren(this); 679 node.visitChildren(this);
650 } 680 }
651 681
652 @override 682 @override
653 void visitPrefixedIdentifier(PrefixedIdentifier node) { 683 void visitPrefixedIdentifier(PrefixedIdentifier node) {
654 final target = node.prefix; 684 final target = node.prefix;
655 if (_rules.isDynamicTarget(target)) { 685 if (_rules.isDynamicTarget(target)) {
656 _recordDynamicInvoke(node, target); 686 _recordDynamicInvoke(node, target);
657 } 687 }
658 node.visitChildren(this); 688 node.visitChildren(this);
659 } 689 }
660 690
661 @override 691 @override
662 void visitDefaultFormalParameter(DefaultFormalParameter node) { 692 void visitDefaultFormalParameter(DefaultFormalParameter node) {
663 _visitMaybeConst(node, (node) { 693 // Check that defaults have the proper subtype.
664 // Check that defaults have the proper subtype. 694 var parameter = node.parameter;
665 var parameter = node.parameter; 695 var parameterType = _rules.elementType(parameter.element);
666 var parameterType = _rules.elementType(parameter.element); 696 assert(parameterType != null);
667 assert(parameterType != null); 697 var defaultValue = node.defaultValue;
668 var defaultValue = node.defaultValue; 698 if (defaultValue == null) {
669 if (defaultValue == null) { 699 if (_rules.maybeNonNullableType(parameterType)) {
670 if (_rules.maybeNonNullableType(parameterType)) { 700 var staticInfo = new InvalidVariableDeclaration(
671 var staticInfo = new InvalidVariableDeclaration( 701 _rules, node.identifier, parameterType);
672 _rules, node.identifier, parameterType); 702 _recordMessage(staticInfo);
673 _recordMessage(staticInfo);
674 }
675 } else {
676 checkAssignment(defaultValue, parameterType);
677 } 703 }
704 } else {
705 checkAssignment(defaultValue, parameterType);
706 }
678 707
679 node.visitChildren(this); 708 node.visitChildren(this);
680 });
681 } 709 }
682 710
683 @override 711 @override
684 void visitFieldFormalParameter(FieldFormalParameter node) { 712 void visitFieldFormalParameter(FieldFormalParameter node) {
685 var element = node.element; 713 var element = node.element;
686 var typeName = node.type; 714 var typeName = node.type;
687 if (typeName != null) { 715 if (typeName != null) {
688 var type = _rules.elementType(element); 716 var type = _rules.elementType(element);
689 var fieldElement = 717 var fieldElement =
690 node.identifier.staticElement as FieldFormalParameterElement; 718 node.identifier.staticElement as FieldFormalParameterElement;
691 var fieldType = _rules.elementType(fieldElement.field); 719 var fieldType = _rules.elementType(fieldElement.field);
692 if (!_rules.isSubTypeOf(type, fieldType)) { 720 if (!_rules.isSubTypeOf(type, fieldType)) {
693 var staticInfo = 721 var staticInfo =
694 new InvalidParameterDeclaration(_rules, node, fieldType); 722 new InvalidParameterDeclaration(_rules, node, fieldType);
695 _recordMessage(staticInfo); 723 _recordMessage(staticInfo);
696 } 724 }
697 } 725 }
698 node.visitChildren(this); 726 node.visitChildren(this);
699 } 727 }
700 728
701 @override 729 @override
702 void visitInstanceCreationExpression(InstanceCreationExpression node) { 730 void visitInstanceCreationExpression(InstanceCreationExpression node) {
703 _visitMaybeConst(node, (node) { 731 var arguments = node.argumentList;
704 var arguments = node.argumentList; 732 var element = node.staticElement;
705 var element = node.staticElement; 733 if (element != null) {
706 if (element != null) { 734 var type = _rules.elementType(node.staticElement);
707 var type = _rules.elementType(node.staticElement); 735 checkArgumentList(arguments, type);
708 checkArgumentList(arguments, type); 736 } else {
709 } else { 737 _recordMessage(new MissingTypeError(node));
710 _recordMessage(new MissingTypeError(node)); 738 }
711 } 739 node.visitChildren(this);
712 node.visitChildren(this);
713 });
714 } 740 }
715 741
716 @override 742 @override
717 void visitVariableDeclarationList(VariableDeclarationList node) { 743 void visitVariableDeclarationList(VariableDeclarationList node) {
718 _visitMaybeConst(node, (node) { 744 TypeName type = node.type;
719 TypeName type = node.type; 745 if (type == null) {
720 if (type == null) { 746 // No checks are needed when the type is var. Although internally the
721 // No checks are needed when the type is var. Although internally the 747 // typing rules may have inferred a more precise type for the variable
722 // typing rules may have inferred a more precise type for the variable 748 // based on the initializer.
723 // based on the initializer. 749 } else {
724 } else { 750 var dartType = getType(type);
725 var dartType = getType(type); 751 for (VariableDeclaration variable in node.variables) {
726 for (VariableDeclaration variable in node.variables) { 752 var initializer = variable.initializer;
727 var initializer = variable.initializer; 753 if (initializer != null) {
728 if (initializer != null) { 754 checkAssignment(initializer, dartType);
729 checkAssignment(initializer, dartType); 755 } else if (_rules.maybeNonNullableType(dartType)) {
730 } else if (_rules.maybeNonNullableType(dartType)) { 756 var element = variable.element;
731 var element = variable.element; 757 if (element is FieldElement && !element.isStatic) {
732 if (element is FieldElement && !element.isStatic) { 758 // Initialized - possibly implicitly - during construction.
733 // Initialized - possibly implicitly - during construction. 759 // Handle this via a runtime check during code generation.
734 // Handle this via a runtime check during code generation.
735 760
736 // TODO(vsm): Detect statically whether this can fail and 761 // TODO(vsm): Detect statically whether this can fail and
737 // report a static error (must fail) or warning (can fail). 762 // report a static error (must fail) or warning (can fail).
738 } else { 763 } else {
739 var staticInfo = 764 var staticInfo =
740 new InvalidVariableDeclaration(_rules, variable, dartType); 765 new InvalidVariableDeclaration(_rules, variable, dartType);
741 _recordMessage(staticInfo); 766 _recordMessage(staticInfo);
742 }
743 } 767 }
744 } 768 }
745 } 769 }
746 node.visitChildren(this); 770 }
747 }); 771 node.visitChildren(this);
748 }
749
750 @override
751 void visitVariableDeclaration(VariableDeclaration node) {
752 _visitMaybeConst(node, super.visitVariableDeclaration);
753 } 772 }
754 773
755 void _checkRuntimeTypeCheck(AstNode node, TypeName typeName) { 774 void _checkRuntimeTypeCheck(AstNode node, TypeName typeName) {
756 var type = getType(typeName); 775 var type = getType(typeName);
757 if (!_rules.isGroundType(type)) { 776 if (!_rules.isGroundType(type)) {
758 _recordMessage(new InvalidRuntimeCheckError(node, type)); 777 _recordMessage(new InvalidRuntimeCheckError(node, type));
759 } 778 }
760 } 779 }
761 780
762 @override 781 @override
(...skipping 115 matching lines...) Expand 10 before | Expand all | Expand 10 after
878 /// Analyzer checks boolean conversions, but we need to check too, because 897 /// Analyzer checks boolean conversions, but we need to check too, because
879 /// it uses the default assignability rules that allow `dynamic` and `Object` 898 /// it uses the default assignability rules that allow `dynamic` and `Object`
880 /// to be assigned to bool with no message. 899 /// to be assigned to bool with no message.
881 void checkBoolean(Expression expr) => 900 void checkBoolean(Expression expr) =>
882 checkAssignment(expr, _rules.provider.boolType); 901 checkAssignment(expr, _rules.provider.boolType);
883 902
884 void checkAssignment(Expression expr, DartType type) { 903 void checkAssignment(Expression expr, DartType type) {
885 if (expr is ParenthesizedExpression) { 904 if (expr is ParenthesizedExpression) {
886 checkAssignment(expr.expression, type); 905 checkAssignment(expr.expression, type);
887 } else { 906 } else {
888 _recordMessage(_rules.checkAssignment(expr, type, _constantContext)); 907 _recordMessage(_rules.checkAssignment(expr, type));
889 } 908 }
890 } 909 }
891 910
892 DartType _specializedBinaryReturnType( 911 DartType _specializedBinaryReturnType(
893 TokenType op, DartType t1, DartType t2, DartType normalReturnType) { 912 TokenType op, DartType t1, DartType t2, DartType normalReturnType) {
894 // This special cases binary return types as per 16.26 and 16.27 of the 913 // This special cases binary return types as per 16.26 and 16.27 of the
895 // Dart language spec. 914 // Dart language spec.
896 switch (op) { 915 switch (op) {
897 case TokenType.PLUS: 916 case TokenType.PLUS:
898 case TokenType.MINUS: 917 case TokenType.MINUS:
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after
949 } else { 968 } else {
950 // Static type error 969 // Static type error
951 staticInfo = new StaticTypeError(_rules, expr, lhsType); 970 staticInfo = new StaticTypeError(_rules, expr, lhsType);
952 } 971 }
953 _recordMessage(staticInfo); 972 _recordMessage(staticInfo);
954 } 973 }
955 974
956 // Check the rhs type 975 // Check the rhs type
957 if (staticInfo is! CoercionInfo) { 976 if (staticInfo is! CoercionInfo) {
958 var paramType = paramTypes.first; 977 var paramType = paramTypes.first;
959 staticInfo = _rules.checkAssignment( 978 staticInfo = _rules.checkAssignment(expr.rightHandSide, paramType);
960 expr.rightHandSide, paramType, _constantContext);
961 _recordMessage(staticInfo); 979 _recordMessage(staticInfo);
962 } 980 }
963 } 981 }
964 } 982 }
965 983
966 void _recordDynamicInvoke(AstNode node, AstNode target) { 984 void _recordDynamicInvoke(AstNode node, AstNode target) {
967 var dinvoke = new DynamicInvoke(_rules, node); 985 var dinvoke = new DynamicInvoke(_rules, node);
968 _reporter.log(dinvoke); 986 _reporter.log(dinvoke);
969 // TODO(jmesserly): we may eventually want to record if the whole operation 987 // TODO(jmesserly): we may eventually want to record if the whole operation
970 // (node) was dynamic, rather than the target, but this is an easier fit 988 // (node) was dynamic, rather than the target, but this is an easier fit
971 // with what we used to do. 989 // with what we used to do.
972 DynamicInvoke.set(target, true); 990 DynamicInvoke.set(target, true);
973 } 991 }
974 992
975 void _recordMessage(StaticInfo info) { 993 void _recordMessage(StaticInfo info) {
976 if (info == null) return; 994 if (info == null) return;
977 if (info.level >= logger.Level.SEVERE) _failure = true; 995 if (info.level >= logger.Level.SEVERE) _failure = true;
978 _reporter.log(info); 996 _reporter.log(info);
979 if (info is CoercionInfo) { 997 if (info is CoercionInfo) {
980 assert(CoercionInfo.get(info.node) == null); 998 assert(CoercionInfo.get(info.node) == null);
981 CoercionInfo.set(info.node, info); 999 CoercionInfo.set(info.node, info);
982 } 1000 }
983 } 1001 }
984 } 1002 }
OLDNEW
« no previous file with comments | « no previous file | lib/src/checker/rules.dart » ('j') | test/checker/checker_test.dart » ('J')

Powered by Google App Engine
This is Rietveld 408576698