rt/vm/src/main/java/org/apidesign/vm4brwsr/ExportedSymbols.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 13 May 2014 11:27:01 +0200
branchclosure
changeset 1561 f6200b8decc4
parent 1491 4a1398eff4fb
child 1583 89b6b369c13d
permissions -rw-r--r--
Need to make sure all VALUES fields of enums are exported
     1 /**
     2  * Back 2 Browser Bytecode Translator
     3  * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
     4  *
     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.
     8  *
     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.
    13  *
    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.
    17  */
    18 package org.apidesign.vm4brwsr;
    19 
    20 import java.io.IOException;
    21 import java.io.InputStream;
    22 import java.util.HashMap;
    23 import java.util.Map;
    24 import org.apidesign.bck2brwsr.core.ExtraJavaScript;
    25 import org.apidesign.vm4brwsr.ByteCodeParser.AnnotationParser;
    26 import org.apidesign.vm4brwsr.ByteCodeParser.ClassData;
    27 import org.apidesign.vm4brwsr.ByteCodeParser.FieldData;
    28 import org.apidesign.vm4brwsr.ByteCodeParser.MethodData;
    29 
    30 @ExtraJavaScript(processByteCode = false, resource="")
    31 final class ExportedSymbols {
    32     private final Bck2Brwsr.Resources resources;
    33     private final StringArray exported;
    34     private final Map<Object, Boolean> isMarkedAsExportedCache;
    35 
    36     ExportedSymbols(final Bck2Brwsr.Resources resources, StringArray explicitlyExported) {
    37         this.resources = resources;
    38         this.exported = explicitlyExported;
    39 
    40         isMarkedAsExportedCache = new HashMap<Object, Boolean>();
    41     }
    42 
    43     boolean isExported(ClassData classData) throws IOException {
    44         if (exported.contains(classData.getClassName())) {
    45             return true;
    46         }
    47         return classData.isPublic() && isMarkedAsExportedPackage(
    48                                            classData.getPkgName())
    49                    || isMarkedAsExported(classData);
    50     }
    51 
    52     boolean isExported(MethodData methodData) throws IOException {
    53         return isAccessible(methodData.access) && isExported(methodData.cls)
    54                    || isMarkedAsExported(methodData);
    55     }
    56 
    57     boolean isExported(FieldData fieldData) throws IOException {
    58         if (
    59             isAccessible(fieldData.access) && 
    60             isExported(fieldData.cls) || isMarkedAsExported(fieldData)
    61         ) {
    62             return true;
    63         }
    64         if (
    65             fieldData.isStatic() && fieldData.getName().equals("$VALUES") &&
    66             "java/lang/Enum".equals(fieldData.cls.getSuperClassName())
    67         ) {
    68             // enum values need to be exported
    69             return true;
    70         }
    71         return false;
    72     }
    73 
    74     private boolean isMarkedAsExportedPackage(String pkgName) {
    75         if (pkgName == null) {
    76             return false;
    77         }
    78 
    79         final Boolean cachedValue = isMarkedAsExportedCache.get(pkgName);
    80         if (cachedValue != null) {
    81             return cachedValue;
    82         }
    83 
    84         final boolean newValue = resolveIsMarkedAsExportedPackage(pkgName);
    85         isMarkedAsExportedCache.put(pkgName, newValue);
    86 
    87         return newValue;
    88     }
    89 
    90     private boolean isMarkedAsExported(ClassData classData)
    91             throws IOException {
    92         final Boolean cachedValue = isMarkedAsExportedCache.get(classData);
    93         if (cachedValue != null) {
    94             return cachedValue;
    95         }
    96 
    97         final boolean newValue =
    98                 isMarkedAsExported(classData.findAnnotationData(true),
    99                                    classData);
   100         isMarkedAsExportedCache.put(classData, newValue);
   101 
   102         return newValue;
   103     }
   104 
   105     private boolean isMarkedAsExported(MethodData methodData)
   106             throws IOException {
   107         return isMarkedAsExported(methodData.findAnnotationData(true),
   108                                   methodData.cls);
   109     }
   110 
   111     private boolean isMarkedAsExported(FieldData fieldData)
   112             throws IOException {
   113         return isMarkedAsExported(fieldData.findAnnotationData(true),
   114                                   fieldData.cls);
   115     }
   116 
   117     private boolean resolveIsMarkedAsExportedPackage(String pkgName) {
   118         try {
   119             final InputStream is =
   120                     resources.get(pkgName + "/package-info.class");
   121             if (is == null) {
   122                 return false;
   123             }
   124 
   125             try {
   126                 final ClassData pkgInfoClass = new ClassData(is);
   127                 return isMarkedAsExported(
   128                                pkgInfoClass.findAnnotationData(true),
   129                                pkgInfoClass);
   130             } finally {
   131                 is.close();
   132             }
   133         } catch (final IOException e) {
   134             return false;
   135         }
   136     }
   137 
   138     private boolean isMarkedAsExported(byte[] arrData, ClassData cd)
   139             throws IOException {
   140         if (arrData == null) {
   141             return false;
   142         }
   143 
   144         final boolean[] found = { false };
   145         final AnnotationParser annotationParser =
   146                 new AnnotationParser(false, false) {
   147                     @Override
   148                     protected void visitAnnotationStart(
   149                             String type,
   150                             boolean top) {
   151                         if (top && type.equals("Lorg/apidesign/bck2brwsr"
   152                                                    + "/core/Exported;")) {
   153                             found[0] = true;
   154                         }
   155                     }
   156                 };
   157         annotationParser.parse(arrData, cd);
   158         return found[0];
   159     }
   160 
   161     private static boolean isAccessible(int access) {
   162         return (access & (ByteCodeParser.ACC_PUBLIC
   163                               | ByteCodeParser.ACC_PROTECTED)) != 0;
   164     }
   165 }