/*
* Copyright (c) 1999, 2004, 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. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* 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.
*/
package sun.java2d.loops;
import java.awt.Composite;
import java.lang.ref.WeakReference;
import sun.java2d.loops.GraphicsPrimitive;
import sun.java2d.SurfaceData;
import sun.java2d.pipe.Region;
MaskBlit
1) copies rectangle of pixels from one surface to another
2) performs compositing of colors based upon a Composite
parameter
3) blends result of composite with destination using an
alpha coverage mask
4) the mask may be null in which case it should be treated
as if it were an array of all opaque values (0xff)
precise behavior is undefined if the source surface
and the destination surface are the same surface
with overlapping regions of pixels
/**
* MaskBlit
* 1) copies rectangle of pixels from one surface to another
* 2) performs compositing of colors based upon a Composite
* parameter
* 3) blends result of composite with destination using an
* alpha coverage mask
* 4) the mask may be null in which case it should be treated
* as if it were an array of all opaque values (0xff)
*
* precise behavior is undefined if the source surface
* and the destination surface are the same surface
* with overlapping regions of pixels
*/
public class MaskBlit extends GraphicsPrimitive
{
public static final String methodSignature = "MaskBlit(...)".toString();
public static final int primTypeID = makePrimTypeID();
private static RenderCache blitcache = new RenderCache(20);
public static MaskBlit locate(SurfaceType srctype,
CompositeType comptype,
SurfaceType dsttype)
{
return (MaskBlit)
GraphicsPrimitiveMgr.locate(primTypeID,
srctype, comptype, dsttype);
}
public static MaskBlit getFromCache(SurfaceType src,
CompositeType comp,
SurfaceType dst)
{
Object o = blitcache.get(src, comp, dst);
if (o != null) {
return (MaskBlit) o;
}
MaskBlit blit = locate(src, comp, dst);
if (blit == null) {
System.out.println("mask blit loop not found for:");
System.out.println("src: "+src);
System.out.println("comp: "+comp);
System.out.println("dst: "+dst);
} else {
blitcache.put(src, comp, dst, blit);
}
return blit;
}
protected MaskBlit(SurfaceType srctype,
CompositeType comptype,
SurfaceType dsttype)
{
super(methodSignature, primTypeID, srctype, comptype, dsttype);
}
public MaskBlit(long pNativePrim,
SurfaceType srctype,
CompositeType comptype,
SurfaceType dsttype)
{
super(pNativePrim, methodSignature, primTypeID, srctype, comptype, dsttype);
}
All MaskBlit implementors must have this invoker method
/**
* All MaskBlit implementors must have this invoker method
*/
public native void MaskBlit(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
int srcx, int srcy,
int dstx, int dsty,
int width, int height,
byte[] mask, int maskoff, int maskscan);
static {
GraphicsPrimitiveMgr.registerGeneral(new MaskBlit(null, null, null));
}
public GraphicsPrimitive makePrimitive(SurfaceType srctype,
CompositeType comptype,
SurfaceType dsttype)
{
/*
new Throwable().printStackTrace();
System.out.println("Constructing general maskblit for:");
System.out.println("src: "+srctype);
System.out.println("comp: "+comptype);
System.out.println("dst: "+dsttype);
*/
if (CompositeType.Xor.equals(comptype)) {
throw new InternalError("Cannot construct MaskBlit for " +
"XOR mode");
}
General ob = new General(srctype, comptype, dsttype);
setupGeneralBinaryOp(ob);
return ob;
}
private static class General
extends MaskBlit
implements GeneralBinaryOp
{
Blit convertsrc;
Blit convertdst;
MaskBlit performop;
Blit convertresult;
WeakReference srcTmp;
WeakReference dstTmp;
public General(SurfaceType srctype,
CompositeType comptype,
SurfaceType dsttype)
{
super(srctype, comptype, dsttype);
}
public void setPrimitives(Blit srcconverter,
Blit dstconverter,
GraphicsPrimitive genericop,
Blit resconverter)
{
this.convertsrc = srcconverter;
this.convertdst = dstconverter;
this.performop = (MaskBlit) genericop;
this.convertresult = resconverter;
}
public synchronized void MaskBlit(SurfaceData srcData,
SurfaceData dstData,
Composite comp,
Region clip,
int srcx, int srcy,
int dstx, int dsty,
int width, int height,
byte mask[], int offset, int scan)
{
SurfaceData src, dst;
Region opclip;
int sx, sy, dx, dy;
if (convertsrc == null) {
src = srcData;
sx = srcx;
sy = srcy;
} else {
SurfaceData cachedSrc = null;
if (srcTmp != null) {
cachedSrc = (SurfaceData) srcTmp.get();
}
src = convertFrom(convertsrc, srcData, srcx, srcy,
width, height, cachedSrc);
sx = 0;
sy = 0;
if (src != cachedSrc) {
srcTmp = new WeakReference(src);
}
}
if (convertdst == null) {
dst = dstData;
dx = dstx;
dy = dsty;
opclip = clip;
} else {
// assert: convertresult != null
SurfaceData cachedDst = null;
if (dstTmp != null) {
cachedDst = (SurfaceData) dstTmp.get();
}
dst = convertFrom(convertdst, dstData, dstx, dsty,
width, height, cachedDst);
dx = 0;
dy = 0;
opclip = null;
if (dst != cachedDst) {
dstTmp = new WeakReference(dst);
}
}
performop.MaskBlit(src, dst, comp, opclip,
sx, sy, dx, dy, width, height,
mask, offset, scan);
if (convertresult != null) {
// assert: convertdst != null
convertTo(convertresult, dst, dstData, clip,
dstx, dsty, width, height);
}
}
}
public GraphicsPrimitive traceWrap() {
return new TraceMaskBlit(this);
}
private static class TraceMaskBlit extends MaskBlit {
MaskBlit target;
public TraceMaskBlit(MaskBlit target) {
// We need to have the same NativePrim as our
// target in case we are used with a TransformHelper
super(target.getNativePrim(),
target.getSourceType(),
target.getCompositeType(),
target.getDestType());
this.target = target;
}
public GraphicsPrimitive traceWrap() {
return this;
}
public void MaskBlit(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
int srcx, int srcy, int dstx, int dsty,
int width, int height,
byte[] mask, int maskoff, int maskscan)
{
tracePrimitive(target);
target.MaskBlit(src, dst, comp, clip,
srcx, srcy, dstx, dsty, width, height,
mask, maskoff, maskscan);
}
}
}