MethodProbesVisitor.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.internal.flow;

  14. import org.jacoco.core.internal.instr.InstrSupport;
  15. import org.objectweb.asm.Label;
  16. import org.objectweb.asm.MethodVisitor;
  17. import org.objectweb.asm.tree.MethodNode;

  18. /**
  19.  * A {@link MethodVisitor} with additional methods to get probe insertion
  20.  * information.
  21.  */
  22. public abstract class MethodProbesVisitor extends MethodVisitor {

  23.     /**
  24.      * New visitor instance without delegate visitor.
  25.      */
  26.     public MethodProbesVisitor() {
  27.         this(null);
  28.     }

  29.     /**
  30.      * New visitor instance that delegates to the given visitor.
  31.      *
  32.      * @param mv
  33.      *            optional next visitor in chain
  34.      */
  35.     public MethodProbesVisitor(final MethodVisitor mv) {
  36.         super(InstrSupport.ASM_API_VERSION, mv);
  37.     }

  38.     /**
  39.      * Visits an unconditional probe that should be inserted at the current
  40.      * position.
  41.      *
  42.      * @param probeId
  43.      *            id of the probe to insert
  44.      */
  45.     @SuppressWarnings("unused")
  46.     public void visitProbe(final int probeId) {
  47.     }

  48.     /**
  49.      * Visits a jump instruction. A probe with the given id should be inserted
  50.      * in a way that it is executed only when the jump to the given label is
  51.      * executed.
  52.      *
  53.      * @param opcode
  54.      *            the opcode of the type instruction to be visited. This opcode
  55.      *            is either IFEQ, IFNE, IFLT, IFGE, IFGT, IFLE, IF_ICMPEQ,
  56.      *            IF_ICMPNE, IF_ICMPLT, IF_ICMPGE, IF_ICMPGT, IF_ICMPLE,
  57.      *            IF_ACMPEQ, IF_ACMPNE, GOTO, IFNULL or IFNONNULL.
  58.      * @param label
  59.      *            the operand of the instruction to be visited. This operand is
  60.      *            a label that designates the instruction to which the jump
  61.      *            instruction may jump.
  62.      * @param probeId
  63.      *            id of the probe
  64.      * @param frame
  65.      *            stackmap frame status after the execution of the jump
  66.      *            instruction. The instance is only valid with the call of this
  67.      *            method.
  68.      * @see MethodVisitor#visitJumpInsn(int, Label)
  69.      */
  70.     @SuppressWarnings("unused")
  71.     public void visitJumpInsnWithProbe(final int opcode, final Label label,
  72.             final int probeId, final IFrame frame) {
  73.     }

  74.     /**
  75.      * Visits a zero operand instruction with a probe. This event is used only
  76.      * for instructions that terminate the method. Therefore the probe must be
  77.      * inserted before the actual instruction.
  78.      *
  79.      * @param opcode
  80.      *            the opcode of the instruction to be visited. This opcode is
  81.      *            either IRETURN, LRETURN, FRETURN, DRETURN, ARETURN, RETURN or
  82.      *            ATHROW.
  83.      * @param probeId
  84.      *            id of the probe
  85.      * @see MethodVisitor#visitInsn(int)
  86.      */
  87.     @SuppressWarnings("unused")
  88.     public void visitInsnWithProbe(final int opcode, final int probeId) {
  89.     }

  90.     /**
  91.      * Visits a TABLESWITCH instruction with optional probes for each target
  92.      * label. Implementations can be optimized based on the fact that the same
  93.      * target labels will always have the same probe id within a call to this
  94.      * method. The probe id for each label can be obtained with
  95.      * {@link LabelInfo#getProbeId(Label)}.
  96.      *
  97.      * @param min
  98.      *            the minimum key value.
  99.      * @param max
  100.      *            the maximum key value.
  101.      * @param dflt
  102.      *            beginning of the default handler block.
  103.      * @param labels
  104.      *            beginnings of the handler blocks. <code>labels[i]</code> is
  105.      *            the beginning of the handler block for the
  106.      *            <code>min + i</code> key.
  107.      * @param frame
  108.      *            stackmap frame status after the execution of the switch
  109.      *            instruction. The instance is only valid with the call of this
  110.      *            method.
  111.      * @see MethodVisitor#visitTableSwitchInsn(int, int, Label, Label[])
  112.      */
  113.     @SuppressWarnings("unused")
  114.     public void visitTableSwitchInsnWithProbes(final int min, final int max,
  115.             final Label dflt, final Label[] labels, final IFrame frame) {
  116.     }

  117.     /**
  118.      * Visits a LOOKUPSWITCH instruction with optional probes for each target
  119.      * label. Implementations can be optimized based on the fact that the same
  120.      * target labels will always have the same probe id within a call to this
  121.      * method. The probe id for each label can be obtained with
  122.      * {@link LabelInfo#getProbeId(Label)}.
  123.      *
  124.      * @param dflt
  125.      *            beginning of the default handler block.
  126.      * @param keys
  127.      *            the values of the keys.
  128.      * @param labels
  129.      *            beginnings of the handler blocks. <code>labels[i]</code> is
  130.      *            the beginning of the handler block for the
  131.      *            <code>keys[i]</code> key.
  132.      * @param frame
  133.      *            stackmap frame status after the execution of the switch
  134.      *            instruction. The instance is only valid with the call of this
  135.      *            method.
  136.      * @see MethodVisitor#visitLookupSwitchInsn(Label, int[], Label[])
  137.      */
  138.     @SuppressWarnings("unused")
  139.     public void visitLookupSwitchInsnWithProbes(final Label dflt,
  140.             final int[] keys, final Label[] labels, final IFrame frame) {
  141.     }

  142.     /**
  143.      * This method can be overwritten to hook into the process of emitting the
  144.      * instructions of this method as <code>visitX()</code> events.
  145.      *
  146.      * @param methodNode
  147.      *            the content to emit
  148.      * @param methodVisitor
  149.      *            A visitor to emit the content to. Note that this is not
  150.      *            necessarily this visitor instance but some wrapper which
  151.      *            calculates the probes.
  152.      */
  153.     public void accept(final MethodNode methodNode,
  154.             final MethodVisitor methodVisitor) {
  155.         methodNode.accept(methodVisitor);
  156.     }

  157. }