diff --git a/compiler/cli/cli-common/gen/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArgumentsCopyGenerated.kt b/compiler/cli/cli-common/gen/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArgumentsCopyGenerated.kt index b92954f494a99..917b9ba31af9e 100644 --- a/compiler/cli/cli-common/gen/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArgumentsCopyGenerated.kt +++ b/compiler/cli/cli-common/gen/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArgumentsCopyGenerated.kt @@ -19,6 +19,7 @@ fun copyK2JSCompilerArguments(from: K2JSCompilerArguments, to: K2JSCompilerArgum to.friendModulesDisabled = from.friendModulesDisabled to.generateDts = from.generateDts to.generatePolyfills = from.generatePolyfills + to.includeUnavailableSourcesIntoSourceMap = from.includeUnavailableSourcesIntoSourceMap to.includes = from.includes to.irBaseClassInMetadata = from.irBaseClassInMetadata to.irBuildCache = from.irBuildCache diff --git a/compiler/cli/cli-common/src/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArguments.kt b/compiler/cli/cli-common/src/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArguments.kt index 656903fd2c74b..9383fb06e895b 100644 --- a/compiler/cli/cli-common/src/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArguments.kt +++ b/compiler/cli/cli-common/src/org/jetbrains/kotlin/cli/common/arguments/K2JSCompilerArguments.kt @@ -274,6 +274,16 @@ In combination with '-meta-info', this generates both IR and pre-IR versions of field = value } + @Argument( + value = "-Xwasm-source-map-include-mappings-from-unavailable-sources", + description = "Insert source mappings from libraries even if their sources are unavailable on the end-user machine." + ) + var includeUnavailableSourcesIntoSourceMap = false + set(value) { + checkFrozen() + field = value + } + @Argument( value = "-Xir-dce-dump-reachability-info-to-file", valueDescription = "", diff --git a/compiler/cli/cli-js/src/org/jetbrains/kotlin/cli/js/K2JsIrCompiler.kt b/compiler/cli/cli-js/src/org/jetbrains/kotlin/cli/js/K2JsIrCompiler.kt index 096e9f379705d..c6ee376c6f5e4 100644 --- a/compiler/cli/cli-js/src/org/jetbrains/kotlin/cli/js/K2JsIrCompiler.kt +++ b/compiler/cli/cli-js/src/org/jetbrains/kotlin/cli/js/K2JsIrCompiler.kt @@ -209,6 +209,7 @@ class K2JsIrCompiler : CLICompiler() { configuration.put(WasmConfigurationKeys.WASM_GENERATE_WAT, arguments.wasmGenerateWat) configuration.put(WasmConfigurationKeys.WASM_USE_TRAPS_INSTEAD_OF_EXCEPTIONS, arguments.wasmUseTrapsInsteadOfExceptions) configuration.put(WasmConfigurationKeys.WASM_USE_NEW_EXCEPTION_PROPOSAL, arguments.wasmUseNewExceptionProposal) + configuration.putIfNotNull(WasmConfigurationKeys.WASM_TARGET, arguments.wasmTarget?.let(WasmTarget::fromName)) configuration.put(JSConfigurationKeys.OPTIMIZE_GENERATED_JS, arguments.optimizeGeneratedJs) @@ -839,6 +840,7 @@ class K2JsIrCompiler : CLICompiler() { sourceMapContentEmbedding = SourceMapSourceEmbedding.INLINING } configuration.put(JSConfigurationKeys.SOURCE_MAP_EMBED_SOURCES, sourceMapContentEmbedding) + configuration.put(JSConfigurationKeys.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES, arguments.includeUnavailableSourcesIntoSourceMap) if (!arguments.sourceMap && sourceMapEmbedContentString != null) { messageCollector.report(WARNING, "source-map-embed-sources argument has no effect without source map", null) diff --git a/compiler/ir/backend.js/src/org/jetbrains/kotlin/ir/backend/js/SourceMapsInfo.kt b/compiler/ir/backend.js/src/org/jetbrains/kotlin/ir/backend/js/SourceMapsInfo.kt index ae38a2d77e3d2..650db2bf2108f 100644 --- a/compiler/ir/backend.js/src/org/jetbrains/kotlin/ir/backend/js/SourceMapsInfo.kt +++ b/compiler/ir/backend.js/src/org/jetbrains/kotlin/ir/backend/js/SourceMapsInfo.kt @@ -17,6 +17,7 @@ data class SourceMapsInfo( val outputDir: File?, val sourceMapContentEmbedding: SourceMapSourceEmbedding, val namesPolicy: SourceMapNamesPolicy, + val includeUnavailableSourcesIntoSourceMap: Boolean = false ) { companion object { fun from(configuration: CompilerConfiguration): SourceMapsInfo? = @@ -26,7 +27,8 @@ data class SourceMapsInfo( configuration.get(JSConfigurationKeys.SOURCE_MAP_SOURCE_ROOTS, emptyList()), configuration.get(JSConfigurationKeys.OUTPUT_DIR), configuration.get(JSConfigurationKeys.SOURCE_MAP_EMBED_SOURCES, SourceMapSourceEmbedding.INLINING), - configuration.get(JSConfigurationKeys.SOURCEMAP_NAMES_POLICY, SourceMapNamesPolicy.SIMPLE_NAMES) + configuration.get(JSConfigurationKeys.SOURCEMAP_NAMES_POLICY, SourceMapNamesPolicy.SIMPLE_NAMES), + configuration.getBoolean(JSConfigurationKeys.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES), ) } else { null diff --git a/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/ir2wasm/locationUtils.kt b/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/ir2wasm/locationUtils.kt index e1bd464b643b4..a6d31d645c459 100644 --- a/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/ir2wasm/locationUtils.kt +++ b/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/ir2wasm/locationUtils.kt @@ -46,12 +46,23 @@ fun IrElement.getSourceLocation(file: IrFile?, type: LocationType = LocationType if (line < 0 || column < 0) return SourceLocation.NoLocation("startLine or startColumn < 0") - if (file.isIgnoredFile) { - return SourceLocation.IgnoredLocation(path, line, column) - } - + val module = file.module.name.asString() - return SourceLocation.Location(path, line, column) + return if (file.isIgnoredFile) { + SourceLocation.IgnoredLocation( + module, + path, + line, + column + ) + } else { + SourceLocation.Location( + module, + path, + line, + column + ) + } } fun WasmExpressionBuilder.buildUnreachableForVerifier() { diff --git a/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/utils/SourceMapGenerator.kt b/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/utils/SourceMapGenerator.kt index 537c068c64de1..09a88776ba1c4 100644 --- a/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/utils/SourceMapGenerator.kt +++ b/compiler/ir/backend.wasm/src/org/jetbrains/kotlin/backend/wasm/utils/SourceMapGenerator.kt @@ -39,8 +39,12 @@ class SourceMapGenerator( val sourceMapBuilder = SourceMap3Builder(null, { error("This should not be called for Kotlin/Wasm") }, sourceMapsInfo.sourceMapPrefix) - val pathResolver = - SourceFilePathResolver.create(sourceMapsInfo.sourceRoots, sourceMapsInfo.sourceMapPrefix, sourceMapsInfo.outputDir) + val pathResolver = SourceFilePathResolver.create( + sourceMapsInfo.sourceRoots, + sourceMapsInfo.sourceMapPrefix, + sourceMapsInfo.outputDir, + sourceMapsInfo.includeUnavailableSourcesIntoSourceMap + ) var prev: SourceLocation.Location? = null var prevGeneratedLine = 0 @@ -62,12 +66,21 @@ class SourceMapGenerator( // TODO: add the ignored location into "ignoreList" in future is SourceLocation.NoLocation, is SourceLocation.IgnoredLocation -> sourceMapBuilder.addEmptyMapping(generatedLocation.column) is SourceLocation.Location -> { - sourceLocation.apply { - // TODO resulting path goes too deep since temporary directory we compiled first is deeper than final destination. - val relativePath = pathResolver.getPathRelativeToSourceRoots(File(file)).replace(Regex("^\\.\\./"), "") - sourceMapBuilder.addMapping(relativePath, null, { null }, line, column, null, generatedLocation.column) - prev = this - } + // TODO resulting path goes too deep since temporary directory we compiled first is deeper than final destination. + val relativePath = pathResolver + .getPathRelativeToSourceRootsIfExists(sourceLocation.module, File(sourceLocation.file)) + ?.replace(Regex("^\\.\\./"), "") ?: continue + + sourceMapBuilder.addMapping( + relativePath, + null, + { null }, + sourceLocation.line, + sourceLocation.column, + null, + generatedLocation.column + ) + prev = sourceLocation } } diff --git a/compiler/testData/cli/js/jsExtraHelp.out b/compiler/testData/cli/js/jsExtraHelp.out index 454d7229d4af0..f17fac52116f7 100644 --- a/compiler/testData/cli/js/jsExtraHelp.out +++ b/compiler/testData/cli/js/jsExtraHelp.out @@ -10,6 +10,8 @@ where advanced options include: -Xfriend-modules-disabled Disable internal declaration export. -Xgenerate-dts Generate a TypeScript declaration .d.ts file alongside the JS file. This is available only in the IR backend. -Xgenerate-polyfills Generate polyfills for features from the ES6+ standards. + -Xwasm-source-map-include-mappings-from-unavailable-sources + Insert source mappings from libraries even if their sources are unavailable on the end-user machine. -Xinclude= Path to an intermediate library that should be processed in the same manner as source files. -Xir-base-class-in-metadata Write base classes into metadata. -Xir-build-cache Use the compiler to build the cache. diff --git a/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/directives/WasmEnvironmentConfigurationDirectives.kt b/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/directives/WasmEnvironmentConfigurationDirectives.kt index b2038b0cf3a2e..d6ff7ffa413ce 100644 --- a/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/directives/WasmEnvironmentConfigurationDirectives.kt +++ b/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/directives/WasmEnvironmentConfigurationDirectives.kt @@ -44,6 +44,11 @@ object WasmEnvironmentConfigurationDirectives : SimpleDirectivesContainer() { applicability = DirectiveApplicability.Global ) + val SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES by directive( + description = "Insert source mappings from libraries even if their sources are unavailable on the end-user machine", + applicability = DirectiveApplicability.Global + ) + val RUN_THIRD_PARTY_OPTIMIZER by directive( description = "Also run third-party optimizer (for now, only binaryen is supported) after the main compilation", ) diff --git a/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/services/configuration/WasmEnvironmentConfigurator.kt b/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/services/configuration/WasmEnvironmentConfigurator.kt index 011d78ee883c6..8568074e45e97 100644 --- a/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/services/configuration/WasmEnvironmentConfigurator.kt +++ b/compiler/tests-common-new/tests/org/jetbrains/kotlin/test/services/configuration/WasmEnvironmentConfigurator.kt @@ -19,6 +19,7 @@ import org.jetbrains.kotlin.test.directives.JsEnvironmentConfigurationDirectives import org.jetbrains.kotlin.test.directives.JsEnvironmentConfigurationDirectives.SOURCE_MAP_EMBED_SOURCES import org.jetbrains.kotlin.test.directives.WasmEnvironmentConfigurationDirectives import org.jetbrains.kotlin.test.directives.WasmEnvironmentConfigurationDirectives.DISABLE_WASM_EXCEPTION_HANDLING +import org.jetbrains.kotlin.test.directives.WasmEnvironmentConfigurationDirectives.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES import org.jetbrains.kotlin.test.directives.WasmEnvironmentConfigurationDirectives.USE_NEW_EXCEPTION_HANDLING_PROPOSAL import org.jetbrains.kotlin.test.directives.model.DirectivesContainer import org.jetbrains.kotlin.test.directives.model.RegisteredDirectives @@ -93,11 +94,13 @@ abstract class WasmEnvironmentConfigurator(testServices: TestServices) : Environ val sourceDirs = module.files.map { it.originalFile.parent }.distinct() configuration.put(JSConfigurationKeys.SOURCE_MAP_SOURCE_ROOTS, sourceDirs) configuration.put(JSConfigurationKeys.SOURCE_MAP, true) + configuration.put(JSConfigurationKeys.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES, SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES in registeredDirectives) val sourceMapSourceEmbedding = registeredDirectives[SOURCE_MAP_EMBED_SOURCES].singleOrNull() ?: SourceMapSourceEmbedding.NEVER configuration.put(JSConfigurationKeys.SOURCE_MAP_EMBED_SOURCES, sourceMapSourceEmbedding) configuration.put(WasmConfigurationKeys.WASM_USE_TRAPS_INSTEAD_OF_EXCEPTIONS, DISABLE_WASM_EXCEPTION_HANDLING in registeredDirectives) configuration.put(WasmConfigurationKeys.WASM_USE_NEW_EXCEPTION_PROPOSAL, USE_NEW_EXCEPTION_HANDLING_PROPOSAL in registeredDirectives) + } } \ No newline at end of file diff --git a/js/js.frontend/src/org/jetbrains/kotlin/js/config/JSConfigurationKeys.java b/js/js.frontend/src/org/jetbrains/kotlin/js/config/JSConfigurationKeys.java index c7d19e3158fde..d8541a5dabe8c 100644 --- a/js/js.frontend/src/org/jetbrains/kotlin/js/config/JSConfigurationKeys.java +++ b/js/js.frontend/src/org/jetbrains/kotlin/js/config/JSConfigurationKeys.java @@ -41,6 +41,9 @@ public class JSConfigurationKeys { public static final CompilerConfigurationKey SOURCEMAP_NAMES_POLICY = CompilerConfigurationKey.create( "a policy to generate a mapping from generated identifiers to their corresponding original names"); + public static final CompilerConfigurationKey SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES = CompilerConfigurationKey.create( + "insert source mappings from libraries even if their sources are unavailable on the end-user machine"); + public static final CompilerConfigurationKey META_INFO = CompilerConfigurationKey.create("generate .meta.js and .kjsm files"); diff --git a/js/js.sourcemap/src/org/jetbrains/kotlin/js/sourceMap/SourceFilePathResolver.kt b/js/js.sourcemap/src/org/jetbrains/kotlin/js/sourceMap/SourceFilePathResolver.kt index 88b2d3b3d9480..71aa49e718086 100644 --- a/js/js.sourcemap/src/org/jetbrains/kotlin/js/sourceMap/SourceFilePathResolver.kt +++ b/js/js.sourcemap/src/org/jetbrains/kotlin/js/sourceMap/SourceFilePathResolver.kt @@ -7,13 +7,19 @@ package org.jetbrains.kotlin.js.sourceMap import org.jetbrains.kotlin.config.CompilerConfiguration import org.jetbrains.kotlin.js.config.JSConfigurationKeys import org.jetbrains.kotlin.js.config.JsConfig +import org.jetbrains.kotlin.utils.addToStdlib.runIf import java.io.File import java.io.IOException -class SourceFilePathResolver(sourceRoots: List, outputDir: File? = null) { +class SourceFilePathResolver( + sourceRoots: List, + outputDir: File? = null, + private val includeUnavailableSourcesIntoSourceMap: Boolean = false +) { private val sourceRoots = sourceRoots.mapTo(mutableSetOf()) { it.absoluteFile } private val outputDirPathResolver = outputDir?.let(::RelativePathCalculator) private val cache = mutableMapOf() + private val modulesAndTheirSourcesStatus = hashMapOf() @Throws(IOException::class) fun getPathRelativeToSourceRoots(file: File): String { @@ -25,6 +31,12 @@ class SourceFilePathResolver(sourceRoots: List, outputDir: File? = null) { return path } + @Throws(IOException::class) + fun getPathRelativeToSourceRootsIfExists(moduleId: String, file: File): String? { + val moduleSourcesShouldBeAdded = includeUnavailableSourcesIntoSourceMap || modulesAndTheirSourcesStatus.getOrPut(moduleId) { file.exists() } + return runIf(moduleSourcesShouldBeAdded) { getPathRelativeToSourceRoots(file) } + } + @Throws(IOException::class) private fun calculatePathRelativeToSourceRoots(file: File): String { val pathRelativeToOutput = calculatePathRelativeToOutput(file) @@ -57,15 +69,22 @@ class SourceFilePathResolver(sourceRoots: List, outputDir: File? = null) { fun create(configuration: CompilerConfiguration) = create( sourceRoots = configuration.get(JSConfigurationKeys.SOURCE_MAP_SOURCE_ROOTS, emptyList()), sourceMapPrefix = configuration.get(JSConfigurationKeys.SOURCE_MAP_PREFIX, ""), - outputDir = configuration.get(JSConfigurationKeys.OUTPUT_DIR) + outputDir = configuration.get(JSConfigurationKeys.OUTPUT_DIR), + includeUnavailableSourcesIntoSourceMap = configuration.getBoolean(JSConfigurationKeys.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES) ) @JvmStatic - fun create(sourceRoots: List, sourceMapPrefix: String, outputDir: File?): SourceFilePathResolver { + fun create( + sourceRoots: List, + sourceMapPrefix: String, + outputDir: File?, + includeUnavailableSourcesIntoSourceMap: Boolean = false, + ): SourceFilePathResolver { val generateRelativePathsInSourceMap = sourceMapPrefix.isEmpty() && sourceRoots.isEmpty() return SourceFilePathResolver( sourceRoots.map(::File), - outputDir.takeIf { generateRelativePathsInSourceMap } + outputDir.takeIf { generateRelativePathsInSourceMap }, + includeUnavailableSourcesIntoSourceMap ) } } diff --git a/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToBinary.kt b/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToBinary.kt index ad2a8c20869b2..c211c7f753a09 100644 --- a/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToBinary.kt +++ b/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToBinary.kt @@ -737,6 +737,7 @@ private class SourceLocationMappingToBinary( ) : SourceLocationMapping() { override val generatedLocation: SourceLocation.Location by lazy { SourceLocation.Location( + module = "", file = "", line = 0, column = offsets.sumOf { diff --git a/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToText.kt b/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToText.kt index 71f9ef6a28e76..21af6f94c16b7 100644 --- a/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToText.kt +++ b/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/convertors/WasmIrToText.kt @@ -83,7 +83,7 @@ class WasmIrToText( debugInformationGenerator?.addSourceLocation( SourceLocationMappingToText( it, - SourceLocation.Location("", stringBuilder.lineNumber, stringBuilder.columnNumber), + SourceLocation.Location("", "", stringBuilder.lineNumber, stringBuilder.columnNumber), ) ) } diff --git a/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/source/location/SourceLocation.kt b/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/source/location/SourceLocation.kt index 6024d37894d02..1666d2076dfb0 100644 --- a/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/source/location/SourceLocation.kt +++ b/wasm/wasm.ir/src/org/jetbrains/kotlin/wasm/ir/source/location/SourceLocation.kt @@ -9,9 +9,19 @@ sealed class SourceLocation { object NoLocation : SourceLocation() // Both line and column are zero-based - data class Location(val file: String, val line: Int, val column: Int) : SourceLocation() + data class Location( + val module: String, + val file: String, + val line: Int, + val column: Int + ) : SourceLocation() - data class IgnoredLocation(val file: String, val line: Int, val column: Int) : SourceLocation() + data class IgnoredLocation( + val module: String, + val file: String, + val line: Int, + val column: Int + ) : SourceLocation() companion object { @Suppress("FunctionName", "UNUSED_PARAMETER") diff --git a/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractFirWasmTest.kt b/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractFirWasmTest.kt index e1c98f5517424..e00850f62bd40 100644 --- a/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractFirWasmTest.kt +++ b/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractFirWasmTest.kt @@ -132,6 +132,7 @@ open class AbstractFirWasmJsSteppingTest : AbstractFirWasmJsTest( commonConfigurationForWasmBlackBoxCodegenTest() defaultDirectives { +WasmEnvironmentConfigurationDirectives.GENERATE_SOURCE_MAP + +WasmEnvironmentConfigurationDirectives.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES } } } diff --git a/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractK1WasmTest.kt b/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractK1WasmTest.kt index 4169d10bb8139..aa8bbffd74970 100644 --- a/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractK1WasmTest.kt +++ b/wasm/wasm.tests/test/org/jetbrains/kotlin/wasm/test/AbstractK1WasmTest.kt @@ -80,6 +80,7 @@ open class AbstractK1WasmSteppingTest : AbstractK1WasmTest( commonConfigurationForWasmBlackBoxCodegenTest() defaultDirectives { +WasmEnvironmentConfigurationDirectives.GENERATE_SOURCE_MAP + +WasmEnvironmentConfigurationDirectives.SOURCE_MAP_INCLUDE_MAPPINGS_FROM_UNAVAILABLE_FILES } } }