001 /* 002 * Copyright 2010-2015 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.kotlin.codegen.intrinsics; 018 019 import com.intellij.psi.PsiElement; 020 import org.jetbrains.annotations.NotNull; 021 import org.jetbrains.annotations.Nullable; 022 import org.jetbrains.kotlin.codegen.ExpressionCodegen; 023 import org.jetbrains.kotlin.codegen.StackValue; 024 import org.jetbrains.kotlin.psi.JetElement; 025 import org.jetbrains.kotlin.psi.JetExpression; 026 import org.jetbrains.kotlin.psi.JetSuperExpression; 027 import org.jetbrains.kotlin.resolve.calls.model.ResolvedCall; 028 import org.jetbrains.kotlin.resolve.scopes.receivers.ExpressionReceiver; 029 import org.jetbrains.kotlin.resolve.scopes.receivers.ReceiverValue; 030 import org.jetbrains.org.objectweb.asm.Type; 031 import org.jetbrains.org.objectweb.asm.commons.InstructionAdapter; 032 033 import java.util.List; 034 035 import static org.jetbrains.kotlin.resolve.calls.callUtil.CallUtilPackage.getResolvedCallWithAssert; 036 import static org.jetbrains.kotlin.resolve.jvm.AsmTypes.OBJECT_TYPE; 037 038 public class Clone extends IntrinsicMethod { 039 @NotNull 040 @Override 041 protected Type generateImpl( 042 @NotNull ExpressionCodegen codegen, 043 @NotNull InstructionAdapter v, 044 @NotNull Type returnType, 045 @Nullable PsiElement element, 046 @NotNull List<JetExpression> arguments, 047 @NotNull StackValue receiver 048 ) { 049 ResolvedCall<?> resolvedCall = getResolvedCallWithAssert(((JetElement) element), codegen.getBindingContext()); 050 StackValue.receiver(resolvedCall, receiver, codegen, null).put(OBJECT_TYPE, v); 051 if (isSuperCall(resolvedCall)) { 052 v.invokespecial("java/lang/Object", "clone", "()Ljava/lang/Object;", false); 053 } 054 else { 055 v.invokevirtual("java/lang/Object", "clone", "()Ljava/lang/Object;", false); 056 } 057 return OBJECT_TYPE; 058 } 059 060 private static boolean isSuperCall(@NotNull ResolvedCall<?> resolvedCall) { 061 ReceiverValue dispatchReceiver = resolvedCall.getDispatchReceiver(); 062 return dispatchReceiver instanceof ExpressionReceiver && 063 ((ExpressionReceiver) dispatchReceiver).getExpression() instanceof JetSuperExpression; 064 } 065 }