mirror of
https://github.com/compiler-explorer/compiler-explorer.git
synced 2025-12-27 10:33:59 -05:00
- Made parsers stateful instances instead of shared static state (for mllvm options). Fixes #8011 as this is caused by multiple clang-based compilers being run concurrently and stomping over each others' state. - passes `Compiler` to the constructor, which removes some param passing - Added some missing awaits - Tried to get things less dependent on `examples`, only `go` needs it - Spotted that `zig` c++ might have issues in discovery - Fly-by fixed a broken go path in ppc64le_gl122 - removed a redundant override in coccinelle - made the mojo parser actually use the parser it defined - canonified tablegen's special method - I changed the zig parser too but as best I can tell it was broken before (the `1` return value from the command it runs:) ``` ubuntu@ip-172-30-0-164:/infra/.deploy$ /opt/compiler-explorer/zig-0.14.1/zig c++ -mllvm --help-list-hidden /infra/.deploy/examples/c++/default.cpp -S -o /tmp/output.s ... --x86-use-vzeroupper - Minimize AVX to SSE transition penalty --xcore-max-threads=<number> - Maximum number of threads (for emulation thread-local storage) /infra/.deploy/examples/c++/default.cpp:1:1: error: FileNotFound ``` return code 1 (means it's not cached) --------- Co-authored-by: Partouf <partouf@gmail.com>
173 lines
6.4 KiB
TypeScript
173 lines
6.4 KiB
TypeScript
// Copyright (c) 2023, 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 {Command} from 'commander';
|
|
import _ from 'underscore';
|
|
|
|
import {CompilerArguments} from './lib/compiler-arguments.js';
|
|
import * as Parsers from './lib/compilers/argument-parsers.js';
|
|
import {executeDirect} from './lib/exec.js';
|
|
import {logger} from './lib/logger.js';
|
|
import {BaseParser} from './lib/compilers/argument-parsers.js';
|
|
|
|
const program = new Command();
|
|
program
|
|
.name('compiler-args-app.ts')
|
|
.usage('--parser=<compilertype> --exe=<path> [--padding=<number>]')
|
|
.description(
|
|
'Extracts compiler arguments\nFor example: node --no-warnings=ExperimentalWarning --import=tsx compiler-args-app.ts --parser=clang --exe=/opt/compiler-explorer/clang-15.0.0/bin/clang++ --padding=50',
|
|
)
|
|
.requiredOption('--parser <type>', 'Compiler parser type')
|
|
.requiredOption('--exe <path>', 'Path to compiler executable')
|
|
.option('--padding <number>', 'Padding for output formatting', '40')
|
|
.option('--debug', 'Enable debug output');
|
|
|
|
program.parse();
|
|
const opts = program.opts();
|
|
|
|
if (opts.debug) logger.level = 'debug';
|
|
|
|
const compilerParsers = {
|
|
gcc: Parsers.GCCParser,
|
|
clang: Parsers.ClangParser,
|
|
ldc: Parsers.LDCParser,
|
|
erlang: Parsers.ErlangParser,
|
|
pascal: Parsers.PascalParser,
|
|
ispc: Parsers.ISPCParser,
|
|
java: Parsers.JavaParser,
|
|
kotlin: Parsers.KotlinParser,
|
|
scala: Parsers.ScalaParser,
|
|
vc: Parsers.VCParser,
|
|
rust: Parsers.RustParser,
|
|
mrustc: Parsers.MrustcParser,
|
|
num: Parsers.NimParser,
|
|
crystal: Parsers.CrystalParser,
|
|
ts: Parsers.TypeScriptNativeParser,
|
|
turboc: Parsers.TurboCParser,
|
|
toit: Parsers.ToitParser,
|
|
circle: Parsers.CircleParser,
|
|
ghc: Parsers.GHCParser,
|
|
tendra: Parsers.TendraParser,
|
|
golang: Parsers.GolangParser,
|
|
zig: Parsers.ZigParser,
|
|
};
|
|
|
|
class CompilerArgsApp {
|
|
parserName: string;
|
|
executable: string;
|
|
compiler: any;
|
|
pad: number;
|
|
|
|
constructor() {
|
|
this.parserName = opts.parser;
|
|
this.executable = opts.exe;
|
|
this.pad = Number.parseInt(opts.padding, 10);
|
|
this.compiler = {
|
|
compiler: {
|
|
exe: this.executable,
|
|
},
|
|
possibleArguments: new CompilerArguments('some-id'),
|
|
execCompilerCached: async (command: string, args: string[]) => {
|
|
return executeDirect(command, args, {}, fn => fn);
|
|
},
|
|
getDefaultExecOptions: () => {
|
|
return {
|
|
env: process.env,
|
|
cwd: process.cwd(),
|
|
timeout: 10000,
|
|
};
|
|
}
|
|
};
|
|
|
|
if (this.parserName === 'juliawrapper') {
|
|
this.compiler.compilerWrapperPath = 'etc/scripts/julia_wrapper.jl';
|
|
}
|
|
}
|
|
|
|
async getPossibleStdvers() {
|
|
const parser = this.getParser();
|
|
return await parser.getPossibleStdvers();
|
|
}
|
|
|
|
async getPossibleTargets() {
|
|
const parser = this.getParser();
|
|
return await parser.getPossibleTargets();
|
|
}
|
|
|
|
async getPossibleEditions() {
|
|
const parser = this.getParser();
|
|
return await parser.getPossibleEditions();
|
|
}
|
|
|
|
getParser(): BaseParser {
|
|
if (compilerParsers[this.parserName as keyof typeof compilerParsers]) {
|
|
return new (compilerParsers[this.parserName as keyof typeof compilerParsers])(this.compiler);
|
|
}
|
|
console.error('Unknown parser type');
|
|
process.exit(1);
|
|
}
|
|
|
|
async doTheParsing() {
|
|
const parser = this.getParser();
|
|
await parser.parse();
|
|
const options = this.compiler.possibleArguments.possibleArguments;
|
|
if (parser.hasSupportStartsWith(options, '--target=')) {
|
|
console.log('supportsTargetIs');
|
|
} else if (parser.hasSupportStartsWith(options, '--target ')) {
|
|
console.log('supportsTarget');
|
|
} else if (parser.hasSupportStartsWith(options, '-target ')) {
|
|
console.log('supportsHyphenTarget');
|
|
} else if (parser.hasSupportStartsWith(options, '--march=')) {
|
|
console.log('supportsMarch');
|
|
} else {
|
|
console.log('none of the things?');
|
|
}
|
|
}
|
|
|
|
async print() {
|
|
const args = _.keys(this.compiler.possibleArguments.possibleArguments);
|
|
for (const arg of args) {
|
|
const description = this.compiler.possibleArguments.possibleArguments[arg].description;
|
|
console.log(`${arg.padEnd(this.pad, ' ')} ${description}`);
|
|
}
|
|
|
|
console.log('Stdvers:');
|
|
console.log(await this.getPossibleStdvers());
|
|
console.log('Targets:');
|
|
console.log(await this.getPossibleTargets());
|
|
console.log('Editions:');
|
|
console.log(await this.getPossibleEditions());
|
|
|
|
console.log('supportsOptOutput:', !!this.compiler.compiler.supportsOptOutput);
|
|
console.log('supportsStackUsageOutput', !!this.compiler.compiler.supportsStackUsageOutput);
|
|
console.log('optPipeline:', this.compiler.compiler.optPipeline);
|
|
console.log('supportsGccDump', !!this.compiler.compiler.supportsGccDump);
|
|
}
|
|
}
|
|
|
|
const app = new CompilerArgsApp();
|
|
app.doTheParsing()
|
|
.then(() => app.print())
|
|
.catch(e => console.error(e));
|