org.jetbrains.kotlin.incremental.IncrementalFirJvmCompilerRunner.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package org.jetbrains.kotlin.incremental
import com.intellij.ide.highlighter.JavaFileType
import com.intellij.openapi.util.Disposer
import com.intellij.psi.PsiJavaModule
import org.jetbrains.kotlin.KtIoFileSourceFile
import org.jetbrains.kotlin.KtSourceFile
import org.jetbrains.kotlin.KtVirtualFileSourceFile
import org.jetbrains.kotlin.backend.common.extensions.IrGenerationExtension
import org.jetbrains.kotlin.backend.jvm.JvmIrDeserializerImpl
import org.jetbrains.kotlin.build.DEFAULT_KOTLIN_SOURCE_FILES_EXTENSIONS
import org.jetbrains.kotlin.build.report.BuildReporter
import org.jetbrains.kotlin.build.report.metrics.GradleBuildPerformanceMetric
import org.jetbrains.kotlin.build.report.metrics.GradleBuildTime
import org.jetbrains.kotlin.cli.common.*
import org.jetbrains.kotlin.cli.common.arguments.K2JVMCompilerArguments
import org.jetbrains.kotlin.cli.common.config.addKotlinSourceRoot
import org.jetbrains.kotlin.cli.common.config.kotlinSourceRoots
import org.jetbrains.kotlin.cli.common.environment.setIdeaIoUseFallback
import org.jetbrains.kotlin.cli.common.fir.reportToMessageCollector
import org.jetbrains.kotlin.cli.common.messages.CompilerMessageSeverity
import org.jetbrains.kotlin.cli.common.messages.GroupingMessageCollector
import org.jetbrains.kotlin.cli.common.messages.IrMessageCollector
import org.jetbrains.kotlin.cli.common.messages.MessageCollector
import org.jetbrains.kotlin.cli.common.modules.ModuleBuilder
import org.jetbrains.kotlin.cli.jvm.*
import org.jetbrains.kotlin.cli.jvm.compiler.EnvironmentConfigFiles
import org.jetbrains.kotlin.cli.jvm.compiler.findMainClass
import org.jetbrains.kotlin.cli.jvm.compiler.forAllFiles
import org.jetbrains.kotlin.cli.jvm.compiler.pipeline.*
import org.jetbrains.kotlin.cli.jvm.compiler.writeOutputsIfNeeded
import org.jetbrains.kotlin.cli.jvm.config.*
import org.jetbrains.kotlin.cli.jvm.plugins.PluginCliParser
import org.jetbrains.kotlin.config.*
import org.jetbrains.kotlin.diagnostics.DiagnosticReporterFactory
import org.jetbrains.kotlin.fir.backend.jvm.JvmFir2IrExtensions
import org.jetbrains.kotlin.fir.pipeline.FirResult
import org.jetbrains.kotlin.fir.session.environment.AbstractProjectFileSearchScope
import org.jetbrains.kotlin.incremental.components.ExpectActualTracker
import org.jetbrains.kotlin.incremental.components.InlineConstTracker
import org.jetbrains.kotlin.incremental.components.LookupTracker
import org.jetbrains.kotlin.incremental.multiproject.ModulesApiHistory
import org.jetbrains.kotlin.ir.backend.jvm.serialization.JvmIrMangler
import org.jetbrains.kotlin.ir.util.IrMessageLogger
import org.jetbrains.kotlin.load.java.JavaClassesTracker
import org.jetbrains.kotlin.load.kotlin.incremental.components.IncrementalCompilationComponents
import org.jetbrains.kotlin.metadata.jvm.deserialization.JvmMetadataVersion
import org.jetbrains.kotlin.metadata.jvm.deserialization.JvmProtoBufUtil
import org.jetbrains.kotlin.modules.TargetId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.platform.CommonPlatforms
import org.jetbrains.kotlin.platform.jvm.JvmPlatforms
import org.jetbrains.kotlin.progress.CompilationCanceledException
import java.io.File
open class IncrementalFirJvmCompilerRunner(
workingDir: File,
reporter: BuildReporter,
buildHistoryFile: File?,
outputDirs: Collection?,
modulesApiHistory: ModulesApiHistory,
kotlinSourceFilesExtensions: Set = DEFAULT_KOTLIN_SOURCE_FILES_EXTENSIONS,
classpathChanges: ClasspathChanges
) : IncrementalJvmCompilerRunner(
workingDir,
reporter,
false,
buildHistoryFile,
outputDirs,
modulesApiHistory,
kotlinSourceFilesExtensions,
classpathChanges
) {
override fun runCompiler(
sourcesToCompile: List,
args: K2JVMCompilerArguments,
caches: IncrementalJvmCachesManager,
services: Services,
messageCollector: MessageCollector,
allSources: List,
isIncremental: Boolean
): Pair> {
// val isIncremental = true // TODO
val collector = GroupingMessageCollector(messageCollector, args.allWarningsAsErrors)
// from K2JVMCompiler (~)
val moduleName = args.moduleName ?: JvmProtoBufUtil.DEFAULT_MODULE_NAME
val targetId = TargetId(moduleName, "java-production") // TODO: get rid of magic constant
val dirtySources = linkedSetOf().apply { sourcesToCompile.forEach { add(KtIoFileSourceFile(it)) } }
// TODO: probably shoudl be passed along with sourcesToCompile
// TODO: file path normalization
val commonSources = args.commonSources?.mapTo(mutableSetOf(), ::File).orEmpty()
val exitCode = ExitCode.OK
val allCompiledSources = LinkedHashSet()
val rootDisposable = Disposer.newDisposable("Disposable for ${IncrementalFirJvmCompilerRunner::class.simpleName}.runCompiler")
try {
// - configuration
val configuration = CompilerConfiguration().apply {
put(CLIConfigurationKeys.ORIGINAL_MESSAGE_COLLECTOR_KEY, messageCollector)
put(CLIConfigurationKeys.MESSAGE_COLLECTOR_KEY, collector)
put(IrMessageLogger.IR_MESSAGE_LOGGER, IrMessageCollector(collector))
setupCommonArguments(args) { JvmMetadataVersion(*it) }
if (IncrementalCompilation.isEnabledForJvm()) {
putIfNotNull(CommonConfigurationKeys.LOOKUP_TRACKER, services[LookupTracker::class.java])
putIfNotNull(CommonConfigurationKeys.EXPECT_ACTUAL_TRACKER, services[ExpectActualTracker::class.java])
putIfNotNull(CommonConfigurationKeys.INLINE_CONST_TRACKER, services[InlineConstTracker::class.java])
putIfNotNull(
JVMConfigurationKeys.INCREMENTAL_COMPILATION_COMPONENTS,
services[IncrementalCompilationComponents::class.java]
)
putIfNotNull(ClassicFrontendSpecificJvmConfigurationKeys.JAVA_CLASSES_TRACKER, services[JavaClassesTracker::class.java])
}
setupJvmSpecificArguments(args)
}
val paths = computeKotlinPaths(collector, args)
if (collector.hasErrors()) return ExitCode.COMPILATION_ERROR to emptyList()
// -- plugins
val pluginClasspaths = args.pluginClasspaths?.toList() ?: emptyList()
val pluginOptions = args.pluginOptions?.toMutableList() ?: ArrayList()
val pluginConfigurations = args.pluginConfigurations?.toList() ?: emptyList()
// TODO: add scripting support when ready in FIR
val pluginLoadResult = PluginCliParser.loadPluginsSafe(pluginClasspaths, pluginOptions, pluginConfigurations, configuration)
if (pluginLoadResult != ExitCode.OK) return pluginLoadResult to emptyList()
// -- /plugins
with(configuration) {
configureJavaModulesContentRoots(args)
configureStandardLibs(paths, args)
configureAdvancedJvmOptions(args)
configureKlibPaths(args)
configureJdkClasspathRoots()
val destination = File(args.destination ?: ".")
if (destination.path.endsWith(".jar")) {
put(JVMConfigurationKeys.OUTPUT_JAR, destination)
} else {
put(JVMConfigurationKeys.OUTPUT_DIRECTORY, destination)
}
addAll(JVMConfigurationKeys.MODULES, listOf(ModuleBuilder(targetId.name, destination.path, targetId.type)))
configureBaseRoots(args)
configureSourceRootsFromSources(allSources, commonSources, args.javaPackagePrefix)
}
// - /configuration
setIdeaIoUseFallback()
// -AbstractProjectEnvironment-
val projectEnvironment =
createProjectEnvironment(configuration, rootDisposable, EnvironmentConfigFiles.JVM_CONFIG_FILES, messageCollector)
// -sources
val allPlatformSourceFiles = linkedSetOf() // TODO: get from caller
val allCommonSourceFiles = linkedSetOf()
val sourcesByModuleName = mutableMapOf>()
configuration.kotlinSourceRoots.forAllFiles(configuration, projectEnvironment.project) { virtualFile, isCommon, hmppModule ->
val file = KtVirtualFileSourceFile(virtualFile)
if (isCommon) allCommonSourceFiles.add(file)
else allPlatformSourceFiles.add(file)
if (hmppModule != null) {
sourcesByModuleName.getOrPut(hmppModule) { mutableSetOf() }.add(file)
}
}
val diagnosticsReporter = DiagnosticReporterFactory.createPendingReporter()
val performanceManager = configuration[CLIConfigurationKeys.PERF_MANAGER]
val compilerEnvironment = ModuleCompilerEnvironment(projectEnvironment, diagnosticsReporter)
performanceManager?.notifyCompilerInitialized(0, 0, "${targetId.name}-${targetId.type}")
// !! main class - maybe from cache?
var mainClassFqName: FqName? = null
val renderDiagnosticName = configuration.getBoolean(CLIConfigurationKeys.RENDER_DIAGNOSTIC_INTERNAL_NAME)
var incrementalExcludesScope: AbstractProjectFileSearchScope? = null
fun firIncrementalCycle(): FirResult? {
while (true) {
val dirtySourcesByModuleName = sourcesByModuleName.mapValues { (_, sources) ->
sources.filterTo(mutableSetOf()) { dirtySources.any { df -> df.path == it.path } }
}
val groupedSource = GroupedKtSources(
commonSources = allCommonSourceFiles.filter { dirtySources.any { df -> df.path == it.path } },
platformSources = allPlatformSourceFiles.filter { dirtySources.any { df -> df.path == it.path } },
sourcesByModuleName = dirtySourcesByModuleName
)
val compilerInput = ModuleCompilerInput(
targetId,
groupedSource,
CommonPlatforms.defaultCommonPlatform,
JvmPlatforms.unspecifiedJvmPlatform,
configuration
)
performanceManager?.notifyAnalysisStarted()
val analysisResults =
compileModuleToAnalyzedFir(
compilerInput,
projectEnvironment,
emptyList(),
incrementalExcludesScope,
diagnosticsReporter,
performanceManager
)
performanceManager?.notifyAnalysisFinished()
// TODO: consider what to do if many compilations find a main class
if (mainClassFqName == null && configuration.get(JVMConfigurationKeys.OUTPUT_JAR) != null) {
mainClassFqName = findMainClass(analysisResults.outputs.last().fir)
}
// TODO: switch the whole IC to KtSourceFile instead of FIle
dirtySources.forEach {
allCompiledSources.add(File(it.path!!))
}
if (diagnosticsReporter.hasErrors) {
diagnosticsReporter.reportToMessageCollector(messageCollector, renderDiagnosticName)
return null
}
val newDirtySources =
collectNewDirtySources(analysisResults, targetId, configuration, caches, allCompiledSources, reporter)
if (!isIncremental || newDirtySources.isEmpty()) return analysisResults
caches.platformCache.markDirty(newDirtySources)
val newDirtyFilesOutputsScope =
projectEnvironment.getSearchScopeByIoFiles(caches.inputsCache.getOutputForSourceFiles(newDirtySources))
incrementalExcludesScope = incrementalExcludesScope.let {
when {
newDirtyFilesOutputsScope.isEmpty -> it
it == null || it.isEmpty -> newDirtyFilesOutputsScope
else -> it + newDirtyFilesOutputsScope
}
}
caches.inputsCache.removeOutputForSourceFiles(newDirtySources)
newDirtySources.forEach {
dirtySources.add(KtIoFileSourceFile(it))
}
projectEnvironment.localFileSystem.refresh(false)
}
}
val cycleResult = firIncrementalCycle() ?: return ExitCode.COMPILATION_ERROR to allCompiledSources
performanceManager?.notifyGenerationStarted()
performanceManager?.notifyIRTranslationStarted()
val extensions = JvmFir2IrExtensions(configuration, JvmIrDeserializerImpl(), JvmIrMangler)
val irGenerationExtensions = projectEnvironment.project.let { IrGenerationExtension.getInstances(it) }
val (irModuleFragment, components, pluginContext, irActualizedResult) = cycleResult.convertToIrAndActualizeForJvm(
extensions, configuration, compilerEnvironment.diagnosticsReporter, irGenerationExtensions,
)
performanceManager?.notifyIRTranslationFinished()
val irInput = ModuleCompilerIrBackendInput(
targetId,
configuration,
extensions,
irModuleFragment,
components,
pluginContext,
irActualizedResult
)
val codegenOutput = generateCodeFromIr(irInput, compilerEnvironment, performanceManager)
performanceManager?.notifyIRGenerationFinished()
performanceManager?.notifyGenerationFinished()
diagnosticsReporter.reportToMessageCollector(messageCollector, renderDiagnosticName)
writeOutputsIfNeeded(
projectEnvironment.project,
configuration,
messageCollector,
listOf(codegenOutput.generationState),
mainClassFqName
)
} catch (e: CompilationCanceledException) {
collector.report(CompilerMessageSeverity.INFO, "Compilation was canceled", null)
return ExitCode.OK to allCompiledSources
} catch (e: RuntimeException) {
val cause = e.cause
if (cause is CompilationCanceledException) {
collector.report(CompilerMessageSeverity.INFO, "Compilation was canceled", null)
return ExitCode.OK to allCompiledSources
} else {
throw e
}
} finally {
collector.flush()
Disposer.dispose(rootDisposable)
}
return exitCode to allCompiledSources
}
}
fun CompilerConfiguration.configureBaseRoots(args: K2JVMCompilerArguments) {
var isJava9Module = false
args.javaSourceRoots?.forEach {
val file = File(it)
val packagePrefix = args.javaPackagePrefix
addJavaSourceRoot(file, packagePrefix)
if (!isJava9Module && packagePrefix == null && (file.name == PsiJavaModule.MODULE_INFO_FILE ||
(file.isDirectory && file.listFiles()?.any { it.name == PsiJavaModule.MODULE_INFO_FILE } == true))
) {
isJava9Module = true
}
}
args.classpath?.split(File.pathSeparator)?.forEach { classpathRoot ->
add(
CLIConfigurationKeys.CONTENT_ROOTS,
if (isJava9Module) JvmModulePathRoot(File(classpathRoot)) else JvmClasspathRoot(File(classpathRoot))
)
}
// TODO: modularJdkRoot (now seems only processed from the build file
}
fun CompilerConfiguration.configureSourceRootsFromSources(
allSources: Collection, commonSources: Set, javaPackagePrefix: String?
) {
val hmppCliModuleStructure = get(CommonConfigurationKeys.HMPP_MODULE_STRUCTURE)
for (sourceFile in allSources) {
if (sourceFile.name.endsWith(JavaFileType.DOT_DEFAULT_EXTENSION)) {
addJavaSourceRoot(sourceFile, javaPackagePrefix)
} else {
val path = sourceFile.path
addKotlinSourceRoot(path, isCommon = sourceFile in commonSources, hmppCliModuleStructure?.getModuleNameForSource(path))
if (sourceFile.isDirectory) {
addJavaSourceRoot(sourceFile, javaPackagePrefix)
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy