blob: 415ebe8f227580db879928799fa7c5c2e8d7032e [file] [log] [blame]
/*
* Copyright 2006 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package java.lang;
import com.google.gwt.core.client.JavaScriptObject;
import java.lang.reflect.Type;
/**
* Generally unsupported. This class is provided so that the GWT compiler can
* choke down class literal references.
* <p>
* NOTE: The code in this class is very sensitive and should keep its dependencies upon other
* classes to a minimum.
*
* @param <T> the type of the object
*/
public final class Class<T> implements Type {
private static final int PRIMITIVE = 0x00000001;
private static final int INTERFACE = 0x00000002;
private static final int ARRAY = 0x00000004;
private static final int ENUM = 0x00000008;
/**
* Create a Class object for an array.<p>
*
* Arrays are not registered in the prototype table and get the class literal explicitly at
* construction.<p>
*
* NOTE: this method is accessed through JSNI (Violator pattern) to avoid changing the public API.
*/
private static native <T> Class<T> getClassLiteralForArray(Class<?> leafClass, int dimensions) /*-{
var arrayLiterals =
leafClass.@java.lang.Class::arrayLiterals = leafClass.@java.lang.Class::arrayLiterals || [];
return arrayLiterals[dimensions] || (arrayLiterals[dimensions] =
leafClass.@java.lang.Class::createClassLiteralForArray(I)(dimensions));
}-*/;
private <T> Class<T> createClassLiteralForArray(int dimensions) {
Class<T> clazz = new Class<T>();
if (clazz.isClassMetadataEnabled()) {
if (this.isPrimitive()) {
// Primitives have an additional prepended space.
clazz.typeName = this.simpleName.substring(1);
} else {
clazz.typeName = "L" + this.typeName;
}
clazz.simpleName = this.simpleName;
for (int i = 0; i < dimensions; i++) {
clazz.typeName = "[" + clazz.typeName;
clazz.simpleName += "[]";
}
if (!this.isPrimitive()) {
clazz.typeName += ";";
}
} else {
// TODO(rluble): add meaninful ids to arrays when there is no metadata.
synthesizeClassNamesFromTypeId(clazz, null);
}
clazz.modifiers = ARRAY;
clazz.superclass = Object.class;
clazz.componentType = this;
return clazz;
}
/**
* Create a Class object for a class.
*
* @skip
*/
static <T> Class<T> createForClass(String packageName, String className,
JavaScriptObject typeId, Class<? super T> superclass) {
// Initialize here to avoid method inliner
Class<T> clazz = new Class<T>();
if (clazz.isClassMetadataEnabled()) {
initializeNames(clazz, packageName, className);
} else {
synthesizeClassNamesFromTypeId(clazz, typeId);
}
maybeSetClassLiteral(typeId, clazz);
clazz.superclass = superclass;
return clazz;
}
/**
* Create a Class object for an enum.
*
* @skip
*/
static <T> Class<T> createForEnum(String packageName, String className,
JavaScriptObject typeId, Class<? super T> superclass,
JavaScriptObject enumConstantsFunc, JavaScriptObject enumValueOfFunc) {
// Initialize here to avoid method inliner
Class<T> clazz = new Class<T>();
if (clazz.isClassMetadataEnabled()) {
initializeNames(clazz, packageName, className);
} else {
synthesizeClassNamesFromTypeId(clazz, typeId);
}
maybeSetClassLiteral(typeId, clazz);
clazz.modifiers = (enumConstantsFunc != null) ? ENUM : 0;
clazz.superclass = clazz.enumSuperclass = superclass;
clazz.enumConstantsFunc = enumConstantsFunc;
clazz.enumValueOfFunc = enumValueOfFunc;
return clazz;
}
/**
* Create a Class object for an interface.
*
* @skip
*/
static <T> Class<T> createForInterface(String packageName, String className) {
// Initialize here to avoid method inliner
Class<T> clazz = new Class<T>();
if (clazz.isClassMetadataEnabled()) {
initializeNames(clazz, packageName, className);
} else {
synthesizeClassNamesFromTypeId(clazz, null);
}
clazz.modifiers = INTERFACE;
return clazz;
}
/**
* Create a Class object for a primitive.
*
* @skip
*/
static Class<?> createForPrimitive(String packageName, String className,
String primitiveTypeId) {
// Initialize here to avoid method inliner
Class<?> clazz = new Class<Object>();
if (clazz.isClassMetadataEnabled()) {
clazz.typeName = className;
clazz.simpleName = primitiveTypeId;
} else {
synthesizePrimitiveNamesFromTypeId(clazz, primitiveTypeId);
}
clazz.modifiers = PRIMITIVE;
return clazz;
}
/**
* Used by {@link WebModePayloadSink} to create uninitialized instances.
*/
static native JavaScriptObject getPrototypeForClass(Class<?> clazz) /*-{
var typeId = clazz.@java.lang.Class::typeId;
var prototype = @com.google.gwt.lang.JavaClassHierarchySetupUtil::prototypesByTypeId[typeId];
// TODO(rluble): introduce pragma annotation to indicate that this function should not be
// inlined.
clazz = null; // HACK: prevent pruning via inlining by using param as lvalue
return prototype;
}-*/;
public static boolean isClassMetadataEnabled() {
// This body may be replaced by the compiler
return true;
}
/**
* null implies non-instantiable type, with no entries in
* {@link JavaClassHierarchySetupUtil::prototypesByTypeId}.
*/
static native boolean isInstantiable(JavaScriptObject typeId) /*-{
return !!typeId;
}-*/;
/**
* Install class literal into prototype.clazz field (if type is instantiable) such that
* Object.getClass() returning this.clazz returns the literal. Also stores typeId on class literal
* for looking up prototypes given a literal. This is used for deRPC at the moment, but may be
* used to implement Class.newInstance() in the future.
*
* If the prototype for typeId has not yet been created, then install the literal into a
* placeholder array to differentiate the two cases.
*/
static native void maybeSetClassLiteral(JavaScriptObject typeId, Class<?> clazz) /*-{
var proto;
if (!typeId) {
// Type is not instantiable, hence not registered in the metadata table.
return;
}
clazz.@java.lang.Class::typeId = typeId;
// Guarantees virtual method won't be pruned by using a JSNI ref
// This is required because deRPC needs to call it.
var prototype = @java.lang.Class::getPrototypeForClass(Ljava/lang/Class;)(clazz);
// A class literal may be referenced prior to an async-loaded vtable setup
// For example, class literal lives in inital fragment,
// but type is instantiated in another fragment
if (!prototype) {
// Leave a place holder for now to be filled in by __defineClass__ later.
// TODO(rluble): Do not rely on the fact that if the entry is an array it is a placeholder.
@com.google.gwt.lang.JavaClassHierarchySetupUtil::prototypesByTypeId[typeId] = [clazz];
return;
}
// Type already registered in the metadata table, install the class literal in the appropriate
// prototype field.
prototype.@java.lang.Object::___clazz = clazz;
}-*/;
/**
* Initiliazes {@code clazz} names from metadata.
* <p>
* Only called if metadata is NOT disabled.
* <p>
* Written in JSNI to minimize dependencies (on (String)+).
*/
private static native void initializeNames(Class<?> clazz, String packageName,
String className) /*-{
clazz.@java.lang.Class::typeName = packageName + className;
clazz.@java.lang.Class::simpleName = className;
}-*/;
/**
* Initiliazes {@code clazz} names from typeIds.
* <p>
* Only called if metadata IS disabled.
* <p>
* Written in JSNI to minimize dependencies (on toString() and (String)+).
*/
static native void synthesizeClassNamesFromTypeId(Class<?> clazz, JavaScriptObject typeId) /*-{
// The initial "" + in the below code is to prevent clazz.getAnonymousId from
// being autoboxed. The class literal creation code is run very early
// during application start up, before class Integer has been initialized.
clazz.@java.lang.Class::typeName = "Class$" +
(!!typeId ? "S" + typeId : "" + clazz.@java.lang.Class::sequentialId);
clazz.@java.lang.Class::simpleName = clazz.@java.lang.Class::typeName;
}-*/;
/**
* Sets the class object for primitives.
* <p>
* Written in JSNI to minimize dependencies (on (String)+).
*/
static native void synthesizePrimitiveNamesFromTypeId(Class<?> clazz, String primitiveTypeId) /*-{
clazz.@java.lang.Class::typeName = "Class$" + primitiveTypeId;
clazz.@java.lang.Class::simpleName = clazz.@java.lang.Class::typeName;
}-*/;
JavaScriptObject enumValueOfFunc;
int modifiers;
private Class<?> componentType;
@SuppressWarnings("unused")
private JavaScriptObject enumConstantsFunc;
private Class<? super T> enumSuperclass;
private Class<? super T> superclass;
private String simpleName;
private String typeName;
private JavaScriptObject typeId;
private JavaScriptObject arrayLiterals;
// Assign a sequential id to each class literal to avoid calling hashCode which bring Impl as
// a dependency.
private int sequentialId = nextSequentialId++;
private static int nextSequentialId = 1;
/**
* Not publicly instantiable.
*
* @skip
*/
private Class() {
}
public boolean desiredAssertionStatus() {
// This body is ignored by the JJS compiler and a new one is
// synthesized at compile-time based on the actual compilation arguments.
return false;
}
public Class<?> getComponentType() {
return componentType;
}
public native T[] getEnumConstants() /*-{
return this.@java.lang.Class::enumConstantsFunc
&& (this.@java.lang.Class::enumConstantsFunc)();
}-*/;
public String getName() {
return typeName;
}
public String getSimpleName() {
return simpleName;
}
public Class<? super T> getSuperclass() {
if (isClassMetadataEnabled()) {
return superclass;
} else {
return null;
}
}
public boolean isArray() {
return (modifiers & ARRAY) != 0;
}
public boolean isEnum() {
return (modifiers & ENUM) != 0;
}
public boolean isInterface() {
return (modifiers & INTERFACE) != 0;
}
public boolean isPrimitive() {
return (modifiers & PRIMITIVE) != 0;
}
public String toString() {
return (isInterface() ? "interface " : (isPrimitive() ? "" : "class "))
+ getName();
}
/**
* Used by Enum to allow getSuperclass() to be pruned.
*/
Class<? super T> getEnumSuperclass() {
return enumSuperclass;
}
}