8059942: Default implementation of DrawImage.renderImageXform() should be improved for d3d/ogl
Reviewed-by: flar, prr
This commit is contained in:
parent
6c2369366c
commit
246a6da52e
@ -364,8 +364,53 @@ public class DrawImage implements DrawImagePipe
|
|||||||
int sx1, int sy1, int sx2, int sy2,
|
int sx1, int sy1, int sx2, int sy2,
|
||||||
Color bgColor)
|
Color bgColor)
|
||||||
{
|
{
|
||||||
|
final AffineTransform itx;
|
||||||
|
try {
|
||||||
|
itx = tx.createInverse();
|
||||||
|
} catch (final NoninvertibleTransformException ignored) {
|
||||||
|
// Non-invertible transform means no output
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Find the maximum bounds on the destination that will be
|
||||||
|
* affected by the transformed source. First, transform all
|
||||||
|
* four corners of the source and then min and max the resulting
|
||||||
|
* destination coordinates of the transformed corners.
|
||||||
|
* Note that tx already has the offset to sx1,sy1 accounted
|
||||||
|
* for so we use the box (0, 0, sx2-sx1, sy2-sy1) as the
|
||||||
|
* source coordinates.
|
||||||
|
*/
|
||||||
|
final double[] coords = new double[8];
|
||||||
|
/* corner: UL UR LL LR */
|
||||||
|
/* index: 0 1 2 3 4 5 6 7 */
|
||||||
|
/* coord: (0, 0), (w, 0), (0, h), (w, h) */
|
||||||
|
coords[2] = coords[6] = sx2 - sx1;
|
||||||
|
coords[5] = coords[7] = sy2 - sy1;
|
||||||
|
tx.transform(coords, 0, coords, 0, 4);
|
||||||
|
double ddx1, ddy1, ddx2, ddy2;
|
||||||
|
ddx1 = ddx2 = coords[0];
|
||||||
|
ddy1 = ddy2 = coords[1];
|
||||||
|
for (int i = 2; i < coords.length; i += 2) {
|
||||||
|
double d = coords[i];
|
||||||
|
if (ddx1 > d) ddx1 = d;
|
||||||
|
else if (ddx2 < d) ddx2 = d;
|
||||||
|
d = coords[i+1];
|
||||||
|
if (ddy1 > d) ddy1 = d;
|
||||||
|
else if (ddy2 < d) ddy2 = d;
|
||||||
|
}
|
||||||
|
|
||||||
Region clip = sg.getCompClip();
|
Region clip = sg.getCompClip();
|
||||||
SurfaceData dstData = sg.surfaceData;
|
final int dx1 = Math.max((int) Math.floor(ddx1), clip.lox);
|
||||||
|
final int dy1 = Math.max((int) Math.floor(ddy1), clip.loy);
|
||||||
|
final int dx2 = Math.min((int) Math.ceil(ddx2), clip.hix);
|
||||||
|
final int dy2 = Math.min((int) Math.ceil(ddy2), clip.hiy);
|
||||||
|
if (dx2 <= dx1 || dy2 <= dy1) {
|
||||||
|
// empty destination means no output
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final SurfaceData dstData = sg.surfaceData;
|
||||||
SurfaceData srcData = dstData.getSourceSurfaceData(img,
|
SurfaceData srcData = dstData.getSourceSurfaceData(img,
|
||||||
SunGraphics2D.TRANSFORM_GENERIC,
|
SunGraphics2D.TRANSFORM_GENERIC,
|
||||||
sg.imageComp,
|
sg.imageComp,
|
||||||
@ -429,56 +474,13 @@ public class DrawImage implements DrawImagePipe
|
|||||||
// assert(helper != null);
|
// assert(helper != null);
|
||||||
}
|
}
|
||||||
|
|
||||||
AffineTransform itx;
|
|
||||||
try {
|
|
||||||
itx = tx.createInverse();
|
|
||||||
} catch (NoninvertibleTransformException e) {
|
|
||||||
// Non-invertible transform means no output
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Find the maximum bounds on the destination that will be
|
|
||||||
* affected by the transformed source. First, transform all
|
|
||||||
* four corners of the source and then min and max the resulting
|
|
||||||
* destination coordinates of the transformed corners.
|
|
||||||
* Note that tx already has the offset to sx1,sy1 accounted
|
|
||||||
* for so we use the box (0, 0, sx2-sx1, sy2-sy1) as the
|
|
||||||
* source coordinates.
|
|
||||||
*/
|
|
||||||
double coords[] = new double[8];
|
|
||||||
/* corner: UL UR LL LR */
|
|
||||||
/* index: 0 1 2 3 4 5 6 7 */
|
|
||||||
/* coord: (0, 0), (w, 0), (0, h), (w, h) */
|
|
||||||
coords[2] = coords[6] = sx2 - sx1;
|
|
||||||
coords[5] = coords[7] = sy2 - sy1;
|
|
||||||
tx.transform(coords, 0, coords, 0, 4);
|
|
||||||
double ddx1, ddy1, ddx2, ddy2;
|
|
||||||
ddx1 = ddx2 = coords[0];
|
|
||||||
ddy1 = ddy2 = coords[1];
|
|
||||||
for (int i = 2; i < coords.length; i += 2) {
|
|
||||||
double d = coords[i];
|
|
||||||
if (ddx1 > d) ddx1 = d;
|
|
||||||
else if (ddx2 < d) ddx2 = d;
|
|
||||||
d = coords[i+1];
|
|
||||||
if (ddy1 > d) ddy1 = d;
|
|
||||||
else if (ddy2 < d) ddy2 = d;
|
|
||||||
}
|
|
||||||
int dx1 = (int) Math.floor(ddx1);
|
|
||||||
int dy1 = (int) Math.floor(ddy1);
|
|
||||||
int dx2 = (int) Math.ceil(ddx2);
|
|
||||||
int dy2 = (int) Math.ceil(ddy2);
|
|
||||||
|
|
||||||
SurfaceType dstType = dstData.getSurfaceType();
|
SurfaceType dstType = dstData.getSurfaceType();
|
||||||
MaskBlit maskblit;
|
|
||||||
Blit blit;
|
|
||||||
if (sg.compositeState <= SunGraphics2D.COMP_ALPHA) {
|
if (sg.compositeState <= SunGraphics2D.COMP_ALPHA) {
|
||||||
/* NOTE: We either have, or we can make,
|
/* NOTE: We either have, or we can make,
|
||||||
* a MaskBlit for any alpha composite type
|
* a MaskBlit for any alpha composite type
|
||||||
*/
|
*/
|
||||||
maskblit = MaskBlit.getFromCache(SurfaceType.IntArgbPre,
|
MaskBlit maskblit = MaskBlit.getFromCache(SurfaceType.IntArgbPre,
|
||||||
sg.imageComp,
|
sg.imageComp, dstType);
|
||||||
dstType);
|
|
||||||
|
|
||||||
/* NOTE: We can only use the native TransformHelper
|
/* NOTE: We can only use the native TransformHelper
|
||||||
* func to go directly to the dest if both the helper
|
* func to go directly to the dest if both the helper
|
||||||
@ -496,27 +498,19 @@ public class DrawImage implements DrawImagePipe
|
|||||||
null, 0, 0);
|
null, 0, 0);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
blit = null;
|
|
||||||
} else {
|
|
||||||
/* NOTE: We either have, or we can make,
|
|
||||||
* a Blit for any composite type, even Custom
|
|
||||||
*/
|
|
||||||
maskblit = null;
|
|
||||||
blit = Blit.getFromCache(SurfaceType.IntArgbPre,
|
|
||||||
sg.imageComp,
|
|
||||||
dstType);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// We need to transform to a temp image and then copy
|
// We need to transform to a temp image and then copy
|
||||||
// just the pieces that are valid data to the dest.
|
// just the pieces that are valid data to the dest.
|
||||||
BufferedImage tmpimg = new BufferedImage(dx2-dx1, dy2-dy1,
|
final int w = dx2 - dx1;
|
||||||
|
final int h = dy2 - dy1;
|
||||||
|
BufferedImage tmpimg = new BufferedImage(w, h,
|
||||||
BufferedImage.TYPE_INT_ARGB_PRE);
|
BufferedImage.TYPE_INT_ARGB_PRE);
|
||||||
SurfaceData tmpData = SurfaceData.getPrimarySurfaceData(tmpimg);
|
SurfaceData tmpData = SurfaceData.getPrimarySurfaceData(tmpimg);
|
||||||
SurfaceType tmpType = tmpData.getSurfaceType();
|
SurfaceType tmpType = tmpData.getSurfaceType();
|
||||||
MaskBlit tmpmaskblit =
|
MaskBlit tmpmaskblit = MaskBlit.getFromCache(SurfaceType.IntArgbPre,
|
||||||
MaskBlit.getFromCache(SurfaceType.IntArgbPre,
|
CompositeType.SrcNoEa,
|
||||||
CompositeType.SrcNoEa,
|
tmpType);
|
||||||
tmpType);
|
|
||||||
/*
|
/*
|
||||||
* The helper function fills a temporary edges buffer
|
* The helper function fills a temporary edges buffer
|
||||||
* for us with the bounding coordinates of each scanline
|
* for us with the bounding coordinates of each scanline
|
||||||
@ -531,7 +525,7 @@ public class DrawImage implements DrawImagePipe
|
|||||||
*
|
*
|
||||||
* edges thus has to be h*2+2 in length
|
* edges thus has to be h*2+2 in length
|
||||||
*/
|
*/
|
||||||
int edges[] = new int[(dy2-dy1)*2+2];
|
final int[] edges = new int[h * 2 + 2];
|
||||||
// It is important that edges[0]=edges[1]=0 when we call
|
// It is important that edges[0]=edges[1]=0 when we call
|
||||||
// Transform in case it must return early and we would
|
// Transform in case it must return early and we would
|
||||||
// not want to render anything on an error condition.
|
// not want to render anything on an error condition.
|
||||||
@ -539,35 +533,17 @@ public class DrawImage implements DrawImagePipe
|
|||||||
AlphaComposite.Src, null,
|
AlphaComposite.Src, null,
|
||||||
itx, interpType,
|
itx, interpType,
|
||||||
sx1, sy1, sx2, sy2,
|
sx1, sy1, sx2, sy2,
|
||||||
0, 0, dx2-dx1, dy2-dy1,
|
0, 0, w, h,
|
||||||
edges, dx1, dy1);
|
edges, dx1, dy1);
|
||||||
|
|
||||||
/*
|
final Region region = Region.getInstance(dx1, dy1, dx2, dy2, edges);
|
||||||
* Now copy the results, scanline by scanline, into the dest.
|
clip = clip.getIntersection(region);
|
||||||
* The edges array helps us minimize the work.
|
|
||||||
|
/* NOTE: We either have, or we can make,
|
||||||
|
* a Blit for any composite type, even Custom
|
||||||
*/
|
*/
|
||||||
int index = 2;
|
final Blit blit = Blit.getFromCache(tmpType, sg.imageComp, dstType);
|
||||||
for (int y = edges[0]; y < edges[1]; y++) {
|
blit.Blit(tmpData, dstData, sg.composite, clip, 0, 0, dx1, dy1, w, h);
|
||||||
int relx1 = edges[index++];
|
|
||||||
int relx2 = edges[index++];
|
|
||||||
if (relx1 >= relx2) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (maskblit != null) {
|
|
||||||
maskblit.MaskBlit(tmpData, dstData,
|
|
||||||
sg.composite, clip,
|
|
||||||
relx1, y,
|
|
||||||
dx1+relx1, dy1+y,
|
|
||||||
relx2 - relx1, 1,
|
|
||||||
null, 0, 0);
|
|
||||||
} else {
|
|
||||||
blit.Blit(tmpData, dstData,
|
|
||||||
sg.composite, clip,
|
|
||||||
relx1, y,
|
|
||||||
dx1+relx1, dy1+y,
|
|
||||||
relx2 - relx1, 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Render an image using only integer translation
|
// Render an image using only integer translation
|
||||||
|
@ -30,6 +30,8 @@ import java.awt.Shape;
|
|||||||
import java.awt.geom.AffineTransform;
|
import java.awt.geom.AffineTransform;
|
||||||
import java.awt.geom.RectangularShape;
|
import java.awt.geom.RectangularShape;
|
||||||
|
|
||||||
|
import sun.java2d.loops.TransformHelper;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class encapsulates a definition of a two dimensional region which
|
* This class encapsulates a definition of a two dimensional region which
|
||||||
* consists of a number of Y ranges each containing multiple X bands.
|
* consists of a number of Y ranges each containing multiple X bands.
|
||||||
@ -160,6 +162,15 @@ public class Region {
|
|||||||
this.hiy = hiy;
|
this.hiy = hiy;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Region(int lox, int loy, int hix, int hiy, int[] bands, int end) {
|
||||||
|
this.lox = lox;
|
||||||
|
this.loy = loy;
|
||||||
|
this.hix = hix;
|
||||||
|
this.hiy = hiy;
|
||||||
|
this.bands = bands;
|
||||||
|
this.endIndex = end;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a Region object covering the pixels which would be
|
* Returns a Region object covering the pixels which would be
|
||||||
* touched by a fill or clip operation on a Graphics implementation
|
* touched by a fill or clip operation on a Graphics implementation
|
||||||
@ -255,6 +266,44 @@ public class Region {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a Region object with a rectangle of interest specified by the
|
||||||
|
* indicated rectangular area in lox, loy, hix, hiy and edges array, which
|
||||||
|
* is located relative to the rectangular area. Edges array - 0,1 are y
|
||||||
|
* range, 2N,2N+1 are x ranges, 1 per y range.
|
||||||
|
*
|
||||||
|
* @see TransformHelper
|
||||||
|
*/
|
||||||
|
static Region getInstance(final int lox, final int loy, final int hix,
|
||||||
|
final int hiy, final int[] edges) {
|
||||||
|
final int y1 = edges[0];
|
||||||
|
final int y2 = edges[1];
|
||||||
|
if (hiy <= loy || hix <= lox || y2 <= y1) {
|
||||||
|
return EMPTY_REGION;
|
||||||
|
}
|
||||||
|
// rowsNum * (3 + 1 * 2)
|
||||||
|
final int[] bands = new int[(y2 - y1) * 5];
|
||||||
|
int end = 0;
|
||||||
|
int index = 2;
|
||||||
|
for (int y = y1; y < y2; ++y) {
|
||||||
|
final int spanlox = Math.max(clipAdd(lox, edges[index++]), lox);
|
||||||
|
final int spanhix = Math.min(clipAdd(lox, edges[index++]), hix);
|
||||||
|
if (spanlox < spanhix) {
|
||||||
|
final int spanloy = Math.max(clipAdd(loy, y), loy);
|
||||||
|
final int spanhiy = Math.min(clipAdd(spanloy, 1), hiy);
|
||||||
|
if (spanloy < spanhiy) {
|
||||||
|
bands[end++] = spanloy;
|
||||||
|
bands[end++] = spanhiy;
|
||||||
|
bands[end++] = 1; // 1 span per row
|
||||||
|
bands[end++] = spanlox;
|
||||||
|
bands[end++] = spanhix;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return end != 0 ? new Region(lox, loy, hix, hiy, bands, end)
|
||||||
|
: EMPTY_REGION;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a Region object with a rectangle of interest specified
|
* Returns a Region object with a rectangle of interest specified
|
||||||
* by the indicated Rectangle object.
|
* by the indicated Rectangle object.
|
||||||
|
@ -0,0 +1,138 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2014, 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.AlphaComposite;
|
||||||
|
import java.awt.Color;
|
||||||
|
import java.awt.Graphics2D;
|
||||||
|
import java.awt.GraphicsConfiguration;
|
||||||
|
import java.awt.GraphicsEnvironment;
|
||||||
|
import java.awt.Image;
|
||||||
|
import java.awt.Rectangle;
|
||||||
|
import java.awt.image.BufferedImage;
|
||||||
|
import java.awt.image.DataBuffer;
|
||||||
|
import java.awt.image.DataBufferByte;
|
||||||
|
import java.awt.image.DataBufferInt;
|
||||||
|
import java.awt.image.DataBufferShort;
|
||||||
|
import java.awt.image.VolatileImage;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.imageio.ImageIO;
|
||||||
|
|
||||||
|
import static java.awt.Transparency.TRANSLUCENT;
|
||||||
|
import static java.awt.image.BufferedImage.TYPE_INT_ARGB;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @test
|
||||||
|
* @bug 8059942
|
||||||
|
* @summary Tests rotated clip when unmanaged image is drawn to VI.
|
||||||
|
* Results of the blit to compatibleImage are used for comparison.
|
||||||
|
* @author Sergey Bylokhov
|
||||||
|
*/
|
||||||
|
public final class IncorrectUnmanagedImageRotatedClip {
|
||||||
|
|
||||||
|
public static void main(final String[] args) throws IOException {
|
||||||
|
BufferedImage bi = makeUnmanagedBI();
|
||||||
|
fill(bi);
|
||||||
|
test(bi);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void test(final BufferedImage bi) throws IOException {
|
||||||
|
GraphicsEnvironment ge = GraphicsEnvironment
|
||||||
|
.getLocalGraphicsEnvironment();
|
||||||
|
GraphicsConfiguration gc = ge.getDefaultScreenDevice()
|
||||||
|
.getDefaultConfiguration();
|
||||||
|
VolatileImage vi = gc.createCompatibleVolatileImage(500, 200,
|
||||||
|
TRANSLUCENT);
|
||||||
|
BufferedImage gold = gc.createCompatibleImage(500, 200, TRANSLUCENT);
|
||||||
|
// draw to compatible Image
|
||||||
|
draw(bi, gold);
|
||||||
|
// draw to volatile image
|
||||||
|
int attempt = 0;
|
||||||
|
BufferedImage snapshot;
|
||||||
|
while (true) {
|
||||||
|
if (++attempt > 10) {
|
||||||
|
throw new RuntimeException("Too many attempts: " + attempt);
|
||||||
|
}
|
||||||
|
vi.validate(gc);
|
||||||
|
if (vi.validate(gc) != VolatileImage.IMAGE_OK) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
draw(bi, vi);
|
||||||
|
snapshot = vi.getSnapshot();
|
||||||
|
if (vi.contentsLost()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// validate images
|
||||||
|
for (int x = 0; x < gold.getWidth(); ++x) {
|
||||||
|
for (int y = 0; y < gold.getHeight(); ++y) {
|
||||||
|
if (gold.getRGB(x, y) != snapshot.getRGB(x, y)) {
|
||||||
|
ImageIO.write(gold, "png", new File("gold.png"));
|
||||||
|
ImageIO.write(snapshot, "png", new File("bi.png"));
|
||||||
|
throw new RuntimeException("Test failed.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void draw(final BufferedImage from,final Image to) {
|
||||||
|
final Graphics2D g2d = (Graphics2D) to.getGraphics();
|
||||||
|
g2d.setComposite(AlphaComposite.Src);
|
||||||
|
g2d.setColor(Color.ORANGE);
|
||||||
|
g2d.fillRect(0, 0, to.getWidth(null), to.getHeight(null));
|
||||||
|
g2d.rotate(Math.toRadians(45));
|
||||||
|
g2d.clip(new Rectangle(41, 42, 43, 44));
|
||||||
|
g2d.drawImage(from, 50, 50, Color.blue, null);
|
||||||
|
g2d.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static BufferedImage makeUnmanagedBI() {
|
||||||
|
final BufferedImage bi = new BufferedImage(500, 200, TYPE_INT_ARGB);
|
||||||
|
final DataBuffer db = bi.getRaster().getDataBuffer();
|
||||||
|
if (db instanceof DataBufferInt) {
|
||||||
|
((DataBufferInt) db).getData();
|
||||||
|
} else if (db instanceof DataBufferShort) {
|
||||||
|
((DataBufferShort) db).getData();
|
||||||
|
} else if (db instanceof DataBufferByte) {
|
||||||
|
((DataBufferByte) db).getData();
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
bi.setAccelerationPriority(0.0f);
|
||||||
|
} catch (final Throwable ignored) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return bi;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void fill(final Image image) {
|
||||||
|
final Graphics2D graphics = (Graphics2D) image.getGraphics();
|
||||||
|
graphics.setComposite(AlphaComposite.Src);
|
||||||
|
for (int i = 0; i < image.getHeight(null); ++i) {
|
||||||
|
graphics.setColor(new Color(i, 0, 0));
|
||||||
|
graphics.fillRect(0, i, image.getWidth(null), 1);
|
||||||
|
}
|
||||||
|
graphics.dispose();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user