001 /* 002 * Copyright 2010-2014 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.backend.common; 018 019 import org.jetbrains.annotations.NotNull; 020 import org.jetbrains.annotations.Nullable; 021 import org.jetbrains.jet.codegen.bridges.BridgesPackage; 022 import org.jetbrains.jet.lang.descriptors.*; 023 import org.jetbrains.jet.lang.psi.JetDelegationSpecifier; 024 import org.jetbrains.jet.lang.psi.JetExpression; 025 import org.jetbrains.jet.lang.psi.JetSimpleNameExpression; 026 import org.jetbrains.jet.lang.resolve.BindingContext; 027 import org.jetbrains.jet.lang.resolve.DescriptorUtils; 028 import org.jetbrains.jet.lang.resolve.calls.CallResolverUtil; 029 import org.jetbrains.jet.lang.resolve.calls.callUtil.CallUtilPackage; 030 import org.jetbrains.jet.lang.resolve.calls.model.ResolvedCall; 031 import org.jetbrains.jet.lang.resolve.name.Name; 032 import org.jetbrains.jet.lang.types.JetType; 033 import org.jetbrains.jet.lang.types.TypeUtils; 034 import org.jetbrains.jet.lang.types.checker.JetTypeChecker; 035 import org.jetbrains.jet.lang.types.lang.KotlinBuiltIns; 036 037 import java.util.*; 038 039 /** 040 * Backend-independent utility class. 041 */ 042 public class CodegenUtil { 043 044 private CodegenUtil() { 045 } 046 047 // TODO: consider putting predefined method signatures here too. 048 public static final String EQUALS_METHOD_NAME = "equals"; 049 public static final String TO_STRING_METHOD_NAME = "toString"; 050 public static final String HASH_CODE_METHOD_NAME = "hashCode"; 051 052 @Nullable 053 public static FunctionDescriptor getDeclaredFunctionByRawSignature( 054 @NotNull ClassDescriptor owner, 055 @NotNull Name name, 056 @NotNull ClassifierDescriptor returnedClassifier, 057 @NotNull ClassifierDescriptor... valueParameterClassifiers 058 ) { 059 Collection<FunctionDescriptor> functions = owner.getDefaultType().getMemberScope().getFunctions(name); 060 for (FunctionDescriptor function : functions) { 061 if (!CallResolverUtil.isOrOverridesSynthesized(function) 062 && function.getTypeParameters().isEmpty() 063 && valueParameterClassesMatch(function.getValueParameters(), Arrays.asList(valueParameterClassifiers)) 064 && rawTypeMatches(function.getReturnType(), returnedClassifier)) { 065 return function; 066 } 067 } 068 return null; 069 } 070 071 public static FunctionDescriptor getAnyEqualsMethod() { 072 ClassDescriptor anyClass = KotlinBuiltIns.getInstance().getAny(); 073 FunctionDescriptor function = 074 getDeclaredFunctionByRawSignature(anyClass, Name.identifier(EQUALS_METHOD_NAME), 075 KotlinBuiltIns.getInstance().getBoolean(), 076 anyClass); 077 assert function != null; 078 return function; 079 } 080 081 public static FunctionDescriptor getAnyToStringMethod() { 082 ClassDescriptor anyClass = KotlinBuiltIns.getInstance().getAny(); 083 FunctionDescriptor function = 084 getDeclaredFunctionByRawSignature(anyClass, Name.identifier(TO_STRING_METHOD_NAME), 085 KotlinBuiltIns.getInstance().getString()); 086 assert function != null; 087 return function; 088 } 089 090 public static FunctionDescriptor getAnyHashCodeMethod() { 091 ClassDescriptor anyClass = KotlinBuiltIns.getInstance().getAny(); 092 FunctionDescriptor function = 093 getDeclaredFunctionByRawSignature(anyClass, Name.identifier(HASH_CODE_METHOD_NAME), 094 KotlinBuiltIns.getInstance().getInt()); 095 assert function != null; 096 return function; 097 } 098 099 @Nullable 100 public static PropertyDescriptor getDelegatePropertyIfAny(JetExpression expression, ClassDescriptor classDescriptor, BindingContext bindingContext) { 101 PropertyDescriptor propertyDescriptor = null; 102 if (expression instanceof JetSimpleNameExpression) { 103 ResolvedCall<?> call = CallUtilPackage.getResolvedCall(expression, bindingContext); 104 if (call != null) { 105 CallableDescriptor callResultingDescriptor = call.getResultingDescriptor(); 106 if (callResultingDescriptor instanceof ValueParameterDescriptor) { 107 ValueParameterDescriptor valueParameterDescriptor = (ValueParameterDescriptor) callResultingDescriptor; 108 // constructor parameter 109 if (valueParameterDescriptor.getContainingDeclaration() instanceof ConstructorDescriptor) { 110 // constructor of my class 111 if (valueParameterDescriptor.getContainingDeclaration().getContainingDeclaration() == classDescriptor) { 112 propertyDescriptor = bindingContext.get(BindingContext.VALUE_PARAMETER_AS_PROPERTY, valueParameterDescriptor); 113 } 114 } 115 } 116 117 // todo: when and if frontend will allow properties defined not as constructor parameters to be used in delegation specifier 118 } 119 } 120 return propertyDescriptor; 121 } 122 123 public static boolean isFinalPropertyWithBackingField(PropertyDescriptor propertyDescriptor, BindingContext bindingContext) { 124 return propertyDescriptor != null && 125 !propertyDescriptor.isVar() && 126 Boolean.TRUE.equals(bindingContext.get(BindingContext.BACKING_FIELD_REQUIRED, propertyDescriptor)); 127 } 128 129 @NotNull 130 public static Map<FunctionDescriptor, FunctionDescriptor> getTraitMethods(ClassDescriptor descriptor) { 131 Map<FunctionDescriptor, FunctionDescriptor> result = new LinkedHashMap<FunctionDescriptor, FunctionDescriptor>(); 132 for (DeclarationDescriptor declaration : descriptor.getDefaultType().getMemberScope().getAllDescriptors()) { 133 if (!(declaration instanceof CallableMemberDescriptor)) continue; 134 135 CallableMemberDescriptor inheritedMember = (CallableMemberDescriptor) declaration; 136 CallableMemberDescriptor traitMember = BridgesPackage.findTraitImplementation(inheritedMember); 137 if (traitMember == null) continue; 138 139 assert traitMember.getModality() != Modality.ABSTRACT : "Cannot delegate to abstract trait method: " + inheritedMember; 140 141 // inheritedMember can be abstract here. In order for FunctionCodegen to generate the method body, we're creating a copy here 142 // with traitMember's modality 143 CallableMemberDescriptor copy = 144 inheritedMember.copy(inheritedMember.getContainingDeclaration(), traitMember.getModality(), Visibilities.PUBLIC, 145 CallableMemberDescriptor.Kind.DECLARATION, true); 146 147 if (traitMember instanceof SimpleFunctionDescriptor) { 148 result.put((FunctionDescriptor) traitMember, (FunctionDescriptor) copy); 149 } 150 else if (traitMember instanceof PropertyDescriptor) { 151 for (PropertyAccessorDescriptor traitAccessor : ((PropertyDescriptor) traitMember).getAccessors()) { 152 for (PropertyAccessorDescriptor inheritedAccessor : ((PropertyDescriptor) copy).getAccessors()) { 153 if (inheritedAccessor.getClass() == traitAccessor.getClass()) { // same accessor kind 154 result.put(traitAccessor, inheritedAccessor); 155 } 156 } 157 } 158 } 159 } 160 return result; 161 } 162 163 @NotNull 164 public static ClassDescriptor getSuperClassByDelegationSpecifier(@NotNull JetDelegationSpecifier specifier, @NotNull BindingContext bindingContext) { 165 JetType superType = bindingContext.get(BindingContext.TYPE, specifier.getTypeReference()); 166 assert superType != null : "superType should not be null: " + specifier.getText(); 167 168 ClassDescriptor superClassDescriptor = (ClassDescriptor) superType.getConstructor().getDeclarationDescriptor(); 169 assert superClassDescriptor != null : "superClassDescriptor should not be null: " + specifier.getText(); 170 return superClassDescriptor; 171 } 172 173 private static boolean valueParameterClassesMatch( 174 @NotNull List<ValueParameterDescriptor> parameters, 175 @NotNull List<ClassifierDescriptor> classifiers 176 ) { 177 if (parameters.size() != classifiers.size()) return false; 178 for (int i = 0; i < parameters.size(); i++) { 179 ValueParameterDescriptor parameterDescriptor = parameters.get(i); 180 ClassifierDescriptor classDescriptor = classifiers.get(i); 181 if (!rawTypeMatches(parameterDescriptor.getType(), classDescriptor)) { 182 return false; 183 } 184 } 185 return true; 186 } 187 188 private static boolean rawTypeMatches(JetType type, ClassifierDescriptor classifier) { 189 return type.getConstructor().equals(classifier.getTypeConstructor()); 190 } 191 192 public static boolean isEnumValueOfMethod(@NotNull FunctionDescriptor functionDescriptor) { 193 List<ValueParameterDescriptor> methodTypeParameters = functionDescriptor.getValueParameters(); 194 JetType nullableString = TypeUtils.makeNullable(KotlinBuiltIns.getInstance().getStringType()); 195 return DescriptorUtils.ENUM_VALUE_OF.equals(functionDescriptor.getName()) 196 && methodTypeParameters.size() == 1 197 && JetTypeChecker.DEFAULT.isSubtypeOf(methodTypeParameters.get(0).getType(), nullableString); 198 } 199 200 public static boolean isEnumValuesMethod(@NotNull FunctionDescriptor functionDescriptor) { 201 List<ValueParameterDescriptor> methodTypeParameters = functionDescriptor.getValueParameters(); 202 return DescriptorUtils.ENUM_VALUES.equals(functionDescriptor.getName()) 203 && methodTypeParameters.isEmpty(); 204 } 205 }