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;
018    
019    import com.intellij.psi.PsiElement;
020    import org.jetbrains.annotations.NotNull;
021    import org.jetbrains.annotations.Nullable;
022    import org.jetbrains.asm4.AnnotationVisitor;
023    import org.jetbrains.asm4.ClassVisitor;
024    import org.jetbrains.asm4.FieldVisitor;
025    import org.jetbrains.asm4.MethodVisitor;
026    import org.jetbrains.jet.codegen.state.JetTypeMapper;
027    import org.jetbrains.jet.lang.descriptors.*;
028    import org.jetbrains.jet.lang.descriptors.annotations.Annotated;
029    import org.jetbrains.jet.lang.descriptors.annotations.AnnotationArgumentVisitor;
030    import org.jetbrains.jet.lang.descriptors.annotations.AnnotationDescriptor;
031    import org.jetbrains.jet.lang.psi.JetAnnotationEntry;
032    import org.jetbrains.jet.lang.psi.JetClass;
033    import org.jetbrains.jet.lang.psi.JetModifierList;
034    import org.jetbrains.jet.lang.psi.JetModifierListOwner;
035    import org.jetbrains.jet.lang.resolve.BindingContext;
036    import org.jetbrains.jet.lang.resolve.calls.model.ResolvedCall;
037    import org.jetbrains.jet.lang.resolve.constants.*;
038    import org.jetbrains.jet.lang.resolve.constants.StringValue;
039    import org.jetbrains.jet.lang.types.lang.KotlinBuiltIns;
040    
041    import java.lang.annotation.RetentionPolicy;
042    import java.util.List;
043    import java.util.Map;
044    
045    import static org.jetbrains.jet.lang.resolve.BindingContextUtils.descriptorToDeclaration;
046    
047    public abstract class AnnotationCodegen {
048        private final JetTypeMapper typeMapper;
049        private final BindingContext bindingContext;
050    
051        private AnnotationCodegen(JetTypeMapper mapper) {
052            typeMapper = mapper;
053            bindingContext = typeMapper.getBindingContext();
054        }
055    
056        public void genAnnotations(Annotated annotated) {
057            if (annotated == null) {
058                return;
059            }
060    
061            if (!(annotated instanceof DeclarationDescriptor)) {
062                return;
063            }
064    
065            PsiElement psiElement = descriptorToDeclaration(bindingContext, (DeclarationDescriptor) annotated);
066    
067            JetModifierList modifierList = null;
068            if (annotated instanceof ConstructorDescriptor && psiElement instanceof JetClass) {
069                modifierList = ((JetClass) psiElement).getPrimaryConstructorModifierList();
070            }
071            else if (psiElement instanceof JetModifierListOwner) {
072                modifierList = ((JetModifierListOwner) psiElement).getModifierList();
073            }
074    
075            if (modifierList == null) {
076                return;
077            }
078    
079            List<JetAnnotationEntry> annotationEntries = modifierList.getAnnotationEntries();
080            for (JetAnnotationEntry annotationEntry : annotationEntries) {
081                ResolvedCall<? extends CallableDescriptor> resolvedCall =
082                        bindingContext.get(BindingContext.RESOLVED_CALL, annotationEntry.getCalleeExpression());
083                if (resolvedCall == null) continue; // Skipping annotations if they are not resolved. Needed for JetLightClass generation
084    
085                AnnotationDescriptor annotationDescriptor = bindingContext.get(BindingContext.ANNOTATION, annotationEntry);
086                if (annotationDescriptor == null) continue; // Skipping annotations if they are not resolved. Needed for JetLightClass generation
087                if (isVolatile(annotationDescriptor)) continue;
088    
089                genAnnotation(annotationDescriptor);
090            }
091        }
092    
093        private static boolean isVolatile(@NotNull AnnotationDescriptor annotationDescriptor) {
094            ClassifierDescriptor classDescriptor = annotationDescriptor.getType().getConstructor().getDeclarationDescriptor();
095            return KotlinBuiltIns.getInstance().getVolatileAnnotationClass().equals(classDescriptor);
096        }
097    
098        public void generateAnnotationDefaultValue(CompileTimeConstant value) {
099            AnnotationVisitor visitor = visitAnnotation(null, false);  // Parameters are unimportant
100            genCompileTimeValue(null, value, visitor);
101            visitor.visitEnd();
102        }
103    
104        private void genAnnotation(AnnotationDescriptor annotationDescriptor) {
105            ClassifierDescriptor classifierDescriptor = annotationDescriptor.getType().getConstructor().getDeclarationDescriptor();
106            RetentionPolicy rp = getRetentionPolicy(classifierDescriptor, typeMapper);
107            if (rp == RetentionPolicy.SOURCE) {
108                return;
109            }
110    
111            String internalName = typeMapper.mapType(annotationDescriptor.getType()).getDescriptor();
112            AnnotationVisitor annotationVisitor = visitAnnotation(internalName, rp == RetentionPolicy.RUNTIME);
113    
114            genAnnotationArguments(annotationDescriptor, annotationVisitor);
115            annotationVisitor.visitEnd();
116        }
117    
118        private void genAnnotationArguments(AnnotationDescriptor annotationDescriptor, AnnotationVisitor annotationVisitor) {
119            for (Map.Entry<ValueParameterDescriptor, CompileTimeConstant<?>> entry : annotationDescriptor.getAllValueArguments().entrySet()) {
120                ValueParameterDescriptor descriptor = entry.getKey();
121                String name = descriptor.getName().asString();
122                genCompileTimeValue(name, entry.getValue(), annotationVisitor);
123            }
124        }
125    
126        private void genCompileTimeValue(
127                @Nullable final String name,
128                @NotNull CompileTimeConstant<?> value,
129                @NotNull final AnnotationVisitor annotationVisitor
130        ) {
131            AnnotationArgumentVisitor argumentVisitor = new AnnotationArgumentVisitor<Void, Void>() {
132                @Override
133                public Void visitLongValue(@NotNull LongValue value, Void data) {
134                    return visitSimpleValue(value);
135                }
136    
137                @Override
138                public Void visitIntValue(IntValue value, Void data) {
139                    return visitSimpleValue(value);
140                }
141    
142                @Override
143                public Void visitShortValue(ShortValue value, Void data) {
144                    return visitSimpleValue(value);
145                }
146    
147                @Override
148                public Void visitByteValue(ByteValue value, Void data) {
149                    return visitSimpleValue(value);
150                }
151    
152                @Override
153                public Void visitDoubleValue(DoubleValue value, Void data) {
154                    return visitSimpleValue(value);
155                }
156    
157                @Override
158                public Void visitFloatValue(FloatValue value, Void data) {
159                    return visitSimpleValue(value);
160                }
161    
162                @Override
163                public Void visitBooleanValue(BooleanValue value, Void data) {
164                    return visitSimpleValue(value);
165                }
166    
167                @Override
168                public Void visitCharValue(CharValue value, Void data) {
169                    return visitSimpleValue(value);
170                }
171    
172                @Override
173                public Void visitStringValue(StringValue value, Void data) {
174                    return visitSimpleValue(value);
175                }
176    
177                @Override
178                public Void visitEnumValue(EnumValue value, Void data) {
179                    String propertyName = value.getValue().getName().asString();
180                    annotationVisitor.visitEnum(name, typeMapper.mapType(value.getType(KotlinBuiltIns.getInstance())).getDescriptor(), propertyName);
181                    return null;
182                }
183    
184                @Override
185                public Void visitArrayValue(ArrayValue value, Void data) {
186                    AnnotationVisitor visitor = annotationVisitor.visitArray(name);
187                    for (CompileTimeConstant<?> argument : value.getValue()) {
188                        genCompileTimeValue(null, argument, visitor);
189                    }
190                    visitor.visitEnd();
191                    return null;
192                }
193    
194                @Override
195                public Void visitAnnotationValue(AnnotationValue value, Void data) {
196                    String internalAnnName = typeMapper.mapType(value.getValue().getType()).getDescriptor();
197                    AnnotationVisitor visitor = annotationVisitor.visitAnnotation(name, internalAnnName);
198                    genAnnotationArguments(value.getValue(), visitor);
199                    visitor.visitEnd();
200                    return null;
201                }
202    
203                @Override
204                public Void visitJavaClassValue(JavaClassValue value, Void data) {
205                    annotationVisitor.visit(name, typeMapper.mapType(value.getValue()));
206                    return null;
207                }
208    
209                private Void visitSimpleValue(CompileTimeConstant value) {
210                    annotationVisitor.visit(name, value.getValue());
211                    return null;
212                }
213    
214                @Override
215                public Void visitErrorValue(ErrorValue value, Void data) {
216                    return visitUnsupportedValue(value);
217                }
218    
219                @Override
220                public Void visitNullValue(NullValue value, Void data) {
221                    return visitUnsupportedValue(value);
222                }
223    
224                private Void visitUnsupportedValue(CompileTimeConstant value) {
225                    throw new IllegalStateException("Don't know how to compile annotation value " + value);
226                }
227            };
228    
229            value.accept(argumentVisitor, null);
230        }
231    
232        private static RetentionPolicy getRetentionPolicy(ClassifierDescriptor descriptor, JetTypeMapper typeMapper) {
233            for (AnnotationDescriptor annotationDescriptor : descriptor.getAnnotations()) {
234                String internalName = typeMapper.mapType(annotationDescriptor.getType()).getInternalName();
235                if("java/lang/annotation/Retention".equals(internalName)) {
236                    CompileTimeConstant<?> compileTimeConstant = annotationDescriptor.getAllValueArguments().values().iterator().next();
237                    assert compileTimeConstant instanceof EnumValue : "Retention argument should be Enum value " + compileTimeConstant;
238                    PropertyDescriptor propertyDescriptor = ((EnumValue) compileTimeConstant).getValue();
239                    assert "java/lang/annotation/RetentionPolicy".equals(typeMapper.mapType(propertyDescriptor.getType()).getInternalName()) :
240                                                                            "Retention argument should be of type RetentionPolicy";
241                    String propertyDescriptorName = propertyDescriptor.getName().asString();
242                    return RetentionPolicy.valueOf(propertyDescriptorName);
243                }
244            }
245    
246            return RetentionPolicy.CLASS;
247        }
248    
249        abstract AnnotationVisitor visitAnnotation(String descr, boolean visible);
250    
251        public static AnnotationCodegen forClass(final ClassVisitor cv, JetTypeMapper mapper) {
252            return new AnnotationCodegen(mapper) {
253                @Override
254                AnnotationVisitor visitAnnotation(String descr, boolean visible) {
255                    return cv.visitAnnotation(descr, visible);
256                }
257            };
258        }
259    
260        public static AnnotationCodegen forMethod(final MethodVisitor mv, JetTypeMapper mapper) {
261            return new AnnotationCodegen(mapper) {
262                @Override
263                AnnotationVisitor visitAnnotation(String descr, boolean visible) {
264                    return mv.visitAnnotation(descr, visible);
265                }
266            };
267        }
268    
269        public static AnnotationCodegen forField(final FieldVisitor fv, JetTypeMapper mapper) {
270            return new AnnotationCodegen(mapper) {
271                @Override
272                AnnotationVisitor visitAnnotation(String descr, boolean visible) {
273                    return fv.visitAnnotation(descr, visible);
274                }
275            };
276        }
277    
278        public static AnnotationCodegen forParameter(final int parameter, final MethodVisitor mv, JetTypeMapper mapper) {
279            return new AnnotationCodegen(mapper) {
280                @Override
281                AnnotationVisitor visitAnnotation(String descr, boolean visible) {
282                    return mv.visitParameterAnnotation(parameter, descr, visible);
283                }
284            };
285        }
286    
287        public static AnnotationCodegen forAnnotationDefaultValue(final MethodVisitor mv, JetTypeMapper mapper) {
288            return new AnnotationCodegen(mapper) {
289                @Override
290                AnnotationVisitor visitAnnotation(String descr, boolean visible) {
291                    return mv.visitAnnotationDefault();
292                }
293            };
294        }
295    }