Skip to content

Commit fe1d022

Browse files
committed
[wasm2js] Remove all handling for external memory file
We have a lot of support code and complexity in emscripten for handling of external memory files. However, its only ever used in wasm2js mode which is a legacy mode. The only reason we continue to support it here IIUC is because its slightly more space efficient than embedding the data as base64. For small programs like hello_world this is an over codesize win. For larger programs there is a regression in overall size in proportion to the amount of static data in the program.
1 parent c0cda71 commit fe1d022

30 files changed

+130
-631
lines changed

ChangeLog.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,10 @@ See docs/process.md for more on how version tagging works.
2525
only exported on demand. This means that they must be added to
2626
`EXPORTED_RUNTIME_METHODS` for them to appear on the `Module` object. For
2727
now, this only effects users of `STRICT` mode. (#21439)
28+
- Emscripten no longer supports `--memory-init-file` (i.e. extracting static
29+
data into an external .mem file). This feature was only available under
30+
wasm2js (`-sWASM=0`) anyway so this change will only affect users of this
31+
setting. (#21217)
2832

2933
3.1.54 - 02/15/24
3034
-----------------
@@ -62,7 +66,6 @@ See docs/process.md for more on how version tagging works.
6266
- Allow comments in response files. Any line starting with `#` is now ignored.
6367
This is useful when listing exported symbols. (#21330)
6468

65-
6669
3.1.53 - 01/29/24
6770
-----------------
6871
- The llvm version that emscripten uses was updated to 19.0.0 trunk. (#21165)

docs/emcc.txt

-48
Original file line numberDiff line numberDiff line change
@@ -493,49 +493,6 @@ Options that are modified or new in *emcc* are listed below:
493493
Ports repos. After this operation is complete, this process will
494494
exit.
495495

496-
"--memory-init-file 0|1"
497-
[link] Specifies whether to emit a separate memory initialization
498-
file.
499-
500-
Note:
501-
502-
Note that this is only relevant when *not* emitting Wasm, as
503-
Wasm embeds the memory init data in the Wasm binary.
504-
505-
Possible values are:
506-
507-
* "0": Do not emit a separate memory initialization file.
508-
Instead keep the static initialization inside the generated
509-
JavaScript as text. This is the default setting if compiling
510-
with -O0 or -O1 link-time optimization flags.
511-
512-
* "1": Emit a separate memory initialization file in binary
513-
format. This is more efficient than storing it as text inside
514-
JavaScript, but does mean you have another file to publish.
515-
The binary file will also be loaded asynchronously, which
516-
means "main()" will not be called until the file is downloaded
517-
and applied; you cannot call any C functions until it arrives.
518-
This is the default setting when compiling with -O2 or higher.
519-
520-
Note:
521-
522-
The safest way to ensure that it is safe to call C functions
523-
(the initialisation file has loaded) is to call a notifier
524-
function from "main()".
525-
526-
Note:
527-
528-
If you assign a network request to
529-
"Module.memoryInitializerRequest" (before the script runs),
530-
then it will use that request instead of automatically
531-
starting a download for you. This is beneficial in that you
532-
can, in your HTML, fire off a request for the memory init
533-
file before the script actually arrives. For this to work,
534-
the network request should be an XMLHttpRequest with
535-
responseType set to "'arraybuffer'". (You can also put any
536-
other object here, all it must provide is a ".response"
537-
property containing an ArrayBuffer.)
538-
539496
"-Wwarn-absolute-paths"
540497
[compile+link] Enables warnings about the use of absolute paths in
541498
"-I" and "-L" command line directives. This is used to warn against
@@ -604,11 +561,6 @@ Options that are modified or new in *emcc* are listed below:
604561
These rules only apply when linking. When compiling to object code
605562
(See *-c* below) the name of the output file is irrelevant.
606563

607-
Note:
608-
609-
If "--memory-init-file" is used, a **.mem** file will be created
610-
in addition to the generated **.js** and/or **.html** file.
611-
612564
"-c"
613565
[compile] Tells *emcc* to emit an object file which can then be
614566
linked with other object files to produce an executable.

emcc.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -81,7 +81,7 @@
8181
'--bind', '--closure', '--cpuprofiler', '--embed-file',
8282
'--emit-symbol-map', '--emrun', '--exclude-file', '--extern-post-js',
8383
'--extern-pre-js', '--ignore-dynamic-linking', '--js-library',
84-
'--js-transform', '--memory-init-file', '--oformat', '--output_eol',
84+
'--js-transform', '--oformat', '--output_eol',
8585
'--post-js', '--pre-js', '--preload-file', '--profiling-funcs',
8686
'--proxy-to-worker', '--shell-file', '--source-map-base',
8787
'--threadprofiler', '--use-preload-plugins'
@@ -145,7 +145,6 @@ def __init__(self):
145145
self.emrun = False
146146
self.cpu_profiler = False
147147
self.memory_profiler = False
148-
self.memory_init_file = None
149148
self.use_preload_cache = False
150149
self.use_preload_plugins = False
151150
self.valid_abspaths = []
@@ -1316,7 +1315,7 @@ def consume_arg_file():
13161315
ports.show_ports()
13171316
should_exit = True
13181317
elif check_arg('--memory-init-file'):
1319-
options.memory_init_file = int(consume_arg())
1318+
exit_with_error('--memory-init-file is no longer supported')
13201319
elif check_flag('--proxy-to-worker'):
13211320
settings_changes.append('PROXY_TO_WORKER=1')
13221321
elif check_arg('--valid-abspath'):

site/source/docs/compiling/Building-Projects.rst

-1
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,6 @@ which files are produced under which conditions:
9292
- ``emcc ... -o output.wasm`` omits generating either JavaScript or HTML launcher file, and produces a single Wasm file built in standalone mode as if the ``-sSTANDALONE_WASM`` setting had been used. The resulting file expects to be run with the `WASI ABI <https://github.com/WebAssembly/WASI/blob/4712d490fd7662f689af6faa5d718e042f014931/legacy/application-abi.md>`_ - in particular, as soon as you initialize the module you must manually invoke either the ``_start`` export or (in the case of ``--no-entry``) the ``_initialize`` export before doing anything else with it.
9393
- ``emcc ... -o output.{html,js} -sWASM=0`` causes the compiler to target JavaScript, and therefore a ``.wasm`` file is not produced.
9494
- ``emcc ... -o output.{html,js} --emit-symbol-map`` produces a file ``output.{html,js}.symbols`` if WebAssembly is being targeted (``-sWASM=0`` not specified), or if JavaScript is being targeted and ``-Os``, ``-Oz`` or ``-O2`` or higher is specified, but debug level setting is ``-g1`` or lower (i.e. if symbols minification did occur).
95-
- ``emcc ... -o output.{html,js} -sWASM=0 --memory-init-file 1`` causes the generation of ``output.{html,js}.mem`` memory initializer file. Passing ``-O2``, ``-Os`` or ``-Oz`` also implies ``--memory-init-file 1``.
9695
- ``emcc ... -o output.{html,js} -gsource-map`` generates a source map file ``output.wasm.map``. If targeting JavaScript with ``-sWASM=0``, the filename is ``output.{html,js}.map``.
9796
- ``emcc ... -o output.{html,js} --preload-file xxx`` directive generates a preloaded MEMFS filesystem file ``output.data``.
9897
- ``emcc ... -o output.{html,js} -sWASM={0,1} -sSINGLE_FILE`` merges JavaScript and WebAssembly code in the single output file ``output.{html,js}`` (in base64) to produce only one file for deployment. (If paired with ``--preload-file``, the preloaded ``.data`` file still exists as a separate file)

site/source/docs/getting_started/FAQ.rst

+5-6
Original file line numberDiff line numberDiff line change
@@ -308,12 +308,11 @@ function `x` called before runtime initialization``, which is a check enabled in
308308
``ASSERTIONS`` builds.)
309309

310310
Calling a compiled function before a page has fully loaded can result in an
311-
error, if the function relies on files that may not be present (for example the
312-
:ref:`.mem <emcc-memory-init-file>` file and :ref:`preloaded
313-
<emcc-preload-file>` files are loaded asynchronously, and therefore if you just
314-
place some JS that calls compiled code in a ``--post-js``, that code will be
315-
called synchronously at the end of the combined JS file, potentially before the
316-
asynchronous event happens, which is bad).
311+
error, if the function relies on files that may not be present (for example
312+
:ref:`preloaded <emcc-preload-file>` files are loaded asynchronously, and
313+
therefore if you just place some JS that calls compiled code in a ``--post-js``,
314+
that code will be called synchronously at the end of the combined JS file,
315+
potentially before the asynchronous event happens, which is bad).
317316

318317
The easiest way to find out when loading is complete is to add a ``main()``
319318
function, and within it call a JavaScript function to notify your code that

site/source/docs/tools_reference/emcc.rst

-20
Original file line numberDiff line numberDiff line change
@@ -487,24 +487,6 @@ Options that are modified or new in *emcc* are listed below:
487487
[general]
488488
Shows the list of available projects in the Emscripten Ports repos. After this operation is complete, this process will exit.
489489

490-
.. _emcc-memory-init-file:
491-
492-
``--memory-init-file 0|1``
493-
[link]
494-
Specifies whether to emit a separate memory initialization file.
495-
496-
.. note:: Note that this is only relevant when *not* emitting Wasm, as Wasm embeds the memory init data in the Wasm binary.
497-
498-
Possible values are:
499-
500-
- ``0``: Do not emit a separate memory initialization file. Instead keep the static initialization inside the generated JavaScript as text. This is the default setting if compiling with -O0 or -O1 link-time optimization flags.
501-
- ``1``: Emit a separate memory initialization file in binary format. This is more efficient than storing it as text inside JavaScript, but does mean you have another file to publish. The binary file will also be loaded asynchronously, which means ``main()`` will not be called until the file is downloaded and applied; you cannot call any C functions until it arrives. This is the default setting when compiling with -O2 or higher.
502-
503-
.. note:: The :ref:`safest way <faq-when-safe-to-call-compiled-functions>` to ensure that it is safe to call C functions (the initialisation file has loaded) is to call a notifier function from ``main()``.
504-
505-
.. note:: If you assign a network request to ``Module.memoryInitializerRequest`` (before the script runs), then it will use that request instead of automatically starting a download for you. This is beneficial in that you can, in your HTML, fire off a request for the memory init file before the script actually arrives. For this to work, the network request should be an XMLHttpRequest with responseType set to ``'arraybuffer'``. (You can also put any other object here, all it must provide is a ``.response`` property containing an ArrayBuffer.)
506-
507-
508490
``-Wwarn-absolute-paths``
509491
[compile+link]
510492
Enables warnings about the use of absolute paths in ``-I`` and ``-L`` command line directives. This is used to warn against unintentional use of absolute paths, which is sometimes dangerous when referring to nonportable local system headers.
@@ -562,8 +544,6 @@ Options that are modified or new in *emcc* are listed below:
562544
These rules only apply when linking. When compiling to object code (See `-c`
563545
below) the name of the output file is irrelevant.
564546

565-
.. note:: If ``--memory-init-file`` is used, a **.mem** file will be created in addition to the generated **.js** and/or **.html** file.
566-
567547
.. _emcc-c:
568548

569549
``-c``

src/postamble_minimal.js

-7
Original file line numberDiff line numberDiff line change
@@ -221,13 +221,6 @@ WebAssembly.instantiate(Module['wasm'], imports).then((output) => {
221221
updateMemoryViews();
222222
#endif
223223

224-
#if !MEM_INIT_IN_WASM && !SINGLE_FILE
225-
#if ASSERTIONS
226-
if (!Module['mem']) throw 'Must load memory initializer as an ArrayBuffer in to variable Module.mem before adding compiled output .js script to the DOM';
227-
#endif
228-
HEAPU8.set(new Uint8Array(Module['mem']), {{{ GLOBAL_BASE }}});
229-
#endif
230-
231224
initRuntime(wasmExports);
232225
#if PTHREADS
233226
// Export Wasm module for pthread creation to access.

src/preamble.js

-85
Original file line numberDiff line numberDiff line change
@@ -973,9 +973,6 @@ function createWasm() {
973973
#endif
974974
updateMemoryViews();
975975
#endif
976-
#if !MEM_INIT_IN_WASM
977-
runMemoryInitializer();
978-
#endif
979976

980977
#if '$wasmTable' in addedLibraryItems && !RELOCATABLE
981978
wasmTable = wasmExports['__indirect_function_table'];
@@ -1123,88 +1120,6 @@ function getCompilerSetting(name) {
11231120
}
11241121
#endif // RETAIN_COMPILER_SETTINGS
11251122

1126-
#if !MEM_INIT_IN_WASM
1127-
var memoryInitializer = <<< MEM_INITIALIZER >>>;
1128-
1129-
function runMemoryInitializer() {
1130-
#if PTHREADS
1131-
if (ENVIRONMENT_IS_PTHREAD) return;
1132-
#endif
1133-
if (!isDataURI(memoryInitializer)) {
1134-
memoryInitializer = locateFile(memoryInitializer);
1135-
}
1136-
if (ENVIRONMENT_IS_NODE || ENVIRONMENT_IS_SHELL) {
1137-
var data = readBinary(memoryInitializer);
1138-
HEAPU8.set(data, {{{ GLOBAL_BASE }}});
1139-
} else {
1140-
addRunDependency('memory initializer');
1141-
var applyMemoryInitializer = (data) => {
1142-
if (data.byteLength) data = new Uint8Array(data);
1143-
#if ASSERTIONS
1144-
for (var i = 0; i < data.length; i++) {
1145-
assert(HEAPU8[{{{ GLOBAL_BASE }}} + i] === 0, "area for memory initializer should not have been touched before it's loaded");
1146-
}
1147-
#endif
1148-
HEAPU8.set(data, {{{ GLOBAL_BASE }}});
1149-
// Delete the typed array that contains the large blob of the memory initializer request response so that
1150-
// we won't keep unnecessary memory lying around. However, keep the XHR object itself alive so that e.g.
1151-
// its .status field can still be accessed later.
1152-
if (Module['memoryInitializerRequest']) delete Module['memoryInitializerRequest'].response;
1153-
removeRunDependency('memory initializer');
1154-
};
1155-
var doBrowserLoad = () => {
1156-
readAsync(memoryInitializer, applyMemoryInitializer, () => {
1157-
var e = new Error('could not load memory initializer ' + memoryInitializer);
1158-
#if MODULARIZE
1159-
readyPromiseReject(e);
1160-
#else
1161-
throw e;
1162-
#endif
1163-
});
1164-
};
1165-
#if SUPPORT_BASE64_EMBEDDING
1166-
var memoryInitializerBytes = tryParseAsDataURI(memoryInitializer);
1167-
if (memoryInitializerBytes) {
1168-
applyMemoryInitializer(memoryInitializerBytes.buffer);
1169-
} else
1170-
#endif
1171-
if (Module['memoryInitializerRequest']) {
1172-
// a network request has already been created, just use that
1173-
var useRequest = () => {
1174-
var request = Module['memoryInitializerRequest'];
1175-
var response = request.response;
1176-
if (request.status !== 200 && request.status !== 0) {
1177-
#if SUPPORT_BASE64_EMBEDDING
1178-
var data = tryParseAsDataURI(Module['memoryInitializerRequestURL']);
1179-
if (data) {
1180-
response = data.buffer;
1181-
} else {
1182-
#endif
1183-
// If you see this warning, the issue may be that you are using locateFile and defining it in JS. That
1184-
// means that the HTML file doesn't know about it, and when it tries to create the mem init request early, does it to the wrong place.
1185-
// Look in your browser's devtools network console to see what's going on.
1186-
console.warn('a problem seems to have happened with Module.memoryInitializerRequest, status: ' + request.status + ', retrying ' + memoryInitializer);
1187-
doBrowserLoad();
1188-
return;
1189-
#if SUPPORT_BASE64_EMBEDDING
1190-
}
1191-
#endif
1192-
}
1193-
applyMemoryInitializer(response);
1194-
};
1195-
if (Module['memoryInitializerRequest'].response) {
1196-
setTimeout(useRequest, 0); // it's already here; but, apply it asynchronously
1197-
} else {
1198-
Module['memoryInitializerRequest'].addEventListener('load', useRequest); // wait for it
1199-
}
1200-
} else {
1201-
// fetch it from the network ourselves
1202-
doBrowserLoad();
1203-
}
1204-
}
1205-
}
1206-
#endif // MEM_INIT_IN_WASM == 0
1207-
12081123
#if MAIN_MODULE && ASYNCIFY
12091124
// With MAIN_MODULE + ASYNCIFY the normal method of placing stub functions in
12101125
// wasmImports for as-yet-undefined symbols doesn't work since ASYNCIFY then

src/settings.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -959,7 +959,7 @@ var INCOMING_MODULE_JS_API = [
959959
'onCustomMessage', 'onExit', 'onFree', 'onFullScreen', 'onMalloc',
960960
'onRealloc', 'onRuntimeInitialized', 'postMainLoop', 'postRun', 'preInit',
961961
'preMainLoop', 'preRun',
962-
'preinitializedWebGLContext', 'memoryInitializerRequest', 'preloadPlugins',
962+
'preinitializedWebGLContext', 'preloadPlugins',
963963
'print', 'printErr', 'quit', 'setStatus', 'statusMessage', 'stderr',
964964
'stdin', 'stdout', 'thisProgram', 'wasm', 'wasmBinary', 'websocket'
965965
];

src/settings_internal.js

-5
Original file line numberDiff line numberDiff line change
@@ -143,11 +143,6 @@ var AUDIO_WORKLET_FILE = '';
143143
// Base URL the source mapfile, if relevant
144144
var SOURCE_MAP_BASE = '';
145145

146-
// When this is false we use an external memory init file
147-
// See --memory-init-file. When not using wasm2js this flag is ignored, and
148-
// this setting will always be true.
149-
var MEM_INIT_IN_WASM = true;
150-
151146
// If set to 1, src/base64Utils.js will be included in the bundle.
152147
// This is set internally when needed (SINGLE_FILE)
153148
var SUPPORT_BASE64_EMBEDDING = false;

src/shell.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -439,7 +439,7 @@ if (ENVIRONMENT_IS_NODE) {
439439
// Merge back in the overrides
440440
Object.assign(Module, moduleOverrides);
441441
// Free the object hierarchy contained in the overrides, this lets the GC
442-
// reclaim data used e.g. in memoryInitializerRequest, which is a large typed array.
442+
// reclaim data used.
443443
moduleOverrides = null;
444444
#if ASSERTIONS
445445
checkIncomingModuleAPI();

src/shell_minimal.js

+2-8
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,7 @@ if (ENVIRONMENT_IS_NODE && ENVIRONMENT_IS_SHELL) {
8282
#endif
8383

8484
#if !SINGLE_FILE
85-
#if ENVIRONMENT_MAY_BE_NODE && ((WASM == 1 && (!WASM2JS || !MEM_INIT_IN_WASM)) || WASM == 2)
85+
#if ENVIRONMENT_MAY_BE_NODE && ((WASM == 1 && !WASM2JS) || WASM == 2)
8686
// Wasm or Wasm2JS loading:
8787

8888
if (ENVIRONMENT_IS_NODE) {
@@ -95,13 +95,10 @@ if (ENVIRONMENT_IS_NODE) {
9595
Module['wasm'] = fs.readFileSync(__dirname + '/{{{ TARGET_BASENAME }}}.wasm');
9696
#endif
9797
#endif
98-
#if !MEM_INIT_IN_WASM
99-
Module['mem'] = fs.readFileSync(__dirname + '/{{{ TARGET_BASENAME }}}.mem');
100-
#endif
10198
}
10299
#endif
103100

104-
#if ENVIRONMENT_MAY_BE_SHELL && ((WASM == 1 && (!WASM2JS || !MEM_INIT_IN_WASM)) || WASM == 2)
101+
#if ENVIRONMENT_MAY_BE_SHELL && ((WASM == 1 && !WASM2JS) || WASM == 2)
105102
if (ENVIRONMENT_IS_SHELL) {
106103
#if WASM == 2
107104
if (typeof WebAssembly != 'undefined') Module['wasm'] = read('{{{ TARGET_BASENAME }}}.wasm', 'binary');
@@ -111,9 +108,6 @@ if (ENVIRONMENT_IS_SHELL) {
111108
Module['wasm'] = read('{{{ TARGET_BASENAME }}}.wasm', 'binary');
112109
#endif
113110
#endif
114-
#if !MEM_INIT_IN_WASM
115-
Module['mem'] = read('{{{ TARGET_BASENAME }}}.mem', 'binary');
116-
#endif
117111
}
118112
#endif
119113

+6-8
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,8 @@
11
{
2-
"a.html": 567,
3-
"a.html.gz": 379,
4-
"a.js": 17800,
5-
"a.js.gz": 7981,
6-
"a.mem": 3123,
7-
"a.mem.gz": 2693,
8-
"total": 21490,
9-
"total_gz": 11053
2+
"a.html": 354,
3+
"a.html.gz": 266,
4+
"a.js": 22323,
5+
"a.js.gz": 11632,
6+
"total": 22677,
7+
"total_gz": 11898
108
}
+6-8
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,8 @@
11
{
2-
"a.html": 567,
3-
"a.html.gz": 379,
4-
"a.js": 17272,
5-
"a.js.gz": 7814,
6-
"a.mem": 3123,
7-
"a.mem.gz": 2693,
8-
"total": 20962,
9-
"total_gz": 10886
2+
"a.html": 354,
3+
"a.html.gz": 266,
4+
"a.js": 21794,
5+
"a.js.gz": 11450,
6+
"total": 22148,
7+
"total_gz": 11716
108
}

0 commit comments

Comments
 (0)