mirror of
https://github.com/i-net-software/JWebAssembly.git
synced 2025-04-03 04:25:46 +02:00
172 lines
6.5 KiB
Java
172 lines
6.5 KiB
Java
/*
|
|
Copyright 2011 - 2020 Volker Berlin (i-net software)
|
|
|
|
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 de.inetsoftware.classparser;
|
|
|
|
import java.io.DataInputStream;
|
|
import java.io.IOException;
|
|
|
|
/**
|
|
* @author Volker Berlin
|
|
*/
|
|
public class ConstantPool {
|
|
|
|
private final Object[] constantPool;
|
|
|
|
/**
|
|
* https://docs.oracle.com/javase/specs/jvms/se9/html/jvms-4.html#jvms-4.4
|
|
*
|
|
* @param input
|
|
* the stream of the class
|
|
* @throws IOException
|
|
* if any IO error occur
|
|
*/
|
|
ConstantPool( DataInputStream input ) throws IOException {
|
|
int count = input.readUnsignedShort();
|
|
Object[] pool = constantPool = new Object[count];
|
|
for( int i = 1; i < count; i++ ) {
|
|
byte type = input.readByte();
|
|
switch( type ) {
|
|
case 1: //CONSTANT_Utf8
|
|
pool[i] = input.readUTF();
|
|
break;
|
|
case 3: //CONSTANT_Integer
|
|
pool[i] = new Integer( input.readInt() );
|
|
break;
|
|
case 4: //CONSTANT_Float
|
|
pool[i] = new Float( input.readFloat() );
|
|
break;
|
|
case 5: //CONSTANT_Long
|
|
pool[i] = new Long( input.readLong() );
|
|
i++;
|
|
break;
|
|
case 6: //CONSTANT_Double
|
|
pool[i] = new Double( input.readDouble() );
|
|
i++;
|
|
break;
|
|
case 7: //CONSTANT_Class
|
|
case 8: //CONSTANT_String
|
|
case 16: // CONSTANT_MethodType
|
|
case 19: // CONSTANT_Module
|
|
case 20: // CONSTANT_Package
|
|
pool[i] = new int[] { type, input.readUnsignedShort() };
|
|
break;
|
|
case 9: //CONSTANT_Fieldref
|
|
case 10: //CONSTANT_Methodref
|
|
case 11: //CONSTANT_InterfaceMethodref
|
|
case 12: //CONSTANT_NameAndType
|
|
case 18: // CONSTANT_InvokeDynamic
|
|
pool[i] = new int[] { type, input.readUnsignedShort(), input.readUnsignedShort() };
|
|
break;
|
|
case 15: // CONSTANT_MethodHandle
|
|
pool[i] = new int[] { type, input.readUnsignedByte(), input.readUnsignedShort() };
|
|
break;
|
|
default:
|
|
throw new IOException( "Unknown constant pool type: " + type );
|
|
}
|
|
}
|
|
|
|
boolean repeat;
|
|
do {
|
|
repeat = false;
|
|
for( int i = 0; i < count; i++ ) {
|
|
if( pool[i] instanceof int[] ) {
|
|
int[] data = (int[])pool[i];
|
|
switch( data[0] ) {
|
|
case 7: //CONSTANT_Class
|
|
pool[i] = new ConstantClass( (String)pool[data[1]] );
|
|
break;
|
|
case 8: // CONSTANT_String
|
|
case 16: // CONSTANT_MethodType
|
|
case 19: // CONSTANT_Module
|
|
case 20: // CONSTANT_Package
|
|
pool[i] = pool[data[1]];
|
|
break;
|
|
case 9: //CONSTANT_Fieldref
|
|
if( pool[data[1]] instanceof int[] || pool[data[2]] instanceof int[] ) {
|
|
repeat = true;
|
|
} else {
|
|
pool[i] = new ConstantFieldRef( (ConstantClass)pool[data[1]], (ConstantNameAndType)pool[data[2]] );
|
|
}
|
|
break;
|
|
case 10: //CONSTANT_Methodref
|
|
if( pool[data[1]] instanceof int[] || pool[data[2]] instanceof int[] ) {
|
|
repeat = true;
|
|
} else {
|
|
pool[i] = new ConstantMethodRef( (ConstantClass)pool[data[1]], (ConstantNameAndType)pool[data[2]] );
|
|
}
|
|
break;
|
|
case 11: //CONSTANT_InterfaceMethodref
|
|
if( pool[data[1]] instanceof int[] || pool[data[2]] instanceof int[] ) {
|
|
repeat = true;
|
|
} else {
|
|
pool[i] = new ConstantInterfaceRef( (ConstantClass)pool[data[1]], (ConstantNameAndType)pool[data[2]] );
|
|
}
|
|
break;
|
|
case 12: //CONSTANT_NameAndType
|
|
pool[i] = new ConstantNameAndType( (String)pool[data[1]], (String)pool[data[2]] );
|
|
break;
|
|
case 15: // CONSTANT_MethodHandle
|
|
pool[i] = pool[data[2]];
|
|
break;
|
|
case 18: // CONSTANT_InvokeDynamic
|
|
if( pool[data[2]] instanceof int[] ) {
|
|
repeat = true;
|
|
} else {
|
|
pool[i] = new ConstantInvokeDynamic( data[1], (ConstantNameAndType)pool[data[2]] );
|
|
}
|
|
break;
|
|
default:
|
|
throw new IOException( "Unknown constant pool type: " + data[0] );
|
|
}
|
|
}
|
|
}
|
|
} while( repeat );
|
|
}
|
|
|
|
/**
|
|
* Get a object from the pool at the given index.
|
|
*
|
|
* @param index
|
|
* the index
|
|
* @return the object
|
|
*/
|
|
public Object get( int index ) {
|
|
return constantPool[index];
|
|
}
|
|
|
|
/**
|
|
* Set a value in the constant pool.
|
|
*
|
|
* @param index
|
|
* the index
|
|
* @param value
|
|
* the new value
|
|
*/
|
|
void set( int index, Object value ) {
|
|
constantPool[index] = value;
|
|
}
|
|
|
|
/**
|
|
* Get the count of entries in the pool.
|
|
*
|
|
* @return the count
|
|
*/
|
|
int size() {
|
|
return constantPool.length;
|
|
}
|
|
}
|