From e41e2678482525971100a3ade2c31859cf0d9b4b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laurent=20Bourg=C3=A8s?= Date: Tue, 28 Apr 2015 14:23:03 -0700 Subject: [PATCH] 8078464: Path2D storage growth algorithms should be less linear Reviewed-by: flar --- .../share/classes/java/awt/geom/Path2D.java | 158 +++++++++++---- .../geom/Path2D/Path2DCopyConstructor.java | 2 +- jdk/test/java/awt/geom/Path2D/Path2DGrow.java | 188 ++++++++++++++++++ 3 files changed, 309 insertions(+), 39 deletions(-) create mode 100644 jdk/test/java/awt/geom/Path2D/Path2DGrow.java diff --git a/jdk/src/java.desktop/share/classes/java/awt/geom/Path2D.java b/jdk/src/java.desktop/share/classes/java/awt/geom/Path2D.java index 3c406aa19a7..e05d3579fc1 100644 --- a/jdk/src/java.desktop/share/classes/java/awt/geom/Path2D.java +++ b/jdk/src/java.desktop/share/classes/java/awt/geom/Path2D.java @@ -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; } } diff --git a/jdk/test/java/awt/geom/Path2D/Path2DCopyConstructor.java b/jdk/test/java/awt/geom/Path2D/Path2DCopyConstructor.java index 5413d5138c5..5667a0b8b4c 100644 --- a/jdk/test/java/awt/geom/Path2D/Path2DCopyConstructor.java +++ b/jdk/test/java/awt/geom/Path2D/Path2DCopyConstructor.java @@ -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 { diff --git a/jdk/test/java/awt/geom/Path2D/Path2DGrow.java b/jdk/test/java/awt/geom/Path2D/Path2DGrow.java new file mode 100644 index 00000000000..3ccbc15de43 --- /dev/null +++ b/jdk/test/java/awt/geom/Path2D/Path2DGrow.java @@ -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); + } + } +}