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

mockit.external.asm.BootstrapMethods Maven / Gradle / Ivy

Go to download

JMockit is a Java toolkit for automated developer testing. It contains APIs for the creation of the objects to be tested, for mocking dependencies, and for faking external APIs; JUnit (4 & 5) and TestNG test runners are supported. It also contains an advanced code coverage tool.

There is a newer version: 1.49
Show newest version
package mockit.external.asm;

import javax.annotation.*;

/**
 * Generates the "BootstrapMethods" attribute in a class file being written by a {@link ClassWriter}.
 */
final class BootstrapMethods
{
   @Nonnull private final ConstantPoolGeneration cp;

   /**
    * The number of entries in the BootstrapMethods attribute.
    */
   @Nonnegative private final int bootstrapMethodsCount;

   /**
    * The BootstrapMethods attribute.
    */
   @Nonnull private final ByteVector bootstrapMethods;

   @Nonnegative private final int bsmStartCodeIndex;

   BootstrapMethods(@Nonnull ConstantPoolGeneration cp, @Nonnull ClassReader cr) {
      this.cp = cp;

      int attrSize = cr.readInt();
      bootstrapMethods = new ByteVector(attrSize + 62);
      bootstrapMethodsCount = cr.readUnsignedShort();

      bsmStartCodeIndex = cr.codeIndex;
      bootstrapMethods.putByteArray(cr.code, bsmStartCodeIndex, attrSize - 2);
   }

   /**
    * Copies the bootstrap method data from the given {@link ClassReader}.
    */
   void copyBootstrapMethods(@Nonnull ClassReader cr, @Nonnull Item[] items) {
      int previousCodeIndex = cr.codeIndex;
      cr.codeIndex = bsmStartCodeIndex;

      for (int bsmIndex = 0; bsmIndex < bootstrapMethodsCount; bsmIndex++) {
         copyBootstrapMethod(cr, items, bsmIndex);
      }

      cr.codeIndex = previousCodeIndex;
   }

   private void copyBootstrapMethod(@Nonnull ClassReader cr, @Nonnull Item[] items, @Nonnegative int bsmIndex) {
      int position = cr.codeIndex - bsmStartCodeIndex;
      MethodHandle bsm = cr.readMethodHandle();
      int hashCode = bsm.hashCode();

      for (int bsmArgCount = cr.readUnsignedShort(); bsmArgCount > 0; bsmArgCount--) {
         Object bsmArg = cr.readConstItem();
         hashCode ^= bsmArg.hashCode();
      }

      BootstrapMethodItem item = new BootstrapMethodItem(bsmIndex, position, hashCode);
      item.setNext(items);
   }

   /**
    * Adds an invokedynamic reference to the constant pool of the class being built.
    * Does nothing if the constant pool already contains a similar item.
    *
    * @param name    name of the invoked method.
    * @param desc    descriptor of the invoke method.
    * @param bsm     the bootstrap method.
    * @param bsmArgs the bootstrap method constant arguments.
    * @return a new or an already existing invokedynamic type reference item.
    */
   @Nonnull
   InvokeDynamicItem addInvokeDynamicReference(
      @Nonnull String name, @Nonnull String desc, @Nonnull MethodHandle bsm, @Nonnull Object... bsmArgs
   ) {
      ByteVector methods = bootstrapMethods;
      int position = methods.length; // record current position

      MethodHandleItem methodHandleItem = cp.newMethodHandleItem(bsm);
      methods.putShort(methodHandleItem.index);

      int argsLength = bsmArgs.length;
      methods.putShort(argsLength);

      int hashCode = bsm.hashCode();
      hashCode = putBSMArgs(hashCode, bsmArgs);
      hashCode &= 0x7FFFFFFF;

      methods.length = position; // revert to old position

      BootstrapMethodItem bsmItem = getBSMItem(hashCode);
      InvokeDynamicItem result = cp.createInvokeDynamicItem(name, desc, bsmItem.index);
      return result;
   }

   private int putBSMArgs(int hashCode, @Nonnull Object[] bsmArgs) {
      for (Object bsmArg : bsmArgs) {
         hashCode ^= bsmArg.hashCode();

         Item constItem = cp.newConstItem(bsmArg);
         bootstrapMethods.putShort(constItem.index);
      }

      return hashCode;
   }

   @Nonnull
   private BootstrapMethodItem getBSMItem(int hashCode) {
      Item item = cp.getItem(hashCode);

      while (item != null) {
         if ((item instanceof BootstrapMethodItem) && item.hashCode == hashCode) {
            return (BootstrapMethodItem) item;
         }

         item = item.next;
      }

      throw new IllegalStateException("BootstrapMethodItem not found for hash code " + hashCode);
   }

   @Nonnegative
   int getSize() {
      cp.newUTF8("BootstrapMethods");
      return 8 + bootstrapMethods.length;
   }

   void put(@Nonnull ByteVector out) {
      out.putShort(cp.newUTF8("BootstrapMethods"));
      out.putInt(bootstrapMethods.length + 2).putShort(bootstrapMethodsCount);
      out.putByteVector(bootstrapMethods);
   }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy