mirror of
https://github.com/compiler-explorer/compiler-explorer.git
synced 2025-12-27 10:33:59 -05:00
205 lines
8.1 KiB
TypeScript
205 lines
8.1 KiB
TypeScript
// 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 'node:path';
|
|
|
|
import {splitArguments} from '../../shared/common-utils.js';
|
|
import type {ConfiguredOverrides} from '../../types/compilation/compiler-overrides.interfaces.js';
|
|
import type {PreliminaryCompilerInfo} from '../../types/compiler.interfaces.js';
|
|
import type {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces.js';
|
|
import {SelectedLibraryVersion} from '../../types/libraries/libraries.interfaces.js';
|
|
import {unwrap} from '../assert.js';
|
|
import {BaseCompiler} from '../base-compiler.js';
|
|
import {CompilationEnvironment} from '../compilation-env.js';
|
|
|
|
export class AdaCompiler extends BaseCompiler {
|
|
static get key() {
|
|
return 'ada';
|
|
}
|
|
|
|
constructor(info: PreliminaryCompilerInfo, env: CompilationEnvironment) {
|
|
super(info, env);
|
|
|
|
this.outputFilebase = 'example';
|
|
|
|
this.compiler.supportsGccDump = true;
|
|
this.compiler.removeEmptyGccDump = true;
|
|
|
|
// this is not showing-up in the --help, so argument parser doesn't
|
|
// automatically detect the support.
|
|
this.compiler.stackUsageArg = '-fstack-usage';
|
|
this.compiler.supportsStackUsageOutput = true;
|
|
|
|
// used for all GNAT related panes (Expanded code, Tree)
|
|
this.compiler.supportsGnatDebugViews = true;
|
|
}
|
|
|
|
override getExecutableFilename(dirPath: string, outputFilebase: string) {
|
|
// 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?.filters?.binary) {
|
|
return path.join(dirPath, 'example');
|
|
}
|
|
if (key?.filters?.binaryObject) {
|
|
return path.join(dirPath, 'example.o');
|
|
}
|
|
return path.join(dirPath, 'example.s');
|
|
}
|
|
|
|
override prepareArguments(
|
|
userOptions: string[],
|
|
filters: ParseFiltersAndOutputOptions,
|
|
backendOptions: Record<string, any>,
|
|
inputFilename: string,
|
|
outputFilename: string,
|
|
libraries: SelectedLibraryVersion[],
|
|
overrides: ConfiguredOverrides,
|
|
) {
|
|
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 (this.compiler.supportsStackUsageOutput && backendOptions.produceStackUsageInfo) {
|
|
gnatmake_opts.push(unwrap(this.compiler.stackUsageArg));
|
|
}
|
|
|
|
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(splitArguments(this.compiler.options), userOptions)) {
|
|
if (a === '-cargs') {
|
|
part = 1;
|
|
continue;
|
|
}
|
|
if (a === '-largs') {
|
|
part = 2;
|
|
continue;
|
|
}
|
|
if (a === '-bargs') {
|
|
part = 3;
|
|
continue;
|
|
}
|
|
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);
|
|
}
|
|
}
|