Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. /*
  2. * Javassist, a Java-bytecode translator toolkit.
  3. * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved.
  4. *
  5. * The contents of this file are subject to the Mozilla Public License Version
  6. * 1.1 (the "License"); you may not use this file except in compliance with
  7. * the License. Alternatively, the contents of this file may be used under
  8. * the terms of the GNU Lesser General Public License Version 2.1 or later,
  9. * or the Apache License Version 2.0.
  10. *
  11. * Software distributed under the License is distributed on an "AS IS" basis,
  12. * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13. * for the specific language governing rights and limitations under the
  14. * License.
  15. */
  16. package javassist;
  17. import javassist.CtMethod.ConstParameter;
  18. import javassist.bytecode.AccessFlag;
  19. import javassist.bytecode.Bytecode;
  20. import javassist.bytecode.ConstPool;
  21. import javassist.bytecode.ExceptionsAttribute;
  22. import javassist.bytecode.FieldInfo;
  23. import javassist.bytecode.MethodInfo;
  24. import javassist.compiler.CompileError;
  25. import javassist.compiler.Javac;
  26. /**
  27. * A collection of static methods for creating a <code>CtMethod</code>.
  28. * An instance of this class does not make any sense.
  29. *
  30. * @see CtClass#addMethod(CtMethod)
  31. */
  32. public class CtNewMethod {
  33. /**
  34. * Compiles the given source code and creates a method.
  35. * The source code must include not only the method body
  36. * but the whole declaration, for example,
  37. *
  38. * <pre>"public Object id(Object obj) { return obj; }"</pre>
  39. *
  40. * @param src the source text.
  41. * @param declaring the class to which the created method is added.
  42. */
  43. public static CtMethod make(String src, CtClass declaring)
  44. throws CannotCompileException
  45. {
  46. return make(src, declaring, null, null);
  47. }
  48. /**
  49. * Compiles the given source code and creates a method.
  50. * The source code must include not only the method body
  51. * but the whole declaration, for example,
  52. *
  53. * <pre>"public Object id(Object obj) { return obj; }"</pre>
  54. *
  55. * <p>If the source code includes <code>$proceed()</code>, then
  56. * it is compiled into a method call on the specified object.
  57. *
  58. * @param src the source text.
  59. * @param declaring the class to which the created method is added.
  60. * @param delegateObj the source text specifying the object
  61. * that is called on by <code>$proceed()</code>.
  62. * @param delegateMethod the name of the method
  63. * that is called by <code>$proceed()</code>.
  64. */
  65. public static CtMethod make(String src, CtClass declaring,
  66. String delegateObj, String delegateMethod)
  67. throws CannotCompileException
  68. {
  69. Javac compiler = new Javac(declaring);
  70. try {
  71. if (delegateMethod != null)
  72. compiler.recordProceed(delegateObj, delegateMethod);
  73. CtMember obj = compiler.compile(src);
  74. declaring.addLines(src.split("\n").length);
  75. if (obj instanceof CtMethod)
  76. return (CtMethod)obj;
  77. }
  78. catch (CompileError e) {
  79. throw new CannotCompileException(e);
  80. }
  81. throw new CannotCompileException("not a method");
  82. }
  83. /**
  84. * Creates a public (non-static) method. The created method cannot
  85. * be changed to a static method later.
  86. *
  87. * @param returnType the type of the returned value.
  88. * @param mname the method name.
  89. * @param parameters a list of the parameter types.
  90. * @param exceptions a list of the exception types.
  91. * @param body the source text of the method body.
  92. * It must be a block surrounded by <code>{}</code>.
  93. * If it is <code>null</code>, the created method
  94. * does nothing except returning zero or null.
  95. * @param declaring the class to which the created method is added.
  96. * @see #make(int, CtClass, String, CtClass[], CtClass[], String, CtClass)
  97. */
  98. public static CtMethod make(CtClass returnType,
  99. String mname, CtClass[] parameters,
  100. CtClass[] exceptions,
  101. String body, CtClass declaring)
  102. throws CannotCompileException
  103. {
  104. return make(Modifier.PUBLIC, returnType, mname, parameters, exceptions,
  105. body, declaring);
  106. }
  107. /**
  108. * Creates a method. <code>modifiers</code> can contain
  109. * <code>Modifier.STATIC</code>.
  110. *
  111. * @param modifiers access modifiers.
  112. * @param returnType the type of the returned value.
  113. * @param mname the method name.
  114. * @param parameters a list of the parameter types.
  115. * @param exceptions a list of the exception types.
  116. * @param body the source text of the method body.
  117. * It must be a block surrounded by <code>{}</code>.
  118. * If it is <code>null</code>, the created method
  119. * does nothing except returning zero or null.
  120. * @param declaring the class to which the created method is added.
  121. *
  122. * @see Modifier
  123. */
  124. public static CtMethod make(int modifiers, CtClass returnType,
  125. String mname, CtClass[] parameters,
  126. CtClass[] exceptions,
  127. String body, CtClass declaring)
  128. throws CannotCompileException
  129. {
  130. try {
  131. CtMethod cm
  132. = new CtMethod(returnType, mname, parameters, declaring);
  133. cm.setModifiers(modifiers);
  134. cm.setExceptionTypes(exceptions);
  135. cm.setBody(body);
  136. return cm;
  137. }
  138. catch (NotFoundException e) {
  139. throw new CannotCompileException(e);
  140. }
  141. }
  142. /**
  143. * Creates a copy of a method. This method is provided for creating
  144. * a new method based on an existing method.
  145. * This is a convenience method for calling
  146. * {@link CtMethod#CtMethod(CtMethod, CtClass, ClassMap) this constructor}.
  147. * See the description of the constructor for particular behavior of the copying.
  148. *
  149. * @param src the source method.
  150. * @param declaring the class to which the created method is added.
  151. * @param map the hash table associating original class names
  152. * with substituted names.
  153. * It can be <code>null</code>.
  154. *
  155. * @see CtMethod#CtMethod(CtMethod,CtClass,ClassMap)
  156. */
  157. public static CtMethod copy(CtMethod src, CtClass declaring,
  158. ClassMap map) throws CannotCompileException {
  159. return new CtMethod(src, declaring, map);
  160. }
  161. /**
  162. * Creates a copy of a method with a new name.
  163. * This method is provided for creating
  164. * a new method based on an existing method.
  165. * This is a convenience method for calling
  166. * {@link CtMethod#CtMethod(CtMethod, CtClass, ClassMap) this constructor}.
  167. * See the description of the constructor for particular behavior of the copying.
  168. *
  169. * @param src the source method.
  170. * @param name the name of the created method.
  171. * @param declaring the class to which the created method is added.
  172. * @param map the hash table associating original class names
  173. * with substituted names.
  174. * It can be <code>null</code>.
  175. *
  176. * @see CtMethod#CtMethod(CtMethod,CtClass,ClassMap)
  177. */
  178. public static CtMethod copy(CtMethod src, String name, CtClass declaring,
  179. ClassMap map) throws CannotCompileException {
  180. CtMethod cm = new CtMethod(src, declaring, map);
  181. cm.setName(name);
  182. return cm;
  183. }
  184. /**
  185. * Creates a public abstract method.
  186. *
  187. * @param returnType the type of the returned value
  188. * @param mname the method name
  189. * @param parameters a list of the parameter types
  190. * @param exceptions a list of the exception types
  191. * @param declaring the class to which the created method is added.
  192. *
  193. * @see CtMethod#CtMethod(CtClass,String,CtClass[],CtClass)
  194. */
  195. public static CtMethod abstractMethod(CtClass returnType,
  196. String mname,
  197. CtClass[] parameters,
  198. CtClass[] exceptions,
  199. CtClass declaring)
  200. throws NotFoundException
  201. {
  202. CtMethod cm = new CtMethod(returnType, mname, parameters, declaring);
  203. cm.setExceptionTypes(exceptions);
  204. return cm;
  205. }
  206. /**
  207. * Creates a public getter method. The getter method returns the value
  208. * of the specified field in the class to which this method is added.
  209. * The created method is initially not static even if the field is
  210. * static. Change the modifiers if the method should be static.
  211. *
  212. * @param methodName the name of the getter
  213. * @param field the field accessed.
  214. */
  215. public static CtMethod getter(String methodName, CtField field)
  216. throws CannotCompileException
  217. {
  218. FieldInfo finfo = field.getFieldInfo2();
  219. String fieldType = finfo.getDescriptor();
  220. String desc = "()" + fieldType;
  221. ConstPool cp = finfo.getConstPool();
  222. MethodInfo minfo = new MethodInfo(cp, methodName, desc);
  223. minfo.setAccessFlags(AccessFlag.PUBLIC);
  224. Bytecode code = new Bytecode(cp, 2, 1);
  225. try {
  226. String fieldName = finfo.getName();
  227. if ((finfo.getAccessFlags() & AccessFlag.STATIC) == 0) {
  228. code.addAload(0);
  229. code.addGetfield(Bytecode.THIS, fieldName, fieldType);
  230. }
  231. else
  232. code.addGetstatic(Bytecode.THIS, fieldName, fieldType);
  233. code.addReturn(field.getType());
  234. }
  235. catch (NotFoundException e) {
  236. throw new CannotCompileException(e);
  237. }
  238. minfo.setCodeAttribute(code.toCodeAttribute());
  239. CtClass cc = field.getDeclaringClass();
  240. // a stack map is not needed.
  241. return new CtMethod(minfo, cc);
  242. }
  243. /**
  244. * Creates a public setter method. The setter method assigns the
  245. * value of the first parameter to the specified field
  246. * in the class to which this method is added.
  247. * The created method is not static even if the field is
  248. * static. You may not change it to be static
  249. * by <code>setModifiers()</code> in <code>CtBehavior</code>.
  250. *
  251. * @param methodName the name of the setter
  252. * @param field the field accessed.
  253. */
  254. public static CtMethod setter(String methodName, CtField field)
  255. throws CannotCompileException
  256. {
  257. FieldInfo finfo = field.getFieldInfo2();
  258. String fieldType = finfo.getDescriptor();
  259. String desc = "(" + fieldType + ")V";
  260. ConstPool cp = finfo.getConstPool();
  261. MethodInfo minfo = new MethodInfo(cp, methodName, desc);
  262. minfo.setAccessFlags(AccessFlag.PUBLIC);
  263. Bytecode code = new Bytecode(cp, 3, 3);
  264. try {
  265. String fieldName = finfo.getName();
  266. if ((finfo.getAccessFlags() & AccessFlag.STATIC) == 0) {
  267. code.addAload(0);
  268. code.addLoad(1, field.getType());
  269. code.addPutfield(Bytecode.THIS, fieldName, fieldType);
  270. }
  271. else {
  272. code.addLoad(1, field.getType());
  273. code.addPutstatic(Bytecode.THIS, fieldName, fieldType);
  274. }
  275. code.addReturn(null);
  276. }
  277. catch (NotFoundException e) {
  278. throw new CannotCompileException(e);
  279. }
  280. minfo.setCodeAttribute(code.toCodeAttribute());
  281. CtClass cc = field.getDeclaringClass();
  282. // a stack map is not needed.
  283. return new CtMethod(minfo, cc);
  284. }
  285. /**
  286. * Creates a method forwarding to a delegate in
  287. * a super class. The created method calls a method specified
  288. * by <code>delegate</code> with all the parameters passed to the
  289. * created method. If the delegate method returns a value,
  290. * the created method returns that value to the caller.
  291. * The delegate method must be declared in a super class.
  292. *
  293. * <p>The following method is an example of the created method.
  294. *
  295. * <pre>
  296. * int f(int p, int q) {
  297. * return super.f(p, q);
  298. * }</pre>
  299. *
  300. * <p>The name of the created method can be changed by
  301. * <code>setName()</code>.
  302. *
  303. * @param delegate the method that the created method forwards to.
  304. * @param declaring the class to which the created method is
  305. * added.
  306. */
  307. public static CtMethod delegator(CtMethod delegate, CtClass declaring)
  308. throws CannotCompileException
  309. {
  310. try {
  311. return delegator0(delegate, declaring);
  312. }
  313. catch (NotFoundException e) {
  314. throw new CannotCompileException(e);
  315. }
  316. }
  317. private static CtMethod delegator0(CtMethod delegate, CtClass declaring)
  318. throws CannotCompileException, NotFoundException
  319. {
  320. MethodInfo deleInfo = delegate.getMethodInfo2();
  321. String methodName = deleInfo.getName();
  322. String desc = deleInfo.getDescriptor();
  323. ConstPool cp = declaring.getClassFile2().getConstPool();
  324. MethodInfo minfo = new MethodInfo(cp, methodName, desc);
  325. minfo.setAccessFlags(deleInfo.getAccessFlags());
  326. ExceptionsAttribute eattr = deleInfo.getExceptionsAttribute();
  327. if (eattr != null)
  328. minfo.setExceptionsAttribute(
  329. (ExceptionsAttribute)eattr.copy(cp, null));
  330. Bytecode code = new Bytecode(cp, 0, 0);
  331. boolean isStatic = Modifier.isStatic(delegate.getModifiers());
  332. CtClass deleClass = delegate.getDeclaringClass();
  333. CtClass[] params = delegate.getParameterTypes();
  334. int s;
  335. if (isStatic) {
  336. s = code.addLoadParameters(params, 0);
  337. code.addInvokestatic(deleClass, methodName, desc);
  338. }
  339. else {
  340. code.addLoad(0, deleClass);
  341. s = code.addLoadParameters(params, 1);
  342. code.addInvokespecial(deleClass, methodName, desc);
  343. }
  344. code.addReturn(delegate.getReturnType());
  345. code.setMaxLocals(++s);
  346. code.setMaxStack(s < 2 ? 2 : s); // for a 2-word return value
  347. minfo.setCodeAttribute(code.toCodeAttribute());
  348. // a stack map is not needed.
  349. return new CtMethod(minfo, declaring);
  350. }
  351. /**
  352. * Creates a wrapped method. The wrapped method receives parameters
  353. * in the form of an array of <code>Object</code>.
  354. *
  355. * <p>The body of the created method is a copy of the body of the method
  356. * specified by <code>body</code>. However, it is wrapped in
  357. * parameter-conversion code.
  358. *
  359. * <p>The method specified by <code>body</code> must have this singature:
  360. *
  361. * <pre>Object method(Object[] params, &lt;type&gt; cvalue)</pre>
  362. *
  363. * <p>The type of the <code>cvalue</code> depends on
  364. * <code>constParam</code>.
  365. * If <code>constParam</code> is <code>null</code>, the signature
  366. * must be:
  367. *
  368. * <pre>Object method(Object[] params)</pre>
  369. *
  370. * <p>The method body copied from <code>body</code> is wrapped in
  371. * parameter-conversion code, which converts parameters specified by
  372. * <code>parameterTypes</code> into an array of <code>Object</code>.
  373. * The returned value is also converted from the <code>Object</code>
  374. * type to the type specified by <code>returnType</code>. Thus,
  375. * the resulting method body is as follows:
  376. *
  377. * <pre>
  378. * Object[] params = new Object[] { p0, p1, ... };
  379. * &lt;<i>type</i>&gt; cvalue = &lt;<i>constant-value</i>&gt;;
  380. * <i>... copied method body ...</i>
  381. * Object result = &lt;<i>returned value</i>&gt;
  382. * return (<i>&lt;returnType&gt;</i>)result;
  383. * </pre>
  384. *
  385. * <p>The variables <code>p0</code>, <code>p2</code>, ... represent
  386. * formal parameters of the created method.
  387. * The value of <code>cvalue</code> is specified by
  388. * <code>constParam</code>.
  389. *
  390. * <p>If the type of a parameter or a returned value is a primitive
  391. * type, then the value is converted into a wrapper object such as
  392. * <code>java.lang.Integer</code>. If the type of the returned value
  393. * is <code>void</code>, the returned value is discarded.
  394. *
  395. * <p><i>Example:</i>
  396. *
  397. * <pre>
  398. * ClassPool pool = ... ;
  399. * CtClass vec = pool.makeClass("intVector");
  400. * vec.setSuperclass(pool.get("java.util.Vector"));
  401. * CtMethod addMethod = pool.getMethod("Sample", "add0");
  402. *
  403. * CtClass[] argTypes = { CtClass.intType };
  404. * CtMethod m = CtNewMethod.wrapped(CtClass.voidType, "add", argTypes,
  405. * null, addMethod, null, vec);
  406. * vec.addMethod(m);</pre>
  407. *
  408. * <p>where the class <code>Sample</code> is as follows:
  409. *
  410. * <pre>public class Sample extends java.util.Vector {
  411. * public Object add0(Object[] args) {
  412. * super.addElement(args[0]);
  413. * return null;
  414. * }
  415. * }</pre>
  416. *
  417. * <p>This program produces a class <code>intVector</code>:
  418. *
  419. * <pre>public class intVector extends java.util.Vector {
  420. * public void add(int p0) {
  421. * Object[] args = new Object[] { p0 };
  422. * // begin of the copied body
  423. * super.addElement(args[0]);
  424. * Object result = null;
  425. * // end
  426. * }
  427. * }</pre>
  428. *
  429. * <p>Note that the type of the parameter to <code>add()</code> depends
  430. * only on the value of <code>argTypes</code> passed to
  431. * <code>CtNewMethod.wrapped()</code>. Thus, it is easy to
  432. * modify this program to produce a
  433. * <code>StringVector</code> class, which is a vector containing
  434. * only <code>String</code> objects, and other vector classes.
  435. *
  436. * @param returnType the type of the returned value.
  437. * @param mname the method name.
  438. * @param parameterTypes a list of the parameter types.
  439. * @param exceptionTypes a list of the exception types.
  440. * @param body the method body
  441. * (must not be a static method).
  442. * @param constParam the constant parameter
  443. * (maybe <code>null</code>).
  444. * @param declaring the class to which the created method is
  445. * added.
  446. */
  447. public static CtMethod wrapped(CtClass returnType,
  448. String mname,
  449. CtClass[] parameterTypes,
  450. CtClass[] exceptionTypes,
  451. CtMethod body, ConstParameter constParam,
  452. CtClass declaring)
  453. throws CannotCompileException
  454. {
  455. return CtNewWrappedMethod.wrapped(returnType, mname, parameterTypes,
  456. exceptionTypes, body, constParam, declaring);
  457. }
  458. }