From a714c6394679dd88a76b8c991400545b0c366fe5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=B6rg=20Kubitz?= <51790620+jukzi@users.noreply.github.com> Date: Mon, 12 Feb 2024 21:06:42 +0100 Subject: [PATCH] ChangedValueChecker: multithreading fixes (#1187) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * use concurrent datastructures, volatile fConflict * do not spinloop burning cpu Co-authored-by: Jörg Kubitz --- .../refactoring/util/ChangedValueChecker.java | 43 +++++++++---------- 1 file changed, 21 insertions(+), 22 deletions(-) diff --git a/org.eclipse.jdt.core.manipulation/refactoring/org/eclipse/jdt/internal/corext/refactoring/util/ChangedValueChecker.java b/org.eclipse.jdt.core.manipulation/refactoring/org/eclipse/jdt/internal/corext/refactoring/util/ChangedValueChecker.java index d7de24a3b27..887f85ed72b 100644 --- a/org.eclipse.jdt.core.manipulation/refactoring/org/eclipse/jdt/internal/corext/refactoring/util/ChangedValueChecker.java +++ b/org.eclipse.jdt.core.manipulation/refactoring/org/eclipse/jdt/internal/corext/refactoring/util/ChangedValueChecker.java @@ -16,10 +16,10 @@ import java.util.ArrayList; -import java.util.Collections; import java.util.HashSet; import java.util.Set; import java.util.concurrent.ArrayBlockingQueue; +import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ExecutorService; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; @@ -74,7 +74,7 @@ public class ChangedValueChecker extends AbstractChecker { private ASTNode fNode2; - private HashSet fDependSet; + private Set fDependSet; private ASTNode fBodyNode; @@ -82,7 +82,7 @@ public class ChangedValueChecker extends AbstractChecker { private ArrayList fMiddleNodes; - private boolean fConflict; + private volatile boolean fConflict; private Set fPosSet; @@ -98,7 +98,7 @@ public void detectConflict(int startOffset, int endOffset, ASTNode node, fNode2= node; fBodyNode= bodyNode; fConflict= false; - fPosSet= Collections.synchronizedSet(new HashSet<>()); + fPosSet= ConcurrentHashMap.newKeySet(); PathVisitor pathVisitor= new PathVisitor(startOffset, endOffset, fNode2, candidateList); while (fBodyNode != null && (fBodyNode.getStartPosition() + fBodyNode.getLength() < pathVisitor.endOffset || fBodyNode.getStartPosition() > pathVisitor.startOffset)) { @@ -110,7 +110,7 @@ public void detectConflict(int startOffset, int endOffset, ASTNode node, } public void analyzeSelectedExpression(ASTNode selectedExpression) { - fDependSet= new HashSet<>(); + fDependSet= ConcurrentHashMap.newKeySet(); ReadVisitor readVisitor= new ReadVisitor(true); selectedExpression.accept(readVisitor); fDependSet.addAll(readVisitor.readSet); @@ -121,25 +121,24 @@ public boolean hasConflict() { new ArrayBlockingQueue<>(10), new ThreadPoolExecutor.CallerRunsPolicy()); try { for (ASTNode node : fMiddleNodes) { - Position pos= new Position(node.getStartPosition(), node.getLength()); - if (fPosSet.contains(pos)) { - continue; - } - if (fConflict == true) { + if (fConflict) { break; } - threadPool.execute(() -> { - fPosSet.add(pos); - UpdateVisitor uv= new UpdateVisitor(fDependSet, true); - node.accept(uv); - if (uv.hasConflict()) - fConflict= true; - }); + Position pos= new Position(node.getStartPosition(), node.getLength()); + if (fPosSet.add(pos)) { + threadPool.execute(() -> { + UpdateVisitor uv= new UpdateVisitor(fDependSet, true); + node.accept(uv); + if (uv.hasConflict()) { + fConflict= true; + } + }); + } } - if (fConflict == false) { + if (!fConflict) { threadPool.shutdown(); - threadPool.awaitTermination(5, TimeUnit.SECONDS); - while (!threadPool.isTerminated() && fConflict == false) { + while (!threadPool.isTerminated() && !fConflict) { + threadPool.awaitTermination(10, TimeUnit.MILLISECONDS); } } } catch (InterruptedException e) { @@ -506,11 +505,11 @@ class UpdateVisitor extends ASTVisitor { private HashSet updateSet; - private HashSet dependSet; + private Set dependSet; private boolean visitMethodCall; - public UpdateVisitor(HashSet dependSet, boolean visitMethodCall) { + public UpdateVisitor(Set dependSet, boolean visitMethodCall) { this.updateSet= new HashSet<>(); this.visitMethodCall= visitMethodCall; this.dependSet= dependSet;