8316741: BasicStroke.createStrokedShape miter-limits failing on small shapes
Reviewed-by: prr, dnguyen
This commit is contained in:
parent
4cf195f00c
commit
a876beb63d
src/java.desktop/share/classes/sun/java2d/marlin
test/jdk/sun/java2d/marlin
@ -557,6 +557,7 @@ final class Renderer implements DPathConsumer2D, MarlinConst {
|
||||
final int pix_boundsWidth, final int pix_boundsHeight,
|
||||
final int windingRule)
|
||||
{
|
||||
this.rdrCtx.doRender = true;
|
||||
this.windingRule = windingRule;
|
||||
|
||||
// bounds as half-open intervals: minX <= x < maxX and minY <= y < maxY
|
||||
|
@ -78,6 +78,8 @@ final class RendererContext extends ReentrantContext implements MarlinConst {
|
||||
final MarlinCache cache;
|
||||
// flag indicating the shape is stroked (1) or filled (0)
|
||||
int stroking = 0;
|
||||
// flag indicating to render the shape
|
||||
boolean doRender = false;
|
||||
// flag indicating to clip the shape
|
||||
boolean doClip = false;
|
||||
// flag indicating if the path is closed or not (in advance) to handle properly caps
|
||||
@ -169,6 +171,7 @@ final class RendererContext extends ReentrantContext implements MarlinConst {
|
||||
stats.totalOffHeap = 0L;
|
||||
}
|
||||
stroking = 0;
|
||||
doRender = false;
|
||||
doClip = false;
|
||||
closedPath = false;
|
||||
clipInvScale = 0.0d;
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2007, 2022, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2007, 2023, 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
|
||||
@ -170,25 +170,34 @@ final class Stroker implements StartFlagPathConsumer2D, MarlinConst {
|
||||
miterScaledLimit = miterLimit * lineWidth2;
|
||||
this.miterLimitSq = miterScaledLimit * miterScaledLimit;
|
||||
|
||||
final double limitMin = ((this.rdrCtx.clipInvScale == 0.0d) ? JOIN_ERROR
|
||||
: (JOIN_ERROR * this.rdrCtx.clipInvScale))
|
||||
+ lineWidth2;
|
||||
|
||||
this.joinLimitMinSq = limitMin * limitMin;
|
||||
if (rdrCtx.doRender) {
|
||||
final double limitMin = ((this.rdrCtx.clipInvScale == 0.0d) ? JOIN_ERROR
|
||||
: (JOIN_ERROR * this.rdrCtx.clipInvScale))
|
||||
+ lineWidth2;
|
||||
|
||||
this.joinLimitMinSq = limitMin * limitMin;
|
||||
} else {
|
||||
// createStrokedShape(): disable limit checks:
|
||||
this.joinLimitMinSq = 0.0;
|
||||
}
|
||||
} else if (joinStyle == JOIN_ROUND) {
|
||||
// chord: s = 2 r * sin( phi / 2)
|
||||
// height: h = 2 r * sin( phi / 4)^2
|
||||
// small angles (phi < 90):
|
||||
// h = s^2 / (8 r)
|
||||
// so s^2 = (8 h * r)
|
||||
if (rdrCtx.doRender) {
|
||||
// chord: s = 2 r * sin( phi / 2)
|
||||
// height: h = 2 r * sin( phi / 4)^2
|
||||
// small angles (phi < 90):
|
||||
// h = s^2 / (8 r)
|
||||
// so s^2 = (8 h * r)
|
||||
|
||||
// height max (note ROUND_JOIN_ERROR = 8 * JOIN_ERROR)
|
||||
final double limitMin = ((this.rdrCtx.clipInvScale == 0.0d) ? ROUND_JOIN_ERROR
|
||||
: (ROUND_JOIN_ERROR * this.rdrCtx.clipInvScale));
|
||||
// height max (note ROUND_JOIN_ERROR = 8 * JOIN_ERROR)
|
||||
final double limitMin = ((this.rdrCtx.clipInvScale == 0.0d) ? ROUND_JOIN_ERROR
|
||||
: (ROUND_JOIN_ERROR * this.rdrCtx.clipInvScale));
|
||||
|
||||
// chord limit (s^2):
|
||||
this.joinLimitMinSq = limitMin * this.lineWidth2;
|
||||
// chord limit (s^2):
|
||||
this.joinLimitMinSq = limitMin * this.lineWidth2;
|
||||
} else {
|
||||
// createStrokedShape(): disable limit checks:
|
||||
this.joinLimitMinSq = 0.0;
|
||||
}
|
||||
}
|
||||
this.prev = CLOSE;
|
||||
|
||||
|
@ -27,7 +27,7 @@ package sun.java2d.marlin;
|
||||
|
||||
public final class Version {
|
||||
|
||||
private static final String VERSION = "marlin-0.9.4.6-Unsafe-OpenJDK";
|
||||
private static final String VERSION = "marlin-0.9.4.6.1-Unsafe-OpenJDK";
|
||||
|
||||
public static String getVersion() {
|
||||
return VERSION;
|
||||
|
104
test/jdk/sun/java2d/marlin/TestCreateStrokedShapeJoins.java
Normal file
104
test/jdk/sun/java2d/marlin/TestCreateStrokedShapeJoins.java
Normal file
@ -0,0 +1,104 @@
|
||||
/*
|
||||
* Copyright (c) 2023, 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.io.*;
|
||||
import java.awt.*;
|
||||
import java.awt.image.*;
|
||||
import java.awt.geom.*;
|
||||
import java.util.Arrays;
|
||||
import javax.imageio.*;
|
||||
|
||||
/**
|
||||
* @test
|
||||
* @bug 8316741
|
||||
* @summary Verifies that Marlin renderer's Stroker generates properly joins
|
||||
* in createStrokedShape()
|
||||
* @run main TestCreateStrokedShapeJoins
|
||||
*/
|
||||
public class TestCreateStrokedShapeJoins {
|
||||
|
||||
static final boolean SAVE_IMAGE = false;
|
||||
|
||||
private final static int W = 200;
|
||||
|
||||
private final static int[] REF_COUNTS = new int[] {4561, 4790, 5499};
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
final int[] test = new int[] {
|
||||
test(BasicStroke.JOIN_BEVEL),
|
||||
test(BasicStroke.JOIN_ROUND),
|
||||
test(BasicStroke.JOIN_MITER)
|
||||
};
|
||||
|
||||
System.out.println("test: " + Arrays.toString(test));
|
||||
System.out.println("ref: " + Arrays.toString(REF_COUNTS));
|
||||
|
||||
// check results:
|
||||
for (int i = 0; i < REF_COUNTS.length; i++) {
|
||||
if (test[i] != REF_COUNTS[i]) {
|
||||
throw new RuntimeException("Invalid test[" + i + "]: " + test[i] + " != " + REF_COUNTS[i]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static int test(int join) throws Exception {
|
||||
final BufferedImage image = new BufferedImage(W, W, BufferedImage.TYPE_INT_ARGB);
|
||||
final Graphics2D g = image.createGraphics();
|
||||
try {
|
||||
g.setPaint(Color.BLACK);
|
||||
g.fillRect(0, 0, W, W);
|
||||
g.setPaint(Color.WHITE);
|
||||
g.setTransform(new AffineTransform(W, 0, 0, W, 0, 0));
|
||||
|
||||
final BasicStroke stroke = new BasicStroke(0.15f, 0, join, 10);
|
||||
|
||||
final Path2D p = new Path2D.Float();
|
||||
p.moveTo(0.95f, 0.6f);
|
||||
p.lineTo(0.5f, 0.5f);
|
||||
p.lineTo(0.95f, 0.4f);
|
||||
|
||||
final Shape outline = stroke.createStrokedShape(p);
|
||||
g.fill(outline);
|
||||
} finally {
|
||||
g.dispose();
|
||||
}
|
||||
if (SAVE_IMAGE) {
|
||||
final File file = new File("TestCreateStrokedShapeJoins-" + join + ".png");
|
||||
System.out.println("Writing " + file.getAbsolutePath());
|
||||
ImageIO.write(image, "png", file);
|
||||
}
|
||||
int count = 0;
|
||||
|
||||
for (int y = 0; y < W; y++) {
|
||||
for (int x = 0; x < W; x++) {
|
||||
final int rgb = image.getRGB(x, y);
|
||||
final int b = rgb & 0xFF;
|
||||
|
||||
if (b != 0) {
|
||||
count++;
|
||||
}
|
||||
}
|
||||
}
|
||||
return count;
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user