org.gradle.api.internal.project.antbuilder.ClassPathToClassLoaderCache Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-test-kit Show documentation
Show all versions of gradle-test-kit Show documentation
Gradle 6.2.1 API redistribution.
/*
* Copyright 2015 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gradle.api.internal.project.antbuilder;
import org.gradle.api.Action;
import org.gradle.api.internal.classloading.GroovySystemLoader;
import org.gradle.api.internal.classloading.GroovySystemLoaderFactory;
import org.gradle.internal.Factory;
import org.gradle.internal.UncheckedException;
import org.gradle.internal.classpath.ClassPath;
import org.gradle.internal.concurrent.Stoppable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
* A cache which caches classloaders based on their classpath. This cache provides bridging with the classloader cleanup mechanism which makes it more complex than it should: - class loaders can be
* reused, so they *must not* be cleared as long as a cached loader is in cache - once a classloader is discarded from the cache, it must be cleared using a cleanup strategy
*
* It is important that the cleanup only occurs when nobody uses the classloader anymore, which means when no consumer retains a strong reference onto a {@link CachedClassLoader}. If we directly put
* the cached classloader as a value of the map, then it cannot be reclaimed, and will never be cleaned up. If we just use a SoftReference to the cached class loader, then the reference will be
* cleared before we have a chance to clean it up. So we use a PhantomReference to the cached class loader, in addition to the soft reference, to finalize the class loader before it gets kicked off
* the cache.
*/
public class ClassPathToClassLoaderCache implements Stoppable {
private final static Logger LOG = LoggerFactory.getLogger(ClassPathToClassLoaderCache.class);
private final FinalizerThread finalizerThread;
// Protects the following fields
private final Lock lock = new ReentrantLock();
private final Map cacheEntries = new ConcurrentHashMap<>();
private final Set inUseClassLoaders = new HashSet<>();
private final GroovySystemLoaderFactory groovySystemLoaderFactory;
public ClassPathToClassLoaderCache(GroovySystemLoaderFactory groovySystemLoaderFactory) {
this.groovySystemLoaderFactory = groovySystemLoaderFactory;
this.finalizerThread = new FinalizerThread(cacheEntries, lock);
this.finalizerThread.start();
}
@Override
public void stop() {
finalizerThread.exit();
try {
finalizerThread.join();
} catch (InterruptedException e) {
throw UncheckedException.throwAsUncheckedException(e);
}
}
public int size() {
return cacheEntries.size();
}
public boolean isEmpty() {
return cacheEntries.isEmpty();
}
/**
* Provides execution of arbitrary code that consumes a cached class loader in a memory safe manner,
* that is to say making sure that concurrent calls reuse the same classloader, or that the class loader
* is retrieved from cache if available.
*
* It will also make sure that once a cached class loader is unused and removed from cache, memory cleanup
* is done.
*
* The action MUST be done on a CachedClassLoader, and not directly with the ClassLoader, in order for
* a strong reference to be kept on the cached class loader while in use. If we don't do so there are risks
* that the cached entry gets released by the GC before we've finished working with the classloader it
* wraps!
*
* @param libClasspath the classpath for this classloader
* @param gradleApiGroovy the Groovy system used by core Gradle API
* @param antBuilderAdapterGroovy the Groovy system used by the Ant builder adapter
* @param factory the factory to create a new class loader on cache miss
* @param action the action to execute with the cached class loader
*/
public void withCachedClassLoader(ClassPath libClasspath,
GroovySystemLoader gradleApiGroovy,
GroovySystemLoader antBuilderAdapterGroovy,
Factory extends ClassLoader> factory,
Action super CachedClassLoader> action) {
CachedClassLoader cachedClassLoader;
lock.lock();
try {
CacheEntry cacheEntry = cacheEntries.get(libClasspath);
cachedClassLoader = maybeGet(cacheEntry);
if (cachedClassLoader == null) {
if (LOG.isDebugEnabled()) {
LOG.debug("Classloader cache miss for classpath : {}. Creating classloader.", libClasspath.getAsURIs());
}
// Lock is held while creating ClassLoader - nothing else can happen while this is running
ClassLoader classLoader = factory.create();
cachedClassLoader = new CachedClassLoader(libClasspath, classLoader);
cacheEntry = new CacheEntry(libClasspath, cachedClassLoader);
GroovySystemLoader groovySystemForLoader = groovySystemLoaderFactory.forClassLoader(classLoader);
Cleanup cleanup = new Cleanup(libClasspath, cachedClassLoader, finalizerThread.getReferenceQueue(), classLoader, groovySystemForLoader, gradleApiGroovy, antBuilderAdapterGroovy);
finalizerThread.putCleanup(libClasspath, cleanup);
cacheEntries.put(libClasspath, cacheEntry);
} else {
if (LOG.isDebugEnabled()) {
LOG.debug("Classloader found in cache: {}", libClasspath.getAsURIs());
}
}
// in order to make sure that the CacheEntry is not collected
// while the cached class loader is still in use, we need to keep a strong reference onto
// the cached class loader as long as the action is executed
inUseClassLoaders.add(cachedClassLoader);
} finally {
lock.unlock();
}
try {
action.execute(cachedClassLoader);
} finally {
lock.lock();
try {
inUseClassLoaders.remove(cachedClassLoader);
} finally {
lock.unlock();
}
}
}
private CachedClassLoader maybeGet(CacheEntry cacheEntry) {
return cacheEntry != null ? cacheEntry.get() : null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy