blob: 25c14dbdb6fa8722189986533858ceecab8176ec [file] [log] [blame] [raw]
// Copyright (c) 2018, Mitch Kennedy
// 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 {PreliminaryCompilerInfo} from '../../types/compiler.interfaces';
import {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces';
import {BaseCompiler} from '../base-compiler';
import * as utils from '../utils';
export class AdaCompiler extends BaseCompiler {
static get key() {
return 'ada';
}
constructor(info: PreliminaryCompilerInfo, env) {
super(info, env);
this.compiler.supportsGccDump = true;
this.compiler.removeEmptyGccDump = true;
// used for all GNAT related panes (Expanded code, Tree)
this.compiler.supportsGnatDebugViews = true;
}
override getExecutableFilename(dirPath: string, outputFilebase: string, key?) {
// The name here must match the value used in the pragma Source_File
// in the user provided source.
return path.join(dirPath, 'example');
}
override getOutputFilename(dirPath: string, outputFilebase: string, key?: any): string {
// The basename here must match the value used in the pragma Source_File
// in the user provided source.
// Beware that GNAT is picky on output filename:
// - the basename must match the unit name (emits error otherwise)
// - can't do "-c -o foo.out", output must end with .o
// - "foo.o" may be used by intermediary file, so "-o foo.o" will not
// work if building an executable.
if (key && key.filters && key.filters.binary) {
return path.join(dirPath, 'example');
} else if (key && key.filters && key.filters.binaryObject) {
return path.join(dirPath, 'example.o');
} else {
return path.join(dirPath, 'example.s');
}
}
override prepareArguments(
userOptions: string[],
filters: ParseFiltersAndOutputOptions,
backendOptions: Record<string, any>,
inputFilename: string,
outputFilename: string,
libraries,
) {
backendOptions = backendOptions || {};
// super call is needed as it handles the GCC Dump files.
const backend_opts = super.optionsForBackend(backendOptions, outputFilename);
// gnatmake opts name {[-cargs opts] [-bargs opts] [-largs opts] [-margs opts]}
// ^ ^ ^ ^ ^
// | | | | |
// `- inputFilename | | | `-- for gnatmake
// | | `-- for linker
// | `-- for binder (unused here)
// `-- for compiler (gcc)
const gnatmake_opts: string[] = [];
const compiler_opts: string[] = [];
const binder_opts: string[] = [];
const linker_opts: string[] = [''];
if (this.compiler.adarts) {
gnatmake_opts.push(`--RTS=${this.compiler.adarts}`);
}
if (!filters.execute && backendOptions.produceGnatDebug && this.compiler.supportsGnatDebugViews)
// This is using stdout
gnatmake_opts.push('-gnatGL');
if (!filters.execute && backendOptions.produceGnatDebugTree && this.compiler.supportsGnatDebugViews)
// This is also using stdout
gnatmake_opts.push('-gnatdt');
gnatmake_opts.push(
'-g',
'-fdiagnostics-color=always',
'-eS', // output commands to stdout, they are not errors
inputFilename,
);
if (!filters.execute && !filters.binary && !filters.binaryObject) {
gnatmake_opts.push(
'-S', // Generate ASM
'-c', // Compile only
'-fverbose-asm', // Generate verbose ASM showing variables
);
// produce assembly output in outputFilename
compiler_opts.push('-o', outputFilename);
if (this.compiler.intelAsm && filters.intel) {
for (const opt of this.compiler.intelAsm.split(' ')) {
gnatmake_opts.push(opt);
}
}
} else if (filters.binaryObject) {
gnatmake_opts.push(
'-c', // Compile only
);
// produce assembly output in outputFilename
compiler_opts.push('-o', outputFilename);
if (this.compiler.intelAsm && filters.intel) {
for (const opt of this.compiler.intelAsm.split(' ')) {
gnatmake_opts.push(opt);
}
}
// if (this.compiler.intelAsm && filters.intel) {
// options = options.concat(this.compiler.intelAsm.split(' '));
// }
} else {
gnatmake_opts.push('-o', outputFilename);
}
// Spread the options coming from outside (user, backend or config options)
// in the correct list.
let part = 0; // 0: gnatmake, 1: compiler, 2: linker, 3: binder
for (const a of backend_opts.concat(utils.splitArguments(this.compiler.options), userOptions)) {
if (a === '-cargs') {
part = 1;
continue;
} else if (a === '-largs') {
part = 2;
continue;
} else if (a === '-bargs') {
part = 3;
continue;
} else if (a === '-margs') {
part = 0;
continue;
}
if (part === 0) {
gnatmake_opts.push(a);
} else if (part === 1) {
compiler_opts.push(a);
} else if (part === 2) {
linker_opts.push(a);
} else if (part === 3) {
binder_opts.push(a);
}
}
return gnatmake_opts.concat('-cargs', compiler_opts, '-largs', linker_opts, '-bargs', binder_opts);
}
}