ViewVC Help
View File | Revision Log | Show Annotations | Download File | Root Listing
root/jsr166/jsr166/src/test/jtreg/util/Collections/RacingCollections.java
Revision: 1.9
Committed: Mon Jan 8 03:12:03 2018 UTC (6 years, 4 months ago) by jsr166
Branch: MAIN
CVS Tags: HEAD
Changes since 1.8: +40 -3 lines
Log Message:
organize imports

File Contents

# Content
1 /*
2 * Copyright (c) 2006, 2010, 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 /*
25 * @test
26 * @bug 6360946 6360948
27 * @summary Test various operations on concurrently mutating collections
28 * @author Martin Buchholz
29 */
30
31 import java.util.ArrayDeque;
32 import java.util.ArrayList;
33 import java.util.Collection;
34 import java.util.Collections;
35 import java.util.Comparator;
36 import java.util.Deque;
37 import java.util.HashMap;
38 import java.util.HashSet;
39 import java.util.Hashtable;
40 import java.util.LinkedList;
41 import java.util.List;
42 import java.util.Map;
43 import java.util.Queue;
44 import java.util.Set;
45 import java.util.TreeMap;
46 import java.util.TreeSet;
47 import java.util.Vector;
48 import java.util.concurrent.ArrayBlockingQueue;
49 import java.util.concurrent.ConcurrentHashMap;
50 import java.util.concurrent.ConcurrentLinkedDeque;
51 import java.util.concurrent.ConcurrentLinkedQueue;
52 import java.util.concurrent.ConcurrentSkipListMap;
53 import java.util.concurrent.ConcurrentSkipListSet;
54 import java.util.concurrent.CopyOnWriteArrayList;
55 import java.util.concurrent.CopyOnWriteArraySet;
56 import java.util.concurrent.LinkedBlockingDeque;
57 import java.util.concurrent.LinkedBlockingQueue;
58 import java.util.concurrent.LinkedTransferQueue;
59
60 import static java.util.Collections.asLifoQueue;
61 import static java.util.Collections.checkedList;
62 import static java.util.Collections.checkedMap;
63 import static java.util.Collections.checkedSet;
64 import static java.util.Collections.newSetFromMap;
65 import static java.util.Collections.synchronizedList;
66 import static java.util.Collections.synchronizedMap;
67 import static java.util.Collections.synchronizedSet;
68 import static java.util.Collections.unmodifiableList;
69 import static java.util.Collections.unmodifiableMap;
70 import static java.util.Collections.unmodifiableSet;
71
72 public class RacingCollections {
73 /**
74 * How long to run each "race" (in milliseconds).
75 * Turn this up to some higher value like 1000 for stress testing:
76 * java -Dmillis=1000 RacingCollections
77 */
78 static final long defaultWorkTimeMillis = Long.getLong("millis", 10L);
79
80 /**
81 * Whether to print debug information.
82 */
83 static final boolean debug = Boolean.getBoolean("debug");
84
85 static final Integer one = 1;
86 static final Integer two = 2;
87
88 /**
89 * A thread that mutates an object forever, alternating between
90 * being empty and containing singleton "two"
91 */
92 static class Frobber extends CheckedThread {
93 volatile boolean done = false;
94 boolean keepGoing(int i) { return (i % 128 != 0) || ! done; }
95
96 final Object elLoco;
97 Frobber(Object elLoco) {
98 this.elLoco = elLoco;
99 this.start();
100 }
101
102 @SuppressWarnings("unchecked")
103 void clear(Object o) {
104 if (o instanceof Collection)
105 ((Collection<?>)o).clear();
106 else
107 ((Map<?,?>)o).clear();
108 }
109
110 @SuppressWarnings("unchecked")
111 void realRun() {
112 // Mutate elLoco wildly forever, checking occasionally for "done"
113 clear(elLoco);
114 if (elLoco instanceof List) {
115 List<Integer> l = (List<Integer>) elLoco;
116 for (int i = 0; keepGoing(i); i++) {
117 switch (i%2) {
118 case 0: l.add(two); break;
119 case 1: l.add(0, two); break;
120 }
121 switch (i%2) {
122 case 0: l.remove(two); break;
123 case 1: l.remove(0); break;
124 }}}
125 else if (elLoco instanceof Deque) {
126 Deque<Integer> q = (Deque<Integer>) elLoco;
127 for (int i = 0; keepGoing(i); i++) {
128 switch (i%6) {
129 case 0: q.add(two); break;
130 case 1: q.addFirst(two); break;
131 case 2: q.addLast(two); break;
132 case 3: q.offer(two); break;
133 case 4: q.offerFirst(two); break;
134 case 5: q.offerLast(two); break;
135 }
136 switch (i%6) {
137 case 0: q.remove(two); break;
138 case 1: q.removeFirst(); break;
139 case 2: q.removeLast(); break;
140 case 3: q.poll(); break;
141 case 4: q.pollFirst(); break;
142 case 5: q.pollLast(); break;
143 }}}
144 else if (elLoco instanceof Queue) {
145 Queue<Integer> q = (Queue<Integer>) elLoco;
146 for (int i = 0; keepGoing(i); i++) {
147 switch (i%2) {
148 case 0: q.add(two); break;
149 case 1: q.offer(two); break;
150 }
151 switch (i%2) {
152 case 0: q.remove(two); break;
153 case 1: q.poll(); break;
154 }}}
155 else if (elLoco instanceof Map) {
156 Map<Integer, Boolean> m = (Map<Integer, Boolean>) elLoco;
157 for (int i = 0; keepGoing(i); i++) {
158 m.put(two, true);
159 m.remove(two);
160 }}
161 else if (elLoco instanceof Collection) {
162 Collection<Integer> c = (Collection<Integer>) elLoco;
163 for (int i = 0; keepGoing(i); i++) {
164 c.add(two);
165 c.remove(two);
166 }}
167 else { throw new Error("Huh? " + elLoco); }
168 }
169
170 void enoughAlready() {
171 done = true;
172 try { join(); } catch (Throwable t) { unexpected(t); }
173 }
174 }
175
176 private static void checkEqualSanity(Object theRock, Object elLoco) {
177 //equal(theRock, theRock);
178 equal(elLoco, elLoco);
179
180 // It would be nice someday to have theRock and elLoco never "equal",
181 // although the meaning of "equal" for mutating collections
182 // is a bit fuzzy. Uncomment when/if we fix:
183 // 6374942: Improve thread safety of collection .equals() methods
184 //notEqual(theRock, elLoco);
185 //notEqual(elLoco, theRock);
186
187 notEqual(theRock.toString(), elLoco.toString());
188 }
189
190 static class Looper {
191 final long quittingTime;
192 int i = 0;
193 Looper() { this(defaultWorkTimeMillis); }
194 Looper(long workTimeMillis) {
195 quittingTime = System.nanoTime() + workTimeMillis * 1024 * 1024;
196 }
197 boolean keepGoing() {
198 return (i++ % 128 != 0) || (System.nanoTime() - quittingTime < 0);
199 }
200 }
201
202 private static void frob(Object theRock, Object elLoco) {
203 Frobber frobber = new Frobber(elLoco);
204 try {
205 if (theRock instanceof Collection) {
206 @SuppressWarnings("unchecked")
207 Collection<Integer> c = (Collection<Integer>) theRock;
208 if (! c.contains(one))
209 c.add(one);
210 } else {
211 @SuppressWarnings("unchecked")
212 Map<Integer, Boolean> m = (Map<Integer, Boolean>) theRock;
213 if (! m.containsKey(one))
214 m.put(one, true);
215 }
216 for (Looper looper = new Looper(); looper.keepGoing(); )
217 checkEqualSanity(theRock, elLoco);
218 }
219 catch (Throwable t) { unexpected(t); }
220 finally { frobber.enoughAlready(); }
221 }
222
223 private static List<Map<Integer, Boolean>> newConcurrentMaps() {
224 List<Map<Integer, Boolean>> list = new ArrayList<>();
225 list.add(new ConcurrentHashMap<Integer, Boolean>());
226 list.add(new ConcurrentSkipListMap<Integer, Boolean>());
227 return list;
228 }
229
230 private static List<Map<Integer, Boolean>> maps() {
231 List<Map<Integer, Boolean>> list = newConcurrentMaps();
232 list.add(new Hashtable<Integer, Boolean>());
233 list.add(new HashMap<Integer, Boolean>());
234 list.add(new TreeMap<Integer, Boolean>());
235 Comparator<Integer> cmp = new Comparator<>() {
236 public int compare(Integer x, Integer y) {
237 return x - y;
238 }};
239 list.add(new TreeMap<Integer, Boolean>(Collections.reverseOrder(cmp)));
240 return list;
241 }
242
243 private static List<Set<Integer>> newConcurrentSets() {
244 List<Set<Integer>> list = new ArrayList<>();
245 list.add(new ConcurrentSkipListSet<Integer>());
246 list.add(new CopyOnWriteArraySet<Integer>());
247 return list;
248 }
249
250 private static List<Set<Integer>> newSets() {
251 List<Set<Integer>> list = newConcurrentSets();
252 list.add(new HashSet<Integer>());
253 list.add(new TreeSet<Integer>());
254 list.add(new TreeSet<Integer>(Collections.reverseOrder()));
255 return list;
256 }
257
258 private static List<List<Integer>> newConcurrentLists() {
259 List<List<Integer>> list = new ArrayList<>();
260 list.add(new CopyOnWriteArrayList<Integer>());
261 return list;
262 }
263
264 private static List<List<Integer>> newLists() {
265 List<List<Integer>> list = newConcurrentLists();
266 list.add(new Vector<Integer>());
267 list.add(new ArrayList<Integer>());
268 return list;
269 }
270
271 private static List<Queue<Integer>> newConcurrentQueues() {
272 List<Queue<Integer>> list = new ArrayList<>(newConcurrentDeques());
273 list.add(new ArrayBlockingQueue<Integer>(10));
274 list.add(new LinkedBlockingQueue<Integer>(10));
275 list.add(new LinkedTransferQueue<Integer>());
276 list.add(new ConcurrentLinkedQueue<Integer>());
277 return list;
278 }
279
280 private static List<Queue<Integer>> newQueues() {
281 List<Queue<Integer>> list = new ArrayList<>(newDeques());
282 list.add(new LinkedBlockingQueue<Integer>(10));
283 return list;
284 }
285
286 private static List<Deque<Integer>> newConcurrentDeques() {
287 List<Deque<Integer>> list = new ArrayList<>();
288 list.add(new LinkedBlockingDeque<Integer>(10));
289 list.add(new ConcurrentLinkedDeque<Integer>());
290 return list;
291 }
292
293 private static List<Deque<Integer>> newDeques() {
294 List<Deque<Integer>> list = newConcurrentDeques();
295 list.add(new ArrayDeque<Integer>());
296 list.add(new LinkedList<Integer>());
297 return list;
298 }
299
300 private static void describe(Class<?> k, Object x, Object y) {
301 if (debug)
302 System.out.printf("%s: %s, %s%n", k.getSimpleName(),
303 x.getClass().getSimpleName(),
304 y.getClass().getSimpleName());
305 }
306
307 private static void realMain(String[] args) {
308 for (Map<Integer, Boolean> x : maps())
309 for (Map<Integer, Boolean> y : newConcurrentMaps()) {
310 describe(Map.class, x, y);
311 x.put(one, true);
312 frob(x, y);
313 frob(unmodifiableMap(x), y);
314 frob(synchronizedMap(x), y);
315 frob(x, synchronizedMap(y));
316 frob(checkedMap(x, Integer.class, Boolean.class), y);
317 frob(x, checkedMap(y, Integer.class, Boolean.class));
318 x.clear();
319 frob(newSetFromMap(x), newSetFromMap(y));
320 frob(x.keySet(), newSetFromMap(y));
321 }
322
323 for (Set<Integer> x : newSets())
324 for (Set<Integer> y : newConcurrentSets()) {
325 describe(Set.class, x, y);
326 frob(x, y);
327 frob(unmodifiableSet(x), y);
328 frob(synchronizedSet(x), y);
329 frob(x, synchronizedSet(y));
330 frob(checkedSet(x, Integer.class), y);
331 frob(x, checkedSet(y, Integer.class));
332 }
333
334 for (List<Integer> x : newLists())
335 for (List<Integer> y : newConcurrentLists()) {
336 describe(List.class, x, y);
337 frob(x, y);
338 frob(unmodifiableList(x), y);
339 frob(synchronizedList(x), y);
340 frob(x, synchronizedList(y));
341 frob(checkedList(x, Integer.class), y);
342 frob(x, checkedList(y, Integer.class));
343 }
344
345 for (Queue<Integer> x : newQueues())
346 for (Queue<Integer> y : newConcurrentQueues()) {
347 describe(Queue.class, x, y);
348 frob(x, y);
349 }
350
351 for (Deque<Integer> x : newDeques())
352 for (Deque<Integer> y : newConcurrentDeques()) {
353 describe(Deque.class, x, y);
354 frob(asLifoQueue(x), y);
355 frob(x, asLifoQueue(y));
356 }
357 }
358
359 //--------------------- Infrastructure ---------------------------
360 static volatile int passed = 0, failed = 0;
361 static void pass() {passed++;}
362 static void fail() {failed++; Thread.dumpStack();}
363 static void fail(String msg) {System.out.println(msg); fail();}
364 static void unexpected(Throwable t) {failed++; t.printStackTrace();}
365 static void check(boolean cond) {if (cond) pass(); else fail();}
366 static String toString(Object x) {
367 return ((x instanceof Collection) || (x instanceof Map)) ?
368 x.getClass().getName() : x.toString();}
369 static void equal(Object x, Object y) {
370 if (x == null ? y == null : x.equals(y)) pass();
371 else fail(toString(x) + " not equal to " + toString(y));}
372 static void notEqual(Object x, Object y) {
373 if (x == null ? y == null : x.equals(y))
374 fail(toString(x) + " equal to " + toString(y));
375 else pass();}
376 public static void main(String[] args) throws Throwable {
377 try {realMain(args);} catch (Throwable t) {unexpected(t);}
378 System.out.printf("%nPassed = %d, failed = %d%n%n", passed, failed);
379 if (failed > 0) throw new AssertionError("Some tests failed");}
380 private abstract static class CheckedThread extends Thread {
381 abstract void realRun() throws Throwable;
382 public void run() {
383 try { realRun(); } catch (Throwable t) { unexpected(t); }}}
384 }