001    /*
002     * Copyright 2010-2013 JetBrains s.r.o.
003     *
004     * Licensed under the Apache License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.apache.org/licenses/LICENSE-2.0
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    
017    package org.jetbrains.jet.codegen.intrinsics;
018    
019    import com.intellij.psi.PsiElement;
020    import org.jetbrains.annotations.NotNull;
021    import org.jetbrains.jet.codegen.ExpressionCodegen;
022    import org.jetbrains.jet.codegen.StackValue;
023    import org.jetbrains.jet.lang.psi.JetBinaryExpression;
024    import org.jetbrains.jet.lang.psi.JetExpression;
025    import org.jetbrains.org.objectweb.asm.Type;
026    import org.jetbrains.org.objectweb.asm.commons.InstructionAdapter;
027    
028    import java.util.List;
029    
030    import static org.jetbrains.org.objectweb.asm.Type.*;
031    
032    public class RangeTo extends IntrinsicMethod {
033        @NotNull
034        @Override
035        public Type generateImpl(
036                @NotNull ExpressionCodegen codegen,
037                @NotNull InstructionAdapter v,
038                @NotNull Type returnType,
039                PsiElement element,
040                @NotNull List<JetExpression> arguments,
041                @NotNull StackValue receiver
042        ) {
043            v.anew(returnType);
044            v.dup();
045    
046            Type type;
047            if (arguments.size() == 1) {
048                assert receiver instanceof StackValue.CallReceiver :
049                        "Receiver in an intrinsic qualified expression should be CallReceiver: " + receiver + " on " + element.getText();
050                type = parameterType(receiver.type, codegen.expressionType(arguments.get(0)));
051                receiver.put(type, v);
052                codegen.gen(arguments.get(0), type);
053            }
054            else {
055                JetBinaryExpression expression = (JetBinaryExpression) element;
056                type = parameterType(codegen.expressionType(expression.getLeft()), codegen.expressionType(expression.getRight()));
057                codegen.gen(expression.getLeft(), type);
058                codegen.gen(expression.getRight(), type);
059            }
060    
061            v.invokespecial(returnType.getInternalName(), "<init>", Type.getMethodDescriptor(Type.VOID_TYPE, type, type), false);
062    
063            return returnType;
064        }
065    
066        @NotNull
067        private static Type parameterType(@NotNull Type leftType, @NotNull Type rightType) {
068            int left = leftType.getSort();
069            int right = rightType.getSort();
070            if (left == DOUBLE || right == DOUBLE) {
071                return DOUBLE_TYPE;
072            }
073            else if (left == FLOAT || right == FLOAT) {
074                return FLOAT_TYPE;
075            }
076            else if (left == LONG || right == LONG) {
077                return LONG_TYPE;
078            }
079            else if (left == INT || right == INT) {
080                return INT_TYPE;
081            }
082            else if (left == SHORT || right == SHORT) {
083                return SHORT_TYPE;
084            }
085            else if (left == CHAR || right == CHAR) {
086                return CHAR_TYPE;
087            }
088            else if (left == BYTE || right == BYTE) {
089                return BYTE_TYPE;
090            }
091            else {
092                throw new IllegalStateException("RangeTo intrinsic can only work for primitive types: " + leftType + ", " + rightType);
093            }
094        }
095    }