// Copyright (c) 2017, 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 type {ExecutionOptions} from '../../types/compilation/compilation.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 {MapFileReaderDelphi} from '../mapfiles/map-file-delphi.js'; import {PELabelReconstructor} from '../pe32-support.js'; import * as utils from '../utils.js'; import {PascalUtils} from './pascal-utils.js'; export class PascalWinCompiler extends BaseCompiler { static get key() { return 'pascal-win'; } mapFilename: string | null; dprFilename: string; pasUtils: PascalUtils; constructor(info: PreliminaryCompilerInfo, env) { super(info, env); info.supportsFiltersInBinary = true; this.mapFilename = null; this.compileFilename = 'output.pas'; this.dprFilename = 'prog.dpr'; this.pasUtils = new PascalUtils(); } override getSharedLibraryPathsAsArguments() { return []; } override exec(command: string, args: string[], options: ExecutionOptions) { if (process.platform === 'linux' || process.platform === 'darwin') { const wine = this.env.ceProps('wine'); args = args.slice(0); if (command.toLowerCase().endsWith('.exe')) { args.unshift(command); command = wine; } } return super.exec(command, args, options); } override getExecutableFilename(dirPath: string) { return path.join(dirPath, 'prog.exe'); } override getOutputFilename(dirPath: string) { return path.join(dirPath, 'prog.exe'); } override filename(fn) { if (process.platform === 'linux' || process.platform === 'darwin') { return 'Z:' + fn; } else { return super.filename(fn); } } override async objdump(outputFilename, result, maxSize: number, intelAsm) { const dirPath = path.dirname(outputFilename); const execBinary = this.getExecutableFilename(dirPath); if (await utils.fileExists(execBinary)) { outputFilename = execBinary; } else { outputFilename = this.getOutputFilename(path.dirname(outputFilename)); } let args = [...this.compiler.objdumperArgs, '-d', outputFilename]; if (intelAsm) args = args.concat(['-M', 'intel']); return this.exec(this.compiler.objdumper, args, {maxOutput: 1024 * 1024 * 1024}).then(objResult => { if (objResult.code === 0) { result.asm = objResult.stdout; } else { result.asm = ''; } return result; }); } async saveDummyProjectFile(filename: string, unitName: string, unitPath: string) { await fs.writeFile( filename, // prettier-ignore 'program prog;\n' + 'uses ' + unitName + ' in \'' + unitPath + '\';\n' + 'begin\n' + 'end.\n', ); } override async writeAllFiles(dirPath: string, source: string, files: any[], filters: ParseFiltersAndOutputOptions) { let inputFilename; if (this.pasUtils.isProgram(source)) { inputFilename = path.join(dirPath, this.dprFilename); } else { const unitName = this.pasUtils.getUnitname(source); if (unitName) { inputFilename = path.join(dirPath, unitName + '.pas'); } else { inputFilename = path.join(dirPath, this.compileFilename); } } await fs.writeFile(inputFilename, source); if (files && files.length > 0) { await this.writeMultipleFiles(files, dirPath); } return { inputFilename, }; } override async runCompiler( compiler: string, options: string[], inputFilename: string, execOptions: ExecutionOptions & {env: Record}, ) { if (!execOptions) { execOptions = this.getDefaultExecOptions(); } const alreadyHasDPR = path.basename(inputFilename) === this.dprFilename; const tempPath = path.dirname(inputFilename); const projectFile = path.join(tempPath, this.dprFilename); this.mapFilename = path.join(tempPath, 'prog.map'); inputFilename = inputFilename.replace(/\//g, '\\'); if (!alreadyHasDPR) { const unitFilepath = path.basename(inputFilename); const unitName = unitFilepath.replace(/.pas$/i, ''); await this.saveDummyProjectFile(projectFile, unitName, unitFilepath); } options.pop(); options.unshift('-CC', '-W', '-H', '-GD', '-$D+', '-V', '-B'); options.push(projectFile); execOptions.customCwd = tempPath; return this.exec(compiler, options, execOptions).then(result => { return { ...result, inputFilename, stdout: utils.parseOutput(result.stdout, inputFilename), stderr: utils.parseOutput(result.stderr, inputFilename), }; }); } override optionsForFilter(filters: ParseFiltersAndOutputOptions) { filters.binary = true; filters.dontMaskFilenames = true; (filters as any).preProcessBinaryAsmLines = asmLines => { const mapFileReader = new MapFileReaderDelphi(unwrap(this.mapFilename)); const reconstructor = new PELabelReconstructor(asmLines, false, mapFileReader, false); reconstructor.run('output'); return reconstructor.asmLines; }; return []; } }