Use JavaScript imports generated from the imports declarations.

This commit is contained in:
Volker Berlin 2019-07-03 20:09:22 +02:00
parent 5133890a6b
commit d9e8b70d06
10 changed files with 110 additions and 48 deletions

View File

@ -203,14 +203,14 @@ public class JWebAssembly {
/** /**
* Convert the added files to a WebAssembly module in text representation. * Convert the added files to a WebAssembly module in text representation.
* *
* @param output * @param target
* the target for the module data * the target for the module data
* @throws WasmException * @throws WasmException
* if any conversion error occurs * if any conversion error occurs
*/ */
private void compileToText( WasmTarget target ) throws WasmException { private void compileToText( WasmTarget target ) throws WasmException {
try (TextModuleWriter writer = new TextModuleWriter( target, properties )) { try (TextModuleWriter writer = new TextModuleWriter( target, properties )) {
compile( writer ); compile( writer, target );
} catch( Exception ex ) { } catch( Exception ex ) {
throw WasmException.create( ex ); throw WasmException.create( ex );
} }
@ -271,7 +271,7 @@ public class JWebAssembly {
*/ */
private void compileToBinary( WasmTarget target ) throws WasmException { private void compileToBinary( WasmTarget target ) throws WasmException {
try (BinaryModuleWriter writer = new BinaryModuleWriter( target, properties )) { try (BinaryModuleWriter writer = new BinaryModuleWriter( target, properties )) {
compile( writer ); compile( writer, target );
} catch( Exception ex ) { } catch( Exception ex ) {
throw WasmException.create( ex ); throw WasmException.create( ex );
} }
@ -282,13 +282,15 @@ public class JWebAssembly {
* *
* @param writer * @param writer
* the formatter * the formatter
* @param target
* the target for the module data
* @throws IOException * @throws IOException
* if any I/O error occur * if any I/O error occur
* @throws WasmException * @throws WasmException
* if any conversion error occurs * if any conversion error occurs
*/ */
private void compile( ModuleWriter writer ) throws IOException, WasmException { private void compile( ModuleWriter writer, WasmTarget target ) throws IOException, WasmException {
ModuleGenerator generator = new ModuleGenerator( writer, libraries ); ModuleGenerator generator = new ModuleGenerator( writer, target, libraries );
for( URL url : classFiles ) { for( URL url : classFiles ) {
ClassFile classFile = new ClassFile( new BufferedInputStream( url.openStream() ) ); ClassFile classFile = new ClassFile( new BufferedInputStream( url.openStream() ) );
generator.prepare( classFile ); generator.prepare( classFile );

View File

@ -129,6 +129,6 @@ public class JavaScriptWriter {
} }
isFirst = false; isFirst = false;
} }
out.append( "\n}\n" ); out.append( "\n};\nif (typeof module !== \"undefined\") module.exports = wasmImports;" );
} }
} }

View File

@ -38,6 +38,7 @@ import de.inetsoftware.classparser.ConstantClass;
import de.inetsoftware.classparser.MethodInfo; import de.inetsoftware.classparser.MethodInfo;
import de.inetsoftware.jwebassembly.JWebAssembly; import de.inetsoftware.jwebassembly.JWebAssembly;
import de.inetsoftware.jwebassembly.WasmException; import de.inetsoftware.jwebassembly.WasmException;
import de.inetsoftware.jwebassembly.javascript.JavaScriptWriter;
import de.inetsoftware.jwebassembly.module.TypeManager.StructType; import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
import de.inetsoftware.jwebassembly.wasm.AnyType; import de.inetsoftware.jwebassembly.wasm.AnyType;
import de.inetsoftware.jwebassembly.wasm.NamedStorageType; import de.inetsoftware.jwebassembly.wasm.NamedStorageType;
@ -54,6 +55,8 @@ public class ModuleGenerator {
private final ModuleWriter writer; private final ModuleWriter writer;
private final JavaScriptWriter javaScript;
private final URLClassLoader libraries; private final URLClassLoader libraries;
private final JavaMethodWasmCodeBuilder javaCodeBuilder = new JavaMethodWasmCodeBuilder(); private final JavaMethodWasmCodeBuilder javaCodeBuilder = new JavaMethodWasmCodeBuilder();
@ -75,11 +78,14 @@ public class ModuleGenerator {
* *
* @param writer * @param writer
* the target writer * the target writer
* @param target
* the target for the module data
* @param libraries * @param libraries
* libraries * libraries
*/ */
public ModuleGenerator( @Nonnull ModuleWriter writer, @Nonnull List<URL> libraries ) { public ModuleGenerator( @Nonnull ModuleWriter writer, WasmTarget target, @Nonnull List<URL> libraries ) {
this.writer = writer; this.writer = writer;
this.javaScript = new JavaScriptWriter( target );
this.libraries = new URLClassLoader( libraries.toArray( new URL[libraries.size()] ) ); this.libraries = new URLClassLoader( libraries.toArray( new URL[libraries.size()] ) );
javaCodeBuilder.init( types ); javaCodeBuilder.init( types );
((WasmCodeBuilder)watParser).init( types ); ((WasmCodeBuilder)watParser).init( types );
@ -211,6 +217,7 @@ public class ModuleGenerator {
String importName = (String)importAnannotation.get( "name" ); String importName = (String)importAnannotation.get( "name" );
writer.prepareImport( name, impoarModule, importName ); writer.prepareImport( name, impoarModule, importName );
writeMethodSignature( name, true, null ); writeMethodSignature( name, true, null );
javaScript.addImport( impoarModule, importName, importAnannotation );
} }
// init/write the function types // init/write the function types
@ -295,6 +302,7 @@ public class ModuleGenerator {
throw new WasmException( "Missing function: " + next.signatureName, -1 ); throw new WasmException( "Missing function: " + next.signatureName, -1 );
} }
} }
javaScript.finish();
} }
/** /**

View File

@ -1,14 +1,7 @@
#!/usr/bin/env node load( "spiderMonkey.wasm.js" );
var wasm = read( "spiderMonkey.wasm", "binary" ); // https://developer.mozilla.org/en-US/docs/Mozilla/Projects/SpiderMonkey/Shell_global_objects var wasm = read( "spiderMonkey.wasm", "binary" ); // https://developer.mozilla.org/en-US/docs/Mozilla/Projects/SpiderMonkey/Shell_global_objects
var testData = JSON.parse( read( "testdata.json" ) ); var testData = JSON.parse( read( "testdata.json" ) );
var dependencies = {
"global": {},
"env": {}
};
dependencies["global.Math"] = Math;
function callExport(instance) { function callExport(instance) {
var result = {}; var result = {};
for (var method in testData) { for (var method in testData) {
@ -24,7 +17,7 @@ function callExport(instance) {
redirect( original ); redirect( original );
} }
WebAssembly.instantiate( wasm, dependencies ).then( WebAssembly.instantiate( wasm, wasmImports ).then(
obj => callExport(obj.instance), obj => callExport(obj.instance),
reason => console.log(reason) reason => console.log(reason)
); );

View File

@ -1,14 +1,7 @@
#!/usr/bin/env node load( "spiderMonkey.wasm.js" );
var wasm = wasmTextToBinary( read( "spiderMonkey.wat" ) ); var wasm = wasmTextToBinary( read( "spiderMonkey.wat" ) );
var testData = JSON.parse( read( "testdata.json" ) ); var testData = JSON.parse( read( "testdata.json" ) );
var dependencies = {
"global": {},
"env": {}
};
dependencies["global.Math"] = Math;
function callExport(instance) { function callExport(instance) {
var result = {}; var result = {};
for (var method in testData) { for (var method in testData) {
@ -24,7 +17,7 @@ function callExport(instance) {
redirect( original ); redirect( original );
} }
WebAssembly.instantiate( wasm, dependencies ).then( WebAssembly.instantiate( wasm, wasmImports ).then(
obj => callExport(obj.instance), obj => callExport(obj.instance),
reason => console.log(reason) reason => console.log(reason)
); );

View File

@ -3,6 +3,7 @@
var fs = require('fs'); var fs = require('fs');
var wabt = require("wabt")(); var wabt = require("wabt")();
const wasmImports = require( "./test.wasm.js" );
var filename = '{test.wat}'; var filename = '{test.wat}';
var text = fs.readFileSync(filename, "utf8"); var text = fs.readFileSync(filename, "utf8");
var testData = JSON.parse( fs.readFileSync( "testdata.json", "utf8" ) ); var testData = JSON.parse( fs.readFileSync( "testdata.json", "utf8" ) );
@ -11,12 +12,6 @@ var features = {'sat_float_to_int':true, 'sign_extension':true, 'exceptions':tru
var wasm = wabt.parseWat(filename, text, features); var wasm = wabt.parseWat(filename, text, features);
wasm = wasm.toBinary({}).buffer; wasm = wasm.toBinary({}).buffer;
var dependencies = {
"global": {},
"env": {}
};
dependencies["global.Math"] = Math;
function callExport(instance) { function callExport(instance) {
var result = {}; var result = {};
for (var method in testData) { for (var method in testData) {
@ -30,7 +25,7 @@ function callExport(instance) {
fs.writeFileSync( "testresult.json", JSON.stringify(result) ); fs.writeFileSync( "testresult.json", JSON.stringify(result) );
} }
WebAssembly.instantiate( wasm, dependencies ).then( WebAssembly.instantiate( wasm, wasmImports ).then(
obj => callExport(obj.instance), obj => callExport(obj.instance),
reason => console.log(reason) reason => console.log(reason)
); );

View File

@ -26,8 +26,8 @@ public class JavaScriptWriterTest {
"Foo:{\n" + "Foo:{\n" +
"bar:1 + 1\n" + "bar:1 + 1\n" +
"}\n" + "}\n" +
"}\n" + "};\n" +
"", builder.toString() ); "if (typeof module !== \"undefined\") module.exports = wasmImports;", builder.toString() );
} }
@Test @Test
@ -42,8 +42,8 @@ public class JavaScriptWriterTest {
"bar:1 + 1,\n" + "bar:1 + 1,\n" +
"xyz:3\n" + "xyz:3\n" +
"}\n" + "}\n" +
"}\n" + "};\n" +
"", builder.toString() ); "if (typeof module !== \"undefined\") module.exports = wasmImports;", builder.toString() );
} }
@Test @Test
@ -60,8 +60,8 @@ public class JavaScriptWriterTest {
"Foo:{\n" + "Foo:{\n" +
"foo:1 + 1\n" + "foo:1 + 1\n" +
"}\n" + "}\n" +
"}\n" + "};\n" +
"", builder.toString() ); "if (typeof module !== \"undefined\") module.exports = wasmImports;", builder.toString() );
} }
@Test @Test
@ -73,7 +73,7 @@ public class JavaScriptWriterTest {
writer.finish( builder ); writer.finish( builder );
assertEquals( "var wasmImports = {\n" + assertEquals( "var wasmImports = {\n" +
"Foo:Foo\n" + "Foo:Foo\n" +
"}\n" + "};\n" +
"", builder.toString() ); "if (typeof module !== \"undefined\") module.exports = wasmImports;", builder.toString() );
} }
} }

View File

@ -2,16 +2,11 @@
var fs = require('fs'); var fs = require('fs');
const wasmImports = require( "./test.wasm.js" );
var filename = '{test.wasm}'; var filename = '{test.wasm}';
var wasm = fs.readFileSync(filename); var wasm = fs.readFileSync(filename);
var testData = JSON.parse( fs.readFileSync( "testdata.json", "utf8" ) ); var testData = JSON.parse( fs.readFileSync( "testdata.json", "utf8" ) );
var dependencies = {
"global": {},
"env": {}
};
dependencies["global.Math"] = Math;
function callExport(instance) { function callExport(instance) {
var result = {}; var result = {};
for (var method in testData) { for (var method in testData) {
@ -25,7 +20,7 @@ function callExport(instance) {
fs.writeFileSync( "testresult.json", JSON.stringify(result) ); fs.writeFileSync( "testresult.json", JSON.stringify(result) );
} }
WebAssembly.instantiate( wasm, dependencies ).then( WebAssembly.instantiate( wasm, wasmImports ).then(
obj => callExport(obj.instance), obj => callExport(obj.instance),
reason => console.log(reason) reason => console.log(reason)
); );

View File

@ -73,7 +73,7 @@ public class CallFunctions extends AbstractBaseTest {
} }
@Import( module = "global.Math", name = "max" ) @Import( module = "Math", name = "max" )
static int abc( int a, int b) { static int abc( int a, int b) {
return Math.max( a, b ); return Math.max( a, b );
} }

View File

@ -0,0 +1,76 @@
/*
* Copyright 2019 Volker Berlin (i-net software)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package de.inetsoftware.jwebassembly.runtime;
import java.util.ArrayList;
import java.util.Collection;
import org.junit.ClassRule;
import org.junit.runners.Parameterized.Parameters;
import de.inetsoftware.jwebassembly.ScriptEngine;
import de.inetsoftware.jwebassembly.WasmRule;
import de.inetsoftware.jwebassembly.api.annotation.Export;
/**
* @author Volker Berlin
*/
public class MathAPI extends AbstractBaseTest {
@ClassRule
public static WasmRule rule = new WasmRule( TestClass.class );
public MathAPI( ScriptEngine script, String method, Object[] params ) {
super( rule, script, method, params );
}
@Parameters(name="{0}-{1}")
public static Collection<Object[]> data() {
ArrayList<Object[]> list = new ArrayList<>();
for( ScriptEngine[] val : ScriptEngine.testParams() ) {
ScriptEngine script = val[0];
addParam( list, script, "sin0" );
addParam( list, script, "sinPI" );
addParam( list, script, "cos0" );
addParam( list, script, "tan0" );
}
rule.setTestParameters( list );
return list;
}
static class TestClass {
@Export
static double sin0() {
return Math.sin( 0 );
}
@Export
static double sinPI() {
return Math.sin( Math.PI / 2 );
}
@Export
static double cos0() {
return Math.cos( 0 );
}
@Export
static double tan0() {
return Math.tan( 0 );
}
}
}