/*
* $Id$
*
* Copyright (C) 2003-2015 JNode.org
*
* This library is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This library is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
* License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library; If not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package org.jnode.vm.classmgr;
import java.security.ProtectionDomain;
/**
* @author epr
*/
public class VmNormalClass<T> extends VmClassType<T> {
/**
* The offsets of all reference variables in this class
*/
private int[] referenceOffsets;
/**
* The size (in bytes) of an instance of this class
*/
private int objectSize = 0;
/**
* @param name
* @param superClass
* @param loader
* @param typeSize
*/
protected VmNormalClass(String name, VmNormalClass<? super T> superClass, VmClassLoader loader, int typeSize,
ProtectionDomain protectionDomain) {
super(name, superClass, loader, typeSize, protectionDomain);
testClassType();
}
/**
* @param name
* @param superClassName
* @param loader
* @param accessFlags
*/
public VmNormalClass(String name, String superClassName, VmClassLoader loader, int accessFlags,
ProtectionDomain protectionDomain) {
super(name, superClassName, loader, accessFlags, protectionDomain);
testClassType();
}
/**
* Return the size in bytes of instantiations of this class
*
* @return The object size
*/
public final int getObjectSize() {
if (!isPrepared()) {
throw new IllegalStateException("Not initialized yet");
}
return objectSize;
}
/**
* Sets the objectSize.
*
* @param objectSize The objectSize to set
*/
protected final void setObjectSize(int objectSize) {
if (this.objectSize == 0) {
this.objectSize = objectSize;
} else {
throw new IllegalArgumentException("Cannot overwrite object size");
}
}
/**
* Gets the offsets within an instance of this class of all reference non-static member variables.
*
* @return The reference offsets
*/
public final int[] getReferenceOffsets() {
if (!isPrepared()) {
throw new IllegalStateException("Not initialized yet");
}
return referenceOffsets;
}
/**
* Do the prepare action required to instantiate this object
*/
protected void prepareForInstantiation() {
// Step 3: Calculate the object size
final VmNormalClass superCls = getSuperClass();
int sc_size = (superCls != null) ? superCls.getObjectSize() : 0;
objectSize += sc_size;
//System.out.println(getName() + " objsz:" + objectSize + " sc_size:" + sc_size);
// Step 4a: Fix the offset for all declared non-static fields
final int cnt = getNoDeclaredFields();
final int[] superRefOffsets = (superCls != null) ? superCls.getReferenceOffsets() : null;
int refOffsetsSize = (superCls != null) ? superRefOffsets.length : 0;
int startRefIdx = refOffsetsSize;
for (int i = 0; i < cnt; i++) {
final VmField field = getDeclaredField(i);
//fs.resolve(loader);
if (!field.isStatic()) {
final VmInstanceField inf = (VmInstanceField) field;
inf.resolveOffset(sc_size);
if (!field.isPrimitive()) {
if (!field.isAddressType()) {
refOffsetsSize++;
} else {
//System.out.println("Found address in field " + fs.getName());
}
}
}
}
// Step 4b: Create the referenceOffsets field
referenceOffsets = new int[refOffsetsSize];
if (superCls != null) {
System.arraycopy(superRefOffsets, 0, referenceOffsets, 0, startRefIdx);
}
for (int i = 0; i < cnt; i++) {
final VmField field = getDeclaredField(i);
if (!field.isStatic()) {
final VmInstanceField inf = (VmInstanceField) field;
if (!field.isPrimitive()) {
if (!field.isAddressType()) {
referenceOffsets[startRefIdx++] = inf.getOffset();
}
}
final int off = inf.getOffset();
if (off + inf.getTypeSize() > objectSize) {
throw new Error("Invalid offset in class " + getName() + " ofs " + off + " size " + objectSize);
}
}
}
}
/**
* Test if this class is using the right modifiers
*
* @throws RuntimeException
*/
private final void testClassType() throws RuntimeException {
if (isArray()) {
throw new RuntimeException("Not a normal class (array-class)");
}
if (isInterface()) {
throw new RuntimeException("Not a normal class (interface-class)");
}
}
}