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