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; 018 019 import com.intellij.psi.PsiElement; 020 import com.intellij.psi.PsiFile; 021 import kotlin.collections.CollectionsKt; 022 import kotlin.jvm.functions.Function1; 023 import kotlin.text.StringsKt; 024 import org.jetbrains.annotations.NotNull; 025 import org.jetbrains.annotations.Nullable; 026 import org.jetbrains.kotlin.codegen.binding.CalculatedClosure; 027 import org.jetbrains.kotlin.codegen.context.CodegenContext; 028 import org.jetbrains.kotlin.codegen.context.FacadePartWithSourceFile; 029 import org.jetbrains.kotlin.codegen.context.MethodContext; 030 import org.jetbrains.kotlin.codegen.context.RootContext; 031 import org.jetbrains.kotlin.codegen.state.JetTypeMapper; 032 import org.jetbrains.kotlin.descriptors.*; 033 import org.jetbrains.kotlin.load.java.JvmAnnotationNames; 034 import org.jetbrains.kotlin.load.kotlin.JvmMetadataVersion; 035 import org.jetbrains.kotlin.load.kotlin.ModuleMapping; 036 import org.jetbrains.kotlin.load.kotlin.ModuleVisibilityUtilsKt; 037 import org.jetbrains.kotlin.psi.KtFile; 038 import org.jetbrains.kotlin.psi.KtFunction; 039 import org.jetbrains.kotlin.psi.codeFragmentUtil.CodeFragmentUtilKt; 040 import org.jetbrains.kotlin.resolve.BindingContext; 041 import org.jetbrains.kotlin.resolve.DescriptorToSourceUtils; 042 import org.jetbrains.kotlin.resolve.DescriptorUtils; 043 import org.jetbrains.kotlin.resolve.inline.InlineUtil; 044 import org.jetbrains.kotlin.serialization.deserialization.descriptors.DeserializedCallableMemberDescriptor; 045 import org.jetbrains.kotlin.types.KotlinType; 046 import org.jetbrains.org.objectweb.asm.AnnotationVisitor; 047 048 import java.io.File; 049 050 import static org.jetbrains.kotlin.descriptors.Modality.ABSTRACT; 051 import static org.jetbrains.kotlin.descriptors.Modality.FINAL; 052 import static org.jetbrains.kotlin.resolve.jvm.annotations.AnnotationUtilKt.hasJvmFieldAnnotation; 053 054 public class JvmCodegenUtil { 055 056 private JvmCodegenUtil() { 057 } 058 059 public static boolean isJvmInterface(DeclarationDescriptor descriptor) { 060 if (descriptor instanceof ClassDescriptor) { 061 ClassKind kind = ((ClassDescriptor) descriptor).getKind(); 062 return kind == ClassKind.INTERFACE || kind == ClassKind.ANNOTATION_CLASS; 063 } 064 return false; 065 } 066 067 public static boolean isJvmInterface(KotlinType type) { 068 return isJvmInterface(type.getConstructor().getDeclarationDescriptor()); 069 } 070 071 public static boolean isConst(@NotNull CalculatedClosure closure) { 072 return closure.getCaptureThis() == null && closure.getCaptureReceiverType() == null && closure.getCaptureVariables().isEmpty(); 073 } 074 075 private static boolean isCallInsideSameClassAsDeclared(@NotNull CallableMemberDescriptor descriptor, @NotNull CodegenContext context) { 076 boolean isFakeOverride = descriptor.getKind() == CallableMemberDescriptor.Kind.FAKE_OVERRIDE; 077 boolean isDelegate = descriptor.getKind() == CallableMemberDescriptor.Kind.DELEGATION; 078 079 DeclarationDescriptor containingDeclaration = descriptor.getContainingDeclaration().getOriginal(); 080 081 return !isFakeOverride && !isDelegate && 082 (((context.hasThisDescriptor() && containingDeclaration == context.getThisDescriptor()) || 083 ((context.getParentContext() instanceof FacadePartWithSourceFile) 084 && isWithinSameFile(((FacadePartWithSourceFile) context.getParentContext()).getSourceFile(), descriptor))) 085 && context.getContextKind() != OwnerKind.DEFAULT_IMPLS); 086 } 087 088 private static boolean isWithinSameFile( 089 @Nullable KtFile callerFile, 090 @NotNull CallableMemberDescriptor descriptor 091 ) { 092 DeclarationDescriptor containingDeclaration = descriptor.getContainingDeclaration().getOriginal(); 093 if (containingDeclaration instanceof PackageFragmentDescriptor) { 094 PsiElement calleeElement = DescriptorToSourceUtils.descriptorToDeclaration(descriptor); 095 PsiFile calleeFile = calleeElement != null ? calleeElement.getContainingFile() : null; 096 return callerFile != null && callerFile != SourceFile.NO_SOURCE_FILE && calleeFile == callerFile; 097 098 } 099 return false; 100 } 101 102 public static boolean isCallInsideSameModuleAsDeclared( 103 @NotNull CallableMemberDescriptor declarationDescriptor, 104 @NotNull CodegenContext context, 105 @Nullable File outDirectory 106 ) { 107 if (context instanceof RootContext) { 108 return true; 109 } 110 DeclarationDescriptor contextDescriptor = context.getContextDescriptor(); 111 112 CallableMemberDescriptor directMember = getDirectMember(declarationDescriptor); 113 if (directMember instanceof DeserializedCallableMemberDescriptor) { 114 return ModuleVisibilityUtilsKt.isContainedByCompiledPartOfOurModule(directMember, outDirectory); 115 } 116 else { 117 return DescriptorUtils.areInSameModule(directMember, contextDescriptor); 118 } 119 } 120 121 public static boolean hasAbstractMembers(@NotNull ClassDescriptor classDescriptor) { 122 return CollectionsKt.any(DescriptorUtils.getAllDescriptors(classDescriptor.getDefaultType().getMemberScope()), 123 new Function1<DeclarationDescriptor, Boolean>() { 124 @Override 125 public Boolean invoke(DeclarationDescriptor descriptor) { 126 return descriptor instanceof CallableMemberDescriptor && 127 ((CallableMemberDescriptor) descriptor).getModality() == ABSTRACT; 128 } 129 } 130 ); 131 } 132 133 public static boolean isConstOrHasJvmFieldAnnotation(@NotNull PropertyDescriptor propertyDescriptor) { 134 return propertyDescriptor.isConst() || hasJvmFieldAnnotation(propertyDescriptor); 135 } 136 137 public static boolean couldUseDirectAccessToProperty( 138 @NotNull PropertyDescriptor property, 139 boolean forGetter, 140 boolean isDelegated, 141 @NotNull MethodContext contextBeforeInline 142 ) { 143 if (JetTypeMapper.isAccessor(property)) return false; 144 145 CodegenContext context = contextBeforeInline.getFirstCrossInlineOrNonInlineContext(); 146 // Inline functions can't use direct access because a field may not be visible at the call site 147 if (context.isInlineMethodContext()) { 148 return false; 149 } 150 151 // Only properties of the same class can be directly accessed, except when we are evaluating expressions in the debugger 152 if (!isCallInsideSameClassAsDeclared(property, context) && !isDebuggerContext(context)) return false; 153 154 // Delegated and extension properties have no backing fields 155 if (isDelegated || property.getExtensionReceiverParameter() != null) return false; 156 157 // Companion object properties cannot be accessed directly because their backing fields are stored in the containing class 158 if (DescriptorUtils.isCompanionObject(property.getContainingDeclaration())) return false; 159 160 PropertyAccessorDescriptor accessor = forGetter ? property.getGetter() : property.getSetter(); 161 162 // If there's no accessor declared we can use direct access 163 if (accessor == null) return true; 164 165 // If the accessor is non-default (i.e. it has some code) we should call that accessor and not use direct access 166 if (accessor.hasBody()) return false; 167 168 // If the accessor is private or final, it can't be overridden in the subclass and thus we can use direct access 169 return Visibilities.isPrivate(property.getVisibility()) || accessor.getModality() == FINAL; 170 } 171 172 private static boolean isDebuggerContext(@NotNull CodegenContext context) { 173 KtFile file = DescriptorToSourceUtils.getContainingFile(context.getContextDescriptor()); 174 return file != null && CodeFragmentUtilKt.getSuppressDiagnosticsInDebugMode(file); 175 } 176 177 @Nullable 178 public static ClassDescriptor getDispatchReceiverParameterForConstructorCall( 179 @NotNull ConstructorDescriptor descriptor, 180 @Nullable CalculatedClosure closure 181 ) { 182 //for compilation against sources 183 if (closure != null) { 184 return closure.getCaptureThis(); 185 } 186 187 //for compilation against binaries 188 //TODO: It's best to use this code also for compilation against sources 189 // but sometimes structures that have dispatchReceiver (bug?) mapped to static classes 190 ReceiverParameterDescriptor dispatchReceiver = descriptor.getDispatchReceiverParameter(); 191 if (dispatchReceiver != null) { 192 ClassDescriptor expectedThisClass = (ClassDescriptor) dispatchReceiver.getContainingDeclaration(); 193 if (!expectedThisClass.getKind().isSingleton()) { 194 return expectedThisClass; 195 } 196 } 197 198 return null; 199 } 200 201 @NotNull 202 public static CallableMemberDescriptor getDirectMember(@NotNull CallableMemberDescriptor descriptor) { 203 return descriptor instanceof PropertyAccessorDescriptor 204 ? ((PropertyAccessorDescriptor) descriptor).getCorrespondingProperty() 205 : descriptor; 206 } 207 208 public static boolean isArgumentWhichWillBeInlined(@NotNull BindingContext bindingContext, @NotNull DeclarationDescriptor descriptor) { 209 PsiElement declaration = DescriptorToSourceUtils.descriptorToDeclaration(descriptor); 210 return InlineUtil.canBeInlineArgument(declaration) && 211 InlineUtil.isInlinedArgument((KtFunction) declaration, bindingContext, false); 212 } 213 214 @NotNull 215 public static String getModuleName(ModuleDescriptor module) { 216 return StringsKt.removeSurrounding(module.getName().asString(), "<", ">"); 217 } 218 219 @NotNull 220 public static String getMappingFileName(@NotNull String moduleName) { 221 return "META-INF/" + moduleName + "." + ModuleMapping.MAPPING_FILE_EXT; 222 } 223 224 public static void writeAbiVersion(@NotNull AnnotationVisitor av) { 225 av.visit(JvmAnnotationNames.VERSION_FIELD_NAME, JvmMetadataVersion.INSTANCE.toArray()); 226 } 227 }