2009-12-02 19:08:29 -07:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 11:06:52 -07:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2009-05-25 21:36:29 -06:00
|
|
|
package avian;
|
2007-07-30 17:19:05 -06:00
|
|
|
|
2009-09-19 16:21:15 -06:00
|
|
|
import static avian.Stream.read1;
|
|
|
|
import static avian.Stream.read2;
|
|
|
|
|
|
|
|
import java.lang.reflect.Method;
|
|
|
|
import java.lang.reflect.Field;
|
2007-08-10 17:45:47 -06:00
|
|
|
import java.net.URL;
|
|
|
|
import java.net.MalformedURLException;
|
2009-09-19 16:21:15 -06:00
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.ByteArrayInputStream;
|
|
|
|
import java.io.IOException;
|
2007-08-10 17:45:47 -06:00
|
|
|
|
2007-07-30 17:19:05 -06:00
|
|
|
public class SystemClassLoader extends ClassLoader {
|
2009-09-19 16:21:15 -06:00
|
|
|
private static final int LinkFlag = 1 << 8;
|
|
|
|
|
|
|
|
public static native Class defineClass
|
|
|
|
(ClassLoader loader, byte[] b, int offset, int length);
|
|
|
|
|
2007-07-30 17:19:05 -06:00
|
|
|
protected native Class findClass(String name) throws ClassNotFoundException;
|
|
|
|
|
2009-12-22 21:34:04 -07:00
|
|
|
protected native Class reallyFindLoadedClass(String name);
|
2007-08-10 17:45:47 -06:00
|
|
|
|
|
|
|
private native boolean resourceExists(String name);
|
|
|
|
|
2009-12-24 17:58:48 -07:00
|
|
|
private static native Class resolveClass(ClassLoader loader, byte[] spec)
|
|
|
|
throws ClassNotFoundException;
|
|
|
|
|
2007-08-10 17:45:47 -06:00
|
|
|
protected URL findResource(String name) {
|
|
|
|
if (resourceExists(name)) {
|
|
|
|
try {
|
2007-08-19 13:45:51 -06:00
|
|
|
return new URL("resource:" + name);
|
2007-08-10 17:45:47 -06:00
|
|
|
} catch (MalformedURLException ignored) { }
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2009-09-19 16:21:15 -06:00
|
|
|
|
|
|
|
private static Class loadClass(ClassLoader loader,
|
|
|
|
byte[] nameBytes, int offset, int length)
|
|
|
|
{
|
2009-12-24 17:58:48 -07:00
|
|
|
byte[] spec = new byte[length + 1];
|
|
|
|
System.arraycopy(nameBytes, offset, spec, 0, length);
|
|
|
|
|
2009-09-19 16:21:15 -06:00
|
|
|
try {
|
2009-12-24 17:58:48 -07:00
|
|
|
Class c = resolveClass(loader, spec);
|
|
|
|
if (c == null) {
|
|
|
|
throw new NoClassDefFoundError();
|
|
|
|
}
|
|
|
|
return c;
|
2009-09-19 16:21:15 -06:00
|
|
|
} catch (ClassNotFoundException e) {
|
2009-12-24 17:58:48 -07:00
|
|
|
NoClassDefFoundError error = new NoClassDefFoundError
|
|
|
|
(new String(nameBytes, offset, length, false));
|
2009-09-19 16:21:15 -06:00
|
|
|
error.initCause(e);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object parseAnnotationValue(ClassLoader loader,
|
|
|
|
Object pool,
|
|
|
|
InputStream in)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
switch (read1(in)) {
|
|
|
|
case 'Z':
|
|
|
|
return Boolean.valueOf(Singleton.getInt(pool, read2(in) - 1) != 0);
|
|
|
|
|
|
|
|
case 'B':
|
|
|
|
return Byte.valueOf((byte) Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'C':
|
|
|
|
return Character.valueOf((char) Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'S':
|
|
|
|
return Short.valueOf((short) Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'I':
|
|
|
|
return Integer.valueOf(Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'F':
|
|
|
|
return Float.valueOf
|
|
|
|
(Float.intBitsToFloat(Singleton.getInt(pool, read2(in) - 1)));
|
|
|
|
|
|
|
|
case 'J': {
|
|
|
|
return Long.valueOf(Singleton.getLong(pool, read2(in) - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'D': {
|
|
|
|
return Double.valueOf
|
|
|
|
(Double.longBitsToDouble(Singleton.getLong(pool, read2(in) - 1)));
|
|
|
|
}
|
|
|
|
|
|
|
|
case 's': {
|
|
|
|
byte[] data = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
|
|
|
|
return new String(data, 0, data.length - 1, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'e': {
|
|
|
|
byte[] typeName = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
byte[] name = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
|
|
|
|
return Enum.valueOf
|
|
|
|
(loadClass(loader, typeName, 1, typeName.length - 3),
|
|
|
|
new String(name, 0, name.length - 1, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'c':{
|
|
|
|
byte[] name = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
|
|
|
|
return loadClass(loader, name, 1, name.length - 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
case '@':
|
|
|
|
return parseAnnotation(loader, pool, in);
|
|
|
|
|
|
|
|
case '[': {
|
|
|
|
Object[] array = new Object[read2(in)];
|
|
|
|
for (int i = 0; i < array.length; ++i) {
|
|
|
|
array[i] = parseAnnotationValue(loader, pool, in);
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: throw new AssertionError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object[] parseAnnotation(ClassLoader loader,
|
|
|
|
Object pool,
|
|
|
|
InputStream in)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
byte[] typeName = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
Object[] annotation = new Object[(read2(in) + 1) * 2];
|
|
|
|
annotation[1] = loadClass(loader, typeName, 1, typeName.length - 3);
|
|
|
|
|
|
|
|
for (int i = 2; i < annotation.length; i += 2) {
|
|
|
|
byte[] name = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
annotation[i] = new String(name, 0, name.length - 1, false);
|
|
|
|
annotation[i + 1] = parseAnnotationValue(loader, pool, in);
|
|
|
|
}
|
|
|
|
|
|
|
|
return annotation;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object[] parseAnnotationTable(ClassLoader loader,
|
|
|
|
Object pool,
|
|
|
|
InputStream in)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
Object[] table = new Object[read2(in)];
|
|
|
|
for (int i = 0; i < table.length; ++i) {
|
|
|
|
table[i] = parseAnnotation(loader, pool, in);
|
|
|
|
}
|
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void parseAnnotationTable(ClassLoader loader,
|
|
|
|
Addendum addendum)
|
|
|
|
{
|
2009-12-24 17:58:48 -07:00
|
|
|
if (addendum != null && addendum.annotationTable instanceof byte[]) {
|
2009-09-19 16:21:15 -06:00
|
|
|
try {
|
|
|
|
addendum.annotationTable = parseAnnotationTable
|
|
|
|
(loader, addendum.pool, new ByteArrayInputStream
|
|
|
|
((byte[]) addendum.annotationTable));
|
|
|
|
} catch (IOException e) {
|
|
|
|
AssertionError error = new AssertionError();
|
|
|
|
error.initCause(e);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
|
|
|
|
addendum.pool = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int resolveSpec(ClassLoader loader, byte[] spec, int start) {
|
|
|
|
int result;
|
|
|
|
int end;
|
|
|
|
switch (spec[start]) {
|
|
|
|
case 'L':
|
|
|
|
++ start;
|
|
|
|
end = start;
|
|
|
|
while (spec[end] != ';') ++ end;
|
|
|
|
result = end + 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '[':
|
|
|
|
end = start + 1;
|
|
|
|
while (spec[end] == '[') ++ end;
|
|
|
|
switch (spec[end]) {
|
|
|
|
case 'L':
|
|
|
|
++ end;
|
|
|
|
while (spec[end] != ';') ++ end;
|
|
|
|
++ end;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ end;
|
|
|
|
}
|
|
|
|
result = end;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return start + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
loadClass(loader, spec, start, end - start);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static native void acquireClassLock();
|
|
|
|
|
|
|
|
private static native void releaseClassLock();
|
|
|
|
|
|
|
|
public static void link(Class c, ClassLoader loader) {
|
|
|
|
acquireClassLock();
|
|
|
|
try {
|
|
|
|
if ((c.vmFlags & LinkFlag) == 0) {
|
|
|
|
if (c.super_ != null) {
|
|
|
|
link(c.super_, loader);
|
|
|
|
}
|
|
|
|
|
|
|
|
parseAnnotationTable(loader, c.addendum);
|
|
|
|
|
|
|
|
if (c.interfaceTable != null) {
|
|
|
|
int stride = (c.isInterface() ? 1 : 2);
|
|
|
|
for (int i = 0; i < c.interfaceTable.length; i += stride) {
|
|
|
|
link((Class) c.interfaceTable[i], loader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c.methodTable != null) {
|
|
|
|
for (int i = 0; i < c.methodTable.length; ++i) {
|
|
|
|
Method m = c.methodTable[i];
|
|
|
|
|
|
|
|
for (int j = 1; j < m.spec.length;) {
|
|
|
|
j = resolveSpec(loader, m.spec, j);
|
|
|
|
}
|
|
|
|
|
|
|
|
parseAnnotationTable(loader, m.addendum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c.fieldTable != null) {
|
|
|
|
for (int i = 0; i < c.fieldTable.length; ++i) {
|
|
|
|
Field f = c.fieldTable[i];
|
|
|
|
|
|
|
|
resolveSpec(loader, f.spec, 0);
|
|
|
|
|
|
|
|
parseAnnotationTable(loader, f.addendum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
c.vmFlags |= LinkFlag;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
releaseClassLock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void link(Class c) {
|
|
|
|
link(c, c.getClassLoader());
|
|
|
|
}
|
2007-07-30 17:19:05 -06:00
|
|
|
}
|