blob: fe6e59439e79f69ca81cd33632955a23386c702a [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 { AsmRaw } from '../asm-raw';
import { BaseCompiler } from '../base-compiler';
import * as utils 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();
}
getSharedLibraryPathsAsArguments() {
return [];
}
getArgumentParser() {
return BaseParser;
}
optionsForFilter(filters) {
filters.binary = true;
return [];
}
getGeneratedOutputFilename(fn) {
const outputFolder = path.dirname(fn);
const files = fs.readdirSync(outputFolder);
let outputFilename = super.filename(fn);
files.forEach(file => {
if (file[0] !== '.' && file !== this.compileFilename) {
outputFilename = path.join(outputFolder, file);
}
});
return outputFilename;
}
getOutputFilename(dirPath) {
return this.getGeneratedOutputFilename(path.join(dirPath, 'example.asm'));
}
async runCompiler(compiler, options, inputFilename, execOptions) {
if (!execOptions) {
execOptions = this.getDefaultExecOptions();
}
execOptions.customCwd = path.dirname(inputFilename);
const result = await this.exec(compiler, options, execOptions);
result.inputFilename = inputFilename;
result.stdout = utils.parseOutput(result.stdout, inputFilename);
result.stderr = utils.parseOutput(result.stderr, inputFilename);
return result;
}
async runReadelf(fullResult, objectFilename) {
let 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) {
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, inputArch, objectFilename, outputFilename) {
let execOptions = this.getDefaultExecOptions();
execOptions.customCwd = path.dirname(objectFilename);
const options = ['-o', outputFilename];
if (inputArch === 'x86') {
options.push('-m');
options.push('elf_i386');
} else if (inputArch === 'x86_64') {
// default target
} else {
const result = {
code: -1,
step: 'ld',
stderr: [{text: 'Invalid architecture for linking and execution'}],
};
fullResult.buildsteps.push(result);
return result;
}
options.push(objectFilename);
return this.doBuildstepAndAddToResult(fullResult, 'ld', this.env.ceProps('ld'), options, execOptions);
}
getExecutableFilename(dirPath) {
return path.join(dirPath, 'ce-asm-executable');
}
async buildExecutable(compiler, options, inputFilename, execOptions) {
const fullResult = {
code: -1,
executableFilename: '',
buildsteps: [],
};
const compilationResult = await this.runCompiler(compiler, options, inputFilename, execOptions);
compilationResult.step = 'Assembling';
fullResult.buildsteps.push(compilationResult);
const dirPath = path.dirname(inputFilename);
fullResult.executableFilename = this.getExecutableFilename(dirPath);
const objectFilename = this.getOutputFilename(dirPath);
const inputArch = await this.getArchitecture(fullResult, objectFilename);
const ldResult = await this.runLinker(fullResult, inputArch, objectFilename, fullResult.executableFilename);
fullResult.code = ldResult.code;
if (ldResult.stderr.length > 0) {
fullResult.stderr = ldResult.stderr;
}
return fullResult;
}
checkOutputFileAndDoPostProcess(asmResult, outputFilename, filters) {
return this.postProcess(asmResult, outputFilename, filters);
}
getObjdumpOutputFilename(defaultOutputFilename) {
return this.getGeneratedOutputFilename(defaultOutputFilename);
}
}