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

com.autonomousapps.internal.asm.kt Maven / Gradle / Ivy

There is a newer version: 2.7.0
Show newest version
// Copyright (c) 2024. Tony Robalik.
// SPDX-License-Identifier: Apache-2.0
package com.autonomousapps.internal

import com.autonomousapps.Flags
import com.autonomousapps.internal.ClassNames.canonicalize
import com.autonomousapps.internal.asm.*
import com.autonomousapps.internal.kotlin.AccessFlags
import com.autonomousapps.internal.utils.METHOD_DESCRIPTOR_REGEX
import com.autonomousapps.internal.utils.efficient
import com.autonomousapps.internal.utils.genericTypes
import com.autonomousapps.model.internal.intermediates.consumer.MemberAccess
import org.gradle.api.logging.Logger
import java.util.SortedSet
import java.util.concurrent.atomic.AtomicReference
import kotlin.metadata.jvm.Metadata

private val logDebug: Boolean get() = Flags.logBytecodeDebug()
private const val ASM_VERSION = Opcodes.ASM9

/** This will collect the class name and information about annotations. */
internal class ClassNameAndAnnotationsVisitor(private val logger: Logger) : ClassVisitor(ASM_VERSION) {

  private lateinit var className: String
  private lateinit var access: Access
  private var outerClassName: String? = null
  private var superClassName: String? = null

  // private var interfaces: Set? = null
  private val retentionPolicyHolder = AtomicReference("")
  private var isAnnotation = false
  private val methods = mutableSetOf()
  private val innerClasses = mutableSetOf()
  // private val effectivelyPublicFields = mutableSetOf()
  // private val effectivelyPublicMethods = mutableSetOf()

  private var methodCount = 0
  private var fieldCount = 0

  // From old ConstantVisitor
  private val constantClasses = mutableSetOf()

  internal fun getAnalyzedClass(): AnalyzedClass {
    val className = this.className
    val access = this.access
    val hasNoMembers = fieldCount == 0 && methodCount == 0

    return AnalyzedClass(
      className = className,
      outerClassName = outerClassName,
      superClassName = superClassName,
      // interfaces = interfaces.orEmpty(),
      retentionPolicy = retentionPolicyHolder.get(),
      isAnnotation = isAnnotation,
      hasNoMembers = hasNoMembers,
      access = access,
      methods = methods.efficient(),
      innerClasses = innerClasses.efficient(),
      constantClasses = constantClasses.efficient(),
      // effectivelyPublicFields = effectivelyPublicFields,
      // effectivelyPublicMethods = effectivelyPublicMethods,
    )
  }

  override fun visit(
    version: Int,
    access: Int,
    name: String,
    signature: String?,
    superName: String?,
    interfaces: Array?
  ) {
    // This _must_ not be canonicalized, unless we also change accesses to be dotty instead of slashy
    this.superClassName = superName
    // this.interfaces = interfaces?.toSortedSet().orEmpty()

    className = canonicalize(name)
    if (interfaces?.contains("java/lang/annotation/Annotation") == true) {
      isAnnotation = true
    }
    this.access = Access.fromInt(access)

    val implementsClause = if (interfaces.isNullOrEmpty()) {
      ""
    } else {
      " implements ${interfaces.joinToString(", ")}"
    }
    log { "ClassNameAndAnnotationsVisitor#visit: ${this.access} $name extends $superName$implementsClause" }
  }

  override fun visitAnnotation(descriptor: String?, visible: Boolean): AnnotationVisitor? {
    if ("Ljava/lang/annotation/Retention;" == descriptor) {
      log { "- ClassNameAndAnnotationsVisitor#visitAnnotation ($className): descriptor=$descriptor visible=$visible" }
      return RetentionPolicyAnnotationVisitor(logger, className, retentionPolicyHolder)
    }
    return null
  }

  override fun visitMethod(
    access: Int, name: String, descriptor: String, signature: String?, exceptions: Array?
  ): MethodVisitor? {
    log { "- visitMethod: ${Access.fromInt(access)} descriptor=$descriptor name=$name signature=$signature" }

    if (!("()V" == descriptor && ("" == name || "" == name))) {
      // ignore constructors and static initializers
      methodCount++
      methods.add(Method(descriptor))
    }

    // if (isEffectivelyPublic(access)) {
    //   effectivelyPublicMethods.add(
    //     Member.Method(
    //       access = access,
    //       name = name,
    //       descriptor = descriptor,
    //     )
    //   )
    // }

    return null
  }

  override fun visitField(
    access: Int, name: String, descriptor: String, signature: String?, value: Any?
  ): FieldVisitor? {
    log { "- visitField: ${Access.fromInt(access)} descriptor=$descriptor name=$name signature=$signature value=$value" }
    fieldCount++

    // from old ConstantVisitor
    if (isStaticFinal(access)) {
      constantClasses.add(name)
    }

    // if (isEffectivelyPublic(access)) {
    //   effectivelyPublicFields.add(
    //     Member.Field(
    //       access = access,
    //       name = name,
    //       descriptor = descriptor,
    //     )
    //   )
    // }

    return null
  }

  override fun visitOuterClass(owner: String?, name: String?, descriptor: String?) {
    log { "- visitOuterClass: owner=$owner name=$name descriptor=$descriptor" }
  }

  override fun visitInnerClass(name: String, outerName: String?, innerName: String?, access: Int) {
    log { "- visitInnerClass: ${Access.fromInt(access)} name=$name outerName=$outerName innerName=$innerName" }
    if (outerName != null) {
      outerClassName = canonicalize(outerName)
    }
    innerClasses.add(canonicalize(name))
  }

  override fun visitSource(source: String?, debug: String?) {
    log { "- visitSource: source=$source debug=$debug" }
  }

  override fun visitEnd() {
    log { "- visitEnd: fieldCount=$fieldCount methodCount=$methodCount" }
  }

  private fun log(msgProvider: () -> String) {
    if (!logDebug) {
      logger.quiet(msgProvider())
    }
  }

  private class RetentionPolicyAnnotationVisitor(
    private val logger: Logger,
    private val className: String?,
    private val retentionPolicyHolder: AtomicReference
  ) : AnnotationVisitor(ASM_VERSION) {

    private fun log(msgProvider: () -> String) {
      if (!logDebug) {
        logger.quiet(msgProvider())
      }
    }

    override fun visitEnum(name: String?, descriptor: String?, value: String?) {
      if ("Ljava/lang/annotation/RetentionPolicy;" == descriptor) {
        log { "  - RetentionPolicyAnnotationVisitor#visitEnum ($className): $value" }
        retentionPolicyHolder.set(value)
      }
    }
  }
}

internal data class ClassRef(
  val classRef: String,
  val kind: Kind,
) : Comparable {

  enum class Kind {
    ANNOTATION_VISIBLE,
    ANNOTATION_HIDDEN,
    NOT_ANNOTATION,
    ;

    companion object {
      fun annotation(visible: Boolean): Kind = if (visible) ANNOTATION_VISIBLE else ANNOTATION_HIDDEN
    }
  }

  override fun compareTo(other: ClassRef): Int {
    return compareBy { it.classRef }
      .thenBy { it.kind }
      .compare(this, other)
  }
}

/**
 * This will collect the class name and the name of all classes used by this class and the methods of this class.
 */
internal class ClassAnalyzer(private val logger: Logger) : ClassVisitor(ASM_VERSION) {

  var source: String? = null
  lateinit var className: String
  val classes = mutableSetOf()
  private val binaryClasses = mutableMapOf>()

  private val methodAnalyzer = MethodAnalyzer(logger, classes, binaryClasses)
  private val fieldAnalyzer = FieldAnalyzer(logger, classes)

  fun getBinaryClasses(): Map> = binaryClasses

  private fun addClass(className: String?, kind: ClassRef.Kind) {
    classes.addClass(className, kind)
  }

  private fun log(msgProvider: () -> String) {
    if (!logDebug) {
      logger.quiet(msgProvider())
    }
  }

  override fun visitSource(source: String?, debug: String?) {
    log { "- visitSource: source=$source debug=$debug" }
    this.source = source
  }

  override fun visit(
    version: Int,
    access: Int,
    name: String,
    signature: String?,
    superName: String,
    interfaces: Array?
  ) {
    log { "ClassAnalyzer#visit: ${Access.fromInt(access)} $name extends $superName" }
    className = name

    addClass("L$superName;", ClassRef.Kind.NOT_ANNOTATION)
    interfaces?.forEach { i ->
      addClass("L$i;", ClassRef.Kind.NOT_ANNOTATION)
    }
  }

  override fun visitField(
    access: Int,
    name: String,
    descriptor: String,
    signature: String?,
    value: Any?
  ): FieldVisitor {
    log { "ClassAnalyzer#visitField: ${Access.fromInt(access)} $descriptor $name" }
    addClass(descriptor, ClassRef.Kind.NOT_ANNOTATION)

    // TODO probably do this for other `visitX` methods as well
    signature?.genericTypes()?.forEach {
      addClass(it, ClassRef.Kind.NOT_ANNOTATION)
    }

    return fieldAnalyzer
  }

  override fun visitMethod(
    access: Int,
    name: String,
    descriptor: String,
    signature: String?,
    exceptions: Array?
  ): MethodVisitor {
    log { "ClassAnalyzer#visitMethod: ${Access.fromInt(access)} $name $descriptor" }

    METHOD_DESCRIPTOR_REGEX.findAll(descriptor).forEach { result ->
      addClass(result.value, ClassRef.Kind.NOT_ANNOTATION)
    }

    return methodAnalyzer
  }

  override fun visitAnnotation(descriptor: String?, visible: Boolean): AnnotationVisitor {
    log { "ClassAnalyzer#visitAnnotation: descriptor=$descriptor visible=$visible" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitTypeAnnotation(
    typeRef: Int,
    typePath: TypePath?,
    descriptor: String?,
    visible: Boolean
  ): AnnotationVisitor {
    log { "ClassAnalyzer#visitTypeAnnotation: typeRef=$typeRef typePath=$typePath descriptor=$descriptor visible=$visible" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitEnd() {
    log { "\n" }
  }
}

private class MethodAnalyzer(
  private val logger: Logger,
  private val classes: MutableSet,
  private val binaryClasses: MutableMap>,
) : MethodVisitor(ASM_VERSION) {

  private fun addClass(className: String?, kind: ClassRef.Kind) {
    classes.addClass(className, kind)
  }

  private fun log(msgProvider: () -> String) {
    if (!logDebug) {
      logger.quiet(msgProvider())
    }
  }

  override fun visitTypeInsn(opcode: Int, type: String?) {
    log { "- MethodAnalyzer#visitTypeInsn: $type" }

    // Type can look like `java/lang/Enum` or `[Lcom/package/Thing;`, which is fucking weird
    addClass(if (type?.startsWith("[") == true) type else "L$type;", ClassRef.Kind.NOT_ANNOTATION)
  }

  override fun visitFieldInsn(opcode: Int, owner: String, name: String, descriptor: String) {
    log { "- MethodAnalyzer#visitFieldInsn: $owner.$name $descriptor" }

    val field = MemberAccess.Field(
      owner = owner,
      name = name,
      descriptor = descriptor,
    )
    binaryClasses.merge(owner, sortedSetOf(field)) { acc, inc ->
      acc.apply { addAll(inc) }
    }

    addClass("L$owner;", ClassRef.Kind.NOT_ANNOTATION)
    addClass(descriptor, ClassRef.Kind.NOT_ANNOTATION)
  }

  override fun visitMethodInsn(
    opcode: Int,
    owner: String,
    name: String,
    descriptor: String,
    isInterface: Boolean
  ) {
    log { "- MethodAnalyzer#visitMethodInsn: $owner.$name $descriptor" }

    // Owner can look like `java/lang/Enum` or `[Lcom/package/Thing;`, which is fucking weird
    addClass(if (owner.startsWith("[")) owner else "L$owner;", ClassRef.Kind.NOT_ANNOTATION)
    METHOD_DESCRIPTOR_REGEX.findAll(descriptor).forEach { result ->
      addClass(result.value, ClassRef.Kind.NOT_ANNOTATION)
    }

    val method = MemberAccess.Method(
      owner = owner,
      name = name,
      descriptor = descriptor,
    )
    binaryClasses.merge(owner, sortedSetOf(method)) { acc, inc ->
      acc.apply { addAll(inc) }
    }
  }

  override fun visitInvokeDynamicInsn(
    name: String?,
    descriptor: String?,
    bootstrapMethodHandle: Handle?,
    vararg bootstrapMethodArguments: Any?
  ) {
    log { "- MethodAnalyzer#visitInvokeDynamicInsn: $name $descriptor" }
    addClass(descriptor, ClassRef.Kind.NOT_ANNOTATION)
  }

  override fun visitLocalVariable(
    name: String?,
    descriptor: String?,
    signature: String?,
    start: Label?,
    end: Label?,
    index: Int
  ) {
    log { "- MethodAnalyzer#visitLocalVariable: $name $descriptor" }
    // TODO probably do this for other `visitX` methods as well
    signature?.genericTypes()?.forEach {
      addClass(it, ClassRef.Kind.NOT_ANNOTATION)
    }
    addClass(descriptor, ClassRef.Kind.NOT_ANNOTATION)
  }

  override fun visitLocalVariableAnnotation(
    typeRef: Int,
    typePath: TypePath?,
    start: Array?,
    end: Array?,
    index: IntArray?,
    descriptor: String?,
    visible: Boolean
  ): AnnotationVisitor {
    log { "- MethodAnalyzer#visitLocalVariableAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.NOT_ANNOTATION)
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitAnnotation(descriptor: String?, visible: Boolean): AnnotationVisitor {
    log { "- MethodAnalyzer#visitAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitInsnAnnotation(
    typeRef: Int,
    typePath: TypePath?,
    descriptor: String?,
    visible: Boolean
  ): AnnotationVisitor {
    log { "- MethodAnalyzer#visitInsnAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitParameterAnnotation(parameter: Int, descriptor: String?, visible: Boolean): AnnotationVisitor {
    log { "- MethodAnalyzer#visitParameterAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.ANNOTATION_VISIBLE)
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitTypeAnnotation(
    typeRef: Int,
    typePath: TypePath?,
    descriptor: String?,
    visible: Boolean
  ): AnnotationVisitor {
    log { "- MethodAnalyzer#visitTypeAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }

  override fun visitTryCatchBlock(start: Label?, end: Label?, handler: Label?, type: String?) {
    log { "- MethodAnalyzer#visitTryCatchBlock: $type" }
    addClass("L$type;", ClassRef.Kind.NOT_ANNOTATION)
  }

  override fun visitTryCatchAnnotation(
    typeRef: Int,
    typePath: TypePath?,
    descriptor: String?,
    visible: Boolean
  ): AnnotationVisitor {
    log { "- MethodAnalyzer#visitTryCatchAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }
}

private class AnnotationAnalyzer(
  private val visible: Boolean,
  private val logger: Logger,
  private val classes: MutableSet,
  private val level: Int = 0,
  private val arrayName: String? = null
) : AnnotationVisitor(ASM_VERSION) {

  private var arraySize = 0
  private var isTypeAlias = false
  private val arrayElements = mutableSetOf()

  // If this is a visible annotation, then internal references are needed at runtime (as well as compile time).
  // Our poor shorthand for modeling that is to say that reference is a `Kind.NOT_ANNOTATION`
  // nb: this is intentionally if confusingly different from the behavior of `ClassRef.Kind.annotation(visible)`
  private val kind = if (visible) ClassRef.Kind.NOT_ANNOTATION else ClassRef.Kind.ANNOTATION_VISIBLE

  private fun addClass(className: String?, kind: ClassRef.Kind) {
    classes.addClass(className, kind)

    if (arrayName == "d2") {
      arrayElements.addClass(className, kind)
    }
  }

  private fun log(msgProvider: () -> String) {
    if (!logDebug) {
      logger.quiet(msgProvider())
    }
  }

  private fun indent() = "  ".repeat(level)

  override fun visit(name: String?, value: Any?) {
    val valueString = stringValueOfArrayElement(value)
    log { "${indent()}- AnnotationAnalyzer#visit: name=$name, value=(${value?.javaClass?.simpleName}, ${valueString})" }

    if (arrayName != null) {
      arraySize++
      if (valueString == "alias") {
        isTypeAlias = true
      }
    }

    if (value is String) {
      METHOD_DESCRIPTOR_REGEX.findAll(value).forEach { result ->
        addClass(result.value, kind)
      }
    } else if (value is Type) {
      addClass(value.descriptor, kind)
    }
  }

  override fun visitEnum(name: String?, descriptor: String?, value: String?) {
    log { "${indent()}- AnnotationAnalyzer#visitEnum: name=$name, descriptor=$descriptor, value=$value" }
    addClass(descriptor, kind)
  }

  override fun visitAnnotation(name: String?, descriptor: String?): AnnotationVisitor {
    log { "${indent()}- AnnotationAnalyzer#visitAnnotation: name=$name, descriptor=$descriptor" }
    addClass(descriptor, kind)
    return AnnotationAnalyzer(visible, logger, classes, level + 1)
  }

  override fun visitArray(name: String?): AnnotationVisitor {
    log { "${indent()}- AnnotationAnalyzer#visitArray: name=$name" }
    return AnnotationAnalyzer(if (name == "d2") false else visible, logger, classes, level + 1, name)
  }

  override fun visitEnd() {
    if (isTypeAlias()) {
      // Transform the "Kind.ANNOTATION" references into "Kind.NOT_ANNOTATION" references so that our
      // "is this typealias used?" algorithm works.
      classes.addAll(arrayElements.map { ClassRef(it.classRef, ClassRef.Kind.NOT_ANNOTATION) })
    }
  }

  // The elements of the array will look like... (:)
  // { String:MyAlias, String:L/com/example/Aliased;, String:alias }
  private fun isTypeAlias() = arraySize == 3 && isTypeAlias
}

private class FieldAnalyzer(
  private val logger: Logger,
  private val classes: MutableSet,
) : FieldVisitor(ASM_VERSION) {

  private fun log(msgProvider: () -> String) {
    if (!logDebug) {
      logger.quiet(msgProvider())
    }
  }

  private fun addClass(className: String?, kind: ClassRef.Kind) {
    classes.addClass(className, kind)
  }

  override fun visitAnnotation(descriptor: String?, visible: Boolean): AnnotationVisitor {
    log { "- FieldAnalyzer#visitAnnotation: $descriptor" }
    addClass(descriptor, ClassRef.Kind.annotation(visible))
    return AnnotationAnalyzer(visible, logger, classes)
  }
}

private fun MutableSet.addClass(classRef: String?, kind: ClassRef.Kind) {
  classRef?.let {
    // Strip array indicators
    it.replace("[", "")
    // Only add class types (not primitives)
    if (it.startsWith("L")) {
      add(ClassRef(it.substring(1, it.length - 1), kind))
    }
  }
}

/* ===================================================================================================
 * Below here used for parsing kotlin.Metadata with the ultimate goal of listing all inline functions.
 * ===================================================================================================
 */

internal class KotlinClassHeaderBuilder {

  var kind: Int = 1
  var metadataVersion: IntArray? = null
  var bytecodeVersion: IntArray? = null
  var data1 = mutableListOf()
  var data2 = mutableListOf()
  var extraString: String? = null
  var packageName: String? = null
  var extraInt: Int = 0

  fun build(): Metadata {
    return Metadata(
      kind = kind,
      metadataVersion = metadataVersion,
      data1 = data1.toTypedArray(),
      data2 = data2.toTypedArray(),
      extraString = extraString,
      packageName = packageName,
      extraInt = extraInt
    )
  }
}

private const val KOTLIN_METADATA = "Lkotlin/Metadata;"

internal class KotlinMetadataVisitor(
  private val logger: Logger
) : ClassVisitor(ASM_VERSION) {

  internal lateinit var className: String
  internal var builder: KotlinClassHeaderBuilder? = null

  private fun log(msgProvider: () -> String) {
    if (!logDebug) {
      logger.quiet(msgProvider())
    }
  }

  override fun visit(
    version: Int,
    access: Int,
    name: String,
    signature: String?,
    superName: String?,
    interfaces: Array?
  ) {
    log { "KotlinMetadataVisitor#visit: $name extends $superName" }
    className = name
  }

  override fun visitAnnotation(descriptor: String?, visible: Boolean): AnnotationVisitor? {
    log { "KotlinMetadataVisitor#visitAnnotation: descriptor=$descriptor visible=$visible" }
    return if (KOTLIN_METADATA == descriptor) {
      builder = KotlinClassHeaderBuilder()
      KotlinAnnotationVisitor(logger, builder!!)
    } else {
      null
    }
  }

  private class KotlinAnnotationVisitor(
    private val logger: Logger,
    private val builder: KotlinClassHeaderBuilder,
    private val level: Int = 0,
    private val arrayName: String? = null
  ) : AnnotationVisitor(ASM_VERSION) {

    private fun log(msgProvider: () -> String) {
      if (!logDebug) {
        logger.quiet(msgProvider())
      }
    }

    private fun indent() = "  ".repeat(level)

    override fun visit(name: String?, value: Any?) {
      log { "${indent()}- visit: name=$name, value=(${value?.javaClass?.simpleName}, ${stringValueOfArrayElement(value)})" }

      when (name) {
        "k" -> builder.kind = value as Int
        "mv" -> builder.metadataVersion = value as IntArray
        "bv" -> builder.bytecodeVersion = value as IntArray
        "xs" -> builder.extraString = value as String
        "pn" -> builder.packageName = value as String
        "xi" -> builder.extraInt = value as Int
      }

      when (arrayName) {
        "d1" -> builder.data1.add(value as String)
        "d2" -> builder.data2.add(value as String)
      }
    }

    override fun visitArray(name: String?): AnnotationVisitor {
      log { "${indent()}- visitArray: name=$name" }
      return KotlinAnnotationVisitor(logger, builder, level + 1, name)
    }
  }
}

fun stringValueOfArrayElement(value: Any?): String {
  return if (value is String && value.contains("\n")) {
    "..."
  } else {
    value.toString()
  }
}

private fun isStaticFinal(access: Int): Boolean =
  access and Opcodes.ACC_STATIC != 0 && access and Opcodes.ACC_FINAL != 0

private fun isEffectivelyPublic(access: Int): Boolean {
  val flags = AccessFlags(access)
  return flags.isPublic || flags.isProtected
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy