blob: 179cdc48d21eea811128e185d58b9dd5d3dc9fe5 [file] [log] [blame]
/**
* @license
* Copyright 2010 The Emscripten Authors
* SPDX-License-Identifier: MIT
*/
// === Auto-generated postamble setup entry stuff ===
{{{ exportRuntime() }}}
var calledRun;
#if STANDALONE_WASM && MAIN_READS_PARAMS
var mainArgs = undefined;
#endif
dependenciesFulfilled = function runCaller() {
// If run has never been called, and we should call run (INVOKE_RUN is true, and Module.noInitialRun is not false)
if (!calledRun) run();
if (!calledRun) dependenciesFulfilled = runCaller; // try this again later, after new deps are fulfilled
};
#if HAS_MAIN
#if MAIN_READS_PARAMS
function callMain(args = []) {
#else
function callMain() {
#endif
#if ASSERTIONS
assert(runDependencies == 0, 'cannot call main when async dependencies remain! (listen on Module["onRuntimeInitialized"])');
assert(__ATPRERUN__.length == 0, 'cannot call main when preRun functions remain to be called');
#endif
var entryFunction = {{{ getEntryFunction() }}};
#if PROXY_TO_PTHREAD
// With PROXY_TO_PTHREAD make sure we keep the runtime alive until the
// proxied main calls exit (see exitOnMainThread() for where Pop is called).
{{{ runtimeKeepalivePush() }}}
#endif
#if MAIN_MODULE
// Main modules can't tell if they have main() at compile time, since it may
// arrive from a dynamic library.
if (!entryFunction) return;
#endif
#if MAIN_READS_PARAMS && STANDALONE_WASM
mainArgs = [thisProgram].concat(args)
#elif MAIN_READS_PARAMS
args.unshift(thisProgram);
var argc = args.length;
var argv = stackAlloc((argc + 1) * {{{ Runtime.POINTER_SIZE }}});
var argv_ptr = argv >> {{{ POINTER_SHIFT }}};
args.forEach((arg) => {
{{{ POINTER_HEAP }}}[argv_ptr++] = {{{ to64('allocateUTF8OnStack(arg)') }}};
});
{{{ POINTER_HEAP }}}[argv_ptr] = {{{ to64('0') }}};
#else
var argc = 0;
var argv = 0;
#endif // MAIN_READS_PARAMS
try {
#if BENCHMARK
var start = Date.now();
#endif
#if ABORT_ON_WASM_EXCEPTIONS
// See abortWrapperDepth in preamble.js!
abortWrapperDepth += 1;
#endif
#if STANDALONE_WASM
entryFunction();
// _start (in crt1.c) will call exit() if main return non-zero. So we know
// that if we get here main returned zero.
var ret = 0;
#else
var ret = entryFunction(argc, {{{ to64('argv') }}});
#endif // STANDALONE_WASM
#if BENCHMARK
Module.realPrint('main() took ' + (Date.now() - start) + ' milliseconds');
#endif
#if ASYNCIFY == 2
// The current spec of JSPI returns a promise only if the function suspends
// and a plain value otherwise. This will likely change:
// https://github.com/WebAssembly/js-promise-integration/issues/11
Promise.resolve(ret).then((result) => {
exitJS(result, /* implicit = */ true);
}).catch((e) => {
handleException(e);
});
#else
// if we're not running an evented main loop, it's time to exit
exitJS(ret, /* implicit = */ true);
#endif // ASYNCIFY == 2
return ret;
}
catch (e) {
return handleException(e);
}
#if ABORT_ON_WASM_EXCEPTIONS
finally {
// See abortWrapperDepth in preamble.js!
abortWrapperDepth -= 1;
}
#endif
}
#endif // HAS_MAIN
#if STACK_OVERFLOW_CHECK
function stackCheckInit() {
// This is normally called automatically during __wasm_call_ctors but need to
// get these values before even running any of the ctors so we call it redundantly
// here.
#if ASSERTIONS && USE_PTHREADS
// See $establishStackSpace for the equivelent code that runs on a thread
assert(!ENVIRONMENT_IS_PTHREAD);
#endif
#if RELOCATABLE
_emscripten_stack_set_limits({{{ STACK_HIGH }}} , {{{ STACK_LOW }}});
#else
_emscripten_stack_init();
#endif
// TODO(sbc): Move writeStackCookie to native to to avoid this.
writeStackCookie();
}
#endif
#if RELOCATABLE
var dylibsLoaded = false;
#if '$LDSO' in addedLibraryItems
LDSO.init();
#endif
#endif
#if MAIN_READS_PARAMS
function run(args = arguments_) {
#else
function run() {
#endif
if (runDependencies > 0) {
#if RUNTIME_LOGGING
err('run() called, but dependencies remain, so not running');
#endif
return;
}
#if STACK_OVERFLOW_CHECK
#if USE_PTHREADS
if (!ENVIRONMENT_IS_PTHREAD)
#endif
stackCheckInit();
#endif
#if RELOCATABLE
if (!dylibsLoaded) {
// Loading of dynamic libraries needs to happen on each thread, so we can't
// use the normal __ATPRERUN__ mechanism.
#if MAIN_MODULE
preloadDylibs();
#else
reportUndefinedSymbols();
#endif
dylibsLoaded = true;
// Loading dylibs can add run dependencies.
if (runDependencies > 0) {
#if RUNTIME_LOGGING
err('preloadDylibs added run() dependencies, not running yet');
#endif
return;
}
}
#endif
#if WASM_WORKERS
if (ENVIRONMENT_IS_WASM_WORKER) {
#if MODULARIZE
readyPromiseResolve(Module);
#endif // MODULARIZE
return initRuntime();
}
#endif
#if USE_PTHREADS
if (ENVIRONMENT_IS_PTHREAD) {
#if MODULARIZE
// The promise resolve function typically gets called as part of the execution
// of `doRun` below. The workers/pthreads don't execute `doRun` so the
// creation promise can be resolved, marking the pthread-Module as initialized.
readyPromiseResolve(Module);
#endif // MODULARIZE
initRuntime();
startWorker(Module);
return;
}
#endif
preRun();
// a preRun added a dependency, run will be called later
if (runDependencies > 0) {
#if RUNTIME_LOGGING
err('run() called, but dependencies remain, so not running');
#endif
return;
}
function doRun() {
// run may have just been called through dependencies being fulfilled just in this very frame,
// or while the async setStatus time below was happening
if (calledRun) return;
calledRun = true;
Module['calledRun'] = true;
if (ABORT) return;
initRuntime();
#if HAS_MAIN
preMain();
#endif
#if MODULARIZE
readyPromiseResolve(Module);
#endif
#if expectToReceiveOnModule('onRuntimeInitialized')
if (Module['onRuntimeInitialized']) Module['onRuntimeInitialized']();
#endif
#if HAS_MAIN
#if MAIN_READS_PARAMS
if (shouldRunNow) callMain(args);
#else
if (shouldRunNow) callMain();
#endif
#else
#if ASSERTIONS
assert(!Module['_main'], 'compiled without a main, but one is present. if you added it from JS, use Module["onRuntimeInitialized"]');
#endif // ASSERTIONS
#endif // HAS_MAIN
postRun();
}
#if expectToReceiveOnModule('setStatus')
if (Module['setStatus']) {
Module['setStatus']('Running...');
setTimeout(function() {
setTimeout(function() {
Module['setStatus']('');
}, 1);
doRun();
}, 1);
} else
#endif
{
doRun();
}
#if STACK_OVERFLOW_CHECK
checkStackCookie();
#endif
}
#if ASSERTIONS
#if EXIT_RUNTIME == 0
function checkUnflushedContent() {
// Compiler settings do not allow exiting the runtime, so flushing
// the streams is not possible. but in ASSERTIONS mode we check
// if there was something to flush, and if so tell the user they
// should request that the runtime be exitable.
// Normally we would not even include flush() at all, but in ASSERTIONS
// builds we do so just for this check, and here we see if there is any
// content to flush, that is, we check if there would have been
// something a non-ASSERTIONS build would have not seen.
// How we flush the streams depends on whether we are in SYSCALLS_REQUIRE_FILESYSTEM=0
// mode (which has its own special function for this; otherwise, all
// the code is inside libc)
var oldOut = out;
var oldErr = err;
var has = false;
out = err = (x) => {
has = true;
}
try { // it doesn't matter if it fails
#if SYSCALLS_REQUIRE_FILESYSTEM == 0 && '$flush_NO_FILESYSTEM' in addedLibraryItems
flush_NO_FILESYSTEM();
#elif hasExportedSymbol('fflush')
_fflush(0);
#endif
#if '$FS' in addedLibraryItems && '$TTY' in addedLibraryItems
// also flush in the JS FS layer
['stdout', 'stderr'].forEach(function(name) {
var info = FS.analyzePath('/dev/' + name);
if (!info) return;
var stream = info.object;
var rdev = stream.rdev;
var tty = TTY.ttys[rdev];
if (tty && tty.output && tty.output.length) {
has = true;
}
});
#endif
} catch(e) {}
out = oldOut;
err = oldErr;
if (has) {
warnOnce('stdio streams had content in them that was not flushed. you should set EXIT_RUNTIME to 1 (see the FAQ), or make sure to emit a newline when you printf etc.');
#if FILESYSTEM == 0 || SYSCALLS_REQUIRE_FILESYSTEM == 0
warnOnce('(this may also be due to not including full filesystem support - try building with -sFORCE_FILESYSTEM)');
#endif
}
}
#endif // EXIT_RUNTIME
#endif // ASSERTIONS
#if expectToReceiveOnModule('preInit')
if (Module['preInit']) {
if (typeof Module['preInit'] == 'function') Module['preInit'] = [Module['preInit']];
while (Module['preInit'].length > 0) {
Module['preInit'].pop()();
}
}
#endif
#if HAS_MAIN
// shouldRunNow refers to calling main(), not run().
#if INVOKE_RUN
var shouldRunNow = true;
#else
var shouldRunNow = false;
#endif
#if expectToReceiveOnModule('noInitialRun')
if (Module['noInitialRun']) shouldRunNow = false;
#endif
#endif // HAS_MAIN
run();
#if BUILD_AS_WORKER
var workerResponded = false, workerCallbackId = -1;
(function() {
var messageBuffer = null, buffer = 0, bufferSize = 0;
function flushMessages() {
if (!messageBuffer) return;
if (runtimeInitialized) {
var temp = messageBuffer;
messageBuffer = null;
temp.forEach(function(message) {
onmessage(message);
});
}
}
function messageResender() {
flushMessages();
if (messageBuffer) {
setTimeout(messageResender, 100); // still more to do
}
}
onmessage = (msg) => {
// if main has not yet been called (mem init file, other async things), buffer messages
if (!runtimeInitialized) {
if (!messageBuffer) {
messageBuffer = [];
setTimeout(messageResender, 100);
}
messageBuffer.push(msg);
return;
}
flushMessages();
var func = Module['_' + msg.data['funcName']];
if (!func) throw 'invalid worker function to call: ' + msg.data['funcName'];
var data = msg.data['data'];
if (data) {
if (!data.byteLength) data = new Uint8Array(data);
if (!buffer || bufferSize < data.length) {
if (buffer) _free(buffer);
bufferSize = data.length;
buffer = _malloc(data.length);
}
HEAPU8.set(data, buffer);
}
workerResponded = false;
workerCallbackId = msg.data['callbackId'];
if (data) {
func(buffer, data.length);
} else {
func(0, 0);
}
}
})();
#endif
#if STANDALONE_WASM && ASSERTIONS && !WASM_BIGINT
err('warning: running JS from STANDALONE_WASM without WASM_BIGINT will fail if a syscall with i64 is used (in standalone mode we cannot legalize syscalls)');
#endif