vm/src/main/java/org/apidesign/vm4brwsr/Zips.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Thu, 07 Feb 2013 13:41:56 +0100
branchemul
changeset 702 fa42b3d8cbbc
parent 690 8929a6558ae4
child 706 a48961ff3e6b
permissions -rw-r--r--
Recognizing Main-Class attribute, so initializing the VM is just about passing in correct classpath
jaroslav@644
     1
/**
jaroslav@644
     2
 * Back 2 Browser Bytecode Translator
jaroslav@644
     3
 * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jaroslav@644
     4
 *
jaroslav@644
     5
 * This program is free software: you can redistribute it and/or modify
jaroslav@644
     6
 * it under the terms of the GNU General Public License as published by
jaroslav@644
     7
 * the Free Software Foundation, version 2 of the License.
jaroslav@644
     8
 *
jaroslav@644
     9
 * This program is distributed in the hope that it will be useful,
jaroslav@644
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
jaroslav@644
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
jaroslav@644
    12
 * GNU General Public License for more details.
jaroslav@644
    13
 *
jaroslav@644
    14
 * You should have received a copy of the GNU General Public License
jaroslav@644
    15
 * along with this program. Look for COPYING file in the top folder.
jaroslav@644
    16
 * If not, see http://opensource.org/licenses/GPL-2.0.
jaroslav@644
    17
 */
jaroslav@644
    18
package org.apidesign.vm4brwsr;
jaroslav@644
    19
jaroslav@672
    20
import java.io.ByteArrayInputStream;
jaroslav@644
    21
import java.io.IOException;
jaroslav@644
    22
import java.net.URL;
jaroslav@644
    23
import java.util.zip.ZipEntry;
jaroslav@644
    24
import java.util.zip.ZipInputStream;
jaroslav@644
    25
import org.apidesign.bck2brwsr.core.JavaScriptBody;
jaroslav@644
    26
jaroslav@644
    27
/** Conversion from classpath to load function.
jaroslav@644
    28
 *
jaroslav@644
    29
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jaroslav@644
    30
 */
jaroslav@644
    31
final class Zips {
jaroslav@644
    32
    private Zips() {
jaroslav@644
    33
    }
jaroslav@644
    34
    
jaroslav@644
    35
    public static void init() {
jaroslav@644
    36
    }
jaroslav@644
    37
    
jaroslav@702
    38
    public static byte[] loadFromCp(Object[] classpath, String res) throws Exception {
jaroslav@644
    39
        for (int i = 0; i < classpath.length; i++) {
jaroslav@644
    40
            Object c = classpath[i];
jaroslav@644
    41
            if (c instanceof String) {
jaroslav@644
    42
                try {
jaroslav@672
    43
                    String url = (String)c;
jaroslav@672
    44
                    final Zips z = toZip(url);
jaroslav@672
    45
                    c = classpath[i] = z;
jaroslav@672
    46
                    final byte[] man = z.findRes("META-INF/MANIFEST.MF");
jaroslav@672
    47
                    if (man != null) {
jaroslav@702
    48
                        String mainClass = processClassPathAttr(man, url, classpath);
jaroslav@702
    49
                        if (mainClass != null) {
jaroslav@702
    50
                            Class.forName(mainClass);
jaroslav@702
    51
                        }
jaroslav@672
    52
                    }
jaroslav@702
    53
                } catch (Exception ex) {
jaroslav@644
    54
                    classpath[i] = ex;
jaroslav@702
    55
                    throw ex;
jaroslav@644
    56
                }
jaroslav@644
    57
            }
jaroslav@702
    58
            if (res != null && c instanceof Zips) {
jaroslav@644
    59
                Object checkRes = ((Zips)c).findRes(res);
jaroslav@644
    60
                if (checkRes instanceof byte[]) {
jaroslav@644
    61
                    return (byte[])checkRes;
jaroslav@644
    62
                }
jaroslav@644
    63
            }
jaroslav@644
    64
        }
jaroslav@644
    65
        return null;
jaroslav@644
    66
    }
jaroslav@644
    67
jaroslav@644
    68
    @JavaScriptBody(args = { "res" }, body = "var r = this[res]; return r ? r : null;")
jaroslav@644
    69
    private native byte[] findRes(String res);
jaroslav@644
    70
jaroslav@644
    71
    @JavaScriptBody(args = { "res", "arr" }, body = "this[res] = arr;")
jaroslav@644
    72
    private native void putRes(String res, byte[] arr);
jaroslav@644
    73
    
jaroslav@644
    74
    private static Zips toZip(String path) throws IOException {
jaroslav@644
    75
        URL u = new URL(path);
jaroslav@644
    76
        ZipInputStream zip = new ZipInputStream(u.openStream());
jaroslav@644
    77
        Zips z = new Zips();
jaroslav@644
    78
        for (;;) {
jaroslav@644
    79
            ZipEntry entry = zip.getNextEntry();
jaroslav@644
    80
            if (entry == null) {
jaroslav@644
    81
                break;
jaroslav@644
    82
            }
jaroslav@644
    83
            byte[] arr = new byte[4096];
jaroslav@644
    84
            int offset = 0;
jaroslav@644
    85
            for (;;) {
jaroslav@644
    86
                int len = zip.read(arr, offset, arr.length - offset);
jaroslav@644
    87
                if (len == -1) {
jaroslav@644
    88
                    break;
jaroslav@644
    89
                }
jaroslav@644
    90
                offset += len;
jaroslav@644
    91
                if (offset == arr.length) {
jaroslav@644
    92
                    enlargeArray(arr, arr.length + 4096);
jaroslav@644
    93
                }
jaroslav@644
    94
            }
jaroslav@644
    95
            sliceArray(arr, offset);
jaroslav@644
    96
            z.putRes(entry.getName(), arr);
jaroslav@644
    97
        }
jaroslav@644
    98
        return z;
jaroslav@644
    99
    }
jaroslav@644
   100
jaroslav@702
   101
    private static String processClassPathAttr(final byte[] man, String url, Object[] classpath) throws IOException {
jaroslav@702
   102
        try (ParseMan is = new ParseMan(new ByteArrayInputStream(man))) {
jaroslav@672
   103
            String cp = is.toString();
jaroslav@702
   104
            if (cp != null) {
jaroslav@702
   105
                cp = cp.trim();
jaroslav@702
   106
                for (int p = 0; p < cp.length();) {
jaroslav@702
   107
                    int n = cp.indexOf(' ', p);
jaroslav@702
   108
                    if (n == -1) {
jaroslav@702
   109
                        n = cp.length();
jaroslav@702
   110
                    }
jaroslav@702
   111
                    String el = cp.substring(p, n);
jaroslav@702
   112
                    URL u = new URL(new URL(url), el);
jaroslav@702
   113
                    classpath = addToArray(classpath, u.toString());
jaroslav@702
   114
                    p = n + 1;
jaroslav@702
   115
                }
jaroslav@672
   116
            }
jaroslav@702
   117
            return is.getMainClass();
jaroslav@672
   118
        }
jaroslav@672
   119
    }
jaroslav@672
   120
jaroslav@672
   121
    private static Object[] addToArray(Object[] arr, String value) {
jaroslav@672
   122
        final int last = arr.length;
jaroslav@672
   123
        Object[] ret = enlargeArray(arr, last + 1);
jaroslav@672
   124
        ret[last] = value;
jaroslav@672
   125
        return ret;
jaroslav@672
   126
    }
jaroslav@672
   127
jaroslav@672
   128
    @JavaScriptBody(args = { "arr", "len" }, body = "while (arr.length < len) arr.push(null); return arr;throw('Arr: ' + arr);")
jaroslav@672
   129
    private static native Object[] enlargeArray(Object[] arr, int len);
jaroslav@644
   130
    @JavaScriptBody(args = { "arr", "len" }, body = "while (arr.length < len) arr.push(0);")
jaroslav@644
   131
    private static native void enlargeArray(byte[] arr, int len);
jaroslav@644
   132
jaroslav@644
   133
    @JavaScriptBody(args = { "arr", "len" }, body = "arr.splice(len, arr.length - len);")
jaroslav@644
   134
    private static native void sliceArray(byte[] arr, int len);
jaroslav@644
   135
    
jaroslav@644
   136
    
jaroslav@644
   137
}