corda/classpath/java/lang/Class.java

792 lines
22 KiB
Java
Raw Normal View History

2015-03-13 12:52:59 -06:00
/* Copyright (c) 2008-2015, Avian Contributors
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. */
package java.lang;
import avian.VMClass;
import avian.ClassAddendum;
import avian.AnnotationInvocationHandler;
import avian.SystemClassLoader;
import avian.Classes;
import avian.InnerClassReference;
2007-07-26 20:39:53 -06:00
import java.lang.reflect.Constructor;
import java.lang.reflect.GenericDeclaration;
2007-07-23 19:44:20 -06:00
import java.lang.reflect.Method;
import java.lang.reflect.Field;
2007-07-27 17:56:19 -06:00
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.lang.reflect.Proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.SignatureParser;
import java.lang.annotation.Annotation;
import java.io.InputStream;
2007-09-13 21:59:39 -06:00
import java.io.IOException;
import java.net.URL;
import java.util.Arrays;
2014-10-07 16:50:02 +04:00
import java.util.List;
import java.util.ArrayList;
import java.util.LinkedList;
2009-08-04 18:31:14 -06:00
import java.util.Map;
import java.util.HashMap;
import java.security.ProtectionDomain;
import java.security.Permissions;
import java.security.AllPermission;
2007-07-23 19:44:20 -06:00
public final class Class <T>
implements Type, AnnotatedElement, GenericDeclaration
{
private static final int PrimitiveFlag = 1 << 5;
private static final int EnumFlag = 1 << 14;
public final VMClass vmClass;
public Class(VMClass vmClass) {
this.vmClass = vmClass;
}
2007-08-19 20:57:32 -06:00
public String toString() {
2014-10-07 16:50:02 +04:00
String res;
if (isInterface()) res = "interface ";
else if (isAnnotation()) res = "annotation ";
else res = "class ";
return res + getName();
2007-08-19 20:57:32 -06:00
}
private static byte[] replace(int a, int b, byte[] s, int offset,
int length)
{
byte[] array = new byte[length];
for (int i = 0; i < length; ++i) {
byte c = s[i];
array[i] = (byte) (c == a ? b : c);
}
return array;
}
public String getName() {
return getName(vmClass);
}
public static String getName(VMClass c) {
if (c.name == null) {
if ((c.vmFlags & PrimitiveFlag) != 0) {
if (c == Classes.primitiveClass('V')) {
c.name = "void\0".getBytes();
} else if (c == Classes.primitiveClass('Z')) {
c.name = "boolean\0".getBytes();
} else if (c == Classes.primitiveClass('B')) {
c.name = "byte\0".getBytes();
} else if (c == Classes.primitiveClass('C')) {
c.name = "char\0".getBytes();
} else if (c == Classes.primitiveClass('S')) {
c.name = "short\0".getBytes();
} else if (c == Classes.primitiveClass('I')) {
c.name = "int\0".getBytes();
} else if (c == Classes.primitiveClass('F')) {
c.name = "float\0".getBytes();
} else if (c == Classes.primitiveClass('J')) {
c.name = "long\0".getBytes();
} else if (c == Classes.primitiveClass('D')) {
c.name = "double\0".getBytes();
} else {
throw new AssertionError();
}
} else {
throw new AssertionError();
}
}
return Classes.makeString
(replace('/', '.', c.name, 0, c.name.length - 1), 0, c.name.length - 1);
}
2007-07-23 19:44:20 -06:00
public String getCanonicalName() {
if ((vmClass.vmFlags & PrimitiveFlag) != 0) {
return getName();
} else if (isArray()) {
return getComponentType().getCanonicalName() + "[]";
} else {
return getName().replace('$', '.');
}
}
public String getSimpleName() {
if ((vmClass.vmFlags & PrimitiveFlag) != 0) {
return getName();
} else if (isArray()) {
return getComponentType().getSimpleName() + "[]";
} else {
String name = getCanonicalName();
int index = name.lastIndexOf('.');
if (index >= 0) {
return name.substring(index + 1);
} else {
return name;
}
}
}
public T newInstance()
throws IllegalAccessException, InstantiationException
{
try {
return (T) getConstructor().newInstance();
} catch (NoSuchMethodException e) {
throw new RuntimeException(e);
} catch (InvocationTargetException e) {
throw new RuntimeException(e);
}
}
2007-07-30 17:19:05 -06:00
public static Class forName(String name) throws ClassNotFoundException {
return forName(name, true, Method.getCaller().class_.loader);
2007-07-30 17:19:05 -06:00
}
public static Class forName(String name, boolean initialize,
ClassLoader loader)
throws ClassNotFoundException
{
2013-02-21 15:37:17 -07:00
return Classes.forName(name, initialize, loader);
}
2007-07-23 19:44:20 -06:00
public Class getComponentType() {
if (isArray()) {
String n = getName();
if ("[Z".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('Z'));
} else if ("[B".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('B'));
} else if ("[S".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('S'));
} else if ("[C".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('C'));
} else if ("[I".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('I'));
} else if ("[F".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('F'));
} else if ("[J".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('J'));
} else if ("[D".equals(n)) {
return SystemClassLoader.getClass(Classes.primitiveClass('D'));
}
2014-06-27 18:32:20 -06:00
if (vmClass.arrayElementClass == null) throw new AssertionError();
return SystemClassLoader.getClass((VMClass) vmClass.arrayElementClass);
} else {
return null;
}
}
public boolean isAssignableFrom(Class c) {
return Classes.isAssignableFrom(vmClass, c.vmClass);
}
2007-07-23 19:44:20 -06:00
2007-07-26 20:39:53 -06:00
public Field getDeclaredField(String name) throws NoSuchFieldException {
2013-02-21 15:37:17 -07:00
int index = Classes.findField(vmClass, name);
if (index < 0) {
2007-07-26 20:39:53 -06:00
throw new NoSuchFieldException(name);
} else {
2013-02-21 15:37:17 -07:00
return new Field(vmClass.fieldTable[index]);
2007-07-26 20:39:53 -06:00
}
}
public Field getField(String name) throws NoSuchFieldException {
for (VMClass c = vmClass; c != null; c = c.super_) {
2013-02-22 11:55:01 -07:00
int index = Classes.findField(c, name);
2013-02-21 15:37:17 -07:00
if (index >= 0) {
return new Field(vmClass.fieldTable[index]);
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
throw new NoSuchFieldException(name);
}
2007-07-26 20:39:53 -06:00
public Method getDeclaredMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
if (name.startsWith("<")) {
throw new NoSuchMethodException(name);
}
2013-02-21 15:37:17 -07:00
int index = Classes.findMethod(vmClass, name, parameterTypes);
if (index < 0) {
2007-07-26 20:39:53 -06:00
throw new NoSuchMethodException(name);
} else {
2013-02-21 15:37:17 -07:00
return new Method(vmClass.methodTable[index]);
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
2007-07-26 20:39:53 -06:00
public Method getMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
if (name.startsWith("<")) {
throw new NoSuchMethodException(name);
}
for (VMClass c = vmClass; c != null; c = c.super_) {
2013-02-22 11:55:01 -07:00
int index = Classes.findMethod(c, name, parameterTypes);
2013-02-21 15:37:17 -07:00
if (index >= 0) {
return new Method(c.methodTable[index]);
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
throw new NoSuchMethodException(name);
}
2007-07-26 20:39:53 -06:00
public Constructor getConstructor(Class ... parameterTypes)
throws NoSuchMethodException
{
2013-02-22 11:55:01 -07:00
int index = Classes.findMethod(vmClass, "<init>", parameterTypes);
if (index < 0) {
2007-07-27 17:56:19 -06:00
throw new NoSuchMethodException();
} else {
2013-02-22 11:55:01 -07:00
return new Constructor(new Method(vmClass.methodTable[index]));
2007-07-27 17:56:19 -06:00
}
2007-07-26 20:39:53 -06:00
}
public Constructor getDeclaredConstructor(Class ... parameterTypes)
throws NoSuchMethodException
{
Constructor c = null;
Constructor[] constructors = getDeclaredConstructors();
for (int i = 0; i < constructors.length; ++i) {
2013-02-22 11:55:01 -07:00
if (Classes.match(parameterTypes, constructors[i].getParameterTypes())) {
c = constructors[i];
}
}
if (c == null) {
throw new NoSuchMethodException();
} else {
return c;
}
}
2007-07-27 17:56:19 -06:00
private int countConstructors(boolean publicOnly) {
2007-07-26 20:39:53 -06:00
int count = 0;
if (vmClass.methodTable != null) {
for (int i = 0; i < vmClass.methodTable.length; ++i) {
2007-07-28 10:10:13 -06:00
if (((! publicOnly)
|| ((vmClass.methodTable[i].flags & Modifier.PUBLIC))
!= 0)
&& Method.getName(vmClass.methodTable[i]).equals("<init>"))
2007-07-28 10:10:13 -06:00
{
++ count;
}
2007-07-26 20:39:53 -06:00
}
}
2007-07-27 17:56:19 -06:00
return count;
}
2007-07-26 20:39:53 -06:00
2007-07-27 17:56:19 -06:00
public Constructor[] getDeclaredConstructors() {
Constructor[] array = new Constructor[countConstructors(false)];
if (vmClass.methodTable != null) {
Classes.link(vmClass);
2007-07-28 10:10:13 -06:00
int index = 0;
for (int i = 0; i < vmClass.methodTable.length; ++i) {
if (Method.getName(vmClass.methodTable[i]).equals("<init>")) {
array[index++] = new Constructor(new Method(vmClass.methodTable[i]));
2007-07-28 10:10:13 -06:00
}
2007-07-26 20:39:53 -06:00
}
}
return array;
}
2007-07-27 17:56:19 -06:00
public Constructor[] getConstructors() {
Constructor[] array = new Constructor[countConstructors(true)];
if (vmClass.methodTable != null) {
Classes.link(vmClass);
2007-07-28 10:10:13 -06:00
int index = 0;
for (int i = 0; i < vmClass.methodTable.length; ++i) {
if (((vmClass.methodTable[i].flags & Modifier.PUBLIC) != 0)
&& Method.getName(vmClass.methodTable[i]).equals("<init>"))
2007-07-28 10:10:13 -06:00
{
array[index++] = new Constructor(new Method(vmClass.methodTable[i]));
2007-07-28 10:10:13 -06:00
}
2007-07-27 17:56:19 -06:00
}
}
return array;
2007-07-26 20:39:53 -06:00
}
public Field[] getDeclaredFields() {
if (vmClass.fieldTable != null) {
Field[] array = new Field[vmClass.fieldTable.length];
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
array[i] = new Field(vmClass.fieldTable[i]);
}
2007-07-28 10:10:13 -06:00
return array;
} else {
return new Field[0];
}
2007-07-26 20:39:53 -06:00
}
2007-07-27 17:56:19 -06:00
private int countPublicFields() {
int count = 0;
if (vmClass.fieldTable != null) {
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
if (((vmClass.fieldTable[i].flags & Modifier.PUBLIC)) != 0) {
2007-07-28 10:10:13 -06:00
++ count;
}
2007-07-27 17:56:19 -06:00
}
}
return count;
}
public Field[] getFields() {
Field[] array = new Field[countPublicFields()];
if (vmClass.fieldTable != null) {
Classes.link(vmClass);
int ai = 0;
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
if (((vmClass.fieldTable[i].flags & Modifier.PUBLIC)) != 0) {
array[ai++] = new Field(vmClass.fieldTable[i]);
2007-07-28 10:10:13 -06:00
}
2007-07-27 17:56:19 -06:00
}
}
return array;
}
private static void getAllFields(VMClass vmClass, ArrayList<Field> fields) {
if (vmClass.super_ != null) {
getAllFields(vmClass.super_, fields);
}
if (vmClass.fieldTable != null) {
Classes.link(vmClass);
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
fields.add(new Field(vmClass.fieldTable[i]));
}
}
}
public Field[] getAllFields() {
ArrayList<Field> fields = new ArrayList<Field>();
getAllFields(vmClass, fields);
return fields.toArray(new Field[fields.size()]);
}
2007-07-27 17:56:19 -06:00
public Method[] getDeclaredMethods() {
return Classes.getMethods(vmClass, false);
2007-07-27 17:56:19 -06:00
}
public Method[] getMethods() {
return Classes.getMethods(vmClass, true);
2007-07-26 20:39:53 -06:00
}
public Class[] getInterfaces() {
ClassAddendum addendum = vmClass.addendum;
if (addendum != null) {
Object[] table = addendum.interfaceTable;
if (table != null) {
Class[] array = new Class[table.length];
for (int i = 0; i < table.length; ++i) {
array[i] = SystemClassLoader.getClass((VMClass) table[i]);
}
return array;
2007-07-28 10:10:13 -06:00
}
2007-07-26 20:39:53 -06:00
}
return new Class[0];
2007-07-26 20:39:53 -06:00
}
public native Class getEnclosingClass();
public native Method getEnclosingMethod();
public native Constructor getEnclosingConstructor();
2007-11-17 11:39:29 -07:00
public T[] getEnumConstants() {
if (Enum.class.isAssignableFrom(this)) {
2007-11-17 11:39:29 -07:00
try {
return (T[]) getMethod("values").invoke(null);
} catch (Exception e) {
throw new Error();
}
} else {
return null;
}
}
public Class[] getDeclaredClasses() {
ClassAddendum addendum = vmClass.addendum;
if (addendum != null) {
InnerClassReference[] table = addendum.innerClassTable;
if (table != null) {
int count = 0;
for (int i = 0; i < table.length; ++i) {
InnerClassReference reference = table[i];
if (reference.outer != null
&& Arrays.equals(vmClass.name, reference.outer))
{
++ count;
}
}
Class[] result = new Class[count];
for (int i = 0; i < table.length; ++i) {
InnerClassReference reference = table[i];
if (reference.outer != null
&& Arrays.equals(vmClass.name, reference.outer))
{
try {
result[--count] = getClassLoader().loadClass
(new String(reference.inner, 0, reference.inner.length - 1));
} catch (ClassNotFoundException e) {
throw new Error(e);
}
}
}
return result;
}
}
return new Class[0];
}
public Class getDeclaringClass() {
ClassAddendum addendum = vmClass.addendum;
if (addendum != null) {
InnerClassReference[] table = addendum.innerClassTable;
if (table != null) {
for (int i = 0; i < table.length; ++i) {
InnerClassReference reference = table[i];
if (Arrays.equals(vmClass.name, reference.inner)) {
if (reference.outer != null) {
try {
return getClassLoader().loadClass
(new String(reference.outer, 0, reference.outer.length - 1));
} catch (ClassNotFoundException e) {
throw new Error(e);
}
} else {
return null;
}
}
}
}
}
return null;
}
2007-07-26 20:39:53 -06:00
public ClassLoader getClassLoader() {
return vmClass.loader;
2007-07-26 20:39:53 -06:00
}
public int getModifiers() {
ClassAddendum addendum = vmClass.addendum;
if (addendum != null) {
InnerClassReference[] table = addendum.innerClassTable;
if (table != null) {
for (int i = 0; i < table.length; ++i) {
InnerClassReference reference = table[i];
if (Arrays.equals(vmClass.name, reference.inner)) {
return reference.flags;
}
}
}
}
return vmClass.flags;
2007-07-26 20:39:53 -06:00
}
2008-04-01 16:48:58 -06:00
public boolean isInterface() {
return (vmClass.flags & Modifier.INTERFACE) != 0;
2008-04-01 16:48:58 -06:00
}
public boolean isAnnotation() {
return (vmClass.flags & 0x2000) != 0;
}
2007-07-26 20:39:53 -06:00
public Class getSuperclass() {
return (vmClass.super_ == null ? null : SystemClassLoader.getClass(vmClass.super_));
2007-07-26 20:39:53 -06:00
}
private enum ClassType { GLOBAL, MEMBER, LOCAL, ANONYMOUS }
/**
* Determines the class type.
*
* There are four class types: global (no dollar sign), anonymous (only digits after the dollar sign),
* local (starts with digits after the dollar, ends in class name) and member (does not start with digits
* after the dollar sign).
*
* @return the class type
*/
private ClassType getClassType() {
final String name = getName();
// Find the last dollar, as classes can be nested
int dollar = name.lastIndexOf('$');
if (dollar < 0) return ClassType.GLOBAL;
// Find the first non-digit after the dollar, if any
final char[] chars = name.toCharArray();
int skipDigits;
for (skipDigits = dollar + 1; skipDigits < chars.length; skipDigits++) {
if (chars[skipDigits] < '0' || chars[skipDigits] > '9') break;
}
if (skipDigits == chars.length) {
return ClassType.ANONYMOUS;
} else if (skipDigits == dollar + 1) {
return ClassType.MEMBER;
} else {
return ClassType.LOCAL;
}
}
public boolean isAnonymousClass () {
return getClassType() == ClassType.ANONYMOUS;
}
public boolean isLocalClass () {
return getClassType() == ClassType.LOCAL;
}
public boolean isMemberClass () {
return getClassType() == ClassType.MEMBER;
}
2007-07-26 20:39:53 -06:00
public boolean isArray() {
return vmClass.arrayDimensions != 0;
}
public static boolean isInstance(VMClass c, Object o) {
return o != null && Classes.isAssignableFrom
(c, Classes.getVMClass(o));
2007-07-26 20:39:53 -06:00
}
public boolean isInstance(Object o) {
return isInstance(vmClass, o);
2007-07-26 20:39:53 -06:00
}
public boolean isPrimitive() {
return (vmClass.vmFlags & PrimitiveFlag) != 0;
2007-07-26 20:39:53 -06:00
}
public boolean isEnum() {
return getSuperclass() == Enum.class && (vmClass.flags & EnumFlag) != 0;
}
2007-09-13 21:59:39 -06:00
public URL getResource(String path) {
if (path.startsWith("/")) {
path = path.substring(1);
} else {
String name = Classes.makeString
(vmClass.name, 0, vmClass.name.length - 1);
2007-09-13 21:59:39 -06:00
int index = name.lastIndexOf('/');
if (index >= 0) {
path = name.substring(0, index) + "/" + path;
}
}
return getClassLoader().getResource(path);
}
public InputStream getResourceAsStream(String path) {
URL url = getResource(path);
try {
return (url == null ? null : url.openStream());
} catch (IOException e) {
return null;
}
}
public boolean desiredAssertionStatus() {
return false;
}
public <T> Class<? extends T> asSubclass(Class<T> c) {
if (! c.isAssignableFrom(this)) {
throw new ClassCastException();
}
return (Class<? extends T>) this;
}
public T cast(Object o) {
return (T) o;
}
public Package getPackage() {
if ((vmClass.vmFlags & PrimitiveFlag) != 0 || isArray()) {
return null;
} else {
String name = getCanonicalName();
int index = name.lastIndexOf('.');
if (index >= 0) {
return getClassLoader().getPackage(name.substring(0, index));
} else {
return null;
}
}
}
public boolean isAnnotationPresent
(Class<? extends Annotation> class_)
{
return getAnnotation(class_) != null;
}
private static Annotation getAnnotation(VMClass c, Object[] a) {
if (a[0] == null) {
a[0] = Proxy.newProxyInstance
(c.loader, new Class[] { (Class) a[1] },
new AnnotationInvocationHandler(a));
}
return (Annotation) a[0];
}
public <T extends Annotation> T getAnnotation(Class<T> class_) {
for (VMClass c = vmClass; c != null; c = c.super_) {
if (c.addendum != null && c.addendum.annotationTable != null) {
Classes.link(c, c.loader);
Object[] table = (Object[]) c.addendum.annotationTable;
for (int i = 0; i < table.length; ++i) {
Object[] a = (Object[]) table[i];
if (a[1] == class_) {
return (T) getAnnotation(c, a);
}
}
}
}
return null;
}
public Annotation[] getDeclaredAnnotations() {
if (vmClass.addendum.annotationTable != null) {
Classes.link(vmClass);
Object[] table = (Object[]) vmClass.addendum.annotationTable;
Annotation[] array = new Annotation[table.length];
for (int i = 0; i < table.length; ++i) {
array[i] = getAnnotation(vmClass, (Object[]) table[i]);
}
return array;
} else {
return new Annotation[0];
}
}
private int countAnnotations() {
int count = 0;
for (VMClass c = vmClass; c != null; c = c.super_) {
if (c.addendum != null && c.addendum.annotationTable != null) {
count += ((Object[]) c.addendum.annotationTable).length;
}
}
return count;
}
public Annotation[] getAnnotations() {
Annotation[] array = new Annotation[countAnnotations()];
int i = 0;
for (VMClass c = vmClass; c != null; c = c.super_) {
if (c.addendum != null && c.addendum.annotationTable != null) {
Object[] table = (Object[]) c.addendum.annotationTable;
for (int j = 0; j < table.length; ++j) {
array[i++] = getAnnotation(vmClass, (Object[]) table[j]);
}
}
}
return array;
}
public ProtectionDomain getProtectionDomain() {
return Classes.getProtectionDomain(vmClass);
}
public TypeVariable<?>[] getTypeParameters() {
throw new UnsupportedOperationException("not yet implemented");
}
/**
* The first one is the superclass, the others are interfaces
**/
private String[] getGenericTypeSignatures() {
String signature = Classes.toString((byte[]) vmClass.addendum.signature);
final char[] signChars = signature.toCharArray();
// Addendum format:
// <generic args if present>LBaseClass;LIface1;LIface2;...
// We should split it
int i = -1;
// Passing the generic args
int angles = 0;
do {
i++;
if (signChars[i] == '<') angles ++;
else if (signChars[i] == '>') angles --;
} while (angles > 0);
if (signChars[i] == '>') i++;
// Splitting types list
LinkedList<String> typeSigns = new LinkedList<String>();
StringBuilder curTypeSign = new StringBuilder();
for (; i < signChars.length; i++) {
// Counting braces
if (signChars[i] == '<') angles ++;
else if (signChars[i] == '>') angles --;
// Appending character
curTypeSign.append(signChars[i]);
// Splitting
if (angles == 0 && signChars[i] == ';') {
typeSigns.add(curTypeSign.toString());
curTypeSign.setLength(0);
}
}
if (curTypeSign.length() > 0) typeSigns.add(curTypeSign.toString());
String[] res = new String[typeSigns.size()];
return typeSigns.toArray(res);
}
public Type[] getGenericInterfaces() {
if (vmClass.addendum == null || vmClass.addendum.signature == null) {
return getInterfaces();
}
String[] typeSigns = getGenericTypeSignatures();
if (typeSigns.length < 1) {
throw new RuntimeException("Class signature doesn't contain any type");
}
// Parsing types
Type[] res = new Type[typeSigns.length - 1];
for (int i = 0; i < typeSigns.length - 1; i++) {
res[i] = SignatureParser.parse(vmClass.loader, typeSigns[i + 1], this);
}
return res;
}
public Type getGenericSuperclass() {
if (vmClass.addendum == null || vmClass.addendum.signature == null) {
return getSuperclass();
}
String[] typeSigns = getGenericTypeSignatures();
if (typeSigns.length < 1) {
throw new RuntimeException("Class signature doesn't contain any type");
}
return SignatureParser.parse(vmClass.loader, typeSigns[0], this);
}
}