GPUFitnessEvaluator.java

1
package net.bmahe.genetics4j.gpu;
2
3
import java.io.IOException;
4
import java.nio.charset.StandardCharsets;
5
import java.util.ArrayList;
6
import java.util.HashMap;
7
import java.util.List;
8
import java.util.Map;
9
import java.util.Set;
10
import java.util.concurrent.CompletableFuture;
11
import java.util.concurrent.ExecutorService;
12
13
import org.apache.commons.collections4.ListUtils;
14
import org.apache.commons.io.IOUtils;
15
import org.apache.commons.lang3.Validate;
16
import org.apache.commons.lang3.tuple.Pair;
17
import org.apache.logging.log4j.LogManager;
18
import org.apache.logging.log4j.Logger;
19
import org.jocl.CL;
20
import org.jocl.cl_command_queue;
21
import org.jocl.cl_context;
22
import org.jocl.cl_context_properties;
23
import org.jocl.cl_device_id;
24
import org.jocl.cl_kernel;
25
import org.jocl.cl_platform_id;
26
import org.jocl.cl_program;
27
import org.jocl.cl_queue_properties;
28
29
import net.bmahe.genetics4j.core.Genotype;
30
import net.bmahe.genetics4j.core.evaluation.FitnessEvaluator;
31
import net.bmahe.genetics4j.gpu.opencl.DeviceReader;
32
import net.bmahe.genetics4j.gpu.opencl.DeviceUtils;
33
import net.bmahe.genetics4j.gpu.opencl.KernelInfoReader;
34
import net.bmahe.genetics4j.gpu.opencl.OpenCLExecutionContext;
35
import net.bmahe.genetics4j.gpu.opencl.PlatformReader;
36
import net.bmahe.genetics4j.gpu.opencl.PlatformUtils;
37
import net.bmahe.genetics4j.gpu.opencl.model.Device;
38
import net.bmahe.genetics4j.gpu.opencl.model.KernelInfo;
39
import net.bmahe.genetics4j.gpu.opencl.model.Platform;
40
import net.bmahe.genetics4j.gpu.spec.GPUEAConfiguration;
41
import net.bmahe.genetics4j.gpu.spec.GPUEAExecutionContext;
42
import net.bmahe.genetics4j.gpu.spec.Program;
43
44
/**
45
 * GPU-accelerated fitness evaluator that leverages OpenCL for high-performance evolutionary algorithm execution.
46
 * 
47
 * <p>GPUFitnessEvaluator implements the core {@link FitnessEvaluator} interface to provide GPU acceleration for fitness
48
 * computation in evolutionary algorithms. This evaluator manages the complete OpenCL lifecycle, from device discovery
49
 * and kernel compilation to memory management and resource cleanup.
50
 * 
51
 * <p>Key responsibilities include:
52
 * <ul>
53
 * <li><strong>OpenCL initialization</strong>: Platform and device discovery, context creation, and kernel
54
 * compilation</li>
55
 * <li><strong>Resource management</strong>: Managing OpenCL contexts, command queues, programs, and kernels</li>
56
 * <li><strong>Population partitioning</strong>: Distributing work across multiple OpenCL devices</li>
57
 * <li><strong>Asynchronous execution</strong>: Coordinating concurrent GPU operations with CPU-side logic</li>
58
 * <li><strong>Memory lifecycle</strong>: Ensuring proper cleanup of GPU resources</li>
59
 * </ul>
60
 * 
61
 * <p>Architecture overview:
62
 * <ol>
63
 * <li><strong>Initialization ({@link #preEvaluation})</strong>: Discover platforms/devices, compile kernels, create
64
 * contexts</li>
65
 * <li><strong>Evaluation ({@link #evaluate})</strong>: Partition population, execute fitness computation on GPU</li>
66
 * <li><strong>Cleanup ({@link #postEvaluation})</strong>: Release all OpenCL resources and contexts</li>
67
 * </ol>
68
 * 
69
 * <p>Multi-device support:
70
 * <ul>
71
 * <li><strong>Device filtering</strong>: Selects devices based on user-defined criteria (type, capabilities)</li>
72
 * <li><strong>Load balancing</strong>: Automatically distributes population across available devices</li>
73
 * <li><strong>Parallel execution</strong>: Concurrent fitness evaluation on multiple GPUs or devices</li>
74
 * <li><strong>Asynchronous coordination</strong>: Non-blocking execution with CompletableFuture-based results</li>
75
 * </ul>
76
 * 
77
 * <p>Resource management patterns:
78
 * <ul>
79
 * <li><strong>Lazy initialization</strong>: OpenCL resources created only when needed</li>
80
 * <li><strong>Automatic cleanup</strong>: Guaranteed resource release through lifecycle methods</li>
81
 * <li><strong>Error recovery</strong>: Robust handling of OpenCL errors and device failures</li>
82
 * <li><strong>Memory optimization</strong>: Efficient GPU memory usage and transfer patterns</li>
83
 * </ul>
84
 * 
85
 * <p>Example usage in GPU EA system:
86
 * 
87
 * <pre>{@code
88
 * // GPU configuration with OpenCL kernel
89
 * Program fitnessProgram = Program.ofResource("/kernels/optimization.cl");
90
 * GPUEAConfiguration<Double> config = GPUEAConfigurationBuilder.<Double>builder()
91
 * 		.program(fitnessProgram)
92
 * 		.fitness(new MyGPUFitness())
93
 * 		// ... other EA configuration
94
 * 		.build();
95
 * 
96
 * // Execution context with device preferences
97
 * GPUEAExecutionContext<Double> context = GPUEAExecutionContextBuilder.<Double>builder()
98
 * 		.populationSize(2000)
99
 * 		.deviceFilter(device -> device.type() == DeviceType.GPU)
100
 * 		.platformFilter(platform -> platform.profile() == PlatformProfile.FULL_PROFILE)
101
 * 		.build();
102
 * 
103
 * // Evaluator handles all OpenCL lifecycle automatically
104
 * GPUFitnessEvaluator<Double> evaluator = new GPUFitnessEvaluator<>(context, config, executorService);
105
 * 
106
 * // Used by EA system - lifecycle managed automatically
107
 * EASystem<Double> system = EASystemFactory.from(config, context, executorService, evaluator);
108
 * }</pre>
109
 * 
110
 * <p>Performance characteristics:
111
 * <ul>
112
 * <li><strong>Initialization overhead</strong>: One-time setup cost for OpenCL compilation and context creation</li>
113
 * <li><strong>Scalability</strong>: Performance scales with population size and problem complexity</li>
114
 * <li><strong>Memory bandwidth</strong>: Optimal for problems with high computational intensity</li>
115
 * <li><strong>Concurrency</strong>: Supports concurrent evaluation across multiple devices</li>
116
 * </ul>
117
 * 
118
 * <p>Error handling:
119
 * <ul>
120
 * <li><strong>Device failures</strong>: Graceful degradation when devices become unavailable</li>
121
 * <li><strong>Memory errors</strong>: Proper cleanup and error reporting for GPU memory issues</li>
122
 * <li><strong>Compilation errors</strong>: Clear error messages for kernel compilation failures</li>
123
 * <li><strong>Resource leaks</strong>: Guaranteed cleanup even in exceptional circumstances</li>
124
 * </ul>
125
 * 
126
 * @param <T> the type of fitness values produced, must be comparable for selection operations
127
 * @see FitnessEvaluator
128
 * @see GPUEAConfiguration
129
 * @see GPUEAExecutionContext
130
 * @see OpenCLExecutionContext
131
 * @see net.bmahe.genetics4j.gpu.fitness.OpenCLFitness
132
 */
133
public class GPUFitnessEvaluator<T extends Comparable<T>> implements FitnessEvaluator<T> {
134
	public static final Logger logger = LogManager.getLogger(GPUFitnessEvaluator.class);
135
136
	private final GPUEAExecutionContext<T> gpuEAExecutionContext;
137
	private final GPUEAConfiguration<T> gpuEAConfiguration;
138
	private final ExecutorService executorService;
139
140
	private List<Pair<Platform, Device>> selectedPlatformToDevice;
141
142 2 1. <init> : Removed assignment to member variable clContexts → NO_COVERAGE
2. <init> : removed call to java/util/ArrayList::<init> → NO_COVERAGE
	final List<cl_context> clContexts = new ArrayList<>();
143 2 1. <init> : removed call to java/util/ArrayList::<init> → NO_COVERAGE
2. <init> : Removed assignment to member variable clCommandQueues → NO_COVERAGE
	final List<cl_command_queue> clCommandQueues = new ArrayList<>();
144 2 1. <init> : Removed assignment to member variable clPrograms → NO_COVERAGE
2. <init> : removed call to java/util/ArrayList::<init> → NO_COVERAGE
	final List<cl_program> clPrograms = new ArrayList<>();
145 2 1. <init> : Removed assignment to member variable clKernels → NO_COVERAGE
2. <init> : removed call to java/util/ArrayList::<init> → NO_COVERAGE
	final List<Map<String, cl_kernel>> clKernels = new ArrayList<>();
146 2 1. <init> : Removed assignment to member variable clExecutionContexts → NO_COVERAGE
2. <init> : removed call to java/util/ArrayList::<init> → NO_COVERAGE
	final List<OpenCLExecutionContext> clExecutionContexts = new ArrayList<>();
147
148
	/**
149
	 * Constructs a GPU fitness evaluator with the specified configuration and execution context.
150
	 * 
151
	 * <p>Initializes the evaluator with GPU-specific configuration and execution parameters. The evaluator will use the
152
	 * provided executor service for coordinating asynchronous operations between CPU and GPU components.
153
	 * 
154
	 * <p>The constructor performs minimal initialization - the actual OpenCL setup occurs during
155
	 * {@link #preEvaluation()} to follow the fitness evaluator lifecycle pattern.
156
	 * 
157
	 * @param _gpuEAExecutionContext the GPU execution context with device filters and population settings
158
	 * @param _gpuEAConfiguration    the GPU EA configuration with OpenCL program and fitness function
159
	 * @param _executorService       the executor service for managing asynchronous operations
160
	 * @throws IllegalArgumentException if any parameter is null
161
	 */
162
	public GPUFitnessEvaluator(final GPUEAExecutionContext<T> _gpuEAExecutionContext,
163
			final GPUEAConfiguration<T> _gpuEAConfiguration,
164
			final ExecutorService _executorService) {
165
		Validate.notNull(_gpuEAExecutionContext);
166
		Validate.notNull(_gpuEAConfiguration);
167
		Validate.notNull(_executorService);
168
169 1 1. <init> : Removed assignment to member variable gpuEAExecutionContext → NO_COVERAGE
		this.gpuEAExecutionContext = _gpuEAExecutionContext;
170 1 1. <init> : Removed assignment to member variable gpuEAConfiguration → NO_COVERAGE
		this.gpuEAConfiguration = _gpuEAConfiguration;
171 1 1. <init> : Removed assignment to member variable executorService → NO_COVERAGE
		this.executorService = _executorService;
172
173 2 1. <init> : removed call to org/jocl/CL::setExceptionsEnabled → NO_COVERAGE
2. <init> : Substituted 1 with 0 → NO_COVERAGE
		CL.setExceptionsEnabled(true);
174
	}
175
176
	private String loadResource(final String filename) {
177
		Validate.notBlank(filename);
178
179
		try {
180 3 1. loadResource : replaced call to org/apache/commons/io/IOUtils::resourceToString with argument → NO_COVERAGE
2. loadResource : replaced return value with "" for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::loadResource → NO_COVERAGE
3. loadResource : removed call to org/apache/commons/io/IOUtils::resourceToString → NO_COVERAGE
			return IOUtils.resourceToString(filename, StandardCharsets.UTF_8);
181
		} catch (IOException e) {
182 1 1. loadResource : removed call to java/lang/IllegalStateException::<init> → NO_COVERAGE
			throw new IllegalStateException("Unable to load resource " + filename, e);
183
		}
184
	}
185
186
	private List<String> grabProgramSources() {
187 1 1. grabProgramSources : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::program → NO_COVERAGE
		final Program programSpec = gpuEAConfiguration.program();
188
189
		logger.info("Load program source: {}", programSpec);
190
191 1 1. grabProgramSources : removed call to java/util/ArrayList::<init> → NO_COVERAGE
		final List<String> sources = new ArrayList<>();
192
193 2 1. grabProgramSources : removed call to net/bmahe/genetics4j/gpu/spec/Program::content → NO_COVERAGE
2. grabProgramSources : removed call to java/util/List::addAll → NO_COVERAGE
		sources.addAll(programSpec.content());
194
195 8 1. lambda$grabProgramSources$0 : replaced call to net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::loadResource with argument → NO_COVERAGE
2. grabProgramSources : removed call to net/bmahe/genetics4j/gpu/spec/Program::resources → NO_COVERAGE
3. lambda$grabProgramSources$0 : replaced return value with "" for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$grabProgramSources$0 → NO_COVERAGE
4. grabProgramSources : replaced call to java/util/stream/Stream::map with receiver → NO_COVERAGE
5. lambda$grabProgramSources$0 : removed call to net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::loadResource → NO_COVERAGE
6. grabProgramSources : removed call to java/util/Set::stream → NO_COVERAGE
7. grabProgramSources : removed call to java/util/stream/Stream::map → NO_COVERAGE
8. grabProgramSources : removed call to java/util/stream/Stream::forEach → NO_COVERAGE
		programSpec.resources().stream().map(resource -> loadResource(resource)).forEach(program -> {
196 1 1. lambda$grabProgramSources$1 : removed call to java/util/List::add → NO_COVERAGE
			sources.add(program);
197
		});
198
199 1 1. grabProgramSources : replaced return value with Collections.emptyList for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::grabProgramSources → NO_COVERAGE
		return sources;
200
	}
201
202
	/**
203
	 * Initializes OpenCL resources and prepares GPU devices for fitness evaluation.
204
	 * 
205
	 * <p>This method performs the complete OpenCL initialization sequence:
206
	 * <ol>
207
	 * <li><strong>Platform discovery</strong>: Enumerates available OpenCL platforms</li>
208
	 * <li><strong>Device filtering</strong>: Selects devices based on configured filters</li>
209
	 * <li><strong>Context creation</strong>: Creates OpenCL contexts for selected devices</li>
210
	 * <li><strong>Queue setup</strong>: Creates command queues with profiling and out-of-order execution</li>
211
	 * <li><strong>Program compilation</strong>: Compiles OpenCL kernels from source code</li>
212
	 * <li><strong>Kernel preparation</strong>: Creates kernel objects and queries execution info</li>
213
	 * <li><strong>Fitness initialization</strong>: Calls lifecycle hooks on the fitness function</li>
214
	 * </ol>
215
	 * 
216
	 * <p>Device selection process:
217
	 * <ul>
218
	 * <li>Applies platform filters to discovered OpenCL platforms</li>
219
	 * <li>Enumerates devices for each qualifying platform</li>
220
	 * <li>Applies device filters to select appropriate devices</li>
221
	 * <li>Validates that at least one device is available</li>
222
	 * </ul>
223
	 * 
224
	 * <p>The method creates separate OpenCL contexts for each selected device to enable concurrent execution and optimal
225
	 * resource utilization. Each context includes compiled programs and kernel objects ready for fitness evaluation.
226
	 * 
227
	 * @throws IllegalStateException if no compatible devices are found
228
	 * @throws RuntimeException      if OpenCL initialization, program compilation, or kernel creation fails
229
	 */
230
	@Override
231
	public void preEvaluation() {
232
		logger.trace("Init...");
233 1 1. preEvaluation : removed call to net/bmahe/genetics4j/core/evaluation/FitnessEvaluator::preEvaluation → NO_COVERAGE
		FitnessEvaluator.super.preEvaluation();
234
235 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/PlatformReader::<init> → NO_COVERAGE
		final var platformReader = new PlatformReader();
236 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/DeviceReader::<init> → NO_COVERAGE
		final var deviceReader = new DeviceReader();
237 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/KernelInfoReader::<init> → NO_COVERAGE
		final var kernelInfoReader = new KernelInfoReader();
238
239 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/PlatformUtils::numPlatforms → NO_COVERAGE
		final int numPlatforms = PlatformUtils.numPlatforms();
240
		logger.info("Found {} platforms", numPlatforms);
241
242 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/PlatformUtils::platformIds → NO_COVERAGE
		final List<cl_platform_id> platformIds = PlatformUtils.platformIds(numPlatforms);
243
244
		logger.info("Selecting platform and devices");
245 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAExecutionContext::platformFilters → NO_COVERAGE
		final var platformFilters = gpuEAExecutionContext.platformFilters();
246 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAExecutionContext::deviceFilters → NO_COVERAGE
		final var deviceFilters = gpuEAExecutionContext.deviceFilters();
247
248 1 1. preEvaluation : removed call to java/util/List::stream → NO_COVERAGE
		selectedPlatformToDevice = platformIds.stream()
249
				.map(platformReader::read)
250 2 1. preEvaluation : replaced call to java/util/stream/Stream::filter with receiver → NO_COVERAGE
2. preEvaluation : removed call to java/util/stream/Stream::filter → NO_COVERAGE
				.filter(platformFilters)
251 2 1. preEvaluation : replaced call to java/util/stream/Stream::flatMap with receiver → NO_COVERAGE
2. preEvaluation : removed call to java/util/stream/Stream::flatMap → NO_COVERAGE
				.flatMap(platform -> {
252 1 1. lambda$preEvaluation$3 : removed call to net/bmahe/genetics4j/gpu/opencl/model/Platform::platformId → NO_COVERAGE
					final var platformId = platform.platformId();
253 1 1. lambda$preEvaluation$3 : removed call to net/bmahe/genetics4j/gpu/opencl/DeviceUtils::numDevices → NO_COVERAGE
					final int numDevices = DeviceUtils.numDevices(platformId);
254
					logger.trace("\tPlatform {}: {} devices", platform.name(), numDevices);
255
256 1 1. lambda$preEvaluation$3 : removed call to net/bmahe/genetics4j/gpu/opencl/DeviceUtils::getDeviceIds → NO_COVERAGE
					final var deviceIds = DeviceUtils.getDeviceIds(platformId, numDevices);
257 6 1. lambda$preEvaluation$3 : replaced return value with Stream.empty for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$3 → NO_COVERAGE
2. lambda$preEvaluation$3 : removed call to java/util/stream/Stream::map → NO_COVERAGE
3. lambda$preEvaluation$3 : removed call to java/util/List::stream → NO_COVERAGE
4. lambda$preEvaluation$2 : removed call to org/apache/commons/lang3/tuple/Pair::of → NO_COVERAGE
5. lambda$preEvaluation$3 : replaced call to java/util/stream/Stream::map with receiver → NO_COVERAGE
6. lambda$preEvaluation$2 : replaced return value with null for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$2 → NO_COVERAGE
					return deviceIds.stream().map(deviceId -> Pair.of(platform, deviceId));
258
				})
259 2 1. preEvaluation : replaced call to java/util/stream/Stream::map with receiver → NO_COVERAGE
2. preEvaluation : removed call to java/util/stream/Stream::map → NO_COVERAGE
				.map(platformToDeviceId -> {
260 1 1. lambda$preEvaluation$4 : removed call to org/apache/commons/lang3/tuple/Pair::getLeft → NO_COVERAGE
					final var platform = platformToDeviceId.getLeft();
261 1 1. lambda$preEvaluation$4 : removed call to net/bmahe/genetics4j/gpu/opencl/model/Platform::platformId → NO_COVERAGE
					final var platformId = platform.platformId();
262 1 1. lambda$preEvaluation$4 : removed call to org/apache/commons/lang3/tuple/Pair::getRight → NO_COVERAGE
					final var deviceID = platformToDeviceId.getRight();
263
264 3 1. lambda$preEvaluation$4 : replaced return value with null for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$4 → NO_COVERAGE
2. lambda$preEvaluation$4 : removed call to net/bmahe/genetics4j/gpu/opencl/DeviceReader::read → NO_COVERAGE
3. lambda$preEvaluation$4 : removed call to org/apache/commons/lang3/tuple/Pair::of → NO_COVERAGE
					return Pair.of(platform, deviceReader.read(platformId, deviceID));
265
				})
266 6 1. lambda$preEvaluation$5 : removed call to java/util/function/Predicate::test → NO_COVERAGE
2. preEvaluation : removed call to java/util/stream/Stream::filter → NO_COVERAGE
3. lambda$preEvaluation$5 : replaced boolean return with true for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$5 → NO_COVERAGE
4. lambda$preEvaluation$5 : replaced boolean return with false for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$5 → NO_COVERAGE
5. lambda$preEvaluation$5 : removed call to org/apache/commons/lang3/tuple/Pair::getRight → NO_COVERAGE
6. preEvaluation : replaced call to java/util/stream/Stream::filter with receiver → NO_COVERAGE
				.filter(platformToDevice -> deviceFilters.test(platformToDevice.getRight()))
267 2 1. preEvaluation : Removed assignment to member variable selectedPlatformToDevice → NO_COVERAGE
2. preEvaluation : removed call to java/util/stream/Stream::toList → NO_COVERAGE
				.toList();
268
269
		if (logger.isTraceEnabled()) {
270
			logger.trace("============================");
271
			logger.trace("Selected devices:");
272 1 1. preEvaluation : removed call to java/util/List::forEach → NO_COVERAGE
			selectedPlatformToDevice.forEach(pd -> {
273
				logger.trace("{}", pd.getLeft());
274
				logger.trace("\t{}", pd.getRight());
275
			});
276
			logger.trace("============================");
277
		}
278
279
		Validate.isTrue(selectedPlatformToDevice.size() > 0);
280
281 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::grabProgramSources → NO_COVERAGE
		final List<String> programs = grabProgramSources();
282 3 1. preEvaluation : removed call to java/util/List::size → NO_COVERAGE
2. preEvaluation : replaced call to java/util/List::toArray with argument → NO_COVERAGE
3. preEvaluation : removed call to java/util/List::toArray → NO_COVERAGE
		final String[] programsArr = programs.toArray(new String[programs.size()]);
283
284
		for (final var platformAndDevice : selectedPlatformToDevice) {
285 1 1. preEvaluation : removed call to org/apache/commons/lang3/tuple/Pair::getLeft → NO_COVERAGE
			final var platform = platformAndDevice.getLeft();
286 1 1. preEvaluation : removed call to org/apache/commons/lang3/tuple/Pair::getRight → NO_COVERAGE
			final var device = platformAndDevice.getRight();
287
288
			logger.info("Processing platform [{}] / device [{}]", platform.name(), device.name());
289
290
			logger.info("\tCreating context");
291 1 1. preEvaluation : removed call to org/jocl/cl_context_properties::<init> → NO_COVERAGE
			cl_context_properties contextProperties = new cl_context_properties();
292 3 1. preEvaluation : Substituted 4228 with 4229 → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/model/Platform::platformId → NO_COVERAGE
3. preEvaluation : removed call to org/jocl/cl_context_properties::addProperty → NO_COVERAGE
			contextProperties.addProperty(CL.CL_CONTEXT_PLATFORM, platform.platformId());
293
294 3 1. preEvaluation : Substituted 1 with 0 → NO_COVERAGE
2. preEvaluation : Substituted 0 with 1 → NO_COVERAGE
3. preEvaluation : Substituted 1 with 0 → NO_COVERAGE
			final cl_context context = CL
295 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/model/Device::deviceId → NO_COVERAGE
2. preEvaluation : removed call to org/jocl/CL::clCreateContext → NO_COVERAGE
					.clCreateContext(contextProperties, 1, new cl_device_id[] { device.deviceId() }, null, null, null);
296
297
			logger.info("\tCreating command queue");
298 1 1. preEvaluation : removed call to org/jocl/cl_queue_properties::<init> → NO_COVERAGE
			final cl_queue_properties queueProperties = new cl_queue_properties();
299 3 1. preEvaluation : Substituted 4243 with 4244 → NO_COVERAGE
2. preEvaluation : Substituted 3 with 4 → NO_COVERAGE
3. preEvaluation : removed call to org/jocl/cl_queue_properties::addProperty → NO_COVERAGE
			queueProperties.addProperty(
300
					CL.CL_QUEUE_PROPERTIES,
301
						CL.CL_QUEUE_PROFILING_ENABLE | CL.CL_QUEUE_OUT_OF_ORDER_EXEC_MODE_ENABLE);
302
			final cl_command_queue commandQueue = CL
303 2 1. preEvaluation : removed call to org/jocl/CL::clCreateCommandQueueWithProperties → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/model/Device::deviceId → NO_COVERAGE
					.clCreateCommandQueueWithProperties(context, device.deviceId(), queueProperties, null);
304
305
			logger.info("\tCreate program");
306 1 1. preEvaluation : removed call to org/jocl/CL::clCreateProgramWithSource → NO_COVERAGE
			final cl_program program = CL.clCreateProgramWithSource(context, programsArr.length, programsArr, null, null);
307
308 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::program → NO_COVERAGE
			final var programSpec = gpuEAConfiguration.program();
309 3 1. preEvaluation : replaced call to java/util/Optional::orElse with argument → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/Program::buildOptions → NO_COVERAGE
3. preEvaluation : removed call to java/util/Optional::orElse → NO_COVERAGE
			final var buildOptions = programSpec.buildOptions().orElse(null);
310
			logger.info("\tBuilding program with options: {}", buildOptions);
311 3 1. preEvaluation : replaced call to org/jocl/CL::clBuildProgram with argument → NO_COVERAGE
2. preEvaluation : removed call to org/jocl/CL::clBuildProgram → NO_COVERAGE
3. preEvaluation : Substituted 0 with 1 → NO_COVERAGE
			CL.clBuildProgram(program, 0, null, buildOptions, null, null);
312
313 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/Program::kernelNames → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::program → NO_COVERAGE
			final Set<String> kernelNames = gpuEAConfiguration.program().kernelNames();
314
315 1 1. preEvaluation : removed call to java/util/HashMap::<init> → NO_COVERAGE
			final Map<String, cl_kernel> kernels = new HashMap<>();
316 1 1. preEvaluation : removed call to java/util/HashMap::<init> → NO_COVERAGE
			final Map<String, KernelInfo> kernelInfos = new HashMap<>();
317
			for (final String kernelName : kernelNames) {
318
319
				logger.info("\tCreate kernel {}", kernelName);
320 1 1. preEvaluation : removed call to org/jocl/CL::clCreateKernel → NO_COVERAGE
				final cl_kernel kernel = CL.clCreateKernel(program, kernelName, null);
321
				Validate.notNull(kernel);
322
323 2 1. preEvaluation : replaced call to java/util/Map::put with argument → NO_COVERAGE
2. preEvaluation : removed call to java/util/Map::put → NO_COVERAGE
				kernels.put(kernelName, kernel);
324
325 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/model/Device::deviceId → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/KernelInfoReader::read → NO_COVERAGE
				final var kernelInfo = kernelInfoReader.read(device.deviceId(), kernel, kernelName);
326
				logger.trace("\t{}", kernelInfo);
327 2 1. preEvaluation : replaced call to java/util/Map::put with argument → NO_COVERAGE
2. preEvaluation : removed call to java/util/Map::put → NO_COVERAGE
				kernelInfos.put(kernelName, kernelInfo);
328
			}
329
330 1 1. preEvaluation : removed call to java/util/List::add → NO_COVERAGE
			clContexts.add(context);
331 1 1. preEvaluation : removed call to java/util/List::add → NO_COVERAGE
			clCommandQueues.add(commandQueue);
332 1 1. preEvaluation : removed call to java/util/List::add → NO_COVERAGE
			clKernels.add(kernels);
333 1 1. preEvaluation : removed call to java/util/List::add → NO_COVERAGE
			clPrograms.add(program);
334
335 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext::builder → NO_COVERAGE
			final var openCLExecutionContext = OpenCLExecutionContext.builder()
336 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::platform → NO_COVERAGE
2. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::platform with receiver → NO_COVERAGE
					.platform(platform)
337 2 1. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::device with receiver → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::device → NO_COVERAGE
					.device(device)
338 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clContext → NO_COVERAGE
2. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clContext with receiver → NO_COVERAGE
					.clContext(context)
339 2 1. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clCommandQueue with receiver → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clCommandQueue → NO_COVERAGE
					.clCommandQueue(commandQueue)
340 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernels → NO_COVERAGE
2. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernels with receiver → NO_COVERAGE
					.kernels(kernels)
341 2 1. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernelInfos with receiver → NO_COVERAGE
2. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernelInfos → NO_COVERAGE
					.kernelInfos(kernelInfos)
342 2 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clProgram → NO_COVERAGE
2. preEvaluation : replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clProgram with receiver → NO_COVERAGE
					.clProgram(program)
343 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::build → NO_COVERAGE
					.build();
344
345 1 1. preEvaluation : removed call to java/util/List::add → NO_COVERAGE
			clExecutionContexts.add(openCLExecutionContext);
346
		}
347
348 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::fitness → NO_COVERAGE
		final var fitness = gpuEAConfiguration.fitness();
349 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeAllEvaluations → NO_COVERAGE
		fitness.beforeAllEvaluations();
350
		for (final OpenCLExecutionContext clExecutionContext : clExecutionContexts) {
351 1 1. preEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeAllEvaluations → NO_COVERAGE
			fitness.beforeAllEvaluations(clExecutionContext, executorService);
352
		}
353
	}
354
355
	/**
356
	 * Evaluates fitness for a population of genotypes using GPU acceleration.
357
	 * 
358
	 * <p>This method implements the core fitness evaluation logic by distributing the population across available OpenCL
359
	 * devices and executing fitness computation concurrently. The evaluation process follows these steps:
360
	 * 
361
	 * <ol>
362
	 * <li><strong>Population partitioning</strong>: Divides genotypes across available devices</li>
363
	 * <li><strong>Parallel dispatch</strong>: Submits evaluation tasks to each device asynchronously</li>
364
	 * <li><strong>GPU execution</strong>: Executes OpenCL kernels for fitness computation</li>
365
	 * <li><strong>Result collection</strong>: Gathers fitness values from all devices</li>
366
	 * <li><strong>Result aggregation</strong>: Combines results preserving original order</li>
367
	 * </ol>
368
	 * 
369
	 * <p>Load balancing strategy:
370
	 * <ul>
371
	 * <li>Automatically calculates partition size based on population and device count</li>
372
	 * <li>Round-robin assignment of partitions to devices for balanced workload</li>
373
	 * <li>Asynchronous execution allows devices to work at their optimal pace</li>
374
	 * </ul>
375
	 * 
376
	 * <p>The method coordinates with the configured fitness function through lifecycle hooks:
377
	 * <ul>
378
	 * <li>{@code beforeEvaluation()}: Called before each device partition evaluation</li>
379
	 * <li>{@code compute()}: Executes the actual GPU fitness computation</li>
380
	 * <li>{@code afterEvaluation()}: Called after each device partition completes</li>
381
	 * </ul>
382
	 * 
383
	 * <p>Concurrency and performance:
384
	 * <ul>
385
	 * <li>Multiple devices execute evaluation partitions concurrently</li>
386
	 * <li>CompletableFuture-based coordination for non-blocking execution</li>
387
	 * <li>Automatic workload distribution across available GPU resources</li>
388
	 * </ul>
389
	 * 
390
	 * @param generation the current generation number for context and logging
391
	 * @param genotypes  the population of genotypes to evaluate
392
	 * @return fitness values corresponding to each genotype in the same order
393
	 * @throws IllegalArgumentException if genotypes is null or empty
394
	 * @throws RuntimeException         if GPU evaluation fails or OpenCL errors occur
395
	 */
396
	@Override
397
	public List<T> evaluate(final long generation, final List<Genotype> genotypes) {
398
399 1 1. evaluate : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::fitness → NO_COVERAGE
		final var fitness = gpuEAConfiguration.fitness();
400
401
		/**
402
		 * TODO make it configurable from execution context
403
		 */
404 5 1. evaluate : removed call to java/util/List::size → NO_COVERAGE
2. evaluate : Replaced double division with multiplication → NO_COVERAGE
3. evaluate : removed call to java/util/List::size → NO_COVERAGE
4. evaluate : removed call to java/lang/Math::ceil → NO_COVERAGE
5. evaluate : replaced call to java/lang/Math::ceil with argument → NO_COVERAGE
		final int partitionSize = (int) (Math.ceil((double) genotypes.size() / clExecutionContexts.size()));
405 2 1. evaluate : replaced call to org/apache/commons/collections4/ListUtils::partition with argument → NO_COVERAGE
2. evaluate : removed call to org/apache/commons/collections4/ListUtils::partition → NO_COVERAGE
		final var subGenotypes = ListUtils.partition(genotypes, partitionSize);
406
		logger.debug("Genotype decomposed in {} partition(s)", subGenotypes.size());
407
		if (logger.isTraceEnabled()) {
408 6 1. evaluate : removed conditional - replaced comparison check with true → NO_COVERAGE
2. evaluate : negated conditional → NO_COVERAGE
3. evaluate : changed conditional boundary → NO_COVERAGE
4. evaluate : Substituted 0 with 1 → NO_COVERAGE
5. evaluate : removed call to java/util/List::size → NO_COVERAGE
6. evaluate : removed conditional - replaced comparison check with false → NO_COVERAGE
			for (int i = 0; i < subGenotypes.size(); i++) {
409 1 1. evaluate : removed call to java/util/List::get → NO_COVERAGE
				final List<Genotype> subGenotype = subGenotypes.get(i);
410
				logger.trace("\tPartition {} with {} elements", i, subGenotype.size());
411
			}
412
		}
413
414 1 1. evaluate : removed call to java/util/ArrayList::<init> → NO_COVERAGE
		final List<CompletableFuture<List<T>>> subResultsCF = new ArrayList<>();
415 6 1. evaluate : removed conditional - replaced comparison check with false → NO_COVERAGE
2. evaluate : removed call to java/util/List::size → NO_COVERAGE
3. evaluate : Substituted 0 with 1 → NO_COVERAGE
4. evaluate : removed conditional - replaced comparison check with true → NO_COVERAGE
5. evaluate : negated conditional → NO_COVERAGE
6. evaluate : changed conditional boundary → NO_COVERAGE
		for (int i = 0; i < subGenotypes.size(); i++) {
416 3 1. evaluate : removed call to java/util/List::size → NO_COVERAGE
2. evaluate : removed call to java/util/List::get → NO_COVERAGE
3. evaluate : Replaced integer modulus with multiplication → NO_COVERAGE
			final var openCLExecutionContext = clExecutionContexts.get(i % clExecutionContexts.size());
417 1 1. evaluate : removed call to java/util/List::get → NO_COVERAGE
			final var subGenotype = subGenotypes.get(i);
418
419 1 1. evaluate : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeEvaluation → NO_COVERAGE
			fitness.beforeEvaluation(generation, subGenotype);
420 1 1. evaluate : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeEvaluation → NO_COVERAGE
			fitness.beforeEvaluation(openCLExecutionContext, executorService, generation, subGenotype);
421
422 1 1. evaluate : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::compute → NO_COVERAGE
			final var resultsCF = fitness.compute(openCLExecutionContext, executorService, generation, subGenotype)
423 2 1. evaluate : removed call to java/util/concurrent/CompletableFuture::thenApply → NO_COVERAGE
2. evaluate : replaced call to java/util/concurrent/CompletableFuture::thenApply with receiver → NO_COVERAGE
					.thenApply((results) -> {
424
425 1 1. lambda$evaluate$7 : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterEvaluation → NO_COVERAGE
						fitness.afterEvaluation(openCLExecutionContext, executorService, generation, subGenotype);
426 1 1. lambda$evaluate$7 : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterEvaluation → NO_COVERAGE
						fitness.afterEvaluation(generation, subGenotype);
427
428 1 1. lambda$evaluate$7 : replaced return value with Collections.emptyList for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$evaluate$7 → NO_COVERAGE
						return results;
429
					});
430
431 1 1. evaluate : removed call to java/util/List::add → NO_COVERAGE
			subResultsCF.add(resultsCF);
432
		}
433
434 2 1. evaluate : removed call to java/util/ArrayList::<init> → NO_COVERAGE
2. evaluate : removed call to java/util/List::size → NO_COVERAGE
		final List<T> resultsEvaluation = new ArrayList<>(genotypes.size());
435
		for (final CompletableFuture<List<T>> subResultCF : subResultsCF) {
436 1 1. evaluate : removed call to java/util/concurrent/CompletableFuture::join → NO_COVERAGE
			final var fitnessResults = subResultCF.join();
437 1 1. evaluate : removed call to java/util/List::addAll → NO_COVERAGE
			resultsEvaluation.addAll(fitnessResults);
438
		}
439 1 1. evaluate : replaced return value with Collections.emptyList for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::evaluate → NO_COVERAGE
		return resultsEvaluation;
440
	}
441
442
	/**
443
	 * Cleans up OpenCL resources and releases GPU memory after evaluation completion.
444
	 * 
445
	 * <p>This method performs comprehensive cleanup of all OpenCL resources in the proper order to prevent memory leaks
446
	 * and ensure clean shutdown. The cleanup sequence follows OpenCL best practices for resource deallocation:
447
	 * 
448
	 * <ol>
449
	 * <li><strong>Fitness cleanup</strong>: Calls lifecycle hooks on the fitness function</li>
450
	 * <li><strong>Kernel release</strong>: Releases all compiled kernel objects</li>
451
	 * <li><strong>Program release</strong>: Releases compiled OpenCL programs</li>
452
	 * <li><strong>Queue release</strong>: Releases command queues and pending operations</li>
453
	 * <li><strong>Context release</strong>: Releases OpenCL contexts and associated memory</li>
454
	 * <li><strong>Reference cleanup</strong>: Clears internal data structures and references</li>
455
	 * </ol>
456
	 * 
457
	 * <p>Resource management guarantees:
458
	 * <ul>
459
	 * <li>All GPU memory allocations are properly released</li>
460
	 * <li>OpenCL objects are released in dependency order to avoid errors</li>
461
	 * <li>No resource leaks occur even if individual cleanup operations fail</li>
462
	 * <li>Evaluator returns to a clean state ready for potential reinitialization</li>
463
	 * </ul>
464
	 * 
465
	 * <p>The method coordinates with the configured fitness function to ensure any fitness-specific resources (buffers,
466
	 * textures, etc.) are also properly cleaned up through the {@code afterAllEvaluations()} lifecycle hooks.
467
	 * 
468
	 * @throws RuntimeException if cleanup operations fail (logged but not propagated to prevent interference with EA
469
	 *                          system shutdown)
470
	 */
471
	@Override
472
	public void postEvaluation() {
473
474 1 1. postEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::fitness → NO_COVERAGE
		final var fitness = gpuEAConfiguration.fitness();
475
476
		for (final OpenCLExecutionContext clExecutionContext : clExecutionContexts) {
477 1 1. postEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterAllEvaluations → NO_COVERAGE
			fitness.afterAllEvaluations(clExecutionContext, executorService);
478
		}
479 1 1. postEvaluation : removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterAllEvaluations → NO_COVERAGE
		fitness.afterAllEvaluations();
480
481
		logger.debug("Releasing kernels");
482
483
		for (final Map<String, cl_kernel> kernels : clKernels) {
484 1 1. postEvaluation : removed call to java/util/Map::values → NO_COVERAGE
			for (final cl_kernel clKernel : kernels.values()) {
485 1 1. postEvaluation : removed call to org/jocl/CL::clReleaseKernel → NO_COVERAGE
				CL.clReleaseKernel(clKernel);
486
			}
487
		}
488 1 1. postEvaluation : removed call to java/util/List::clear → NO_COVERAGE
		clKernels.clear();
489
490
		logger.debug("Releasing programs");
491
		for (final cl_program clProgram : clPrograms) {
492 1 1. postEvaluation : removed call to org/jocl/CL::clReleaseProgram → NO_COVERAGE
			CL.clReleaseProgram(clProgram);
493
		}
494 1 1. postEvaluation : removed call to java/util/List::clear → NO_COVERAGE
		clPrograms.clear();
495
496
		logger.debug("Releasing command queues");
497
		for (final cl_command_queue clCommandQueue : clCommandQueues) {
498 1 1. postEvaluation : removed call to org/jocl/CL::clReleaseCommandQueue → NO_COVERAGE
			CL.clReleaseCommandQueue(clCommandQueue);
499
		}
500 1 1. postEvaluation : removed call to java/util/List::clear → NO_COVERAGE
		clCommandQueues.clear();
501
502
		logger.debug("Releasing contexts");
503
		for (final cl_context clContext : clContexts) {
504 1 1. postEvaluation : removed call to org/jocl/CL::clReleaseContext → NO_COVERAGE
			CL.clReleaseContext(clContext);
505
		}
506 1 1. postEvaluation : removed call to java/util/List::clear → NO_COVERAGE
		clContexts.clear();
507
508 1 1. postEvaluation : removed call to java/util/List::clear → NO_COVERAGE
		clExecutionContexts.clear();
509 1 1. postEvaluation : Removed assignment to member variable selectedPlatformToDevice → NO_COVERAGE
		selectedPlatformToDevice = null;
510
511 1 1. postEvaluation : removed call to net/bmahe/genetics4j/core/evaluation/FitnessEvaluator::postEvaluation → NO_COVERAGE
		FitnessEvaluator.super.postEvaluation();
512
	}
513
}

Mutations

142

1.1
Location : <init>
Killed by : none
Removed assignment to member variable clContexts → NO_COVERAGE

2.2
Location : <init>
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

143

1.1
Location : <init>
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

2.2
Location : <init>
Killed by : none
Removed assignment to member variable clCommandQueues → NO_COVERAGE

144

1.1
Location : <init>
Killed by : none
Removed assignment to member variable clPrograms → NO_COVERAGE

2.2
Location : <init>
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

145

1.1
Location : <init>
Killed by : none
Removed assignment to member variable clKernels → NO_COVERAGE

2.2
Location : <init>
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

146

1.1
Location : <init>
Killed by : none
Removed assignment to member variable clExecutionContexts → NO_COVERAGE

2.2
Location : <init>
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

169

1.1
Location : <init>
Killed by : none
Removed assignment to member variable gpuEAExecutionContext → NO_COVERAGE

170

1.1
Location : <init>
Killed by : none
Removed assignment to member variable gpuEAConfiguration → NO_COVERAGE

171

1.1
Location : <init>
Killed by : none
Removed assignment to member variable executorService → NO_COVERAGE

173

1.1
Location : <init>
Killed by : none
removed call to org/jocl/CL::setExceptionsEnabled → NO_COVERAGE

2.2
Location : <init>
Killed by : none
Substituted 1 with 0 → NO_COVERAGE

180

1.1
Location : loadResource
Killed by : none
replaced call to org/apache/commons/io/IOUtils::resourceToString with argument → NO_COVERAGE

2.2
Location : loadResource
Killed by : none
replaced return value with "" for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::loadResource → NO_COVERAGE

3.3
Location : loadResource
Killed by : none
removed call to org/apache/commons/io/IOUtils::resourceToString → NO_COVERAGE

182

1.1
Location : loadResource
Killed by : none
removed call to java/lang/IllegalStateException::<init> → NO_COVERAGE

187

1.1
Location : grabProgramSources
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::program → NO_COVERAGE

191

1.1
Location : grabProgramSources
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

193

1.1
Location : grabProgramSources
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/Program::content → NO_COVERAGE

2.2
Location : grabProgramSources
Killed by : none
removed call to java/util/List::addAll → NO_COVERAGE

195

1.1
Location : lambda$grabProgramSources$0
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::loadResource with argument → NO_COVERAGE

2.2
Location : grabProgramSources
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/Program::resources → NO_COVERAGE

3.3
Location : lambda$grabProgramSources$0
Killed by : none
replaced return value with "" for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$grabProgramSources$0 → NO_COVERAGE

4.4
Location : grabProgramSources
Killed by : none
replaced call to java/util/stream/Stream::map with receiver → NO_COVERAGE

5.5
Location : lambda$grabProgramSources$0
Killed by : none
removed call to net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::loadResource → NO_COVERAGE

6.6
Location : grabProgramSources
Killed by : none
removed call to java/util/Set::stream → NO_COVERAGE

7.7
Location : grabProgramSources
Killed by : none
removed call to java/util/stream/Stream::map → NO_COVERAGE

8.8
Location : grabProgramSources
Killed by : none
removed call to java/util/stream/Stream::forEach → NO_COVERAGE

196

1.1
Location : lambda$grabProgramSources$1
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

199

1.1
Location : grabProgramSources
Killed by : none
replaced return value with Collections.emptyList for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::grabProgramSources → NO_COVERAGE

233

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/core/evaluation/FitnessEvaluator::preEvaluation → NO_COVERAGE

235

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/PlatformReader::<init> → NO_COVERAGE

236

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/DeviceReader::<init> → NO_COVERAGE

237

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/KernelInfoReader::<init> → NO_COVERAGE

239

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/PlatformUtils::numPlatforms → NO_COVERAGE

242

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/PlatformUtils::platformIds → NO_COVERAGE

245

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAExecutionContext::platformFilters → NO_COVERAGE

246

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAExecutionContext::deviceFilters → NO_COVERAGE

248

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::stream → NO_COVERAGE

250

1.1
Location : preEvaluation
Killed by : none
replaced call to java/util/stream/Stream::filter with receiver → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/stream/Stream::filter → NO_COVERAGE

251

1.1
Location : preEvaluation
Killed by : none
replaced call to java/util/stream/Stream::flatMap with receiver → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/stream/Stream::flatMap → NO_COVERAGE

252

1.1
Location : lambda$preEvaluation$3
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/model/Platform::platformId → NO_COVERAGE

253

1.1
Location : lambda$preEvaluation$3
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/DeviceUtils::numDevices → NO_COVERAGE

256

1.1
Location : lambda$preEvaluation$3
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/DeviceUtils::getDeviceIds → NO_COVERAGE

257

1.1
Location : lambda$preEvaluation$3
Killed by : none
replaced return value with Stream.empty for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$3 → NO_COVERAGE

2.2
Location : lambda$preEvaluation$3
Killed by : none
removed call to java/util/stream/Stream::map → NO_COVERAGE

3.3
Location : lambda$preEvaluation$3
Killed by : none
removed call to java/util/List::stream → NO_COVERAGE

4.4
Location : lambda$preEvaluation$2
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::of → NO_COVERAGE

5.5
Location : lambda$preEvaluation$3
Killed by : none
replaced call to java/util/stream/Stream::map with receiver → NO_COVERAGE

6.6
Location : lambda$preEvaluation$2
Killed by : none
replaced return value with null for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$2 → NO_COVERAGE

259

1.1
Location : preEvaluation
Killed by : none
replaced call to java/util/stream/Stream::map with receiver → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/stream/Stream::map → NO_COVERAGE

260

1.1
Location : lambda$preEvaluation$4
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::getLeft → NO_COVERAGE

261

1.1
Location : lambda$preEvaluation$4
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/model/Platform::platformId → NO_COVERAGE

262

1.1
Location : lambda$preEvaluation$4
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::getRight → NO_COVERAGE

264

1.1
Location : lambda$preEvaluation$4
Killed by : none
replaced return value with null for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$4 → NO_COVERAGE

2.2
Location : lambda$preEvaluation$4
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/DeviceReader::read → NO_COVERAGE

3.3
Location : lambda$preEvaluation$4
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::of → NO_COVERAGE

266

1.1
Location : lambda$preEvaluation$5
Killed by : none
removed call to java/util/function/Predicate::test → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/stream/Stream::filter → NO_COVERAGE

3.3
Location : lambda$preEvaluation$5
Killed by : none
replaced boolean return with true for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$5 → NO_COVERAGE

4.4
Location : lambda$preEvaluation$5
Killed by : none
replaced boolean return with false for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$preEvaluation$5 → NO_COVERAGE

5.5
Location : lambda$preEvaluation$5
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::getRight → NO_COVERAGE

6.6
Location : preEvaluation
Killed by : none
replaced call to java/util/stream/Stream::filter with receiver → NO_COVERAGE

267

1.1
Location : preEvaluation
Killed by : none
Removed assignment to member variable selectedPlatformToDevice → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/stream/Stream::toList → NO_COVERAGE

272

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::forEach → NO_COVERAGE

281

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::grabProgramSources → NO_COVERAGE

282

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
replaced call to java/util/List::toArray with argument → NO_COVERAGE

3.3
Location : preEvaluation
Killed by : none
removed call to java/util/List::toArray → NO_COVERAGE

285

1.1
Location : preEvaluation
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::getLeft → NO_COVERAGE

286

1.1
Location : preEvaluation
Killed by : none
removed call to org/apache/commons/lang3/tuple/Pair::getRight → NO_COVERAGE

291

1.1
Location : preEvaluation
Killed by : none
removed call to org/jocl/cl_context_properties::<init> → NO_COVERAGE

292

1.1
Location : preEvaluation
Killed by : none
Substituted 4228 with 4229 → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/model/Platform::platformId → NO_COVERAGE

3.3
Location : preEvaluation
Killed by : none
removed call to org/jocl/cl_context_properties::addProperty → NO_COVERAGE

294

1.1
Location : preEvaluation
Killed by : none
Substituted 1 with 0 → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
Substituted 0 with 1 → NO_COVERAGE

3.3
Location : preEvaluation
Killed by : none
Substituted 1 with 0 → NO_COVERAGE

295

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/model/Device::deviceId → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to org/jocl/CL::clCreateContext → NO_COVERAGE

298

1.1
Location : preEvaluation
Killed by : none
removed call to org/jocl/cl_queue_properties::<init> → NO_COVERAGE

299

1.1
Location : preEvaluation
Killed by : none
Substituted 4243 with 4244 → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
Substituted 3 with 4 → NO_COVERAGE

3.3
Location : preEvaluation
Killed by : none
removed call to org/jocl/cl_queue_properties::addProperty → NO_COVERAGE

303

1.1
Location : preEvaluation
Killed by : none
removed call to org/jocl/CL::clCreateCommandQueueWithProperties → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/model/Device::deviceId → NO_COVERAGE

306

1.1
Location : preEvaluation
Killed by : none
removed call to org/jocl/CL::clCreateProgramWithSource → NO_COVERAGE

308

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::program → NO_COVERAGE

309

1.1
Location : preEvaluation
Killed by : none
replaced call to java/util/Optional::orElse with argument → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/Program::buildOptions → NO_COVERAGE

3.3
Location : preEvaluation
Killed by : none
removed call to java/util/Optional::orElse → NO_COVERAGE

311

1.1
Location : preEvaluation
Killed by : none
replaced call to org/jocl/CL::clBuildProgram with argument → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to org/jocl/CL::clBuildProgram → NO_COVERAGE

3.3
Location : preEvaluation
Killed by : none
Substituted 0 with 1 → NO_COVERAGE

313

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/Program::kernelNames → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::program → NO_COVERAGE

315

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/HashMap::<init> → NO_COVERAGE

316

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/HashMap::<init> → NO_COVERAGE

320

1.1
Location : preEvaluation
Killed by : none
removed call to org/jocl/CL::clCreateKernel → NO_COVERAGE

323

1.1
Location : preEvaluation
Killed by : none
replaced call to java/util/Map::put with argument → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/Map::put → NO_COVERAGE

325

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/model/Device::deviceId → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/KernelInfoReader::read → NO_COVERAGE

327

1.1
Location : preEvaluation
Killed by : none
replaced call to java/util/Map::put with argument → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to java/util/Map::put → NO_COVERAGE

330

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

331

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

332

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

333

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

335

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext::builder → NO_COVERAGE

336

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::platform → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::platform with receiver → NO_COVERAGE

337

1.1
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::device with receiver → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::device → NO_COVERAGE

338

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clContext → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clContext with receiver → NO_COVERAGE

339

1.1
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clCommandQueue with receiver → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clCommandQueue → NO_COVERAGE

340

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernels → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernels with receiver → NO_COVERAGE

341

1.1
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernelInfos with receiver → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::kernelInfos → NO_COVERAGE

342

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clProgram → NO_COVERAGE

2.2
Location : preEvaluation
Killed by : none
replaced call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::clProgram with receiver → NO_COVERAGE

343

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/opencl/OpenCLExecutionContext$Builder::build → NO_COVERAGE

345

1.1
Location : preEvaluation
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

348

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::fitness → NO_COVERAGE

349

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeAllEvaluations → NO_COVERAGE

351

1.1
Location : preEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeAllEvaluations → NO_COVERAGE

399

1.1
Location : evaluate
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::fitness → NO_COVERAGE

404

1.1
Location : evaluate
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
Replaced double division with multiplication → NO_COVERAGE

3.3
Location : evaluate
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

4.4
Location : evaluate
Killed by : none
removed call to java/lang/Math::ceil → NO_COVERAGE

5.5
Location : evaluate
Killed by : none
replaced call to java/lang/Math::ceil with argument → NO_COVERAGE

405

1.1
Location : evaluate
Killed by : none
replaced call to org/apache/commons/collections4/ListUtils::partition with argument → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
removed call to org/apache/commons/collections4/ListUtils::partition → NO_COVERAGE

408

1.1
Location : evaluate
Killed by : none
removed conditional - replaced comparison check with true → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
negated conditional → NO_COVERAGE

3.3
Location : evaluate
Killed by : none
changed conditional boundary → NO_COVERAGE

4.4
Location : evaluate
Killed by : none
Substituted 0 with 1 → NO_COVERAGE

5.5
Location : evaluate
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

6.6
Location : evaluate
Killed by : none
removed conditional - replaced comparison check with false → NO_COVERAGE

409

1.1
Location : evaluate
Killed by : none
removed call to java/util/List::get → NO_COVERAGE

414

1.1
Location : evaluate
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

415

1.1
Location : evaluate
Killed by : none
removed conditional - replaced comparison check with false → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

3.3
Location : evaluate
Killed by : none
Substituted 0 with 1 → NO_COVERAGE

4.4
Location : evaluate
Killed by : none
removed conditional - replaced comparison check with true → NO_COVERAGE

5.5
Location : evaluate
Killed by : none
negated conditional → NO_COVERAGE

6.6
Location : evaluate
Killed by : none
changed conditional boundary → NO_COVERAGE

416

1.1
Location : evaluate
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
removed call to java/util/List::get → NO_COVERAGE

3.3
Location : evaluate
Killed by : none
Replaced integer modulus with multiplication → NO_COVERAGE

417

1.1
Location : evaluate
Killed by : none
removed call to java/util/List::get → NO_COVERAGE

419

1.1
Location : evaluate
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeEvaluation → NO_COVERAGE

420

1.1
Location : evaluate
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::beforeEvaluation → NO_COVERAGE

422

1.1
Location : evaluate
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::compute → NO_COVERAGE

423

1.1
Location : evaluate
Killed by : none
removed call to java/util/concurrent/CompletableFuture::thenApply → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
replaced call to java/util/concurrent/CompletableFuture::thenApply with receiver → NO_COVERAGE

425

1.1
Location : lambda$evaluate$7
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterEvaluation → NO_COVERAGE

426

1.1
Location : lambda$evaluate$7
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterEvaluation → NO_COVERAGE

428

1.1
Location : lambda$evaluate$7
Killed by : none
replaced return value with Collections.emptyList for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::lambda$evaluate$7 → NO_COVERAGE

431

1.1
Location : evaluate
Killed by : none
removed call to java/util/List::add → NO_COVERAGE

434

1.1
Location : evaluate
Killed by : none
removed call to java/util/ArrayList::<init> → NO_COVERAGE

2.2
Location : evaluate
Killed by : none
removed call to java/util/List::size → NO_COVERAGE

436

1.1
Location : evaluate
Killed by : none
removed call to java/util/concurrent/CompletableFuture::join → NO_COVERAGE

437

1.1
Location : evaluate
Killed by : none
removed call to java/util/List::addAll → NO_COVERAGE

439

1.1
Location : evaluate
Killed by : none
replaced return value with Collections.emptyList for net/bmahe/genetics4j/gpu/GPUFitnessEvaluator::evaluate → NO_COVERAGE

474

1.1
Location : postEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/GPUEAConfiguration::fitness → NO_COVERAGE

477

1.1
Location : postEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterAllEvaluations → NO_COVERAGE

479

1.1
Location : postEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/gpu/spec/fitness/OpenCLFitness::afterAllEvaluations → NO_COVERAGE

484

1.1
Location : postEvaluation
Killed by : none
removed call to java/util/Map::values → NO_COVERAGE

485

1.1
Location : postEvaluation
Killed by : none
removed call to org/jocl/CL::clReleaseKernel → NO_COVERAGE

488

1.1
Location : postEvaluation
Killed by : none
removed call to java/util/List::clear → NO_COVERAGE

492

1.1
Location : postEvaluation
Killed by : none
removed call to org/jocl/CL::clReleaseProgram → NO_COVERAGE

494

1.1
Location : postEvaluation
Killed by : none
removed call to java/util/List::clear → NO_COVERAGE

498

1.1
Location : postEvaluation
Killed by : none
removed call to org/jocl/CL::clReleaseCommandQueue → NO_COVERAGE

500

1.1
Location : postEvaluation
Killed by : none
removed call to java/util/List::clear → NO_COVERAGE

504

1.1
Location : postEvaluation
Killed by : none
removed call to org/jocl/CL::clReleaseContext → NO_COVERAGE

506

1.1
Location : postEvaluation
Killed by : none
removed call to java/util/List::clear → NO_COVERAGE

508

1.1
Location : postEvaluation
Killed by : none
removed call to java/util/List::clear → NO_COVERAGE

509

1.1
Location : postEvaluation
Killed by : none
Removed assignment to member variable selectedPlatformToDevice → NO_COVERAGE

511

1.1
Location : postEvaluation
Killed by : none
removed call to net/bmahe/genetics4j/core/evaluation/FitnessEvaluator::postEvaluation → NO_COVERAGE

Active mutators

Tests examined


Report generated by PIT 1.20.3