29852ecd80
Reviewed-by: alanb, mcimadamore, kvn
363 lines
12 KiB
Java
363 lines
12 KiB
Java
/*
|
|
* Copyright (c) 2010, 2011, Oracle and/or its affiliates. All rights reserved.
|
|
* 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
|
|
@bug 5089429 6982632 8145808
|
|
@summary Checks that we don't crash if rendering operations and state
|
|
changes are performed on a graphics context from different threads.
|
|
|
|
@author Dmitri.Trembovetski@sun.com area=Graphics
|
|
@run main MTGraphicsAccessTest
|
|
*/
|
|
|
|
import java.awt.*;
|
|
import java.awt.image.*;
|
|
import java.awt.geom.*;
|
|
import java.util.concurrent.atomic.AtomicInteger;
|
|
|
|
public class MTGraphicsAccessTest {
|
|
|
|
// in seconds
|
|
static final long STANDALONE_RUN_TIME = 20;
|
|
static final long JTREG_RUN_TIME = 7;
|
|
|
|
static boolean standaloneMode;
|
|
static boolean allowExceptions = true;
|
|
static long testRunTime;
|
|
|
|
volatile boolean done;
|
|
AtomicInteger stillRunning = new AtomicInteger(0);
|
|
volatile int numexceptions;
|
|
|
|
Graphics2D sharedGraphics;
|
|
BufferedImage sharedBI =
|
|
new BufferedImage(50, 50, BufferedImage.TYPE_INT_RGB);
|
|
|
|
static final Paint colors[] = {
|
|
Color.red,
|
|
new Color(0x7f, 0xff, 0x00, 0x7f),
|
|
new GradientPaint(0, 0, Color.red,
|
|
50, 50, new Color(0x7f, 0xff, 0x00, 0x7f)),
|
|
};
|
|
static final Font fonts[] = {
|
|
new Font("Dialog", Font.PLAIN, 12),
|
|
new Font("Dialog", Font.BOLD, 16),
|
|
new Font("Dialog", Font.ITALIC, 18),
|
|
};
|
|
static final AlphaComposite comps[] = {
|
|
AlphaComposite.getInstance(AlphaComposite.SRC_OVER, 1.0f),
|
|
AlphaComposite.Src,
|
|
AlphaComposite.Xor,
|
|
AlphaComposite.getInstance(AlphaComposite.SRC_OVER, 0.5f),
|
|
null,
|
|
};
|
|
static final Stroke strokes[] = {
|
|
new BasicStroke(),
|
|
new BasicStroke(0.0f),
|
|
new BasicStroke(2.0f),
|
|
new BasicStroke(2.0f, BasicStroke.CAP_ROUND,
|
|
BasicStroke.JOIN_BEVEL),
|
|
new BasicStroke(5.0f, BasicStroke.CAP_SQUARE,
|
|
BasicStroke.JOIN_ROUND),
|
|
new BasicStroke(0.0f, BasicStroke.CAP_ROUND,
|
|
BasicStroke.JOIN_ROUND, 0,
|
|
new float[]{0,6,0,6}, 0),
|
|
};
|
|
static final AffineTransform transforms[] = {
|
|
new AffineTransform(),
|
|
AffineTransform.getRotateInstance(10.0),
|
|
AffineTransform.getShearInstance(10.0, 4.0),
|
|
AffineTransform.getScaleInstance(1.1, 1.2),
|
|
AffineTransform.getScaleInstance(3.0, 2.0),
|
|
};
|
|
|
|
public MTGraphicsAccessTest() {
|
|
BufferedImage bi =
|
|
new BufferedImage(50, 50, BufferedImage.TYPE_INT_RGB);
|
|
sharedGraphics = (Graphics2D)bi.getGraphics();
|
|
|
|
done = false;
|
|
numexceptions = 0;
|
|
|
|
for (int i = 0; i < (standaloneMode ? stateChangers.length : 3); i++) {
|
|
(new TesterThread(stateChangers[i])).start();
|
|
}
|
|
for (int i = 0; i < (standaloneMode ? renderTests.length : 5); i++) {
|
|
(new TesterThread(renderTests[i])).start();
|
|
}
|
|
|
|
mysleep(testRunTime);
|
|
done = true;
|
|
while (stillRunning.get() > 0) { mysleep(500); }
|
|
|
|
if (numexceptions == 0) {
|
|
System.err.println("Test passed");
|
|
} else if (!allowExceptions) {
|
|
throw new RuntimeException("Test failed with "+
|
|
numexceptions+" exceptions");
|
|
} else {
|
|
System.err.println("Test finished with "+
|
|
numexceptions+" exceptions");
|
|
}
|
|
}
|
|
|
|
private void mysleep(long time) {
|
|
try {
|
|
// add +/-5ms variance to increase randomness
|
|
Thread.sleep(time + (long)(5 - Math.random()*10));
|
|
} catch (InterruptedException e) {};
|
|
}
|
|
|
|
public static void usage(String message) {
|
|
if (message != null) {
|
|
System.err.println(message);
|
|
}
|
|
System.err.println("Usage: MTGraphicsAccessTest [-full] "+
|
|
"[-time N/forever] [-help]");
|
|
System.err.println(" -full: run full suite of tests "+
|
|
"(default: limited number of tests is run)");
|
|
System.err.println(" -time N: test duration in seconds/forever"+
|
|
" (default: "+JTREG_RUN_TIME+"s for the short suite, "+
|
|
STANDALONE_RUN_TIME+"s for the full suite)");
|
|
System.err.println(" -help: print this help page");
|
|
System.exit(1);
|
|
}
|
|
|
|
public static void main(String[] args) {
|
|
boolean testRunSet = false;
|
|
for (int i = 0; i < args.length; i++) {
|
|
if ("-full".equals(args[i])) {
|
|
standaloneMode = true;
|
|
System.err.println("Running complete list of tests");
|
|
} else if ("-noexc".equals(args[i])) {
|
|
allowExceptions = false;
|
|
} else if ("-time".equals(args[i])) {
|
|
try {
|
|
String time = args[++i];
|
|
if ("forever".equals(time)) {
|
|
testRunTime = (Long.MAX_VALUE - 20)/1000;
|
|
} else {
|
|
testRunTime = 1000*Integer.parseInt(time);
|
|
}
|
|
testRunSet = true;
|
|
} catch (NumberFormatException e) {
|
|
usage("Can't parse number of seconds: " + args[i]);
|
|
} catch (ArrayIndexOutOfBoundsException e1) {
|
|
usage("Missing the 'seconds' argument for -time parameter");
|
|
}
|
|
} else if ("-help".equals(args[i])) {
|
|
usage(null);
|
|
} else {
|
|
usage("Unknown argument:" + args[i]);
|
|
}
|
|
}
|
|
|
|
if (!testRunSet) {
|
|
testRunTime = 1000 *
|
|
(standaloneMode ? STANDALONE_RUN_TIME : JTREG_RUN_TIME);
|
|
}
|
|
|
|
System.err.println("Approximate test run time: "+
|
|
testRunTime/1000+" seconds");
|
|
|
|
new MTGraphicsAccessTest();
|
|
}
|
|
|
|
class TesterThread extends Thread {
|
|
Runnable testRunnable;
|
|
|
|
public TesterThread(Runnable testRunnable) {
|
|
stillRunning.incrementAndGet();
|
|
this.testRunnable = testRunnable;
|
|
}
|
|
|
|
public void run() {
|
|
try {
|
|
while (!done) {
|
|
try {
|
|
testRunnable.run();
|
|
Thread.yield();
|
|
} catch (Throwable t) {
|
|
numexceptions++;
|
|
t.printStackTrace();
|
|
}
|
|
}
|
|
} finally {
|
|
stillRunning.decrementAndGet();
|
|
}
|
|
}
|
|
}
|
|
|
|
final Runnable stateChangers[] = {
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.setClip(10, 10, 30, 30);
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.setClip(10, 10, 30, 30);
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
int c = 0;
|
|
public void run() {
|
|
sharedGraphics.setPaint(colors[c++ % colors.length]);
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
boolean AA = false;
|
|
public void run() {
|
|
if (AA) {
|
|
sharedGraphics.setRenderingHint(
|
|
RenderingHints.KEY_ANTIALIASING,
|
|
RenderingHints.VALUE_ANTIALIAS_ON);
|
|
} else {
|
|
sharedGraphics.setRenderingHint(
|
|
RenderingHints.KEY_ANTIALIASING,
|
|
RenderingHints.VALUE_ANTIALIAS_OFF);
|
|
}
|
|
AA = !AA;
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
int t = 0;
|
|
public void run() {
|
|
sharedGraphics.setTransform(
|
|
transforms[t++ % transforms.length]);
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
int c = 0;
|
|
public void run() {
|
|
AlphaComposite comp = comps[c++ % comps.length];
|
|
if (comp == null) {
|
|
sharedGraphics.setXORMode(Color.green);
|
|
} else {
|
|
sharedGraphics.setComposite(comp);
|
|
}
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
int s = 0;
|
|
public void run() {
|
|
sharedGraphics.setStroke(strokes[s++ % strokes.length]);
|
|
mysleep(10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
int f = 0;
|
|
public void run() {
|
|
sharedGraphics.setFont(fonts[f++ % fonts.length]);
|
|
mysleep(10);
|
|
}
|
|
},
|
|
};
|
|
|
|
final Runnable renderTests[] = {
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawLine(10, 10, 30, 30);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawLine(10, 10, 30, 30);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawRect(10, 10, 30, 30);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.fillRect(10, 10, 30, 30);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawString("Stuff", 10, 10);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.draw3DRect(10, 10, 30, 30, true);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawImage(sharedBI, 10, 10, null);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.fill3DRect(10, 10, 30, 30, false);
|
|
}
|
|
},
|
|
// REMIND: copyArea doesn't work when transform is set..
|
|
// new Runnable() {
|
|
// public void run() {
|
|
// sharedGraphics.copyArea(10, 10, 30, 30, 20, 20);
|
|
// }
|
|
// },
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawRoundRect(10, 10, 30, 30, 20, 20);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.fillRoundRect(10, 10, 30, 30, 20, 20);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawArc(10, 10, 30, 30, 0, 90);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.fillArc(10, 10, 30, 30, 0, 90);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.drawOval(10, 10, 30, 30);
|
|
}
|
|
},
|
|
new Runnable() {
|
|
public void run() {
|
|
sharedGraphics.fillOval(10, 10, 30, 30);
|
|
}
|
|
}
|
|
};
|
|
}
|