You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

WeavingURLClassLoader.java 6.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. /* *******************************************************************
  2. * Copyright (c) 2004 IBM Corporation
  3. * All rights reserved.
  4. * This program and the accompanying materials are made available
  5. * under the terms of the Eclipse Public License v1.0
  6. * which accompanies this distribution and is available at
  7. * http://www.eclipse.org/legal/epl-v10.html
  8. *
  9. * Contributors:
  10. * Matthew Webster, Adrian Colyer,
  11. * Martin Lippert initial implementation
  12. * Andy Clement
  13. * Abraham Nevado
  14. * ******************************************************************/
  15. package org.aspectj.weaver.loadtime;
  16. import java.io.File;
  17. import java.io.IOException;
  18. import java.net.MalformedURLException;
  19. import java.net.URL;
  20. import java.security.CodeSource;
  21. import java.util.ArrayList;
  22. import java.util.HashMap;
  23. import java.util.List;
  24. import java.util.Map;
  25. import java.util.StringTokenizer;
  26. import org.aspectj.bridge.AbortException;
  27. import org.aspectj.weaver.bcel.ExtensibleURLClassLoader;
  28. import org.aspectj.weaver.tools.Trace;
  29. import org.aspectj.weaver.tools.TraceFactory;
  30. import org.aspectj.weaver.tools.WeavingAdaptor;
  31. import org.aspectj.weaver.tools.WeavingClassLoader;
  32. public class WeavingURLClassLoader extends ExtensibleURLClassLoader implements WeavingClassLoader {
  33. public static final String WEAVING_CLASS_PATH = "aj.class.path";
  34. public static final String WEAVING_ASPECT_PATH = "aj.aspect.path";
  35. private URL[] aspectURLs;
  36. private WeavingAdaptor adaptor;
  37. private boolean initializingAdaptor;
  38. private Map generatedClasses = new HashMap(); /* String -> byte[] */
  39. private static Trace trace = TraceFactory.getTraceFactory().getTrace(WeavingURLClassLoader.class);
  40. /*
  41. * This constructor is needed when using "-Djava.system.class.loader".
  42. */
  43. public WeavingURLClassLoader(ClassLoader parent) {
  44. this(getURLs(getClassPath()), getURLs(getAspectPath()), parent);
  45. // System.out.println("? WeavingURLClassLoader.WeavingURLClassLoader()");
  46. }
  47. public WeavingURLClassLoader(URL[] urls, ClassLoader parent) {
  48. super(urls, parent);
  49. if (trace.isTraceEnabled())
  50. trace.enter("<init>", this, new Object[] { urls, parent });
  51. // System.out.println("WeavingURLClassLoader.WeavingURLClassLoader()");
  52. if (trace.isTraceEnabled())
  53. trace.exit("<init>");
  54. }
  55. public WeavingURLClassLoader(URL[] classURLs, URL[] aspectURLs, ClassLoader parent) {
  56. super(classURLs, parent);
  57. // System.out.println("> WeavingURLClassLoader.WeavingURLClassLoader() classURLs=" + Arrays.asList(classURLs));
  58. this.aspectURLs = aspectURLs;
  59. /*
  60. * If either we nor our parent is using an ASPECT_PATH use a new-style adaptor
  61. */
  62. if (this.aspectURLs.length > 0 || getParent() instanceof WeavingClassLoader) {
  63. try {
  64. adaptor = new WeavingAdaptor(this);
  65. } catch (ExceptionInInitializerError ex) {
  66. ex.printStackTrace(System.out);
  67. throw ex;
  68. }
  69. }
  70. // System.out.println("< WeavingURLClassLoader.WeavingURLClassLoader() adaptor=" + adaptor);
  71. }
  72. private static String getAspectPath() {
  73. return System.getProperty(WEAVING_ASPECT_PATH, "");
  74. }
  75. private static String getClassPath() {
  76. return System.getProperty(WEAVING_CLASS_PATH, "");
  77. }
  78. private static URL[] getURLs(String path) {
  79. List urlList = new ArrayList();
  80. for (StringTokenizer t = new StringTokenizer(path, File.pathSeparator); t.hasMoreTokens();) {
  81. File f = new File(t.nextToken().trim());
  82. try {
  83. if (f.exists()) {
  84. URL url = f.toURL();
  85. if (url != null)
  86. urlList.add(url);
  87. }
  88. } catch (MalformedURLException e) {
  89. }
  90. }
  91. URL[] urls = new URL[urlList.size()];
  92. urlList.toArray(urls);
  93. return urls;
  94. }
  95. protected void addURL(URL url) {
  96. if (adaptor == null) {
  97. createAdaptor();
  98. }
  99. adaptor.addURL(url);
  100. super.addURL(url);
  101. }
  102. /**
  103. * Override to weave class using WeavingAdaptor
  104. */
  105. protected Class defineClass(String name, byte[] b, CodeSource cs) throws IOException {
  106. if (trace.isTraceEnabled())
  107. trace.enter("defineClass", this, new Object[] { name, b, cs });
  108. // System.err.println("? WeavingURLClassLoader.defineClass(" + name + ", [" + b.length + "])");
  109. byte orig[] = b;
  110. /* Avoid recursion during adaptor initialization */
  111. if (!initializingAdaptor) {
  112. /* Need to defer creation because of possible recursion during constructor execution */
  113. if (adaptor == null && !initializingAdaptor) {
  114. createAdaptor();
  115. }
  116. try {
  117. b = adaptor.weaveClass(name, b, false);
  118. } catch (AbortException ex) {
  119. trace.error("defineClass", ex);
  120. throw ex;
  121. } catch (Throwable th) {
  122. trace.error("defineClass", th);
  123. }
  124. }
  125. Class clazz;
  126. // On error, define the original form of the class and log the issue
  127. try {
  128. clazz= super.defineClass(name, b, cs);
  129. } catch (Throwable th) {
  130. trace.error("Weaving class problem. Original class has been returned. The error was caused because of: " + th, th);
  131. clazz= super.defineClass(name, orig, cs);
  132. }
  133. if (trace.isTraceEnabled())
  134. trace.exit("defineClass", clazz);
  135. return clazz;
  136. }
  137. private void createAdaptor() {
  138. DefaultWeavingContext weavingContext = new DefaultWeavingContext(this) {
  139. /* Ensures consistent LTW messages for testing */
  140. public String getClassLoaderName() {
  141. ClassLoader loader = getClassLoader();
  142. return loader.getClass().getName();
  143. }
  144. };
  145. ClassLoaderWeavingAdaptor clwAdaptor = new ClassLoaderWeavingAdaptor();
  146. initializingAdaptor = true;
  147. clwAdaptor.initialize(this, weavingContext);
  148. initializingAdaptor = false;
  149. adaptor = clwAdaptor;
  150. }
  151. /**
  152. * Override to find classes generated by WeavingAdaptor
  153. */
  154. protected byte[] getBytes(String name) throws IOException {
  155. byte[] bytes = super.getBytes(name);
  156. if (bytes == null) {
  157. // return adaptor.findClass(name);
  158. return (byte[]) generatedClasses.remove(name);
  159. }
  160. return bytes;
  161. }
  162. /**
  163. * Implement method from WeavingClassLoader
  164. */
  165. public URL[] getAspectURLs() {
  166. return aspectURLs;
  167. }
  168. public void acceptClass (String name, byte[] classBytes, byte[] weavedBytes) {
  169. generatedClasses.put(name, weavedBytes);
  170. }
  171. // protected synchronized Class loadClass(String name, boolean resolve) throws ClassNotFoundException {
  172. // System.err.println("> WeavingURLClassLoader.loadClass() name=" + name);
  173. // Class clazz= super.loadClass(name, resolve);
  174. // System.err.println("< WeavingURLClassLoader.loadClass() clazz=" + clazz + ", loader=" + clazz.getClassLoader());
  175. // return clazz;
  176. // }
  177. // private interface ClassPreProcessorAdaptor extends ClassPreProcessor {
  178. // public void addURL(URL url);
  179. // }
  180. //
  181. // private class WeavingAdaptorPreProcessor implements ClassPreProcessorAdaptor {
  182. //
  183. // private WeavingAdaptor adaptor;
  184. //
  185. // public WeavingAdaptorPreProcessor (WeavingClassLoader wcl) {
  186. // adaptor = new WeavingAdaptor(wcl);
  187. // }
  188. //
  189. // public void initialize() {
  190. // }
  191. //
  192. // public byte[] preProcess(String className, byte[] bytes, ClassLoader classLoader) {
  193. // return adaptor.weaveClass(className,bytes);
  194. // }
  195. //
  196. // public void addURL(URL url) {
  197. //
  198. // }
  199. // }
  200. }