@@ -49,18 +49,24 @@ public class OOMEInAQS extends Thread {
49
49
static final Condition condition = mainLock .newCondition ();
50
50
static final CountDownLatch started = new CountDownLatch (1 );
51
51
static final CountDownLatch filled = new CountDownLatch (1 );
52
+ static final CountDownLatch canFill = new CountDownLatch (NTHREADS );
52
53
static volatile Object data ;
54
+ static volatile Throwable exception ;
53
55
static int turn ;
54
56
55
57
/**
56
58
* For each of NTHREADS threads, REPS times: Take turns
57
- * executing. Introduce OOM using fillHeap during runs.
59
+ * executing. Introduce OOM using fillHeap during runs. In
60
+ * addition to testing AQS, the CountDownLatches ensure that
61
+ * methods execute at least once before OutOfMemory occurs, to
62
+ * avoid uncontrollable impact of OOME during class-loading.
58
63
*/
59
64
public static void main (String [] args ) throws Throwable {
60
65
OOMEInAQS [] threads = new OOMEInAQS [NTHREADS ];
61
66
for (int i = 0 ; i < NTHREADS ; ++i )
62
67
(threads [i ] = new OOMEInAQS (i )).start ();
63
68
started .countDown ();
69
+ canFill .await ();
64
70
long t0 = System .nanoTime ();
65
71
data = fillHeap ();
66
72
filled .countDown ();
@@ -69,6 +75,9 @@ public static void main(String[] args) throws Throwable {
69
75
threads [i ].join ();
70
76
data = null ; // free heap before reporting and terminating
71
77
System .gc ();
78
+ Throwable ex = exception ;
79
+ if (ex != null )
80
+ throw ex ;
72
81
System .out .println (
73
82
"fillHeap time: " + (t1 - t0 ) / 1000_000 +
74
83
" millis, whole test time: " + (System .nanoTime () - t0 ) / 1000_000 +
@@ -89,21 +98,26 @@ public void run() {
89
98
try {
90
99
started .await ();
91
100
for (int i = 0 ; i < NREPS ; i ++) {
101
+ lock .lock ();
92
102
try {
93
- lock .lock ();
94
103
while (turn != id )
95
104
cond .await ();
96
105
turn = nextId ;
97
106
cond .signalAll ();
98
107
} finally {
99
108
lock .unlock ();
100
109
}
101
- if (i == 2 ) // Subsequent AQS methods encounter OOME
110
+ if (i == 2 ) { // Subsequent AQS methods encounter OOME
111
+ canFill .countDown ();
102
112
filled .await ();
113
+ }
103
114
}
104
- } catch (Throwable ex ) { // Could be InterruptedExeption or OOME
105
115
data = null ;
106
- System .exit (0 ); // avoid getting stuck trying to recover
116
+ System .gc (); // avoid getting stuck while exiting
117
+ } catch (Throwable ex ) {
118
+ data = null ;
119
+ System .gc (); // avoid nested OOME
120
+ exception = ex ;
107
121
}
108
122
}
109
123
0 commit comments