Skip to content

Commit 9731b1c

Browse files
committedApr 10, 2024
8327137: Add test for ConcurrentModificationException in BasicDirectoryModel
Reviewed-by: serb, tr
1 parent c5150c7 commit 9731b1c

File tree

1 file changed

+273
-0
lines changed

1 file changed

+273
-0
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,273 @@
1+
/*
2+
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import java.io.IOException;
25+
import java.nio.file.Files;
26+
import java.nio.file.Path;
27+
import java.nio.file.Paths;
28+
import java.util.ArrayList;
29+
import java.util.List;
30+
import java.util.Timer;
31+
import java.util.TimerTask;
32+
import java.util.concurrent.BrokenBarrierException;
33+
import java.util.concurrent.CyclicBarrier;
34+
import java.util.concurrent.atomic.AtomicReference;
35+
import java.util.stream.IntStream;
36+
import java.util.stream.LongStream;
37+
38+
import javax.swing.JFileChooser;
39+
40+
/*
41+
* @test
42+
* @bug 8323670 8307091 8240690
43+
* @requires os.family == "mac" | os.family == "linux"
44+
* @summary Verifies thread-safety of BasicDirectoryModel (JFileChooser)
45+
* @run main/othervm -Djava.awt.headless=true ConcurrentModification
46+
*/
47+
public final class ConcurrentModification extends ThreadGroup {
48+
/** Initial number of files. */
49+
private static final long NUMBER_OF_FILES = 50;
50+
/** Maximum number of files created on a timer tick. */
51+
private static final long LIMIT_FILES = 10;
52+
53+
/** Timer period (delay) for creating new files. */
54+
private static final long TIMER_PERIOD = 250;
55+
56+
/**
57+
* Number of threads running {@code fileChooser.rescanCurrentDirectory()}.
58+
*/
59+
private static final int NUMBER_OF_THREADS = 5;
60+
/** Number of repeated calls to {@code rescanCurrentDirectory}. */
61+
private static final int NUMBER_OF_REPEATS = 2_000;
62+
/** Maximum amount a thread waits before initiating rescan. */
63+
private static final long LIMIT_SLEEP = 100;
64+
65+
66+
/** The barrier to start all the scanner threads simultaneously. */
67+
private static final CyclicBarrier start = new CyclicBarrier(NUMBER_OF_THREADS);
68+
/** The barrier to wait for all the scanner threads to complete, plus main thread. */
69+
private static final CyclicBarrier end = new CyclicBarrier(NUMBER_OF_THREADS + 1);
70+
71+
/** List of scanner threads. */
72+
private static final List<Thread> threads = new ArrayList<>(NUMBER_OF_THREADS);
73+
74+
/**
75+
* Stores an exception caught by any of the threads.
76+
* If more exceptions are caught, they're added as suppressed exceptions.
77+
*/
78+
private static final AtomicReference<Throwable> exception =
79+
new AtomicReference<>();
80+
81+
/**
82+
* Stores an {@code IOException} thrown while removing the files.
83+
*/
84+
private static final AtomicReference<IOException> ioException =
85+
new AtomicReference<>();
86+
87+
88+
public static void main(String[] args) throws Throwable {
89+
try {
90+
// Start the test in its own thread group to catch and handle
91+
// all thrown exceptions, in particular in
92+
// BasicDirectoryModel.FilesLoader which is created by Swing.
93+
ThreadGroup threadGroup = new ConcurrentModification();
94+
Thread runner = new Thread(threadGroup,
95+
ConcurrentModification::wrapper,
96+
"Test Runner");
97+
runner.start();
98+
runner.join();
99+
} catch (Throwable throwable) {
100+
handleException(throwable);
101+
}
102+
103+
if (ioException.get() != null) {
104+
System.err.println("An error occurred while removing files:");
105+
ioException.get().printStackTrace();
106+
}
107+
108+
if (exception.get() != null) {
109+
throw exception.get();
110+
}
111+
}
112+
113+
private static void wrapper() {
114+
final long timeStart = System.currentTimeMillis();
115+
try {
116+
runTest(timeStart);
117+
} catch (Throwable throwable) {
118+
handleException(throwable);
119+
} finally {
120+
System.out.printf("Duration: %,d\n",
121+
(System.currentTimeMillis() - timeStart));
122+
}
123+
}
124+
125+
private static void runTest(final long timeStart) throws Throwable {
126+
final Path temp = Files.createDirectory(Paths.get("fileChooser-concurrency-" + timeStart));
127+
128+
final Timer timer = new Timer("File creator");
129+
130+
try {
131+
createFiles(temp);
132+
133+
final JFileChooser fc = new JFileChooser(temp.toFile());
134+
135+
IntStream.range(0, NUMBER_OF_THREADS)
136+
.forEach(i -> {
137+
Thread thread = new Thread(new Scanner(fc));
138+
threads.add(thread);
139+
thread.start();
140+
});
141+
142+
timer.scheduleAtFixedRate(new CreateFilesTimerTask(temp),
143+
0, TIMER_PERIOD);
144+
145+
end.await();
146+
} catch (Throwable e) {
147+
threads.forEach(Thread::interrupt);
148+
throw e;
149+
} finally {
150+
timer.cancel();
151+
152+
deleteFiles(temp);
153+
deleteFile(temp);
154+
}
155+
}
156+
157+
158+
private ConcurrentModification() {
159+
super("bdmConcurrency");
160+
}
161+
162+
@Override
163+
public void uncaughtException(Thread t, Throwable e) {
164+
handleException(t, e);
165+
}
166+
167+
private static void handleException(Throwable throwable) {
168+
handleException(Thread.currentThread(), throwable);
169+
}
170+
171+
private static void handleException(final Thread thread,
172+
final Throwable throwable) {
173+
System.err.println("Exception in " + thread.getName() + ": "
174+
+ throwable.getClass()
175+
+ (throwable.getMessage() != null
176+
? ": " + throwable.getMessage()
177+
: ""));
178+
if (!exception.compareAndSet(null, throwable)) {
179+
exception.get().addSuppressed(throwable);
180+
}
181+
threads.stream()
182+
.filter(t -> t != thread)
183+
.forEach(Thread::interrupt);
184+
}
185+
186+
187+
private record Scanner(JFileChooser fileChooser)
188+
implements Runnable {
189+
190+
@Override
191+
public void run() {
192+
try {
193+
start.await();
194+
195+
int counter = 0;
196+
try {
197+
do {
198+
fileChooser.rescanCurrentDirectory();
199+
Thread.sleep((long) (Math.random() * LIMIT_SLEEP));
200+
} while (++counter < NUMBER_OF_REPEATS
201+
&& !Thread.interrupted());
202+
} catch (InterruptedException e) {
203+
// Just exit the loop
204+
}
205+
} catch (Throwable throwable) {
206+
handleException(throwable);
207+
} finally {
208+
try {
209+
end.await();
210+
} catch (InterruptedException | BrokenBarrierException e) {
211+
handleException(e);
212+
}
213+
}
214+
}
215+
}
216+
217+
private static void createFiles(final Path parent) {
218+
createFiles(parent, 0, NUMBER_OF_FILES);
219+
}
220+
221+
private static void createFiles(final Path parent,
222+
final long start,
223+
final long end) {
224+
LongStream.range(start, end)
225+
.forEach(n -> createFile(parent.resolve(n + ".file")));
226+
}
227+
228+
private static void createFile(final Path file) {
229+
try {
230+
Files.createFile(file);
231+
} catch (IOException e) {
232+
throw new RuntimeException(e);
233+
}
234+
}
235+
236+
private static void deleteFiles(final Path parent) throws IOException {
237+
try (var stream = Files.walk(parent)) {
238+
stream.filter(p -> p != parent)
239+
.forEach(ConcurrentModification::deleteFile);
240+
}
241+
}
242+
243+
private static void deleteFile(final Path file) {
244+
try {
245+
Files.delete(file);
246+
} catch (IOException e) {
247+
if (!ioException.compareAndSet(null, e)) {
248+
ioException.get().addSuppressed(e);
249+
}
250+
}
251+
}
252+
253+
private static final class CreateFilesTimerTask extends TimerTask {
254+
private final Path temp;
255+
private long no;
256+
257+
public CreateFilesTimerTask(Path temp) {
258+
this.temp = temp;
259+
no = NUMBER_OF_FILES;
260+
}
261+
262+
@Override
263+
public void run() {
264+
try {
265+
long count = (long) (Math.random() * LIMIT_FILES);
266+
createFiles(temp, no, no + count);
267+
no += count;
268+
} catch (Throwable t) {
269+
handleException(t);
270+
}
271+
}
272+
}
273+
}

0 commit comments

Comments
 (0)
Please sign in to comment.