mirror of
https://github.com/compiler-explorer/compiler-explorer.git
synced 2025-12-27 07:04:04 -05:00
add support for erlang to beam bytecode (#3094)
* add support for erlang to beam bytecode * changed call by 'this' to overridden functions * Update base-compiler.js Co-authored-by: Patrick Quist <partouf@gmail.com>
This commit is contained in:
20
etc/config/erlang.defaults.properties
Normal file
20
etc/config/erlang.defaults.properties
Normal file
@@ -0,0 +1,20 @@
|
||||
# Default settings for Erlang
|
||||
compilers=&erlang
|
||||
compilerType=erlang
|
||||
versionFlag=
|
||||
objdumper=
|
||||
instructionSet=beam
|
||||
|
||||
group.erlang.compilers=erlcdefault
|
||||
compiler.erlcdefault.exe=/usr/bin/erl
|
||||
compiler.erlcdefault.runtime=/usr/bin/erl
|
||||
compiler.erlcdefault.name=erl default
|
||||
|
||||
defaultCompiler=erlcdefault
|
||||
demangler=
|
||||
postProcess=
|
||||
options=-S
|
||||
supportsBinary=false
|
||||
needsMulti=false
|
||||
supportsExecute=false
|
||||
interpreted=true
|
||||
4
examples/erlang/default.erl
Normal file
4
examples/erlang/default.erl
Normal file
@@ -0,0 +1,4 @@
|
||||
-module(square).
|
||||
-export([square/1]).
|
||||
|
||||
square(A) -> A*A.
|
||||
@@ -37,6 +37,7 @@ export { CrystalCompiler } from './crystal';
|
||||
export { DefaultCompiler } from './default';
|
||||
export { DMDCompiler } from './dmd';
|
||||
export { EllccCompiler } from './ellcc';
|
||||
export { ErlangCompiler } from './erlang';
|
||||
export { EWARMCompiler } from './ewarm';
|
||||
export { EWAVRCompiler } from './ewavr';
|
||||
export { FakeCompiler } from './fake-for-test';
|
||||
|
||||
@@ -157,6 +157,13 @@ export class ClangParser extends BaseParser {
|
||||
}
|
||||
}
|
||||
|
||||
export class ErlangParser extends BaseParser {
|
||||
static async parse(compiler) {
|
||||
await ErlangParser.getOptions(compiler, '-help');
|
||||
return compiler;
|
||||
}
|
||||
}
|
||||
|
||||
export class PascalParser extends BaseParser {
|
||||
static async parse(compiler) {
|
||||
await PascalParser.getOptions(compiler, '-help');
|
||||
|
||||
96
lib/compilers/erlang.js
Normal file
96
lib/compilers/erlang.js
Normal file
@@ -0,0 +1,96 @@
|
||||
// Copyright (c) 2021, 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 { BaseCompiler } from '../base-compiler';
|
||||
import { logger } from '../logger';
|
||||
|
||||
import { ErlangParser } from './argument-parsers';
|
||||
|
||||
export class ErlangCompiler extends BaseCompiler {
|
||||
static get key() { return 'erlang'; }
|
||||
|
||||
/* eslint-disable no-unused-vars */
|
||||
optionsForFilter(filters, outputFilename) {
|
||||
return [
|
||||
'-noshell',
|
||||
'-eval',
|
||||
'{ok, Input} = init:get_argument(input),' +
|
||||
'{ok, _, Output} = compile:file(Input, [\'S\', binary, no_line_info, report]),' +
|
||||
'Binary = io_lib:format("~p", [Output]),' +
|
||||
`file:write_file("${outputFilename}", list_to_binary(Binary)),` +
|
||||
'halt().',
|
||||
];
|
||||
}
|
||||
/* eslint-enable no-unused-vars */
|
||||
|
||||
async runCompiler(compiler, options, inputFilename, execOptions) {
|
||||
if (!execOptions) {
|
||||
execOptions = this.getDefaultExecOptions();
|
||||
}
|
||||
|
||||
if (!execOptions.customCwd) {
|
||||
execOptions.customCwd = path.dirname(inputFilename);
|
||||
}
|
||||
|
||||
const result = await this.exec(compiler, options.concat(['-input', inputFilename]), execOptions);
|
||||
result.inputFilename = inputFilename;
|
||||
const transformedInput = result.filenameTransform(inputFilename);
|
||||
this.parseCompilationOutput(result, transformedInput);
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
getVersion() {
|
||||
logger.info(`Gathering ${this.compiler.id} version information on ${this.compiler.exe}...`);
|
||||
if (this.compiler.explicitVersion) {
|
||||
logger.debug(`${this.compiler.id} has forced version output: ${this.compiler.explicitVersion}`);
|
||||
return {stdout: [this.compiler.explicitVersion], stderr: [], code: 0};
|
||||
}
|
||||
const execOptions = this.getDefaultExecOptions();
|
||||
const versionCmd = this.compilerProps(`compiler.${this.compiler.id}.runtime`);
|
||||
execOptions.timeoutMs = 0; // No timeout for --version. A sort of workaround for slow EFS/NFS on the prod site
|
||||
execOptions.ldPath = this.getSharedLibraryPathsAsLdLibraryPaths([]);
|
||||
|
||||
try {
|
||||
return this.execCompilerCached(
|
||||
versionCmd,
|
||||
['-noshell', '-eval', 'io:fwrite("~s~n", [erlang:system_info(otp_release)]), halt().'],
|
||||
execOptions,
|
||||
);
|
||||
} catch (err) {
|
||||
logger.error(`Unable to get version for compiler '${this.compiler.exe}' - ${err}`);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
getOutputFilename(dirPath, outputFilebase) {
|
||||
return path.join(dirPath, `${outputFilebase}.S`);
|
||||
}
|
||||
|
||||
getArgumentParser() {
|
||||
return ErlangParser;
|
||||
}
|
||||
}
|
||||
@@ -116,6 +116,12 @@ export const languages = {
|
||||
extensions: ['.d'],
|
||||
alias: [],
|
||||
},
|
||||
erlang: {
|
||||
name: 'Erlang',
|
||||
monaco: 'erlang',
|
||||
extensions: ['.erl', '.hrl'],
|
||||
alias: [],
|
||||
},
|
||||
go: {
|
||||
name: 'Go',
|
||||
monaco: 'go',
|
||||
|
||||
Reference in New Issue
Block a user