From 3227aad604f040cdced47d7a8462acf563f52ef6 Mon Sep 17 00:00:00 2001 From: Andy Clement Date: Thu, 13 Jan 2022 17:57:51 -0800 Subject: Couple of basic tests for the repository optimizations --- .../apache/bcel/util/ClassLoaderRepository.java | 37 +++++----- .../classfile/tests/ClassloaderRepositoryTest.java | 84 ++++++++++++++++++++++ 2 files changed, 105 insertions(+), 16 deletions(-) (limited to 'bcel-builder') diff --git a/bcel-builder/src/main/java/org/aspectj/apache/bcel/util/ClassLoaderRepository.java b/bcel-builder/src/main/java/org/aspectj/apache/bcel/util/ClassLoaderRepository.java index e8d8f309e..cc3290b29 100644 --- a/bcel-builder/src/main/java/org/aspectj/apache/bcel/util/ClassLoaderRepository.java +++ b/bcel-builder/src/main/java/org/aspectj/apache/bcel/util/ClassLoaderRepository.java @@ -94,16 +94,17 @@ public class ClassLoaderRepository implements Repository { // 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"); + public static boolean useSharedCache = + System.getProperty("org.aspectj.apache.bcel.useSharedCache", "true").equalsIgnoreCase("true"); - //Cache not found classes as well to prevent unnecessary file I/O operations - public static final boolean useUnavailableClassesCache = + // Cache not found classes as well to prevent unnecessary file I/O operations + public static boolean useUnavailableClassesCache = System.getProperty("org.aspectj.apache.bcel.useUnavailableClassesCache", "false").equalsIgnoreCase("true"); - //Ignore cache clear requests to not build up the cache over and over again - public static final boolean ignoreCacheClearRequests = + // Ignore cache clear requests to not build up the cache over and over again + public static boolean ignoreCacheClearRequests = System.getProperty("org.aspectj.apache.bcel.ignoreCacheClearRequests", "false").equalsIgnoreCase("true"); - //Second cache for the unavailable classes + // Second cache for the unavailable classes private static Set unavailableClasses = new HashSet(); private static int cacheHitsShared = 0; @@ -113,6 +114,7 @@ public class ClassLoaderRepository implements Repository { private int classesLoadedCount = 0; private int misses = 0; private int cacheHitsLocal = 0; + private int unavailableClassesCacheHits = 0; private int missLocalEvicted = 0; // Misses in local cache access due to reference GC public ClassLoaderRepository(java.lang.ClassLoader loader) { @@ -290,26 +292,26 @@ public class ClassLoaderRepository implements Repository { } /** - * Lookup a JavaClass object from the Class Name provided. + * Lookup a JavaClass object from the classname provided. */ public JavaClass loadClass(String className) throws ClassNotFoundException { - - //Quick evaluation of unavailable classes to prevent unnecessary file I/O - if (useUnavailableClassesCache && unavailableClasses.contains(className)) + // Quick evaluation of unavailable classes to prevent unnecessary file I/O + if (useUnavailableClassesCache && unavailableClasses.contains(className)) { + unavailableClassesCacheHits++; throw new ClassNotFoundException(className + " not found."); + } - // translate to a URL long time = System.currentTimeMillis(); java.net.URL url = toURL(className); timeManipulatingURLs += (System.currentTimeMillis() - time); if (url == null) { - if (useUnavailableClassesCache) + if (useUnavailableClassesCache) { unavailableClasses.add(className); + } throw new ClassNotFoundException(className + " not found - unable to determine URL"); } JavaClass clazz = null; - // Look in the appropriate cache if (useSharedCache) { clazz = findClassShared(url); @@ -386,7 +388,7 @@ public class ClassLoaderRepository implements Repository { */ public long[] reportStats() { return new long[] { timeSpentLoading, timeManipulatingURLs, classesLoadedCount, cacheHitsShared, missSharedEvicted, - cacheHitsLocal, missLocalEvicted, sharedCache.size() }; + cacheHitsLocal, missLocalEvicted, sharedCache.size(), unavailableClassesCacheHits }; } /** @@ -400,6 +402,7 @@ public class ClassLoaderRepository implements Repository { cacheHitsShared = 0; missSharedEvicted = 0; missLocalEvicted = 0; + unavailableClassesCacheHits = 0; misses = 0; clear(); } @@ -411,10 +414,12 @@ public class ClassLoaderRepository implements Repository { /** Clear all entries from the local cache */ public void clear() { if (!ignoreCacheClearRequests) { - if (useSharedCache) + if (useSharedCache) { sharedCache.clear(); - else + } else { localCache.clear(); + } + unavailableClasses.clear(); } } diff --git a/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/ClassloaderRepositoryTest.java b/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/ClassloaderRepositoryTest.java index 05e9f8f73..bcf37d498 100644 --- a/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/ClassloaderRepositoryTest.java +++ b/bcel-builder/src/test/java/org/aspectj/apache/bcel/classfile/tests/ClassloaderRepositoryTest.java @@ -10,6 +10,8 @@ import junit.framework.TestCase; /* * Tests create a simple classloader repository configuration and check sharing of information. + * + * @author Andy Clement */ public class ClassloaderRepositoryTest extends TestCase { @@ -69,6 +71,88 @@ public class ClassloaderRepositoryTest extends TestCase { } } + // ClassLoaderRepository.ignoreCacheClearRequests + public void testIgnoreCacheClearRequests() throws Exception { + ClassLoaderRepository.useSharedCache = false; + try { + // the 'normal' flow with ignore in default of false + ClassLoaderRepository.ignoreCacheClearRequests = false; + try { + ClassLoaderRepository repository = setupRepository(); + repository.loadClass("java.lang.String"); + long localCacheHits = repository.reportStats()[5]; + assertEquals(0, localCacheHits); + repository.clear(); + repository.loadClass("java.lang.String"); + localCacheHits = repository.reportStats()[5]; + assertEquals(0, localCacheHits); // cache was cleared, so no hit + } finally { + ClassLoaderRepository.ignoreCacheClearRequests = false; + } + // with ignore cache clear turned on + ClassLoaderRepository.ignoreCacheClearRequests = true; + try { + ClassLoaderRepository repository = setupRepository(); + repository.loadClass("java.lang.String"); + long localCacheHits = repository.reportStats()[5]; + assertEquals(0, localCacheHits); + repository.clear(); + repository.loadClass("java.lang.String"); + localCacheHits = repository.reportStats()[5]; + assertEquals(1, localCacheHits); + } finally { + ClassLoaderRepository.ignoreCacheClearRequests = false; + } + } finally { + ClassLoaderRepository.useSharedCache = true; + } + } + + // ClassLoaderRepository.useUnavailableClassesCache + public void testUnavailableClassesCache() throws Exception { + ClassLoaderRepository.useUnavailableClassesCache = false; + try { + ClassLoaderRepository repository = setupRepository(); + attemptLoadThatWillFail(repository); + for (int i = 0; i < 1000; i++) { + attemptLoadThatWillFail(repository); + } + assertEquals(0, repository.reportStats()[8]); + } finally { + ClassLoaderRepository.useUnavailableClassesCache = false; // back to default + } + + ClassLoaderRepository.useUnavailableClassesCache = true; + try { + ClassLoaderRepository repository = setupRepository(); + assertNotNull(repository.loadClass("java.lang.String")); + attemptLoadThatWillFail(repository); + for (int i = 0; i < 1000; i++) { + attemptLoadThatWillFail(repository); + } + assertEquals(1000,repository.reportStats()[8]); + } finally { + ClassLoaderRepository.useUnavailableClassesCache = false; + } + // If checking the report stats for time spent manipulating URLs it will be massively reduced + } + + private ClassLoaderRepository setupRepository() throws Exception { + ClassLoader cl = Thread.currentThread().getContextClassLoader(); + ClassLoader res = new URLClassLoader(new URL[] {}, cl); + ClassLoaderRepository rep = new ClassLoaderRepository(res); + return rep; + } + + private void attemptLoadThatWillFail(ClassLoaderRepository repository) { + try { + repository.loadClass("this.is.made.up"); + throw new IllegalStateException("Should not have found 'this.is.made.up'"); + } catch (ClassNotFoundException cnfe) { + // ... expected ... + } + } + public void tearDown() throws Exception { super.tearDown(); System.err.println("Rep1: "+rep1.reportStats()); -- cgit v1.2.3