package org.jetbrains.jet.lang.types; import com.intellij.psi.PsiElement; import com.intellij.psi.tree.IElementType; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import org.jetbrains.jet.JetNodeTypes; import org.jetbrains.jet.lang.JetSemanticServices; import org.jetbrains.jet.lang.cfg.JetFlowInformationProvider; import org.jetbrains.jet.lang.psi.*; import org.jetbrains.jet.lang.resolve.*; import org.jetbrains.jet.lexer.JetTokens; import org.jetbrains.jet.resolve.DescriptorUtil; import java.util.*; /** * @author abreslav */ public class JetTypeInferrer { private static final Map unaryOperationNames = new HashMap(); static { unaryOperationNames.put(JetTokens.PLUSPLUS, "inc"); unaryOperationNames.put(JetTokens.MINUSMINUS, "dec"); unaryOperationNames.put(JetTokens.PLUS, "plus"); unaryOperationNames.put(JetTokens.MINUS, "minus"); unaryOperationNames.put(JetTokens.EXCL, "not"); } private static final Map binaryOperationNames = new HashMap(); static { binaryOperationNames.put(JetTokens.MUL, "times"); binaryOperationNames.put(JetTokens.PLUS, "plus"); binaryOperationNames.put(JetTokens.MINUS, "minus"); binaryOperationNames.put(JetTokens.DIV, "div"); binaryOperationNames.put(JetTokens.PERC, "mod"); binaryOperationNames.put(JetTokens.ARROW, "arrow"); binaryOperationNames.put(JetTokens.RANGE, "rangeTo"); } private static final Set comparisonOperations = new HashSet(Arrays.asList(JetTokens.LT, JetTokens.GT, JetTokens.LTEQ, JetTokens.GTEQ)); private static final Set equalsOperations = new HashSet(Arrays.asList(JetTokens.EQEQ, JetTokens.EXCLEQ)); private static final Set inOperations = new HashSet(Arrays.asList(JetTokens.IN_KEYWORD, JetTokens.NOT_IN)); public static final Map assignmentOperationNames = new HashMap(); static { assignmentOperationNames.put(JetTokens.MULTEQ, "timesAssign"); assignmentOperationNames.put(JetTokens.DIVEQ, "divAssign"); assignmentOperationNames.put(JetTokens.PERCEQ, "modAssign"); assignmentOperationNames.put(JetTokens.PLUSEQ, "plusAssign"); assignmentOperationNames.put(JetTokens.MINUSEQ, "minusAssign"); } private static final Map assignmentOperationCounterparts = new HashMap(); static { assignmentOperationCounterparts.put(JetTokens.MULTEQ, JetTokens.MUL); assignmentOperationCounterparts.put(JetTokens.DIVEQ, JetTokens.DIV); assignmentOperationCounterparts.put(JetTokens.PERCEQ, JetTokens.PERC); assignmentOperationCounterparts.put(JetTokens.PLUSEQ, JetTokens.PLUS); assignmentOperationCounterparts.put(JetTokens.MINUSEQ, JetTokens.MINUS); } private final Map typeCache = new HashMap(); private final BindingTrace trace; private final JetSemanticServices semanticServices; private final TypeResolver typeResolver; private final ClassDescriptorResolver classDescriptorResolver; private final JetFlowInformationProvider flowInformationProvider; public JetTypeInferrer(@NotNull BindingTrace trace, @NotNull JetFlowInformationProvider flowInformationProvider, @NotNull JetSemanticServices semanticServices) { this.trace = new CachedBindingTrace(trace); this.semanticServices = semanticServices; this.typeResolver = new TypeResolver(trace, semanticServices); this.classDescriptorResolver = semanticServices.getClassDescriptorResolver(trace); this.flowInformationProvider = flowInformationProvider; } @NotNull public JetType safeGetType(@NotNull final JetScope scope, @NotNull JetExpression expression, final boolean preferBlock) { JetType type = getType(scope, expression, preferBlock); if (type != null) { return type; } return ErrorUtils.createErrorType("Type for " + expression.getText()); } @Nullable public JetType getType(@NotNull final JetScope scope, @NotNull JetExpression expression, final boolean preferBlock) { return new TypeInferrerVisitor(scope, preferBlock).getType(expression); } @Nullable private List getTypes(JetScope scope, List indexExpressions) { List argumentTypes = new ArrayList(); TypeInferrerVisitor typeInferrerVisitor = new TypeInferrerVisitor(scope, false); for (JetExpression indexExpression : indexExpressions) { JetType type = typeInferrerVisitor.getType(indexExpression); if (type == null) { return null; } argumentTypes.add(type); typeInferrerVisitor.resetResult(); // TODO : recreate? } return argumentTypes; } @Nullable private FunctionDescriptor lookupFunction( @NotNull JetScope scope, @NotNull JetReferenceExpression reference, @NotNull String name, @NotNull JetType receiverType, @NotNull List argumentTypes, boolean reportUnresolved) { OverloadDomain overloadDomain = semanticServices.getOverloadResolver().getOverloadDomain(receiverType, scope, name); overloadDomain = wrapForTracing(overloadDomain, reference, null, reportUnresolved); OverloadResolutionResult resolutionResult = overloadDomain.getFunctionDescriptorForPositionedArguments(Collections.emptyList(), argumentTypes); return resolutionResult.isSuccess() ? resolutionResult.getFunctionDescriptor() : null; } private OverloadDomain getOverloadDomain( @NotNull final JetScope scope, @NotNull JetExpression calleeExpression, @Nullable PsiElement argumentList ) { final OverloadDomain[] result = new OverloadDomain[1]; final JetSimpleNameExpression[] reference = new JetSimpleNameExpression[1]; calleeExpression.accept(new JetVisitor() { @Override public void visitHashQualifiedExpression(JetHashQualifiedExpression expression) { // a#b -- create a domain for all overloads of b in a throw new UnsupportedOperationException(); // TODO } @Override public void visitPredicateExpression(JetPredicateExpression expression) { // overload lookup for checking, but the type is receiver's type + nullable throw new UnsupportedOperationException(); // TODO } @Override public void visitQualifiedExpression(JetQualifiedExpression expression) { // . or ?. JetType receiverType = getType(scope, expression.getReceiverExpression(), false); checkNullSafety(receiverType, expression); JetExpression selectorExpression = expression.getSelectorExpression(); if (selectorExpression instanceof JetSimpleNameExpression) { JetSimpleNameExpression referenceExpression = (JetSimpleNameExpression) selectorExpression; String referencedName = referenceExpression.getReferencedName(); if (receiverType != null && referencedName != null) { result[0] = semanticServices.getOverloadResolver().getOverloadDomain(receiverType, scope, referencedName); reference[0] = referenceExpression; } } else { throw new UnsupportedOperationException(); // TODO } } @Override public void visitSimpleNameExpression(JetSimpleNameExpression expression) { // a -- create a hierarchical lookup domain for this.a String referencedName = expression.getReferencedName(); if (referencedName != null) { result[0] = semanticServices.getOverloadResolver().getOverloadDomain(null, scope, referencedName); reference[0] = expression; } } @Override public void visitExpression(JetExpression expression) { // create a dummy domain for the type of e throw new UnsupportedOperationException(expression.getText()); // TODO } @Override public void visitJetElement(JetElement elem) { semanticServices.getErrorHandler().genericError(elem.getNode(), "Unsupported in call element"); // TODO : Message } }); return wrapForTracing(result[0], reference[0], argumentList, true); } private void checkNullSafety(JetType receiverType, JetQualifiedExpression expression) { if (receiverType != null) { boolean namespaceType = receiverType instanceof NamespaceType; boolean nullable = !namespaceType && receiverType.isNullable(); if (nullable && expression.getOperationSign() == JetTokens.DOT) { semanticServices.getErrorHandler().genericError(expression.getOperationTokenNode(), "Only safe calls (?.) are allowed on a nullable receiver of type " + receiverType); } else if (!nullable && expression.getOperationSign() == JetTokens.SAFE_ACCESS) { if (namespaceType) { semanticServices.getErrorHandler().genericError(expression.getOperationTokenNode(), "Safe calls are not allowed on namespaces"); } else { semanticServices.getErrorHandler().genericWarning(expression.getOperationTokenNode(), "Unnecessary safe call on a non-null receiver of type " + receiverType); } } } } private OverloadDomain wrapForTracing( @Nullable final OverloadDomain overloadDomain, final JetReferenceExpression referenceExpression, @Nullable final PsiElement argumentList, final boolean reportErrors) { if (overloadDomain == null) return OverloadDomain.EMPTY; assert referenceExpression != null; return new OverloadDomain() { @NotNull @Override public OverloadResolutionResult getFunctionDescriptorForNamedArguments(@NotNull List typeArguments, @NotNull Map valueArgumentTypes, @Nullable JetType functionLiteralArgumentType) { OverloadResolutionResult resolutionResult = overloadDomain.getFunctionDescriptorForNamedArguments(typeArguments, valueArgumentTypes, functionLiteralArgumentType); report(resolutionResult); return resolutionResult; } @NotNull @Override public OverloadResolutionResult getFunctionDescriptorForPositionedArguments(@NotNull List typeArguments, @NotNull List positionedValueArgumentTypes) { OverloadResolutionResult resolutionResult = overloadDomain.getFunctionDescriptorForPositionedArguments(typeArguments, positionedValueArgumentTypes); report(resolutionResult); return resolutionResult; } private void report(OverloadResolutionResult resolutionResult) { if (resolutionResult.isSuccess() || resolutionResult.singleFunction()) { trace.recordReferenceResolution(referenceExpression, resolutionResult.getFunctionDescriptor()); } if (reportErrors) { switch (resolutionResult.getResultCode()) { case NAME_NOT_FOUND: semanticServices.getErrorHandler().unresolvedReference(referenceExpression); break; case SINGLE_FUNCTION_ARGUMENT_MISMATCH: if (argumentList != null) { // TODO : More helpful message. NOTE: there's a separate handling for this for constructors semanticServices.getErrorHandler().genericError(argumentList.getNode(), "Arguments do not match " + DescriptorUtil.renderPresentableText(resolutionResult.getFunctionDescriptor())); } else { semanticServices.getErrorHandler().unresolvedReference(referenceExpression); } break; case AMBIGUITY: if (argumentList != null) { // TODO : More helpful message. NOTE: there's a separate handling for this for constructors semanticServices.getErrorHandler().genericError(argumentList.getNode(), "Overload ambiguity [TODO : more helpful message]"); } else { semanticServices.getErrorHandler().unresolvedReference(referenceExpression); } break; default: // Not a success } } } @Override public boolean isEmpty() { return overloadDomain.isEmpty(); } }; } @NotNull public JetType getFunctionReturnType(@NotNull JetScope outerScope, JetFunction function, FunctionDescriptor functionDescriptor) { Map typeMap = getReturnedExpressions(outerScope, function, functionDescriptor); Collection types = typeMap.values(); return types.isEmpty() ? JetStandardClasses.getNothingType() : semanticServices.getTypeChecker().commonSupertype(types); } private JetType getCachedType(@NotNull JetExpression expression) { // assert typeCache.containsKey(expression) : "No type cached for " + expression.getText(); return typeCache.get(expression); } public void checkFunctionReturnType(@NotNull JetScope outerScope, @NotNull JetFunction function, @NotNull FunctionDescriptor functionDescriptor) { Map typeMap = getReturnedExpressions(outerScope, function, functionDescriptor); if (typeMap.isEmpty()) { return; // The function returns Nothing } JetType expectedReturnType = functionDescriptor.getUnsubstitutedReturnType(); for (Map.Entry entry : typeMap.entrySet()) { JetType actualType = entry.getValue(); JetElement element = entry.getKey(); JetTypeChecker typeChecker = semanticServices.getTypeChecker(); if (!typeChecker.isSubtypeOf(actualType, expectedReturnType)) { if (typeChecker.isConvertibleBySpecialConversion(actualType, expectedReturnType)) { if (expectedReturnType.getConstructor().equals(JetStandardClasses.getUnitType().getConstructor()) && element.getParent() instanceof JetReturnExpression) { semanticServices.getErrorHandler().genericError(element.getNode(), "This function must return a value of type Unit"); } } else { if (element == function) { JetExpression bodyExpression = function.getBodyExpression(); assert bodyExpression != null; semanticServices.getErrorHandler().genericError(bodyExpression.getNode(), "This function must return a value of type " + expectedReturnType); } else if (element instanceof JetExpression) { JetExpression expression = (JetExpression) element; semanticServices.getErrorHandler().typeMismatch(expression, expectedReturnType, actualType); } else { semanticServices.getErrorHandler().genericError(element.getNode(), "This function must return a value of type " + expectedReturnType); } } } } } private Map getReturnedExpressions(JetScope outerScope, JetFunction function, FunctionDescriptor functionDescriptor) { JetExpression bodyExpression = function.getBodyExpression(); assert bodyExpression != null; JetScope functionInnerScope = FunctionDescriptorUtil.getFunctionInnerScope(outerScope, functionDescriptor, semanticServices); getType(functionInnerScope, bodyExpression, function.hasBlockBody()); Collection returnedExpressions = new ArrayList(); Collection elementsReturningUnit = new ArrayList(); flowInformationProvider.collectReturnedInformation(function, returnedExpressions, elementsReturningUnit); Map typeMap = new HashMap(); for (JetExpression returnedExpression : returnedExpressions) { JetType cachedType = getCachedType(returnedExpression); if (cachedType != null) { typeMap.put(returnedExpression, cachedType); } } for (JetElement jetElement : elementsReturningUnit) { typeMap.put(jetElement, JetStandardClasses.getUnitType()); } return typeMap; } @Nullable private JetType getBlockReturnedType(@NotNull JetScope outerScope, @NotNull List block) { if (block.isEmpty()) { return JetStandardClasses.getUnitType(); } DeclarationDescriptor containingDescriptor = outerScope.getContainingDeclaration(); WritableScope scope = semanticServices.createWritableScope(outerScope, containingDescriptor); return getBlockReturnedTypeWithWritableScope(scope, block); } private JetType getBlockReturnedTypeWithWritableScope(@NotNull WritableScope scope, @NotNull List block) { assert !block.isEmpty(); TypeInferrerVisitorWithWritableScope blockLevelVisitor = new TypeInferrerVisitorWithWritableScope(scope, true); JetType result = null; for (JetElement statement : block) { result = blockLevelVisitor.getType((JetExpression) statement); blockLevelVisitor.resetResult(); // TODO : maybe it's better to recreate the visitors with the same scope? } return result; } private void collectAllReturnTypes(JetWhenExpression whenExpression, JetScope scope, List result) { for (JetWhenEntry entry : whenExpression.getEntries()) { JetWhenExpression subWhen = entry.getSubWhen(); if (subWhen != null) { collectAllReturnTypes(subWhen, scope, result); } else { JetExpression resultExpression = entry.getExpression(); if (resultExpression != null) { result.add(getType(scope, resultExpression, true)); } } } } private class TypeInferrerVisitor extends JetVisitor { private final JetScope scope; private final boolean preferBlock; protected JetType result; private TypeInferrerVisitor(@NotNull JetScope scope, boolean preferBlock) { this.scope = scope; this.preferBlock = preferBlock; } @Nullable public JetType getType(@NotNull final JetScope scope, @NotNull JetExpression expression, final boolean preferBlock) { TypeInferrerVisitor visitor; if (this.scope == scope && this.preferBlock == preferBlock && result == null) { visitor = this; } else { visitor = new TypeInferrerVisitor(scope, preferBlock); } JetType type = visitor.getType(expression); visitor.result = null; return type; } @Nullable public JetType getType(@NotNull JetExpression expression) { assert result == null; expression.accept(this); if (result != null) { trace.recordExpressionType(expression, result); } return result; } public void resetResult() { result = null; } //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// @Override public void visitSimpleNameExpression(JetSimpleNameExpression expression) { // TODO : other members // TODO : type substitutions??? String referencedName = expression.getReferencedName(); if (referencedName != null) { PropertyDescriptor property = scope.getProperty(referencedName); if (property != null) { trace.recordReferenceResolution(expression, property); result = property.getOutType(); if (result == null) { semanticServices.getErrorHandler().genericError(expression.getNode(), "This property is not readable in this context"); } return; } else { NamespaceDescriptor namespace = scope.getNamespace(referencedName); if (namespace != null) { trace.recordReferenceResolution(expression, namespace); result = namespace.getNamespaceType(); return; } } semanticServices.getErrorHandler().unresolvedReference(expression); } } @Override public void visitFunctionLiteralExpression(JetFunctionLiteralExpression expression) { if (preferBlock && !expression.hasParameterSpecification()) { trace.recordBlock(expression); result = getBlockReturnedType(scope, expression.getBody()); return; } FunctionDescriptorImpl functionDescriptor = new FunctionDescriptorImpl(scope.getContainingDeclaration(), Collections.emptyList(), ""); JetTypeReference returnTypeRef = expression.getReturnTypeRef(); JetTypeReference receiverTypeRef = expression.getReceiverTypeRef(); final JetType receiverType; if (receiverTypeRef != null) { receiverType = typeResolver.resolveType(scope, receiverTypeRef); } else { receiverType = scope.getThisType(); } List body = expression.getBody(); final Map parameterDescriptors = new HashMap(); List parameterTypes = new ArrayList(); for (JetParameter parameter : expression.getParameters()) { JetTypeReference typeReference = parameter.getTypeReference(); if (typeReference == null) { throw new UnsupportedOperationException("Type inference for parameters is not implemented yet"); } PropertyDescriptor propertyDescriptor = classDescriptorResolver.resolveValueParameterDescriptor(functionDescriptor, scope, parameter); parameterDescriptors.put(parameter.getName(), propertyDescriptor); parameterTypes.add(propertyDescriptor.getOutType()); } JetType returnType; if (returnTypeRef != null) { returnType = typeResolver.resolveType(scope, returnTypeRef); } else { WritableScope writableScope = semanticServices.createWritableScope(scope, functionDescriptor); for (PropertyDescriptor propertyDescriptor : parameterDescriptors.values()) { writableScope.addPropertyDescriptor(propertyDescriptor); } writableScope.setThisType(receiverType); returnType = getBlockReturnedType(writableScope, body); } JetType effectiveReceiverType = receiverTypeRef == null ? null : receiverType; JetType safeReturnType = returnType == null ? ErrorUtils.createErrorType("") : returnType; result = JetStandardClasses.getFunctionType(null, effectiveReceiverType, parameterTypes, safeReturnType); } @Override public void visitParenthesizedExpression(JetParenthesizedExpression expression) { JetExpression inner = expression.getExpression(); if (inner != null) { result = getType(scope, inner, false); } } @Override public void visitConstantExpression(JetConstantExpression expression) { IElementType elementType = expression.getNode().getElementType(); JetStandardLibrary standardLibrary = semanticServices.getStandardLibrary(); if (elementType == JetNodeTypes.INTEGER_CONSTANT) { if (expression.getValue() instanceof Long) { result = standardLibrary.getLongType(); } else { result = standardLibrary.getIntType(); } // TODO : other ranges } else if (elementType == JetNodeTypes.LONG_CONSTANT) { result = standardLibrary.getLongType(); } else if (elementType == JetNodeTypes.FLOAT_CONSTANT) { String text = expression.getText(); assert text.length() > 0; char lastChar = text.charAt(text.length() - 1); if (lastChar == 'f' || lastChar == 'F') { result = standardLibrary.getFloatType(); } else { result = standardLibrary.getDoubleType(); } } else if (elementType == JetNodeTypes.BOOLEAN_CONSTANT) { result = standardLibrary.getBooleanType(); } else if (elementType == JetNodeTypes.CHARACTER_CONSTANT) { result = standardLibrary.getCharType(); } else if (elementType == JetNodeTypes.STRING_CONSTANT) { result = standardLibrary.getStringType(); } else if (elementType == JetNodeTypes.NULL) { result = JetStandardClasses.getNullableNothingType(); } else { throw new IllegalArgumentException("Unsupported constant: " + expression); } } @Override public void visitThrowExpression(JetThrowExpression expression) { result = JetStandardClasses.getNothingType(); } @Override public void visitReturnExpression(JetReturnExpression expression) { JetExpression returnedExpression = expression.getReturnedExpression(); JetType returnedType; if (returnedExpression != null) { returnedType = getType(scope, returnedExpression, false); } else { returnedType = JetStandardClasses.getUnitType(); } result = JetStandardClasses.getNothingType(); } @Override public void visitBreakExpression(JetBreakExpression expression) { result = JetStandardClasses.getNothingType(); } @Override public void visitContinueExpression(JetContinueExpression expression) { result = JetStandardClasses.getNothingType(); } @Override public void visitTypeofExpression(JetTypeofExpression expression) { throw new UnsupportedOperationException("Return some reflection interface"); // TODO } @Override public void visitBinaryWithTypeRHSExpression(JetBinaryExpressionWithTypeRHS expression) { IElementType operationType = expression.getOperationSign().getReferencedNameElementType(); JetType actualType = getType(scope, expression.getLeft(), false); JetTypeReference right = expression.getRight(); if (right != null) { JetType targetType = typeResolver.resolveType(scope, right); if (operationType == JetTokens.COLON) { if (actualType != null && !semanticServices.getTypeChecker().isSubtypeOf(actualType, targetType)) { semanticServices.getErrorHandler().typeMismatch(expression.getLeft(), targetType, actualType); } } else if (operationType == JetTokens.AS_KEYWORD) { // TODO : Check for cast impossibility } else { semanticServices.getErrorHandler().genericError(expression.getOperationSign().getNode(), "Unsupported binary operation"); } result = targetType; } } @Override public void visitTupleExpression(JetTupleExpression expression) { List entries = expression.getEntries(); List types = new ArrayList(); for (JetExpression entry : entries) { types.add(getType(scope, entry, false)); } // TODO : labels result = JetStandardClasses.getTupleType(types); } @Override public void visitThisExpression(JetThisExpression expression) { // TODO : qualified this, e.g. this@Foo JetType thisType = null; String labelName = expression.getLabelName(); if (labelName != null) { Collection declarationsByLabel = scope.getDeclarationsByLabel(labelName); int size = declarationsByLabel.size(); if (size == 1) { DeclarationDescriptor declarationDescriptor = declarationsByLabel.iterator().next(); if (declarationDescriptor instanceof ClassDescriptor) { ClassDescriptor classDescriptor = (ClassDescriptor) declarationDescriptor; thisType = classDescriptor.getDefaultType(); } else { throw new UnsupportedOperationException(); // TODO } } else if (size == 0) { semanticServices.getErrorHandler().unresolvedReference(expression.getTargetLabel()); } else { JetSimpleNameExpression labelElement = expression.getTargetLabel(); assert labelElement != null; semanticServices.getErrorHandler().genericError(labelElement.getNode(), "Ambiguous label"); } } else { thisType = scope.getThisType(); } if (thisType != null) { if (JetStandardClasses.isNothing(thisType)) { semanticServices.getErrorHandler().genericError(expression.getNode(), "'this' is not defined in this context"); } else { JetTypeReference superTypeQualifier = expression.getSuperTypeQualifier(); if (superTypeQualifier != null) { JetTypeElement superTypeElement = superTypeQualifier.getTypeElement(); // Errors are reported by the parser if (superTypeElement instanceof JetUserType) { JetUserType typeElement = (JetUserType) superTypeElement; ClassifierDescriptor classifierCandidate = typeResolver.resolveClass(scope, typeElement); if (classifierCandidate instanceof ClassDescriptor) { ClassDescriptor superclass = (ClassDescriptor) classifierCandidate; Collection supertypes = thisType.getConstructor().getSupertypes(); Map substitutionContext = TypeUtils.buildSubstitutionContext(thisType); for (JetType declaredSupertype : supertypes) { if (declaredSupertype.getConstructor().equals(superclass.getTypeConstructor())) { result = TypeSubstitutor.INSTANCE.safeSubstitute(substitutionContext, declaredSupertype, Variance.INVARIANT); break; } } if (result == null) { semanticServices.getErrorHandler().genericError(superTypeElement.getNode(), "Not a superclass"); } } } } else { result = thisType; } } } } @Override public void visitBlockExpression(JetBlockExpression expression) { result = getBlockReturnedType(scope, expression.getStatements()); } @Override public void visitWhenExpression(JetWhenExpression expression) { // TODO :change scope according to the bound value in the when header List expressions = new ArrayList(); collectAllReturnTypes(expression, scope, expressions); result = semanticServices.getTypeChecker().commonSupertype(expressions); } @Override public void visitTryExpression(JetTryExpression expression) { JetExpression tryBlock = expression.getTryBlock(); List catchClauses = expression.getCatchClauses(); JetFinallySection finallyBlock = expression.getFinallyBlock(); List types = new ArrayList(); if (finallyBlock == null) { for (JetCatchClause catchClause : catchClauses) { // TODO: change scope here JetExpression catchBody = catchClause.getCatchBody(); if (catchBody != null) { types.add(getType(scope, catchBody, true)); } } } else { types.add(getType(scope, finallyBlock.getFinalExpression(), true)); } types.add(getType(scope, tryBlock, true)); result = semanticServices.getTypeChecker().commonSupertype(types); } @Override public void visitIfExpression(JetIfExpression expression) { checkCondition(scope, expression.getCondition()); // TODO : change types according to is and null checks JetExpression elseBranch = expression.getElse(); JetExpression thenBranch = expression.getThen(); if (elseBranch == null) { if (thenBranch != null) { getType(scope, thenBranch, true); result = JetStandardClasses.getUnitType(); } } else if (thenBranch == null) { getType(scope, elseBranch, true); result = JetStandardClasses.getUnitType(); } else { JetType thenType = getType(scope, thenBranch, true); JetType elseType = getType(scope, elseBranch, true); if (thenType == null) { result = elseType; } else if (elseType == null) { result = thenType; } else { result = semanticServices.getTypeChecker().commonSupertype(Arrays.asList(thenType, elseType)); } } } private void checkCondition(@NotNull JetScope scope, @Nullable JetExpression condition) { if (condition != null) { JetType conditionType = getType(scope, condition, false); if (conditionType != null && !isBoolean(conditionType)) { semanticServices.getErrorHandler().genericError(condition.getNode(), "Condition must be of type Boolean, but was of type " + conditionType); } } } @Override public void visitWhileExpression(JetWhileExpression expression) { checkCondition(scope, expression.getCondition()); JetExpression body = expression.getBody(); if (body != null) { getType(scope, body, true); } result = JetStandardClasses.getUnitType(); } @Override public void visitDoWhileExpression(JetDoWhileExpression expression) { JetExpression body = expression.getBody(); JetScope conditionScope = scope; if (body instanceof JetFunctionLiteralExpression) { JetFunctionLiteralExpression function = (JetFunctionLiteralExpression) body; if (!function.hasParameterSpecification()) { WritableScope writableScope = semanticServices.createWritableScope(scope, scope.getContainingDeclaration()); conditionScope = writableScope; getBlockReturnedTypeWithWritableScope(writableScope, function.getBody()); trace.recordBlock(function); } else { getType(scope, body, true); } } else if (body != null) { WritableScope writableScope = semanticServices.createWritableScope(scope, scope.getContainingDeclaration()); conditionScope = writableScope; getBlockReturnedTypeWithWritableScope(writableScope, Collections.singletonList(body)); } checkCondition(conditionScope, expression.getCondition()); result = JetStandardClasses.getUnitType(); } @Override public void visitForExpression(JetForExpression expression) { JetParameter loopParameter = expression.getLoopParameter(); JetExpression loopRange = expression.getLoopRange(); JetType loopRangeType = getType(scope, loopRange, false); JetType expectedParameterType = null; if (loopRangeType != null) { if (!semanticServices.getTypeChecker().isSubtypeOf(loopRangeType, semanticServices.getStandardLibrary().getIterableType(JetStandardClasses.getNullableAnyType()))) { semanticServices.getErrorHandler().genericError(loopRange.getNode(), "Expecting an Iterable, but found " + loopRangeType); } else { TypeProjection typeProjection = loopRangeType.getArguments().get(0); if (!typeProjection.getProjectionKind().allowsOutPosition()) { expectedParameterType = JetStandardClasses.getDefaultBound(); } else { expectedParameterType = typeProjection.getType(); } } } WritableScope loopScope = semanticServices.createWritableScope(scope, scope.getContainingDeclaration()); JetTypeReference typeReference = loopParameter.getTypeReference(); PropertyDescriptor propertyDescriptor; if (typeReference != null) { propertyDescriptor = classDescriptorResolver.resolveValueParameterDescriptor(scope.getContainingDeclaration(), scope, loopParameter); JetType actualParameterType = propertyDescriptor.getOutType(); if (expectedParameterType != null && actualParameterType != null && !semanticServices.getTypeChecker().isSubtypeOf(expectedParameterType, actualParameterType)) { semanticServices.getErrorHandler().genericError(typeReference.getNode(), "The loop iterates over values of type " + expectedParameterType + " but the parameter is declared to be " + actualParameterType); } } else { if (expectedParameterType == null) { expectedParameterType = ErrorUtils.createErrorType("Error"); } propertyDescriptor = classDescriptorResolver.resolveValueParameterDescriptor(scope.getContainingDeclaration(), loopParameter, expectedParameterType); } loopScope.addPropertyDescriptor(propertyDescriptor); JetExpression body = expression.getBody(); if (body != null) { getType(loopScope, body, true); // TODO } result = JetStandardClasses.getUnitType(); } @Override public void visitNewExpression(JetNewExpression expression) { // TODO : type argument inference JetTypeReference typeReference = expression.getTypeReference(); if (typeReference != null) { JetTypeElement typeElement = typeReference.getTypeElement(); if (typeElement instanceof JetUserType) { JetUserType userType = (JetUserType) typeElement; // TODO : to infer constructor parameters, one will need to // 1) resolve a _class_ from the typeReference // 2) rely on the overload domain of constructors of this class to infer type arguments // For now we assume that the type arguments are provided, and thus the typeReference can be // resolved into a valid type JetType receiverType = typeResolver.resolveType(scope, typeReference); DeclarationDescriptor declarationDescriptor = receiverType.getConstructor().getDeclarationDescriptor(); if (declarationDescriptor instanceof ClassDescriptor) { ClassDescriptor classDescriptor = (ClassDescriptor) declarationDescriptor; for (JetTypeProjection typeProjection : userType.getTypeArguments()) { switch (typeProjection.getProjectionKind()) { case IN: case OUT: case STAR: // TODO : Bug in the editor semanticServices.getErrorHandler().genericError(typeProjection.getProjectionNode(), "Projections are not allowed in constructor type arguments"); break; case NONE: break; } } JetSimpleNameExpression referenceExpression = userType.getReferenceExpression(); if (referenceExpression != null) { // When one writes 'new Array(...)' this does not make much sense, and an instance // of 'Array' must be created anyway. // Thus, we should either prohibit projections in type arguments in such contexts, // or treat them as an automatic upcast to the desired type, i.e. for the user not // to be forced to write // val a : Array = new Array(...) // NOTE: Array may be a bad example here, some classes may have substantial functionality // not involving their type parameters // // The code below upcasts the type automatically List typeArguments = receiverType.getArguments(); List projectionsStripped = new ArrayList(); for (TypeProjection typeArgument : typeArguments) { if (typeArgument.getProjectionKind() != Variance.INVARIANT) { projectionsStripped.add(new TypeProjection(typeArgument.getType())); } else projectionsStripped.add(typeArgument); } FunctionGroup constructors = classDescriptor.getConstructors(projectionsStripped); OverloadDomain constructorsOverloadDomain = semanticServices.getOverloadResolver().getOverloadDomain(constructors); JetType constructorReturnedType = resolveOverloads( scope, wrapForTracing(constructorsOverloadDomain, referenceExpression, expression.getArgumentList(), false), Collections.emptyList(), expression.getArguments(), expression.getFunctionLiteralArguments()); if (constructorReturnedType == null && !ErrorUtils.isErrorType(receiverType)) { trace.recordReferenceResolution(referenceExpression, receiverType.getConstructor().getDeclarationDescriptor()); // TODO : more helpful message JetArgumentList argumentList = expression.getArgumentList(); if (argumentList != null) { semanticServices.getErrorHandler().genericError(argumentList.getNode(), "Cannot find an overload for these arguments"); } constructorReturnedType = receiverType; } // If no upcast needed: result = constructorReturnedType; // Automatic upcast: // result = receiverType; } } else { semanticServices.getErrorHandler().genericError(expression.getNode(), "Calling a constructor is only supported for ordinary classes"); // TODO : review the message } } else { if (typeElement != null) { semanticServices.getErrorHandler().genericError(typeElement.getNode(), "Calling a constructor is only supported for ordinary classes"); // TODO : Better message } } } } @Override public void visitHashQualifiedExpression(JetHashQualifiedExpression expression) { throw new UnsupportedOperationException(); // TODO } @Override public void visitQualifiedExpression(JetQualifiedExpression expression) { // TODO : functions JetExpression selectorExpression = expression.getSelectorExpression(); JetExpression receiverExpression = expression.getReceiverExpression(); JetType receiverType = getType(scope, receiverExpression, false); if (receiverType != null) { checkNullSafety(receiverType, expression); JetType selectorReturnType = getSelectorReturnType(receiverType, selectorExpression); if (expression.getOperationSign() == JetTokens.QUEST) { if (selectorReturnType != null && !isBoolean(selectorReturnType)) { // TODO : more comprehensible error message semanticServices.getErrorHandler().typeMismatch(selectorExpression, semanticServices.getStandardLibrary().getBooleanType(), selectorReturnType); } result = TypeUtils.makeNullable(receiverType); } else { result = selectorReturnType; } if (selectorExpression != null && result != null) { trace.recordExpressionType(selectorExpression, result); } } } private JetType getSelectorReturnType(JetType receiverType, JetExpression selectorExpression) { JetScope compositeScope = new ScopeWithReceiver(scope, receiverType); if (selectorExpression instanceof JetCallExpression) { JetCallExpression callExpression = (JetCallExpression) selectorExpression; OverloadDomain overloadDomain = getOverloadDomain(compositeScope, callExpression.getCalleeExpression(), callExpression.getValueArgumentList()); return resolveOverloads(scope, callExpression, overloadDomain); } else if (selectorExpression instanceof JetSimpleNameExpression) { return getType(compositeScope, selectorExpression, false); } else if (selectorExpression != null) { // TODO : not a simple name -> resolve in scope, expect property type or a function type semanticServices.getErrorHandler().genericError(selectorExpression.getNode(), "Unsupported selector element type: " + selectorExpression); } return receiverType; } @Override public void visitCallExpression(JetCallExpression expression) { JetExpression calleeExpression = expression.getCalleeExpression(); OverloadDomain overloadDomain = getOverloadDomain(scope, calleeExpression, expression.getValueArgumentList()); result = resolveOverloads(scope, expression, overloadDomain); } @Nullable private JetType resolveOverloads(JetScope scope, JetCallExpression expression, OverloadDomain overloadDomain) { List typeArguments = expression.getTypeArguments(); List valueArguments = expression.getValueArguments(); List functionLiteralArguments = expression.getFunctionLiteralArguments(); return resolveOverloads(scope, overloadDomain, typeArguments, valueArguments, functionLiteralArguments); } @Nullable private JetType resolveOverloads( @NotNull JetScope scope, @NotNull OverloadDomain overloadDomain, @NotNull List typeArguments, @NotNull List valueArguments, @NotNull List functionLiteralArguments) { // 1) ends with a name -> (scope, name) to look up // 2) ends with something else -> just check types for (JetTypeProjection typeArgument : typeArguments) { if (typeArgument.getProjectionKind() != JetProjectionKind.NONE) { semanticServices.getErrorHandler().genericError(typeArgument.getNode(), "Projections are not allowed on type parameters for methods"); // TODO : better positioning } } boolean someNamed = false; for (JetArgument argument : valueArguments) { if (argument.isNamed()) { someNamed = true; break; } } // JetExpression functionLiteralArgument = functionLiteralArguments.isEmpty() ? null : functionLiteralArguments.get(0); // TODO : must be a check assert functionLiteralArguments.size() <= 1; if (someNamed) { // TODO : check that all are named throw new UnsupportedOperationException(); // TODO // result = overloadDomain.getFunctionDescriptorForNamedArguments(typeArguments, valueArguments, functionLiteralArgument); } else { List types = new ArrayList(); for (JetTypeProjection projection : typeArguments) { // TODO : check that there's no projection JetTypeReference typeReference = projection.getTypeReference(); if (typeReference != null) { types.add(typeResolver.resolveType(scope, typeReference)); } } List positionedValueArguments = new ArrayList(); for (JetArgument argument : valueArguments) { JetExpression argumentExpression = argument.getArgumentExpression(); if (argumentExpression != null) { positionedValueArguments.add(argumentExpression); } } positionedValueArguments.addAll(functionLiteralArguments); List valueArgumentTypes = new ArrayList(); for (JetExpression valueArgument : positionedValueArguments) { valueArgumentTypes.add(safeGetType(scope, valueArgument, false)); } OverloadResolutionResult resolutionResult = overloadDomain.getFunctionDescriptorForPositionedArguments(types, valueArgumentTypes); if (resolutionResult.isSuccess()) { return resolutionResult.getFunctionDescriptor().getUnsubstitutedReturnType(); } } return null; } @Override public void visitIsExpression(JetIsExpression expression) { // TODO : patterns and everything result = semanticServices.getStandardLibrary().getBooleanType(); } @Override public void visitUnaryExpression(JetUnaryExpression expression) { JetSimpleNameExpression operationSign = expression.getOperationSign(); if (JetTokens.LABELS.contains(operationSign.getReferencedNameElementType())) { // TODO : Some processing for the label? result = getType(expression.getBaseExpression()); return; } IElementType operationType = operationSign.getReferencedNameElementType(); String name = unaryOperationNames.get(operationType); if (name == null) { semanticServices.getErrorHandler().genericError(operationSign.getNode(), "Unknown unary operation"); } else { JetType receiverType = getType(scope, expression.getBaseExpression(), false); if (receiverType != null) { FunctionDescriptor functionDescriptor = lookupFunction(scope, expression.getOperationSign(), name, receiverType, Collections.emptyList(), true); if (functionDescriptor != null) { JetType returnType = functionDescriptor.getUnsubstitutedReturnType(); if (operationType == JetTokens.PLUSPLUS || operationType == JetTokens.MINUSMINUS) { if (!semanticServices.getTypeChecker().isSubtypeOf(returnType, receiverType)) { semanticServices.getErrorHandler().genericError(operationSign.getNode(), name + " must return " + receiverType + " but returns " + returnType); } // TODO : Maybe returnType? result = receiverType; } else { result = returnType; } } } } } @Override public void visitBinaryExpression(JetBinaryExpression expression) { JetSimpleNameExpression operationSign = expression.getOperationReference(); JetExpression left = expression.getLeft(); JetExpression right = expression.getRight(); IElementType operationType = operationSign.getReferencedNameElementType(); if (operationType == JetTokens.IDENTIFIER) { String referencedName = operationSign.getReferencedName(); if (referencedName != null) { result = getTypeForBinaryCall(expression, referencedName, scope, true); } } else if (binaryOperationNames.containsKey(operationType)) { result = getTypeForBinaryCall(expression, binaryOperationNames.get(operationType), scope, true); } else if (operationType == JetTokens.EQ) { visitAssignment(expression); } else if (assignmentOperationNames.containsKey(operationType)) { visitAssignmentOperation(expression); } else if (comparisonOperations.contains(operationType)) { JetType compareToReturnType = getTypeForBinaryCall(expression, "compareTo", scope, true); if (compareToReturnType != null) { TypeConstructor constructor = compareToReturnType.getConstructor(); JetStandardLibrary standardLibrary = semanticServices.getStandardLibrary(); TypeConstructor intTypeConstructor = standardLibrary.getInt().getTypeConstructor(); if (constructor.equals(intTypeConstructor)) { result = standardLibrary.getBooleanType(); } else { semanticServices.getErrorHandler().genericError(operationSign.getNode(), "compareTo must return Int, but returns " + compareToReturnType); } } } else if (equalsOperations.contains(operationType)) { String name = "equals"; if (right != null) { JetType leftType = getType(scope, left, false); if (leftType != null) { JetType rightType = getType(scope, right, false); if (rightType != null) { FunctionDescriptor equals = lookupFunction( scope, operationSign, "equals", leftType, Collections.singletonList(JetStandardClasses.getNullableAnyType()), false); if (equals != null) { if (ensureBooleanResult(operationSign, name, equals.getUnsubstitutedReturnType())) { ensureNonemptyIntersectionOfOperandTypes(expression); } } else { semanticServices.getErrorHandler().genericError(operationSign.getNode(), "No method 'equals(Any?) : Boolean' available"); } } } } result = semanticServices.getStandardLibrary().getBooleanType(); } else if (operationType == JetTokens.EQEQEQ || operationType == JetTokens.EXCLEQEQEQ) { ensureNonemptyIntersectionOfOperandTypes(expression); // TODO : Check comparison pointlessness result = semanticServices.getStandardLibrary().getBooleanType(); } else if (inOperations.contains(operationType)) { if (right == null) { result = ErrorUtils.createErrorType("No right argument"); // TODO return; } String name = "contains"; JetType containsType = getTypeForBinaryCall(scope, right, expression.getOperationReference(), expression.getLeft(), name, true); ensureBooleanResult(operationSign, name, containsType); result = semanticServices.getStandardLibrary().getBooleanType(); } else if (operationType == JetTokens.ANDAND || operationType == JetTokens.OROR) { JetType leftType = getType(scope, left, false); JetType rightType = right == null ? null : getType(scope, right, false); if (leftType != null && !isBoolean(leftType)) { semanticServices.getErrorHandler().typeMismatch(left, semanticServices.getStandardLibrary().getBooleanType(), leftType); } if (rightType != null && !isBoolean(rightType)) { semanticServices.getErrorHandler().typeMismatch(right, semanticServices.getStandardLibrary().getBooleanType(), rightType); } result = semanticServices.getStandardLibrary().getBooleanType(); } else if (operationType == JetTokens.ELVIS) { JetType leftType = getType(scope, left, false); JetType rightType = right == null ? null : getType(scope, right, false); if (leftType != null) { if (!leftType.isNullable()) { semanticServices.getErrorHandler().genericWarning(left.getNode(), "Elvis operator (?:) is always returns the left operand of non-nullable type " + leftType); } if (rightType != null) { result = TypeUtils.makeNullableAsSpecified(semanticServices.getTypeChecker().commonSupertype(leftType, rightType), rightType.isNullable()); } } } else { semanticServices.getErrorHandler().genericError(operationSign.getNode(), "Unknown operation"); } } private void ensureNonemptyIntersectionOfOperandTypes(JetBinaryExpression expression) { JetSimpleNameExpression operationSign = expression.getOperationReference(); JetExpression left = expression.getLeft(); JetExpression right = expression.getRight(); // TODO : duplicated effort for == and != JetType leftType = getType(scope, left, false); if (leftType != null && right != null) { JetType rightType = getType(scope, right, false); if (rightType != null) { JetType intersect = TypeUtils.intersect(semanticServices.getTypeChecker(), new HashSet(Arrays.asList(leftType, rightType))); if (intersect == null) { semanticServices.getErrorHandler().genericError(expression.getNode(), "Operator " + operationSign.getReferencedName() + " cannot be applied to " + leftType + " and " + rightType); } } } } protected void visitAssignmentOperation(JetBinaryExpression expression) { assignmentIsNotAnExpressionError(expression); } protected void visitAssignment(JetBinaryExpression expression) { assignmentIsNotAnExpressionError(expression); } private void assignmentIsNotAnExpressionError(JetBinaryExpression expression) { semanticServices.getErrorHandler().genericError(expression.getNode(), "Assignments are not expressions, and only expressions are allowed in this context"); } private boolean ensureBooleanResult(JetSimpleNameExpression operationSign, String name, JetType resultType) { if (resultType != null) { // TODO : Relax? if (!isBoolean(resultType)) { semanticServices.getErrorHandler().genericError(operationSign.getNode(), "'" + name + "' must return Boolean but returns " + resultType); return false; } } return true; } private boolean isBoolean(@NotNull JetType type) { return semanticServices.getTypeChecker().isConvertibleTo(type, semanticServices.getStandardLibrary().getBooleanType()); } @Override public void visitArrayAccessExpression(JetArrayAccessExpression expression) { JetExpression arrayExpression = expression.getArrayExpression(); JetType receiverType = getType(scope, arrayExpression, false); List indexExpressions = expression.getIndexExpressions(); List argumentTypes = getTypes(scope, indexExpressions); if (argumentTypes == null) return; if (receiverType != null) { FunctionDescriptor functionDescriptor = lookupFunction(scope, expression, "get", receiverType, argumentTypes, true); if (functionDescriptor != null) { result = functionDescriptor.getUnsubstitutedReturnType(); } } } @Nullable protected JetType getTypeForBinaryCall( @NotNull JetBinaryExpression expression, @NotNull String name, @NotNull JetScope scope, boolean reportUnresolved) { JetExpression left = expression.getLeft(); JetExpression right = expression.getRight(); if (right == null) { return null; } JetSimpleNameExpression operationSign = expression.getOperationReference(); return getTypeForBinaryCall(scope, left, operationSign, right, name, reportUnresolved); } @Nullable private JetType getTypeForBinaryCall( @NotNull JetScope scope, @NotNull JetExpression left, @NotNull JetSimpleNameExpression operationSign, @NotNull JetExpression right, @NotNull String name, boolean reportUnresolved) { JetType leftType = getType(scope, left, false); JetType rightType = getType(scope, right, false); if (leftType == null || rightType == null) { return null; } FunctionDescriptor functionDescriptor = lookupFunction(scope, operationSign, name, leftType, Collections.singletonList(rightType), reportUnresolved); if (functionDescriptor != null) { if (leftType.isNullable()) { // TODO : better error message for '1 + nullableVar' case semanticServices.getErrorHandler().genericError(operationSign.getNode(), "Infix call corresponds to a dot-qualified call '" + left.getText() + "." + name + "(" + right.getText() + ")'" + " which is not allowed on a nullable receiver '" + right.getText() + "'." + " Use '?.'-qualified call instead"); } return functionDescriptor.getUnsubstitutedReturnType(); } return null; } @Override public void visitDeclaration(JetDeclaration dcl) { semanticServices.getErrorHandler().genericError(dcl.getNode(), "Declarations are not allowed in this position"); } @Override public void visitJetElement(JetElement elem) { semanticServices.getErrorHandler().genericError(elem.getNode(), "Unsupported element: " + elem + " " + elem.getClass().getCanonicalName()); } } private class TypeInferrerVisitorWithWritableScope extends TypeInferrerVisitor { private final WritableScope scope; public TypeInferrerVisitorWithWritableScope(@NotNull WritableScope scope, boolean preferBlock) { super(scope, preferBlock); this.scope = scope; } @Override public void visitProperty(JetProperty property) { JetPropertyAccessor getter = property.getGetter(); if (getter != null) { semanticServices.getErrorHandler().genericError(getter.getNode(), "Local variables are not allowed to have getters"); } JetPropertyAccessor setter = property.getSetter(); if (setter != null) { semanticServices.getErrorHandler().genericError(setter.getNode(), "Local variables are not allowed to have setters"); } PropertyDescriptor propertyDescriptor = classDescriptorResolver.resolvePropertyDescriptor(scope.getContainingDeclaration(), scope, property); JetExpression initializer = property.getInitializer(); if (property.getPropertyTypeRef() != null && initializer != null) { JetType initializerType = getType(scope, initializer, false); JetType outType = propertyDescriptor.getOutType(); if (outType != null && initializerType != null && !semanticServices.getTypeChecker().isConvertibleTo(initializerType, outType)) { semanticServices.getErrorHandler().typeMismatch(initializer, outType, initializerType); } } scope.addPropertyDescriptor(propertyDescriptor); } @Override public void visitFunction(JetFunction function) { scope.addFunctionDescriptor(classDescriptorResolver.resolveFunctionDescriptor(scope.getContainingDeclaration(), scope, function)); } @Override public void visitClass(JetClass klass) { super.visitClass(klass); // TODO } @Override public void visitExtension(JetExtension extension) { super.visitExtension(extension); // TODO } @Override public void visitTypedef(JetTypedef typedef) { super.visitTypedef(typedef); // TODO } @Override public void visitDeclaration(JetDeclaration dcl) { visitJetElement(dcl); } @Override protected void visitAssignmentOperation(JetBinaryExpression expression) { IElementType operationType = expression.getOperationReference().getReferencedNameElementType(); String name = assignmentOperationNames.get(operationType); JetType assignmentOperationType = getTypeForBinaryCall(expression, name, scope, false); if (assignmentOperationType == null) { String counterpartName = binaryOperationNames.get(assignmentOperationCounterparts.get(operationType)); getTypeForBinaryCall(expression, counterpartName, scope, true); } result = null; // not an element } @Override protected void visitAssignment(JetBinaryExpression expression) { JetExpression left = expression.getLeft(); JetExpression deparenthesized = JetPsiUtil.deparenthesize(left); JetExpression right = expression.getRight(); if (deparenthesized instanceof JetArrayAccessExpression) { JetArrayAccessExpression arrayAccessExpression = (JetArrayAccessExpression) deparenthesized; resolveArrayAccessToLValue(arrayAccessExpression, right, expression.getOperationReference()); } else { JetType leftType = getType(scope, left, false); if (right != null) { JetType rightType = getType(scope, right, false); if (rightType != null && leftType != null && !semanticServices.getTypeChecker().isConvertibleTo(rightType, leftType)) { semanticServices.getErrorHandler().typeMismatch(right, leftType, rightType); } } } result = null; // This is not an element } private void resolveArrayAccessToLValue(JetArrayAccessExpression arrayAccessExpression, JetExpression rightHandSide, JetSimpleNameExpression operationSign) { List argumentTypes = getTypes(scope, arrayAccessExpression.getIndexExpressions()); if (argumentTypes == null) return; JetType rhsType = getType(scope, rightHandSide, false); if (rhsType == null) return; argumentTypes.add(rhsType); JetType receiverType = getType(scope, arrayAccessExpression.getArrayExpression(), false); if (receiverType == null) return; // TODO : nasty hack: effort is duplicated lookupFunction(scope, arrayAccessExpression, "set", receiverType, argumentTypes, true); FunctionDescriptor functionDescriptor = lookupFunction(scope, operationSign, "set", receiverType, argumentTypes, true); if (functionDescriptor != null) { result = functionDescriptor.getUnsubstitutedReturnType(); } } @Override public void visitJetElement(JetElement elem) { semanticServices.getErrorHandler().genericError(elem.getNode(), "Unsupported element in a block: " + elem + " " + elem.getClass().getCanonicalName()); } } private class CachedBindingTrace extends BindingTrace { private final BindingTrace originalTrace; public CachedBindingTrace(BindingTrace originalTrace) { this.originalTrace = originalTrace; } public void recordExpressionType(@NotNull JetExpression expression, @NotNull JetType type) { originalTrace.recordExpressionType(expression, type); typeCache.put(expression, type); } public void recordReferenceResolution(@NotNull JetReferenceExpression expression, @NotNull DeclarationDescriptor descriptor) { originalTrace.recordReferenceResolution(expression, descriptor); } public void recordLabelResolution(@NotNull JetReferenceExpression expression, @NotNull PsiElement element) { originalTrace.recordLabelResolution(expression, element); } public void recordDeclarationResolution(@NotNull PsiElement declaration, @NotNull DeclarationDescriptor descriptor) { originalTrace.recordDeclarationResolution(declaration, descriptor); } public void recordTypeResolution(@NotNull JetTypeReference typeReference, @NotNull JetType type) { originalTrace.recordTypeResolution(typeReference, type); } public void setToplevelScope(JetScope toplevelScope) { originalTrace.setToplevelScope(toplevelScope); } public void recordBlock(JetFunctionLiteralExpression expression) { originalTrace.recordBlock(expression); } public void removeReferenceResolution(@NotNull JetReferenceExpression referenceExpression) { originalTrace.removeReferenceResolution(referenceExpression); } } }