Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[embind] Export embind exports as ESM exports for MODULARIZE=instance. #23404

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 11 additions & 1 deletion src/lib/libembind.js
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,13 @@ var LibraryEmbind = {
$PureVirtualError: undefined,
$GenericWireTypeSize: {{{ 2 * POINTER_SIZE }}},
#if EMBIND_AOT
$InvokerFunctions: '<<< EMBIND_AOT_OUTPUT >>>',
$InvokerFunctions: '<<< EMBIND_AOT_INVOKERS >>>',
#if MODULARIZE == 'instance'
// embindUpdateExports is called in the generated code after embind is
// initialized. It will link the bindings to the ES module exports.
$embindUpdateExports: '<<< EMBIND_AOT_UPDATE_EXPORTS >>>',
$embindUpdateExports__postset: 'addOnInit(embindUpdateExports);',
#endif
#endif
// If register_type is used, emval will be registered multiple times for
// different type id's, but only a single type object is needed on the JS side
Expand Down Expand Up @@ -2362,4 +2368,8 @@ var LibraryEmbind = {
},
};

#if MODULARIZE == 'instance'
extraLibraryFuncs.push('$embindUpdateExports');
#endif

addToLibrary(LibraryEmbind);
21 changes: 19 additions & 2 deletions src/lib/libembind_gen.js
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ var LibraryEmbind = {
},
$FunctionDefinition__deps: ['$createJsInvoker', '$createJsInvokerSignature', '$emittedFunctions'],
$FunctionDefinition: class {
hasPublicSymbol = true;
constructor(name, returnType, argumentTypes, functionIndex, thisType = null, isNonnullReturn = false, isAsync = false) {
this.name = name;
this.returnType = returnType;
Expand Down Expand Up @@ -157,6 +158,7 @@ var LibraryEmbind = {
}
},
$ClassDefinition: class {
hasPublicSymbol = true;
constructor(typeId, name, base = null) {
this.typeId = typeId;
this.name = name;
Expand Down Expand Up @@ -268,6 +270,7 @@ var LibraryEmbind = {
}
},
$ConstantDefinition: class {
hasPublicSymbol = true;
constructor(type, name) {
this.type = type;
this.name = name;
Expand All @@ -278,6 +281,7 @@ var LibraryEmbind = {
}
},
$EnumDefinition: class {
hasPublicSymbol = true;
constructor(typeId, name) {
this.typeId = typeId;
this.name = name;
Expand Down Expand Up @@ -454,14 +458,27 @@ var LibraryEmbind = {

print() {
const out = ['{\n'];
const publicSymbols = [];
for (const def of this.definitions) {
if (def.hasPublicSymbol) {
publicSymbols.push(def.name);
}
if (!def.printJs) {
continue;
}
def.printJs(out);
}
out.push('}')
console.log(out.join(''));
out.push('}\n');
let updateExports = '() => {\n';
for (const publicSymbol of publicSymbols) {
updateExports += `__exp_${publicSymbol} = Module['${publicSymbol}'];\n`
}
updateExports += '}\n';
console.log(JSON.stringify({
'invokers': out.join(''),
publicSymbols,
updateExports
}));
}
},

Expand Down
3 changes: 3 additions & 0 deletions src/settings_internal.js
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,9 @@ var USE_ASAN = false;
// Whether embind has been enabled.
var EMBIND = false;

// Symbols that embind exports. Only available when using EMBIND_AOT.
var EMBIND_EXPORTS = [];

// Whether a TypeScript definition file has been requested.
var EMIT_TSD = false;

Expand Down
26 changes: 26 additions & 0 deletions test/modularize_instance_embind.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
#include <stdio.h>
#include <emscripten.h>
#include <emscripten/bind.h>

using namespace emscripten;

void foo() {
printf("foo\n");
}

struct Bar {
void print() {
printf("bar\n");
}
};

int main() {
printf("main\n");
}

EMSCRIPTEN_BINDINGS(xxx) {
function("foo", &foo);
class_<Bar>("Bar")
.constructor<>()
.function("print", &Bar::print);
}
20 changes: 19 additions & 1 deletion test/test_other.py
Original file line number Diff line number Diff line change
Expand Up @@ -477,6 +477,24 @@ def test_modularize_instance(self, args):

self.assertContained('main1\nmain2\nfoo\nbar\nbaz\n', self.run_js('runner.mjs'))

def test_modularize_instance_embind(self):
self.run_process([EMCC, test_file('modularize_instance_embind.cpp'),
'-sMODULARIZE=instance',
'-lembind',
'-sEMBIND_AOT',
'-o', 'modularize_instance_embind.mjs'])

create_file('runner.mjs', '''
import init, { foo, Bar } from "./modularize_instance_embind.mjs";
await init();
foo();
const bar = new Bar();
bar.print();
bar.delete();
''')

self.assertContained('main\nfoo\nbar\n', self.run_js('runner.mjs'))

def test_emcc_out_file(self):
# Verify that "-ofile" works in addition to "-o" "file"
self.run_process([EMCC, '-c', '-ofoo.o', test_file('hello_world.c')])
Expand Down Expand Up @@ -3402,7 +3420,7 @@ def test_embind_tsgen_end_to_end(self, opts, tsc_opts):
# Check that TypeScript generation works and that the program is runs as
# expected.
self.emcc(test_file('other/embind_tsgen.cpp'),
['-o', 'embind_tsgen.js', '-lembind', '--emit-tsd', 'embind_tsgen.d.ts'] + opts)
['-o', 'embind_tsgen.js', '-sEMBIND_AOT', '-lembind', '--emit-tsd', 'embind_tsgen.d.ts'] + opts)

# Test that the output compiles with a TS file that uses the defintions.
shutil.copyfile(test_file('other/embind_tsgen_main.ts'), 'main.ts')
Expand Down
10 changes: 7 additions & 3 deletions tools/link.py
Original file line number Diff line number Diff line change
Expand Up @@ -2047,9 +2047,13 @@ def phase_emit_tsd(options, wasm_target, js_target, js_syms, metadata, linker_in
def phase_embind_aot(wasm_target, js_syms, linker_inputs):
out = run_embind_gen(wasm_target, js_syms, {}, linker_inputs)
if DEBUG:
write_file(in_temp('embind_aot.js'), out)
write_file(in_temp('embind_aot.json'), out)
out = json.loads(out)
src = read_file(final_js)
src = do_replace(src, '<<< EMBIND_AOT_OUTPUT >>>', out)
src = do_replace(src, '<<< EMBIND_AOT_INVOKERS >>>', out['invokers'])
if settings.MODULARIZE == 'instance':
settings.EMBIND_EXPORTS = out['publicSymbols']
src = do_replace(src, '<<< EMBIND_AOT_UPDATE_EXPORTS >>>', out['updateExports'])
write_file(final_js, src)


Expand Down Expand Up @@ -2488,7 +2492,7 @@ def modularize():
# Export using a UMD style export, or ES6 exports if selected
if settings.EXPORT_ES6:
if settings.MODULARIZE == 'instance':
exports = settings.EXPORTED_FUNCTIONS + settings.EXPORTED_RUNTIME_METHODS
exports = settings.EXPORTED_FUNCTIONS + settings.EXPORTED_RUNTIME_METHODS + settings.EMBIND_EXPORTS
# Declare a top level var for each export so that code in the init function
# can assign to it and update the live module bindings.
src += 'var ' + ', '.join(['__exp_' + export for export in exports]) + ';\n'
Expand Down
Loading