blob: bc957b912db0157b440dd508d30fa7e56ce2cb60 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2009, 2019 Mountainminds GmbH & Co. KG and Contributors
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Marc R. Hoffmann - initial API and implementation
*
*******************************************************************************/
package org.jacoco.core.internal.instr;
import org.jacoco.core.internal.flow.ClassProbesAdapter;
import org.jacoco.core.runtime.IExecutionDataAccessorGenerator;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.Opcodes;
/**
* Factory to find a suitable strategy to access the probe array for a given
* class.
*/
public final class ProbeArrayStrategyFactory {
private ProbeArrayStrategyFactory() {
}
/**
* Creates a suitable strategy instance for the class described by the given
* reader. Created instance must be used only to process a class or
* interface for which it has been created and must be used only once.
*
* @param classId
* class identifier
* @param reader
* reader to get information about the class
* @param accessorGenerator
* accessor to the coverage runtime
* @return strategy instance
*/
public static IProbeArrayStrategy createFor(final long classId,
final ClassReader reader,
final IExecutionDataAccessorGenerator accessorGenerator) {
final String className = reader.getClassName();
final int version = InstrSupport.getMajorVersion(reader);
if (isInterfaceOrModule(reader)) {
final ProbeCounter counter = getProbeCounter(reader);
if (counter.getCount() == 0) {
return new NoneProbeArrayStrategy();
}
// BEGIN android-change
// See https://github.com/jacoco/jacoco/issues/1151
// if (version >= Opcodes.V11 && counter.hasMethods()) {
// return new CondyProbeArrayStrategy(className, true, classId,
// accessorGenerator);
// }
// END android-change
if (version >= Opcodes.V1_8 && counter.hasMethods()) {
return new InterfaceFieldProbeArrayStrategy(className, classId,
counter.getCount(), accessorGenerator);
} else {
return new LocalProbeArrayStrategy(className, classId,
counter.getCount(), accessorGenerator);
}
} else {
// BEGIN android-change
// See https://github.com/jacoco/jacoco/issues/1151
// if (version >= Opcodes.V11) {
// return new CondyProbeArrayStrategy(className, false, classId,
// accessorGenerator);
// }
// END android-change
return new ClassFieldProbeArrayStrategy(className, classId,
InstrSupport.needsFrames(version), accessorGenerator);
}
}
private static boolean isInterfaceOrModule(final ClassReader reader) {
return (reader.getAccess()
& (Opcodes.ACC_INTERFACE | Opcodes.ACC_MODULE)) != 0;
}
private static ProbeCounter getProbeCounter(final ClassReader reader) {
final ProbeCounter counter = new ProbeCounter();
reader.accept(new ClassProbesAdapter(counter, false), 0);
return counter;
}
}