blob: c7b0b7ecffbd0cb87581fbe196a7d73aa2902467 [file] [log] [blame] [raw]
// Copyright (c) 2022, 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 {ExecutionOptions} from '../../types/compilation/compilation.interfaces';
import {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces';
import {BaseCompiler} from '../base-compiler';
import {logger} from '../logger';
import {AsmParserZ88dk} from '../parsers/asm-parser-z88dk';
import * as utils from '../utils';
export class z88dkCompiler extends BaseCompiler {
static get key() {
return 'z88dk';
}
constructor(compilerInfo, env) {
super(compilerInfo, env);
this.outputFilebase = 'example';
this.asm = new AsmParserZ88dk(this.compilerProps);
}
public override getOutputFilename(dirPath: string, outputFilebase: string, key?: any): string {
let filename;
if (key && key.backendOptions && key.backendOptions.customOutputFilename) {
filename = key.backendOptions.customOutputFilename;
} else if (key && key.filters.binary) {
filename = `${outputFilebase}`;
} else {
filename = `${outputFilebase}.c.asm`;
}
if (dirPath) {
return path.join(dirPath, filename);
} else {
return filename;
}
}
public override orderArguments(
options,
inputFilename,
libIncludes,
libOptions,
libPaths,
libLinks,
userOptions,
staticLibLinks,
) {
return userOptions.concat(
options,
[this.filename(inputFilename)],
libIncludes,
libOptions,
libPaths,
libLinks,
staticLibLinks,
);
}
protected override optionsForFilter(filters: ParseFiltersAndOutputOptions, outputFilename: string): string[] {
if (!filters.binary) {
return ['-S'];
} else {
return ['-o', outputFilename + '.s', '-create-app'];
}
}
override getDefaultExecOptions(): ExecutionOptions {
const opts = super.getDefaultExecOptions();
opts.env.ZCCCFG = path.join(path.dirname(this.compiler.exe), '../share/z88dk/lib/config');
opts.env.PATH = process.env.PATH + path.delimiter + path.dirname(this.compiler.exe);
return opts;
}
override getObjdumpOutputFilename(defaultOutputFilename) {
return defaultOutputFilename;
}
getTapefilename() {
return `${this.outputFilebase}.tap`;
}
getSmsfilename() {
return `${this.outputFilebase}.sms`;
}
override async objdump(
outputFilename,
result: any,
maxSize: number,
intelAsm,
demangle,
filters: ParseFiltersAndOutputOptions,
) {
outputFilename = this.getObjdumpOutputFilename(outputFilename);
// sometimes (with +z80 for example) the .bin file is written and the .s file is empty
if (!(await utils.fileExists(outputFilename + '.bin'))) {
if (!(await utils.fileExists(outputFilename + '.s'))) {
result.asm = '<No output file ' + outputFilename + '.s>';
return result;
} else {
outputFilename += '.s';
}
} else {
outputFilename += '.bin';
}
const args = [outputFilename];
if (this.externalparser) {
const objResult = await this.externalparser.objdumpAndParseAssembly(result.dirPath, args, filters);
if (objResult.parsingTime !== undefined) {
objResult.objdumpTime = parseInt(result.execTime) - parseInt(result.parsingTime);
delete objResult.execTime;
}
result = {...result, ...objResult};
} else {
const execOptions: ExecutionOptions = {
maxOutput: maxSize,
customCwd: (result.dirPath as string) || path.dirname(outputFilename),
};
const objResult = await this.exec(this.compiler.objdumper, args, execOptions);
if (objResult.code !== 0) {
logger.error(`Error executing objdump ${this.compiler.objdumper}`, objResult);
result.asm = `<No output: objdump returned ${objResult.code}>`;
} else {
result.objdumpTime = objResult.execTime;
result.asm = this.postProcessObjdumpOutput(objResult.stdout);
}
}
if (result.code === 0 && filters.binary) {
const tapeFilepath = path.join(result.dirPath, this.getTapefilename());
if (await utils.fileExists(tapeFilepath)) {
const file_buffer = await fs.readFile(tapeFilepath);
const binary_base64 = file_buffer.toString('base64');
result.speccytape = binary_base64;
}
const smsFilepath = path.join(result.dirPath, this.getSmsfilename());
if (await utils.fileExists(smsFilepath)) {
const file_buffer = await fs.readFile(smsFilepath);
const binary_base64 = file_buffer.toString('base64');
result.miraclesms = binary_base64;
}
}
return result;
}
}