/hg/icedtea6: Backport of S7019861.

dlila at icedtea.classpath.org dlila at icedtea.classpath.org
Wed Mar 16 12:06:19 PDT 2011


changeset b69c6e3f3d78 in /hg/icedtea6
details: http://icedtea.classpath.org/hg/icedtea6?cmd=changeset;node=b69c6e3f3d78
author: Denis Lila <dlila at redhat.com>
date: Wed Mar 16 14:11:16 2011 -0400

	Backport of S7019861.


diffstat:

 ChangeLog                                       |    8 +
 Makefile.am                                     |    3 +-
 NEWS                                            |    1 +
 patches/openjdk/7019861-AA-regression-fix.patch |  497 ++++++++++++++++++++++++
 4 files changed, 508 insertions(+), 1 deletions(-)

diffs (truncated from 545 to 500 lines):

diff -r 7cc732051121 -r b69c6e3f3d78 ChangeLog
--- a/ChangeLog	Wed Mar 16 09:50:47 2011 -0400
+++ b/ChangeLog	Wed Mar 16 14:11:16 2011 -0400
@@ -1,3 +1,11 @@
+2011-03-16  Denis Lila  <dlila at redhat.com>
+
+	* Makefile.am: Add patch.
+	* patches/openjdk/7019861-AA-regression-fix.patch:
+	Backport.
+	* NEWS:
+	Updated with backport.
+
 2011-03-16  Denis Lila  <dlila at redhat.com>
 
 	* Makefile.am: Add patches.
diff -r 7cc732051121 -r b69c6e3f3d78 Makefile.am
--- a/Makefile.am	Wed Mar 16 09:50:47 2011 -0400
+++ b/Makefile.am	Wed Mar 16 14:11:16 2011 -0400
@@ -328,7 +328,8 @@
 	patches/jtreg-hotspot-bug-6196102.patch \
 	patches/jtreg-double-to-string.patch \
 	patches/openjdk/7023591-AAShapePipe.patch \
-	patches/openjdk/7027667-AAShapePipeRegTest.patch
+	patches/openjdk/7027667-AAShapePipeRegTest.patch \
+	patches/openjdk/7019861-AA-regression-fix.patch
 
 if WITH_ALT_HSBUILD
 ICEDTEA_PATCHES += \
diff -r 7cc732051121 -r b69c6e3f3d78 NEWS
--- a/NEWS	Wed Mar 16 09:50:47 2011 -0400
+++ b/NEWS	Wed Mar 16 14:11:16 2011 -0400
@@ -22,6 +22,7 @@
   - CACAO PR157: ARM SMP Assertion thinlock failed.
 * Backports
   - S7023591, S7027667: Clipped antialiased rectangles are filled, not drawn.
+  - S7019861: Last scanline skpped when doing AA.
 
 New in release 1.10 (2011-XX-XX):
 
diff -r 7cc732051121 -r b69c6e3f3d78 patches/openjdk/7019861-AA-regression-fix.patch
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/patches/openjdk/7019861-AA-regression-fix.patch	Wed Mar 16 14:11:16 2011 -0400
@@ -0,0 +1,502 @@
+# HG changeset patch
+# User dlila
+# Date 1300223102 14400
+# Node ID 253f3bc64961944b1886a243a662bd929e9ab605
+# Parent  fd8b81c558d307a984ad2dcd06b7de092c3f01f5
+7019861: Last scanline is skipped in pisces.Renderer.
+Summary: not skipping it anymore.
+Reviewed-by: flar
+
+diff -r fd8b81c558d3 -r 253f3bc64961 src/share/classes/sun/java2d/pisces/Helpers.java
+--- openjdk.orig/jdk/src/share/classes/sun/java2d/pisces/Helpers.java	Tue Mar 15 15:15:10 2011 -0400
++++ openjdk/jdk/src/share/classes/sun/java2d/pisces/Helpers.java	Tue Mar 15 17:05:02 2011 -0400
+@@ -154,9 +154,6 @@
+     // These use a hardcoded factor of 2 for increasing sizes. Perhaps this
+     // should be provided as an argument.
+     static float[] widenArray(float[] in, final int cursize, final int numToAdd) {
+-        if (in == null) {
+-            return new float[5 * numToAdd];
+-        }
+         if (in.length >= cursize + numToAdd) {
+             return in;
+         }
+diff -r fd8b81c558d3 -r 253f3bc64961 src/share/classes/sun/java2d/pisces/PiscesTileGenerator.java
+--- openjdk.orig/jdk/src/share/classes/sun/java2d/pisces/PiscesTileGenerator.java	Tue Mar 15 15:15:10 2011 -0400
++++ openjdk/jdk/src/share/classes/sun/java2d/pisces/PiscesTileGenerator.java	Tue Mar 15 17:05:02 2011 -0400
+@@ -191,8 +191,7 @@
+                     System.out.println("len = "+runLen);
+                     System.out.print(cache.toString());
+                     e0.printStackTrace();
+-                    System.exit(1);
+-                    return;
++                    throw e0;
+                 }
+ 
+                 int rx0 = cx;
+@@ -215,8 +214,7 @@
+                         System.out.println("len = "+runLen);
+                         System.out.print(cache.toString());
+                         e.printStackTrace();
+-                        System.exit(1);
+-                        return;
++                        throw e;
+                     }
+                 }
+                 pos += 2;
+@@ -250,4 +248,5 @@
+      * No further calls will be made on this instance.
+      */
+     public void dispose() {}
+-}
+\ No newline at end of file
++}
++
+diff -r fd8b81c558d3 -r 253f3bc64961 src/share/classes/sun/java2d/pisces/Renderer.java
+--- openjdk.orig/jdk/src/share/classes/sun/java2d/pisces/Renderer.java	Tue Mar 15 15:15:10 2011 -0400
++++ openjdk/jdk/src/share/classes/sun/java2d/pisces/Renderer.java	Tue Mar 15 17:05:02 2011 -0400
+@@ -47,16 +47,16 @@
+ 
+         private static final int INIT_CROSSINGS_SIZE = 10;
+ 
+-        private ScanlineIterator() {
++        // Preconditions: Only subpixel scanlines in the range
++        // (start <= subpixel_y <= end) will be evaluated. No
++        // edge may have a valid (i.e. inside the supplied clip)
++        // crossing that would be generated outside that range.
++        private ScanlineIterator(int start, int end) {
+             crossings = new int[INIT_CROSSINGS_SIZE];
+             edgePtrs = new int[INIT_CROSSINGS_SIZE];
+ 
+-            // We don't care if we clip some of the line off with ceil, since
+-            // no scan line crossings will be eliminated (in fact, the ceil is
+-            // the y of the first scan line crossing).
+-            final int minY = getFirstScanLineCrossing();
+-            nextY = minY;
+-            maxY = getScanLineCrossingEnd()-1;
++            nextY = start;
++            maxY = end;
+             edgeCount = 0;
+         }
+ 
+@@ -148,6 +148,7 @@
+     // don't just set NULL to -1, because we want NULL+NEXT to be negative.
+     private static final int NULL = -SIZEOF_EDGE;
+     private float[] edges = null;
++    private static final int INIT_NUM_EDGES = 8;
+     private int[] edgeBuckets = null;
+     private int[] edgeBucketCounts = null; // 2*newedges + (1 if pruning needed)
+     private int numEdges;
+@@ -156,7 +157,7 @@
+     private static final float INC_BND = 8f;
+ 
+     // each bucket is a linked list. this method adds eptr to the
+-    // start "bucket"th linked list.
++    // start of the "bucket"th linked list.
+     private void addEdgeToBucket(final int eptr, final int bucket) {
+         edges[eptr+NEXT] = edgeBuckets[bucket];
+         edgeBuckets[bucket] = eptr;
+@@ -168,7 +169,8 @@
+     // X0, Y0, D*[X|Y], COUNT; not variables used for computing scanline crossings).
+     private void quadBreakIntoLinesAndAdd(float x0, float y0,
+                                           final Curve c,
+-                                          final float x2, final float y2) {
++                                          final float x2, final float y2)
++    {
+         final float QUAD_DEC_BND = 32;
+         final int countlg = 4;
+         int count = 1 << countlg;
+@@ -204,7 +206,8 @@
+     // here, but then too many numbers are passed around.
+     private void curveBreakIntoLinesAndAdd(float x0, float y0,
+                                            final Curve c,
+-                                           final float x3, final float y3) {
++                                           final float x3, final float y3)
++    {
+         final int countlg = 3;
+         int count = 1 << countlg;
+ 
+@@ -259,8 +262,6 @@
+         }
+     }
+ 
+-    // Preconditions: y2 > y1 and the curve must cross some scanline
+-    // i.e.: y1 <= y < y2 for some y such that boundsMinY <= y < boundsMaxY
+     private void addLine(float x1, float y1, float x2, float y2) {
+         float or = 1; // orientation of the line. 1 if y increases, 0 otherwise.
+         if (y2 < y1) {
+@@ -272,12 +273,11 @@
+             x1 = or;
+             or = 0;
+         }
+-        final int firstCrossing = Math.max((int) Math.ceil(y1), boundsMinY);
++        final int firstCrossing = Math.max((int)Math.ceil(y1), boundsMinY);
+         final int lastCrossing = Math.min((int)Math.ceil(y2), boundsMaxY);
+         if (firstCrossing >= lastCrossing) {
+             return;
+         }
+-
+         if (y1 < edgeMinY) { edgeMinY = y1; }
+         if (y2 > edgeMaxY) { edgeMaxY = y2; }
+ 
+@@ -297,22 +297,10 @@
+         edges[ptr+OR] = or;
+         edges[ptr+CURX] = x1 + (firstCrossing - y1) * slope;
+         edges[ptr+SLOPE] = slope;
+-        edges[ptr+YMAX] = y2;
++        edges[ptr+YMAX] = lastCrossing;
+         final int bucketIdx = firstCrossing - boundsMinY;
+         addEdgeToBucket(ptr, bucketIdx);
+-        if (lastCrossing < boundsMaxY) {
+-            edgeBucketCounts[lastCrossing - boundsMinY] |= 1;
+-        }
+-    }
+-
+-    // preconditions: should not be called before the last line has been added
+-    // to the edge list (even though it will return a correct answer at that
+-    // point in time, it's not meant to be used that way).
+-    private int getFirstScanLineCrossing() {
+-        return Math.max(boundsMinY, (int)Math.ceil(edgeMinY));
+-    }
+-    private int getScanLineCrossingEnd() {
+-        return Math.min(boundsMaxY, (int)Math.ceil(edgeMaxY));
++        edgeBucketCounts[lastCrossing - boundsMinY] |= 1;
+     }
+ 
+ // END EDGE LIST
+@@ -366,9 +354,11 @@
+         this.boundsMaxX = (pix_boundsX + pix_boundsWidth) * SUBPIXEL_POSITIONS_X;
+         this.boundsMaxY = (pix_boundsY + pix_boundsHeight) * SUBPIXEL_POSITIONS_Y;
+ 
++        edges = new float[INIT_NUM_EDGES * SIZEOF_EDGE];
++        numEdges = 0;
+         edgeBuckets = new int[boundsMaxY - boundsMinY];
+         java.util.Arrays.fill(edgeBuckets, NULL);
+-        edgeBucketCounts = new int[edgeBuckets.length];
++        edgeBucketCounts = new int[edgeBuckets.length + 1];
+     }
+ 
+     private float tosubpixx(float pix_x) {
+@@ -394,7 +384,7 @@
+         y0 = y1;
+     }
+ 
+-    Curve c = new Curve();
++    private Curve c = new Curve();
+     @Override public void curveTo(float x1, float y1,
+                                   float x2, float y2,
+                                   float x3, float y3)
+@@ -431,8 +421,8 @@
+         throw new InternalError("Renderer does not use a native consumer.");
+     }
+ 
+-    private void _endRendering(final int pix_bboxx0, final int pix_bboxy0,
+-                               final int pix_bboxx1, final int pix_bboxy1)
++    private void _endRendering(final int pix_bboxx0, final int pix_bboxx1,
++                               int ymin, int ymax)
+     {
+         // Mask to determine the relevant bit of the crossing sum
+         // 0x1 if EVEN_ODD, all bits if NON_ZERO
+@@ -455,7 +445,7 @@
+         int pix_minX = Integer.MAX_VALUE;
+ 
+         int y = boundsMinY; // needs to be declared here so we emit the last row properly.
+-        ScanlineIterator it = this.new ScanlineIterator();
++        ScanlineIterator it = this.new ScanlineIterator(ymin, ymax);
+         for ( ; it.hasNext(); ) {
+             int numCrossings = it.next();
+             int[] crossings = it.crossings;
+@@ -477,7 +467,7 @@
+                 int curxo = crossings[i];
+                 int curx = curxo >> 1;
+                 // to turn {0, 1} into {-1, 1}, multiply by 2 and subtract 1.
+-                int crorientation = ((curxo & 0x1) << 1) -1;
++                int crorientation = ((curxo & 0x1) << 1) - 1;
+                 if ((sum & mask) != 0) {
+                     int x0 = Math.max(prev, bboxx0);
+                     int x1 = Math.min(curx, bboxx1);
+@@ -541,7 +531,7 @@
+         }
+ 
+         this.cache = new PiscesCache(pminX, pminY, pmaxX, pmaxY);
+-        _endRendering(pminX, pminY, pmaxX, pmaxY);
++        _endRendering(pminX, pmaxX, spminY, spmaxY);
+     }
+ 
+     public PiscesCache getCache() {
+diff -r fd8b81c558d3 -r 253f3bc64961 src/share/classes/sun/java2d/pisces/Stroker.java
+--- openjdk.orig/jdk/src/share/classes/sun/java2d/pisces/Stroker.java	Tue Mar 15 15:15:10 2011 -0400
++++ openjdk/jdk/src/share/classes/sun/java2d/pisces/Stroker.java	Tue Mar 15 17:05:02 2011 -0400
+@@ -764,6 +764,11 @@
+     private static final int MAX_N_CURVES = 11;
+     private float[] subdivTs = new float[MAX_N_CURVES - 1];
+ 
++    // If this class is compiled with ecj, then Hotspot crashes when OSR
++    // compiling this function. See bugs 7004570 and 6675699
++    // TODO: until those are fixed, we should work around that by
++    // manually inlining this into curveTo and quadTo.
++/******************************* WORKAROUND **********************************
+     private void somethingTo(final int type) {
+         // need these so we can update the state at the end of this method
+         final float xf = middle[type-2], yf = middle[type-1];
+@@ -866,6 +871,7 @@
+         this.cy0 = yf;
+         this.prev = DRAWING_OP_TO;
+     }
++****************************** END WORKAROUND *******************************/
+ 
+     // finds values of t where the curve in pts should be subdivided in order
+     // to get good offset curves a distance of w away from the middle curve.
+@@ -932,18 +938,168 @@
+         middle[2] = x1;  middle[3] = y1;
+         middle[4] = x2;  middle[5] = y2;
+         middle[6] = x3;  middle[7] = y3;
+-        somethingTo(8);
+-    }
+ 
+-    @Override public long getNativeConsumer() {
+-        throw new InternalError("Stroker doesn't use a native consumer");
++        // inlined version of somethingTo(8);
++        // See the TODO on somethingTo
++
++        // need these so we can update the state at the end of this method
++        final float xf = middle[6], yf = middle[7];
++        float dxs = middle[2] - middle[0];
++        float dys = middle[3] - middle[1];
++        float dxf = middle[6] - middle[4];
++        float dyf = middle[7] - middle[5];
++
++        boolean p1eqp2 = (dxs == 0f && dys == 0f);
++        boolean p3eqp4 = (dxf == 0f && dyf == 0f);
++        if (p1eqp2) {
++            dxs = middle[4] - middle[0];
++            dys = middle[5] - middle[1];
++            if (dxs == 0f && dys == 0f) {
++                dxs = middle[6] - middle[0];
++                dys = middle[7] - middle[1];
++            }
++        }
++        if (p3eqp4) {
++            dxf = middle[6] - middle[2];
++            dyf = middle[7] - middle[3];
++            if (dxf == 0f && dyf == 0f) {
++                dxf = middle[6] - middle[0];
++                dyf = middle[7] - middle[1];
++            }
++        }
++        if (dxs == 0f && dys == 0f) {
++            // this happens iff the "curve" is just a point
++            lineTo(middle[0], middle[1]);
++            return;
++        }
++
++        // if these vectors are too small, normalize them, to avoid future
++        // precision problems.
++        if (Math.abs(dxs) < 0.1f && Math.abs(dys) < 0.1f) {
++            float len = (float)Math.sqrt(dxs*dxs + dys*dys);
++            dxs /= len;
++            dys /= len;
++        }
++        if (Math.abs(dxf) < 0.1f && Math.abs(dyf) < 0.1f) {
++            float len = (float)Math.sqrt(dxf*dxf + dyf*dyf);
++            dxf /= len;
++            dyf /= len;
++        }
++
++        computeOffset(dxs, dys, lineWidth2, offset[0]);
++        final float mx = offset[0][0];
++        final float my = offset[0][1];
++        drawJoin(cdx, cdy, cx0, cy0, dxs, dys, cmx, cmy, mx, my);
++
++        int nSplits = findSubdivPoints(middle, subdivTs, 8, lineWidth2);
++
++        int kind = 0;
++        Iterator<Integer> it = Curve.breakPtsAtTs(middle, 8, subdivTs, nSplits);
++        while(it.hasNext()) {
++            int curCurveOff = it.next();
++
++            kind = computeOffsetCubic(middle, curCurveOff, lp, rp);
++            if (kind != 0) {
++                emitLineTo(lp[0], lp[1]);
++                switch(kind) {
++                case 8:
++                    emitCurveTo(lp[0], lp[1], lp[2], lp[3], lp[4], lp[5], lp[6], lp[7], false);
++                    emitCurveTo(rp[0], rp[1], rp[2], rp[3], rp[4], rp[5], rp[6], rp[7], true);
++                    break;
++                case 4:
++                    emitLineTo(lp[2], lp[3]);
++                    emitLineTo(rp[0], rp[1], true);
++                    break;
++                }
++                emitLineTo(rp[kind - 2], rp[kind - 1], true);
++            }
++        }
++
++        this.cmx = (lp[kind - 2] - rp[kind - 2]) / 2;
++        this.cmy = (lp[kind - 1] - rp[kind - 1]) / 2;
++        this.cdx = dxf;
++        this.cdy = dyf;
++        this.cx0 = xf;
++        this.cy0 = yf;
++        this.prev = DRAWING_OP_TO;
+     }
+ 
+     @Override public void quadTo(float x1, float y1, float x2, float y2) {
+         middle[0] = cx0; middle[1] = cy0;
+         middle[2] = x1;  middle[3] = y1;
+         middle[4] = x2;  middle[5] = y2;
+-        somethingTo(6);
++
++        // inlined version of somethingTo(8);
++        // See the TODO on somethingTo
++
++        // need these so we can update the state at the end of this method
++        final float xf = middle[4], yf = middle[5];
++        float dxs = middle[2] - middle[0];
++        float dys = middle[3] - middle[1];
++        float dxf = middle[4] - middle[2];
++        float dyf = middle[5] - middle[3];
++        if ((dxs == 0f && dys == 0f) || (dxf == 0f && dyf == 0f)) {
++            dxs = dxf = middle[4] - middle[0];
++            dys = dyf = middle[5] - middle[1];
++        }
++        if (dxs == 0f && dys == 0f) {
++            // this happens iff the "curve" is just a point
++            lineTo(middle[0], middle[1]);
++            return;
++        }
++        // if these vectors are too small, normalize them, to avoid future
++        // precision problems.
++        if (Math.abs(dxs) < 0.1f && Math.abs(dys) < 0.1f) {
++            float len = (float)Math.sqrt(dxs*dxs + dys*dys);
++            dxs /= len;
++            dys /= len;
++        }
++        if (Math.abs(dxf) < 0.1f && Math.abs(dyf) < 0.1f) {
++            float len = (float)Math.sqrt(dxf*dxf + dyf*dyf);
++            dxf /= len;
++            dyf /= len;
++        }
++
++        computeOffset(dxs, dys, lineWidth2, offset[0]);
++        final float mx = offset[0][0];
++        final float my = offset[0][1];
++        drawJoin(cdx, cdy, cx0, cy0, dxs, dys, cmx, cmy, mx, my);
++
++        int nSplits = findSubdivPoints(middle, subdivTs, 6, lineWidth2);
++
++        int kind = 0;
++        Iterator<Integer> it = Curve.breakPtsAtTs(middle, 6, subdivTs, nSplits);
++        while(it.hasNext()) {
++            int curCurveOff = it.next();
++
++            kind = computeOffsetQuad(middle, curCurveOff, lp, rp);
++            if (kind != 0) {
++                emitLineTo(lp[0], lp[1]);
++                switch(kind) {
++                case 6:
++                    emitQuadTo(lp[0], lp[1], lp[2], lp[3], lp[4], lp[5], false);
++                    emitQuadTo(rp[0], rp[1], rp[2], rp[3], rp[4], rp[5], true);
++                    break;
++                case 4:
++                    emitLineTo(lp[2], lp[3]);
++                    emitLineTo(rp[0], rp[1], true);
++                    break;
++                }
++                emitLineTo(rp[kind - 2], rp[kind - 1], true);
++            }
++        }
++
++        this.cmx = (lp[kind - 2] - rp[kind - 2]) / 2;
++        this.cmy = (lp[kind - 1] - rp[kind - 1]) / 2;
++        this.cdx = dxf;
++        this.cdy = dyf;
++        this.cx0 = xf;
++        this.cy0 = yf;
++        this.prev = DRAWING_OP_TO;
++    }
++
++    @Override public long getNativeConsumer() {
++        throw new InternalError("Stroker doesn't use a native consumer");
+     }
+ 
+     // a stack of polynomial curves where each curve shares endpoints with
+diff -r fd8b81c558d3 -r 253f3bc64961 test/sun/java2d/pisces/Renderer/Test7019861.java
+--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
++++ openjdk/jdk/test/sun/java2d/pisces/Renderer/Test7019861.java	Tue Mar 15 17:05:02 2011 -0400
+@@ -0,0 +1,76 @@
++/*
++ * Copyright (c) 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     7019861
++ *
++ * @summary Verifies that the last scanline isn't skipped when doing
++ *          antialiased rendering.
++ *
++ * @run     main Test7019861



More information about the distro-pkg-dev mailing list