All Downloads are FREE. Search and download functionalities are using the official Maven repository.

proguard.classfile.visitor.ParallelAllClassVisitor Maven / Gradle / Ivy

Go to download

ProGuardCORE is a free library to read, analyze, modify, and write Java class files.

There is a newer version: 9.1.7
Show newest version
/*
 * ProGuardCORE -- library to process Java bytecode.
 *
 * Copyright (c) 2002-2020 Guardsquare NV
 *
 * 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 proguard.classfile.visitor;

import proguard.classfile.*;

import java.util.*;
import java.util.concurrent.*;


/**
 * This {@link ClassPoolVisitor} will visit all Clazz instances of the class pool
 * in a parallel way. For each thread, a separate {@link ClassVisitor} will be
 * created using {@link ClassVisitorFactory#createClassVisitor()}.
 * 

* The number of parallel threads is coupled to the number of available * processors: *

 *     parallel_threads = Runtime.getRuntime().availableProcessors() - 1;
 * 
*

* It is possible to override the number of threads by setting the * environment variable {@code parallel.threads} to an integer > 0. * * @author Thomas Neidhart */ public class ParallelAllClassVisitor implements ClassPoolVisitor { private static final int THREAD_COUNT; static { Integer threads = null; try { String threadCountString = System.getProperty("parallel.threads"); if (threadCountString != null) { threads = Integer.parseInt(threadCountString); } } catch (Exception ex) {} threads = threads == null ? Runtime.getRuntime().availableProcessors() - 1 : Math.min(threads.intValue(), Runtime.getRuntime().availableProcessors()); THREAD_COUNT = threads.intValue(); } /** * A factory for {@link ClassVisitor} objects. */ public interface ClassVisitorFactory { /** * Creates a ClassVisitor that will be used during * parallel visiting of classes in a ClassPool. */ ClassVisitor createClassVisitor(); } private final ClassVisitorFactory classVisitorFactory; /** * Create a new ParallelAllClassVisitor that will use the given factory * to visit all classes in a ClassPool in a parallel way. */ public ParallelAllClassVisitor(ClassVisitorFactory classVisitorFactory) { this.classVisitorFactory = classVisitorFactory; } // Implementations for ClassPoolVisitor. public void visitClassPool(ClassPool classPool) { if (THREAD_COUNT <= 1) { // Fallback to single thread execution if the thread count // was overridden by an environment variable. classPool.classesAccept(classVisitorFactory.createClassVisitor()); } else { ExecutorService executor = Executors.newFixedThreadPool(THREAD_COUNT, new MyThreadFactory()); MyThreadedClassVisitor classVisitor = new MyThreadedClassVisitor(executor); classPool.classesAccept(classVisitor); try { // Shutdown the executor service to release memory. executor.shutdown(); // Rethrow any exception that was thrown in the executor threads. classVisitor.awaitTermination(); } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw new RuntimeException("Parallel execution is taking too long", e); } catch (ExecutionException e) { throw e.getCause() instanceof RuntimeException ? (RuntimeException)e.getCause() : new RuntimeException(e.getCause()); } } } private class MyThreadFactory implements ThreadFactory { private int threadCounter = 0; public Thread newThread(Runnable runnable) { return new MyClassVisitorThread(++threadCounter, runnable); } } private class MyClassVisitorThread extends Thread { private final ClassVisitor classVisitor = classVisitorFactory.createClassVisitor(); public MyClassVisitorThread(int counter, Runnable runnable) { super(runnable, "Parallel Class Visitor " + counter); } } private static class MyThreadedClassVisitor implements ClassVisitor { private final ExecutorService executorService; private final List futures = new ArrayList<>(); public MyThreadedClassVisitor(ExecutorService executorService) { this.executorService = executorService; } public void awaitTermination() throws ExecutionException, InterruptedException { for (Future future : futures) { future.get(); } } // Implementations for ClassVisitor. @Override public void visitAnyClass(Clazz clazz) { submitClassToExecutorService(clazz); } private void submitClassToExecutorService(final Clazz clazz) { futures.add(executorService.submit(() -> { MyClassVisitorThread thread = (MyClassVisitorThread)Thread.currentThread(); clazz.accept(thread.classVisitor); })); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy