rt/vm/src/main/java/org/apidesign/vm4brwsr/ClassPath.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 14 May 2014 15:57:52 +0200
branchclosure
changeset 1575 08c8821c962d
parent 1554 7ba27baf5f3f
child 1576 3e9bfed0d88e
permissions -rw-r--r--
First step towards calling directly into ClassPath and using VMLazy only for JIT compilation
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@1551
    22
import java.io.InputStream;
jaroslav@644
    23
import java.net.URL;
jaroslav@1575
    24
import org.apidesign.bck2brwsr.core.Exported;
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@1550
    31
final class ClassPath {
jaroslav@1550
    32
    private ClassPath() {
jaroslav@644
    33
    }
jaroslav@644
    34
    
jaroslav@644
    35
    public static void init() {
jaroslav@644
    36
    }
jaroslav@729
    37
    @JavaScriptBody(args = { "arr" }, body = "return arr.length;")
jaroslav@729
    38
    private static native int length(Object arr);
jaroslav@729
    39
    @JavaScriptBody(args = { "arr", "index" }, body = "return arr[index];")
jaroslav@729
    40
    private static native Object at(Object arr, int index);
jaroslav@729
    41
    @JavaScriptBody(args = { "arr", "index", "value" }, body = "arr[index] = value; return value;")
jaroslav@729
    42
    private static native Object set(Object arr, int index, Object value);
jaroslav@644
    43
    
jaroslav@1554
    44
    private static boolean doingToZip;
jaroslav@1575
    45
    
jaroslav@1575
    46
    
jaroslav@1575
    47
    @Exported static byte[] loadBytes(Object loader, String name, Object[] arguments, int skip) throws Exception {
jaroslav@1575
    48
        return ClassPath.loadFromCp(arguments, name, skip);
jaroslav@1575
    49
    }
jaroslav@1575
    50
    static byte[] loadFromCp(Object classpath, String res, int skip) 
jaroslav@729
    51
    throws IOException, ClassNotFoundException {
jaroslav@729
    52
        for (int i = 0; i < length(classpath); i++) {
jaroslav@729
    53
            Object c = at(classpath, i);
jaroslav@1554
    54
            if (c instanceof String && !doingToZip) {
jaroslav@644
    55
                try {
jaroslav@1554
    56
                    doingToZip = true;
jaroslav@672
    57
                    String url = (String)c;
jaroslav@1551
    58
                    final Bck2Brwsr.Resources z = toZip(url);
jaroslav@729
    59
                    c = set(classpath, i, z);
jaroslav@1551
    60
                    final byte[] man = readBytes(z, "META-INF/MANIFEST.MF");
jaroslav@672
    61
                    if (man != null) {
jaroslav@702
    62
                        String mainClass = processClassPathAttr(man, url, classpath);
jaroslav@743
    63
//                        if (mainClass != null) {
jaroslav@743
    64
//                            Class.forName(mainClass);
jaroslav@743
    65
//                        }
jaroslav@672
    66
                    }
jaroslav@743
    67
                } catch (IOException ex) {
jaroslav@729
    68
                    set(classpath, i, ex);
jaroslav@746
    69
                    log("Cannot load " + c + " - " + ex.getClass().getName() + ":" + ex.getMessage());
jaroslav@1554
    70
                } finally {
jaroslav@1554
    71
                    doingToZip = false;
jaroslav@644
    72
                }
jaroslav@644
    73
            }
jaroslav@729
    74
            if (res != null) {
jaroslav@729
    75
                byte[] checkRes;
jaroslav@1551
    76
                if (c instanceof Bck2Brwsr.Resources) {
jaroslav@1551
    77
                    checkRes = readBytes((Bck2Brwsr.Resources)c, res);
jaroslav@1387
    78
                    if (checkRes != null && --skip < 0) {
jaroslav@1387
    79
                        return checkRes;
jaroslav@1387
    80
                    }
jaroslav@729
    81
                } else {
jaroslav@1375
    82
                    checkRes = callFunction(c, res, skip);
jaroslav@1387
    83
                    if (checkRes != null) {
jaroslav@1387
    84
                        return checkRes;
jaroslav@1387
    85
                    }
jaroslav@644
    86
                }
jaroslav@644
    87
            }
jaroslav@644
    88
        }
jaroslav@644
    89
        return null;
jaroslav@644
    90
    }
jaroslav@706
    91
    
jaroslav@1375
    92
    @JavaScriptBody(args = { "fn", "res", "skip" }, body = 
jaroslav@1375
    93
        "if (typeof fn === 'function') return fn(res, skip);\n"
jaroslav@729
    94
      + "return null;"
jaroslav@729
    95
    )
jaroslav@1375
    96
    private static native byte[] callFunction(Object fn, String res, int skip);
jaroslav@729
    97
    
jaroslav@1358
    98
    @JavaScriptBody(args = { "msg" }, body = "if (typeof console !== 'undefined') console.log(msg.toString());")
jaroslav@706
    99
    private static native void log(String msg);
jaroslav@706
   100
jaroslav@729
   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@729
   121
    private static Object addToArray(Object arr, String value) {
jaroslav@729
   122
        final int last = length(arr);
jaroslav@729
   123
        Object ret = enlargeArray(arr, last + 1);
jaroslav@729
   124
        set(ret, last, value);
jaroslav@672
   125
        return ret;
jaroslav@672
   126
    }
jaroslav@672
   127
jaroslav@729
   128
    @JavaScriptBody(args = { "arr", "len" }, body = "while (arr.length < len) arr.push(null); return arr;")
jaroslav@729
   129
    private static native Object enlargeArray(Object arr, int len);
jaroslav@1554
   130
jaroslav@1551
   131
    private static Bck2Brwsr.Resources toZip(String path) throws IOException {
jaroslav@1551
   132
        URL u = new URL(path);
jaroslav@1551
   133
        byte[] zipData = (byte[]) u.getContent(new Class[]{byte[].class});
jaroslav@1551
   134
        Bck2Brwsr.Resources r;
jaroslav@1551
   135
        try {
jaroslav@1551
   136
            Class<?> fastJar = Class.forName("org.apidesign.bck2brwsr.vmzip.ZipResources");
jaroslav@1551
   137
            return (Bck2Brwsr.Resources) fastJar.getConstructor(byte[].class).newInstance(zipData);
jaroslav@1551
   138
        } catch (Exception ex) {
jaroslav@1551
   139
            log("Reading JARs is only possible with enum.zip module included: " + ex.getMessage());
jaroslav@1551
   140
            ex.printStackTrace();
jaroslav@1551
   141
            throw new IOException(ex);
jaroslav@1551
   142
        }
jaroslav@1551
   143
    }
jaroslav@1551
   144
    
jaroslav@1551
   145
    private static byte[] readBytes(Bck2Brwsr.Resources r, String res) throws IOException {
jaroslav@1551
   146
        InputStream is = r.get(res);
jaroslav@1551
   147
        if (is == null) {
jaroslav@1551
   148
            return null;
jaroslav@1551
   149
        }
jaroslav@1551
   150
        byte[] arr = new byte[is.available()];
jaroslav@1551
   151
        int off = 0;
jaroslav@1551
   152
        for (;;) {
jaroslav@1551
   153
            int len = is.read(arr, off, arr.length - off);
jaroslav@1551
   154
            if (len == -1) {
jaroslav@1551
   155
                break;
jaroslav@1551
   156
            }
jaroslav@1551
   157
            off += len;
jaroslav@1551
   158
        }
jaroslav@1551
   159
        is.close();
jaroslav@1551
   160
        return arr;
jaroslav@1551
   161
    }
jaroslav@644
   162
}