FileDocCategorySizeDatePackage
ArrayClassInfo.javaAPI DocphoneME MR2 API (J2ME)6412Wed May 02 17:59:48 BST 2007vm

ArrayClassInfo.java

/*
 *   
 *
 * Copyright  1990-2007 Sun Microsystems, Inc. All Rights Reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER
 * 
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License version
 * 2 only, as published by the Free Software Foundation.
 * 
 * This program 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
 * General Public License version 2 for more details (a copy is
 * included at /legal/license.txt).
 * 
 * You should have received a copy of the GNU General Public License
 * version 2 along with this work; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA
 * 
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
 * Clara, CA 95054 or visit www.sun.com if you need additional
 * information or have any questions.
 */

package vm;
import components.*;
import util.*;
import jcc.Const;

/*
 * An array is a class.
 * It is a subclass of java.lang.Object.
 * It has all the class-related runtime data structures, or at least
 * may of them.
 * It is not read in from class files, but is made up on-the-fly
 * when the classresolver sees a reference to a classname starting with
 * "[".
 *
 * In order to resolve such references early, we must do likewise here.
 */
public
class ArrayClassInfo extends ClassInfo {

    private static int nFake = 0;
    public int    	 arrayClassNumber;
    public int    	 depth;
    public int    	 baseType;
    public String        baseName;
    public ClassConstant baseClass;
    public ClassConstant subarrayClass;
    

    /*
     * Given signature s,
     * fill in
     * depth ( i.e. number of opening [ )
     * basetype ( i.e. thing after the last [ )
     * and baseClass ( if basetype is a classtype )
     */
    private void fillInTypeInfo( String s ) throws DataFormatException {
	int index = 0;
	char c;
	while( ( c = s.charAt( index ) ) == Const.SIGC_ARRAY )
	    index++;
	depth = index;
	switch( c ){
	case Const.SIGC_INT:
	    baseType = Const.T_INT;
	    baseName = "Int";
	    break;
	case Const.SIGC_LONG:
	    baseType = Const.T_LONG;
	    baseName = "Long";
	    break;
	case Const.SIGC_FLOAT:
	    baseType = Const.T_FLOAT;
	    baseName = "Float";
	    break;
	case Const.SIGC_DOUBLE:
	    baseType = Const.T_DOUBLE;
	    baseName = "Double";
	    break;
	case Const.SIGC_BOOLEAN:
	    baseType = Const.T_BOOLEAN;
	    baseName = "Boolean";
	    break;
	case Const.SIGC_BYTE:
	    baseType = Const.T_BYTE;
	    baseName = "Byte";
	    break;
	case Const.SIGC_CHAR:
	    baseType = Const.T_CHAR;
	    baseName = "Char";
	    break;
	case Const.SIGC_SHORT:
	    baseType = Const.T_SHORT;
	    baseName = "Short";
	    break;
	case Const.SIGC_CLASS:
	    baseType = Const.T_CLASS;	    
	    int start = ++index;
	    while (s.charAt(index) != Const.SIGC_ENDCLASS) {
		index++;
	    }

	    baseClass = new ClassConstant(
		new UnicodeConstant( s.substring( start, index ) ) 
	    );
	    break;

	default:
	    throw new DataFormatException(Localizer.getString("arrayclassinfo.malformed_array_type_string.dataformatexception",s));
	}
	// For EVM, we want to know the sub-class, which is different from
	// the base class.
	if ( depth > 1 ){
	    subarrayClass = new ClassConstant( new UnicodeConstant( s.substring(1) ) );
	} else if ( (depth == 1 ) && ( baseClass != null ) ){
	    subarrayClass = baseClass;
	}

    }

    private void fakeConstantPool(){
	// 0 -- unused
	// 1 -- depth as integer
	// 2 -- base type tag as integer
	// 3 -- base class ref, if a ref type
	// 4 -- subarray class ref, if depth > 1
	//      base type if depth==1 && its a class ref array type
	// 5 -- self class ref
	// 6 -- UTF8 for entry 3
	// 7 -- UTF8 for entry 4
	// 8 -- UTF8 for entry 5

	// MAKE A CHANGE FOR KVM
	
	if (false) {
	    int nconstants = 9;
	    constants = new ConstantObject[nconstants];
	    constants[1] = new SingleValueConstant( depth );
	    constants[2] = new SingleValueConstant( baseType );
	    if ( baseClass == null ){
		constants[3] = new SingleValueConstant( 0 );
		constants[6] = new SingleValueConstant( 0 );
	    } else {
		constants[3] = baseClass;
		constants[6] = baseClass.name;
	    }
	    // for EVM, point at the subarray type.
	    if ( subarrayClass == null ){
		constants[4] = new SingleValueConstant( 0 );
		constants[7] = new SingleValueConstant( 0 );
	    } else {
		constants[4] = subarrayClass;
		constants[7] = subarrayClass.name;
	    }
	    constants[5] = thisClass; // EVM 4 => 5
	    constants[8] = thisClass.name; // EVM 5 => 7
	} else { 
	    int nconstants = 2;
	    constants = new ConstantObject[nconstants];
	    if (subarrayClass == null) { 
		constants[1] = new SingleValueConstant(baseType);
	    } else { 
		constants[1] = subarrayClass;
	    }
	}
    }

    public ArrayClassInfo( boolean v, String s ) throws DataFormatException {
	super(v);
	arrayClassNumber = nFake++;
	fillInTypeInfo( s );
	className = s;
	thisClass = new ClassConstant( new UnicodeConstant( s ) );
	superClassInfo = lookupClass(/*NOI18N*/"java/lang/Object");
	superClass = superClassInfo.thisClass;
	access = Const.ACC_FINAL|Const.ACC_ABSTRACT|Const.ACC_PUBLIC;
	fakeConstantPool();
	methods = new MethodInfo[0];
	fields  = new FieldInfo[0];
	enterClass( s );
    }

    public boolean
    countReferences( boolean v ){
	// some entries are magic. The others
	// can get shared or whacked.
	if (false) { 
	    constants[1].incReference();	// depth
	    constants[2].incReference();	// baseType code
	    constants[3].incReference();	// base class pointer
	    constants[4].incReference();	// subarray class pointer
	} else { 
	    constants[1].incReference();	// base type 
	}
	return super.countReferences( v );
    }

    public void
    externalize( ConstantPool p ){
	if (verbose){
	    log.println(Localizer.getString("arrayclassinfo.externalizing_class", className));
	}
	externalizeConstants( p );
	// DONT DO THISthisClass = (ClassConstant)p.dup( thisClass );
	if ( superClass != null ){
	    superClass = (ClassConstant)p.dup( superClass );
	    // THIS SHOULD WORK.
	}
    }

    protected String createGenericNativeName() { 
        return /*NOI18N*/"fakeArray" + arrayClassNumber;
    }
}