// Copyright (c) 2021, Compiler Explorer Authors // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are met: // // * Redistributions of source code must retain the above copyright notice, // this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright // notice, this list of conditions and the following disclaimer in the // documentation and/or other materials provided with the distribution. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE // POSSIBILITY OF SUCH DAMAGE. import path from 'path'; import fs from 'fs-extra'; import {BaseCompiler} from '../base-compiler'; import {DotNetAsmParser} from '../parsers/asm-parser-dotnet'; class DotNetCompiler extends BaseCompiler { private rID: string; private targetFramework: string; private buildConfig: string; private nugetPackagesPath: string; private clrBuildDir: string; private additionalSources: string; private langVersion: string; constructor(compilerInfo, env) { super(compilerInfo, env); this.rID = this.compilerProps(`compiler.${this.compiler.id}.runtimeId`); this.targetFramework = this.compilerProps(`compiler.${this.compiler.id}.targetFramework`); this.buildConfig = this.compilerProps(`compiler.${this.compiler.id}.buildConfig`); this.nugetPackagesPath = this.compilerProps(`compiler.${this.compiler.id}.nugetPackages`); this.clrBuildDir = this.compilerProps(`compiler.${this.compiler.id}.clrDir`); this.additionalSources = this.compilerProps(`compiler.${this.compiler.id}.additionalSources`); this.langVersion = this.compilerProps(`compiler.${this.compiler.id}.langVersion`); this.asm = new DotNetAsmParser(); } get compilerOptions() { return ['publish', '-c', this.buildConfig, '--self-contained', '--runtime', this.rID, '-v', 'q', '--nologo']; } get configurableOptions() { return [ '--targetos', '--targetarch', '--instruction-set', '--singlemethodtypename', '--singlemethodname', '--singlemethodindex', '--singlemethodgenericarg', '--codegenopt', '--codegen-options', ]; } get configurableSwitches() { return [ '-O', '--optimize', '--Od', '--optimize-disabled', '--Os', '--optimize-space', '--Ot', '--optimize-time', ]; } override async runCompiler(compiler, options, inputFileName, execOptions) { if (!execOptions) { execOptions = this.getDefaultExecOptions(); } const programDir = path.dirname(inputFileName); const sourceFile = path.basename(inputFileName); const projectFilePath = path.join(programDir, `CompilerExplorer${this.lang.extensions[0]}proj`); const crossgen2Path = path.join(this.clrBuildDir, 'crossgen2', 'crossgen2.dll'); const programPublishPath = path.join( programDir, 'bin', this.buildConfig, this.targetFramework, this.rID, 'publish', ); const programDllPath = path.join(programPublishPath, 'CompilerExplorer.dll'); const projectFileContent = ` ${this.targetFramework} true CompilerExplorer ${this.langVersion} false ${this.langVersion === 'preview' ? 'true' : 'false'} https://api.nuget.org/v3/index.json;${this.additionalSources ? this.additionalSources : ''} `; execOptions.env.DOTNET_CLI_TELEMETRY_OPTOUT = 'true'; execOptions.env.DOTNET_SKIP_FIRST_TIME_EXPERIENCE = 'true'; execOptions.env.NUGET_PACKAGES = this.nugetPackagesPath; execOptions.env.DOTNET_NOLOGO = 'true'; execOptions.customCwd = programDir; await fs.writeFile(projectFilePath, projectFileContent); const crossgen2Options: string[] = []; const configurableOptions = this.configurableOptions; for (const configurableOption of configurableOptions) { const optionIndex = options.indexOf(configurableOption); if (optionIndex === -1 || optionIndex === options.length - 1) { continue; } crossgen2Options.push(options[optionIndex], options[optionIndex + 1]); } const configurableSwitches = this.configurableSwitches; for (const configurableSwitch of configurableSwitches) { const switchIndex = options.indexOf(configurableSwitch); if (switchIndex === -1) { continue; } crossgen2Options.push(options[switchIndex]); } const compilerResult = await super.runCompiler(compiler, this.compilerOptions, inputFileName, execOptions); if (compilerResult.code !== 0) { return compilerResult; } const crossgen2Result = await this.runCrossgen2( compiler, execOptions, crossgen2Path, programPublishPath, programDllPath, crossgen2Options, this.getOutputFilename(programDir, this.outputFilebase), ); if (crossgen2Result.code !== 0) { return crossgen2Result; } return compilerResult; } override optionsForFilter() { return this.compilerOptions; } async runCrossgen2(compiler, execOptions, crossgen2Path, publishPath, dllPath, options, outputPath) { const crossgen2Options = [ crossgen2Path, '-r', path.join(publishPath, '*'), dllPath, '-o', 'CompilerExplorer.r2r.dll', '--codegenopt', 'NgenDisasm=*', '--codegenopt', 'JitDiffableDasm=1', '--parallelism', '1', '--inputbubble', '--compilebubblegenerics', ].concat(options); const result = await this.exec(compiler, crossgen2Options, execOptions); result.inputFilename = dllPath; const transformedInput = result.filenameTransform(dllPath); this.parseCompilationOutput(result, transformedInput); await fs.writeFile( outputPath, result.stdout.map(o => o.text).reduce((a, n) => `${a}\n${n}`), ); return result; } } export class CSharpCompiler extends DotNetCompiler { static get key() { return 'csharp'; } } export class FSharpCompiler extends DotNetCompiler { static get key() { return 'fsharp'; } } export class VBCompiler extends DotNetCompiler { static get key() { return 'vb'; } }