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

kanela.agent.util.classloader.PreInitializeClasses Maven / Gradle / Ivy

/*
 * =========================================================================================
 * Copyright © 2013-2019 the kamon project 
 *
 * 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 kanela.agent.util.classloader;

import java.util.List;
import java.util.ArrayList;

import com.blogspot.mydailyjava.weaklockfree.WeakConcurrentMap;
import io.vavr.control.Try;
import kanela.agent.util.log.Logger;
import lombok.Value;
import lombok.val;

// This implementation is adapted from:
// https://github.com/glowroot/glowroot/blob/v0.9.20/agent/core/src/main/java/org/glowroot/agent/weaving/PreInitializeWeavingClasses.java

// "There are some things that agents are allowed to do that simply should not be permitted"
//
// -- http://mail.openjdk.java.net/pipermail/hotspot-dev/2012-March/005464.html
//
// in particular (at least prior to parallel class loading in JDK 7) initializing other classes
// inside of a ClassFileTransformer.transform() method occasionally leads to deadlocks
//
// this is still a problem in JDK 7+, since parallel class loading must be opted in to by custom
// class loaders, see ClassLoader.registerAsParallelCapable()
//
// to avoid initializing other classes inside of the transform() method, all classes referenced from
// WeavingClassFileTransformer are pre-initialized (and all classes referenced from those classes,
// etc)
@Value
public class PreInitializeClasses {
    // this is probably not needed, since preInitializeLinkedHashMapKeySetAndKeySetIterator() is
    // only called a single time, but just to be safe ...
    public static volatile Object toPreventDeadCodeElimination;

    public static void preInitializeClasses(ClassLoader loader) {
        for (String type : usedTypes()) {
            initialize(type, loader);
        }
        preExpiryMapKeySetAndKeySetIterator();
    }

    private static void initialize(String type, ClassLoader loader) {
        Try.of(() -> Class.forName(type, true, loader))
                .onFailure((cause) -> Logger.warn(() -> "class not found: " + type, cause));
    }

    public static List usedTypes() {
        val types = new ArrayList();
        types.addAll(getJavaUsedTypes());
        types.addAll(getBytebuddyUsedTypes());
        return types;
    }

    private static String prependByteBuddyPrefix(String s) {
        return "kanela.agent.libs.net.bytebuddy." + s;
    }

    private static List getJavaUsedTypes() {
        val types = new ArrayList();
        types.add("java.util.concurrent.locks.LockSupport");
        types.add("java.util.concurrent.ThreadLocalRandom");
        types.add("java.util.concurrent.locks.AbstractQueuedSynchronizer$Node");
        return types;
    }

    // List obtained from running:
    // grep -rh net.bytebuddy ./agent/src/ | sed s"/import net.bytebuddy.//" | sort -u
    // and adapted manually
    private static List getBytebuddyUsedTypes() {
        val types = new ArrayList();
        types.add(prependByteBuddyPrefix("agent.builder.ResettableClassFileTransformer"));
        types.add(prependByteBuddyPrefix("agent.ByteBuddyAgent"));
        types.add(prependByteBuddyPrefix("asm.Advice"));
        types.add(prependByteBuddyPrefix("asm.AsmVisitorWrapper"));
        types.add(prependByteBuddyPrefix("ByteBuddy"));
        types.add(prependByteBuddyPrefix("ClassFileVersion"));
        types.add(prependByteBuddyPrefix("description.ByteCodeElement"));
        types.add(prependByteBuddyPrefix("description.field.FieldDescription"));
        types.add(prependByteBuddyPrefix("description.field.FieldList"));
        types.add(prependByteBuddyPrefix("description.method.MethodDescription"));
        types.add(prependByteBuddyPrefix("description.method.MethodList"));
        types.add(prependByteBuddyPrefix("description.NamedElement"));
        types.add(prependByteBuddyPrefix("description.type.TypeDescription"));
        types.add(prependByteBuddyPrefix("dynamic.ClassFileLocator"));
        types.add(prependByteBuddyPrefix("dynamic.DynamicType"));
        types.add(prependByteBuddyPrefix("dynamic.loading.ClassInjector"));
        types.add(prependByteBuddyPrefix("dynamic.scaffold.MethodGraph"));
        types.add(prependByteBuddyPrefix("dynamic.scaffold.TypeValidation"));
        types.add(prependByteBuddyPrefix("implementation.bytecode.StackManipulation"));
        types.add(prependByteBuddyPrefix("implementation.Implementation"));
        types.add(prependByteBuddyPrefix("implementation.MethodDelegation"));
        types.add(prependByteBuddyPrefix("jar.asm.ClassReader"));
        types.add(prependByteBuddyPrefix("jar.asm.ClassVisitor"));
        types.add(prependByteBuddyPrefix("jar.asm.ClassWriter"));
        types.add(prependByteBuddyPrefix("jar.asm.commons.AdviceAdapter"));
        types.add(prependByteBuddyPrefix("jar.asm.commons.ClassRemapper"));
        types.add(prependByteBuddyPrefix("jar.asm.commons.Method"));
        types.add(prependByteBuddyPrefix("jar.asm.commons.MethodRemapper"));
        types.add(prependByteBuddyPrefix("jar.asm.commons.SimpleRemapper"));
        types.add(prependByteBuddyPrefix("jar.asm.Label"));
        types.add(prependByteBuddyPrefix("jar.asm.MethodVisitor"));
        types.add(prependByteBuddyPrefix("jar.asm.Opcodes"));
        types.add(prependByteBuddyPrefix("jar.asm.tree.ClassNode"));
        types.add(prependByteBuddyPrefix("jar.asm.tree.MethodNode"));
        types.add(prependByteBuddyPrefix("jar.asm.Type"));
        types.add(prependByteBuddyPrefix("matcher.ElementMatcher"));
        types.add(prependByteBuddyPrefix("matcher.ElementMatchers"));
        types.add(prependByteBuddyPrefix("pool.TypePool"));
        types.add(prependByteBuddyPrefix("utility.JavaModule"));
        types.add(prependByteBuddyPrefix("utility.OpenedClassReader"));
        return types;
    }

    private static void preExpiryMapKeySetAndKeySetIterator() {
        toPreventDeadCodeElimination = new WeakConcurrentMap(false).iterator();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy