All Downloads are FREE. Search and download functionalities are using the official Maven repository.

serp.bytecode.ClassConstantInstruction Maven / Gradle / Ivy

There is a newer version: 10.0.0-M3
Show newest version
package serp.bytecode;

import java.util.*;

import serp.bytecode.lowlevel.*;

/**
 * Pseudo-instruction used to place {@link Class} objects onto the stack.
 * This logical instruction may actually involve a large chunk of code, and
 * may even add static synthetic fields and methods to the owning class.
 * Therefore, once the type of class being loaded is set, it cannot
 * be changed. Also, this instruction is invalid as the target of
 * any jump instruction or exception handler.
 *
 * @author Abe White
 */
public class ClassConstantInstruction {
    private static final Class[] _params = new Class[] { String.class };
    private static final Map _wrappers = new HashMap();
    static {
        _wrappers.put(byte.class.getName(), Byte.class);
        _wrappers.put(boolean.class.getName(), Boolean.class);
        _wrappers.put(char.class.getName(), Character.class);
        _wrappers.put(double.class.getName(), Double.class);
        _wrappers.put(float.class.getName(), Float.class);
        _wrappers.put(int.class.getName(), Integer.class);
        _wrappers.put(long.class.getName(), Long.class);
        _wrappers.put(short.class.getName(), Short.class);
    }

    private Instruction _ins = null;
    private Code _code = null;
    private BCClass _class = null;
    private boolean _invalid = false;

    ClassConstantInstruction(BCClass bc, Code code, Instruction nop) {
        _class = bc;
        _code = code;
        _ins = nop;
    }

    /**
     * Set the type of class being loaded.
     *
     * @return the first Instruction of the block added by setting the type
     * @throws IllegalStateException if type has already been set
     */
    public Instruction setClass(String name) {
        name = _class.getProject().getNameCache().getExternalForm(name, false);
        setClassName(name, getWrapperClass(name));
        return _ins;
    }

    /**
     * Set the type of class being loaded.
     *
     * @return the first Instruction of the block added by setting the type
     * @throws IllegalStateException if type has already been set
     */
    public Instruction setClass(Class type) {
        return setClass(type.getName());
    }

    /**
     * Set the type of class being loaded.
     *
     * @return the first Instruction of the block added by setting the type
     * @throws IllegalStateException if type has already been set
     */
    public Instruction setClass(BCClass type) {
        return setClass(type.getName());
    }

    /**
     * Set the name of the class to load.
     */
    private void setClassName(String name, Class wrapper) {
        if (_invalid)
            throw new IllegalStateException();

        // remember the position of the code iterator
        Instruction before = (_code.hasNext()) ? _code.next() : null;
        _code.before(_ins);
        _code.next();
        if (wrapper != null)
            _code.getstatic().setField(wrapper, "TYPE", Class.class);
        else
            setObject(name);

        // move to the old position
        if (before != null)
            _code.before(before);
        else
            _code.afterLast();
        _invalid = true;
    }

    /**
     * Adds fields and methods as necessary to load a class constant of
     * an object type.
     */
    private void setObject(String name) {
        BCField field = addClassField(name);
        BCMethod method = addClassLoadMethod();

        // copied from the way jikes loads classes
        _code.getstatic().setField(field);
        JumpInstruction ifnull = _code.ifnull();
        _code.getstatic().setField(field);
        JumpInstruction go2 = _code.go2();
        ifnull.setTarget(_code.constant().setValue(name));
        _code.invokestatic().setMethod(method);
        _code.dup();
        _code.putstatic().setField(field);
        go2.setTarget(_code.nop());
    }

    /**
     * Adds a static field to hold the loaded class constant.
     */
    private BCField addClassField(String name) {
        String fieldName = "class$L" 
            + name.replace('.', '$').replace('[', '$').replace(';', '$');
        BCField field = _class.getDeclaredField(fieldName);
        if (field == null) {
            field = _class.declareField(fieldName, Class.class);
            field.makePackage();
            field.setStatic(true);
            field.setSynthetic(true);
        }
        return field;
    }

    /**
     * Adds the standard class$ method used inernally by classes
     * to load class constants for object types.
     */
    private BCMethod addClassLoadMethod() {
        BCMethod method = _class.getDeclaredMethod("class$", _params);
        if (method != null)
            return method;

        // add the special synthetic method
        method = _class.declareMethod("class$", Class.class, _params);
        method.setStatic(true);
        method.makePackage();
        method.setSynthetic(true);

        // copied directly from the output of the jikes compiler
        Code code = method.getCode(true);
        code.setMaxStack(3);
        code.setMaxLocals(2);

        Instruction tryStart = code.aload().setLocal(0);
        code.invokestatic().setMethod(Class.class, "forName", Class.class, 
            _params);
        Instruction tryEnd = code.areturn();
        Instruction handlerStart = code.astore().setLocal(1);
        code.anew().setType(NoClassDefFoundError.class);
        code.dup();
        code.aload().setLocal(1);
        code.invokevirtual().setMethod(Throwable.class, "getMessage", 
            String.class, null);
        code.invokespecial().setMethod(NoClassDefFoundError.class, "", 
            void.class, _params);
        code.athrow();
        code.addExceptionHandler(tryStart, tryEnd, handlerStart,
            ClassNotFoundException.class);
        return method;
    }

    /**
     * Return the wrapper type for the given primitive class, or null
     * if the given name is not a primitive type. The given name should
     * be in external form.
     */
    private static Class getWrapperClass(String name) {
        if (name == null)
            return null;
        return (Class) _wrappers.get(name);
    }
}