// Copyright (c) 2018, Microsoft Corporation // 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 _ from 'underscore'; import type {ExecutionOptions} from '../../types/compilation/compilation.interfaces.js'; import type {ConfiguredOverrides} from '../../types/compilation/compiler-overrides.interfaces.js'; import type {PreliminaryCompilerInfo} from '../../types/compiler.interfaces.js'; import type {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces.js'; import {unwrap} from '../assert.js'; import {BaseCompiler} from '../base-compiler.js'; import {MapFileReaderVS} from '../mapfiles/map-file-vs.js'; import {AsmParser} from '../parsers/asm-parser.js'; import {PELabelReconstructor} from '../pe32-support.js'; import * as utils from '../utils.js'; export class Win32Compiler extends BaseCompiler { static get key() { return 'win32'; } binaryAsmParser: AsmParser; constructor(compilerInfo: PreliminaryCompilerInfo, env) { super(compilerInfo, env); this.binaryAsmParser = new AsmParser(this.compilerProps); } override getStdverFlags(): string[] { return ['/std:']; } override getExecutableFilename(dirPath: string, outputFilebase: string, key?) { return this.getOutputFilename(dirPath, outputFilebase, key) + '.exe'; } override getObjdumpOutputFilename(defaultOutputFilename: string) { return this.getExecutableFilename(path.dirname(defaultOutputFilename), 'output'); } override getSharedLibraryPathsAsArguments(libraries) { const libPathFlag = this.compiler.libpathFlag || '/LIBPATH:'; return this.getSharedLibraryPaths(libraries).map(path => libPathFlag + path); } override getSharedLibraryLinks(libraries: any[]): string[] { return _.flatten( libraries .map(selectedLib => [selectedLib, this.findLibVersion(selectedLib)]) .filter(([selectedLib, foundVersion]) => !!foundVersion) .map(([selectedLib, foundVersion]) => { return foundVersion.liblink.filter(Boolean).map(lib => `"${lib}.lib"`); }) .map(([selectedLib, foundVersion]) => selectedLib), ); } override getStaticLibraryLinks(libraries) { return super.getSortedStaticLibraries(libraries).map(lib => { return '"' + lib + '.lib"'; }); } override prepareArguments( userOptions: string[], filters: ParseFiltersAndOutputOptions, backendOptions: Record, inputFilename: string, outputFilename: string, libraries, overrides: ConfiguredOverrides, ) { let options = this.optionsForFilter(filters, outputFilename, userOptions); backendOptions = backendOptions || {}; if (this.compiler.options) { options = options.concat(utils.splitArguments(this.compiler.options)); } if (this.compiler.supportsOptOutput && backendOptions.produceOptInfo) { options = options.concat(unwrap(this.compiler.optArg)); } const libIncludes = this.getIncludeArguments(libraries, path.dirname(inputFilename)); const libOptions = this.getLibraryOptions(libraries); let libLinks: any[] = []; let libPaths: string[] = []; let preLink: string[] = []; let staticlibLinks: string[] = []; if (filters.binary) { preLink = ['/link']; libLinks = this.getSharedLibraryLinks(libraries); libPaths = this.getSharedLibraryPathsAsArguments(libraries); staticlibLinks = this.getStaticLibraryLinks(libraries); } userOptions = this.filterUserOptions(userOptions) || []; return options.concat( libIncludes, libOptions, userOptions, [this.filename(inputFilename)], preLink, libPaths, libLinks, staticlibLinks, ); } override optionsForFilter(filters: ParseFiltersAndOutputOptions, outputFilename: string, userOptions?: string[]) { if (filters.binary) { const mapFilename = outputFilename + '.map'; const mapFileReader = new MapFileReaderVS(mapFilename); (filters as any).preProcessBinaryAsmLines = asmLines => { const reconstructor = new PELabelReconstructor(asmLines, false, mapFileReader); reconstructor.run('output.s.obj'); return reconstructor.asmLines; }; return [ '/nologo', '/FA', '/Fa' + this.filename(outputFilename.replace(/\.exe$/, '')), '/Fo' + this.filename(outputFilename.replace(/\.exe$/, '') + '.obj'), '/Fm' + this.filename(mapFilename), '/Fe' + this.filename(this.getExecutableFilename(path.dirname(outputFilename), 'output')), ]; } else { return [ '/nologo', '/FA', '/c', '/Fa' + this.filename(outputFilename), '/Fo' + this.filename(outputFilename + '.obj'), ]; } } override async processAsm(result, filters /*, options*/) { if (filters.binary) { filters.dontMaskFilenames = true; return this.binaryAsmParser.process(result.asm, filters); } else { return this.asm.process(result.asm, filters); } } override exec(compiler: string, args: string[], options_: ExecutionOptions) { const options = Object.assign({}, options_); options.env = Object.assign({}, options.env); if (this.compiler.includePath) { options.env['INCLUDE'] = this.compiler.includePath; } if (this.compiler.libPath) { options.env['LIB'] = this.compiler.libPath.join(';'); } for (const [env, to] of this.compiler.envVars) { options.env[env] = to; } return super.exec(compiler, args, options); } }