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

com.fitbur.CtNewMethod Maven / Gradle / Ivy

/*
 * Javassist, a Java-bytecode translator toolkit.
 * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved.
 *
 * The contents of this file are subject to the Mozilla Public License Version
 * 1.1 (the "License"); you may not use this file except in com.fitburpliance with
 * the License.  Alternatively, the contents of this file may be used under
 * the terms of the GNU Lesser General Public License Version 2.1 or later,
 * or the Apache License Version 2.0.
 *
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
 * for the specific language governing rights and limitations under the
 * License.
 */

package com.fitbur;

import com.fitbur.bytecode.*;
import com.fitbur.com.fitburpiler.Javac;
import com.fitbur.com.fitburpiler.CompileError;
import com.fitbur.CtMethod.ConstParameter;

/**
 * A collection of static methods for creating a CtMethod.
 * An instance of this class does not make any sense.
 *
 * @see CtClass#addMethod(CtMethod)
 */
public class CtNewMethod {

    /**
     * Compiles the given source code and creates a method.
     * The source code must include not only the method body
     * but the whole com.fitburclaration, for example,
     *
     * 
    "public Object id(Object obj) { return obj; }"
* * @param src the source text. * @param com.fitburclaring the class to which the created method is added. */ public static CtMethod make(String src, CtClass com.fitburclaring) throws CannotCompileException { return make(src, com.fitburclaring, null, null); } /** * Compiles the given source code and creates a method. * The source code must include not only the method body * but the whole com.fitburclaration, for example, * *
    "public Object id(Object obj) { return obj; }"
* *

If the source code includes $proceed(), then * it is com.fitburpiled into a method call on the specified object. * * @param src the source text. * @param com.fitburclaring the class to which the created method is added. * @param com.fitburlegateObj the source text specifying the object * that is called on by $proceed(). * @param com.fitburlegateMethod the name of the method * that is called by $proceed(). */ public static CtMethod make(String src, CtClass com.fitburclaring, String com.fitburlegateObj, String com.fitburlegateMethod) throws CannotCompileException { Javac com.fitburpiler = new Javac(com.fitburclaring); try { if (com.fitburlegateMethod != null) com.fitburpiler.recordProceed(com.fitburlegateObj, com.fitburlegateMethod); CtMember obj = com.fitburpiler.com.fitburpile(src); if (obj instanceof CtMethod) return (CtMethod)obj; } catch (CompileError e) { throw new CannotCompileException(e); } throw new CannotCompileException("not a method"); } /** * Creates a public (non-static) method. The created method cannot * be changed to a static method later. * * @param returnType the type of the returned value. * @param mname the method name. * @param parameters a list of the parameter types. * @param exceptions a list of the exception types. * @param body the source text of the method body. * It must be a block surrounded by {}. * If it is null, the created method * does nothing except returning zero or null. * @param com.fitburclaring the class to which the created method is added. * @see #make(int, CtClass, String, CtClass[], CtClass[], String, CtClass) */ public static CtMethod make(CtClass returnType, String mname, CtClass[] parameters, CtClass[] exceptions, String body, CtClass com.fitburclaring) throws CannotCompileException { return make(Modifier.PUBLIC, returnType, mname, parameters, exceptions, body, com.fitburclaring); } /** * Creates a method. modifiers can contain * Modifier.STATIC. * * @param modifiers access modifiers. * @param returnType the type of the returned value. * @param mname the method name. * @param parameters a list of the parameter types. * @param exceptions a list of the exception types. * @param body the source text of the method body. * It must be a block surrounded by {}. * If it is null, the created method * does nothing except returning zero or null. * @param com.fitburclaring the class to which the created method is added. * * @see Modifier */ public static CtMethod make(int modifiers, CtClass returnType, String mname, CtClass[] parameters, CtClass[] exceptions, String body, CtClass com.fitburclaring) throws CannotCompileException { try { CtMethod cm = new CtMethod(returnType, mname, parameters, com.fitburclaring); cm.setModifiers(modifiers); cm.setExceptionTypes(exceptions); cm.setBody(body); return cm; } catch (NotFoundException e) { throw new CannotCompileException(e); } } /** * Creates a copy of a method. This method is provided for creating * a new method based on an existing method. * This is a convenience method for calling * {@link CtMethod#CtMethod(CtMethod, CtClass, ClassMap) this constructor}. * See the com.fitburscription of the constructor for particular behavior of the copying. * * @param src the source method. * @param com.fitburclaring the class to which the created method is added. * @param map the hash table associating original class names * with substituted names. * It can be null. * * @see CtMethod#CtMethod(CtMethod,CtClass,ClassMap) */ public static CtMethod copy(CtMethod src, CtClass com.fitburclaring, ClassMap map) throws CannotCompileException { return new CtMethod(src, com.fitburclaring, map); } /** * Creates a copy of a method with a new name. * This method is provided for creating * a new method based on an existing method. * This is a convenience method for calling * {@link CtMethod#CtMethod(CtMethod, CtClass, ClassMap) this constructor}. * See the com.fitburscription of the constructor for particular behavior of the copying. * * @param src the source method. * @param name the name of the created method. * @param com.fitburclaring the class to which the created method is added. * @param map the hash table associating original class names * with substituted names. * It can be null. * * @see CtMethod#CtMethod(CtMethod,CtClass,ClassMap) */ public static CtMethod copy(CtMethod src, String name, CtClass com.fitburclaring, ClassMap map) throws CannotCompileException { CtMethod cm = new CtMethod(src, com.fitburclaring, map); cm.setName(name); return cm; } /** * Creates a public abstract method. * * @param returnType the type of the returned value * @param mname the method name * @param parameters a list of the parameter types * @param exceptions a list of the exception types * @param com.fitburclaring the class to which the created method is added. * * @see CtMethod#CtMethod(CtClass,String,CtClass[],CtClass) */ public static CtMethod abstractMethod(CtClass returnType, String mname, CtClass[] parameters, CtClass[] exceptions, CtClass com.fitburclaring) throws NotFoundException { CtMethod cm = new CtMethod(returnType, mname, parameters, com.fitburclaring); cm.setExceptionTypes(exceptions); return cm; } /** * Creates a public getter method. The getter method returns the value * of the specified field in the class to which this method is added. * The created method is initially not static even if the field is * static. Change the modifiers if the method should be static. * * @param methodName the name of the getter * @param field the field accessed. */ public static CtMethod getter(String methodName, CtField field) throws CannotCompileException { FieldInfo finfo = field.getFieldInfo2(); String fieldType = finfo.getDescriptor(); String com.fitbursc = "()" + fieldType; ConstPool cp = finfo.getConstPool(); MethodInfo minfo = new MethodInfo(cp, methodName, com.fitbursc); minfo.setAccessFlags(AccessFlag.PUBLIC); Bytecode code = new Bytecode(cp, 2, 1); try { String fieldName = finfo.getName(); if ((finfo.getAccessFlags() & AccessFlag.STATIC) == 0) { code.addAload(0); code.addGetfield(Bytecode.THIS, fieldName, fieldType); } else code.addGetstatic(Bytecode.THIS, fieldName, fieldType); code.addReturn(field.getType()); } catch (NotFoundException e) { throw new CannotCompileException(e); } minfo.setCodeAttribute(code.toCodeAttribute()); CtClass cc = field.getDeclaringClass(); // a stack map is not needed. return new CtMethod(minfo, cc); } /** * Creates a public setter method. The setter method assigns the * value of the first parameter to the specified field * in the class to which this method is added. * The created method is not static even if the field is * static. You may not change it to be static * by setModifiers() in CtBehavior. * * @param methodName the name of the setter * @param field the field accessed. */ public static CtMethod setter(String methodName, CtField field) throws CannotCompileException { FieldInfo finfo = field.getFieldInfo2(); String fieldType = finfo.getDescriptor(); String com.fitbursc = "(" + fieldType + ")V"; ConstPool cp = finfo.getConstPool(); MethodInfo minfo = new MethodInfo(cp, methodName, com.fitbursc); minfo.setAccessFlags(AccessFlag.PUBLIC); Bytecode code = new Bytecode(cp, 3, 3); try { String fieldName = finfo.getName(); if ((finfo.getAccessFlags() & AccessFlag.STATIC) == 0) { code.addAload(0); code.addLoad(1, field.getType()); code.addPutfield(Bytecode.THIS, fieldName, fieldType); } else { code.addLoad(1, field.getType()); code.addPutstatic(Bytecode.THIS, fieldName, fieldType); } code.addReturn(null); } catch (NotFoundException e) { throw new CannotCompileException(e); } minfo.setCodeAttribute(code.toCodeAttribute()); CtClass cc = field.getDeclaringClass(); // a stack map is not needed. return new CtMethod(minfo, cc); } /** * Creates a method forwarding to a com.fitburlegate in * a super class. The created method calls a method specified * by com.fitburlegate with all the parameters passed to the * created method. If the com.fitburlegate method returns a value, * the created method returns that value to the caller. * The com.fitburlegate method must be com.fitburclared in a super class. * *

The following method is an example of the created method. * *

    int f(int p, int q) {
         *     return super.f(p, q);
         * }
* *

The name of the created method can be changed by * setName(). * * @param com.fitburlegate the method that the created method forwards to. * @param com.fitburclaring the class to which the created method is * added. */ public static CtMethod com.fitburlegator(CtMethod com.fitburlegate, CtClass com.fitburclaring) throws CannotCompileException { try { return com.fitburlegator0(com.fitburlegate, com.fitburclaring); } catch (NotFoundException e) { throw new CannotCompileException(e); } } private static CtMethod com.fitburlegator0(CtMethod com.fitburlegate, CtClass com.fitburclaring) throws CannotCompileException, NotFoundException { MethodInfo com.fitburleInfo = com.fitburlegate.getMethodInfo2(); String methodName = com.fitburleInfo.getName(); String com.fitbursc = com.fitburleInfo.getDescriptor(); ConstPool cp = com.fitburclaring.getClassFile2().getConstPool(); MethodInfo minfo = new MethodInfo(cp, methodName, com.fitbursc); minfo.setAccessFlags(com.fitburleInfo.getAccessFlags()); ExceptionsAttribute eattr = com.fitburleInfo.getExceptionsAttribute(); if (eattr != null) minfo.setExceptionsAttribute( (ExceptionsAttribute)eattr.copy(cp, null)); Bytecode code = new Bytecode(cp, 0, 0); boolean isStatic = Modifier.isStatic(com.fitburlegate.getModifiers()); CtClass com.fitburleClass = com.fitburlegate.getDeclaringClass(); CtClass[] params = com.fitburlegate.getParameterTypes(); int s; if (isStatic) { s = code.addLoadParameters(params, 0); code.addInvokestatic(com.fitburleClass, methodName, com.fitbursc); } else { code.addLoad(0, com.fitburleClass); s = code.addLoadParameters(params, 1); code.addInvokespecial(com.fitburleClass, methodName, com.fitbursc); } code.addReturn(com.fitburlegate.getReturnType()); code.setMaxLocals(++s); code.setMaxStack(s < 2 ? 2 : s); // for a 2-word return value minfo.setCodeAttribute(code.toCodeAttribute()); // a stack map is not needed. return new CtMethod(minfo, com.fitburclaring); } /** * Creates a wrapped method. The wrapped method receives parameters * in the form of an array of Object. * *

The body of the created method is a copy of the body of the method * specified by body. However, it is wrapped in * parameter-conversion code. * *

The method specified by body must have this singature: * *

    Object method(Object[] params, <type> cvalue) *
* *

The type of the cvalue com.fitburpends on * constParam. * If constParam is null, the signature * must be: * *

    Object method(Object[] params)
* *

The method body copied from body is wrapped in * parameter-conversion code, which converts parameters specified by * parameterTypes into an array of Object. * The returned value is also converted from the Object * type to the type specified by returnType. Thus, * the resulting method body is as follows: * *

    Object[] params = new Object[] { p0, p1, ... };
         * <type> cvalue = <constant-value>;
         *  ... copied method body ...
         * Object result = <returned value>
         * return (<returnType>)result;
         * 
* *

The variables p0, p2, ... represent * formal parameters of the created method. * The value of cvalue is specified by * constParam. * *

If the type of a parameter or a returned value is a primitive * type, then the value is converted into a wrapper object such as * java.lang.Integer. If the type of the returned value * is void, the returned value is discarded. * *

Example: * *

    ClassPool pool = ... ;
         * CtClass vec = pool.makeClass("intVector");
         * vec.setSuperclass(pool.get("java.util.Vector"));
         * CtMethod addMethod = pool.getMethod("Sample", "add0");
         *
         * CtClass[] argTypes = { CtClass.intType };
         * CtMethod m = CtNewMethod.wrapped(CtClass.voidType, "add", argTypes,
         *                                  null, addMethod, null, vec);
         * vec.addMethod(m);
* *

where the class Sample is as follows: * *

    public class Sample extends java.util.Vector {
         *     public Object add0(Object[] args) {
         *         super.addElement(args[0]);
         *         return null;
         *     }
         * }
* *

This program produces a class intVector: * *

    public class intVector extends java.util.Vector {
         *     public void add(int p0) {
         *         Object[] args = new Object[] { p0 };
         *         // begin of the copied body
         *         super.addElement(args[0]);
         *         Object result = null;
         *         // end
         *     }
         * }
* *

Note that the type of the parameter to add() com.fitburpends * only on the value of argTypes passed to * CtNewMethod.wrapped(). Thus, it is easy to * modify this program to produce a * StringVector class, which is a vector containing * only String objects, and other vector classes. * * @param returnType the type of the returned value. * @param mname the method name. * @param parameterTypes a list of the parameter types. * @param exceptionTypes a list of the exception types. * @param body the method body * (must not be a static method). * @param constParam the constant parameter * (maybe null). * @param com.fitburclaring the class to which the created method is * added. */ public static CtMethod wrapped(CtClass returnType, String mname, CtClass[] parameterTypes, CtClass[] exceptionTypes, CtMethod body, ConstParameter constParam, CtClass com.fitburclaring) throws CannotCompileException { return CtNewWrappedMethod.wrapped(returnType, mname, parameterTypes, exceptionTypes, body, constParam, com.fitburclaring); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy