8078464: Path2D storage growth algorithms should be less linear
Reviewed-by: flar
This commit is contained in:
parent
6dd76600b3
commit
e41e267848
@ -101,6 +101,8 @@ public abstract class Path2D implements Shape, Cloneable {
|
||||
|
||||
static final int INIT_SIZE = 20;
|
||||
static final int EXPAND_MAX = 500;
|
||||
static final int EXPAND_MAX_COORDS = EXPAND_MAX * 2;
|
||||
static final int EXPAND_MIN = 10; // ensure > 6 (cubics)
|
||||
|
||||
/**
|
||||
* Constructs a new empty {@code Path2D} object.
|
||||
@ -141,6 +143,42 @@ public abstract class Path2D implements Shape, Cloneable {
|
||||
abstract int rectCrossings(double rxmin, double rymin,
|
||||
double rxmax, double rymax);
|
||||
|
||||
static byte[] expandPointTypes(byte[] oldPointTypes, int needed) {
|
||||
final int oldSize = oldPointTypes.length;
|
||||
final int newSizeMin = oldSize + needed;
|
||||
if (newSizeMin < oldSize) {
|
||||
// hard overflow failure - we can't even accommodate
|
||||
// new items without overflowing
|
||||
throw new ArrayIndexOutOfBoundsException(
|
||||
"pointTypes exceeds maximum capacity !");
|
||||
}
|
||||
// growth algorithm computation
|
||||
int grow = oldSize;
|
||||
if (grow > EXPAND_MAX) {
|
||||
grow = Math.max(EXPAND_MAX, oldSize >> 3); // 1/8th min
|
||||
} else if (grow < EXPAND_MIN) {
|
||||
grow = EXPAND_MIN;
|
||||
}
|
||||
assert grow > 0;
|
||||
|
||||
int newSize = oldSize + grow;
|
||||
if (newSize < newSizeMin) {
|
||||
// overflow in growth algorithm computation
|
||||
newSize = Integer.MAX_VALUE;
|
||||
}
|
||||
while (true) {
|
||||
try {
|
||||
// try allocating the larger array
|
||||
return Arrays.copyOf(oldPointTypes, newSize);
|
||||
} catch (OutOfMemoryError oome) {
|
||||
if (newSize == newSizeMin) {
|
||||
throw oome;
|
||||
}
|
||||
}
|
||||
newSize = newSizeMin + (newSize - newSizeMin) / 2;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* The {@code Float} class defines a geometric path with
|
||||
* coordinates stored in single precision floating point.
|
||||
@ -279,31 +317,53 @@ public abstract class Path2D implements Shape, Cloneable {
|
||||
floatCoords[coordindex+1]);
|
||||
}
|
||||
|
||||
@Override
|
||||
void needRoom(boolean needMove, int newCoords) {
|
||||
if (needMove && numTypes == 0) {
|
||||
if ((numTypes == 0) && needMove) {
|
||||
throw new IllegalPathStateException("missing initial moveto "+
|
||||
"in path definition");
|
||||
}
|
||||
int size = pointTypes.length;
|
||||
if (numTypes >= size) {
|
||||
int grow = size;
|
||||
if (grow > EXPAND_MAX) {
|
||||
grow = EXPAND_MAX;
|
||||
} else if (grow == 0) {
|
||||
grow = 1;
|
||||
}
|
||||
pointTypes = Arrays.copyOf(pointTypes, size+grow);
|
||||
if (numTypes >= pointTypes.length) {
|
||||
pointTypes = expandPointTypes(pointTypes, 1);
|
||||
}
|
||||
size = floatCoords.length;
|
||||
if (numCoords + newCoords > size) {
|
||||
int grow = size;
|
||||
if (grow > EXPAND_MAX * 2) {
|
||||
grow = EXPAND_MAX * 2;
|
||||
if (numCoords > (floatCoords.length - newCoords)) {
|
||||
floatCoords = expandCoords(floatCoords, newCoords);
|
||||
}
|
||||
}
|
||||
|
||||
static float[] expandCoords(float[] oldCoords, int needed) {
|
||||
final int oldSize = oldCoords.length;
|
||||
final int newSizeMin = oldSize + needed;
|
||||
if (newSizeMin < oldSize) {
|
||||
// hard overflow failure - we can't even accommodate
|
||||
// new items without overflowing
|
||||
throw new ArrayIndexOutOfBoundsException(
|
||||
"coords exceeds maximum capacity !");
|
||||
}
|
||||
// growth algorithm computation
|
||||
int grow = oldSize;
|
||||
if (grow > EXPAND_MAX_COORDS) {
|
||||
grow = Math.max(EXPAND_MAX_COORDS, oldSize >> 3); // 1/8th min
|
||||
} else if (grow < EXPAND_MIN) {
|
||||
grow = EXPAND_MIN;
|
||||
}
|
||||
assert grow > needed;
|
||||
|
||||
int newSize = oldSize + grow;
|
||||
if (newSize < newSizeMin) {
|
||||
// overflow in growth algorithm computation
|
||||
newSize = Integer.MAX_VALUE;
|
||||
}
|
||||
while (true) {
|
||||
try {
|
||||
// try allocating the larger array
|
||||
return Arrays.copyOf(oldCoords, newSize);
|
||||
} catch (OutOfMemoryError oome) {
|
||||
if (newSize == newSizeMin) {
|
||||
throw oome;
|
||||
}
|
||||
}
|
||||
if (grow < newCoords) {
|
||||
grow = newCoords;
|
||||
}
|
||||
floatCoords = Arrays.copyOf(floatCoords, size+grow);
|
||||
newSize = newSizeMin + (newSize - newSizeMin) / 2;
|
||||
}
|
||||
}
|
||||
|
||||
@ -1126,31 +1186,53 @@ public abstract class Path2D implements Shape, Cloneable {
|
||||
doubleCoords[coordindex+1]);
|
||||
}
|
||||
|
||||
@Override
|
||||
void needRoom(boolean needMove, int newCoords) {
|
||||
if (needMove && numTypes == 0) {
|
||||
if ((numTypes == 0) && needMove) {
|
||||
throw new IllegalPathStateException("missing initial moveto "+
|
||||
"in path definition");
|
||||
}
|
||||
int size = pointTypes.length;
|
||||
if (numTypes >= size) {
|
||||
int grow = size;
|
||||
if (grow > EXPAND_MAX) {
|
||||
grow = EXPAND_MAX;
|
||||
} else if (grow == 0) {
|
||||
grow = 1;
|
||||
}
|
||||
pointTypes = Arrays.copyOf(pointTypes, size+grow);
|
||||
if (numTypes >= pointTypes.length) {
|
||||
pointTypes = expandPointTypes(pointTypes, 1);
|
||||
}
|
||||
size = doubleCoords.length;
|
||||
if (numCoords + newCoords > size) {
|
||||
int grow = size;
|
||||
if (grow > EXPAND_MAX * 2) {
|
||||
grow = EXPAND_MAX * 2;
|
||||
if (numCoords > (doubleCoords.length - newCoords)) {
|
||||
doubleCoords = expandCoords(doubleCoords, newCoords);
|
||||
}
|
||||
}
|
||||
|
||||
static double[] expandCoords(double[] oldCoords, int needed) {
|
||||
final int oldSize = oldCoords.length;
|
||||
final int newSizeMin = oldSize + needed;
|
||||
if (newSizeMin < oldSize) {
|
||||
// hard overflow failure - we can't even accommodate
|
||||
// new items without overflowing
|
||||
throw new ArrayIndexOutOfBoundsException(
|
||||
"coords exceeds maximum capacity !");
|
||||
}
|
||||
// growth algorithm computation
|
||||
int grow = oldSize;
|
||||
if (grow > EXPAND_MAX_COORDS) {
|
||||
grow = Math.max(EXPAND_MAX_COORDS, oldSize >> 3); // 1/8th min
|
||||
} else if (grow < EXPAND_MIN) {
|
||||
grow = EXPAND_MIN;
|
||||
}
|
||||
assert grow > needed;
|
||||
|
||||
int newSize = oldSize + grow;
|
||||
if (newSize < newSizeMin) {
|
||||
// overflow in growth algorithm computation
|
||||
newSize = Integer.MAX_VALUE;
|
||||
}
|
||||
while (true) {
|
||||
try {
|
||||
// try allocating the larger array
|
||||
return Arrays.copyOf(oldCoords, newSize);
|
||||
} catch (OutOfMemoryError oome) {
|
||||
if (newSize == newSizeMin) {
|
||||
throw oome;
|
||||
}
|
||||
}
|
||||
if (grow < newCoords) {
|
||||
grow = newCoords;
|
||||
}
|
||||
doubleCoords = Arrays.copyOf(doubleCoords, size+grow);
|
||||
newSize = newSizeMin + (newSize - newSizeMin) / 2;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -37,7 +37,7 @@ import java.util.Arrays;
|
||||
* @bug 8076419
|
||||
* @summary Check Path2D copy constructor (trims arrays)
|
||||
* and constructor with zero capacity
|
||||
* @run main Path2DTrimCopy
|
||||
* @run main Path2DCopyConstructor
|
||||
*/
|
||||
public class Path2DCopyConstructor {
|
||||
|
||||
|
188
jdk/test/java/awt/geom/Path2D/Path2DGrow.java
Normal file
188
jdk/test/java/awt/geom/Path2D/Path2DGrow.java
Normal file
@ -0,0 +1,188 @@
|
||||
/*
|
||||
* Copyright (c) 2015, 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.geom.GeneralPath;
|
||||
import java.awt.geom.Path2D;
|
||||
|
||||
/**
|
||||
* @test
|
||||
* @bug 8078464
|
||||
* @summary Check the growth algorithm (needRoom) in Path2D implementations
|
||||
* @run main Path2DGrow
|
||||
*/
|
||||
public class Path2DGrow {
|
||||
|
||||
public static final int N = 1000 * 1000;
|
||||
|
||||
public static boolean verbose = false;
|
||||
public static boolean force = false;
|
||||
|
||||
static void echo(String msg) {
|
||||
System.out.println(msg);
|
||||
}
|
||||
|
||||
static void log(String msg) {
|
||||
if (verbose || force) {
|
||||
echo(msg);
|
||||
}
|
||||
}
|
||||
|
||||
public static void main(String argv[]) {
|
||||
verbose = (argv.length != 0);
|
||||
|
||||
testEmptyDoublePaths();
|
||||
testDoublePaths();
|
||||
|
||||
testEmptyFloatPaths();
|
||||
testFloatPaths();
|
||||
|
||||
testEmptyGeneralPath();
|
||||
testGeneralPath();
|
||||
}
|
||||
|
||||
static void testEmptyDoublePaths() {
|
||||
echo("\n - Test(Path2D.Double[0]) ---");
|
||||
test(() -> new Path2D.Double(Path2D.WIND_NON_ZERO, 0));
|
||||
}
|
||||
|
||||
static void testDoublePaths() {
|
||||
echo("\n - Test(Path2D.Double) ---");
|
||||
test(() -> new Path2D.Double());
|
||||
}
|
||||
|
||||
static void testEmptyFloatPaths() {
|
||||
echo("\n - Test(Path2D.Float[0]) ---");
|
||||
test(() -> new Path2D.Float(Path2D.WIND_NON_ZERO, 0));
|
||||
}
|
||||
|
||||
static void testFloatPaths() {
|
||||
echo("\n - Test(Path2D.Float) ---");
|
||||
test(() -> new Path2D.Float());
|
||||
}
|
||||
|
||||
static void testEmptyGeneralPath() {
|
||||
echo("\n - Test(GeneralPath[0]) ---");
|
||||
test(() -> new GeneralPath(Path2D.WIND_NON_ZERO, 0));
|
||||
}
|
||||
|
||||
static void testGeneralPath() {
|
||||
echo("\n - Test(GeneralPath) ---");
|
||||
test(() -> new GeneralPath());
|
||||
}
|
||||
|
||||
interface PathFactory {
|
||||
Path2D makePath();
|
||||
}
|
||||
|
||||
static void test(PathFactory pf) {
|
||||
long start, end;
|
||||
|
||||
for (int n = 1; n <= N; n *= 10) {
|
||||
force = (n == N);
|
||||
|
||||
start = System.nanoTime();
|
||||
testAddMoves(pf.makePath(), n);
|
||||
end = System.nanoTime();
|
||||
log("testAddMoves[" + n + "] duration= "
|
||||
+ (1e-6 * (end - start)) + " ms.");
|
||||
|
||||
start = System.nanoTime();
|
||||
testAddLines(pf.makePath(), n);
|
||||
end = System.nanoTime();
|
||||
log("testAddLines[" + n + "] duration= "
|
||||
+ (1e-6 * (end - start)) + " ms.");
|
||||
|
||||
start = System.nanoTime();
|
||||
testAddQuads(pf.makePath(), n);
|
||||
end = System.nanoTime();
|
||||
log("testAddQuads[" + n + "] duration= "
|
||||
+ (1e-6 * (end - start)) + " ms.");
|
||||
|
||||
start = System.nanoTime();
|
||||
testAddCubics(pf.makePath(), n);
|
||||
end = System.nanoTime();
|
||||
log("testAddCubics[" + n + "] duration= "
|
||||
+ (1e-6 * (end - start)) + " ms.");
|
||||
|
||||
start = System.nanoTime();
|
||||
testAddMoveAndCloses(pf.makePath(), n);
|
||||
end = System.nanoTime();
|
||||
log("testAddMoveAndCloses[" + n + "] duration= "
|
||||
+ (1e-6 * (end - start)) + " ms.");
|
||||
}
|
||||
}
|
||||
|
||||
static void addMove(Path2D p2d, int i) {
|
||||
p2d.moveTo(1.0 * i, 0.5 * i);
|
||||
}
|
||||
|
||||
static void addLine(Path2D p2d, int i) {
|
||||
p2d.lineTo(1.1 * i, 2.3 * i);
|
||||
}
|
||||
|
||||
static void addCubic(Path2D p2d, int i) {
|
||||
p2d.curveTo(1.1 * i, 1.2 * i, 1.3 * i, 1.4 * i, 1.5 * i, 1.6 * i);
|
||||
}
|
||||
|
||||
static void addQuad(Path2D p2d, int i) {
|
||||
p2d.quadTo(1.1 * i, 1.2 * i, 1.3 * i, 1.4 * i);
|
||||
}
|
||||
|
||||
static void addClose(Path2D p2d) {
|
||||
p2d.closePath();
|
||||
}
|
||||
|
||||
static void testAddMoves(Path2D pathA, int n) {
|
||||
for (int i = 0; i < n; i++) {
|
||||
addMove(pathA, i);
|
||||
}
|
||||
}
|
||||
|
||||
static void testAddLines(Path2D pathA, int n) {
|
||||
addMove(pathA, 0);
|
||||
for (int i = 0; i < n; i++) {
|
||||
addLine(pathA, i);
|
||||
}
|
||||
}
|
||||
|
||||
static void testAddQuads(Path2D pathA, int n) {
|
||||
addMove(pathA, 0);
|
||||
for (int i = 0; i < n; i++) {
|
||||
addQuad(pathA, i);
|
||||
}
|
||||
}
|
||||
|
||||
static void testAddCubics(Path2D pathA, int n) {
|
||||
addMove(pathA, 0);
|
||||
for (int i = 0; i < n; i++) {
|
||||
addCubic(pathA, i);
|
||||
}
|
||||
}
|
||||
|
||||
static void testAddMoveAndCloses(Path2D pathA, int n) {
|
||||
for (int i = 0; i < n; i++) {
|
||||
addMove(pathA, i);
|
||||
addClose(pathA);
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user