2 Java 4 Browser Bytecode Translator
3 Copyright (C) 2012-2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation, version 2 of the License.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. Look for COPYING file in the top folder.
16 If not, see http://opensource.org/licenses/GPL-2.0.
18 package org.apidesign.java4browser;
20 import java.io.IOException;
21 import java.io.InputStream;
22 import java.util.List;
23 import org.netbeans.modules.classfile.ByteCodes;
24 import org.netbeans.modules.classfile.ClassFile;
25 import org.netbeans.modules.classfile.Code;
26 import org.netbeans.modules.classfile.Method;
27 import org.netbeans.modules.classfile.Parameter;
29 /** Translator of the code inside class files to JavaScript.
31 * @author Jaroslav Tulach <jtulach@netbeans.org>
33 public final class ByteCodeToJavaScript {
34 private final ClassFile jc;
35 private final Appendable out;
37 private ByteCodeToJavaScript(ClassFile jc, Appendable out) {
42 /** Converts a given class file to a JavaScript version.
43 * @param fileName the name of the file we are reading
44 * @param classFile input stream with code of the .class file
45 * @param out a {@link StringBuilder} or similar to generate the output to
46 * @throws IOException if something goes wrong during read or write or translating
48 public static void compile(String fileName, InputStream classFile, Appendable out) throws IOException {
49 ClassFile jc = new ClassFile(classFile, true);
50 ByteCodeToJavaScript compiler = new ByteCodeToJavaScript(jc, out);
51 for (Method m : jc.getMethods()) {
53 compiler.generateStaticMethod(m);
57 private void generateStaticMethod(Method m) throws IOException {
58 out.append("function ").append(
59 jc.getName().getExternalName().replace('.', '_')
63 out.append(m.getReturnType());
64 List<Parameter> args = m.getParameters();
65 for (Parameter t : args) {
66 out.append(t.getDescriptor());
70 for (int i = 0; i < args.size(); i++) {
72 out.append("arg").append(String.valueOf(i));
75 out.append(") {").append("\n var ");
76 final Code code = m.getCode();
77 int len = code.getMaxLocals();
79 for (int i = 0; i < len; i++) {
81 out.append("var").append(String.valueOf(i));
84 out.append(";\n var stack = new Array(");
85 out.append(Integer.toString(code.getMaxStack()));
87 produceCode(code.getByteCodes());
88 out.append(";\nreturn 1;");
92 private void produceCode(byte[] byteCodes) throws IOException {
93 for (int i = 0; i < byteCodes.length; i++) {
96 final int c = (byteCodes[i] + 256) % 256;
98 case ByteCodes.bc_aload_0:
99 case ByteCodes.bc_iload_0:
100 out.append("stack.push(arg0);");
102 case ByteCodes.bc_aload_1:
103 case ByteCodes.bc_iload_1:
104 out.append("stack.push(arg1);");
106 case ByteCodes.bc_iadd:
107 out.append("stack.push(stack.pop() + stack.pop());");
109 case ByteCodes.bc_ireturn:
110 out.append("return stack.pop();");
113 for (int j = prev; j <= i; j++) {
115 final int cc = (byteCodes[j] + 256) % 256;
116 out.append(Integer.toString(cc));