2012-12-19 19:21:15 -08:00
|
|
|
/*
|
2013-01-25 16:09:14 -08:00
|
|
|
* Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved.
|
2012-12-19 19:21:15 -08:00
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
|
|
* accompanied this code).
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License version
|
|
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
|
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
|
|
* questions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test Test8004741.java
|
|
|
|
* @bug 8004741
|
|
|
|
* @summary Missing compiled exception handle table entry for multidimensional array allocation
|
2013-01-25 16:09:14 -08:00
|
|
|
* @run main/othervm -Xmx64m -Xbatch -XX:+IgnoreUnrecognizedVMOptions -XX:-TieredCompilation -XX:+StressCompiledExceptionHandlers -XX:+SafepointALot -XX:GuaranteedSafepointInterval=100 Test8004741
|
2012-12-19 19:21:15 -08:00
|
|
|
* @run main/othervm -Xmx64m -Xbatch -XX:+IgnoreUnrecognizedVMOptions -XX:-TieredCompilation -XX:+StressCompiledExceptionHandlers Test8004741
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.util.*;
|
|
|
|
|
|
|
|
public class Test8004741 extends Thread {
|
|
|
|
|
2013-01-25 16:09:14 -08:00
|
|
|
static int passed = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loop forever allocating 2-d arrays.
|
|
|
|
* Catches and rethrows all exceptions; in the case of ThreadDeath, increments passed.
|
|
|
|
* Note that passed is incremented here because this is the exception handler with
|
|
|
|
* the smallest scope; we only want to declare success in the case where it is highly
|
|
|
|
* likely that the test condition
|
|
|
|
* (exception in 2-d array alloc interrupted by ThreadDeath)
|
|
|
|
* actually occurs.
|
|
|
|
*/
|
2012-12-19 19:21:15 -08:00
|
|
|
static int[][] test(int a, int b) throws Exception {
|
2013-01-25 16:09:14 -08:00
|
|
|
int[][] ar;
|
2012-12-19 19:21:15 -08:00
|
|
|
try {
|
|
|
|
ar = new int[a][b];
|
2013-01-25 16:09:14 -08:00
|
|
|
} catch (ThreadDeath e) {
|
|
|
|
System.out.println("test got ThreadDeath");
|
|
|
|
passed++;
|
2012-12-19 19:21:15 -08:00
|
|
|
throw(e);
|
|
|
|
}
|
|
|
|
return ar;
|
|
|
|
}
|
|
|
|
|
2013-01-25 16:09:14 -08:00
|
|
|
/* Cookbook wait-notify to track progress of test thread. */
|
|
|
|
Object progressLock = new Object();
|
|
|
|
private static final int NOT_STARTED = 0;
|
|
|
|
private static final int RUNNING = 1;
|
|
|
|
private static final int STOPPING = 2;
|
2012-12-19 19:21:15 -08:00
|
|
|
|
2013-01-25 16:09:14 -08:00
|
|
|
int progressState = NOT_STARTED;
|
|
|
|
|
|
|
|
void toState(int state) {
|
|
|
|
synchronized (progressLock) {
|
|
|
|
progressState = state;
|
|
|
|
progressLock.notify();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void waitFor(int state) {
|
|
|
|
synchronized (progressLock) {
|
|
|
|
while (progressState < state) {
|
|
|
|
try {
|
|
|
|
progressLock.wait();
|
|
|
|
} catch (InterruptedException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
System.out.println("unexpected InterruptedException");
|
|
|
|
fail();
|
2012-12-19 19:21:15 -08:00
|
|
|
}
|
|
|
|
}
|
2013-01-25 16:09:14 -08:00
|
|
|
if (progressState > state) {
|
|
|
|
System.out.println("unexpected test state change, expected " +
|
|
|
|
state + " but saw " + progressState);
|
|
|
|
fail();
|
|
|
|
}
|
|
|
|
}
|
2012-12-19 19:21:15 -08:00
|
|
|
}
|
|
|
|
|
2013-01-25 16:09:14 -08:00
|
|
|
/**
|
|
|
|
* Loops running test until some sort of an exception or error,
|
|
|
|
* expects to see ThreadDeath.
|
|
|
|
*/
|
|
|
|
public void run() {
|
|
|
|
try {
|
|
|
|
// Print before state change, so that other thread is most likely
|
|
|
|
// to see this thread executing calls to test() in a loop.
|
|
|
|
System.out.println("thread running");
|
|
|
|
toState(RUNNING);
|
|
|
|
while (true) {
|
|
|
|
// (2,2) (2,10) (2,100) were observed to tickle the bug;
|
|
|
|
test(2, 100);
|
|
|
|
}
|
|
|
|
} catch (ThreadDeath e) {
|
|
|
|
// nothing to say, passing was incremented by the test.
|
|
|
|
} catch (Throwable e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
System.out.println("unexpected Throwable " + e);
|
|
|
|
fail();
|
2012-12-19 19:21:15 -08:00
|
|
|
}
|
2013-01-25 16:09:14 -08:00
|
|
|
toState(STOPPING);
|
|
|
|
}
|
2012-12-19 19:21:15 -08:00
|
|
|
|
2013-01-25 16:09:14 -08:00
|
|
|
/**
|
|
|
|
* Runs a single trial of the test in a thread.
|
|
|
|
* No single trial is definitive, since the ThreadDeath
|
|
|
|
* exception might not land in the tested region of code.
|
|
|
|
*/
|
|
|
|
public static void threadTest() throws InterruptedException {
|
2012-12-19 19:21:15 -08:00
|
|
|
Test8004741 t = new Test8004741();
|
|
|
|
t.start();
|
2013-01-25 16:09:14 -08:00
|
|
|
t.waitFor(RUNNING);
|
|
|
|
Thread.sleep(100);
|
|
|
|
System.out.println("stopping thread");
|
2012-12-19 19:21:15 -08:00
|
|
|
t.stop();
|
2013-01-25 16:09:14 -08:00
|
|
|
t.waitFor(STOPPING);
|
2012-12-19 19:21:15 -08:00
|
|
|
t.join();
|
2013-01-25 16:09:14 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
|
|
// Warm up "test"
|
|
|
|
// t will never be started.
|
|
|
|
for (int n = 0; n < 11000; n++) {
|
|
|
|
test(2, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Will this sleep help ensure that the compiler is run?
|
|
|
|
Thread.sleep(500);
|
|
|
|
passed = 0;
|
|
|
|
|
|
|
|
try {
|
|
|
|
test(-1, 100);
|
|
|
|
System.out.println("Missing NegativeArraySizeException #1");
|
|
|
|
fail();
|
|
|
|
} catch ( java.lang.NegativeArraySizeException e ) {
|
|
|
|
System.out.println("Saw expected NegativeArraySizeException #1");
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
test(100, -1);
|
|
|
|
fail();
|
|
|
|
System.out.println("Missing NegativeArraySizeException #2");
|
|
|
|
fail();
|
|
|
|
} catch ( java.lang.NegativeArraySizeException e ) {
|
|
|
|
System.out.println("Saw expected NegativeArraySizeException #2");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test repetitions. If the test succeeds-mostly, it succeeds,
|
|
|
|
* as long as it does not crash (the outcome if the exception range
|
|
|
|
* table entry for the array allocation is missing).
|
|
|
|
*/
|
|
|
|
int N = 12;
|
|
|
|
for (int n = 0; n < N; n++) {
|
|
|
|
threadTest();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (passed > N/2) {
|
|
|
|
System.out.println("Saw " + passed + " out of " + N + " possible ThreadDeath hits");
|
2012-12-19 19:21:15 -08:00
|
|
|
System.out.println("PASSED");
|
|
|
|
} else {
|
2013-01-25 16:09:14 -08:00
|
|
|
System.out.println("Too few ThreadDeath hits; expected at least " + N/2 +
|
|
|
|
" but saw only " + passed);
|
|
|
|
fail();
|
2012-12-19 19:21:15 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-25 16:09:14 -08:00
|
|
|
static void fail() {
|
|
|
|
System.out.println("FAILED");
|
|
|
|
System.exit(97);
|
|
|
|
}
|
2012-12-19 19:21:15 -08:00
|
|
|
};
|