package org.aspectj.apache.bcel.util;
/* ====================================================================
* The Apache Software License, Version 1.1
*
* Copyright (c) 2001 The Apache Software Foundation. All rights
* reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* 3. The end-user documentation included with the redistribution,
* if any, must include the following acknowledgment:
* "This product includes software developed by the
* Apache Software Foundation (http://www.apache.org/)."
* Alternately, this acknowledgment may appear in the software itself,
* if and wherever such third-party acknowledgments normally appear.
*
* 4. The names "Apache" and "Apache Software Foundation" and
* "Apache BCEL" must not be used to endorse or promote products
* derived from this software without prior written permission. For
* written permission, please contact apache@apache.org.
*
* 5. Products derived from this software may not be called "Apache",
* "Apache BCEL", nor may "Apache" appear in their name, without
* prior written permission of the Apache Software Foundation.
*
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation. For more
* information on the Apache Software Foundation, please see
* .
*/
import java.io.IOException;
import java.io.InputStream;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.AbstractMap;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.WeakHashMap;
import org.aspectj.apache.bcel.classfile.ClassParser;
import org.aspectj.apache.bcel.classfile.JavaClass;
/**
* The repository maintains information about which classes have
* been loaded.
*
* It loads its data from the ClassLoader implementation
* passed into its constructor.
*
* @see org.aspectj.apache.bcel.Repository
*
* @version $Id: ClassLoaderRepository.java,v 1.9.4.3 2008/05/28 00:02:49 aclement Exp $
* @author M. Dahm
* @author David Dixon-Peugh
*/
public class ClassLoaderRepository implements Repository {
private static java.lang.ClassLoader bootClassLoader = null;
private ClassLoaderReference loaderRef;
// Choice of cache...
private WeakHashMap /**/localCache = new WeakHashMap();
private static SoftHashMap /**/sharedCache = new SoftHashMap(Collections.synchronizedMap(new HashMap()));
// For fast translation of the classname *intentionally not static*
private SoftHashMap /**/ nameMap = new SoftHashMap(new HashMap(), false);
public static boolean useSharedCache =
System.getProperty("org.aspectj.apache.bcel.useSharedCache","true").equalsIgnoreCase("true");
private static int cacheHitsShared = 0;
private static int missSharedEvicted = 0; // Misses in shared cache access due to reference GC
private long timeManipulatingURLs = 0L;
private long timeSpentLoading = 0L;
private int classesLoadedCount = 0;
private int misses = 0;
private int cacheHitsLocal = 0;
private int missLocalEvicted = 0; // Misses in local cache access due to reference GC
public ClassLoaderRepository(java.lang.ClassLoader loader) {
this.loaderRef = new DefaultClassLoaderReference((loader != null) ? loader : getBootClassLoader());
}
public ClassLoaderRepository(ClassLoaderReference loaderRef) {
this.loaderRef = loaderRef;
}
private static synchronized java.lang.ClassLoader getBootClassLoader() {
if (bootClassLoader == null) {
bootClassLoader = new URLClassLoader(new URL[0]);
}
return bootClassLoader;
}
// Can track back to its key
public static class SoftHashMap extends AbstractMap {
private Map map;
boolean recordMiss = true; // only interested in recording miss stats sometimes
private ReferenceQueue rq = new ReferenceQueue();
public SoftHashMap(Map map) { this.map = map; }
public SoftHashMap() { this(new HashMap()); }
public SoftHashMap(Map map, boolean b) { this(map); this.recordMiss=b;}
class SpecialValue extends SoftReference {
private final Object key;
SpecialValue(Object k,Object v) {
super(v,rq);
this.key = k;
}
}
private void processQueue() {
SpecialValue sv = null;
while ((sv = (SpecialValue)rq.poll())!=null) {
map.remove(sv.key);
}
}
public Object get(Object key) {
SpecialValue value = (SpecialValue)map.get(key);
if (value==null) return null;
if (value.get()==null) {
// it got GC'd
map.remove(value.key);
if (recordMiss) missSharedEvicted++;
return null;
} else {
return value.get();
}
}
public Object put(Object k, Object v) {
processQueue();
return map.put(k, new SpecialValue(k,v));
}
public Set entrySet() {
return map.entrySet();
}
public void clear() {
processQueue();
map.clear();
}
public int size() {
processQueue();
return map.size();
}
public Object remove(Object k) {
processQueue();
SpecialValue value = (SpecialValue)map.remove(k);
if (value==null) return null;
if (value.get()!=null) {
return value.get();
}
return null;
}
}
/**
* Store a new JavaClass into this repository as a soft reference and return the reference
*/
private void storeClassAsReference(URL url, JavaClass clazz ) {
if (useSharedCache) {
clazz.setRepository(null); // can't risk setting repository, we'll get in a pickle!
sharedCache.put(url, clazz);
} else {
clazz.setRepository(this);
localCache.put(url, new SoftReference(clazz));
}
}
/**
* Store a new JavaClass into this Repository.
*/
public void storeClass( JavaClass clazz ) {
storeClassAsReference(toURL(clazz.getClassName()),clazz);
}
/**
* Remove class from repository
*/
public void removeClass(JavaClass clazz) {
if (useSharedCache) sharedCache.remove(toURL(clazz.getClassName()));
else localCache.remove(toURL(clazz.getClassName()));
}
/**
* Find an already defined JavaClass in the local cache.
*/
public JavaClass findClass( String className ) {
if (useSharedCache) return findClassShared(toURL(className));
else return findClassLocal(toURL(className));
}
private JavaClass findClassLocal( URL url ) {
Object o = localCache.get(url);
if (o != null) {
o = ((Reference)o).get();
if (o != null) {
return (JavaClass)o;
} else {
missLocalEvicted++;
}
}
return null;
}
/**
* Find an already defined JavaClass in the shared cache.
*/
private JavaClass findClassShared(URL url) {
return (JavaClass)sharedCache.get(url);
}
private URL toURL(String className) {
URL url = (URL)nameMap.get(className);
if (url==null) {
String classFile = className.replace('.', '/');
url = loaderRef.getClassLoader().getResource(classFile + ".class");
nameMap.put(className, url);
}
return url;
}
/**
* Lookup a JavaClass object from the Class Name provided.
*/
public JavaClass loadClass( String className ) throws ClassNotFoundException {
// translate to a URL
long time = System.currentTimeMillis();
java.net.URL url = toURL(className);
timeManipulatingURLs += (System.currentTimeMillis() - time);
if (url==null) throw new ClassNotFoundException(className + " not found - unable to determine URL");
JavaClass clazz = null;
// Look in the appropriate cache
if (useSharedCache) {
clazz = findClassShared(url);
if (clazz != null) { cacheHitsShared++; return clazz; }
} else {
clazz = findClassLocal(url);
if (clazz != null) { cacheHitsLocal++; return clazz; }
}
// Didn't find it in either cache
misses++;
try {
// Load it
String classFile = className.replace('.', '/');
InputStream is = (useSharedCache ? url.openStream() : loaderRef.getClassLoader().getResourceAsStream(classFile + ".class"));
if (is == null) {
throw new ClassNotFoundException(className + " not found "+(url==null?"":"- using url "+url));
}
ClassParser parser = new ClassParser( is, className );
clazz = parser.parse();
// Cache it
storeClassAsReference(url, clazz );
timeSpentLoading += (System.currentTimeMillis() - time);
classesLoadedCount++;
return clazz;
} catch (IOException e) {
throw new ClassNotFoundException( e.toString() );
}
}
/**
* Produce a report on cache usage.
*/
public String report() {
StringBuffer sb = new StringBuffer();
sb.append("BCEL repository report.");
if (useSharedCache) sb.append(" (shared cache)");
else sb.append(" (local cache)");
sb.append(" Total time spent loading: "+timeSpentLoading+"ms.");
sb.append(" Time spent manipulating URLs: "+timeManipulatingURLs+"ms.");
sb.append(" Classes loaded: "+classesLoadedCount+".");
if (useSharedCache) {
sb.append(" Shared cache size: "+sharedCache.size());
sb.append(" Shared cache (hits/missDueToEviction): ("+cacheHitsShared+"/"+missSharedEvicted+").");
} else {
sb.append(" Local cache size: "+localCache.size());
sb.append(" Local cache (hits/missDueToEviction): ("+cacheHitsLocal+"/"+missLocalEvicted+").");
}
return sb.toString();
}
/**
* Returns an array of the stats, for testing, the order is fixed:
* 0=time spent loading (static)
* 1=time spent manipulating URLs (static)
* 2=classes loaded (static)
* 3=cache hits shared (static)
* 4=misses in shared due to eviction (static)
* 5=cache hits local
* 6=misses in local due to eviction
* 7=shared cache size
*/
public long[] reportStats() {
return new long[]{timeSpentLoading,timeManipulatingURLs,classesLoadedCount,
cacheHitsShared,missSharedEvicted,cacheHitsLocal,missLocalEvicted,
sharedCache.size()};
}
/**
* Reset statistics and clear all caches
*/
public void reset() {
timeManipulatingURLs = 0L;
timeSpentLoading = 0L;
classesLoadedCount = 0;
cacheHitsLocal = 0;
cacheHitsShared = 0;
missSharedEvicted = 0;
missLocalEvicted = 0;
misses = 0;
clear();
}
public JavaClass loadClass(Class clazz) throws ClassNotFoundException {
return loadClass(clazz.getName());
}
/** Clear all entries from the local cache */
public void clear() {
if (useSharedCache) sharedCache.clear();
else localCache.clear();
}
}