6d4bd6c6b6
Reviewed-by: azvegint, prr
293 lines
10 KiB
Java
293 lines
10 KiB
Java
/*
|
|
* Copyright (c) 2015, 2024, 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.
|
|
*/
|
|
|
|
import java.awt.BasicStroke;
|
|
import java.awt.Color;
|
|
import java.awt.Graphics2D;
|
|
import java.awt.RenderingHints;
|
|
import java.awt.geom.Path2D;
|
|
import static java.awt.geom.Path2D.WIND_NON_ZERO;
|
|
import java.awt.image.BufferedImage;
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import javax.imageio.ImageIO;
|
|
|
|
/**
|
|
* @test
|
|
* @summary Simple crash rendering test using huge GeneralPaths with the Marlin renderer
|
|
* @run main/othervm -Xmx512m CrashTest
|
|
* @ignore tests that take a long time and consumes 5Gb memory
|
|
* @run main/othervm -Xms4g -Xmx4g CrashTest -slow
|
|
*/
|
|
public class CrashTest {
|
|
|
|
static final boolean SAVE_IMAGE = false;
|
|
static boolean USE_ROUND_CAPS_AND_JOINS = true;
|
|
|
|
public static void main(String[] args) {
|
|
boolean runSlowTests = (args.length != 0 && "-slow".equals(args[0]));
|
|
|
|
// First display which renderer is tested:
|
|
System.setProperty("sun.java2d.renderer.verbose", "true");
|
|
|
|
// try insane image sizes:
|
|
|
|
// subpixel coords may overflow:
|
|
// check MAX_VALUE / (8 * 2); overflow may happen due to orientation flag
|
|
// But as it is impossible to allocate an image larger than 2Gb (byte) then
|
|
// it is also impossible to have rowAAChunk larger than 2Gb !
|
|
|
|
// Disabled test as it consumes 4GB heap + offheap (2Gb) ie > 6Gb !
|
|
if (runSlowTests) {
|
|
testHugeImage((Integer.MAX_VALUE >> 4) - 100, 16);
|
|
}
|
|
|
|
// larger than 23 bits: (RLE)
|
|
testHugeImage(8388608 + 1, 10);
|
|
|
|
if (runSlowTests) {
|
|
test(0.1f, false, 0);
|
|
test(0.1f, true, 7f);
|
|
}
|
|
|
|
// Exceed 2Gb OffHeap buffer for edges:
|
|
try {
|
|
USE_ROUND_CAPS_AND_JOINS = true;
|
|
test(0.1f, true, 0.1f);
|
|
System.out.println("Exception MISSING.");
|
|
}
|
|
catch (Throwable th) {
|
|
if (th instanceof ArrayIndexOutOfBoundsException) {
|
|
System.out.println("ArrayIndexOutOfBoundsException expected.");
|
|
} else {
|
|
throw new RuntimeException("Unexpected exception", th);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void test(final float lineStroke,
|
|
final boolean useDashes,
|
|
final float dashMinLen)
|
|
throws ArrayIndexOutOfBoundsException
|
|
{
|
|
System.out.println("---\n" + "test: "
|
|
+ "lineStroke=" + lineStroke
|
|
+ ", useDashes=" + useDashes
|
|
+", dashMinLen=" + dashMinLen
|
|
);
|
|
|
|
final BasicStroke stroke = createStroke(lineStroke, useDashes, dashMinLen);
|
|
|
|
// TODO: test Dasher.firstSegmentsBuffer resizing ?
|
|
// array.dasher.firstSegmentsBuffer.d_float[2] sum: 6 avg: 3.0 [3 | 3]
|
|
/*
|
|
// Marlin growable arrays:
|
|
= new StatLong("array.dasher.firstSegmentsBuffer.d_float");
|
|
= new StatLong("array.stroker.polystack.curves.d_float");
|
|
= new StatLong("array.stroker.polystack.curveTypes.d_byte");
|
|
= new StatLong("array.marlincache.rowAAChunk.d_byte");
|
|
= new StatLong("array.marlincache.touchedTile.int");
|
|
= new StatLong("array.renderer.alphaline.int");
|
|
= new StatLong("array.renderer.crossings.int");
|
|
= new StatLong("array.renderer.aux_crossings.int");
|
|
= new StatLong("array.renderer.edgeBuckets.int");
|
|
= new StatLong("array.renderer.edgeBucketCounts.int");
|
|
= new StatLong("array.renderer.edgePtrs.int");
|
|
= new StatLong("array.renderer.aux_edgePtrs.int");
|
|
*/
|
|
// size > 8192 (exceed both tile and buckets arrays)
|
|
final int size = 9000;
|
|
System.out.println("image size = " + size);
|
|
|
|
final BufferedImage image = new BufferedImage(size, size, BufferedImage.TYPE_INT_ARGB);
|
|
|
|
final Graphics2D g2d = (Graphics2D) image.getGraphics();
|
|
try {
|
|
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
|
|
g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
|
|
|
|
g2d.setClip(0, 0, size, size);
|
|
g2d.setBackground(Color.WHITE);
|
|
g2d.clearRect(0, 0, size, size);
|
|
|
|
g2d.setStroke(stroke);
|
|
g2d.setColor(Color.BLACK);
|
|
|
|
final long start = System.nanoTime();
|
|
|
|
paint(g2d, size - 10f);
|
|
|
|
final long time = System.nanoTime() - start;
|
|
|
|
System.out.println("paint: duration= " + (1e-6 * time) + " ms.");
|
|
|
|
if (SAVE_IMAGE) {
|
|
try {
|
|
final File file = new File("CrashTest-dash-" + useDashes + ".bmp");
|
|
|
|
System.out.println("Writing file: " + file.getAbsolutePath());
|
|
ImageIO.write(image, "BMP", file);
|
|
} catch (IOException ex) {
|
|
System.out.println("Writing file failure:");
|
|
ex.printStackTrace();
|
|
}
|
|
}
|
|
} finally {
|
|
g2d.dispose();
|
|
}
|
|
}
|
|
|
|
private static void testHugeImage(final int width, final int height)
|
|
throws ArrayIndexOutOfBoundsException
|
|
{
|
|
System.out.println("---\n" + "testHugeImage: "
|
|
+ "width=" + width + ", height=" + height);
|
|
|
|
final BasicStroke stroke = createStroke(2.5f, false, 0);
|
|
|
|
// size > 24bits (exceed both tile and buckets arrays)
|
|
System.out.println("image size = " + width + " x "+height);
|
|
|
|
final BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_BYTE_GRAY);
|
|
|
|
final Graphics2D g2d = (Graphics2D) image.getGraphics();
|
|
try {
|
|
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
|
|
g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
|
|
|
|
g2d.setBackground(Color.WHITE);
|
|
g2d.clearRect(0, 0, width, height);
|
|
|
|
g2d.setStroke(stroke);
|
|
g2d.setColor(Color.BLACK);
|
|
|
|
final Path2D.Float path = new Path2D.Float(WIND_NON_ZERO, 32);
|
|
path.moveTo(0, 0);
|
|
path.lineTo(width, 0);
|
|
path.lineTo(width, height);
|
|
path.lineTo(0, height);
|
|
path.lineTo(0, 0);
|
|
|
|
final long start = System.nanoTime();
|
|
|
|
g2d.draw(path);
|
|
|
|
final long time = System.nanoTime() - start;
|
|
|
|
System.out.println("paint: duration= " + (1e-6 * time) + " ms.");
|
|
|
|
if (SAVE_IMAGE) {
|
|
try {
|
|
final File file = new File("CrashTest-huge-"
|
|
+ width + "x" +height + ".bmp");
|
|
|
|
System.out.println("Writing file: " + file.getAbsolutePath());
|
|
ImageIO.write(image, "BMP", file);
|
|
} catch (IOException ex) {
|
|
System.out.println("Writing file failure:");
|
|
ex.printStackTrace();
|
|
}
|
|
}
|
|
} finally {
|
|
g2d.dispose();
|
|
}
|
|
}
|
|
|
|
private static void paint(final Graphics2D g2d, final float size) {
|
|
final double halfSize = size / 2.0;
|
|
|
|
final Path2D.Float path = new Path2D.Float(WIND_NON_ZERO, 32 * 1024);
|
|
|
|
// show cross:
|
|
path.moveTo(0, 0);
|
|
path.lineTo(size, size);
|
|
|
|
path.moveTo(size, 0);
|
|
path.lineTo(0, size);
|
|
|
|
path.moveTo(0, 0);
|
|
path.lineTo(size, 0);
|
|
|
|
path.moveTo(0, 0);
|
|
path.lineTo(0, size);
|
|
|
|
path.moveTo(0, 0);
|
|
|
|
double r = size;
|
|
|
|
final int ratio = 100;
|
|
int repeats = 1;
|
|
|
|
int n = 0;
|
|
|
|
while (r > 1.0) {
|
|
repeats *= ratio;
|
|
|
|
if (repeats > 10000) {
|
|
repeats = 10000;
|
|
}
|
|
|
|
for (int i = 0; i < repeats; i++) {
|
|
path.lineTo(halfSize - 0.5 * r + i * r / repeats,
|
|
halfSize - 0.5 * r);
|
|
n++;
|
|
path.lineTo(halfSize - 0.5 * r + i * r / repeats + 0.1,
|
|
halfSize + 0.5 * r);
|
|
n++;
|
|
}
|
|
|
|
r -= halfSize;
|
|
}
|
|
System.out.println("draw : " + n + " lines.");
|
|
g2d.draw(path);
|
|
}
|
|
|
|
private static BasicStroke createStroke(final float width,
|
|
final boolean useDashes,
|
|
final float dashMinLen) {
|
|
final float[] dashes;
|
|
|
|
if (useDashes) {
|
|
// huge dash array (exceed Dasher.INITIAL_ARRAY)
|
|
dashes = new float[512];
|
|
|
|
float cur = dashMinLen;
|
|
float step = 0.01f;
|
|
|
|
for (int i = 0; i < dashes.length; i += 2) {
|
|
dashes[i] = cur;
|
|
dashes[i + 1] = cur;
|
|
cur += step;
|
|
}
|
|
} else {
|
|
dashes = null;
|
|
}
|
|
|
|
if (USE_ROUND_CAPS_AND_JOINS) {
|
|
// Use both round Caps & Joins:
|
|
return new BasicStroke(width, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND, 100.0f, dashes, 0.0f);
|
|
}
|
|
return new BasicStroke(width, BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER, 100.0f, dashes, 0.0f);
|
|
}
|
|
}
|