Instrumenter.java

  1. /*******************************************************************************
  2.  * Copyright (c) 2009, 2025 Mountainminds GmbH & Co. KG and Contributors
  3.  * This program and the accompanying materials are made available under
  4.  * the terms of the Eclipse Public License 2.0 which is available at
  5.  * http://www.eclipse.org/legal/epl-2.0
  6.  *
  7.  * SPDX-License-Identifier: EPL-2.0
  8.  *
  9.  * Contributors:
  10.  *    Marc R. Hoffmann - initial API and implementation
  11.  *
  12.  *******************************************************************************/
  13. package org.jacoco.core.instr;

  14. import java.io.ByteArrayOutputStream;
  15. import java.io.IOException;
  16. import java.io.InputStream;
  17. import java.io.OutputStream;
  18. import java.util.zip.CRC32;
  19. import java.util.zip.GZIPInputStream;
  20. import java.util.zip.GZIPOutputStream;
  21. import java.util.zip.ZipEntry;
  22. import java.util.zip.ZipInputStream;
  23. import java.util.zip.ZipOutputStream;

  24. import org.jacoco.core.JaCoCo;
  25. import org.jacoco.core.internal.ContentTypeDetector;
  26. import org.jacoco.core.internal.InputStreams;
  27. import org.jacoco.core.internal.Pack200Streams;
  28. import org.jacoco.core.internal.data.CRC64;
  29. import org.jacoco.core.internal.flow.ClassProbesAdapter;
  30. import org.jacoco.core.internal.instr.ClassInstrumenter;
  31. import org.jacoco.core.internal.instr.IProbeArrayStrategy;
  32. import org.jacoco.core.internal.instr.InstrSupport;
  33. import org.jacoco.core.internal.instr.ProbeArrayStrategyFactory;
  34. import org.jacoco.core.internal.instr.SignatureRemover;
  35. import org.jacoco.core.runtime.IExecutionDataAccessorGenerator;
  36. import org.objectweb.asm.ClassReader;
  37. import org.objectweb.asm.ClassVisitor;
  38. import org.objectweb.asm.ClassWriter;

  39. /**
  40.  * Several APIs to instrument Java class definitions for coverage tracing.
  41.  */
  42. public class Instrumenter {

  43.     private final IExecutionDataAccessorGenerator accessorGenerator;

  44.     private final SignatureRemover signatureRemover;

  45.     /**
  46.      * Creates a new instance based on the given runtime.
  47.      *
  48.      * @param runtime
  49.      *            runtime used by the instrumented classes
  50.      */
  51.     public Instrumenter(final IExecutionDataAccessorGenerator runtime) {
  52.         this.accessorGenerator = runtime;
  53.         this.signatureRemover = new SignatureRemover();
  54.     }

  55.     /**
  56.      * Determines whether signatures should be removed from JAR files. This is
  57.      * typically necessary as instrumentation modifies the class files and
  58.      * therefore invalidates existing JAR signatures. Default is
  59.      * <code>true</code>.
  60.      *
  61.      * @param flag
  62.      *            <code>true</code> if signatures should be removed
  63.      */
  64.     public void setRemoveSignatures(final boolean flag) {
  65.         signatureRemover.setActive(flag);
  66.     }

  67.     private byte[] instrument(final byte[] source) {
  68.         final long classId = CRC64.classId(source);
  69.         final ClassReader reader = InstrSupport.classReaderFor(source);
  70.         final ClassWriter writer = new ClassWriter(reader, 0) {
  71.             @Override
  72.             protected String getCommonSuperClass(final String type1,
  73.                     final String type2) {
  74.                 throw new IllegalStateException();
  75.             }
  76.         };
  77.         final IProbeArrayStrategy strategy = ProbeArrayStrategyFactory
  78.                 .createFor(classId, reader, accessorGenerator);
  79.         final int version = InstrSupport.getMajorVersion(reader);
  80.         final ClassVisitor visitor = new ClassProbesAdapter(
  81.                 new ClassInstrumenter(strategy, writer),
  82.                 InstrSupport.needsFrames(version));
  83.         reader.accept(visitor, ClassReader.EXPAND_FRAMES);
  84.         return writer.toByteArray();
  85.     }

  86.     /**
  87.      * Creates an instrumented version of the given class if possible.
  88.      *
  89.      * @param buffer
  90.      *            definition of the class
  91.      * @param name
  92.      *            a name used for exception messages
  93.      * @return instrumented definition
  94.      * @throws IOException
  95.      *             if the class can't be instrumented
  96.      */
  97.     public byte[] instrument(final byte[] buffer, final String name)
  98.             throws IOException {
  99.         try {
  100.             return instrument(buffer);
  101.         } catch (final RuntimeException e) {
  102.             throw instrumentError(name, e);
  103.         }
  104.     }

  105.     /**
  106.      * Creates an instrumented version of the given class if possible. The
  107.      * provided {@link InputStream} is not closed by this method.
  108.      *
  109.      * @param input
  110.      *            stream to read class definition from
  111.      * @param name
  112.      *            a name used for exception messages
  113.      * @return instrumented definition
  114.      * @throws IOException
  115.      *             if reading data from the stream fails or the class can't be
  116.      *             instrumented
  117.      */
  118.     public byte[] instrument(final InputStream input, final String name)
  119.             throws IOException {
  120.         final byte[] bytes;
  121.         try {
  122.             bytes = InputStreams.readFully(input);
  123.         } catch (final IOException e) {
  124.             throw instrumentError(name, e);
  125.         }
  126.         return instrument(bytes, name);
  127.     }

  128.     /**
  129.      * Creates an instrumented version of the given class file. The provided
  130.      * {@link InputStream} and {@link OutputStream} instances are not closed by
  131.      * this method.
  132.      *
  133.      * @param input
  134.      *            stream to read class definition from
  135.      * @param output
  136.      *            stream to write the instrumented version of the class to
  137.      * @param name
  138.      *            a name used for exception messages
  139.      * @throws IOException
  140.      *             if reading data from the stream fails or the class can't be
  141.      *             instrumented
  142.      */
  143.     public void instrument(final InputStream input, final OutputStream output,
  144.             final String name) throws IOException {
  145.         output.write(instrument(input, name));
  146.     }

  147.     private IOException instrumentError(final String name,
  148.             final Exception cause) {
  149.         final IOException ex = new IOException(
  150.                 String.format("Error while instrumenting %s with JaCoCo %s/%s.",
  151.                         name, JaCoCo.VERSION, JaCoCo.COMMITID_SHORT));
  152.         ex.initCause(cause);
  153.         return ex;
  154.     }

  155.     /**
  156.      * Creates an instrumented version of the given resource depending on its
  157.      * type. Class files and the content of archive files are instrumented. All
  158.      * other files are copied without modification. The provided
  159.      * {@link InputStream} and {@link OutputStream} instances are not closed by
  160.      * this method.
  161.      *
  162.      * @param input
  163.      *            stream to contents from
  164.      * @param output
  165.      *            stream to write the instrumented version of the contents
  166.      * @param name
  167.      *            a name used for exception messages
  168.      * @return number of instrumented classes
  169.      * @throws IOException
  170.      *             if reading data from the stream fails or a class can't be
  171.      *             instrumented
  172.      */
  173.     public int instrumentAll(final InputStream input, final OutputStream output,
  174.             final String name) throws IOException {
  175.         final ContentTypeDetector detector;
  176.         try {
  177.             detector = new ContentTypeDetector(input);
  178.         } catch (final IOException e) {
  179.             throw instrumentError(name, e);
  180.         }
  181.         switch (detector.getType()) {
  182.         case ContentTypeDetector.CLASSFILE:
  183.             instrument(detector.getInputStream(), output, name);
  184.             return 1;
  185.         case ContentTypeDetector.ZIPFILE:
  186.             return instrumentZip(detector.getInputStream(), output, name);
  187.         case ContentTypeDetector.GZFILE:
  188.             return instrumentGzip(detector.getInputStream(), output, name);
  189.         case ContentTypeDetector.PACK200FILE:
  190.             return instrumentPack200(detector.getInputStream(), output, name);
  191.         default:
  192.             copy(detector.getInputStream(), output, name);
  193.             return 0;
  194.         }
  195.     }

  196.     private int instrumentZip(final InputStream input,
  197.             final OutputStream output, final String name) throws IOException {
  198.         final ZipInputStream zipin = new ZipInputStream(input);
  199.         final ZipOutputStream zipout = new ZipOutputStream(output);
  200.         ZipEntry entry;
  201.         int count = 0;
  202.         while ((entry = nextEntry(zipin, name)) != null) {
  203.             final String entryName = entry.getName();
  204.             if (signatureRemover.removeEntry(entryName)) {
  205.                 continue;
  206.             }

  207.             final ZipEntry newEntry = new ZipEntry(entryName);
  208.             newEntry.setMethod(entry.getMethod());
  209.             switch (entry.getMethod()) {
  210.             case ZipEntry.DEFLATED:
  211.                 zipout.putNextEntry(newEntry);
  212.                 count += filterOrInstrument(zipin, zipout, name, entryName);
  213.                 break;
  214.             case ZipEntry.STORED:
  215.                 // Uncompressed entries must be processed in-memory to calculate
  216.                 // mandatory entry size and CRC
  217.                 final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
  218.                 count += filterOrInstrument(zipin, buffer, name, entryName);
  219.                 final byte[] bytes = buffer.toByteArray();
  220.                 newEntry.setSize(bytes.length);
  221.                 newEntry.setCompressedSize(bytes.length);
  222.                 newEntry.setCrc(crc(bytes));
  223.                 zipout.putNextEntry(newEntry);
  224.                 zipout.write(bytes);
  225.                 break;
  226.             default:
  227.                 throw new AssertionError(entry.getMethod());
  228.             }
  229.             zipout.closeEntry();
  230.         }
  231.         zipout.finish();
  232.         return count;
  233.     }

  234.     private int filterOrInstrument(final InputStream in, final OutputStream out,
  235.             final String name, final String entryName) throws IOException {
  236.         if (signatureRemover.filterEntry(entryName, in, out)) {
  237.             return 0;
  238.         } else {
  239.             return instrumentAll(in, out, name + "@" + entryName);
  240.         }
  241.     }

  242.     private static long crc(final byte[] data) {
  243.         final CRC32 crc = new CRC32();
  244.         crc.update(data);
  245.         return crc.getValue();
  246.     }

  247.     private ZipEntry nextEntry(final ZipInputStream input,
  248.             final String location) throws IOException {
  249.         try {
  250.             return input.getNextEntry();
  251.         } catch (final IOException e) {
  252.             throw instrumentError(location, e);
  253.         } catch (final IllegalArgumentException e) {
  254.             // might be thrown in JDK versions below 23 - see
  255.             // https://bugs.openjdk.org/browse/JDK-8321156
  256.             // https://github.com/openjdk/jdk/commit/20c71ceacdcb791f5b70cda456bdc47bdd9acf6c
  257.             throw instrumentError(location, e);
  258.         }
  259.     }

  260.     private int instrumentGzip(final InputStream input,
  261.             final OutputStream output, final String name) throws IOException {
  262.         final GZIPInputStream gzipInputStream;
  263.         try {
  264.             gzipInputStream = new GZIPInputStream(input);
  265.         } catch (final IOException e) {
  266.             throw instrumentError(name, e);
  267.         }
  268.         final GZIPOutputStream gzout = new GZIPOutputStream(output);
  269.         final int count = instrumentAll(gzipInputStream, gzout, name);
  270.         gzout.finish();
  271.         return count;
  272.     }

  273.     private int instrumentPack200(final InputStream input,
  274.             final OutputStream output, final String name) throws IOException {
  275.         final InputStream unpackedInput;
  276.         try {
  277.             unpackedInput = Pack200Streams.unpack(input);
  278.         } catch (final IOException e) {
  279.             throw instrumentError(name, e);
  280.         }
  281.         final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
  282.         final int count = instrumentAll(unpackedInput, buffer, name);
  283.         Pack200Streams.pack(buffer.toByteArray(), output);
  284.         return count;
  285.     }

  286.     private void copy(final InputStream input, final OutputStream output,
  287.             final String name) throws IOException {
  288.         final byte[] buffer = new byte[1024];
  289.         int len;
  290.         while ((len = read(input, buffer, name)) != -1) {
  291.             output.write(buffer, 0, len);
  292.         }
  293.     }

  294.     private int read(final InputStream input, final byte[] buffer,
  295.             final String name) throws IOException {
  296.         try {
  297.             return input.read(buffer);
  298.         } catch (final IOException e) {
  299.             throw instrumentError(name, e);
  300.         }
  301.     }

  302. }