aboutsummaryrefslogtreecommitdiffstats
path: root/bcel-builder
diff options
context:
space:
mode:
authorAndy Clement <aclement@pivotal.io>2016-06-06 12:00:12 -0700
committerAndy Clement <aclement@pivotal.io>2016-06-06 12:00:12 -0700
commitf8b86ff2c03a77e47e87573b59bc43c57cfdee38 (patch)
treebab01913bce4b8be028821ac90c596a613c4bee1 /bcel-builder
parent87de66e902d8fa1818f9debd12cd5195ccdb19ac (diff)
downloadaspectj-f8b86ff2c03a77e47e87573b59bc43c57cfdee38.tar.gz
aspectj-f8b86ff2c03a77e47e87573b59bc43c57cfdee38.zip
[j9] Basic read/write of Module attribute
Diffstat (limited to 'bcel-builder')
-rw-r--r--bcel-builder/src/org/aspectj/apache/bcel/Constants.java10
-rw-r--r--bcel-builder/src/org/aspectj/apache/bcel/classfile/Module.java432
-rwxr-xr-xbcel-builder/testdata/modules/cpl.sh24
-rw-r--r--bcel-builder/testdata/modules/one/module-info.classbin0 -> 124 bytes
-rw-r--r--bcel-builder/testdata/modules/one/module-info.java1
-rw-r--r--bcel-builder/testdata/modules/two/a/module-info.classbin0 -> 126 bytes
-rw-r--r--bcel-builder/testdata/modules/two/a/module-info.java2
-rw-r--r--bcel-builder/testdata/modules/two/b/module-info.classbin0 -> 126 bytes
-rw-r--r--bcel-builder/testdata/modules/two/b/module-info.java2
-rw-r--r--bcel-builder/testdata/modules/two/c/module-info.classbin0 -> 126 bytes
-rw-r--r--bcel-builder/testdata/modules/two/c/module-info.java2
-rw-r--r--bcel-builder/testdata/modules/two/d/module-info.classbin0 -> 150 bytes
-rw-r--r--bcel-builder/testdata/modules/two/d/module-info.java4
-rw-r--r--bcel-builder/testdata/modules/two/e/C1.java3
-rw-r--r--bcel-builder/testdata/modules/two/e/C2.java3
-rw-r--r--bcel-builder/testdata/modules/two/e/C3.java3
-rw-r--r--bcel-builder/testdata/modules/two/e/com/foo1/C1.classbin0 -> 187 bytes
-rw-r--r--bcel-builder/testdata/modules/two/e/com/foo2/C2.classbin0 -> 187 bytes
-rw-r--r--bcel-builder/testdata/modules/two/e/com/foo3/C3.classbin0 -> 187 bytes
-rw-r--r--bcel-builder/testdata/modules/two/e/module-info.classbin0 -> 193 bytes
-rw-r--r--bcel-builder/testdata/modules/two/e/module-info.java5
-rw-r--r--bcel-builder/testdata/modules/two/f/I1.java3
-rw-r--r--bcel-builder/testdata/modules/two/f/com/foo1/I1.classbin0 -> 94 bytes
-rw-r--r--bcel-builder/testdata/modules/two/f/module-info.classbin0 -> 145 bytes
-rw-r--r--bcel-builder/testdata/modules/two/f/module-info.java3
-rw-r--r--bcel-builder/testdata/modules/two/g/C1.java3
-rw-r--r--bcel-builder/testdata/modules/two/g/C2.java3
-rw-r--r--bcel-builder/testdata/modules/two/g/I1.java3
-rw-r--r--bcel-builder/testdata/modules/two/g/I2.java3
-rw-r--r--bcel-builder/testdata/modules/two/g/com/foo1/C1.classbin0 -> 206 bytes
-rw-r--r--bcel-builder/testdata/modules/two/g/com/foo1/I1.classbin0 -> 94 bytes
-rw-r--r--bcel-builder/testdata/modules/two/g/com/foo2/C2.classbin0 -> 206 bytes
-rw-r--r--bcel-builder/testdata/modules/two/g/com/foo2/I2.classbin0 -> 94 bytes
-rw-r--r--bcel-builder/testdata/modules/two/g/module-info.classbin0 -> 202 bytes
-rw-r--r--bcel-builder/testdata/modules/two/g/module-info.java4
-rw-r--r--bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/BcelTestCase.java14
-rw-r--r--bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/ModuleTest.java129
37 files changed, 652 insertions, 4 deletions
diff --git a/bcel-builder/src/org/aspectj/apache/bcel/Constants.java b/bcel-builder/src/org/aspectj/apache/bcel/Constants.java
index 4499284c2..8d8427006 100644
--- a/bcel-builder/src/org/aspectj/apache/bcel/Constants.java
+++ b/bcel-builder/src/org/aspectj/apache/bcel/Constants.java
@@ -80,6 +80,8 @@ public interface Constants {
public final static short MINOR_1_7 = 0;
public final static short MAJOR_1_8 = 52;
public final static short MINOR_1_8 = 0;
+ public final static short MAJOR_1_9 = 53;
+ public final static short MINOR_1_9 = 0;
// Defaults
public final static short MAJOR = MAJOR_1_1;
public final static short MINOR = MINOR_1_1;
@@ -105,12 +107,18 @@ public interface Constants {
public final static short ACC_INTERFACE = 0x0200;
public final static short ACC_ABSTRACT = 0x0400;
public final static short ACC_STRICT = 0x0800;
-
+
public final static short ACC_ANNOTATION = 0x2000;
public final static short ACC_ENUM = 0x4000;
+ public final static int ACC_MODULE = 0x8000;
public final static short ACC_BRIDGE = 0x0040;
public final static short ACC_VARARGS = 0x0080;
+ // module related
+ public final static int MODULE_ACC_PUBLIC = 0x0020;
+ public final static int MODULE_ACC_SYNTHETIC = 0x1000;
+ public final static int MODULE_ACC_MANDATED = 0x8000;
+
// Applies to classes compiled by new compilers only
public final static short ACC_SUPER = 0x0020;
diff --git a/bcel-builder/src/org/aspectj/apache/bcel/classfile/Module.java b/bcel-builder/src/org/aspectj/apache/bcel/classfile/Module.java
new file mode 100644
index 000000000..217f5fbd0
--- /dev/null
+++ b/bcel-builder/src/org/aspectj/apache/bcel/classfile/Module.java
@@ -0,0 +1,432 @@
+/* ====================================================================
+ * The Apache Software License, Version 1.1
+ *
+ * Copyright (c) 2001 The Apache Software Foundation. All rights
+ * reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in
+ * the documentation and/or other materials provided with the
+ * distribution.
+ *
+ * 3. The end-user documentation included with the redistribution,
+ * if any, must include the following acknowledgment:
+ * "This product includes software developed by the
+ * Apache Software Foundation (http://www.apache.org/)."
+ * Alternately, this acknowledgment may appear in the software itself,
+ * if and wherever such third-party acknowledgments normally appear.
+ *
+ * 4. The names "Apache" and "Apache Software Foundation" and
+ * "Apache BCEL" must not be used to endorse or promote products
+ * derived from this software without prior written permission. For
+ * written permission, please contact apache@apache.org.
+ *
+ * 5. Products derived from this software may not be called "Apache",
+ * "Apache BCEL", nor may "Apache" appear in their name, without
+ * prior written permission of the Apache Software Foundation.
+ *
+ * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
+ * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
+ * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
+ * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
+ * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ * ====================================================================
+ *
+ * This software consists of voluntary contributions made by many
+ * individuals on behalf of the Apache Software Foundation. For more
+ * information on the Apache Software Foundation, please see
+ * <http://www.apache.org/>.
+ */
+package org.aspectj.apache.bcel.classfile;
+
+import java.io.ByteArrayInputStream;
+import java.io.DataInputStream;
+import java.io.DataOutputStream;
+import java.io.IOException;
+
+import org.aspectj.apache.bcel.Constants;
+import org.aspectj.apache.bcel.classfile.Module.Export;
+
+/**
+ * This class is derived from <em>Attribute</em> and represents the module
+ * information captured in a class file.
+ * http://cr.openjdk.java.net/~mr/jigsaw/spec/lang-vm.html
+ *
+ * @author Andy Clement
+ */
+public final class Module extends Attribute {
+
+ private static final String[] NO_MODULE_NAMES = {};
+
+ private byte[] moduleInfo;
+ private int ptr;
+ private boolean unpacked = false;
+ private Require[] requires;
+ private Export[] exports;
+ private Uses[] uses;
+ private Provide[] provides;
+
+ /**
+ * Build a Module attribute from a previously Unknown attribute.
+ */
+ public Module(Unknown unknown) {
+ super(unknown.getTag(), unknown.getNameIndex(), unknown.getLength(), unknown.getConstantPool());
+ moduleInfo = unknown.getBytes();
+ }
+
+ public class Require {
+
+ private final int moduleNameIndex;
+ private final int requiresFlags;
+
+ public Require(int moduleNameIndex, int requiresFlags) {
+ this.moduleNameIndex = moduleNameIndex;
+ this.requiresFlags = requiresFlags;
+ }
+
+ public String getModuleName() {
+ return cpool.getConstantUtf8(moduleNameIndex).getStringValue();
+ }
+
+ public int getRequiresFlags() {
+ return requiresFlags;
+ }
+
+ public String getRequiresFlagsAsString() {
+ StringBuilder s = new StringBuilder();
+ if ((requiresFlags & Constants.MODULE_ACC_PUBLIC)!=0) {
+ s.append("public ");
+ }
+ if ((requiresFlags & Constants.MODULE_ACC_SYNTHETIC)!=0) {
+ s.append("synthetic ");
+ }
+ if ((requiresFlags & Constants.MODULE_ACC_MANDATED)!=0) {
+ s.append("mandated ");
+ }
+ return s.toString();
+ }
+
+ public String toString() {
+ return "requires "+getRequiresFlagsAsString()+getModuleName();
+ }
+
+ }
+
+
+ public class Export {
+
+ private final int exportedPackageNameIndex;
+ private final int[] toModuleNameIndices;
+
+ public Export(int exportedPackageNameIndex, int[] toModuleNameIndices) {
+ this.exportedPackageNameIndex = exportedPackageNameIndex;
+ this.toModuleNameIndices = toModuleNameIndices;
+ }
+
+ public String getExportedPackage() {
+ return cpool.getConstantUtf8(exportedPackageNameIndex).getStringValue();
+ }
+
+ public String[] getToModuleNames() {
+ if (toModuleNameIndices==null) {
+ return NO_MODULE_NAMES;
+ }
+ String[] toModuleNames = new String[toModuleNameIndices.length];
+ for (int i=0;i<toModuleNameIndices.length;i++) {
+ toModuleNames[i] = cpool.getConstantUtf8(toModuleNameIndices[i]).getStringValue();
+ }
+ return toModuleNames;
+ }
+
+ public String toString() {
+ StringBuilder s =new StringBuilder();
+ s.append("exports ").append(getExportedPackage().replace('/', '.'));
+ String[] toModules = getToModuleNames();
+ if (toModules.length!=0) {
+ s.append(" to ");
+ for (int i=0;i<toModules.length;i++) {
+ if (i>0) {
+ s.append(", ");
+ }
+ s.append(toModules[i]);
+ }
+ }
+ return s.toString().trim();
+ }
+ }
+
+ public class Provide {
+ private final int providedTypeIndex;
+ private final int withTypeIndex;
+
+ public Provide(int providedTypeIndex, int withTypeIndex) {
+ this.providedTypeIndex = providedTypeIndex;
+ this.withTypeIndex = withTypeIndex;
+ }
+
+ public String getProvidedType() {
+ return cpool.getConstantString_CONSTANTClass(providedTypeIndex);
+ }
+
+ public int getProvidedTypeIndex() {
+ return providedTypeIndex;
+ }
+
+ public String getWithType() {
+ return cpool.getConstantString_CONSTANTClass(withTypeIndex);
+ }
+
+ public int getWithTypeIndex() {
+ return withTypeIndex;
+ }
+
+ public String toString() {
+ StringBuilder s =new StringBuilder();
+ s.append("provides ").append(getProvidedType().replace('/', '.'));
+ s.append(" with ").append(getWithType().replace('/','.'));
+ return s.toString();
+ }
+ }
+
+ public class Uses {
+ private final int typeNameIndex;
+
+ public Uses(int typeNameIndex) {
+ this.typeNameIndex = typeNameIndex;
+ }
+
+ public String getTypeName() {
+ return cpool.getConstantString_CONSTANTClass(typeNameIndex);
+ }
+
+ public int getTypeNameIndex() {
+ return typeNameIndex;
+ }
+
+ public String toString() {
+ StringBuilder s =new StringBuilder();
+ s.append("uses ").append(getTypeName().replace('/', '.'));
+ return s.toString().trim();
+ }
+ }
+
+ private final int readInt() {
+ return ((moduleInfo[ptr++] & 0xFF) << 24) + ((moduleInfo[ptr++] & 0xFF) << 16)
+ + ((moduleInfo[ptr++] & 0xFF) << 8) + (moduleInfo[ptr++] & 0xFF);
+ }
+
+ private final int readUnsignedShort() {
+ return ((moduleInfo[ptr++] & 0xff) << 8) + (moduleInfo[ptr++] & 0xff);
+ }
+
+ private final int readUnsignedShort(int offset) {
+ return ((moduleInfo[offset++] & 0xff) << 8) + (moduleInfo[offset] & 0xff);
+ }
+
+ private void ensureUnpacked() {
+ if (!unpacked) {
+ ptr = 0;
+ int count = readUnsignedShort();
+ requires = new Require[count];
+ for (int i = 0; i < count; i++) {
+ requires[i] = new Require(readUnsignedShort(), readUnsignedShort());
+ }
+ count = readUnsignedShort();
+ exports = new Export[count];
+ for (int i = 0; i < count; i++) {
+ int index = readUnsignedShort();
+ int toCount = readUnsignedShort();
+ int[] to = new int[toCount];
+ for (int j = 0; j < toCount; j++) {
+ to[j] = readUnsignedShort();
+ }
+ exports[i] = new Export(index, to);
+ }
+ count = readUnsignedShort();
+ uses = new Uses[count];
+ for (int i = 0; i < count; i++) {
+ uses[i] = new Uses(readUnsignedShort());
+ }
+ count = readUnsignedShort();
+ provides = new Provide[count];
+ for (int i = 0; i < count; i++) {
+ provides[i] = new Provide(readUnsignedShort(), readUnsignedShort());
+ }
+ unpacked = true;
+ }
+ }
+
+ @Override
+ public final void dump(DataOutputStream file) throws IOException {
+ super.dump(file);
+ if (!unpacked) {
+ file.write(moduleInfo);
+ } else {
+ file.writeShort(requires.length);
+ for (int i = 0; i < requires.length; i++) {
+ file.writeShort(requires[i].moduleNameIndex);
+ file.writeShort(requires[i].requiresFlags);
+ }
+ file.writeShort(exports.length);
+ for (Export export : exports) {
+ file.writeShort(export.exportedPackageNameIndex);
+ int[] toIndices = export.toModuleNameIndices;
+ file.writeShort(toIndices.length);
+ for (int index : toIndices) {
+ file.writeShort(index);
+ }
+ }
+ file.writeShort(uses.length);
+ for (Uses use : uses) {
+ file.writeShort(use.getTypeNameIndex());
+ }
+ file.writeShort(provides.length);
+ for (Provide provide : provides) {
+ file.writeShort(provide.providedTypeIndex);
+ file.writeShort(provide.withTypeIndex);
+ }
+ }
+ }
+
+ public String toStringRequires() {
+ StringBuilder s = new StringBuilder();
+ s.append('#').append(requires.length);
+ if (requires.length > 0) {
+ for (Require require : requires) {
+ s.append(' ');
+ s.append(require.moduleNameIndex).append(':').append(require.requiresFlags);
+ }
+ }
+ return s.toString();
+ }
+
+ public String toStringExports() {
+ StringBuilder s = new StringBuilder();
+ s.append('#').append(exports.length);
+ if (exports.length > 0) {
+ for (Export export : exports) {
+ s.append(' ');
+ s.append(export.exportedPackageNameIndex).append(":[");
+ int[] toIndices = export.toModuleNameIndices;
+ for (int i = 0; i < toIndices.length; i++) {
+ if (i > 0)
+ s.append(',');
+ s.append(toIndices[i]);
+ }
+ s.append("]");
+ }
+ }
+ return s.toString();
+ }
+
+ public String toStringUses() {
+ StringBuilder s = new StringBuilder();
+ s.append('#').append(uses.length);
+ if (uses.length > 0) {
+ for (Uses use : uses) {
+ s.append(' ');
+ s.append(use.getTypeName());
+ }
+ }
+ return s.toString();
+ }
+
+ public String toStringProvides() {
+ StringBuilder s = new StringBuilder();
+ s.append('#').append(provides.length);
+ if (provides.length > 0) {
+ for (Provide provide : provides) {
+ s.append(' ');
+ s.append(provide.providedTypeIndex).append(':').append(provide.withTypeIndex);
+ }
+ }
+ return s.toString();
+ }
+
+ @Override
+ public final String toString() {
+ StringBuilder s = new StringBuilder();
+ ensureUnpacked();
+ s.append("Module(");
+ if (requires.length != 0) {
+ s.append("requires=");
+ s.append(toStringRequires());
+ s.append(" ");
+ }
+ if (exports.length != 0) {
+ s.append("exports=");
+ s.append(toStringExports());
+ s.append(" ");
+ }
+ if (uses.length != 0) {
+ s.append("uses=");
+ s.append(toStringUses());
+ s.append(" ");
+ }
+ if (provides.length != 0) {
+ s.append("provides=");
+ s.append(toStringProvides());
+ s.append(" ");
+ }
+ return s.toString().trim()+")";
+ }
+
+ /**
+ * @return deep copy of this attribute //
+ */
+ // @Override
+ // public Attribute copy(ConstantPool constant_pool) {
+ // return (SourceFile) clone();
+ // }
+ @Override
+ public void accept(ClassVisitor v) {
+ v.visitSourceFile(this);
+ }
+
+ public Require[] getRequires() {
+ ensureUnpacked();
+ return requires;
+ }
+
+ public String[] getRequiredModuleNames() {
+ ensureUnpacked();
+ String[] results = new String[requires.length];
+ for (int i=0;i<requires.length;i++) {
+ results[i] = cpool.getConstantUtf8(requires[i].moduleNameIndex).getStringValue();
+ }
+ return results;
+ }
+
+ public byte[] getBytes() {
+ return moduleInfo;
+ }
+
+ public Export[] getExports() {
+ ensureUnpacked();
+ return exports;
+ }
+
+ public Uses[] getUses() {
+ ensureUnpacked();
+ return uses;
+ }
+
+ public Provide[] getProvides() {
+ ensureUnpacked();
+ return provides;
+ }
+}
diff --git a/bcel-builder/testdata/modules/cpl.sh b/bcel-builder/testdata/modules/cpl.sh
new file mode 100755
index 000000000..0ac21453d
--- /dev/null
+++ b/bcel-builder/testdata/modules/cpl.sh
@@ -0,0 +1,24 @@
+cd one
+javac module-info.java
+cd ..
+cd two/a
+javac module-info.java
+cd ../..
+cd two/b
+javac module-info.java
+cd ../..
+cd two/c
+javac module-info.java
+cd ../..
+cd two/d
+javac module-info.java -modulepath ../a:../b
+cd ../..
+cd two/e
+javac module-info.java C1.java C2.java C3.java -d . -modulepath ../a:../b
+cd ../..
+cd two/f
+javac module-info.java I1.java -d .
+cd ../..
+cd two/g
+javac module-info.java I1.java I2.java C1.java C2.java -d .
+cd ../..
diff --git a/bcel-builder/testdata/modules/one/module-info.class b/bcel-builder/testdata/modules/one/module-info.class
new file mode 100644
index 000000000..0088bd13a
--- /dev/null
+++ b/bcel-builder/testdata/modules/one/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/one/module-info.java b/bcel-builder/testdata/modules/one/module-info.java
new file mode 100644
index 000000000..d0913deeb
--- /dev/null
+++ b/bcel-builder/testdata/modules/one/module-info.java
@@ -0,0 +1 @@
+module one {}
diff --git a/bcel-builder/testdata/modules/two/a/module-info.class b/bcel-builder/testdata/modules/two/a/module-info.class
new file mode 100644
index 000000000..c0003b8f6
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/a/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/a/module-info.java b/bcel-builder/testdata/modules/two/a/module-info.java
new file mode 100644
index 000000000..8ab4a7dc9
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/a/module-info.java
@@ -0,0 +1,2 @@
+module a.b.c {
+}
diff --git a/bcel-builder/testdata/modules/two/b/module-info.class b/bcel-builder/testdata/modules/two/b/module-info.class
new file mode 100644
index 000000000..6721be124
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/b/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/b/module-info.java b/bcel-builder/testdata/modules/two/b/module-info.java
new file mode 100644
index 000000000..308ba05bb
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/b/module-info.java
@@ -0,0 +1,2 @@
+module b.c.d {
+}
diff --git a/bcel-builder/testdata/modules/two/c/module-info.class b/bcel-builder/testdata/modules/two/c/module-info.class
new file mode 100644
index 000000000..0c81601d5
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/c/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/c/module-info.java b/bcel-builder/testdata/modules/two/c/module-info.java
new file mode 100644
index 000000000..501d75a06
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/c/module-info.java
@@ -0,0 +1,2 @@
+module c.d.e {
+}
diff --git a/bcel-builder/testdata/modules/two/d/module-info.class b/bcel-builder/testdata/modules/two/d/module-info.class
new file mode 100644
index 000000000..358624633
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/d/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/d/module-info.java b/bcel-builder/testdata/modules/two/d/module-info.java
new file mode 100644
index 000000000..5b9446eed
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/d/module-info.java
@@ -0,0 +1,4 @@
+module d.e.f {
+ requires a.b.c;
+ requires public b.c.d;
+}
diff --git a/bcel-builder/testdata/modules/two/e/C1.java b/bcel-builder/testdata/modules/two/e/C1.java
new file mode 100644
index 000000000..eb087840d
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/C1.java
@@ -0,0 +1,3 @@
+package com.foo1;
+
+public class C1 {}
diff --git a/bcel-builder/testdata/modules/two/e/C2.java b/bcel-builder/testdata/modules/two/e/C2.java
new file mode 100644
index 000000000..528b87e02
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/C2.java
@@ -0,0 +1,3 @@
+package com.foo2;
+
+public class C2 {}
diff --git a/bcel-builder/testdata/modules/two/e/C3.java b/bcel-builder/testdata/modules/two/e/C3.java
new file mode 100644
index 000000000..205f75fac
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/C3.java
@@ -0,0 +1,3 @@
+package com.foo3;
+
+public class C3 {}
diff --git a/bcel-builder/testdata/modules/two/e/com/foo1/C1.class b/bcel-builder/testdata/modules/two/e/com/foo1/C1.class
new file mode 100644
index 000000000..6b5bb5fd4
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/com/foo1/C1.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/e/com/foo2/C2.class b/bcel-builder/testdata/modules/two/e/com/foo2/C2.class
new file mode 100644
index 000000000..d9eec8bcd
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/com/foo2/C2.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/e/com/foo3/C3.class b/bcel-builder/testdata/modules/two/e/com/foo3/C3.class
new file mode 100644
index 000000000..d64d2fc98
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/com/foo3/C3.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/e/module-info.class b/bcel-builder/testdata/modules/two/e/module-info.class
new file mode 100644
index 000000000..565bbbea7
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/e/module-info.java b/bcel-builder/testdata/modules/two/e/module-info.java
new file mode 100644
index 000000000..230097050
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/e/module-info.java
@@ -0,0 +1,5 @@
+module e.f.g {
+ exports com.foo1;
+ exports com.foo2 to a.b.c;
+ exports com.foo3 to a.b.c, b.c.d;
+}
diff --git a/bcel-builder/testdata/modules/two/f/I1.java b/bcel-builder/testdata/modules/two/f/I1.java
new file mode 100644
index 000000000..88029d83c
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/f/I1.java
@@ -0,0 +1,3 @@
+package com.foo1;
+
+public interface I1 {}
diff --git a/bcel-builder/testdata/modules/two/f/com/foo1/I1.class b/bcel-builder/testdata/modules/two/f/com/foo1/I1.class
new file mode 100644
index 000000000..da13682c4
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/f/com/foo1/I1.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/f/module-info.class b/bcel-builder/testdata/modules/two/f/module-info.class
new file mode 100644
index 000000000..9377bb8cc
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/f/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/f/module-info.java b/bcel-builder/testdata/modules/two/f/module-info.java
new file mode 100644
index 000000000..78978622c
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/f/module-info.java
@@ -0,0 +1,3 @@
+module f.g.h {
+ uses com.foo1.I1;
+}
diff --git a/bcel-builder/testdata/modules/two/g/C1.java b/bcel-builder/testdata/modules/two/g/C1.java
new file mode 100644
index 000000000..e25a57ec8
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/C1.java
@@ -0,0 +1,3 @@
+package com.foo1;
+
+public class C1 implements I1 {}
diff --git a/bcel-builder/testdata/modules/two/g/C2.java b/bcel-builder/testdata/modules/two/g/C2.java
new file mode 100644
index 000000000..100c37b18
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/C2.java
@@ -0,0 +1,3 @@
+package com.foo2;
+
+public class C2 implements I2 {}
diff --git a/bcel-builder/testdata/modules/two/g/I1.java b/bcel-builder/testdata/modules/two/g/I1.java
new file mode 100644
index 000000000..88029d83c
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/I1.java
@@ -0,0 +1,3 @@
+package com.foo1;
+
+public interface I1 {}
diff --git a/bcel-builder/testdata/modules/two/g/I2.java b/bcel-builder/testdata/modules/two/g/I2.java
new file mode 100644
index 000000000..bd52da822
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/I2.java
@@ -0,0 +1,3 @@
+package com.foo2;
+
+public interface I2 {}
diff --git a/bcel-builder/testdata/modules/two/g/com/foo1/C1.class b/bcel-builder/testdata/modules/two/g/com/foo1/C1.class
new file mode 100644
index 000000000..dfc944836
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/com/foo1/C1.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/g/com/foo1/I1.class b/bcel-builder/testdata/modules/two/g/com/foo1/I1.class
new file mode 100644
index 000000000..da13682c4
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/com/foo1/I1.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/g/com/foo2/C2.class b/bcel-builder/testdata/modules/two/g/com/foo2/C2.class
new file mode 100644
index 000000000..aae7ea279
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/com/foo2/C2.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/g/com/foo2/I2.class b/bcel-builder/testdata/modules/two/g/com/foo2/I2.class
new file mode 100644
index 000000000..c31533fbf
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/com/foo2/I2.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/g/module-info.class b/bcel-builder/testdata/modules/two/g/module-info.class
new file mode 100644
index 000000000..415a54036
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/module-info.class
Binary files differ
diff --git a/bcel-builder/testdata/modules/two/g/module-info.java b/bcel-builder/testdata/modules/two/g/module-info.java
new file mode 100644
index 000000000..00633c573
--- /dev/null
+++ b/bcel-builder/testdata/modules/two/g/module-info.java
@@ -0,0 +1,4 @@
+module g.h.i {
+ provides com.foo1.I1 with com.foo1.C1;
+ provides com.foo2.I2 with com.foo2.C2;
+}
diff --git a/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/BcelTestCase.java b/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/BcelTestCase.java
index 3311c5386..ef4b8a9a7 100644
--- a/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/BcelTestCase.java
+++ b/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/BcelTestCase.java
@@ -1,5 +1,5 @@
/* *******************************************************************
- * Copyright (c) 2004, 2013 IBM, VMware
+ * Copyright (c) 2004 - 2016 IBM, VMware, Contributors
* All rights reserved.
* This program and the accompanying materials are made available
* under the terms of the Eclipse Public License v1.0
@@ -34,8 +34,7 @@ import org.aspectj.apache.bcel.util.SyntheticRepository;
/**
* Super class for the Java5 tests, includes various helper methods.
*/
-
-public class BcelTestCase extends TestCase {
+public abstract class BcelTestCase extends TestCase {
private boolean verbose = false;
@@ -175,4 +174,13 @@ public class BcelTestCase extends TestCase {
return null;
}
+ public Attribute getAttribute(Attribute[] attrs, String name) {
+ for (Attribute attr: attrs) {
+ if (attr.getName().equals(name)) {
+ return attr;
+ }
+ }
+ return null;
+ }
+
}
diff --git a/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/ModuleTest.java b/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/ModuleTest.java
new file mode 100644
index 000000000..d6e2b456b
--- /dev/null
+++ b/bcel-builder/testsrc/org/aspectj/apache/bcel/classfile/tests/ModuleTest.java
@@ -0,0 +1,129 @@
+/* *******************************************************************
+ * Copyright (c) 2016 Contributors
+ * All rights reserved.
+ * This program and the accompanying materials are made available
+ * under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ * Andy Clement - initial implementation
+ * ******************************************************************/
+
+package org.aspectj.apache.bcel.classfile.tests;
+
+import java.io.ByteArrayOutputStream;
+import java.io.DataOutputStream;
+
+import org.aspectj.apache.bcel.Constants;
+import org.aspectj.apache.bcel.classfile.Attribute;
+import org.aspectj.apache.bcel.classfile.ClassParser;
+import org.aspectj.apache.bcel.classfile.JavaClass;
+import org.aspectj.apache.bcel.classfile.Module;
+import org.aspectj.apache.bcel.classfile.Module.Export;
+import org.aspectj.apache.bcel.classfile.Module.Provide;
+import org.aspectj.apache.bcel.classfile.Module.Require;
+import org.aspectj.apache.bcel.classfile.Module.Uses;
+import org.aspectj.apache.bcel.classfile.SourceFile;
+import org.aspectj.apache.bcel.classfile.Unknown;
+
+/**
+ * http://cr.openjdk.java.net/~mr/jigsaw/spec/lang-vm.html
+ *
+ * @author Andy Clement
+ */
+public class ModuleTest extends BcelTestCase {
+
+ public void testLoadSimpleModuleClass() throws Exception {
+ String moduleFilename = "testdata/modules/one/module-info.class";
+ ClassParser classParser = new ClassParser(moduleFilename);
+ JavaClass javaClass = classParser.parse();
+ assertNotNull(javaClass);
+ assertEquals(Constants.MAJOR_1_9,javaClass.getMajor());
+ assertEquals(Constants.MINOR_1_9,javaClass.getMinor());
+ assertEquals(Constants.ACC_MODULE,javaClass.getModifiers());
+ assertEquals(0,javaClass.getSuperclassNameIndex());
+ assertEquals(0,javaClass.getInterfaceIndices().length);
+ assertEquals(0,javaClass.getFields().length);
+ assertEquals(0,javaClass.getMethods().length);
+ Attribute[] attrs = javaClass.getAttributes();
+ assertEquals(2,attrs.length);
+ SourceFile sourceFile = (SourceFile) getAttribute(attrs,Constants.ATTR_SOURCE_FILE);
+ Module moduleAttr = new Module((Unknown)getAttribute(attrs,Constants.ATTR_UNKNOWN));
+ byte[] originalData = moduleAttr.getBytes();
+ String[] requiredModuleNames = moduleAttr.getRequiredModuleNames();
+ assertEquals(1,requiredModuleNames.length);
+ assertEquals("java.base",requiredModuleNames[0]);
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ moduleAttr.dump(new DataOutputStream(baos));
+ byte[] newData = baos.toByteArray();
+ // The 6 offset here is because the newdata includes the 2byte cpool pointer for the name 'Module'
+ // and the 4byte int length field for the attribute data
+ if (newData.length!=originalData.length+6) {
+ fail("Expected the length of the original attribute ("+originalData.length+") to match the new written length ("+newData.length+")");
+ }
+ for (int i=0;i<originalData.length;i++) {
+ if (originalData[i]!=newData[i+6]) {
+ fail("byte mismatch at position "+i+" of "+newData.length);
+ }
+ }
+ }
+
+
+ public void testRequires() throws Exception {
+ Module moduleAttr = getModuleAttribute("testdata/modules/two/d/module-info.class");
+ Require[] requires = moduleAttr.getRequires();
+ assertEquals(3,requires.length);
+ assertEquals("requires mandated java.base",requires[0].toString());
+ assertEquals("requires a.b.c",requires[1].toString());
+ assertEquals("requires public b.c.d",requires[2].toString());
+ assertEquals("java.base",requires[0].getModuleName());
+ assertEquals("a.b.c",requires[1].getModuleName());
+ assertEquals("b.c.d",requires[2].getModuleName());
+ }
+
+ public void testExports() throws Exception {
+ Module moduleAttr = getModuleAttribute("testdata/modules/two/e/module-info.class");
+ Export[] exports = moduleAttr.getExports();
+ assertEquals(3,exports.length);
+ assertEquals("exports com.foo1",exports[0].toString());
+ assertEquals("exports com.foo2 to a.b.c",exports[1].toString());
+ assertEquals("exports com.foo3 to b.c.d, a.b.c",exports[2].toString());
+ assertEquals("com/foo1",exports[0].getExportedPackage());
+ assertEquals("com/foo2",exports[1].getExportedPackage());
+ assertEquals("com/foo3",exports[2].getExportedPackage());
+ assertEquals("a.b.c",exports[1].getToModuleNames()[0]);
+ assertEquals("b.c.d",exports[2].getToModuleNames()[0]);
+ assertEquals("a.b.c",exports[2].getToModuleNames()[1]);
+ }
+
+ public void testUses() throws Exception {
+ Module moduleAttr = getModuleAttribute("testdata/modules/two/f/module-info.class");
+ Uses[] uses = moduleAttr.getUses();
+ assertEquals(1,uses.length);
+ assertEquals("com/foo1/I1",uses[0].getTypeName());
+ assertEquals("uses com.foo1.I1",uses[0].toString());
+ }
+
+ public void testProvides() throws Exception {
+ Module moduleAttr = getModuleAttribute("testdata/modules/two/g/module-info.class");
+ Provide[] provides = moduleAttr.getProvides();
+ assertEquals(2,provides.length);
+ assertEquals("provides com.foo1.I1 with com.foo1.C1",provides[0].toString());
+ assertEquals("provides com.foo2.I2 with com.foo2.C2",provides[1].toString());
+ assertEquals("com/foo1/I1",provides[0].getProvidedType());
+ assertEquals("com/foo1/C1",provides[0].getWithType());
+ assertEquals("com/foo2/I2",provides[1].getProvidedType());
+ assertEquals("com/foo2/C2",provides[1].getWithType());
+ }
+
+ // ---
+
+ private Module getModuleAttribute(String moduleInfoClass) throws Exception {
+ ClassParser classParser = new ClassParser(moduleInfoClass);
+ JavaClass javaClass = classParser.parse();
+ Module moduleAttr = new Module((Unknown)getAttribute(javaClass.getAttributes(),Constants.ATTR_UNKNOWN));
+ return moduleAttr;
+ }
+
+}