blob: e5cd8d29c64f31a4cccd8c350b346ad74ad6ef41 [file] [log] [blame] [edit]
#!/usr/bin/env node
/**
* @license
* Copyright 2010 The Emscripten Authors
* SPDX-License-Identifier: MIT
*/
// LLVM => JavaScript compiler, main entry point
import * as fs from 'node:fs';
import * as path from 'node:path';
import * as url from 'node:url';
import {Benchmarker, applySettings, assert, loadSettingsFile, printErr, read} from './utility.mjs';
function find(filename) {
assert(filename);
const dirname = url.fileURLToPath(new URL('.', import.meta.url));
const prefixes = [dirname, process.cwd()];
for (let i = 0; i < prefixes.length; ++i) {
const combined = path.join(prefixes[i], filename);
if (fs.existsSync(combined)) {
return combined;
}
}
return filename;
}
// Load default settings
loadSettingsFile(find('settings.js'));
loadSettingsFile(find('settings_internal.js'));
const argv = process.argv.slice(2);
const symbolsOnlyArg = argv.indexOf('--symbols-only');
if (symbolsOnlyArg != -1) {
argv.splice(symbolsOnlyArg, 1);
}
// Load settings from JSON passed on the command line
const settingsFile = argv[0];
assert(settingsFile);
const user_settings = JSON.parse(read(settingsFile));
applySettings(user_settings);
export const symbolsOnly = symbolsOnlyArg != -1;
// In case compiler.mjs is run directly (as in gen_sig_info)
// ALL_INCOMING_MODULE_JS_API might not be populated yet.
if (!ALL_INCOMING_MODULE_JS_API.length) {
ALL_INCOMING_MODULE_JS_API = INCOMING_MODULE_JS_API;
}
EXPORTED_FUNCTIONS = new Set(EXPORTED_FUNCTIONS);
WASM_EXPORTS = new Set(WASM_EXPORTS);
SIDE_MODULE_EXPORTS = new Set(SIDE_MODULE_EXPORTS);
INCOMING_MODULE_JS_API = new Set(INCOMING_MODULE_JS_API);
ALL_INCOMING_MODULE_JS_API = new Set(ALL_INCOMING_MODULE_JS_API);
EXPORTED_RUNTIME_METHODS = new Set(EXPORTED_RUNTIME_METHODS);
WEAK_IMPORTS = new Set(WEAK_IMPORTS);
if (symbolsOnly) {
INCLUDE_FULL_LIBRARY = 1;
}
// Side modules are pure wasm and have no JS
assert(
!SIDE_MODULE || (ASYNCIFY && symbolsOnly),
'JS compiler should only run on side modules if asyncify is used.',
);
// Load compiler code
// We can't use static import statements here because several of these
// file depend on having the settings defined in the global scope (which
// we do dynamically above.
await import('./modules.mjs');
await import('./parseTools.mjs');
if (!STRICT) {
await import('./parseTools_legacy.mjs');
}
const jsifier = await import('./jsifier.mjs');
// ===============================
// Main
// ===============================
const B = new Benchmarker();
try {
jsifier.runJSify(symbolsOnly);
B.print('glue');
} catch (err) {
if (err.toString().includes('Aborting compilation due to previous errors')) {
// Compiler failed on user error, don't print the stacktrace in this case.
printErr(err);
} else {
// Compiler failed on internal compiler error!
printErr('Internal compiler error in src/compiler.mjs!');
printErr('Please create a bug report at https://github.com/emscripten-core/emscripten/issues/');
printErr(
'with a log of the build and the input files used to run. Exception message: "' +
(err.stack || err),
);
}
// Work around a node.js bug where stdout buffer is not flushed at process exit:
// Instead of process.exit() directly, wait for stdout flush event.
// See https://github.com/joyent/node/issues/1669 and https://github.com/emscripten-core/emscripten/issues/2582
// Workaround is based on https://github.com/RReverser/acorn/commit/50ab143cecc9ed71a2d66f78b4aec3bb2e9844f6
process.stdout.once('drain', () => process.exit(1));
// Make sure to print something to force the drain event to occur, in case the
// stdout buffer was empty.
console.log(' ');
// Work around another node bug where sometimes 'drain' is never fired - make
// another effort to emit the exit status, after a significant delay (if node
// hasn't fired drain by then, give up)
setTimeout(() => process.exit(1), 500);
}