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.google.common.collect.Sets;
020    import com.intellij.util.containers.MultiMap;
021    import org.jetbrains.annotations.NotNull;
022    import org.jetbrains.kotlin.codegen.state.GenerationState;
023    import org.jetbrains.kotlin.fileClasses.JvmFileClassInfo;
024    import org.jetbrains.kotlin.name.FqName;
025    import org.jetbrains.kotlin.progress.ProgressIndicatorAndCompilationCanceledStatus;
026    import org.jetbrains.kotlin.psi.KtFile;
027    import org.jetbrains.kotlin.psi.KtScript;
028    import org.jetbrains.kotlin.resolve.ScriptNameUtil;
029    import org.jetbrains.org.objectweb.asm.Type;
030    
031    import java.util.Collection;
032    import java.util.HashSet;
033    import java.util.Set;
034    
035    import static org.jetbrains.kotlin.codegen.binding.CodegenBinding.registerClassNameForScript;
036    
037    public class KotlinCodegenFacade {
038    
039        public static void prepareForCompilation(@NotNull GenerationState state) {
040            for (KtFile file : state.getFiles()) {
041                if (file.isScript()) {
042                    KtScript script = file.getScript();
043                    assert script != null;
044    
045                    FqName name = ScriptNameUtil.classNameForScript(script);
046                    Type type = AsmUtil.asmTypeByFqNameWithoutInnerClasses(name);
047                    registerClassNameForScript(state.getBindingTrace(), script, type, state.getFileClassesProvider());
048                }
049            }
050    
051            state.beforeCompile();
052        }
053    
054        public static void compileCorrectFiles(
055                @NotNull GenerationState state,
056                @NotNull CompilationErrorHandler errorHandler
057        ) {
058            ProgressIndicatorAndCompilationCanceledStatus.checkCanceled();
059    
060            prepareForCompilation(state);
061    
062            ProgressIndicatorAndCompilationCanceledStatus.checkCanceled();
063    
064            doGenerateFiles(state.getFiles(), state, errorHandler);
065        }
066    
067        public static void doGenerateFiles(
068                @NotNull Collection<KtFile> files,
069                @NotNull GenerationState state,
070                @NotNull CompilationErrorHandler errorHandler
071        ) {
072            MultiMap<FqName, KtFile> filesInPackages = new MultiMap<FqName, KtFile>();
073            MultiMap<FqName, KtFile> filesInMultifileClasses = new MultiMap<FqName, KtFile>();
074    
075            for (KtFile file : files) {
076                if (file == null) throw new IllegalArgumentException("A null file given for compilation");
077    
078                JvmFileClassInfo fileClassInfo = state.getFileClassesProvider().getFileClassInfo(file);
079    
080                if (fileClassInfo.getWithJvmMultifileClass()) {
081                    filesInMultifileClasses.putValue(fileClassInfo.getFacadeClassFqName(), file);
082                }
083                else {
084                    filesInPackages.putValue(file.getPackageFqName(), file);
085                }
086            }
087    
088            Set<FqName> obsoleteMultifileClasses = new HashSet<FqName>(state.getObsoleteMultifileClasses());
089            for (FqName multifileClassFqName : Sets.union(filesInMultifileClasses.keySet(), obsoleteMultifileClasses)) {
090                doCheckCancelled(state);
091                generateMultifileClass(state, multifileClassFqName, filesInMultifileClasses.get(multifileClassFqName), errorHandler);
092            }
093    
094            Set<FqName> packagesWithObsoleteParts = new HashSet<FqName>(state.getPackagesWithObsoleteParts());
095            for (FqName packageFqName : Sets.union(packagesWithObsoleteParts, filesInPackages.keySet())) {
096                doCheckCancelled(state);
097                generatePackage(state, packageFqName, filesInPackages.get(packageFqName), errorHandler);
098            }
099    
100            doCheckCancelled(state);
101            state.getFactory().done();
102        }
103    
104        private static void doCheckCancelled(GenerationState state) {
105            if (state.getClassBuilderMode() == ClassBuilderMode.FULL) {
106                ProgressIndicatorAndCompilationCanceledStatus.checkCanceled();
107            }
108        }
109    
110        public static void generatePackage(
111                @NotNull GenerationState state,
112                @NotNull FqName packageFqName,
113                @NotNull Collection<KtFile> jetFiles,
114                @NotNull CompilationErrorHandler errorHandler
115        ) {
116            // We do not really generate package class, but use old package fqName to identify package in module-info.
117            //FqName packageClassFqName = PackageClassUtils.getPackageClassFqName(packageFqName);
118            PackageCodegen codegen = state.getFactory().forPackage(packageFqName, jetFiles);
119            codegen.generate(errorHandler);
120        }
121    
122        private static void generateMultifileClass(
123                @NotNull GenerationState state,
124                @NotNull FqName multifileClassFqName,
125                @NotNull Collection<KtFile> files,
126                @NotNull CompilationErrorHandler handler
127        ) {
128            MultifileClassCodegen codegen = state.getFactory().forMultifileClass(multifileClassFqName, files);
129            codegen.generate(handler);
130        }
131    
132        private KotlinCodegenFacade() {}
133    }