288 lines
8.8 KiB
Java
Raw Normal View History

2017-04-02 10:04:23 +02:00
/*
2018-05-30 18:57:36 +02:00
* Copyright 2017 - 2018 Volker Berlin (i-net software)
2017-04-02 10:04:23 +02:00
*
* 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;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
2017-04-02 10:04:23 +02:00
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
2018-10-08 20:46:35 +02:00
import java.io.OutputStreamWriter;
2017-04-02 13:50:01 +02:00
import java.net.MalformedURLException;
import java.net.URL;
2018-10-08 20:46:35 +02:00
import java.nio.charset.StandardCharsets;
2017-04-02 10:04:23 +02:00
import java.util.ArrayList;
2018-10-07 18:57:41 +02:00
import java.util.HashMap;
2017-04-02 10:04:23 +02:00
import java.util.List;
import javax.annotation.Nonnull;
import de.inetsoftware.classparser.ClassFile;
import de.inetsoftware.jwebassembly.binary.BinaryModuleWriter;
import de.inetsoftware.jwebassembly.module.ModuleGenerator;
2017-04-02 10:04:23 +02:00
import de.inetsoftware.jwebassembly.module.ModuleWriter;
2019-03-30 22:23:36 +01:00
import de.inetsoftware.jwebassembly.module.WasmTarget;
2017-04-02 10:04:23 +02:00
import de.inetsoftware.jwebassembly.text.TextModuleWriter;
/**
* The main class of the compiler.
*
* @author Volker Berlin
*/
public class JWebAssembly {
2018-10-07 18:57:41 +02:00
private final List<URL> classFiles = new ArrayList<>();
private final HashMap<String, String> properties = new HashMap<>();
2017-04-02 10:04:23 +02:00
2018-11-21 19:44:05 +01:00
private final List<URL> libraries = new ArrayList<>();
2018-11-16 17:46:10 +01:00
/**
* Property for adding debug names to the output if true.
*/
public static final String DEBUG_NAMES = "DebugNames";
2018-11-16 17:46:10 +01:00
/**
* The name of the annotation for import functions.
*/
public static final String IMPORT_ANNOTATION = "de.inetsoftware.jwebassembly.api.annotation.Import";
/**
* The name of the annotation for export functions.
*/
public static final String EXPORT_ANNOTATION = "de.inetsoftware.jwebassembly.api.annotation.Export";
2018-11-16 20:14:43 +01:00
/**
* The name of the annotation for native WASM code in text format.
*/
public static final String TEXTCODE_ANNOTATION = "de.inetsoftware.jwebassembly.api.annotation.WasmTextCode";
/**
* The name of the annotation for replacing a single method of the Java runtime.
*/
public static final String REPLACE_ANNOTATION = "de.inetsoftware.jwebassembly.api.annotation.Replace";
2017-04-02 10:04:23 +02:00
/**
* Create a instance.
*/
public JWebAssembly() {
}
/**
* Add a classFile to compile
*
* @param classFile
* the file
*/
public void addFile( @Nonnull File classFile ) {
2017-04-02 13:50:01 +02:00
try {
classFiles.add( classFile.toURI().toURL() );
} catch( MalformedURLException ex ) {
throw new IllegalArgumentException( ex );
}
}
/**
* Add a classFile to compile
*
* @param classFile
* the file
*/
public void addFile( @Nonnull URL classFile ) {
2017-04-02 10:04:23 +02:00
classFiles.add( classFile );
}
2018-10-07 18:57:41 +02:00
/**
* Set property to control the behavior of the compiler
*
* @param key
* the key
* @param value
* the new value
*/
public void setProperty( String key, String value ) {
properties.put( key, value );
}
/**
* Get the value of a property.
*
* @param key
* the key
* @return the current value
*/
public String getProperty( String key ) {
return properties.get( key );
}
2017-04-02 10:04:23 +02:00
/**
2018-11-21 19:44:05 +01:00
* Add a jar or zip file as library to the compiler. Methods from the library will be add to the wasm only when used.
*
* @param library
* a archive file
*/
public void addLibrary( @Nonnull File library ) {
try {
2019-04-04 18:54:52 +02:00
addLibrary( library.toURI().toURL() );
2018-11-21 19:44:05 +01:00
} catch( MalformedURLException ex ) {
throw new IllegalArgumentException( ex );
}
}
/**
* Add a jar or zip file as library to the compiler. Methods from the library will be add to the wasm only when used.
*
* @param library
* a archive file
*/
public void addLibrary( @Nonnull URL library ) {
libraries.add( library );
}
/**
2017-04-02 10:04:23 +02:00
* Convert the added files to a WebAssembly module in text representation.
*
* @return the module as string
* @throws WasmException
* if any conversion error occurs
*/
public String compileToText() throws WasmException {
StringBuilder output = new StringBuilder();
try {
compileToText( output );
return output.toString();
} catch( Exception ex ) {
System.err.println( output );
throw ex;
}
2017-04-02 10:04:23 +02:00
}
2018-10-08 20:46:35 +02:00
/**
* Convert the added files to a WebAssembly module in text representation.
*
* @param file
* the target for the module data
* @throws WasmException
* if any conversion error occurs
*/
public void compileToText( File file ) throws WasmException {
try (OutputStreamWriter output = new OutputStreamWriter( new BufferedOutputStream( new FileOutputStream( file ) ), StandardCharsets.UTF_8 )) {
compileToText( output );
2018-10-08 20:46:35 +02:00
} catch( Exception ex ) {
throw WasmException.create( ex );
}
}
2017-04-02 10:04:23 +02:00
/**
* Convert the added files to a WebAssembly module in text representation.
*
* @param output
* the target for the module data
* @throws WasmException
* if any conversion error occurs
*/
public void compileToText( Appendable output ) throws WasmException {
try (TextModuleWriter writer = new TextModuleWriter( output, properties )) {
2017-04-02 10:04:23 +02:00
compile( writer );
} catch( Exception ex ) {
throw WasmException.create( ex );
}
}
/**
* Convert the added files to a WebAssembly module in binary representation.
*
* @return the module as string
* @throws WasmException
* if any conversion error occurs
*/
public byte[] compileToBinary() throws WasmException {
ByteArrayOutputStream output = new ByteArrayOutputStream();
compileToBinary( output );
return output.toByteArray();
}
/**
* Convert the added files to a WebAssembly module in binary representation.
*
* @param file
* the target for the module data
* @throws WasmException
* if any conversion error occurs
*/
public void compileToBinary( File file ) throws WasmException {
2019-03-30 22:23:36 +01:00
try (WasmTarget target = new WasmTarget( file ) ) {
compileToBinary( target );
2017-04-02 10:04:23 +02:00
} catch( Exception ex ) {
throw WasmException.create( ex );
}
}
/**
* Convert the added files to a WebAssembly module in binary representation.
*
* @param output
* the target for the module data
* @throws WasmException
* if any conversion error occurs
*/
public void compileToBinary( OutputStream output ) throws WasmException {
2019-03-30 22:23:36 +01:00
try (WasmTarget target = new WasmTarget( output )) {
compileToBinary( target );
} catch( Exception ex ) {
throw WasmException.create( ex );
}
}
/**
* Convert the added files to a WebAssembly module in binary representation.
*
* @param target
* the target for the module data
* @throws WasmException
* if any conversion error occurs
*/
private void compileToBinary( WasmTarget target ) throws WasmException {
try (BinaryModuleWriter writer = new BinaryModuleWriter( target, properties )) {
2017-04-02 10:04:23 +02:00
compile( writer );
} catch( Exception ex ) {
throw WasmException.create( ex );
}
}
/**
* Convert the added files to a WebAssembly module.
*
* @param writer
* the formatter
* @throws IOException
* if any I/O error occur
* @throws WasmException
* if any conversion error occurs
*/
private void compile( ModuleWriter writer ) throws IOException, WasmException {
2018-11-21 19:44:05 +01:00
ModuleGenerator generator = new ModuleGenerator( writer, libraries );
2017-04-17 12:10:56 +02:00
for( URL url : classFiles ) {
ClassFile classFile = new ClassFile( new BufferedInputStream( url.openStream() ) );
generator.prepare( classFile );
2017-04-17 12:10:56 +02:00
}
generator.prepareFinish();
generator.finish();
2017-04-02 10:04:23 +02:00
}
}