ProbeArrayStrategyFactory.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.instr;

  14. import org.jacoco.core.internal.flow.ClassProbesAdapter;
  15. import org.jacoco.core.runtime.IExecutionDataAccessorGenerator;
  16. import org.objectweb.asm.ClassReader;
  17. import org.objectweb.asm.Opcodes;

  18. /**
  19.  * Factory to find a suitable strategy to access the probe array for a given
  20.  * class.
  21.  */
  22. public final class ProbeArrayStrategyFactory {

  23.     private ProbeArrayStrategyFactory() {
  24.     }

  25.     /**
  26.      * Creates a suitable strategy instance for the class described by the given
  27.      * reader. Created instance must be used only to process a class or
  28.      * interface for which it has been created and must be used only once.
  29.      *
  30.      * @param classId
  31.      *            class identifier
  32.      * @param reader
  33.      *            reader to get information about the class
  34.      * @param accessorGenerator
  35.      *            accessor to the coverage runtime
  36.      * @return strategy instance
  37.      */
  38.     public static IProbeArrayStrategy createFor(final long classId,
  39.             final ClassReader reader,
  40.             final IExecutionDataAccessorGenerator accessorGenerator) {

  41.         final String className = reader.getClassName();
  42.         final int version = InstrSupport.getMajorVersion(reader);

  43.         if (isInterfaceOrModule(reader)) {
  44.             final ProbeCounter counter = getProbeCounter(reader);
  45.             if (counter.getCount() == 0) {
  46.                 return new NoneProbeArrayStrategy();
  47.             }
  48.             if (version >= Opcodes.V11 && counter.hasMethods()) {
  49.                 return new CondyProbeArrayStrategy(className, true, classId,
  50.                         accessorGenerator);
  51.             }
  52.             if (version >= Opcodes.V1_8 && counter.hasMethods()) {
  53.                 return new InterfaceFieldProbeArrayStrategy(className, classId,
  54.                         counter.getCount(), accessorGenerator);
  55.             } else {
  56.                 return new LocalProbeArrayStrategy(className, classId,
  57.                         counter.getCount(), accessorGenerator);
  58.             }
  59.         } else {
  60.             if (version >= Opcodes.V11) {
  61.                 return new CondyProbeArrayStrategy(className, false, classId,
  62.                         accessorGenerator);
  63.             }
  64.             return new ClassFieldProbeArrayStrategy(className, classId,
  65.                     InstrSupport.needsFrames(version), accessorGenerator);
  66.         }
  67.     }

  68.     private static boolean isInterfaceOrModule(final ClassReader reader) {
  69.         return (reader.getAccess()
  70.                 & (Opcodes.ACC_INTERFACE | Opcodes.ACC_MODULE)) != 0;
  71.     }

  72.     private static ProbeCounter getProbeCounter(final ClassReader reader) {
  73.         final ProbeCounter counter = new ProbeCounter();
  74.         reader.accept(new ClassProbesAdapter(counter, false), 0);
  75.         return counter;
  76.     }

  77. }