blob: c7334e26d0086052533fca6355b3d01f76675671 [file] [log] [blame] [raw]
// Copyright (c) 2018, 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 fs from 'fs';
import path from 'path';
import _ from 'underscore';
import {BuildResult, BuildStep} from '../../types/compilation/compilation.interfaces';
import {ParseFilters} from '../../types/features/filters.interfaces';
import {BaseCompiler} from '../base-compiler';
import {AsmRaw} from '../parsers/asm-raw';
import {fileExists, splitArguments} from '../utils';
import {BaseParser} from './argument-parsers';
export class AssemblyCompiler extends BaseCompiler {
static get key() {
return 'assembly';
}
constructor(info, env) {
super(info, env);
this.asm = new AsmRaw();
}
override getSharedLibraryPathsAsArguments() {
return [];
}
override getArgumentParser() {
return BaseParser;
}
override optionsForFilter(filters): string[] {
filters.binary = true;
return [];
}
getGeneratedOutputFilename(fn) {
const outputFolder = path.dirname(fn);
const files = fs.readdirSync(outputFolder);
let outputFilename = super.filename(fn);
for (const file of files) {
if (file[0] !== '.' && file !== this.compileFilename) {
outputFilename = path.join(outputFolder, file);
}
}
return outputFilename;
}
override getOutputFilename(dirPath) {
return this.getGeneratedOutputFilename(path.join(dirPath, 'example.asm'));
}
async runReadelf(fullResult, objectFilename) {
const execOptions = this.getDefaultExecOptions();
execOptions.customCwd = path.dirname(objectFilename);
return await this.doBuildstepAndAddToResult(
fullResult,
'readelf',
this.env.ceProps('readelf'),
['-h', objectFilename],
execOptions,
);
}
async getArchitecture(fullResult, objectFilename): Promise<string | false> {
const result = await this.runReadelf(fullResult, objectFilename);
const output = result.stdout.map(line => line.text).join('\n');
if (output.includes('ELF32') && output.includes('80386')) {
return 'x86';
} else if (output.includes('ELF64') && output.includes('X86-64')) {
return 'x86_64';
} else if (output.includes('Mach-O 64-bit x86-64')) {
// note: this is to support readelf=objdump on Mac
return 'x86_64';
}
return false;
}
async runLinker(
fullResult: BuildResult,
inputArch: string | false,
objectFilename: string,
outputFilename: string,
extraOptions: string[],
) {
const execOptions = this.getDefaultExecOptions();
execOptions.customCwd = path.dirname(objectFilename);
let options = ['-o', outputFilename];
if (inputArch === 'x86') {
options.push('-m', 'elf_i386');
} else if (inputArch === 'x86_64') {
// default target
} else {
const result: BuildStep = {
code: -1,
okToCache: false,
execTime: '0',
step: 'ld',
timedOut: false,
filenameTransform: f => f,
stderr: [{text: 'Invalid architecture for linking and execution'}],
stdout: [],
};
if (fullResult.buildsteps) fullResult.buildsteps.push(result);
return result;
}
options.push(objectFilename);
options = options.concat(extraOptions);
return this.doBuildstepAndAddToResult(fullResult, 'ld', this.env.ceProps('ld'), options, execOptions);
}
override getExecutableFilename(dirPath) {
return path.join(dirPath, 'ce-asm-executable');
}
override prepareArguments(
userOptions: string[],
filters: ParseFilters,
backendOptions: Record<string, any>,
inputFilename: string,
outputFilename: string,
libraries,
) {
let options = this.optionsForFilter(filters);
backendOptions = backendOptions || {};
options = options.concat(this.optionsForBackend(backendOptions, outputFilename));
if (this.compiler.options) {
options = options.concat(splitArguments(this.compiler.options));
}
if (this.compiler.supportsOptOutput && backendOptions.produceOptInfo) {
options = options.concat(this.compiler.optArg);
}
const libIncludes = this.getIncludeArguments(libraries);
userOptions = this.filterUserOptions(userOptions) || [];
options = this.fixIncompatibleOptions(options, userOptions);
return this.orderArguments(options, inputFilename, libIncludes, [], [], [], userOptions, []);
}
override async buildExecutableInFolder(key, dirPath): Promise<BuildResult> {
const buildEnvironment = this.setupBuildEnvironment(key, dirPath, true);
const writeSummary = await this.writeAllFiles(dirPath, key.source, key.files, key.filters);
const inputFilename = writeSummary.inputFilename;
const outputFilename = this.getExecutableFilename(dirPath);
const buildFilters: ParseFilters = Object.assign({}, key.filters);
buildFilters.binary = true;
buildFilters.execute = false;
const compilerArguments = _.compact(
this.prepareArguments(
key.options,
buildFilters,
key.backendOptions,
inputFilename,
outputFilename,
key.libraries,
),
);
const execOptions = this.getDefaultExecOptions();
execOptions.ldPath = this.getSharedLibraryPathsAsLdLibraryPaths(key.libraries);
const downloads = await buildEnvironment;
const result = await this.buildExecutable(key.compiler.exe, compilerArguments, inputFilename, execOptions);
const fullResult: BuildResult = {
...result,
buildsteps: [],
downloads,
executableFilename: outputFilename,
compilationOptions: compilerArguments,
};
const objectFilename = this.getOutputFilename(dirPath);
if (objectFilename !== inputFilename && (await fileExists(objectFilename))) {
const inputArch = await this.getArchitecture(fullResult, objectFilename);
const libOpts = this.getSharedLibraryLinks(key.libraries);
const ldResult = await this.runLinker(fullResult, inputArch, objectFilename, outputFilename, libOpts);
fullResult.stderr = fullResult.stderr.concat(ldResult.stderr);
}
return fullResult;
}
override checkOutputFileAndDoPostProcess(asmResult, outputFilename, filters) {
return this.postProcess(asmResult, outputFilename, filters);
}
override getObjdumpOutputFilename(defaultOutputFilename) {
return this.getGeneratedOutputFilename(defaultOutputFilename);
}
}