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

serp.bytecode.BCMethod Maven / Gradle / Ivy

package serp.bytecode;

import serp.bytecode.visitor.*;
import serp.util.*;

/**
 * A method of a class.
 *
 * @author Abe White
 */
public class BCMethod extends BCMember implements VisitAcceptor {
    BCMethod(BCClass owner) {
        super(owner);
    }

    /////////////////////
    // Access operations
    /////////////////////

    /**
     * Manipulate the method access flags.
     */
    public boolean isSynchronized() {
        return (getAccessFlags() & Constants.ACCESS_SYNCHRONIZED) > 0;
    }

    /**
     * Manipulate the method access flags.
     */
    public void setSynchronized(boolean on) {
        if (on)
            setAccessFlags(getAccessFlags() | Constants.ACCESS_SYNCHRONIZED);
        else
            setAccessFlags(getAccessFlags() & ~Constants.ACCESS_SYNCHRONIZED);
    }

    /**
     * Manipulate the method access flags.
     */
    public boolean isNative() {
        return (getAccessFlags() & Constants.ACCESS_NATIVE) > 0;
    }

    /**
     * Manipulate the method access flags.
     */
    public void setNative(boolean on) {
        if (on)
            setAccessFlags(getAccessFlags() | Constants.ACCESS_NATIVE);
        else
            setAccessFlags(getAccessFlags() & ~Constants.ACCESS_NATIVE);
    }

    /**
     * Manipulate the method access flags.
     */
    public boolean isAbstract() {
        return (getAccessFlags() & Constants.ACCESS_ABSTRACT) > 0;
    }

    /**
     * Manipulate the method access flags.
     */
    public void setAbstract(boolean on) {
        if (on)
            setAccessFlags(getAccessFlags() | Constants.ACCESS_ABSTRACT);
        else
            setAccessFlags(getAccessFlags() & ~Constants.ACCESS_ABSTRACT);
    }

    /**
     * Manipulate the method access flags.
     */
    public boolean isStrict() {
        return (getAccessFlags() & Constants.ACCESS_STRICT) > 0;
    }

    /**
     * Manipulate the method access flags.
     */
    public void setStrict(boolean on) {
        if (on)
            setAccessFlags(getAccessFlags() | Constants.ACCESS_STRICT);
        else
            setAccessFlags(getAccessFlags() & ~Constants.ACCESS_STRICT);
    }

    /**
     * Manipulate the method access flags.
     */
    public boolean isVarArgs() {
        return (getAccessFlags() & Constants.ACCESS_VARARGS) > 0;
    }

    /**
     * Manipulate the method access flags.
     */
    public void setVarArgs(boolean on) {
        if (on)
            setAccessFlags(getAccessFlags() | Constants.ACCESS_VARARGS);
        else
            setAccessFlags(getAccessFlags() & ~Constants.ACCESS_VARARGS);
    }

    /**
     * Manipulate the method access flags.
     */
    public boolean isBridge() {
        return (getAccessFlags() & Constants.ACCESS_BRIDGE) > 0;
    }

    /**
     * Manipulate the method access flags.
     */
    public void setBridge(boolean on) {
        if (on)
            setAccessFlags(getAccessFlags() | Constants.ACCESS_BRIDGE);
        else
            setAccessFlags(getAccessFlags() & ~Constants.ACCESS_BRIDGE);
    }

    /////////////////////
    // Return operations
    /////////////////////

    /**
     * Return the name of the type returned by this method. The name
     * will be given in a form suitable for a {@link Class#forName} call.
     *
     * @see BCMember#getDescriptor
     */
    public String getReturnName() {
        return getProject().getNameCache().getExternalForm(getProject().
            getNameCache().getDescriptorReturnName(getDescriptor()), false);
    }

    /**
     * Return the {@link Class} object for the return type of this method.
     *
     * @see BCMember#getDescriptor
     */
    public Class getReturnType() {
        return Strings.toClass(getReturnName(), getClassLoader());
    }

    /**
     * Return the bytecode for the return type of this method.
     *
     * @see BCMember#getDescriptor
     */
    public BCClass getReturnBC() {
        return getProject().loadClass(getReturnName(), getClassLoader());
    }

    /**
     * Set the return type of this method.
     */
    public void setReturn(String name) {
        setDescriptor(getProject().getNameCache().getDescriptor(name, 
            getParamNames()));
    }

    /**
     * Set the return type of this method.
     */
    public void setReturn(Class type) {
        setReturn(type.getName());
    }

    /**
     * Set the return type of this method.
     */
    public void setReturn(BCClass type) {
        setReturn(type.getName());
    }

    ////////////////////////
    // Parameter operations
    ////////////////////////

    /**
     * Return the names of all the parameter types for this method. The names
     * will be returned in a form suitable for a {@link Class#forName} call.
     *
     * @see BCMember#getDescriptor
     */
    public String[] getParamNames() {
        // get the parameter types from the descriptor
        String[] params = getProject().getNameCache().getDescriptorParamNames
            (getDescriptor());

        // convert them to external form
        for (int i = 0; i < params.length; i++)
            params[i] = getProject().getNameCache().getExternalForm(params[i], 
                false);
        return params;
    }

    /**
     * Return the {@link Class} objects for all the parameter types for this
     * method.
     *
     * @see BCMember#getDescriptor
     */
    public Class[] getParamTypes() {
        String[] paramNames = getParamNames();
        Class[] params = new Class[paramNames.length];
        for (int i = 0; i < paramNames.length; i++)
            params[i] = Strings.toClass(paramNames[i], getClassLoader());
        return params;
    }

    /**
     * Return the bytecode for all the parameter types for this method.
     *
     * @see BCMember#getDescriptor
     */
    public BCClass[] getParamBCs() {
        String[] paramNames = getParamNames();
        BCClass[] params = new BCClass[paramNames.length];
        for (int i = 0; i < paramNames.length; i++)
            params[i] = getProject().loadClass(paramNames[i], getClassLoader());
        return params;
    }

    /**
     * Set the parameter types of this method.
     *
     * @see BCMember#setDescriptor
     */
    public void setParams(String[] names) {
        if (names == null)
            names = new String[0];
        setDescriptor(getProject().getNameCache().getDescriptor(getReturnName(),
            names));
    }

    /**
     * Set the parameter type of this method.
     *
     * @see BCMember#setDescriptor
     */
    public void setParams(Class[] types) {
        if (types == null)
            setParams((String[]) null);
        else {
            String[] names = new String[types.length];
            for (int i = 0; i < types.length; i++)
                names[i] = types[i].getName();
            setParams(names);
        }
    }

    /**
     * Set the parameter type of this method.
     *
     * @see BCMember#setDescriptor
     */
    public void setParams(BCClass[] types) {
        if (types == null)
            setParams((String[]) null);
        else {
            String[] names = new String[types.length];
            for (int i = 0; i < types.length; i++)
                names[i] = types[i].getName();
            setParams(names);
        }
    }

    /**
     * Add a parameter type to this method.
     */
    public void addParam(String type) {
        String[] origParams = getParamNames();
        String[] params = new String[origParams.length + 1];
        for (int i = 0; i < origParams.length; i++)
            params[i] = origParams[i];
        params[origParams.length] = type;
        setParams(params);
    }

    /**
     * Add a parameter type to this method.
     */
    public void addParam(Class type) {
        addParam(type.getName());
    }

    /**
     * Add a parameter type to this method.
     */
    public void addParam(BCClass type) {
        addParam(type.getName());
    }

    /**
     * Add a parameter type to this method.
     *
     * @see java.util.List#add(int,Object)
     */
    public void addParam(int pos, String type) {
        String[] origParams = getParamNames();
        if ((pos < 0) || (pos >= origParams.length))
            throw new IndexOutOfBoundsException("pos = " + pos);

        String[] params = new String[origParams.length + 1];
        for (int i = 0, index = 0; i < params.length; i++) {
            if (i == pos)
                params[i] = type;
            else
                params[i] = origParams[index++];
        }
        setParams(params);
    }

    /**
     * Add a parameter type to this method.
     *
     * @see java.util.List#add(int,Object)
     */
    public void addParam(int pos, Class type) {
        addParam(pos, type.getName());
    }

    /**
     * Add a parameter type to this method.
     *
     * @see java.util.List#add(int,Object)
     */
    public void addParam(int pos, BCClass type) {
        addParam(pos, type.getName());
    }

    /**
     * Change a parameter type of this method.
     *
     * @see java.util.List#set(int,Object)
     */
    public void setParam(int pos, String type) {
        String[] origParams = getParamNames();
        if ((pos < 0) || (pos >= origParams.length))
            throw new IndexOutOfBoundsException("pos = " + pos);

        String[] params = new String[origParams.length];
        for (int i = 0; i < params.length; i++) {
            if (i == pos)
                params[i] = type;
            else
                params[i] = origParams[i];
        }
        setParams(params);
    }

    /**
     * Change a parameter type of this method.
     *
     * @see java.util.List#set(int,Object)
     */
    public void setParam(int pos, Class type) {
        setParam(pos, type.getName());
    }

    /**
     * Change a parameter type of this method.
     *
     * @see java.util.List#set(int,Object)
     */
    public void setParam(int pos, BCClass type) {
        setParam(pos, type.getName());
    }

    /**
     * Clear all parameters from this method.
     */
    public void clearParams() {
        setParams((String[]) null);
    }

    /**
     * Remove a parameter from this method.
     */
    public void removeParam(int pos) {
        String[] origParams = getParamNames();
        if ((pos < 0) || (pos >= origParams.length))
            throw new IndexOutOfBoundsException("pos = " + pos);

        String[] params = new String[origParams.length - 1];

        for (int i = 0, index = 0; i < origParams.length; i++)
            if (i != pos)
                params[index++] = origParams[i];
        setParams(params);
    }

    ///////////////////////
    // Convenience methods
    ///////////////////////

    /**
     * Return the checked exceptions information for the method.
     * Acts internally through the {@link Attributes} interface.
     *
     * @param add if true, a new exceptions attribute will be added
     * if not already present
     * @return the exceptions information, or null if none and the
     * add param is set to false
     */
    public Exceptions getExceptions(boolean add) {
        Exceptions exceptions = (Exceptions) getAttribute
            (Constants.ATTR_EXCEPTIONS);
        if (!add || (exceptions != null))
            return exceptions;

        if (exceptions == null)
            exceptions = (Exceptions) addAttribute(Constants.ATTR_EXCEPTIONS);
        return exceptions;
    }

    /**
     * Remove the exceptions attribute for the method.
     * Acts internally through the {@link Attributes} interface.
     *
     * @return true if there was a value to remove
     */
    public boolean removeExceptions() {
        return removeAttribute(Constants.ATTR_EXCEPTIONS);
    }

    /**
     * Return the code for the method. If the code already exists, its
     * iterator will be reset to the first instruction.
     * Acts internally through the {@link Attributes} interface.
     *
     * @param add if true, a new code attribute will be added
     * if not already present
     * @return the code for the metohd, or null if none and the
     * add param is set to false
     */
    public Code getCode(boolean add) {
        Code code = (Code) getAttribute(Constants.ATTR_CODE);
        if (code != null) {
            code.beforeFirst();
            return code;
        }
        if (!add)
            return null;
        return (Code) addAttribute(Constants.ATTR_CODE);
    }

    /**
     * Remove the code attribute from the method.
     * Acts internally through the {@link Attributes} interface.
     *
     * @return true if there was a value to remove
     */
    public boolean removeCode() {
        return removeAttribute(Constants.ATTR_CODE);
    }

    ////////////////////////////////
    // VisitAcceptor implementation
    ////////////////////////////////

    public void acceptVisit(BCVisitor visit) {
        visit.enterBCMethod(this);
        visitAttributes(visit);
        visit.exitBCMethod(this);
    }

    void initialize(String name, String descriptor) {
        super.initialize(name, descriptor);
        makePublic();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy