5.10f) {
+ File f = new File("/etc/release");
+ FileInputStream fis = new FileInputStream(f);
+ InputStreamReader isr
+ = new InputStreamReader(fis, "ISO-8859-1");
+ BufferedReader br = new BufferedReader(isr);
+ String line = br.readLine();
+ if (line.indexOf("OpenSolaris") >= 0) {
+ isOpenSolaris = true;
+ }
+ fis.close();
+ }
+ } catch (Exception e) {
+ }
} else if ("Windows".equals(osName)) {
isWindows = true;
}
@@ -174,11 +193,7 @@ public abstract class SunGraphicsEnvironment extends GraphicsEnvironment
noType1Font = "true".
equals(System.getProperty("sun.java2d.noType1Font"));
- if (isOpenJDK()) {
- String[] fontInfo = FontManager.getDefaultPlatformFont();
- defaultFontName = fontInfo[0];
- defaultFontFileName = fontInfo[1];
- } else {
+ if (!isOpenJDK()) {
defaultFontName = lucidaFontName;
if (useAbsoluteFontFileNames()) {
defaultFontFileName =
@@ -244,6 +259,11 @@ public abstract class SunGraphicsEnvironment extends GraphicsEnvironment
* that might be specified.
*/
fontConfig = createFontConfiguration();
+ if (isOpenJDK()) {
+ String[] fontInfo = FontManager.getDefaultPlatformFont();
+ defaultFontName = fontInfo[0];
+ defaultFontFileName = fontInfo[1];
+ }
getPlatformFontPathFromFontConfig();
String extraFontPath = fontConfig.getExtraFontPath();
@@ -1069,7 +1089,7 @@ public abstract class SunGraphicsEnvironment extends GraphicsEnvironment
String fontFileName =
getFileNameFromPlatformName(platformFontName);
String[] nativeNames = null;
- if (fontFileName == null) {
+ if (fontFileName == null || fontFileName.equals(platformFontName)){
/* No file located, so register using the platform name,
* i.e. as a native font.
*/
@@ -1294,4 +1314,18 @@ public abstract class SunGraphicsEnvironment extends GraphicsEnvironment
/*
* ----END DISPLAY CHANGE SUPPORT----
*/
+
+ /**
+ * Returns true if FlipBufferStrategy with COPIED buffer contents
+ * is preferred for this peer's GraphicsConfiguration over
+ * BlitBufferStrategy, false otherwise.
+ *
+ * The reason FlipBS could be preferred is that in some configurations
+ * an accelerated copy to the screen is supported (like Direct3D 9)
+ *
+ * @return true if flip strategy should be used, false otherwise
+ */
+ public boolean isFlipStrategyPreferred(ComponentPeer peer) {
+ return false;
+ }
}
diff --git a/jdk/src/share/classes/sun/java2d/Surface.java b/jdk/src/share/classes/sun/java2d/Surface.java
new file mode 100644
index 00000000000..7a0fa44c600
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/Surface.java
@@ -0,0 +1,34 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d;
+
+/**
+ * This is a tag interface for a surface.
+ * @see sun.java2d.SurfaceData
+ * @see sun.java2d.pipe.hw.AccelSurface
+ */
+public interface Surface {
+}
diff --git a/jdk/src/share/classes/sun/java2d/SurfaceData.java b/jdk/src/share/classes/sun/java2d/SurfaceData.java
index fa4429dd240..84acfbd8f0f 100644
--- a/jdk/src/share/classes/sun/java2d/SurfaceData.java
+++ b/jdk/src/share/classes/sun/java2d/SurfaceData.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -93,7 +93,7 @@ import sun.awt.image.SurfaceManager;
* retrieved the tracker.
*/
public abstract class SurfaceData
- implements Transparency, DisposerTarget, StateTrackable
+ implements Transparency, DisposerTarget, StateTrackable, Surface
{
private long pData;
private boolean valid;
diff --git a/jdk/src/share/classes/sun/java2d/SurfaceDataProxy.java b/jdk/src/share/classes/sun/java2d/SurfaceDataProxy.java
index 66ee6344491..f94419ca478 100644
--- a/jdk/src/share/classes/sun/java2d/SurfaceDataProxy.java
+++ b/jdk/src/share/classes/sun/java2d/SurfaceDataProxy.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -521,7 +521,7 @@ public abstract class SurfaceDataProxy
CompositeType.SrcNoEa,
dstType);
blitbg.BlitBg(srcData, dstData,
- AlphaComposite.Src, null, bgColor,
+ AlphaComposite.Src, null, bgColor.getRGB(),
0, 0, 0, 0, w, h);
dstData.markDirty();
}
diff --git a/jdk/src/share/classes/sun/java2d/loops/BlitBg.java b/jdk/src/share/classes/sun/java2d/loops/BlitBg.java
index a64e8e81f10..77ae4b364eb 100644
--- a/jdk/src/share/classes/sun/java2d/loops/BlitBg.java
+++ b/jdk/src/share/classes/sun/java2d/loops/BlitBg.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2004 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -109,7 +109,7 @@ public class BlitBg extends GraphicsPrimitive
*/
public native void BlitBg(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
- Color bgColor,
+ int bgColor,
int srcx, int srcy,
int dstx, int dsty,
int width, int height);
@@ -142,19 +142,19 @@ public class BlitBg extends GraphicsPrimitive
compositeType = comptype;
}
+ @Override
public void BlitBg(SurfaceData srcData,
SurfaceData dstData,
Composite comp,
Region clip,
- Color bgColor,
+ int bgArgb,
int srcx, int srcy,
int dstx, int dsty,
int width, int height)
{
ColorModel dstModel = dstData.getColorModel();
- if (!dstModel.hasAlpha() &&
- bgColor.getTransparency() != Transparency.OPAQUE)
- {
+ boolean bgHasAlpha = (bgArgb >>> 24) != 0xff;
+ if (!dstModel.hasAlpha() && bgHasAlpha) {
dstModel = ColorModel.getRGBdefault();
}
WritableRaster wr =
@@ -163,6 +163,7 @@ public class BlitBg extends GraphicsPrimitive
BufferedImage bimg =
new BufferedImage(dstModel, wr, isPremult, null);
SurfaceData tmpData = BufImgSurfaceData.createData(bimg);
+ Color bgColor = new Color(bgArgb, bgHasAlpha);
SunGraphics2D sg2d = new SunGraphics2D(tmpData, bgColor, bgColor,
defaultFont);
FillRect fillop = FillRect.locate(SurfaceType.AnyColor,
@@ -201,9 +202,10 @@ public class BlitBg extends GraphicsPrimitive
return this;
}
+ @Override
public void BlitBg(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
- Color bgColor,
+ int bgColor,
int srcx, int srcy, int dstx, int dsty,
int width, int height)
{
diff --git a/jdk/src/share/classes/sun/java2d/loops/GeneralRenderer.java b/jdk/src/share/classes/sun/java2d/loops/GeneralRenderer.java
index 8c7a4862a62..6a7eade32a4 100644
--- a/jdk/src/share/classes/sun/java2d/loops/GeneralRenderer.java
+++ b/jdk/src/share/classes/sun/java2d/loops/GeneralRenderer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1998-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1998-2008 Sun Microsystems, Inc. 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
@@ -159,6 +159,10 @@ public final class GeneralRenderer {
{
int mx, my, x1, y1;
int[] tmp = null;
+
+ if (nPoints <= 0) {
+ return;
+ }
mx = x1 = xPoints[off] + transx;
my = y1 = yPoints[off] + transy;
while (--nPoints > 0) {
diff --git a/jdk/src/share/classes/sun/java2d/opengl/OGLBufImgOps.java b/jdk/src/share/classes/sun/java2d/opengl/OGLBufImgOps.java
index 2b62bd4736b..1becb763c92 100644
--- a/jdk/src/share/classes/sun/java2d/opengl/OGLBufImgOps.java
+++ b/jdk/src/share/classes/sun/java2d/opengl/OGLBufImgOps.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -35,6 +35,7 @@ import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
import sun.java2d.loops.CompositeType;
import sun.java2d.pipe.BufferedBufImgOps;
+import static sun.java2d.opengl.OGLContext.OGLContextCaps.*;
class OGLBufImgOps extends BufferedBufImgOps {
@@ -94,7 +95,7 @@ class OGLBufImgOps extends BufferedBufImgOps {
OGLSurfaceData oglSrc = (OGLSurfaceData)srcData;
OGLGraphicsConfig gc = oglSrc.getOGLGraphicsConfig();
if (oglSrc.getType() != OGLSurfaceData.TEXTURE ||
- !gc.isCapPresent(OGLContext.CAPS_EXT_BIOP_SHADER))
+ !gc.isCapPresent(CAPS_EXT_BIOP_SHADER))
{
return false;
}
diff --git a/jdk/src/share/classes/sun/java2d/opengl/OGLContext.java b/jdk/src/share/classes/sun/java2d/opengl/OGLContext.java
index 99bff126f92..5ac6afbbc1c 100644
--- a/jdk/src/share/classes/sun/java2d/opengl/OGLContext.java
+++ b/jdk/src/share/classes/sun/java2d/opengl/OGLContext.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -25,91 +25,24 @@
package sun.java2d.opengl;
-import java.awt.AlphaComposite;
-import java.awt.Composite;
-import java.awt.Paint;
-import java.awt.geom.AffineTransform;
-import sun.java2d.SunGraphics2D;
import sun.java2d.pipe.BufferedContext;
-import sun.java2d.pipe.Region;
import sun.java2d.pipe.RenderBuffer;
import sun.java2d.pipe.RenderQueue;
+import sun.java2d.pipe.hw.ContextCapabilities;
import static sun.java2d.pipe.BufferedOpCodes.*;
+import static sun.java2d.pipe.hw.ContextCapabilities.*;
/**
* Note that the RenderQueue lock must be acquired before calling any of
* the methods in this class.
*/
-class OGLContext extends BufferedContext {
+public class OGLContext extends BufferedContext {
- /** Indicates that the context has no capabilities. */
- static final int CAPS_EMPTY = (0 << 0);
- /** Indicates that the context is doublebuffered. */
- static final int CAPS_DOUBLEBUFFERED = (1 << 0);
- /** Indicates that the context supports a stored alpha channel. */
- static final int CAPS_STORED_ALPHA = (1 << 1);
- /** Indicates the presence of the GL_ARB_multitexture extension. */
- static final int CAPS_EXT_MULTITEXTURE = (1 << 2);
- /** Indicates the presence of the GL_ARB_texture_non_power_of_two ext. */
- static final int CAPS_EXT_TEXNONPOW2 = (1 << 3);
- /**
- * Indicates the presence of the GL_EXT_framebuffer_object extension.
- * This cap will only be set if the fbobject system property has been
- * enabled and we are able to create an FBO with depth buffer.
- */
- static final int CAPS_EXT_FBOBJECT = (1 << 4);
- /**
- * Indicates the presence of the GL_ARB_fragment_shader extension.
- * This cap will only be set if the lcdshader system property has been
- * enabled and the hardware supports the minimum number of texture units.
- */
- static final int CAPS_EXT_LCD_SHADER = (1 << 5);
- /** Indicates the presence of the GL_ARB_texture_rectangle extension. */
- static final int CAPS_EXT_TEXRECT = (1 << 6);
- /**
- * Indicates the presence of the GL_ARB_fragment_shader extension.
- * This cap will only be set if the biopshader system property has been
- * enabled and the hardware meets our minimum requirements.
- */
- static final int CAPS_EXT_BIOP_SHADER = (1 << 7);
- /**
- * Indicates the presence of the GL_ARB_fragment_shader extension.
- * This cap will only be set if the gradshader system property has been
- * enabled and the hardware meets our minimum requirements.
- */
- static final int CAPS_EXT_GRAD_SHADER = (1 << 8);
+ private final OGLGraphicsConfig config;
- OGLContext(RenderQueue rq) {
+ OGLContext(RenderQueue rq, OGLGraphicsConfig config) {
super(rq);
- }
-
- /**
- * Fetches the OGLContext associated with the current GraphicsConfig
- * and validates the context using the given parameters. Most rendering
- * operations will call this method first in order to set the necessary
- * state before issuing rendering commands.
- */
- static void validateContext(OGLSurfaceData srcData,
- OGLSurfaceData dstData,
- Region clip, Composite comp,
- AffineTransform xform,
- Paint paint, SunGraphics2D sg2d,
- int flags)
- {
- // assert rq.lock.isHeldByCurrentThread();
- OGLContext oglc = dstData.getContext();
- oglc.validate(srcData, dstData,
- clip, comp, xform, paint, sg2d, flags);
- }
-
- /**
- * Simplified version of validateContext() that disables all context
- * state settings.
- */
- static void validateContext(OGLSurfaceData dstData) {
- // assert rq.lock.isHeldByCurrentThread();
- validateContext(dstData, dstData,
- null, null, null, null, null, NO_CONTEXT_FLAGS);
+ this.config = config;
}
/**
@@ -160,19 +93,128 @@ class OGLContext extends BufferedContext {
static void invalidateCurrentContext() {
// assert OGLRenderQueue.getInstance().lock.isHeldByCurrentThread();
- // first invalidate the context reference at the native level, and
+ // invalidate the current Java-level context so that we
+ // revalidate everything the next time around
+ if (currentContext != null) {
+ currentContext.invalidateContext();
+ currentContext = null;
+ }
+
+ // invalidate the context reference at the native level, and
// then flush the queue so that we have no pending operations
// dependent on the current context
OGLRenderQueue rq = OGLRenderQueue.getInstance();
rq.ensureCapacity(4);
rq.getBuffer().putInt(INVALIDATE_CONTEXT);
rq.flushNow();
+ }
- // then invalidate the current Java-level context so that we
- // revalidate everything the next time around
- if (currentContext != null) {
- currentContext.invalidateSurfaces();
- currentContext = null;
+ public RenderQueue getRenderQueue() {
+ return OGLRenderQueue.getInstance();
+ }
+
+ /**
+ * Returns a string representing adapter id (vendor, renderer, version).
+ * Must be called on the rendering thread.
+ *
+ * @return an id string for the adapter
+ */
+ static final native String getOGLIdString();
+
+ @Override
+ public void saveState() {
+ // assert rq.lock.isHeldByCurrentThread();
+
+ // reset all attributes of this and current contexts
+ invalidateContext();
+ invalidateCurrentContext();
+
+ setScratchSurface(config);
+
+ // save the state on the native level
+ rq.ensureCapacity(4);
+ buf.putInt(SAVE_STATE);
+ rq.flushNow();
+ }
+
+ @Override
+ public void restoreState() {
+ // assert rq.lock.isHeldByCurrentThread();
+
+ // reset all attributes of this and current contexts
+ invalidateContext();
+ invalidateCurrentContext();
+
+ setScratchSurface(config);
+
+ // restore the state on the native level
+ rq.ensureCapacity(4);
+ buf.putInt(RESTORE_STATE);
+ rq.flushNow();
+ }
+
+ static class OGLContextCaps extends ContextCapabilities {
+ /**
+ * Indicates the presence of the GL_EXT_framebuffer_object extension.
+ * This cap will only be set if the fbobject system property has been
+ * enabled and we are able to create an FBO with depth buffer.
+ */
+ static final int CAPS_EXT_FBOBJECT =
+ (CAPS_RT_TEXTURE_ALPHA | CAPS_RT_TEXTURE_OPAQUE);
+ /** Indicates that the context supports a stored alpha channel. */
+ static final int CAPS_STORED_ALPHA = CAPS_RT_PLAIN_ALPHA;
+ /** Indicates that the context is doublebuffered. */
+ static final int CAPS_DOUBLEBUFFERED = (FIRST_PRIVATE_CAP << 0);
+ /**
+ * Indicates the presence of the GL_ARB_fragment_shader extension.
+ * This cap will only be set if the lcdshader system property has been
+ * enabled and the hardware supports the minimum number of texture units
+ */
+ static final int CAPS_EXT_LCD_SHADER = (FIRST_PRIVATE_CAP << 1);
+ /**
+ * Indicates the presence of the GL_ARB_fragment_shader extension.
+ * This cap will only be set if the biopshader system property has been
+ * enabled and the hardware meets our minimum requirements.
+ */
+ static final int CAPS_EXT_BIOP_SHADER = (FIRST_PRIVATE_CAP << 2);
+ /**
+ * Indicates the presence of the GL_ARB_fragment_shader extension.
+ * This cap will only be set if the gradshader system property has been
+ * enabled and the hardware meets our minimum requirements.
+ */
+ static final int CAPS_EXT_GRAD_SHADER = (FIRST_PRIVATE_CAP << 3);
+ /** Indicates the presence of the GL_ARB_texture_rectangle extension. */
+ static final int CAPS_EXT_TEXRECT = (FIRST_PRIVATE_CAP << 4);
+
+ OGLContextCaps(int caps, String adapterId) {
+ super(caps, adapterId);
+ }
+
+ @Override
+ public String toString() {
+ StringBuffer buf = new StringBuffer(super.toString());
+ if ((caps & CAPS_EXT_FBOBJECT) != 0) {
+ buf.append("CAPS_EXT_FBOBJECT|");
+ }
+ if ((caps & CAPS_STORED_ALPHA) != 0) {
+ buf.append("CAPS_STORED_ALPHA|");
+ }
+ if ((caps & CAPS_DOUBLEBUFFERED) != 0) {
+ buf.append("CAPS_DOUBLEBUFFERED|");
+ }
+ if ((caps & CAPS_EXT_LCD_SHADER) != 0) {
+ buf.append("CAPS_EXT_LCD_SHADER|");
+ }
+ if ((caps & CAPS_EXT_BIOP_SHADER) != 0) {
+ buf.append("CAPS_BIOP_SHADER|");
+ }
+ if ((caps & CAPS_EXT_GRAD_SHADER) != 0) {
+ buf.append("CAPS_EXT_GRAD_SHADER|");
+ }
+ if ((caps & CAPS_EXT_TEXRECT) != 0) {
+ buf.append("CAPS_EXT_TEXRECT|");
+ }
+ return buf.toString();
}
}
}
diff --git a/jdk/src/share/classes/sun/java2d/opengl/OGLGraphicsConfig.java b/jdk/src/share/classes/sun/java2d/opengl/OGLGraphicsConfig.java
index 32de60627bd..498ca868cc0 100644
--- a/jdk/src/share/classes/sun/java2d/opengl/OGLGraphicsConfig.java
+++ b/jdk/src/share/classes/sun/java2d/opengl/OGLGraphicsConfig.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -27,13 +27,16 @@ package sun.java2d.opengl;
import sun.java2d.SurfaceData;
import sun.awt.image.SurfaceManager;
+import sun.java2d.pipe.hw.AccelGraphicsConfig;
/**
* This interface collects the methods that are provided by both
* GLXGraphicsConfig and WGLGraphicsConfig, making it easier to invoke these
* methods directly from OGLSurfaceData.
*/
-interface OGLGraphicsConfig extends SurfaceManager.ProxiedGraphicsConfig {
+interface OGLGraphicsConfig extends
+ AccelGraphicsConfig, SurfaceManager.ProxiedGraphicsConfig
+{
OGLContext getContext();
long getNativeConfigInfo();
boolean isCapPresent(int cap);
diff --git a/jdk/src/share/classes/sun/java2d/opengl/OGLPaints.java b/jdk/src/share/classes/sun/java2d/opengl/OGLPaints.java
index 4dc828e998a..47206348230 100644
--- a/jdk/src/share/classes/sun/java2d/opengl/OGLPaints.java
+++ b/jdk/src/share/classes/sun/java2d/opengl/OGLPaints.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -39,6 +39,7 @@ import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
import sun.java2d.loops.CompositeType;
import static sun.java2d.pipe.BufferedPaints.*;
+import static sun.java2d.opengl.OGLContext.OGLContextCaps.*;
abstract class OGLPaints {
@@ -170,7 +171,7 @@ abstract class OGLPaints {
OGLSurfaceData dstData = (OGLSurfaceData)sg2d.surfaceData;
OGLGraphicsConfig gc = dstData.getOGLGraphicsConfig();
- if (!gc.isCapPresent(OGLContext.CAPS_EXT_GRAD_SHADER)) {
+ if (!gc.isCapPresent(CAPS_EXT_GRAD_SHADER)) {
return false;
}
diff --git a/jdk/src/share/classes/sun/java2d/opengl/OGLRenderer.java b/jdk/src/share/classes/sun/java2d/opengl/OGLRenderer.java
index a3afd13e3bf..9fc678797e7 100644
--- a/jdk/src/share/classes/sun/java2d/opengl/OGLRenderer.java
+++ b/jdk/src/share/classes/sun/java2d/opengl/OGLRenderer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -30,6 +30,7 @@ import java.awt.geom.Path2D;
import sun.java2d.SunGraphics2D;
import sun.java2d.loops.GraphicsPrimitive;
import sun.java2d.pipe.BufferedRenderPipe;
+import sun.java2d.pipe.ParallelogramPipe;
import sun.java2d.pipe.RenderQueue;
import sun.java2d.pipe.SpanIterator;
import static sun.java2d.pipe.BufferedOpCodes.*;
@@ -51,6 +52,15 @@ class OGLRenderer extends BufferedRenderPipe {
null, sg2d.paint, sg2d, ctxflags);
}
+ @Override
+ protected void validateContextAA(SunGraphics2D sg2d) {
+ int ctxflags = OGLContext.NO_CONTEXT_FLAGS;
+ OGLSurfaceData dstData = (OGLSurfaceData)sg2d.surfaceData;
+ OGLContext.validateContext(dstData, dstData,
+ sg2d.getCompClip(), sg2d.composite,
+ null, sg2d.paint, sg2d, ctxflags);
+ }
+
void copyArea(SunGraphics2D sg2d,
int x, int y, int w, int h, int dx, int dy)
{
@@ -88,6 +98,31 @@ class OGLRenderer extends BufferedRenderPipe {
super(oglr.rq);
this.oglr = oglr;
}
+ public ParallelogramPipe getAAParallelogramPipe() {
+ final ParallelogramPipe realpipe = oglr.getAAParallelogramPipe();
+ return new ParallelogramPipe() {
+ public void fillParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ GraphicsPrimitive.tracePrimitive("OGLFillAAParallelogram");
+ realpipe.fillParallelogram(sg2d,
+ x, y, dx1, dy1, dx2, dy2);
+ }
+ public void drawParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2)
+ {
+ GraphicsPrimitive.tracePrimitive("OGLDrawAAParallelogram");
+ realpipe.drawParallelogram(sg2d,
+ x, y, dx1, dy1, dx2, dy2,
+ lw1, lw2);
+ }
+ };
+ }
protected void validateContext(SunGraphics2D sg2d) {
oglr.validateContext(sg2d);
}
@@ -130,6 +165,23 @@ class OGLRenderer extends BufferedRenderPipe {
GraphicsPrimitive.tracePrimitive("OGLFillSpans");
oglr.fillSpans(sg2d, si, transx, transy);
}
+ public void fillParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ GraphicsPrimitive.tracePrimitive("OGLFillParallelogram");
+ oglr.fillParallelogram(sg2d, x, y, dx1, dy1, dx2, dy2);
+ }
+ public void drawParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2)
+ {
+ GraphicsPrimitive.tracePrimitive("OGLDrawParallelogram");
+ oglr.drawParallelogram(sg2d, x, y, dx1, dy1, dx2, dy2, lw1, lw2);
+ }
public void copyArea(SunGraphics2D sg2d,
int x, int y, int w, int h, int dx, int dy)
{
diff --git a/jdk/src/share/classes/sun/java2d/opengl/OGLSurfaceData.java b/jdk/src/share/classes/sun/java2d/opengl/OGLSurfaceData.java
index a47918700ce..78f16f7dcca 100644
--- a/jdk/src/share/classes/sun/java2d/opengl/OGLSurfaceData.java
+++ b/jdk/src/share/classes/sun/java2d/opengl/OGLSurfaceData.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -27,23 +27,26 @@ package sun.java2d.opengl;
import java.awt.AlphaComposite;
import java.awt.GraphicsEnvironment;
+import java.awt.Rectangle;
import java.awt.Transparency;
import java.awt.image.ColorModel;
import java.awt.image.Raster;
import sun.awt.SunHints;
import sun.awt.image.PixelConverter;
+import sun.java2d.pipe.hw.AccelSurface;
import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
import sun.java2d.SurfaceDataProxy;
+import sun.java2d.loops.CompositeType;
import sun.java2d.loops.GraphicsPrimitive;
import sun.java2d.loops.MaskFill;
import sun.java2d.loops.SurfaceType;
-import sun.java2d.pipe.PixelToShapeConverter;
+import sun.java2d.pipe.ParallelogramPipe;
+import sun.java2d.pipe.PixelToParallelogramConverter;
import sun.java2d.pipe.RenderBuffer;
-import sun.java2d.pipe.RenderQueue;
import sun.java2d.pipe.TextPipe;
import static sun.java2d.pipe.BufferedOpCodes.*;
-import static sun.java2d.opengl.OGLContext.*;
+import static sun.java2d.opengl.OGLContext.OGLContextCaps.*;
/**
* This class describes an OpenGL "surface", that is, a region of pixels
@@ -92,17 +95,16 @@ import static sun.java2d.opengl.OGLContext.*;
* FLIP_BACKBUFFER OpenGLSurface
* FBOBJECT OpenGLSurfaceRTT
*/
-public abstract class OGLSurfaceData extends SurfaceData {
+public abstract class OGLSurfaceData extends SurfaceData
+ implements AccelSurface {
/**
* OGL-specific surface types
+ *
+ * @see sun.java2d.pipe.hw.AccelSurface
*/
- public static final int UNDEFINED = 0;
- public static final int WINDOW = 1;
- public static final int PBUFFER = 2;
- public static final int TEXTURE = 3;
- public static final int FLIP_BACKBUFFER = 4;
- public static final int FBOBJECT = 5;
+ public static final int PBUFFER = RT_PLAIN;
+ public static final int FBOBJECT = RT_TEXTURE;
/**
* Pixel formats
@@ -148,11 +150,14 @@ public abstract class OGLSurfaceData extends SurfaceData {
private static boolean isGradShaderEnabled;
private OGLGraphicsConfig graphicsConfig;
- private int textureTarget;
protected int type;
+ // these fields are set from the native code when the surface is
+ // initialized
+ private int nativeWidth, nativeHeight;
protected static OGLRenderer oglRenderPipe;
- protected static PixelToShapeConverter oglTxRenderPipe;
+ protected static PixelToParallelogramConverter oglTxRenderPipe;
+ protected static ParallelogramPipe oglAAPgramPipe;
protected static OGLTextRenderer oglTextPipe;
protected static OGLDrawImage oglImagePipe;
@@ -170,6 +175,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
int width, int height);
private native int getTextureTarget(long pData);
+ private native int getTextureID(long pData);
static {
if (!GraphicsEnvironment.isHeadless()) {
@@ -203,9 +209,14 @@ public abstract class OGLSurfaceData extends SurfaceData {
oglRenderPipe = new OGLRenderer(rq);
if (GraphicsPrimitive.tracingEnabled()) {
oglTextPipe = oglTextPipe.traceWrap();
- oglRenderPipe = oglRenderPipe.traceWrap();
+ //The wrapped oglRenderPipe will wrap the AA pipe as well...
+ //oglAAPgramPipe = oglRenderPipe.traceWrap();
}
- oglTxRenderPipe = new PixelToShapeConverter(oglRenderPipe);
+ oglAAPgramPipe = oglRenderPipe.getAAParallelogramPipe();
+ oglTxRenderPipe =
+ new PixelToParallelogramConverter(oglRenderPipe,
+ oglRenderPipe,
+ 1.0, 0.25, true);
OGLBlitLoops.register();
OGLMaskFill.register();
@@ -282,9 +293,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
break;
}
- if (success) {
- textureTarget = getTextureTarget(getNativeOps());
- } else {
+ if (!success) {
throw new OutOfMemoryError("can't create offscreen surface");
}
}
@@ -323,7 +332,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
* Returns the OGLContext for the GraphicsConfig associated with this
* surface.
*/
- final OGLContext getContext() {
+ public final OGLContext getContext() {
return graphicsConfig.getContext();
}
@@ -337,7 +346,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
/**
* Returns one of the surface type constants defined above.
*/
- final int getType() {
+ public final int getType() {
return type;
}
@@ -346,8 +355,41 @@ public abstract class OGLSurfaceData extends SurfaceData {
* for that texture (either GL_TEXTURE_2D or GL_TEXTURE_RECTANGLE_ARB).
* Otherwise, this method will return zero.
*/
- final int getTextureTarget() {
- return textureTarget;
+ public final int getTextureTarget() {
+ return getTextureTarget(getNativeOps());
+ }
+
+ /**
+ * If this surface is backed by a texture object, returns the texture ID
+ * for that texture.
+ * Otherwise, this method will return zero.
+ */
+ public final int getTextureID() {
+ return getTextureID(getNativeOps());
+ }
+
+ /**
+ * Returns native resource of specified {@code resType} associated with
+ * this surface.
+ *
+ * Specifically, for {@code OGLSurfaceData} this method returns the
+ * the following:
+ *
+ * TEXTURE - texture id
+ *
+ *
+ * Note: the resource returned by this method is only valid on the rendering
+ * thread.
+ *
+ * @return native resource of specified type or 0L if
+ * such resource doesn't exist or can not be retrieved.
+ * @see sun.java2d.pipe.hw.AccelSurface#getNativeResource
+ */
+ public long getNativeResource(int resType) {
+ if (resType == TEXTURE) {
+ return getTextureID();
+ }
+ return 0L;
}
public Raster getRaster(int x, int y, int w, int h) {
@@ -366,7 +408,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
*/
public boolean canRenderLCDText(SunGraphics2D sg2d) {
return
- graphicsConfig.isCapPresent(OGLContext.CAPS_EXT_LCD_SHADER) &&
+ graphicsConfig.isCapPresent(CAPS_EXT_LCD_SHADER) &&
sg2d.compositeState <= SunGraphics2D.COMP_ISCOPY &&
sg2d.paintState <= SunGraphics2D.PAINT_OPAQUECOLOR;
}
@@ -405,7 +447,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
validated = true;
}
- PixelToShapeConverter txPipe = null;
+ PixelToParallelogramConverter txPipe = null;
OGLRenderer nonTxPipe = null;
if (sg2d.antialiasHint != SunHints.INTVAL_ANTIALIAS_ON) {
@@ -422,12 +464,28 @@ public abstract class OGLSurfaceData extends SurfaceData {
// custom paints handled by super.validatePipe() below
}
} else {
- if (sg2d.paintState <= sg2d.PAINT_ALPHACOLOR &&
- sg2d.compositeState == sg2d.COMP_XOR)
- {
- // install the solid pipes when AA and XOR are both enabled
- txPipe = oglTxRenderPipe;
- nonTxPipe = oglRenderPipe;
+ if (sg2d.paintState <= sg2d.PAINT_ALPHACOLOR) {
+ if (graphicsConfig.isCapPresent(CAPS_PS30) &&
+ (sg2d.imageComp == CompositeType.SrcOverNoEa ||
+ sg2d.imageComp == CompositeType.SrcOver))
+ {
+ if (!validated) {
+ super.validatePipe(sg2d);
+ validated = true;
+ }
+ PixelToParallelogramConverter aaConverter =
+ new PixelToParallelogramConverter(sg2d.shapepipe,
+ oglAAPgramPipe,
+ 1.0/8.0, 0.499,
+ false);
+ sg2d.drawpipe = aaConverter;
+ sg2d.fillpipe = aaConverter;
+ sg2d.shapepipe = aaConverter;
+ } else if (sg2d.compositeState == sg2d.COMP_XOR) {
+ // install the solid pipes when AA and XOR are both enabled
+ txPipe = oglTxRenderPipe;
+ nonTxPipe = oglRenderPipe;
+ }
}
// other cases handled by super.validatePipe() below
}
@@ -443,7 +501,11 @@ public abstract class OGLSurfaceData extends SurfaceData {
sg2d.drawpipe = nonTxPipe;
sg2d.fillpipe = nonTxPipe;
}
- sg2d.shapepipe = nonTxPipe;
+ // Note that we use the transforming pipe here because it
+ // will examine the shape and possibly perform an optimized
+ // operation if it can be simplified. The simplifications
+ // will be valid for all STROKE and TRANSFORM types.
+ sg2d.shapepipe = txPipe;
} else {
if (!validated) {
super.validatePipe(sg2d);
@@ -472,7 +534,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
* validation code will choose a more general software-based loop.
*/
if (!OGLPaints.isValid(sg2d) ||
- !graphicsConfig.isCapPresent(CAPS_EXT_MULTITEXTURE))
+ !graphicsConfig.isCapPresent(CAPS_MULTITEXTURE))
{
return null;
}
@@ -564,7 +626,7 @@ public abstract class OGLSurfaceData extends SurfaceData {
* when using the basic GL_TEXTURE_2D target.
*/
boolean isTexNonPow2Available() {
- return graphicsConfig.isCapPresent(OGLContext.CAPS_EXT_TEXNONPOW2);
+ return graphicsConfig.isCapPresent(CAPS_TEXNONPOW2);
}
/**
@@ -573,6 +635,16 @@ public abstract class OGLSurfaceData extends SurfaceData {
* GL_ARB_texture_rectangle extension is present).
*/
boolean isTexRectAvailable() {
- return graphicsConfig.isCapPresent(OGLContext.CAPS_EXT_TEXRECT);
+ return graphicsConfig.isCapPresent(CAPS_EXT_TEXRECT);
+ }
+
+ public Rectangle getNativeBounds() {
+ OGLRenderQueue rq = OGLRenderQueue.getInstance();
+ rq.lock();
+ try {
+ return new Rectangle(nativeWidth, nativeHeight);
+ } finally {
+ rq.unlock();
+ }
}
}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/BufferedContext.java b/jdk/src/share/classes/sun/java2d/pipe/BufferedContext.java
index d14b4277d4d..05989523083 100644
--- a/jdk/src/share/classes/sun/java2d/pipe/BufferedContext.java
+++ b/jdk/src/share/classes/sun/java2d/pipe/BufferedContext.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -26,12 +26,13 @@
package sun.java2d.pipe;
import java.awt.AlphaComposite;
+import java.awt.Color;
import java.awt.Composite;
import java.awt.Paint;
import java.awt.geom.AffineTransform;
+import sun.java2d.pipe.hw.AccelSurface;
import sun.java2d.InvalidPipeException;
import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
import sun.java2d.loops.XORComposite;
import static sun.java2d.pipe.BufferedOpCodes.*;
import static sun.java2d.pipe.BufferedRenderPipe.BYTES_PER_SPAN;
@@ -43,8 +44,10 @@ import static sun.java2d.pipe.BufferedRenderPipe.BYTES_PER_SPAN;
* single thread. Note that the RenderQueue lock must be acquired before
* calling the validate() method (or any other method in this class). See
* the RenderQueue class comments for a sample usage scenario.
+ *
+ * @see RenderQueue
*/
-public class BufferedContext {
+public abstract class BufferedContext {
/*
* The following flags help the internals of validate() determine
@@ -82,19 +85,66 @@ public class BufferedContext {
*/
protected static BufferedContext currentContext;
- private SurfaceData validatedSrcData;
- private SurfaceData validatedDstData;
+ private AccelSurface validatedSrcData;
+ private AccelSurface validatedDstData;
private Region validatedClip;
private Composite validatedComp;
private Paint validatedPaint;
+ private boolean isValidatedPaintAColor;
+ private int validatedRGB;
private int validatedFlags;
private boolean xformInUse;
+ private int transX;
+ private int transY;
protected BufferedContext(RenderQueue rq) {
this.rq = rq;
this.buf = rq.getBuffer();
}
+ /**
+ * Fetches the BufferedContextContext associated with the dst. surface
+ * and validates the context using the given parameters. Most rendering
+ * operations will call this method first in order to set the necessary
+ * state before issuing rendering commands.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @throws InvalidPipeException if either src or dest surface is not valid
+ * or lost
+ * @see RenderQueue#lock
+ * @see RenderQueue#unlock
+ */
+ public static void validateContext(AccelSurface srcData,
+ AccelSurface dstData,
+ Region clip, Composite comp,
+ AffineTransform xform,
+ Paint paint, SunGraphics2D sg2d,
+ int flags)
+ {
+ // assert rq.lock.isHeldByCurrentThread();
+ BufferedContext d3dc = dstData.getContext();
+ d3dc.validate(srcData, dstData,
+ clip, comp, xform, paint, sg2d, flags);
+ }
+
+ /**
+ * Fetches the BufferedContextassociated with the surface
+ * and disables all context state settings.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @throws InvalidPipeException if the surface is not valid
+ * or lost
+ * @see RenderQueue#lock
+ * @see RenderQueue#unlock
+ */
+ public static void validateContext(AccelSurface surface) {
+ // assert rt.lock.isHeldByCurrentThread();
+ validateContext(surface, surface,
+ null, null, null, null, null, NO_CONTEXT_FLAGS);
+ }
+
/**
* Validates the given parameters against the current state for this
* context. If this context is not current, it will be made current
@@ -106,19 +156,47 @@ public class BufferedContext {
* Note that the SunGraphics2D parameter is only used for the purposes
* of validating a (non-null) Paint parameter. In all other cases it
* is safe to pass a null SunGraphics2D and it will be ignored.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @throws InvalidPipeException if either src or dest surface is not valid
+ * or lost
*/
- public void validate(SurfaceData srcData, SurfaceData dstData,
+ public void validate(AccelSurface srcData, AccelSurface dstData,
Region clip, Composite comp,
AffineTransform xform,
Paint paint, SunGraphics2D sg2d, int flags)
{
// assert rq.lock.isHeldByCurrentThread();
- boolean updateClip = (clip != validatedClip);
- boolean updatePaint = (paint != validatedPaint);
+ boolean updateClip = false;
+ boolean updatePaint = false;
- if (!dstData.isValid()) {
- throw new InvalidPipeException("bounds changed");
+ if (!dstData.isValid() ||
+ dstData.isSurfaceLost() || srcData.isSurfaceLost())
+ {
+ invalidateContext();
+ throw new InvalidPipeException("bounds changed or surface lost");
+ }
+
+ if (paint instanceof Color) {
+ // REMIND: not 30-bit friendly
+ int newRGB = ((Color)paint).getRGB();
+ if (isValidatedPaintAColor) {
+ if (newRGB != validatedRGB) {
+ validatedRGB = newRGB;
+ updatePaint = true;
+ }
+ } else {
+ validatedRGB = newRGB;
+ updatePaint = true;
+ isValidatedPaintAColor = true;
+ }
+ } else if (validatedPaint != paint) {
+ updatePaint = true;
+ // this should be set when we are switching from paint to color
+ // in which case this condition will be true
+ isValidatedPaintAColor = false;
}
if ((currentContext != this) ||
@@ -147,9 +225,18 @@ public class BufferedContext {
}
// validate clip
- if (updateClip) {
+ if ((clip != validatedClip) || updateClip) {
if (clip != null) {
- setClip(clip);
+ if (updateClip ||
+ validatedClip == null ||
+ !(validatedClip.isRectangular() && clip.isRectangular()) ||
+ ((clip.getLoX() != validatedClip.getLoX() ||
+ clip.getLoY() != validatedClip.getLoY() ||
+ clip.getHiX() != validatedClip.getHiX() ||
+ clip.getHiY() != validatedClip.getHiY())))
+ {
+ setClip(clip);
+ }
} else {
resetClip();
}
@@ -173,14 +260,29 @@ public class BufferedContext {
}
// validate transform
+ boolean txChanged = false;
if (xform == null) {
if (xformInUse) {
resetTransform();
xformInUse = false;
+ txChanged = true;
+ } else if (sg2d != null) {
+ if (transX != sg2d.transX || transY != sg2d.transY) {
+ txChanged = true;
+ }
+ }
+ if (sg2d != null) {
+ transX = sg2d.transX;
+ transY = sg2d.transY;
}
} else {
setTransform(xform);
xformInUse = true;
+ txChanged = true;
+ }
+ // non-Color paints may require paint revalidation
+ if (!isValidatedPaintAColor && txChanged) {
+ updatePaint = true;
}
// validate paint
@@ -194,6 +296,7 @@ public class BufferedContext {
}
// mark dstData dirty
+ // REMIND: is this really needed now? we do it in SunGraphics2D..
dstData.markDirty();
}
@@ -201,13 +304,20 @@ public class BufferedContext {
* Invalidates the surfaces associated with this context. This is
* useful when the context is no longer needed, and we want to break
* the chain caused by these surface references.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @see RenderQueue#lock
+ * @see RenderQueue#unlock
*/
public void invalidateSurfaces() {
validatedSrcData = null;
validatedDstData = null;
}
- private void setSurfaces(SurfaceData srcData, SurfaceData dstData) {
+ private void setSurfaces(AccelSurface srcData,
+ AccelSurface dstData)
+ {
// assert rq.lock.isHeldByCurrentThread();
rq.ensureCapacityAndAlignment(20, 4);
buf.putInt(SET_SURFACES);
@@ -304,4 +414,54 @@ public class BufferedContext {
buf.putDouble(xform.getTranslateX());
buf.putDouble(xform.getTranslateY());
}
+
+ /**
+ * Resets this context's surfaces and all attributes.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @see RenderQueue#lock
+ * @see RenderQueue#unlock
+ */
+ public void invalidateContext() {
+ resetTransform();
+ resetComposite();
+ resetClip();
+ invalidateSurfaces();
+ validatedComp = null;
+ validatedClip = null;
+ validatedPaint = null;
+ xformInUse = false;
+ }
+
+ /**
+ * Returns a singleton {@code RenderQueue} object used by the rendering
+ * pipeline.
+ *
+ * @return a render queue
+ * @see RenderQueue
+ */
+ public abstract RenderQueue getRenderQueue();
+
+ /**
+ * Saves the the state of this context.
+ * It may reset the current context.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @see RenderQueue#lock
+ * @see RenderQueue#unlock
+ */
+ public abstract void saveState();
+
+ /**
+ * Restores the native state of this context.
+ * It may reset the current context.
+ *
+ * Note: must be called while the RenderQueue lock is held.
+ *
+ * @see RenderQueue#lock
+ * @see RenderQueue#unlock
+ */
+ public abstract void restoreState();
}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/BufferedOpCodes.java b/jdk/src/share/classes/sun/java2d/pipe/BufferedOpCodes.java
index 0523d27bd76..44de9603a84 100644
--- a/jdk/src/share/classes/sun/java2d/pipe/BufferedOpCodes.java
+++ b/jdk/src/share/classes/sun/java2d/pipe/BufferedOpCodes.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -32,10 +32,14 @@ public class BufferedOpCodes {
public static final int DRAW_POLY = 12;
public static final int DRAW_PIXEL = 13;
public static final int DRAW_SCANLINES = 14;
+ public static final int DRAW_PARALLELOGRAM = 15;
+ public static final int DRAW_AAPARALLELOGRAM = 16;
// fill ops
public static final int FILL_RECT = 20;
public static final int FILL_SPANS = 21;
+ public static final int FILL_PARALLELOGRAM = 22;
+ public static final int FILL_AAPARALLELOGRAM = 23;
// copy-related ops
public static final int COPY_AREA = 30;
@@ -67,6 +71,9 @@ public class BufferedOpCodes {
public static final int DISPOSE_CONFIG = 74;
public static final int INVALIDATE_CONTEXT = 75;
public static final int SYNC = 76;
+ public static final int RESTORE_DEVICES = 77;
+ public static final int SAVE_STATE = 78;
+ public static final int RESTORE_STATE = 79;
// multibuffering ops
public static final int SWAP_BUFFERS = 80;
diff --git a/jdk/src/share/classes/sun/java2d/pipe/BufferedRenderPipe.java b/jdk/src/share/classes/sun/java2d/pipe/BufferedRenderPipe.java
index 71feb4b9ba4..3961edc69e0 100644
--- a/jdk/src/share/classes/sun/java2d/pipe/BufferedRenderPipe.java
+++ b/jdk/src/share/classes/sun/java2d/pipe/BufferedRenderPipe.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -25,6 +25,7 @@
package sun.java2d.pipe;
+import java.awt.BasicStroke;
import java.awt.Polygon;
import java.awt.Shape;
import java.awt.geom.AffineTransform;
@@ -33,6 +34,7 @@ import java.awt.geom.Ellipse2D;
import java.awt.geom.Path2D;
import java.awt.geom.IllegalPathStateException;
import java.awt.geom.PathIterator;
+import java.awt.geom.Rectangle2D;
import java.awt.geom.RoundRectangle2D;
import sun.java2d.SunGraphics2D;
import sun.java2d.loops.ProcessPath;
@@ -51,8 +53,10 @@ import static sun.java2d.pipe.BufferedOpCodes.*;
* simply delegate to draw(Shape) and fill(Shape), respectively.
*/
public abstract class BufferedRenderPipe
- implements PixelDrawPipe, PixelFillPipe, ShapeDrawPipe
+ implements PixelDrawPipe, PixelFillPipe, ShapeDrawPipe, ParallelogramPipe
{
+ ParallelogramPipe aapgrampipe = new AAParallelogramPipe();
+
static final int BYTES_PER_POLY_POINT = 8;
static final int BYTES_PER_SCANLINE = 12;
static final int BYTES_PER_SPAN = 16;
@@ -67,12 +71,17 @@ public abstract class BufferedRenderPipe
this.drawHandler = new BufferedDrawHandler();
}
+ public ParallelogramPipe getAAParallelogramPipe() {
+ return aapgrampipe;
+ }
+
/**
* Validates the state in the provided SunGraphics2D object and sets up
* any special resources for this operation (e.g. enabling gradient
* shading).
*/
protected abstract void validateContext(SunGraphics2D sg2d);
+ protected abstract void validateContextAA(SunGraphics2D sg2d);
public void drawLine(SunGraphics2D sg2d,
int x1, int y1, int x2, int y2)
@@ -398,6 +407,98 @@ public abstract class BufferedRenderPipe
}
}
+ public void fillParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ rq.lock();
+ try {
+ validateContext(sg2d);
+ rq.ensureCapacity(28);
+ buf.putInt(FILL_PARALLELOGRAM);
+ buf.putFloat((float) x);
+ buf.putFloat((float) y);
+ buf.putFloat((float) dx1);
+ buf.putFloat((float) dy1);
+ buf.putFloat((float) dx2);
+ buf.putFloat((float) dy2);
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ public void drawParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2)
+ {
+ rq.lock();
+ try {
+ validateContext(sg2d);
+ rq.ensureCapacity(36);
+ buf.putInt(DRAW_PARALLELOGRAM);
+ buf.putFloat((float) x);
+ buf.putFloat((float) y);
+ buf.putFloat((float) dx1);
+ buf.putFloat((float) dy1);
+ buf.putFloat((float) dx2);
+ buf.putFloat((float) dy2);
+ buf.putFloat((float) lw1);
+ buf.putFloat((float) lw2);
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ private class AAParallelogramPipe implements ParallelogramPipe {
+ public void fillParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ rq.lock();
+ try {
+ validateContextAA(sg2d);
+ rq.ensureCapacity(28);
+ buf.putInt(FILL_AAPARALLELOGRAM);
+ buf.putFloat((float) x);
+ buf.putFloat((float) y);
+ buf.putFloat((float) dx1);
+ buf.putFloat((float) dy1);
+ buf.putFloat((float) dx2);
+ buf.putFloat((float) dy2);
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ public void drawParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2)
+ {
+ rq.lock();
+ try {
+ validateContextAA(sg2d);
+ rq.ensureCapacity(36);
+ buf.putInt(DRAW_AAPARALLELOGRAM);
+ buf.putFloat((float) x);
+ buf.putFloat((float) y);
+ buf.putFloat((float) dx1);
+ buf.putFloat((float) dy1);
+ buf.putFloat((float) dx2);
+ buf.putFloat((float) dy2);
+ buf.putFloat((float) lw1);
+ buf.putFloat((float) lw2);
+ } finally {
+ rq.unlock();
+ }
+ }
+ }
+
public void draw(SunGraphics2D sg2d, Shape s) {
if (sg2d.strokeState == sg2d.STROKE_THIN) {
if (s instanceof Polygon) {
diff --git a/jdk/src/share/classes/sun/java2d/pipe/DrawImage.java b/jdk/src/share/classes/sun/java2d/pipe/DrawImage.java
index 21397906da3..43b17ac6be3 100644
--- a/jdk/src/share/classes/sun/java2d/pipe/DrawImage.java
+++ b/jdk/src/share/classes/sun/java2d/pipe/DrawImage.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2001-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2001-2008 Sun Microsystems, Inc. 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
@@ -955,7 +955,7 @@ public class DrawImage implements DrawImagePipe
} else {
BlitBg blit = BlitBg.getFromCache(srcType, comp, dstType);
blit.BlitBg(srcData, dstData, sg.composite, clipRegion,
- bgColor, sx, sy, dx, dy, w, h);
+ bgColor.getRGB(), sx, sy, dx, dy, w, h);
}
}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/ParallelogramPipe.java b/jdk/src/share/classes/sun/java2d/pipe/ParallelogramPipe.java
new file mode 100644
index 00000000000..33278dd904f
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/ParallelogramPipe.java
@@ -0,0 +1,66 @@
+/*
+ * Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe;
+
+import sun.java2d.SunGraphics2D;
+
+/**
+ * This interface defines the set of calls that pipeline objects
+ * can use to pass on responsibility for drawing arbitrary
+ * parallelogram shapes.
+ * Six floating point numbers are provided and the parallelogram
+ * is defined as the quadrilateral with the following vertices:
+ *
+ * origin: (x, y)
+ * => (x+dx1, y+dy1)
+ * => (x+dx1+dx2, y+dy1+dy2)
+ * => (x+dx2, y+dy2)
+ * => origin
+ *
+ */
+public interface ParallelogramPipe {
+ public void fillParallelogram(SunGraphics2D sg,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2);
+
+ /**
+ * Draw a Parallelogram with the indicated line widths
+ * assuming a standard BasicStroke with MITER joins.
+ * lw1 specifies the width of the stroke along the dx1,dy1
+ * vector and lw2 specifies the width of the stroke along
+ * the dx2,dy2 vector.
+ * This is equivalent to outsetting the indicated
+ * parallelogram by lw/2 pixels, then insetting the
+ * same parallelogram by lw/2 pixels and filling the
+ * difference between the outer and inner parallelograms.
+ */
+ public void drawParallelogram(SunGraphics2D sg,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2);
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/PixelToParallelogramConverter.java b/jdk/src/share/classes/sun/java2d/pipe/PixelToParallelogramConverter.java
new file mode 100644
index 00000000000..dcbebb2fbf3
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/PixelToParallelogramConverter.java
@@ -0,0 +1,417 @@
+/*
+ * Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe;
+
+import java.awt.Shape;
+import java.awt.BasicStroke;
+import java.awt.geom.Line2D;
+import java.awt.geom.Rectangle2D;
+import java.awt.geom.AffineTransform;
+import sun.java2d.SunGraphics2D;
+import sun.awt.SunHints;
+
+/**
+ * This class converts calls to the basic pixel rendering methods
+ * into calls to the methods on a ParallelogramPipe.
+ * Most calls are transformed into calls to the fill(Shape) method
+ * by the parent PixelToShapeConverter class, but some calls are
+ * transformed into calls to fill/drawParallelogram().
+ */
+public class PixelToParallelogramConverter extends PixelToShapeConverter
+ implements ShapeDrawPipe
+{
+ ParallelogramPipe outrenderer;
+ double minPenSize;
+ double normPosition;
+ double normRoundingBias;
+ boolean adjustfill;
+
+ /**
+ * @param shapepipe pipeline to forward shape calls to
+ * @param pgrampipe pipeline to forward parallelogram calls to
+ * (and drawLine calls if possible)
+ * @param minPenSize minimum pen size for dropout control
+ * @param normPosition sub-pixel location to normalize endpoints
+ * for STROKE_NORMALIZE cases
+ * @param adjustFill boolean to control whethere normalization
+ * constants are also applied to fill operations
+ * (normally true for non-AA, false for AA)
+ */
+ public PixelToParallelogramConverter(ShapeDrawPipe shapepipe,
+ ParallelogramPipe pgrampipe,
+ double minPenSize,
+ double normPosition,
+ boolean adjustfill)
+ {
+ super(shapepipe);
+ outrenderer = pgrampipe;
+ this.minPenSize = minPenSize;
+ this.normPosition = normPosition;
+ this.normRoundingBias = 0.5 - normPosition;
+ this.adjustfill = adjustfill;
+ }
+
+ public void drawLine(SunGraphics2D sg2d,
+ int x1, int y1, int x2, int y2)
+ {
+ if (!drawGeneralLine(sg2d, x1, y1, x2, y2)) {
+ super.drawLine(sg2d, x1, y1, x2, y2);
+ }
+ }
+
+ public void drawRect(SunGraphics2D sg2d,
+ int x, int y, int w, int h)
+ {
+ if (w >= 0 && h >= 0) {
+ if (sg2d.strokeState < SunGraphics2D.STROKE_CUSTOM) {
+ BasicStroke bs = ((BasicStroke) sg2d.stroke);
+ if (w > 0 && h > 0) {
+ if (bs.getLineJoin() == BasicStroke.JOIN_MITER &&
+ bs.getDashArray() == null)
+ {
+ double lw = bs.getLineWidth();
+ drawRectangle(sg2d, x, y, w, h, lw);
+ return;
+ }
+ } else {
+ // Note: This calls the integer version which
+ // will verify that the local drawLine optimizations
+ // work and call super.drawLine(), if not.
+ drawLine(sg2d, x, y, x+w, y+h);
+ return;
+ }
+ }
+ super.drawRect(sg2d, x, y, w, h);
+ }
+ }
+
+ public void fillRect(SunGraphics2D sg2d,
+ int x, int y, int w, int h)
+ {
+ if (w > 0 && h > 0) {
+ fillRectangle(sg2d, x, y, w, h);
+ }
+ }
+
+ public void draw(SunGraphics2D sg2d, Shape s) {
+ if (sg2d.strokeState < SunGraphics2D.STROKE_CUSTOM) {
+ BasicStroke bs = ((BasicStroke) sg2d.stroke);
+ if (s instanceof Rectangle2D) {
+ if (bs.getLineJoin() == BasicStroke.JOIN_MITER &&
+ bs.getDashArray() == null)
+ {
+ Rectangle2D r2d = (Rectangle2D) s;
+ double w = r2d.getWidth();
+ double h = r2d.getHeight();
+ double x = r2d.getX();
+ double y = r2d.getY();
+ if (w >= 0 && h >= 0) {
+ double lw = bs.getLineWidth();
+ drawRectangle(sg2d, x, y, w, h, lw);
+ }
+ return;
+ }
+ } else if (s instanceof Line2D) {
+ Line2D l2d = (Line2D) s;
+ if (drawGeneralLine(sg2d,
+ l2d.getX1(), l2d.getY1(),
+ l2d.getX2(), l2d.getY2()))
+ {
+ return;
+ }
+ }
+ }
+
+ outpipe.draw(sg2d, s);
+ }
+
+ public void fill(SunGraphics2D sg2d, Shape s) {
+ if (s instanceof Rectangle2D) {
+ Rectangle2D r2d = (Rectangle2D) s;
+ double w = r2d.getWidth();
+ double h = r2d.getHeight();
+ if (w > 0 && h > 0) {
+ double x = r2d.getX();
+ double y = r2d.getY();
+ fillRectangle(sg2d, x, y, w, h);
+ }
+ return;
+ }
+
+ outpipe.fill(sg2d, s);
+ }
+
+ static double len(double x, double y) {
+ return ((x == 0) ? Math.abs(y)
+ : ((y == 0) ? Math.abs(x)
+ : Math.sqrt(x * x + y * y)));
+ }
+
+ double normalize(double v) {
+ return Math.floor(v + normRoundingBias) + normPosition;
+ }
+
+ public boolean drawGeneralLine(SunGraphics2D sg2d,
+ double x1, double y1,
+ double x2, double y2)
+ {
+ if (sg2d.strokeState == SunGraphics2D.STROKE_CUSTOM ||
+ sg2d.strokeState == SunGraphics2D.STROKE_THINDASHED)
+ {
+ return false;
+ }
+ BasicStroke bs = (BasicStroke) sg2d.stroke;
+ int cap = bs.getEndCap();
+ if (cap == BasicStroke.CAP_ROUND || bs.getDashArray() != null) {
+ // TODO: we could construct the GeneralPath directly
+ // for CAP_ROUND and save a lot of processing in that case...
+ // And again, we would need to deal with dropout control...
+ return false;
+ }
+ double lw = bs.getLineWidth();
+ // Save the original dx, dy in case we need it to transform
+ // the linewidth as a perpendicular vector below
+ double dx = x2 - x1;
+ double dy = y2 - y1;
+ switch (sg2d.transformState) {
+ case SunGraphics2D.TRANSFORM_GENERIC:
+ case SunGraphics2D.TRANSFORM_TRANSLATESCALE:
+ {
+ double coords[] = {x1, y1, x2, y2};
+ sg2d.transform.transform(coords, 0, coords, 0, 2);
+ x1 = coords[0];
+ y1 = coords[1];
+ x2 = coords[2];
+ y2 = coords[3];
+ }
+ break;
+ case SunGraphics2D.TRANSFORM_ANY_TRANSLATE:
+ case SunGraphics2D.TRANSFORM_INT_TRANSLATE:
+ {
+ double tx = sg2d.transform.getTranslateX();
+ double ty = sg2d.transform.getTranslateY();
+ x1 += tx;
+ y1 += ty;
+ x2 += tx;
+ y2 += ty;
+ }
+ break;
+ case SunGraphics2D.TRANSFORM_ISIDENT:
+ break;
+ default:
+ throw new InternalError("unknown TRANSFORM state...");
+ }
+ if (sg2d.strokeHint != SunHints.INTVAL_STROKE_PURE) {
+ if (sg2d.strokeState == SunGraphics2D.STROKE_THIN &&
+ outrenderer instanceof PixelDrawPipe)
+ {
+ // PixelDrawPipes will add sg2d.transXY so we need to factor
+ // that out...
+ int ix1 = (int) Math.floor(x1 - sg2d.transX);
+ int iy1 = (int) Math.floor(y1 - sg2d.transY);
+ int ix2 = (int) Math.floor(x2 - sg2d.transX);
+ int iy2 = (int) Math.floor(y2 - sg2d.transY);
+ ((PixelDrawPipe)outrenderer).drawLine(sg2d, ix1, iy1, ix2, iy2);
+ return true;
+ }
+ x1 = normalize(x1);
+ y1 = normalize(y1);
+ x2 = normalize(x2);
+ y2 = normalize(y2);
+ }
+ if (sg2d.transformState >= SunGraphics2D.TRANSFORM_TRANSLATESCALE) {
+ // Transform the linewidth...
+ // calculate the scaling factor for a unit vector
+ // perpendicular to the original user space line.
+ double len = len(dx, dy);
+ if (len == 0) {
+ dx = len = 1;
+ // dy = 0; already
+ }
+ // delta transform the transposed (90 degree rotated) unit vector
+ double unitvector[] = {dy/len, -dx/len};
+ sg2d.transform.deltaTransform(unitvector, 0, unitvector, 0, 1);
+ lw *= len(unitvector[0], unitvector[1]);
+ }
+ lw = Math.max(lw, minPenSize);
+ dx = x2 - x1;
+ dy = y2 - y1;
+ double len = len(dx, dy);
+ double udx, udy;
+ if (len == 0) {
+ if (cap == BasicStroke.CAP_BUTT) {
+ return true;
+ }
+ udx = lw;
+ udy = 0;
+ } else {
+ udx = lw * dx / len;
+ udy = lw * dy / len;
+ }
+ double px = x1 + udy / 2.0;
+ double py = y1 - udx / 2.0;
+ if (cap == BasicStroke.CAP_SQUARE) {
+ px -= udx / 2.0;
+ py -= udy / 2.0;
+ dx += udx;
+ dy += udy;
+ }
+ outrenderer.fillParallelogram(sg2d, px, py, -udy, udx, dx, dy);
+ return true;
+ }
+
+ public void fillRectangle(SunGraphics2D sg2d,
+ double rx, double ry,
+ double rw, double rh)
+ {
+ double px, py;
+ double dx1, dy1, dx2, dy2;
+ AffineTransform txform = sg2d.transform;
+ dx1 = txform.getScaleX();
+ dy1 = txform.getShearY();
+ dx2 = txform.getShearX();
+ dy2 = txform.getScaleY();
+ px = rx * dx1 + ry * dx2 + txform.getTranslateX();
+ py = rx * dy1 + ry * dy2 + txform.getTranslateY();
+ dx1 *= rw;
+ dy1 *= rw;
+ dx2 *= rh;
+ dy2 *= rh;
+ if (adjustfill &&
+ sg2d.strokeState < SunGraphics2D.STROKE_CUSTOM &&
+ sg2d.strokeHint != SunHints.INTVAL_STROKE_PURE)
+ {
+ double newx = normalize(px);
+ double newy = normalize(py);
+ dx1 = normalize(px + dx1) - newx;
+ dy1 = normalize(py + dy1) - newy;
+ dx2 = normalize(px + dx2) - newx;
+ dy2 = normalize(py + dy2) - newy;
+ px = newx;
+ py = newy;
+ }
+ outrenderer.fillParallelogram(sg2d, px, py, dx1, dy1, dx2, dy2);
+ }
+
+ public void drawRectangle(SunGraphics2D sg2d,
+ double rx, double ry,
+ double rw, double rh,
+ double lw)
+ {
+ double px, py;
+ double dx1, dy1, dx2, dy2;
+ double lw1, lw2;
+ AffineTransform txform = sg2d.transform;
+ dx1 = txform.getScaleX();
+ dy1 = txform.getShearY();
+ dx2 = txform.getShearX();
+ dy2 = txform.getScaleY();
+ px = rx * dx1 + ry * dx2 + txform.getTranslateX();
+ py = rx * dy1 + ry * dy2 + txform.getTranslateY();
+ // lw along dx1,dy1 scale by transformed length of dx2,dy2 vectors
+ // and vice versa
+ lw1 = len(dx1, dy1) * lw;
+ lw2 = len(dx2, dy2) * lw;
+ dx1 *= rw;
+ dy1 *= rw;
+ dx2 *= rh;
+ dy2 *= rh;
+ if (sg2d.strokeState < SunGraphics2D.STROKE_CUSTOM &&
+ sg2d.strokeHint != SunHints.INTVAL_STROKE_PURE)
+ {
+ double newx = normalize(px);
+ double newy = normalize(py);
+ dx1 = normalize(px + dx1) - newx;
+ dy1 = normalize(py + dy1) - newy;
+ dx2 = normalize(px + dx2) - newx;
+ dy2 = normalize(py + dy2) - newy;
+ px = newx;
+ py = newy;
+ }
+ lw1 = Math.max(lw1, minPenSize);
+ lw2 = Math.max(lw2, minPenSize);
+ double len1 = len(dx1, dy1);
+ double len2 = len(dx2, dy2);
+ if (lw1 >= len1 || lw2 >= len2) {
+ // The line widths are large enough to consume the
+ // entire hole in the middle of the parallelogram
+ // so we can just fill the outer parallelogram.
+ fillOuterParallelogram(sg2d,
+ px, py, dx1, dy1, dx2, dy2,
+ len1, len2, lw1, lw2);
+ } else {
+ outrenderer.drawParallelogram(sg2d,
+ px, py, dx1, dy1, dx2, dy2,
+ lw1 / len1, lw2 / len2);
+ }
+ }
+
+ /**
+ * This utility function handles the case where a drawRectangle
+ * operation discovered that the interior hole in the rectangle
+ * or parallelogram has been completely filled in by the stroke
+ * width. It calculates the outer parallelogram of the stroke
+ * and issues a single fillParallelogram request to fill it.
+ */
+ public void fillOuterParallelogram(SunGraphics2D sg2d,
+ double px, double py,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double len1, double len2,
+ double lw1, double lw2)
+ {
+ double udx1 = dx1 / len1;
+ double udy1 = dy1 / len1;
+ double udx2 = dx2 / len2;
+ double udy2 = dy2 / len2;
+ if (len1 == 0) {
+ // len1 is 0, replace udxy1 with perpendicular of udxy2
+ if (len2 == 0) {
+ // both are 0, use a unit Y vector for udxy2
+ udx2 = 0;
+ udy2 = 1;
+ }
+ udx1 = udy2;
+ udy1 = -udx2;
+ } else if (len2 == 0) {
+ // len2 is 0, replace udxy2 with perpendicular of udxy1
+ udx2 = udy1;
+ udy2 = -udx1;
+ }
+ udx1 *= lw1;
+ udy1 *= lw1;
+ udx2 *= lw2;
+ udy2 *= lw2;
+ px -= (udx1 + udx2) / 2;
+ py -= (udy1 + udy2) / 2;
+ dx1 += udx1;
+ dy1 += udy1;
+ dx2 += udx2;
+ dy2 += udy2;
+
+ outrenderer.fillParallelogram(sg2d, px, py, dx1, dy1, dx2, dy2);
+ }
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/AccelDeviceEventListener.java b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelDeviceEventListener.java
new file mode 100644
index 00000000000..a6cacbdeebc
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelDeviceEventListener.java
@@ -0,0 +1,59 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+/**
+ * An interface for receiving notifications about imminent accelerated device's
+ * events. Upon receiving such event appropriate actions can be taken (for
+ * example, resources associated with the device can be freed).
+ */
+public interface AccelDeviceEventListener {
+ /**
+ * Called when the device is about to be reset.
+ *
+ * One must release all native resources associated with the device which
+ * prevent the device from being reset (such as Default Pool resources for
+ * the D3D pipeline).
+ *
+ * It is safe to remove the listener while in the call back.
+ *
+ * Note: this method is called on the rendering thread,
+ * do not call into user code, do not take RQ lock!
+ */
+ public void onDeviceReset();
+
+ /**
+ * Called when the device is about to be disposed of.
+ *
+ * One must release all native resources associated with the device.
+ *
+ * It is safe to remove the listener while in the call back.
+ *
+ * Note: this method is called on the rendering thread,
+ * do not call into user code, do not take RQ lock!
+ */
+ public void onDeviceDispose();
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/AccelDeviceEventNotifier.java b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelDeviceEventNotifier.java
new file mode 100644
index 00000000000..73a387a6c8a
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelDeviceEventNotifier.java
@@ -0,0 +1,167 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * This class is used to notify listeners about accelerated device's
+ * events such as device reset or dispose that are about to occur.
+ */
+public class AccelDeviceEventNotifier {
+
+ private static AccelDeviceEventNotifier theInstance;
+
+ /**
+ * A device is about to be reset. The listeners have to release all
+ * resources associated with the device which are required for the device
+ * to be reset.
+ */
+ public static final int DEVICE_RESET = 0;
+
+ /**
+ * A device is about to be disposed. The listeners have to release all
+ * resources associated with the device.
+ */
+ public static final int DEVICE_DISPOSED = 1;
+
+ private final Map listeners;
+
+ private AccelDeviceEventNotifier() {
+ listeners = Collections.synchronizedMap(
+ new HashMap(1));
+ }
+
+ /**
+ * Returns a singleton of AccelDeviceEventNotifier if it exists. If the
+ * passed boolean is false and singleton doesn't exist yet, null is
+ * returned. If the passed boolean is {@code true} and singleton doesn't
+ * exist it will be created and returned.
+ *
+ * @param create whether to create a singleton instance if doesn't yet
+ * exist
+ * @return a singleton instance or null
+ */
+ private static synchronized
+ AccelDeviceEventNotifier getInstance(boolean create)
+ {
+ if (theInstance == null && create) {
+ theInstance = new AccelDeviceEventNotifier();
+ }
+ return theInstance;
+ }
+
+ /**
+ * Called to indicate that a device event had occured.
+ * If a singleton exists, the listeners (those associated with
+ * the device) will be notified.
+ *
+ * @param screen a screen number of the device which is a source of
+ * the event
+ * @param eventType a type of the event
+ * @see #DEVICE_DISPOSED
+ * @see #DEVICE_RESET
+ */
+ public static final void eventOccured(int screen, int eventType) {
+ AccelDeviceEventNotifier notifier = getInstance(false);
+ if (notifier != null) {
+ notifier.notifyListeners(eventType, screen);
+ }
+ }
+
+ /**
+ * Adds the listener associated with a device on particular screen.
+ *
+ * Note: the listener must be removed as otherwise it will forever
+ * be referenced by the notifier.
+ *
+ * @param l the listener
+ * @param screen the screen number indicating which device the listener is
+ * interested in.
+ */
+ public static final void addListener(AccelDeviceEventListener l,int screen){
+ getInstance(true).add(l, screen);
+ }
+
+ /**
+ * Removes the listener.
+ *
+ * @param l the listener
+ */
+ public static final void removeListener(AccelDeviceEventListener l) {
+ getInstance(true).remove(l);
+ }
+
+ private final void add(AccelDeviceEventListener theListener, int screen) {
+ listeners.put(theListener, screen);
+ }
+ private final void remove(AccelDeviceEventListener theListener) {
+ listeners.remove(theListener);
+ }
+
+ /**
+ * Notifies the listeners associated with the screen's device about the
+ * event.
+ *
+ * Implementation note: the current list of listeners is first duplicated
+ * which allows the listeners to remove themselves during the iteration.
+ *
+ * @param screen a screen number with which the device which is a source of
+ * the event is associated with
+ * @param eventType a type of the event
+ * @see #DEVICE_DISPOSED
+ * @see #DEVICE_RESET
+ */
+ private final void notifyListeners(int deviceEventType, int screen) {
+ HashMap listClone;
+ Set cloneSet;
+
+ synchronized(listeners) {
+ listClone =
+ new HashMap(listeners);
+ }
+
+ cloneSet = listClone.keySet();
+ Iterator itr = cloneSet.iterator();
+ while (itr.hasNext()) {
+ AccelDeviceEventListener current = itr.next();
+ Integer i = listClone.get(current);
+ // only notify listeners which are interested in this device
+ if (i != null && i.intValue() != screen) {
+ continue;
+ }
+ if (deviceEventType == DEVICE_RESET) {
+ current.onDeviceReset();
+ } else if (deviceEventType == DEVICE_DISPOSED) {
+ current.onDeviceDispose();
+ }
+ }
+ }
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/AccelGraphicsConfig.java b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelGraphicsConfig.java
new file mode 100644
index 00000000000..ab799d94724
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelGraphicsConfig.java
@@ -0,0 +1,95 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+import java.awt.image.VolatileImage;
+
+/**
+ * Implementors of this interface provida a way to create a
+ * {@code VolatileImage} whose destination surface is an
+ * {@link AccelSurface} of specified type.
+ *
+ * @see AccelSurface
+ */
+public interface AccelGraphicsConfig extends BufferedContextProvider {
+ /**
+ * Returns a VolatileImage with specified width, height, transparency
+ * and guaranteed accelerated surface type. If such image can not be created
+ * (out of vram error, specific surface type is not supported) null
+ * is returned.
+ *
+ * Note: if {@link AccelSurface#TEXTURE} type is requested, rendering
+ * to the image will be denied by throwing
+ * {@code UnsupportedOperationException }
+ * from {@link java.awt.image.VolatileImage#getGraphics} and
+ * {@link java.awt.image.VolatileImage#createGraphics}
+ *
+ * @param width the width of the returned {@code VolatileImage}
+ * @param height the height of the returned {@code VolatileImage}
+ * @param transparency the specified transparency mode
+ * @param type requested accelerated surface type as specified by constants
+ * in AccelSurface interface
+ * @return a {@code VolatileImage} backed up by requested accelerated
+ * surface type or null
+ * @throws IllegalArgumentException if the transparency is not a valid value
+ * @see AccelSurface#TEXTURE
+ * @see AccelSurface#RT_PLAIN
+ * @see AccelSurface#RT_TEXTURE
+ */
+ public VolatileImage createCompatibleVolatileImage(int width, int height,
+ int transparency,
+ int type);
+ /**
+ * Returns object representing capabilities of the context associated
+ * with this {@code AccelGraphicsConfig}.
+ *
+ * @return ContextCapabilities object representing caps
+ * @see ContextCapabilities
+ */
+ public ContextCapabilities getContextCapabilities();
+
+ /**
+ * Adds an {@code AccelDeviceEventListener} to listen to accelerated
+ * device's (which is associated with this {@code AccelGraphicsConfig})
+ * events.
+ *
+ * Note: a hard link to the listener may be kept so it must be explicitly
+ * removed via {@link #removeDeviceEventListener()}.
+ *
+ * @param l the listener
+ * @see AccelDeviceEventListener
+ */
+ public void addDeviceEventListener(AccelDeviceEventListener l);
+
+ /**
+ * Removes an {@code AccelDeviceEventListener} from the list of listeners
+ * for this device's events.
+ *
+ * @param l the listener
+ * @see AccelDeviceEventListener
+ */
+ public void removeDeviceEventListener(AccelDeviceEventListener l);
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/AccelSurface.java b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelSurface.java
new file mode 100644
index 00000000000..c8692cb5ecb
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelSurface.java
@@ -0,0 +1,136 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+import java.awt.Rectangle;
+import sun.java2d.Surface;
+
+/**
+ * Abstraction for a hardware accelerated surface.
+ */
+public interface AccelSurface extends BufferedContextProvider, Surface {
+ /**
+ * Undefined
+ */
+ public static final int UNDEFINED = 0;
+ /**
+ * Window (or window substitute) surface
+ */
+ public static final int WINDOW = 1;
+ /**
+ * Render-To Plain surface (pbuffer for OpenGL, Render Target surface
+ * for Direct3D)
+ */
+ public static final int RT_PLAIN = 2;
+ /**
+ * Texture surface
+ */
+ public static final int TEXTURE = 3;
+ /**
+ * A back-buffer surface (SwapChain surface for Direct3D, backbuffer for
+ * OpenGL)
+ */
+ public static final int FLIP_BACKBUFFER = 4;
+ /**
+ * Render-To Texture surface (fbobject for OpenGL, texture with render-to
+ * attribute for Direct3D)
+ */
+ public static final int RT_TEXTURE = 5;
+
+ /**
+ * Returns {@code int} representing surface's type as defined by constants
+ * in this interface.
+ *
+ * @return an integer representing this surface's type
+ * @see AccelSurface#UNDEFINED
+ * @see AccelSurface#WINDOW
+ * @see AccelSurface#RT_PLAIN
+ * @see AccelSurface#TEXTURE
+ * @see AccelSurface#FLIP_BACKBUFFER
+ * @see AccelSurface#RT_TEXTURE
+ */
+ public int getType();
+
+ /**
+ * Returns a pointer to the native surface data associated with this
+ * surface.
+ * Note: this pointer is only valid on the rendering thread.
+ *
+ * @return pointer to the native surface's data
+ */
+ public long getNativeOps();
+
+ /**
+ * Returns a pointer to the real native resource
+ * of the specified type associated with this AccelSurface.
+ * Note: this pointer is only valid on the rendering thread.
+ *
+ * @param resType the type of the requested resource
+ * @return a long containing a pointer to the native resource of the
+ * specified type or 0L if such resource doesn't exist for this surface
+ */
+ public long getNativeResource(int resType);
+
+ /**
+ * Marks this surface dirty.
+ */
+ public void markDirty();
+
+ /**
+ * Returns whether the pipeline considers this surface valid. A surface
+ * may become invalid if it is disposed of, or resized.
+ *
+ * @return true if valid, false otherwise
+ */
+ public boolean isValid();
+
+ /**
+ * Returns whether this surface is lost. The return value is only valid
+ * on the render thread, meaning that even if this method returns
+ * {@code true} it could be lost in the next moment unless it is called
+ * on the rendering thread.
+ *
+ * @return true if the surface is known to be lost, false otherwise
+ */
+ public boolean isSurfaceLost();
+
+ /**
+ * Returns the requested bounds of the destination surface. The real bounds
+ * of the native accelerated surface may differ. Use
+ * {@link #getNativeBounds} to get the bounds of the native surface.
+ *
+ * @return Rectangle representing java surface's bounds
+ */
+ public Rectangle getBounds();
+
+ /**
+ * Returns real bounds of the native surface, which may differ from those
+ * returned by {@link #getBounds}.
+ *
+ * @return Rectangle representing native surface's bounds
+ */
+ public Rectangle getNativeBounds();
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/AccelTypedVolatileImage.java b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelTypedVolatileImage.java
new file mode 100644
index 00000000000..e855201d77b
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/AccelTypedVolatileImage.java
@@ -0,0 +1,74 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import sun.awt.image.SunVolatileImage;
+import static sun.java2d.pipe.hw.AccelSurface.*;
+
+/**
+ * This is an image with forced type of the accelerated surface.
+ */
+public class AccelTypedVolatileImage extends SunVolatileImage {
+
+ /**
+ * Creates a volatile image with specified type of accelerated surface.
+ *
+ * @param graphicsConfig a GraphicsConfiguration for which this image should
+ * be created.
+ * @param width width
+ * @param height width
+ * @param transparency type of {@link java.awt.Transparency transparency}
+ * requested for the image
+ * @param accType type of the desired accelerated surface as defined in
+ * AccelSurface interface
+ * @see sun.java2d.pipe.hw.AccelSurface
+ */
+ public AccelTypedVolatileImage(GraphicsConfiguration graphicsConfig,
+ int width, int height, int transparency,
+ int accType)
+ {
+ super(null, graphicsConfig, width, height, null, transparency,
+ null, accType);
+ }
+
+ /**
+ * {@inheritDoc}
+ *
+ * This method will throw {@code UnsupportedOperationException} if it this
+ * image's destination surface can not be rendered to.
+ */
+ @Override
+ public Graphics2D createGraphics() {
+ if (getForcedAccelSurfaceType() == TEXTURE) {
+ throw new UnsupportedOperationException("Can't render " +
+ "to a non-RT Texture");
+ }
+ return super.createGraphics();
+ }
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/BufferedContextProvider.java b/jdk/src/share/classes/sun/java2d/pipe/hw/BufferedContextProvider.java
new file mode 100644
index 00000000000..86df94442e0
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/BufferedContextProvider.java
@@ -0,0 +1,45 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+import sun.java2d.pipe.BufferedContext;
+
+/**
+ * Classes implementing this interface can provide the {@code BufferedContext}
+ * associated with or used by them.
+ *
+ * @see sun.java2d.pipe.BufferedContext
+ */
+public interface BufferedContextProvider {
+ /**
+ * Retrieves a context associated with object implementing this
+ * interface.
+ *
+ * @return associated context
+ * @see sun.java2d.pipe.BufferedContext
+ */
+ public BufferedContext getContext();
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/ContextCapabilities.java b/jdk/src/share/classes/sun/java2d/pipe/hw/ContextCapabilities.java
new file mode 100644
index 00000000000..5f17028b0d2
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/ContextCapabilities.java
@@ -0,0 +1,128 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+/**
+ * Represents a set of capabilities of a BufferedContext and associated
+ * AccelGraphicsConfig.
+ *
+ * @see AccelGraphicsConfig
+ */
+public class ContextCapabilities {
+ /** Indicates that the context has no capabilities. */
+ public static final int CAPS_EMPTY = (0 << 0);
+ /** Indicates that the context supports RT surfaces with alpha channel. */
+ public static final int CAPS_RT_PLAIN_ALPHA = (1 << 1);
+ /** Indicates that the context supports RT textures with alpha channel. */
+ public static final int CAPS_RT_TEXTURE_ALPHA = (1 << 2);
+ /** Indicates that the context supports opaque RT textures. */
+ public static final int CAPS_RT_TEXTURE_OPAQUE = (1 << 3);
+ /** Indicates that the context supports multitexturing. */
+ public static final int CAPS_MULTITEXTURE = (1 << 4);
+ /** Indicates that the context supports non-pow2 texture dimensions. */
+ public static final int CAPS_TEXNONPOW2 = (1 << 5);
+ /** Indicates that the context supports non-square textures. */
+ public static final int CAPS_TEXNONSQUARE = (1 << 6);
+ /** Indicates that the context supports pixel shader 2.0 or better. */
+ public static final int CAPS_PS20 = (1 << 7);
+ /** Indicates that the context supports pixel shader 3.0 or better. */
+ public static final int CAPS_PS30 = (1 << 8);
+ /*
+ * Pipeline contexts should use this for defining pipeline-specific
+ * capabilities, for example:
+ * int CAPS_D3D_1 = (FIRST_PRIVATE_CAP << 0);
+ * int CAPS_D3D_2 = (FIRST_PRIVATE_CAP << 1);
+ */
+ protected static final int FIRST_PRIVATE_CAP = (1 << 16);
+
+ protected final int caps;
+ protected final String adapterId;
+
+ /**
+ * Constructs a {@code ContextCapabilities} object.
+ * @param caps an {@code int} representing the capabilities
+ * @param a {@code String} representing the name of the adapter, or null,
+ * in which case the adapterId will be set to "unknown adapter".
+ */
+ protected ContextCapabilities(int caps, String adapterId) {
+ this.caps = caps;
+ this.adapterId = adapterId != null ? adapterId : "unknown adapter";
+ }
+
+ /**
+ * Returns a string representing the name of the graphics adapter if such
+ * could be determined. It is guaranteed to never return {@code null}.
+ * @return string representing adapter id
+ */
+ public String getAdapterId() {
+ return adapterId;
+ }
+
+ /**
+ * Returns an {@code int} with capabilities (OR-ed constants defined in
+ * this class and its pipeline-specific subclasses).
+ * @return capabilities as {@code int}
+ */
+ public int getCaps() {
+ return caps;
+ }
+
+ @Override
+ public String toString() {
+ StringBuffer buf =
+ new StringBuffer("ContextCapabilities: adapter=" +
+ adapterId+", caps=");
+ if (caps == CAPS_EMPTY) {
+ buf.append("CAPS_EMPTY");
+ } else {
+ if ((caps & CAPS_RT_PLAIN_ALPHA) != 0) {
+ buf.append("CAPS_RT_PLAIN_ALPHA|");
+ }
+ if ((caps & CAPS_RT_TEXTURE_ALPHA) != 0) {
+ buf.append("CAPS_RT_TEXTURE_ALPHA|");
+ }
+ if ((caps & CAPS_RT_TEXTURE_OPAQUE) != 0) {
+ buf.append("CAPS_RT_TEXTURE_OPAQUE|");
+ }
+ if ((caps & CAPS_MULTITEXTURE) != 0) {
+ buf.append("CAPS_MULTITEXTURE|");
+ }
+ if ((caps & CAPS_TEXNONPOW2) != 0) {
+ buf.append("CAPS_TEXNONPOW2|");
+ }
+ if ((caps & CAPS_TEXNONSQUARE) != 0) {
+ buf.append("CAPS_TEXNONSQUARE|");
+ }
+ if ((caps & CAPS_PS20) != 0) {
+ buf.append("CAPS_PS20|");
+ }
+ if ((caps & CAPS_PS30) != 0) {
+ buf.append("CAPS_PS30|");
+ }
+ }
+ return buf.toString();
+ }
+}
diff --git a/jdk/src/share/classes/sun/java2d/pipe/hw/ExtendedBufferCapabilities.java b/jdk/src/share/classes/sun/java2d/pipe/hw/ExtendedBufferCapabilities.java
new file mode 100644
index 00000000000..65eb305ab6c
--- /dev/null
+++ b/jdk/src/share/classes/sun/java2d/pipe/hw/ExtendedBufferCapabilities.java
@@ -0,0 +1,154 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.pipe.hw;
+
+import java.awt.BufferCapabilities;
+import java.awt.ImageCapabilities;
+
+/**
+ * Provides extended BufferStrategy capabilities, allowing to specify
+ * the type of vertical refresh synchronization for a buffer strategy.
+ *
+ * This BS capability is always page flipping because v-sync is only relevant
+ * to flipping buffer strategies.
+ *
+ * Note that asking for a v-synced BS doesn't necessarily guarantee that it will
+ * be v-synced since the vsync capability may be disabled in the driver, or
+ * there may be other restriction (like a number of v-synced buffer strategies
+ * allowed per vm). Because of this {@code createBufferStrategy} doesn't
+ * throw {@code AWTException} when a v-synced BS could not be created when
+ * requested.
+ *
+ * @see java.awt.Canvas#createBufferStrategy(int, BufferCapabilities)
+ * @see java.awt.Window#createBufferStrategy(int, BufferCapabilities)
+ */
+public class ExtendedBufferCapabilities extends BufferCapabilities {
+
+ /**
+ * Type of synchronization on vertical retrace.
+ */
+ public static enum VSyncType {
+ /**
+ * Use the default v-sync mode appropriate for given BufferStrategy
+ * and situation.
+ */
+ VSYNC_DEFAULT(0),
+
+ /**
+ * Synchronize flip on vertical retrace.
+ */
+ VSYNC_ON(1),
+
+ /**
+ * Do not synchronize flip on vertical retrace.
+ */
+ VSYNC_OFF(2);
+
+ /**
+ * Used to identify the v-sync type (independent of the constants
+ * order as opposed to {@code ordinal()}).
+ */
+ public int id() {
+ return id;
+ }
+
+ private VSyncType(int id) {
+ this.id = id;
+ }
+ private int id;
+ }
+
+ private VSyncType vsync;
+
+ /**
+ * Creates an ExtendedBufferCapabilities object with front/back/flip caps
+ * from the passed cap, and VSYNC_DEFAULT v-sync mode.
+ */
+ public ExtendedBufferCapabilities(BufferCapabilities caps) {
+ super(caps.getFrontBufferCapabilities(),
+ caps.getBackBufferCapabilities(),
+ caps.getFlipContents());
+
+ this.vsync = VSyncType.VSYNC_DEFAULT;
+ }
+
+ /**
+ * Creates an ExtendedBufferCapabilities instance with front/back/flip caps
+ * from the passed caps, and VSYNC_DEFAULT v-sync mode.
+ */
+ public ExtendedBufferCapabilities(ImageCapabilities front,
+ ImageCapabilities back, FlipContents flip)
+ {
+ super(front, back, flip);
+
+ this.vsync = VSyncType.VSYNC_DEFAULT;
+ }
+
+ /**
+ * Creates an ExtendedBufferCapabilities instance with front/back/flip caps
+ * from the passed image/flip caps, and the v-sync type.
+ */
+ public ExtendedBufferCapabilities(ImageCapabilities front,
+ ImageCapabilities back, FlipContents flip,
+ VSyncType t)
+ {
+ super(front, back, flip);
+
+ this.vsync = t;
+ }
+
+ /**
+ * Creates an ExtendedBufferCapabilities instance with front/back/flip caps
+ * from the passed cap, and the passed v-sync mode.
+ */
+ public ExtendedBufferCapabilities(BufferCapabilities caps, VSyncType t) {
+ super(caps.getFrontBufferCapabilities(),
+ caps.getBackBufferCapabilities(),
+ caps.getFlipContents());
+
+ this.vsync = t;
+ }
+
+ /**
+ * Creates an ExtendedBufferCapabilities instance with front/back/flip caps
+ * from the object, and passed v-sync mode.
+ */
+ public ExtendedBufferCapabilities derive(VSyncType t) {
+ return new ExtendedBufferCapabilities(this, t);
+ }
+
+ /**
+ * Returns the type of v-sync requested by this capabilities instance.
+ */
+ public VSyncType getVSync() {
+ return vsync;
+ }
+
+ @Override
+ public final boolean isPageFlipping() {
+ return true;
+ }
+}
diff --git a/jdk/src/share/classes/sun/text/resources/FormatData_sv.java b/jdk/src/share/classes/sun/text/resources/FormatData_sv.java
index a5f25230202..f02245a55da 100644
--- a/jdk/src/share/classes/sun/text/resources/FormatData_sv.java
+++ b/jdk/src/share/classes/sun/text/resources/FormatData_sv.java
@@ -104,6 +104,12 @@ public class FormatData_sv extends ListResourceBundle {
"l\u00f6" // abb Saturday
}
},
+ { "AmPmMarkers",
+ new String[] {
+ "fm", // am marker
+ "em" // pm marker
+ }
+ },
{ "NumberElements",
new String[] {
",", // decimal separator
diff --git a/jdk/src/share/native/sun/font/AccelGlyphCache.c b/jdk/src/share/native/sun/font/AccelGlyphCache.c
index abd2d881a35..73e94a91c38 100644
--- a/jdk/src/share/native/sun/font/AccelGlyphCache.c
+++ b/jdk/src/share/native/sun/font/AccelGlyphCache.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -45,6 +45,19 @@
* list describing the cache as new glyphs are added. Platform specific
* glyph caching code is responsible for actually creating the accelerated
* memory surface that will contain the individual glyph images.
+ *
+ * Each glyph contains a reference to a list of cell infos - one per glyph
+ * cache. There may be multiple glyph caches (for example, one per graphics
+ * adapter), so if the glyph is cached on two devices its cell list will
+ * consists of two elements corresponding to different glyph caches.
+ *
+ * The platform-specific glyph caching code is supposed to use
+ * GetCellInfoForCache method for retrieving cache infos from the glyph's list.
+ *
+ * Note that if it is guaranteed that there will be only one global glyph
+ * cache then it one does not have to use AccelGlyphCache_GetCellInfoForCache
+ * for retrieving cell info for the glyph, but instead just use the struct's
+ * field directly.
*/
GlyphCacheInfo *
AccelGlyphCache_Init(jint width, jint height,
@@ -86,8 +99,11 @@ AccelGlyphCache_Init(jint width, jint height,
* "virtual" glyph cache is available for the glyph image. Platform specific
* glyph caching code is responsible for actually caching the glyph image
* in the associated accelerated memory surface.
+ *
+ * Returns created cell info if it was successfully created and added to the
+ * cache and glyph's cell lists, NULL otherwise.
*/
-void
+CacheCellInfo *
AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
{
CacheCellInfo *cellinfo = NULL;
@@ -99,7 +115,7 @@ AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
if ((glyph->width > cache->cellWidth) ||
(glyph->height > cache->cellHeight))
{
- return;
+ return NULL;
}
if (!cache->isFull) {
@@ -126,9 +142,8 @@ AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
// create new CacheCellInfo
cellinfo = (CacheCellInfo *)malloc(sizeof(CacheCellInfo));
if (cellinfo == NULL) {
- glyph->cellInfo = NULL;
J2dTraceLn(J2D_TRACE_ERROR, "could not allocate CellInfo");
- return;
+ return NULL;
}
cellinfo->cacheInfo = cache;
@@ -136,6 +151,8 @@ AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
cellinfo->timesRendered = 0;
cellinfo->x = x;
cellinfo->y = y;
+ cellinfo->leftOff = 0;
+ cellinfo->rightOff = 0;
cellinfo->tx1 = (jfloat)cellinfo->x / cache->width;
cellinfo->ty1 = (jfloat)cellinfo->y / cache->height;
cellinfo->tx2 = cellinfo->tx1 + ((jfloat)w / cache->width);
@@ -152,6 +169,7 @@ AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
// add the new cell to the end of the list
cache->tail = cellinfo;
cellinfo->next = NULL;
+ cellinfo->nextGCI = NULL;
}
}
@@ -210,9 +228,9 @@ AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
cache->Flush();
}
- // if the cell is occupied, notify the base glyph that its
- // cached version is about to be kicked out
- cellinfo->glyphInfo->cellInfo = NULL;
+ // if the cell is occupied, notify the base glyph that the
+ // cached version for this cache is about to be kicked out
+ AccelGlyphCache_RemoveCellInfo(cellinfo->glyphInfo, cellinfo);
}
// update cellinfo with glyph's occupied region information
@@ -221,8 +239,9 @@ AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, GlyphInfo *glyph)
cellinfo->ty2 = cellinfo->ty1 + ((jfloat)h / cache->height);
}
- // update the glyph's reference to its cache cell
- glyph->cellInfo = cellinfo;
+ // add cache cell to the glyph's cells list
+ AccelGlyphCache_AddCellInfo(glyph, cellinfo);
+ return cellinfo;
}
/**
@@ -251,10 +270,145 @@ AccelGlyphCache_Invalidate(GlyphCacheInfo *cache)
while (cellinfo != NULL) {
if (cellinfo->glyphInfo != NULL) {
// if the cell is occupied, notify the base glyph that its
- // cached version is about to be invalidated
- cellinfo->glyphInfo->cellInfo = NULL;
- cellinfo->glyphInfo = NULL;
+ // cached version for this cache is about to be invalidated
+ AccelGlyphCache_RemoveCellInfo(cellinfo->glyphInfo, cellinfo);
}
cellinfo = cellinfo->next;
}
}
+
+/**
+ * Invalidates and frees all cells and the cache itself. The "cache" pointer
+ * becomes invalid after this function returns.
+ */
+void
+AccelGlyphCache_Free(GlyphCacheInfo *cache)
+{
+ CacheCellInfo *cellinfo;
+
+ J2dTraceLn(J2D_TRACE_INFO, "AccelGlyphCache_Free");
+
+ if (cache == NULL) {
+ return;
+ }
+
+ // flush any pending vertices that may be depending on the current
+ // glyph cache
+ if (cache->Flush != NULL) {
+ cache->Flush();
+ }
+
+ while (cache->head != NULL) {
+ cellinfo = cache->head;
+ if (cellinfo->glyphInfo != NULL) {
+ // if the cell is occupied, notify the base glyph that its
+ // cached version for this cache is about to be invalidated
+ AccelGlyphCache_RemoveCellInfo(cellinfo->glyphInfo, cellinfo);
+ }
+ cache->head = cellinfo->next;
+ free(cellinfo);
+ }
+ free(cache);
+}
+
+/**
+ * Add cell info to the head of the glyph's list of cached cells.
+ */
+void
+AccelGlyphCache_AddCellInfo(GlyphInfo *glyph, CacheCellInfo *cellInfo)
+{
+ // assert (glyph != NULL && cellInfo != NULL)
+ J2dTraceLn(J2D_TRACE_INFO, "AccelGlyphCache_AddCellInfo");
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " glyph 0x%x: adding cell 0x%x to the list",
+ glyph, cellInfo);
+
+ cellInfo->glyphInfo = glyph;
+ cellInfo->nextGCI = glyph->cellInfo;
+ glyph->cellInfo = cellInfo;
+}
+
+/**
+ * Removes cell info from the glyph's list of cached cells.
+ */
+void
+AccelGlyphCache_RemoveCellInfo(GlyphInfo *glyph, CacheCellInfo *cellInfo)
+{
+ CacheCellInfo *currCellInfo = glyph->cellInfo;
+ CacheCellInfo *prevInfo = NULL;
+ // assert (glyph!= NULL && glyph->cellInfo != NULL && cellInfo != NULL)
+ J2dTraceLn(J2D_TRACE_INFO, "AccelGlyphCache_RemoveCellInfo");
+ do {
+ if (currCellInfo == cellInfo) {
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ " glyph 0x%x: removing cell 0x%x from glyph's list",
+ glyph, currCellInfo);
+ if (prevInfo == NULL) { // it's the head, chop-chop
+ glyph->cellInfo = currCellInfo->nextGCI;
+ } else {
+ prevInfo->nextGCI = currCellInfo->nextGCI;
+ }
+ currCellInfo->glyphInfo = NULL;
+ currCellInfo->nextGCI = NULL;
+ return;
+ }
+ prevInfo = currCellInfo;
+ currCellInfo = currCellInfo->nextGCI;
+ } while (currCellInfo != NULL);
+ J2dTraceLn2(J2D_TRACE_WARNING, "AccelGlyphCache_RemoveCellInfo: "\
+ "no cell 0x%x in glyph 0x%x's cell list",
+ cellInfo, glyph);
+}
+
+/**
+ * Removes cell info from the glyph's list of cached cells.
+ */
+JNIEXPORT void
+AccelGlyphCache_RemoveAllCellInfos(GlyphInfo *glyph)
+{
+ CacheCellInfo *currCell, *prevCell;
+
+ J2dTraceLn(J2D_TRACE_INFO, "AccelGlyphCache_RemoveAllCellInfos");
+
+ if (glyph == NULL || glyph->cellInfo == NULL) {
+ return;
+ }
+
+ // invalidate all of this glyph's accelerated cache cells
+ currCell = glyph->cellInfo;
+ do {
+ currCell->glyphInfo = NULL;
+ prevCell = currCell;
+ currCell = currCell->nextGCI;
+ prevCell->nextGCI = NULL;
+ } while (currCell != NULL);
+
+ glyph->cellInfo = NULL;
+}
+
+/**
+ * Returns cell info associated with particular cache from the glyph's list of
+ * cached cells.
+ */
+CacheCellInfo *
+AccelGlyphCache_GetCellInfoForCache(GlyphInfo *glyph, GlyphCacheInfo *cache)
+{
+ // assert (glyph != NULL && cache != NULL)
+ J2dTraceLn(J2D_TRACE_VERBOSE2, "AccelGlyphCache_GetCellInfoForCache");
+
+ if (glyph->cellInfo != NULL) {
+ CacheCellInfo *cellInfo = glyph->cellInfo;
+ do {
+ if (cellInfo->cacheInfo == cache) {
+ J2dTraceLn3(J2D_TRACE_VERBOSE2,
+ " glyph 0x%x: found cell 0x%x for cache 0x%x",
+ glyph, cellInfo, cache);
+ return cellInfo;
+ }
+ cellInfo = cellInfo->nextGCI;
+ } while (cellInfo != NULL);
+ }
+ J2dTraceLn2(J2D_TRACE_VERBOSE2, " glyph 0x%x: no cell for cache 0x%x",
+ glyph, cache);
+ return NULL;
+}
+
diff --git a/jdk/src/share/native/sun/font/AccelGlyphCache.h b/jdk/src/share/native/sun/font/AccelGlyphCache.h
index fa62dc044b7..2a4f0bac150 100644
--- a/jdk/src/share/native/sun/font/AccelGlyphCache.h
+++ b/jdk/src/share/native/sun/font/AccelGlyphCache.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -26,6 +26,10 @@
#ifndef AccelGlyphCache_h_Included
#define AccelGlyphCache_h_Included
+#ifdef __cplusplus
+extern "C" {
+#endif
+
#include "jni.h"
#include "fontscalerdefs.h"
@@ -48,10 +52,18 @@ typedef struct {
struct _CacheCellInfo {
GlyphCacheInfo *cacheInfo;
struct GlyphInfo *glyphInfo;
+ // next cell info in the cache's list
CacheCellInfo *next;
+ // REMIND: find better name?
+ // next cell info in the glyph's cell list (next Glyph Cache Info)
+ CacheCellInfo *nextGCI;
jint timesRendered;
jint x;
jint y;
+ // number of pixels from the left or right edge not considered touched
+ // by the glyph
+ jint leftOff;
+ jint rightOff;
jfloat tx1;
jfloat ty1;
jfloat tx2;
@@ -62,9 +74,24 @@ GlyphCacheInfo *
AccelGlyphCache_Init(jint width, jint height,
jint cellWidth, jint cellHeight,
FlushFunc *func);
-void
+CacheCellInfo *
AccelGlyphCache_AddGlyph(GlyphCacheInfo *cache, struct GlyphInfo *glyph);
void
AccelGlyphCache_Invalidate(GlyphCacheInfo *cache);
+void
+AccelGlyphCache_AddCellInfo(struct GlyphInfo *glyph, CacheCellInfo *cellInfo);
+void
+AccelGlyphCache_RemoveCellInfo(struct GlyphInfo *glyph, CacheCellInfo *cellInfo);
+CacheCellInfo *
+AccelGlyphCache_GetCellInfoForCache(struct GlyphInfo *glyph,
+ GlyphCacheInfo *cache);
+JNIEXPORT void
+AccelGlyphCache_RemoveAllCellInfos(struct GlyphInfo *glyph);
+void
+AccelGlyphCache_Free(GlyphCacheInfo *cache);
+
+#ifdef __cplusplus
+};
+#endif
#endif /* AccelGlyphCache_h_Included */
diff --git a/jdk/src/share/native/sun/font/freetypeScaler.c b/jdk/src/share/native/sun/font/freetypeScaler.c
index a3229df5dd1..5f6385b2868 100644
--- a/jdk/src/share/native/sun/font/freetypeScaler.c
+++ b/jdk/src/share/native/sun/font/freetypeScaler.c
@@ -770,11 +770,9 @@ Java_sun_font_FreetypeFontScaler_getGlyphImageNative(
glyphInfo->topLeftX = (float) ftglyph->bitmap_left;
glyphInfo->topLeftY = (float) -ftglyph->bitmap_top;
- if (context->aaType == TEXT_AA_LCD_HRGB ||
- context->aaType == TEXT_AA_LCD_HBGR) {
+ if (ftglyph->bitmap.pixel_mode == FT_PIXEL_MODE_LCD) {
glyphInfo->width = width/3;
- } else if (context->aaType == TEXT_AA_LCD_VRGB ||
- context->aaType == TEXT_AA_LCD_VBGR) {
+ } else if (ftglyph->bitmap.pixel_mode == FT_PIXEL_MODE_LCD_V) {
glyphInfo->height = glyphInfo->height/3;
}
diff --git a/jdk/src/share/native/sun/font/sunFont.c b/jdk/src/share/native/sun/font/sunFont.c
index 7c00561f684..af8aa669808 100644
--- a/jdk/src/share/native/sun/font/sunFont.c
+++ b/jdk/src/share/native/sun/font/sunFont.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -35,6 +35,8 @@
#include "sun_font_StrikeCache.h"
static void *theNullScalerContext = NULL;
+extern void AccelGlyphCache_RemoveAllCellInfos(GlyphInfo *glyph);
+
JNIEXPORT jlong JNICALL
Java_sun_font_NullFontScaler_getNullScalerContext
@@ -294,7 +296,7 @@ JNIEXPORT void JNICALL Java_sun_font_StrikeCache_freeIntMemory
GlyphInfo *ginfo = (GlyphInfo *)ptrs[i];
if (ginfo->cellInfo != NULL) {
// invalidate this glyph's accelerated cache cell
- ginfo->cellInfo->glyphInfo = NULL;
+ AccelGlyphCache_RemoveAllCellInfos(ginfo);
}
free((void*)ginfo);
}
@@ -324,8 +326,7 @@ JNIEXPORT void JNICALL Java_sun_font_StrikeCache_freeLongMemory
if (ptrs[i] != 0L) {
GlyphInfo *ginfo = (GlyphInfo *) jlong_to_ptr(ptrs[i]);
if (ginfo->cellInfo != NULL) {
- // invalidate this glyph's accelerated cache cell
- ginfo->cellInfo->glyphInfo = NULL;
+ AccelGlyphCache_RemoveAllCellInfos(ginfo);
}
free((void*)ginfo);
}
diff --git a/jdk/src/share/native/sun/java2d/ShaderList.c b/jdk/src/share/native/sun/java2d/ShaderList.c
new file mode 100644
index 00000000000..8831f2abd06
--- /dev/null
+++ b/jdk/src/share/native/sun/java2d/ShaderList.c
@@ -0,0 +1,143 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include
+#include
+
+#include "ShaderList.h"
+#include "Trace.h"
+
+/**
+ * Creates a new ShaderInfo that wraps the given fragment program handle
+ * and related data and stores it at the front of the provided ShaderList.
+ * If the addition causes the ShaderList to outgrow its defined capacity,
+ * the least-recently used item in the list (including its fragment program
+ * object) will be disposed.
+ */
+void
+ShaderList_AddProgram(ShaderList *programList,
+ jlong programID,
+ jint compType, jint compMode, jint flags)
+{
+ ShaderInfo *info;
+
+ J2dTraceLn(J2D_TRACE_INFO, "ShaderList_AddProgram");
+
+ // create new ShaderInfo
+ info = (ShaderInfo *)malloc(sizeof(ShaderInfo));
+ if (info == NULL) {
+ J2dTraceLn(J2D_TRACE_ERROR,
+ "OGLContext_AddProgram: could not allocate ShaderInfo");
+ return;
+ }
+
+ // fill in the information
+ info->next = programList->head;
+ info->programID = programID;
+ info->compType = compType;
+ info->compMode = compMode;
+ info->flags = flags;
+
+ // insert it at the head of the list
+ programList->head = info;
+
+ // run through the list and see if we need to delete the least
+ // recently used item
+ {
+ int i = 1;
+ ShaderInfo *prev = NULL;
+ ShaderInfo *curr = info->next;
+ while (curr != NULL) {
+ if (i >= programList->maxItems) {
+ prev->next = NULL;
+ programList->dispose(curr->programID);
+ free(curr);
+ break;
+ }
+ i++;
+ prev = curr;
+ curr = curr->next;
+ }
+ }
+}
+
+/**
+ * Locates a fragment program handle given a list of shader programs
+ * (ShaderInfos), using the provided composite state and flags as search
+ * parameters. The "flags" parameter is a bitwise-or'd value that helps
+ * differentiate one program for another; the interpretation of this value
+ * varies depending on the type of shader (BufImgOp, Paint, etc) but here
+ * it is only used to find another ShaderInfo with that same "flags" value.
+ * If no matching program can be located, this method returns 0.
+ */
+jlong
+ShaderList_FindProgram(ShaderList *programList,
+ jint compType, jint compMode, jint flags)
+{
+ ShaderInfo *prev = NULL;
+ ShaderInfo *info = programList->head;
+
+ J2dTraceLn(J2D_TRACE_INFO, "ShaderList_FindProgram");
+
+ while (info != NULL) {
+ if (compType == info->compType &&
+ compMode == info->compMode &&
+ flags == info->flags)
+ {
+ // it's a match: move it to the front of the list (if it's not
+ // there already) and patch up the links
+ if (info != programList->head) {
+ prev->next = info->next;
+ info->next = programList->head;
+ programList->head = info;
+ }
+ return info->programID;
+ }
+ prev = info;
+ info = info->next;
+ }
+ return 0;
+}
+
+/**
+ * Disposes all entries (and their associated shader program objects)
+ * contained in the given ShaderList.
+ */
+void
+ShaderList_Dispose(ShaderList *programList)
+{
+ ShaderInfo *info = programList->head;
+
+ J2dTraceLn(J2D_TRACE_INFO, "ShaderList_Dispose");
+
+ while (info != NULL) {
+ ShaderInfo *tmp = info->next;
+ programList->dispose(info->programID);
+ free(info);
+ info = tmp;
+ }
+
+ programList->head = NULL;
+}
diff --git a/jdk/src/share/native/sun/java2d/ShaderList.h b/jdk/src/share/native/sun/java2d/ShaderList.h
new file mode 100644
index 00000000000..f46190fcde9
--- /dev/null
+++ b/jdk/src/share/native/sun/java2d/ShaderList.h
@@ -0,0 +1,85 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef ShaderList_h_Included
+#define ShaderList_h_Included
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#include "jni.h"
+#include "jlong.h"
+
+typedef void (ShaderDisposeFunc)(jlong programID);
+
+/**
+ * The following structures are used to maintain a list of fragment program
+ * objects and their associated attributes. Each logical shader (e.g.
+ * RadialGradientPaint shader, ConvolveOp shader) can have a number of
+ * different variants depending on a number of factors, such as whether
+ * antialiasing is enabled or the current composite mode. Since the number
+ * of possible combinations of these factors is in the hundreds, we need
+ * some way to create fragment programs on an as-needed basis, and also
+ * keep them in a limited sized cache to avoid creating too many objects.
+ *
+ * The ShaderInfo structure keeps a reference to the fragment program's
+ * handle, as well as some other values that help differentiate one ShaderInfo
+ * from another. ShaderInfos can be chained together to form a linked list.
+ *
+ * The ShaderList structure acts as a cache for ShaderInfos, placing
+ * most-recently used items at the front, and removing items from the
+ * cache when its size exceeds the "maxItems" limit.
+ */
+typedef struct _ShaderInfo ShaderInfo;
+
+typedef struct {
+ ShaderInfo *head;
+ ShaderDisposeFunc *dispose;
+ jint maxItems;
+} ShaderList;
+
+struct _ShaderInfo {
+ ShaderInfo *next;
+ jlong programID;
+ jint compType;
+ jint compMode;
+ jint flags;
+};
+
+void ShaderList_AddProgram(ShaderList *programList,
+ jlong programID,
+ jint compType, jint compMode,
+ jint flags);
+jlong ShaderList_FindProgram(ShaderList *programList,
+ jint compType, jint compMode,
+ jint flags);
+void ShaderList_Dispose(ShaderList *programList);
+
+#ifdef __cplusplus
+};
+#endif
+
+#endif /* ShaderList_h_Included */
diff --git a/jdk/src/share/native/sun/java2d/Trace.h b/jdk/src/share/native/sun/java2d/Trace.h
index bc2596db013..e4c3bde35ea 100644
--- a/jdk/src/share/native/sun/java2d/Trace.h
+++ b/jdk/src/share/native/sun/java2d/Trace.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -62,12 +62,18 @@ J2dTraceInit();
#define J2dTrace3(level, string, arg1, arg2, arg3)
#define J2dTrace4(level, string, arg1, arg2, arg3, arg4)
#define J2dTrace5(level, string, arg1, arg2, arg3, arg4, arg5)
+#define J2dTrace6(level, string, arg1, arg2, arg3, arg4, arg5, arg6)
+#define J2dTrace7(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7)
+#define J2dTrace8(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8)
#define J2dTraceLn(level, string)
#define J2dTraceLn1(level, string, arg1)
#define J2dTraceLn2(level, string, arg1, arg2)
#define J2dTraceLn3(level, string, arg1, arg2, arg3)
#define J2dTraceLn4(level, string, arg1, arg2, arg3, arg4)
#define J2dTraceLn5(level, string, arg1, arg2, arg3, arg4, arg5)
+#define J2dTraceLn6(level, string, arg1, arg2, arg3, arg4, arg5, arg6)
+#define J2dTraceLn7(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7)
+#define J2dTraceLn8(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8)
#else /* DEBUG */
#define J2dTrace(level, string) { \
J2dTraceImpl(level, JNI_FALSE, string); \
@@ -87,6 +93,15 @@ J2dTraceInit();
#define J2dTrace5(level, string, arg1, arg2, arg3, arg4, arg5) { \
J2dTraceImpl(level, JNI_FALSE, string, arg1, arg2, arg3, arg4, arg5); \
}
+#define J2dTrace6(level, string, arg1, arg2, arg3, arg4, arg5, arg6) { \
+ J2dTraceImpl(level, JNI_FALSE, string, arg1, arg2, arg3, arg4, arg5, arg6); \
+ }
+#define J2dTrace7(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7) { \
+ J2dTraceImpl(level, JNI_FALSE, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7); \
+ }
+#define J2dTrace8(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8) { \
+ J2dTraceImpl(level, JNI_FALSE, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8); \
+ }
#define J2dTraceLn(level, string) { \
J2dTraceImpl(level, JNI_TRUE, string); \
}
@@ -105,6 +120,15 @@ J2dTraceInit();
#define J2dTraceLn5(level, string, arg1, arg2, arg3, arg4, arg5) { \
J2dTraceImpl(level, JNI_TRUE, string, arg1, arg2, arg3, arg4, arg5); \
}
+#define J2dTraceLn6(level, string, arg1, arg2, arg3, arg4, arg5, arg6) { \
+ J2dTraceImpl(level, JNI_TRUE, string, arg1, arg2, arg3, arg4, arg5, arg6); \
+ }
+#define J2dTraceLn7(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7) { \
+ J2dTraceImpl(level, JNI_TRUE, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7); \
+ }
+#define J2dTraceLn8(level, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8) { \
+ J2dTraceImpl(level, JNI_TRUE, string, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8); \
+ }
#endif /* DEBUG */
diff --git a/jdk/src/share/native/sun/java2d/loops/BlitBg.c b/jdk/src/share/native/sun/java2d/loops/BlitBg.c
index c5bccf24a84..892d5660683 100644
--- a/jdk/src/share/native/sun/java2d/loops/BlitBg.c
+++ b/jdk/src/share/native/sun/java2d/loops/BlitBg.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2000-2003 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2000-2008 Sun Microsystems, Inc. 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
@@ -31,12 +31,12 @@
/*
* Class: sun_java2d_loops_BlitBg
* Method: BlitBg
- * Signature: (Lsun/java2d/SurfaceData;Lsun/java2d/SurfaceData;Ljava/awt/Composite;Ljava/awt/Color;IIIIII)V
+ * Signature: (Lsun/java2d/SurfaceData;Lsun/java2d/SurfaceData;Ljava/awt/Composite;IIIIIII)V
*/
JNIEXPORT void JNICALL Java_sun_java2d_loops_BlitBg_BlitBg
(JNIEnv *env, jobject self,
jobject srcData, jobject dstData,
- jobject comp, jobject clip, jobject bgColor,
+ jobject comp, jobject clip, jint bgColor,
jint srcx, jint srcy, jint dstx, jint dsty, jint width, jint height)
{
SurfaceDataOps *srcOps;
@@ -93,7 +93,7 @@ JNIEXPORT void JNICALL Java_sun_java2d_loops_BlitBg_BlitBg
Region_IntersectBounds(&clipInfo, &dstInfo.bounds);
if (!Region_IsEmpty(&clipInfo)) {
- jint bgpixel = GrPrim_ColorGetRGB(env, bgColor);
+ jint bgpixel = bgColor;
srcOps->GetRasInfo(env, srcOps, &srcInfo);
dstOps->GetRasInfo(env, dstOps, &dstInfo);
if (pPrim->pDstType->pixelFor) {
diff --git a/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.c b/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.c
index a36a2885747..d6c100c0fe4 100644
--- a/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.c
+++ b/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2000-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2000-2008 Sun Microsystems, Inc. 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
@@ -50,7 +50,6 @@ static jfieldID eargbID;
static jfieldID clipRegionID;
static jfieldID compositeID;
static jfieldID lcdTextContrastID;
-static jfieldID valueID;
static jfieldID xorPixelID;
static jfieldID xorColorID;
static jfieldID alphaMaskID;
@@ -64,6 +63,8 @@ static jfieldID m10ID;
static jfieldID m11ID;
static jfieldID m12ID;
+static jmethodID getRgbID;
+
static jboolean InitPrimTypes(JNIEnv *env);
static jboolean InitSurfaceTypes(JNIEnv *env, jclass SurfaceType);
static jboolean InitCompositeTypes(JNIEnv *env, jclass CompositeType);
@@ -114,7 +115,7 @@ Java_sun_java2d_loops_GraphicsPrimitiveMgr_initIDs
"Ljava/awt/Composite;");
lcdTextContrastID =
(*env)->GetFieldID(env, SG2D, "lcdTextContrast", "I");
- valueID = (*env)->GetFieldID(env, Color, "value", "I");
+ getRgbID = (*env)->GetMethodID(env, Color, "getRGB", "()I");
xorPixelID = (*env)->GetFieldID(env, XORComp, "xorPixel", "I");
xorColorID = (*env)->GetFieldID(env, XORComp, "xorColor",
"Ljava/awt/Color;");
@@ -138,7 +139,6 @@ Java_sun_java2d_loops_GraphicsPrimitiveMgr_initIDs
sg2dStrokeHintID = (*env)->GetFieldID(env, SG2D, "strokeHint", "I");
fid = (*env)->GetStaticFieldID(env, SHints, "INTVAL_STROKE_PURE", "I");
sunHints_INTVAL_STROKE_PURE = (*env)->GetStaticIntField(env, SHints, fid);
-
}
void GrPrim_RefineBounds(SurfaceDataBounds *bounds, jint transX, jint transY,
@@ -467,7 +467,7 @@ GrPrim_CompGetXorColor(JNIEnv *env, jobject comp)
jint rgb;
color = (*env)->GetObjectField(env, comp, xorColorID);
- rgb = (*env)->GetIntField(env, color, valueID);
+ rgb = (*env)->CallIntMethod(env, color, getRgbID);
(*env)->DeleteLocalRef(env, color);
return rgb;
@@ -492,11 +492,6 @@ GrPrim_Sg2dGetEaRGB(JNIEnv *env, jobject sg2d)
return (*env)->GetIntField(env, sg2d, eargbID);
}
-jint GrPrim_ColorGetRGB(JNIEnv *env, jobject color)
-{
- return (*env)->GetIntField(env, color, valueID);
-}
-
JNIEXPORT jint JNICALL
GrPrim_Sg2dGetLCDTextContrast(JNIEnv *env, jobject sg2d)
{
diff --git a/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.h b/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.h
index 2d567632290..c3393cc4562 100644
--- a/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.h
+++ b/jdk/src/share/native/sun/java2d/loops/GraphicsPrimitiveMgr.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2000-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2000-2008 Sun Microsystems, Inc. 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
@@ -501,7 +501,6 @@ extern JNIEXPORT jint JNICALL
GrPrim_Sg2dGetEaRGB(JNIEnv *env, jobject sg2d);
extern JNIEXPORT jint JNICALL
GrPrim_Sg2dGetLCDTextContrast(JNIEnv *env, jobject sg2d);
-extern jint GrPrim_ColorGetRGB(JNIEnv *env, jobject color);
/*
* Data structure and functions to retrieve and use
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLContext.c b/jdk/src/share/native/sun/java2d/opengl/OGLContext.c
index 33bc3e82700..3a845b0f920 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLContext.c
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLContext.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -852,28 +852,58 @@ OGLContext_GetExtensionInfo(JNIEnv *env, jint *caps)
J2dTraceLn(J2D_TRACE_INFO, "OGLContext_GetExtensionInfo");
+ *caps |= CAPS_TEXNONSQUARE;
if (OGLContext_IsExtensionAvailable(e, "GL_ARB_multitexture")) {
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_MULTITEXTURE;
+ *caps |= CAPS_MULTITEXTURE;
}
if (OGLContext_IsExtensionAvailable(e, "GL_ARB_texture_non_power_of_two")){
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_TEXNONPOW2;
+ *caps |= CAPS_TEXNONPOW2;
}
- if (OGLContext_IsExtensionAvailable(e, "GL_ARB_texture_rectangle")) {
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_TEXRECT;
+ // 6656574: Use of the GL_ARB_texture_rectangle extension by Java 2D
+ // complicates any third-party libraries that try to interact with
+ // the OGL pipeline (and we've run into driver bugs in the past related
+ // to this extension), so for now we will disable its use by default (unless
+ // forced). We will still make use of the GL_ARB_texture_non_power_of_two
+ // extension when available, which is the better choice going forward
+ // anyway.
+ if (OGLContext_IsExtensionAvailable(e, "GL_ARB_texture_rectangle") &&
+ getenv("J2D_OGL_TEXRECT") != NULL)
+ {
+ *caps |= CAPS_EXT_TEXRECT;
}
if (OGLContext_IsFBObjectExtensionAvailable(env, e)) {
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_FBOBJECT;
+ *caps |= CAPS_EXT_FBOBJECT;
}
if (OGLContext_IsLCDShaderSupportAvailable(env, fragShaderAvail)) {
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_LCD_SHADER;
+ *caps |= CAPS_EXT_LCD_SHADER | CAPS_PS20;
}
if (OGLContext_IsBIOpShaderSupportAvailable(env, fragShaderAvail)) {
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_BIOP_SHADER;
+ *caps |= CAPS_EXT_BIOP_SHADER | CAPS_PS20;
}
if (OGLContext_IsGradShaderSupportAvailable(env, fragShaderAvail)) {
- *caps |= sun_java2d_opengl_OGLContext_CAPS_EXT_GRAD_SHADER;
+ *caps |= CAPS_EXT_GRAD_SHADER | CAPS_PS20;
+ }
+ if (OGLContext_IsExtensionAvailable(e, "GL_NV_fragment_program")) {
+ // this is an Nvidia board, at least PS 2.0, but we can't
+ // use the "max instructions" heuristic since GeForce FX
+ // boards report 1024 even though they're only PS 2.0,
+ // so we'll check the following, which does imply PS 3.0
+ if (OGLContext_IsExtensionAvailable(e, "GL_NV_fragment_program2")) {
+ *caps |= CAPS_PS30;
+ }
+ } else {
+ // for all other boards, we look at the "max instructions"
+ // count reported by the GL_ARB_fragment_program extension
+ // as a heuristic for detecting PS 3.0 compatible hardware
+ if (OGLContext_IsExtensionAvailable(e, "GL_ARB_fragment_program")) {
+ GLint instr;
+ j2d_glGetProgramivARB(GL_FRAGMENT_PROGRAM_ARB,
+ GL_MAX_PROGRAM_INSTRUCTIONS_ARB, &instr);
+ if (instr > 512) {
+ *caps |= CAPS_PS30;
+ }
+ }
}
-
// stuff vendor descriptor in the upper bits of the caps
if (vendor != NULL) {
if (strncmp(vendor, "ATI", 3) == 0) {
@@ -883,8 +913,10 @@ OGLContext_GetExtensionInfo(JNIEnv *env, jint *caps)
} else if (strncmp(vendor, "Sun", 3) == 0) {
vcap = OGLC_VENDOR_SUN;
}
+ // REMIND: new in 7 - check if needs fixing
+ *caps |= ((vcap & OGLC_VCAP_MASK) << OGLC_VCAP_OFFSET);
}
- *caps |= ((vcap & OGLC_VCAP_MASK) << OGLC_VCAP_OFFSET);
+
}
/**
@@ -983,4 +1015,49 @@ OGLContext_CreateFragmentProgram(const char *fragmentShaderSource)
return fragmentProgram;
}
+/*
+ * Class: sun_java2d_opengl_OGLContext
+ * Method: getOGLIdString
+ * Signature: ()Ljava/lang/String;
+ */
+JNIEXPORT jstring JNICALL Java_sun_java2d_opengl_OGLContext_getOGLIdString
+ (JNIEnv *env, jclass oglcc)
+{
+ char *vendor, *renderer, *version;
+ char *pAdapterId;
+ jobject ret = NULL;
+ int len;
+
+ J2dTraceLn(J2D_TRACE_INFO, "OGLContext_getOGLIdString");
+
+ vendor = (char*)j2d_glGetString(GL_VENDOR);
+ if (vendor == NULL) {
+ vendor = "Unknown Vendor";
+ }
+ renderer = (char*)j2d_glGetString(GL_RENDERER);
+ if (renderer == NULL) {
+ renderer = "Unknown Renderer";
+ }
+ version = (char*)j2d_glGetString(GL_VERSION);
+ if (version == NULL) {
+ version = "unknown version";
+ }
+
+ // 'vendor renderer (version)0'
+ len = strlen(vendor) + 1 + strlen(renderer) + 1 + 1+strlen(version)+1 + 1;
+ pAdapterId = malloc(len);
+ if (pAdapterId != NULL) {
+
+ jio_snprintf(pAdapterId, len, "%s %s (%s)", vendor, renderer, version);
+
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " id=%s", pAdapterId);
+
+ ret = JNU_NewStringPlatform(env, pAdapterId);
+
+ free(pAdapterId);
+ }
+
+ return ret;
+}
+
#endif /* !HEADLESS */
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLContext.h b/jdk/src/share/native/sun/java2d/opengl/OGLContext.h
index 0e0236f1d4b..78b89a53aa5 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLContext.h
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLContext.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -28,6 +28,7 @@
#include "sun_java2d_pipe_BufferedContext.h"
#include "sun_java2d_opengl_OGLContext.h"
+#include "sun_java2d_opengl_OGLContext_OGLContextCaps.h"
#include "j2d_md.h"
#include "J2D_GL/gl.h"
@@ -95,14 +96,51 @@ typedef struct {
#define OGLC_USE_MASK \
sun_java2d_pipe_BufferedContext_USE_MASK
+/**
+ * See OGLContext.java for more on these flags.
+ */
+#define CAPS_EMPTY \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_EMPTY
+#define CAPS_RT_PLAIN_ALPHA \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_RT_PLAIN_ALPHA
+#define CAPS_RT_TEXTURE_ALPHA \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_RT_TEXTURE_ALPHA
+#define CAPS_RT_TEXTURE_OPAQUE \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_RT_TEXTURE_OPAQUE
+#define CAPS_MULTITEXTURE \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_MULTITEXTURE
+#define CAPS_TEXNONPOW2 \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_TEXNONPOW2
+#define CAPS_TEXNONSQUARE \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_TEXNONSQUARE
+#define CAPS_PS20 \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_PS20
+#define CAPS_PS30 \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_PS30
+#define LAST_SHARED_CAP \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_LAST_SHARED_CAP
+#define CAPS_EXT_FBOBJECT \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_EXT_FBOBJECT
+#define CAPS_STORED_ALPHA \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_STORED_ALPHA
+#define CAPS_DOUBLEBUFFERED \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_DOUBLEBUFFERED
+#define CAPS_EXT_LCD_SHADER \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_EXT_LCD_SHADER
+#define CAPS_EXT_BIOP_SHADER \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_EXT_BIOP_SHADER
+#define CAPS_EXT_GRAD_SHADER \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_EXT_GRAD_SHADER
+#define CAPS_EXT_TEXRECT \
+ sun_java2d_opengl_OGLContext_OGLContextCaps_CAPS_EXT_TEXRECT
+
/**
* Evaluates to true if the given capability bitmask is present for the
* given OGLContext. Note that only the constant name needs to be passed as
* a parameter, as this macro will automatically prepend the full package
* and class name to the constant name.
*/
-#define OGLC_IS_CAP_PRESENT(oglc, cap) \
- (((oglc)->caps & (sun_java2d_opengl_OGLContext_##cap)) != 0)
+#define OGLC_IS_CAP_PRESENT(oglc, cap) (((oglc)->caps & (cap)) != 0)
/**
* At startup we will embed one of the following values in the caps field
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLFuncs.h b/jdk/src/share/native/sun/java2d/opengl/OGLFuncs.h
index 3d22a0c5c72..a63f35b8329 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLFuncs.h
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLFuncs.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -91,8 +91,12 @@ typedef void (GLAPIENTRY *glPixelStoreiType)(GLenum pname, GLint param);
typedef void (GLAPIENTRY *glPixelTransferfType)(GLenum pname, GLfloat param);
typedef void (GLAPIENTRY *glPixelZoomType)(GLfloat xfactor, GLfloat yfactor);
typedef void (GLAPIENTRY *glPolygonOffsetType)(GLfloat factor, GLfloat units);
+typedef void (GLAPIENTRY *glPopAttribType)(void);
+typedef void (GLAPIENTRY *glPopClientAttribType)(void);
typedef void (GLAPIENTRY *glPopMatrixType)(void);
typedef void (GLAPIENTRY *glPrioritizeTexturesType)(GLsizei n, const GLuint *textures, const GLclampf *priorities);
+typedef void (GLAPIENTRY *glPushAttribType)(GLbitfield);
+typedef void (GLAPIENTRY *glPushClientAttribType)(GLbitfield);
typedef void (GLAPIENTRY *glPushMatrixType)(void);
typedef void (GLAPIENTRY *glRasterPos2iType)(GLint x, GLint y);
typedef void (GLAPIENTRY *glReadBufferType)(GLenum mode);
@@ -155,8 +159,10 @@ typedef void (GLAPIENTRY *glUniform2fARBType)(GLint, GLfloat, GLfloat);
typedef void (GLAPIENTRY *glUniform3fARBType)(GLint, GLfloat, GLfloat, GLfloat);
typedef void (GLAPIENTRY *glUniform3fvARBType)(GLint, GLsizei, const GLfloat *);
typedef void (GLAPIENTRY *glUniform4fARBType)(GLint, GLfloat, GLfloat, GLfloat, GLfloat);
+typedef void (GLAPIENTRY *glUniform4fvARBType)(GLint, GLsizei, const GLfloat *);
typedef GLint (GLAPIENTRY *glGetUniformLocationARBType)(GLhandleARB, const GLcharARB *);
typedef void (GLAPIENTRY *glGetInfoLogARBType)(GLhandleARB, GLsizei, GLsizei *, GLcharARB *);
+typedef void (GLAPIENTRY *glGetProgramivARBType)(GLenum, GLenum, GLint *);
typedef void (GLAPIENTRY *glGetObjectParameterivARBType)(GLhandleARB, GLenum, GLint *);
typedef GLhandleARB (GLAPIENTRY *glCreateProgramObjectARBType)(void);
typedef void (GLAPIENTRY *glAttachObjectARBType)(GLhandleARB, GLhandleARB);
@@ -243,8 +249,12 @@ typedef void (GLAPIENTRY *glDeleteObjectARBType)(GLhandleARB);
OGL_##action##_FUNC(glPixelTransferf); \
OGL_##action##_FUNC(glPixelZoom); \
OGL_##action##_FUNC(glPolygonOffset); \
+ OGL_##action##_FUNC(glPopAttrib); \
+ OGL_##action##_FUNC(glPopClientAttrib); \
OGL_##action##_FUNC(glPopMatrix); \
OGL_##action##_FUNC(glPrioritizeTextures); \
+ OGL_##action##_FUNC(glPushAttrib); \
+ OGL_##action##_FUNC(glPushClientAttrib); \
OGL_##action##_FUNC(glPushMatrix); \
OGL_##action##_FUNC(glRasterPos2i); \
OGL_##action##_FUNC(glReadBuffer); \
@@ -298,7 +308,9 @@ typedef void (GLAPIENTRY *glDeleteObjectARBType)(GLhandleARB);
OGL_##action##_EXT_FUNC(glUniform3fARB); \
OGL_##action##_EXT_FUNC(glUniform3fvARB); \
OGL_##action##_EXT_FUNC(glUniform4fARB); \
+ OGL_##action##_EXT_FUNC(glUniform4fvARB); \
OGL_##action##_EXT_FUNC(glGetUniformLocationARB); \
+ OGL_##action##_EXT_FUNC(glGetProgramivARB); \
OGL_##action##_EXT_FUNC(glGetInfoLogARB); \
OGL_##action##_EXT_FUNC(glGetObjectParameterivARB); \
OGL_##action##_EXT_FUNC(glDeleteObjectARB);
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.c b/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.c
index 0ded24d51e0..ffd0ebcb4e5 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.c
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -88,8 +88,9 @@ Java_sun_java2d_opengl_OGLRenderQueue_flushBuffer
while (b < end) {
jint opcode = NEXT_INT(b);
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "OGLRenderQueue_flushBuffer: opcode=%d", opcode);
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ "OGLRenderQueue_flushBuffer: opcode=%d, rem=%d",
+ opcode, (end-b));
switch (opcode) {
@@ -148,6 +149,40 @@ Java_sun_java2d_opengl_OGLRenderQueue_flushBuffer
SKIP_BYTES(b, count * BYTES_PER_SCANLINE);
}
break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_PARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+ jfloat lwr21 = NEXT_FLOAT(b);
+ jfloat lwr12 = NEXT_FLOAT(b);
+ OGLRenderer_DrawParallelogram(oglc,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12,
+ lwr21, lwr12);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_AAPARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+ jfloat lwr21 = NEXT_FLOAT(b);
+ jfloat lwr12 = NEXT_FLOAT(b);
+ OGLRenderer_DrawAAParallelogram(oglc, dstOps,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12,
+ lwr21, lwr12);
+ }
+ break;
// fill ops
case sun_java2d_pipe_BufferedOpCodes_FILL_RECT:
@@ -166,6 +201,34 @@ Java_sun_java2d_opengl_OGLRenderQueue_flushBuffer
SKIP_BYTES(b, count * BYTES_PER_SPAN);
}
break;
+ case sun_java2d_pipe_BufferedOpCodes_FILL_PARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+ OGLRenderer_FillParallelogram(oglc,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_FILL_AAPARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+ OGLRenderer_FillAAParallelogram(oglc, dstOps,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12);
+ }
+ break;
// text-related ops
case sun_java2d_pipe_BufferedOpCodes_DRAW_GLYPH_LIST:
@@ -438,6 +501,31 @@ Java_sun_java2d_opengl_OGLRenderQueue_flushBuffer
dstOps = NULL;
}
break;
+ case sun_java2d_pipe_BufferedOpCodes_SAVE_STATE:
+ {
+ j2d_glPushAttrib(GL_ALL_ATTRIB_BITS);
+ j2d_glPushClientAttrib(GL_CLIENT_ALL_ATTRIB_BITS);
+ j2d_glMatrixMode(GL_MODELVIEW);
+ j2d_glPushMatrix();
+ j2d_glMatrixMode(GL_PROJECTION);
+ j2d_glPushMatrix();
+ j2d_glMatrixMode(GL_TEXTURE);
+ j2d_glPushMatrix();
+ }
+ break;
+
+ case sun_java2d_pipe_BufferedOpCodes_RESTORE_STATE:
+ {
+ j2d_glPopAttrib();
+ j2d_glPopClientAttrib();
+ j2d_glMatrixMode(GL_MODELVIEW);
+ j2d_glPopMatrix();
+ j2d_glMatrixMode(GL_PROJECTION);
+ j2d_glPopMatrix();
+ j2d_glMatrixMode(GL_TEXTURE);
+ j2d_glPopMatrix();
+ }
+ break;
case sun_java2d_pipe_BufferedOpCodes_SYNC:
{
sync = JNI_TRUE;
@@ -691,6 +779,9 @@ OGLRenderQueue_CheckPreviousOp(jint op)
return;
}
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ "OGLRenderQueue_CheckPreviousOp: new op=%d", op);
+
switch (previousOp) {
case GL_TEXTURE_2D:
case GL_TEXTURE_RECTANGLE_ARB:
@@ -718,6 +809,9 @@ OGLRenderQueue_CheckPreviousOp(jint op)
case OGL_STATE_GLYPH_OP:
OGLTR_DisableGlyphVertexCache(oglc);
break;
+ case OGL_STATE_PGRAM_OP:
+ OGLRenderer_DisableAAParallelogramProgram();
+ break;
case OGL_STATE_RESET:
case OGL_STATE_CHANGE:
// No-op
@@ -745,6 +839,9 @@ OGLRenderQueue_CheckPreviousOp(jint op)
case OGL_STATE_GLYPH_OP:
OGLTR_EnableGlyphVertexCache(oglc);
break;
+ case OGL_STATE_PGRAM_OP:
+ OGLRenderer_EnableAAParallelogramProgram();
+ break;
case OGL_STATE_RESET:
case OGL_STATE_CHANGE:
// No-op
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.h b/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.h
index e70d740158c..512f6292acd 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.h
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLRenderQueue.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -89,6 +89,13 @@
*/
#define OGL_STATE_GLYPH_OP -4
+/*
+ * Parameter passed to the CHECK_PREVIOUS_OP() macro to indicate that the
+ * following operation represents an operation that renders a
+ * parallelogram via a fragment program (see OGLRenderer).
+ */
+#define OGL_STATE_PGRAM_OP -5
+
/*
* Initializes the "previous operation" state to its default value.
*/
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.c b/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.c
index 93b1a6e65dc..ac844b3eb6e 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.c
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -27,6 +27,7 @@
#include
#include
+#include
#include "sun_java2d_opengl_OGLRenderer.h"
@@ -326,4 +327,486 @@ OGLRenderer_FillSpans(OGLContext *oglc, jint spanCount, jint *spans)
}
}
+#define FILL_PGRAM(fx11, fy11, dx21, dy21, dx12, dy12) \
+ do { \
+ j2d_glVertex2f(fx11, fy11); \
+ j2d_glVertex2f(fx11 + dx21, fy11 + dy21); \
+ j2d_glVertex2f(fx11 + dx21 + dx12, fy11 + dy21 + dy12); \
+ j2d_glVertex2f(fx11 + dx12, fy11 + dy12); \
+ } while (0)
+
+void
+OGLRenderer_FillParallelogram(OGLContext *oglc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12)
+{
+ J2dTraceLn6(J2D_TRACE_INFO,
+ "OGLRenderer_FillParallelogram "
+ "(x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f)",
+ fx11, fy11,
+ dx21, dy21,
+ dx12, dy12);
+
+ RETURN_IF_NULL(oglc);
+
+ CHECK_PREVIOUS_OP(GL_QUADS);
+
+ FILL_PGRAM(fx11, fy11, dx21, dy21, dx12, dy12);
+}
+
+void
+OGLRenderer_DrawParallelogram(OGLContext *oglc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lwr21, jfloat lwr12)
+{
+ // dx,dy for line width in the "21" and "12" directions.
+ jfloat ldx21 = dx21 * lwr21;
+ jfloat ldy21 = dy21 * lwr21;
+ jfloat ldx12 = dx12 * lwr12;
+ jfloat ldy12 = dy12 * lwr12;
+
+ // calculate origin of the outer parallelogram
+ jfloat ox11 = fx11 - (ldx21 + ldx12) / 2.0f;
+ jfloat oy11 = fy11 - (ldy21 + ldy12) / 2.0f;
+
+ J2dTraceLn8(J2D_TRACE_INFO,
+ "OGLRenderer_DrawParallelogram "
+ "(x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f lwr1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f lwr2=%6.2f)",
+ fx11, fy11,
+ dx21, dy21, lwr21,
+ dx12, dy12, lwr12);
+
+ RETURN_IF_NULL(oglc);
+
+ CHECK_PREVIOUS_OP(GL_QUADS);
+
+ // Only need to generate 4 quads if the interior still
+ // has a hole in it (i.e. if the line width ratio was
+ // less than 1.0)
+ if (lwr21 < 1.0f && lwr12 < 1.0f) {
+ // Note: "TOP", "BOTTOM", "LEFT" and "RIGHT" here are
+ // relative to whether the dxNN variables are positive
+ // and negative. The math works fine regardless of
+ // their signs, but for conceptual simplicity the
+ // comments will refer to the sides as if the dxNN
+ // were all positive. "TOP" and "BOTTOM" segments
+ // are defined by the dxy21 deltas. "LEFT" and "RIGHT"
+ // segments are defined by the dxy12 deltas.
+
+ // Each segment includes its starting corner and comes
+ // to just short of the following corner. Thus, each
+ // corner is included just once and the only lengths
+ // needed are the original parallelogram delta lengths
+ // and the "line width deltas". The sides will cover
+ // the following relative territories:
+ //
+ // T T T T T R
+ // L R
+ // L R
+ // L R
+ // L R
+ // L B B B B B
+
+ // TOP segment, to left side of RIGHT edge
+ // "width" of original pgram, "height" of hor. line size
+ fx11 = ox11;
+ fy11 = oy11;
+ FILL_PGRAM(fx11, fy11, dx21, dy21, ldx12, ldy12);
+
+ // RIGHT segment, to top of BOTTOM edge
+ // "width" of vert. line size , "height" of original pgram
+ fx11 = ox11 + dx21;
+ fy11 = oy11 + dy21;
+ FILL_PGRAM(fx11, fy11, ldx21, ldy21, dx12, dy12);
+
+ // BOTTOM segment, from right side of LEFT edge
+ // "width" of original pgram, "height" of hor. line size
+ fx11 = ox11 + dx12 + ldx21;
+ fy11 = oy11 + dy12 + ldy21;
+ FILL_PGRAM(fx11, fy11, dx21, dy21, ldx12, ldy12);
+
+ // LEFT segment, from bottom of TOP edge
+ // "width" of vert. line size , "height" of inner pgram
+ fx11 = ox11 + ldx12;
+ fy11 = oy11 + ldy12;
+ FILL_PGRAM(fx11, fy11, ldx21, ldy21, dx12, dy12);
+ } else {
+ // The line width ratios were large enough to consume
+ // the entire hole in the middle of the parallelogram
+ // so we can just issue one large quad for the outer
+ // parallelogram.
+ dx21 += ldx21;
+ dy21 += ldy21;
+ dx12 += ldx12;
+ dy12 += ldy12;
+ FILL_PGRAM(ox11, oy11, dx21, dy21, dx12, dy12);
+ }
+}
+
+static GLhandleARB aaPgramProgram = 0;
+
+/*
+ * This shader fills the space between an outer and inner parallelogram.
+ * It can be used to draw an outline by specifying both inner and outer
+ * values. It fills pixels by estimating what portion falls inside the
+ * outer shape, and subtracting an estimate of what portion falls inside
+ * the inner shape. Specifying both inner and outer values produces a
+ * standard "wide outline". Specifying an inner shape that falls far
+ * outside the outer shape allows the same shader to fill the outer
+ * shape entirely since pixels that fall within the outer shape are never
+ * inside the inner shape and so they are filled based solely on their
+ * coverage of the outer shape.
+ *
+ * The setup code renders this shader over the bounds of the outer
+ * shape (or the only shape in the case of a fill operation) and
+ * sets the texture 0 coordinates so that 0,0=>0,1=>1,1=>1,0 in those
+ * texture coordinates map to the four corners of the parallelogram.
+ * Similarly the texture 1 coordinates map the inner shape to the
+ * unit square as well, but in a different coordinate system.
+ *
+ * When viewed in the texture coordinate systems the parallelograms
+ * we are filling are unit squares, but the pixels have then become
+ * tiny parallelograms themselves. Both of the texture coordinate
+ * systems are affine transforms so the rate of change in X and Y
+ * of the texture coordinates are essentially constants and happen
+ * to correspond to the size and direction of the slanted sides of
+ * the distorted pixels relative to the "square mapped" boundary
+ * of the parallelograms.
+ *
+ * The shader uses the dFdx() and dFdy() functions to measure the "rate
+ * of change" of these texture coordinates and thus gets an accurate
+ * measure of the size and shape of a pixel relative to the two
+ * parallelograms. It then uses the bounds of the size and shape
+ * of a pixel to intersect with the unit square to estimate the
+ * coverage of the pixel. Unfortunately, without a lot more work
+ * to calculate the exact area of intersection between a unit
+ * square (the original parallelogram) and a parallelogram (the
+ * distorted pixel), this shader only approximates the pixel
+ * coverage, but emperically the estimate is very useful and
+ * produces visually pleasing results, if not theoretically accurate.
+ */
+static const char *aaPgramShaderSource =
+ "void main() {"
+ // Calculate the vectors for the "legs" of the pixel parallelogram
+ // for the outer parallelogram.
+ " vec2 oleg1 = dFdx(gl_TexCoord[0].st);"
+ " vec2 oleg2 = dFdy(gl_TexCoord[0].st);"
+ // Calculate the bounds of the distorted pixel parallelogram.
+ " vec2 corner = gl_TexCoord[0].st - (oleg1+oleg2)/2.0;"
+ " vec2 omin = min(corner, corner+oleg1);"
+ " omin = min(omin, corner+oleg2);"
+ " omin = min(omin, corner+oleg1+oleg2);"
+ " vec2 omax = max(corner, corner+oleg1);"
+ " omax = max(omax, corner+oleg2);"
+ " omax = max(omax, corner+oleg1+oleg2);"
+ // Calculate the vectors for the "legs" of the pixel parallelogram
+ // for the inner parallelogram.
+ " vec2 ileg1 = dFdx(gl_TexCoord[1].st);"
+ " vec2 ileg2 = dFdy(gl_TexCoord[1].st);"
+ // Calculate the bounds of the distorted pixel parallelogram.
+ " corner = gl_TexCoord[1].st - (ileg1+ileg2)/2.0;"
+ " vec2 imin = min(corner, corner+ileg1);"
+ " imin = min(imin, corner+ileg2);"
+ " imin = min(imin, corner+ileg1+ileg2);"
+ " vec2 imax = max(corner, corner+ileg1);"
+ " imax = max(imax, corner+ileg2);"
+ " imax = max(imax, corner+ileg1+ileg2);"
+ // Clamp the bounds of the parallelograms to the unit square to
+ // estimate the intersection of the pixel parallelogram with
+ // the unit square. The ratio of the 2 rectangle areas is a
+ // reasonable estimate of the proportion of coverage.
+ " vec2 o1 = clamp(omin, 0.0, 1.0);"
+ " vec2 o2 = clamp(omax, 0.0, 1.0);"
+ " float oint = (o2.y-o1.y)*(o2.x-o1.x);"
+ " float oarea = (omax.y-omin.y)*(omax.x-omin.x);"
+ " vec2 i1 = clamp(imin, 0.0, 1.0);"
+ " vec2 i2 = clamp(imax, 0.0, 1.0);"
+ " float iint = (i2.y-i1.y)*(i2.x-i1.x);"
+ " float iarea = (imax.y-imin.y)*(imax.x-imin.x);"
+ // Proportion of pixel in outer shape minus the proportion
+ // of pixel in the inner shape == the coverage of the pixel
+ // in the area between the two.
+ " float coverage = oint/oarea - iint / iarea;"
+ " gl_FragColor = gl_Color * coverage;"
+ "}";
+
+#define ADJUST_PGRAM(V1, DV, V2) \
+ do { \
+ if ((DV) >= 0) { \
+ (V2) += (DV); \
+ } else { \
+ (V1) += (DV); \
+ } \
+ } while (0)
+
+// Invert the following transform:
+// DeltaT(0, 0) == (0, 0)
+// DeltaT(1, 0) == (DX1, DY1)
+// DeltaT(0, 1) == (DX2, DY2)
+// DeltaT(1, 1) == (DX1+DX2, DY1+DY2)
+// TM00 = DX1, TM01 = DX2, (TM02 = X11)
+// TM10 = DY1, TM11 = DY2, (TM12 = Y11)
+// Determinant = TM00*TM11 - TM01*TM10
+// = DX1*DY2 - DX2*DY1
+// Inverse is:
+// IM00 = TM11/det, IM01 = -TM01/det
+// IM10 = -TM10/det, IM11 = TM00/det
+// IM02 = (TM01 * TM12 - TM11 * TM02) / det,
+// IM12 = (TM10 * TM02 - TM00 * TM12) / det,
+
+#define DECLARE_MATRIX(MAT) \
+ jfloat MAT ## 00, MAT ## 01, MAT ## 02, MAT ## 10, MAT ## 11, MAT ## 12
+
+#define GET_INVERTED_MATRIX(MAT, X11, Y11, DX1, DY1, DX2, DY2, RET_CODE) \
+ do { \
+ jfloat det = DX1*DY2 - DX2*DY1; \
+ if (det == 0) { \
+ RET_CODE; \
+ } \
+ MAT ## 00 = DY2/det; \
+ MAT ## 01 = -DX2/det; \
+ MAT ## 10 = -DY1/det; \
+ MAT ## 11 = DX1/det; \
+ MAT ## 02 = (DX2 * Y11 - DY2 * X11) / det; \
+ MAT ## 12 = (DY1 * X11 - DX1 * Y11) / det; \
+ } while (0)
+
+#define TRANSFORM(MAT, TX, TY, X, Y) \
+ do { \
+ TX = (X) * MAT ## 00 + (Y) * MAT ## 01 + MAT ## 02; \
+ TY = (X) * MAT ## 10 + (Y) * MAT ## 11 + MAT ## 12; \
+ } while (0)
+
+void
+OGLRenderer_FillAAParallelogram(OGLContext *oglc, OGLSDOps *dstOps,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12)
+{
+ DECLARE_MATRIX(om);
+ // parameters for parallelogram bounding box
+ jfloat bx11, by11, bx22, by22;
+ // parameters for uv texture coordinates of parallelogram corners
+ jfloat u11, v11, u12, v12, u21, v21, u22, v22;
+
+ J2dTraceLn6(J2D_TRACE_INFO,
+ "OGLRenderer_FillAAParallelogram "
+ "(x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f)",
+ fx11, fy11,
+ dx21, dy21,
+ dx12, dy12);
+
+ RETURN_IF_NULL(oglc);
+ RETURN_IF_NULL(dstOps);
+
+ GET_INVERTED_MATRIX(om, fx11, fy11, dx21, dy21, dx12, dy12,
+ return);
+
+ CHECK_PREVIOUS_OP(OGL_STATE_PGRAM_OP);
+
+ bx11 = bx22 = fx11;
+ by11 = by22 = fy11;
+ ADJUST_PGRAM(bx11, dx21, bx22);
+ ADJUST_PGRAM(by11, dy21, by22);
+ ADJUST_PGRAM(bx11, dx12, bx22);
+ ADJUST_PGRAM(by11, dy12, by22);
+ bx11 = (jfloat) floor(bx11);
+ by11 = (jfloat) floor(by11);
+ bx22 = (jfloat) ceil(bx22);
+ by22 = (jfloat) ceil(by22);
+
+ TRANSFORM(om, u11, v11, bx11, by11);
+ TRANSFORM(om, u21, v21, bx22, by11);
+ TRANSFORM(om, u12, v12, bx11, by22);
+ TRANSFORM(om, u22, v22, bx22, by22);
+
+ j2d_glBegin(GL_QUADS);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, u11, v11);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, 5.f, 5.f);
+ j2d_glVertex2f(bx11, by11);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, u21, v21);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, 6.f, 5.f);
+ j2d_glVertex2f(bx22, by11);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, u22, v22);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, 6.f, 6.f);
+ j2d_glVertex2f(bx22, by22);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, u12, v12);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, 5.f, 6.f);
+ j2d_glVertex2f(bx11, by22);
+ j2d_glEnd();
+}
+
+void
+OGLRenderer_FillAAParallelogramInnerOuter(OGLContext *oglc, OGLSDOps *dstOps,
+ jfloat ox11, jfloat oy11,
+ jfloat ox21, jfloat oy21,
+ jfloat ox12, jfloat oy12,
+ jfloat ix11, jfloat iy11,
+ jfloat ix21, jfloat iy21,
+ jfloat ix12, jfloat iy12)
+{
+ DECLARE_MATRIX(om);
+ DECLARE_MATRIX(im);
+ // parameters for parallelogram bounding box
+ jfloat bx11, by11, bx22, by22;
+ // parameters for uv texture coordinates of outer parallelogram corners
+ jfloat ou11, ov11, ou12, ov12, ou21, ov21, ou22, ov22;
+ // parameters for uv texture coordinates of inner parallelogram corners
+ jfloat iu11, iv11, iu12, iv12, iu21, iv21, iu22, iv22;
+
+ RETURN_IF_NULL(oglc);
+ RETURN_IF_NULL(dstOps);
+
+ GET_INVERTED_MATRIX(im, ix11, iy11, ix21, iy21, ix12, iy12,
+ // inner parallelogram is degenerate
+ // therefore it encloses no area
+ // fill outer
+ OGLRenderer_FillAAParallelogram(oglc, dstOps,
+ ox11, oy11,
+ ox21, oy21,
+ ox12, oy12);
+ return);
+ GET_INVERTED_MATRIX(om, ox11, oy11, ox21, oy21, ox12, oy12,
+ return);
+
+ CHECK_PREVIOUS_OP(OGL_STATE_PGRAM_OP);
+
+ bx11 = bx22 = ox11;
+ by11 = by22 = oy11;
+ ADJUST_PGRAM(bx11, ox21, bx22);
+ ADJUST_PGRAM(by11, oy21, by22);
+ ADJUST_PGRAM(bx11, ox12, bx22);
+ ADJUST_PGRAM(by11, oy12, by22);
+ bx11 = (jfloat) floor(bx11);
+ by11 = (jfloat) floor(by11);
+ bx22 = (jfloat) ceil(bx22);
+ by22 = (jfloat) ceil(by22);
+
+ TRANSFORM(om, ou11, ov11, bx11, by11);
+ TRANSFORM(om, ou21, ov21, bx22, by11);
+ TRANSFORM(om, ou12, ov12, bx11, by22);
+ TRANSFORM(om, ou22, ov22, bx22, by22);
+
+ TRANSFORM(im, iu11, iv11, bx11, by11);
+ TRANSFORM(im, iu21, iv21, bx22, by11);
+ TRANSFORM(im, iu12, iv12, bx11, by22);
+ TRANSFORM(im, iu22, iv22, bx22, by22);
+
+ j2d_glBegin(GL_QUADS);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, ou11, ov11);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, iu11, iv11);
+ j2d_glVertex2f(bx11, by11);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, ou21, ov21);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, iu21, iv21);
+ j2d_glVertex2f(bx22, by11);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, ou22, ov22);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, iu22, iv22);
+ j2d_glVertex2f(bx22, by22);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE0_ARB, ou12, ov12);
+ j2d_glMultiTexCoord2fARB(GL_TEXTURE1_ARB, iu12, iv12);
+ j2d_glVertex2f(bx11, by22);
+ j2d_glEnd();
+}
+
+void
+OGLRenderer_DrawAAParallelogram(OGLContext *oglc, OGLSDOps *dstOps,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lwr21, jfloat lwr12)
+{
+ // dx,dy for line width in the "21" and "12" directions.
+ jfloat ldx21, ldy21, ldx12, ldy12;
+ // parameters for "outer" parallelogram
+ jfloat ofx11, ofy11, odx21, ody21, odx12, ody12;
+ // parameters for "inner" parallelogram
+ jfloat ifx11, ify11, idx21, idy21, idx12, idy12;
+
+ J2dTraceLn8(J2D_TRACE_INFO,
+ "OGLRenderer_DrawAAParallelogram "
+ "(x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f lwr1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f lwr2=%6.2f)",
+ fx11, fy11,
+ dx21, dy21, lwr21,
+ dx12, dy12, lwr12);
+
+ RETURN_IF_NULL(oglc);
+ RETURN_IF_NULL(dstOps);
+
+ // calculate true dx,dy for line widths from the "line width ratios"
+ ldx21 = dx21 * lwr21;
+ ldy21 = dy21 * lwr21;
+ ldx12 = dx12 * lwr12;
+ ldy12 = dy12 * lwr12;
+
+ // calculate coordinates of the outer parallelogram
+ ofx11 = fx11 - (ldx21 + ldx12) / 2.0f;
+ ofy11 = fy11 - (ldy21 + ldy12) / 2.0f;
+ odx21 = dx21 + ldx21;
+ ody21 = dy21 + ldy21;
+ odx12 = dx12 + ldx12;
+ ody12 = dy12 + ldy12;
+
+ // Only process the inner parallelogram if the line width ratio
+ // did not consume the entire interior of the parallelogram
+ // (i.e. if the width ratio was less than 1.0)
+ if (lwr21 < 1.0f && lwr12 < 1.0f) {
+ // calculate coordinates of the inner parallelogram
+ ifx11 = fx11 + (ldx21 + ldx12) / 2.0f;
+ ify11 = fy11 + (ldy21 + ldy12) / 2.0f;
+ idx21 = dx21 - ldx21;
+ idy21 = dy21 - ldy21;
+ idx12 = dx12 - ldx12;
+ idy12 = dy12 - ldy12;
+
+ OGLRenderer_FillAAParallelogramInnerOuter(oglc, dstOps,
+ ofx11, ofy11,
+ odx21, ody21,
+ odx12, ody12,
+ ifx11, ify11,
+ idx21, idy21,
+ idx12, idy12);
+ } else {
+ OGLRenderer_FillAAParallelogram(oglc, dstOps,
+ ofx11, ofy11,
+ odx21, ody21,
+ odx12, ody12);
+ }
+}
+
+void
+OGLRenderer_EnableAAParallelogramProgram()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "OGLRenderer_EnableAAParallelogramProgram");
+
+ if (aaPgramProgram == 0) {
+ aaPgramProgram = OGLContext_CreateFragmentProgram(aaPgramShaderSource);
+ if (aaPgramProgram == 0) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "OGLRenderer_EnableAAParallelogramProgram: "
+ "error creating program");
+ return;
+ }
+ }
+ j2d_glUseProgramObjectARB(aaPgramProgram);
+}
+
+void
+OGLRenderer_DisableAAParallelogramProgram()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "OGLRenderer_DisableAAParallelogramProgram");
+
+ j2d_glUseProgramObjectARB(0);
+}
+
#endif /* !HEADLESS */
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.h b/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.h
index 6b97ba16637..207ee51335b 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.h
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLRenderer.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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
@@ -46,9 +46,31 @@ void OGLRenderer_DrawPoly(OGLContext *oglc,
jint *xPoints, jint *yPoints);
void OGLRenderer_DrawScanlines(OGLContext *oglc,
jint count, jint *scanlines);
+void OGLRenderer_DrawParallelogram(OGLContext *oglc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lw21, jfloat lw12);
+void OGLRenderer_DrawAAParallelogram(OGLContext *oglc, OGLSDOps *dstOps,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lw21, jfloat lw12);
+
void OGLRenderer_FillRect(OGLContext *oglc,
jint x, jint y, jint w, jint h);
void OGLRenderer_FillSpans(OGLContext *oglc,
jint count, jint *spans);
+void OGLRenderer_FillParallelogram(OGLContext *oglc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12);
+void OGLRenderer_FillAAParallelogram(OGLContext *oglc, OGLSDOps *dstOps,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12);
+
+void OGLRenderer_EnableAAParallelogramProgram();
+void OGLRenderer_DisableAAParallelogramProgram();
#endif /* OGLRenderer_h_Included */
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.c b/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.c
index fe9d37ac643..2b45a610be3 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.c
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -41,6 +41,8 @@ extern jlong OGLSD_GetNativeConfigInfo(OGLSDOps *oglsdo);
extern jboolean OGLSD_InitOGLWindow(JNIEnv *env, OGLSDOps *oglsdo);
extern void OGLSD_DestroyOGLSurface(JNIEnv *env, OGLSDOps *oglsdo);
+void OGLSD_SetNativeDimensions(JNIEnv *env, OGLSDOps *oglsdo, jint w, jint h);
+
/**
* This table contains the "pixel formats" for all system memory surfaces
* that OpenGL is capable of handling, indexed by the "PF_" constants defined
@@ -269,6 +271,9 @@ Java_sun_java2d_opengl_OGLSurfaceData_initTexture
return JNI_FALSE;
}
+ OGLSD_SetNativeDimensions(env, oglsdo,
+ oglsdo->textureWidth, oglsdo->textureHeight);
+
oglsdo->drawableType = OGLSD_TEXTURE;
// other fields (e.g. width, height) are set in OGLSD_InitTextureObject()
@@ -427,6 +432,9 @@ Java_sun_java2d_opengl_OGLSurfaceData_initFBObject
oglsdo->fbobjectID = fbobjectID;
oglsdo->depthID = depthID;
+ OGLSD_SetNativeDimensions(env, oglsdo,
+ oglsdo->textureWidth, oglsdo->textureHeight);
+
// framebuffer objects differ from other OpenGL surfaces in that the
// value passed to glRead/DrawBuffer() must be GL_COLOR_ATTACHMENTn_EXT,
// rather than GL_FRONT (or GL_BACK)
@@ -476,6 +484,8 @@ Java_sun_java2d_opengl_OGLSurfaceData_initFlipBackbuffer
// explicitly use BACK_LEFT rather than BACK...
oglsdo->activeBuffer = GL_BACK_LEFT;
+ OGLSD_SetNativeDimensions(env, oglsdo, oglsdo->width, oglsdo->height);
+
return JNI_TRUE;
}
@@ -497,6 +507,45 @@ Java_sun_java2d_opengl_OGLSurfaceData_getTextureTarget
return (jint)oglsdo->textureTarget;
}
+JNIEXPORT jint JNICALL
+Java_sun_java2d_opengl_OGLSurfaceData_getTextureID
+ (JNIEnv *env, jobject oglsd,
+ jlong pData)
+{
+ OGLSDOps *oglsdo = (OGLSDOps *)jlong_to_ptr(pData);
+
+ J2dTraceLn(J2D_TRACE_INFO, "OGLSurfaceData_getTextureID");
+
+ if (oglsdo == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "OGLSurfaceData_getTextureID: ops are null");
+ return 0L;
+ }
+
+ return (jint)oglsdo->textureID;
+}
+
+/**
+ * Initializes nativeWidth/Height fields of the surfaceData object with
+ * passed arguments.
+ */
+void
+OGLSD_SetNativeDimensions(JNIEnv *env, OGLSDOps *oglsdo,
+ jint width, jint height)
+{
+ jobject sdObject;
+
+ sdObject = (*env)->NewLocalRef(env, oglsdo->sdOps.sdObject);
+ if (sdObject == NULL) {
+ return;
+ }
+
+ JNU_SetFieldByName(env, NULL, sdObject, "nativeWidth", "I", width);
+ JNU_SetFieldByName(env, NULL, sdObject, "nativeHeight", "I", height);
+
+ (*env)->DeleteLocalRef(env, sdObject);
+}
+
/**
* Disposes of all native resources associated with this surface.
*/
diff --git a/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.h b/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.h
index 17205ce0654..553913bbbdc 100644
--- a/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.h
+++ b/jdk/src/share/native/sun/java2d/opengl/OGLSurfaceData.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -28,6 +28,7 @@
#include "java_awt_image_AffineTransformOp.h"
#include "sun_java2d_opengl_OGLSurfaceData.h"
+#include "sun_java2d_pipe_hw_AccelSurface.h"
#include "J2D_GL/gl.h"
#include "SurfaceData.h"
@@ -215,12 +216,12 @@ struct _OGLSDOps {
* These are shorthand names for the surface type constants defined in
* OGLSurfaceData.java.
*/
-#define OGLSD_UNDEFINED sun_java2d_opengl_OGLSurfaceData_UNDEFINED
-#define OGLSD_WINDOW sun_java2d_opengl_OGLSurfaceData_WINDOW
-#define OGLSD_PBUFFER sun_java2d_opengl_OGLSurfaceData_PBUFFER
-#define OGLSD_TEXTURE sun_java2d_opengl_OGLSurfaceData_TEXTURE
-#define OGLSD_FLIP_BACKBUFFER sun_java2d_opengl_OGLSurfaceData_FLIP_BACKBUFFER
-#define OGLSD_FBOBJECT sun_java2d_opengl_OGLSurfaceData_FBOBJECT
+#define OGLSD_UNDEFINED sun_java2d_pipe_hw_AccelSurface_UNDEFINED
+#define OGLSD_WINDOW sun_java2d_pipe_hw_AccelSurface_WINDOW
+#define OGLSD_PBUFFER sun_java2d_pipe_hw_AccelSurface_RT_PLAIN
+#define OGLSD_TEXTURE sun_java2d_pipe_hw_AccelSurface_TEXTURE
+#define OGLSD_FLIP_BACKBUFFER sun_java2d_pipe_hw_AccelSurface_FLIP_BACKBUFFER
+#define OGLSD_FBOBJECT sun_java2d_pipe_hw_AccelSurface_RT_TEXTURE
/**
* These are shorthand names for the filtering method constants used by
diff --git a/jdk/src/share/native/sun/java2d/pipe/BufferedMaskBlit.c b/jdk/src/share/native/sun/java2d/pipe/BufferedMaskBlit.c
index 42933cd0675..c11172bc68c 100644
--- a/jdk/src/share/native/sun/java2d/pipe/BufferedMaskBlit.c
+++ b/jdk/src/share/native/sun/java2d/pipe/BufferedMaskBlit.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -59,7 +59,6 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
{
SurfaceDataOps *srcOps = (SurfaceDataOps *)jlong_to_ptr(pSrcOps);
SurfaceDataRasInfo srcInfo;
- unsigned char *pMask;
unsigned char *bbuf;
jint *pBuf;
@@ -97,13 +96,6 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
return bpos;
}
- pMask = (*env)->GetPrimitiveArrayCritical(env, maskArray, 0);
- if (pMask == NULL) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "BufferedMaskBlit_enqueueTile: cannot lock mask array");
- return bpos;
- }
-
srcInfo.bounds.x1 = srcx;
srcInfo.bounds.y1 = srcy;
srcInfo.bounds.x2 = srcx + width;
@@ -112,8 +104,6 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
if (srcOps->Lock(env, srcOps, &srcInfo, SD_LOCK_READ) != SD_SUCCESS) {
J2dRlsTraceLn(J2D_TRACE_WARNING,
"BufferedMaskBlit_enqueueTile: could not acquire lock");
- (*env)->ReleasePrimitiveArrayCritical(env, maskArray,
- pMask, JNI_ABORT);
return bpos;
}
@@ -129,6 +119,15 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
PtrCoord(srcInfo.rasBase,
srcInfo.bounds.x1, srcInfo.pixelStride,
srcInfo.bounds.y1, srcInfo.scanStride);
+ unsigned char *pMask =
+ (*env)->GetPrimitiveArrayCritical(env, maskArray, 0);
+ if (pMask == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "BufferedMaskBlit_enqueueTile: cannot lock mask array");
+ SurfaceData_InvokeRelease(env, srcOps, &srcInfo);
+ SurfaceData_InvokeUnlock(env, srcOps, &srcInfo);
+ return bpos;
+ }
width = srcInfo.bounds.x2 - srcInfo.bounds.x1;
height = srcInfo.bounds.y2 - srcInfo.bounds.y1;
@@ -166,18 +165,22 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
do {
jint w = width;
do {
- jubyte pathA = *pMask++;
+ jint pathA = *pMask++;
if (!pathA) {
pBuf[0] = 0;
} else {
- jint cr, cg, cb, ca;
- jubyte r, g, b, a;
- LoadIntArgbTo4ByteArgb(pSrc, c, 0, ca, cr, cg, cb);
- a = MUL8(ca, pathA);
- r = MUL8(cr, a);
- g = MUL8(cg, a);
- b = MUL8(cb, a);
- pBuf[0] = (a << 24) | (r << 16) | (g << 8) | b;
+ jint pixel = pSrc[0];
+ if (pathA == 0xff && (pixel >> 24) + 1 == 0) {
+ pBuf[0] = pixel;
+ } else {
+ jint r, g, b, a;
+ ExtractIntDcmComponents1234(pixel, a, r, g, b);
+ a = MUL8(pathA, a);
+ r = MUL8(a, r);
+ g = MUL8(a, g);
+ b = MUL8(a, b);
+ pBuf[0] = (a << 24) | (r << 16) | (g << 8) | b;
+ }
}
pSrc = PtrAddBytes(pSrc, srcPixelStride);
pBuf++;
@@ -191,17 +194,17 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
do {
jint w = width;
do {
- jubyte pathA = *pMask++;
+ jint pathA = *pMask++;
if (!pathA) {
pBuf[0] = 0;
} else if (pathA == 0xff) {
pBuf[0] = pSrc[0];
} else {
- jubyte r, g, b, a;
- a = MUL8((pSrc[0] >> 24) & 0xff, pathA);
- r = MUL8((pSrc[0] >> 16) & 0xff, pathA);
- g = MUL8((pSrc[0] >> 8) & 0xff, pathA);
- b = MUL8((pSrc[0] >> 0) & 0xff, pathA);
+ jint r, g, b, a;
+ a = MUL8(pathA, (pSrc[0] >> 24) & 0xff);
+ r = MUL8(pathA, (pSrc[0] >> 16) & 0xff);
+ g = MUL8(pathA, (pSrc[0] >> 8) & 0xff);
+ b = MUL8(pathA, (pSrc[0] >> 0) & 0xff);
pBuf[0] = (a << 24) | (r << 16) | (g << 8) | b;
}
pSrc = PtrAddBytes(pSrc, srcPixelStride);
@@ -216,17 +219,18 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
do {
jint w = width;
do {
- jubyte pathA = *pMask++;
+ jint pathA = *pMask++;
if (!pathA) {
pBuf[0] = 0;
+ } else if (pathA == 0xff) {
+ pBuf[0] = pSrc[0] | 0xff000000;
} else {
- jint cr, cg, cb;
- jubyte r, g, b, a;
- LoadIntRgbTo3ByteRgb(pSrc, c, 0, cr, cg, cb);
+ jint r, g, b, a;
+ LoadIntRgbTo3ByteRgb(pSrc, c, 0, r, g, b);
a = pathA;
- r = MUL8(cr, a);
- g = MUL8(cg, a);
- b = MUL8(cb, a);
+ r = MUL8(a, r);
+ g = MUL8(a, g);
+ b = MUL8(a, b);
pBuf[0] = (a << 24) | (r << 16) | (g << 8) | b;
}
pSrc = PtrAddBytes(pSrc, srcPixelStride);
@@ -241,17 +245,16 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
do {
jint w = width;
do {
- jubyte pathA = *pMask++;
+ jint pathA = *pMask++;
if (!pathA) {
pBuf[0] = 0;
} else {
- jint cr, cg, cb;
- jubyte r, g, b, a;
- LoadIntBgrTo3ByteRgb(pSrc, c, 0, cr, cg, cb);
+ jint r, g, b, a;
+ LoadIntBgrTo3ByteRgb(pSrc, c, 0, r, g, b);
a = pathA;
- r = MUL8(cr, a);
- g = MUL8(cg, a);
- b = MUL8(cb, a);
+ r = MUL8(a, r);
+ g = MUL8(a, g);
+ b = MUL8(a, b);
pBuf[0] = (a << 24) | (r << 16) | (g << 8) | b;
}
pSrc = PtrAddBytes(pSrc, srcPixelStride);
@@ -269,14 +272,14 @@ Java_sun_java2d_pipe_BufferedMaskBlit_enqueueTile
// increment current byte position
bpos += width * height * sizeof(jint);
+
+ (*env)->ReleasePrimitiveArrayCritical(env, maskArray,
+ pMask, JNI_ABORT);
}
SurfaceData_InvokeRelease(env, srcOps, &srcInfo);
}
SurfaceData_InvokeUnlock(env, srcOps, &srcInfo);
- (*env)->ReleasePrimitiveArrayCritical(env, maskArray,
- pMask, JNI_ABORT);
-
// return the current byte position
return bpos;
}
diff --git a/jdk/src/solaris/classes/sun/awt/X11/XComponentPeer.java b/jdk/src/solaris/classes/sun/awt/X11/XComponentPeer.java
index 55b14ab65c6..651d10a2905 100644
--- a/jdk/src/solaris/classes/sun/awt/X11/XComponentPeer.java
+++ b/jdk/src/solaris/classes/sun/awt/X11/XComponentPeer.java
@@ -70,9 +70,12 @@ import sun.awt.*;
import sun.awt.event.IgnorePaintEvent;
import sun.awt.image.SunVolatileImage;
import sun.awt.image.ToolkitImage;
+import sun.java2d.BackBufferCapsProvider;
import sun.java2d.pipe.Region;
-public class XComponentPeer extends XWindow implements ComponentPeer, DropTargetPeer {
+public class XComponentPeer extends XWindow implements ComponentPeer, DropTargetPeer,
+ BackBufferCapsProvider
+{
/* FIX ME: these constants copied from java.awt.KeyboardFocusManager */
static final int SNFH_FAILURE = 0;
static final int SNFH_SUCCESS_HANDLED = 1;
@@ -1274,25 +1277,37 @@ public class XComponentPeer extends XWindow implements ComponentPeer, DropTarget
* native windowing system specific actions.
*/
+ private BufferCapabilities backBufferCaps;
+
public void createBuffers(int numBuffers, BufferCapabilities caps)
throws AWTException
{
if (buffersLog.isLoggable(Level.FINE)) {
buffersLog.fine("createBuffers(" + numBuffers + ", " + caps + ")");
}
+ // set the caps first, they're used when creating the bb
+ backBufferCaps = caps;
backBuffer = graphicsConfig.createBackBuffer(this, numBuffers, caps);
xBackBuffer = graphicsConfig.createBackBufferImage(target,
backBuffer);
}
- public void flip(BufferCapabilities.FlipContents flipAction) {
+ @Override
+ public BufferCapabilities getBackBufferCaps() {
+ return backBufferCaps;
+ }
+
+ public void flip(int x1, int y1, int x2, int y2,
+ BufferCapabilities.FlipContents flipAction)
+ {
if (buffersLog.isLoggable(Level.FINE)) {
buffersLog.fine("flip(" + flipAction + ")");
}
if (backBuffer == 0) {
throw new IllegalStateException("Buffers have not been created");
}
- graphicsConfig.flip(this, target, xBackBuffer, flipAction);
+ graphicsConfig.flip(this, target, xBackBuffer,
+ x1, y1, x2, y2, flipAction);
}
public Image getBackBuffer() {
diff --git a/jdk/src/solaris/classes/sun/awt/X11/XEmbedChildProxyPeer.java b/jdk/src/solaris/classes/sun/awt/X11/XEmbedChildProxyPeer.java
index b2a342757bd..e4500043ae1 100644
--- a/jdk/src/solaris/classes/sun/awt/X11/XEmbedChildProxyPeer.java
+++ b/jdk/src/solaris/classes/sun/awt/X11/XEmbedChildProxyPeer.java
@@ -257,7 +257,7 @@ public class XEmbedChildProxyPeer implements ComponentPeer, XEventDispatcher{
public void createBuffers(int numBuffers, BufferCapabilities caps)
throws AWTException { }
public Image getBackBuffer() { return null; }
- public void flip(BufferCapabilities.FlipContents flipAction) { }
+ public void flip(int x1, int y1, int x2, int y2, BufferCapabilities.FlipContents flipAction) { }
public void destroyBuffers() { }
/**
diff --git a/jdk/src/solaris/classes/sun/awt/X11GraphicsConfig.java b/jdk/src/solaris/classes/sun/awt/X11GraphicsConfig.java
index ef3c6995ea3..8ea5717ac2a 100644
--- a/jdk/src/solaris/classes/sun/awt/X11GraphicsConfig.java
+++ b/jdk/src/solaris/classes/sun/awt/X11GraphicsConfig.java
@@ -27,20 +27,17 @@ package sun.awt;
import java.awt.AWTException;
import java.awt.BufferCapabilities;
+import java.awt.BufferCapabilities.FlipContents;
import java.awt.Component;
import java.awt.Toolkit;
import java.awt.GraphicsConfiguration;
import java.awt.GraphicsDevice;
import java.awt.Image;
import java.awt.ImageCapabilities;
-import java.awt.image.DataBuffer;
import java.awt.Transparency;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.DirectColorModel;
-import java.awt.image.ImageProducer;
-import java.awt.image.IndexColorModel;
-import java.awt.image.Raster;
import java.awt.image.VolatileImage;
import java.awt.image.WritableRaster;
import java.awt.geom.AffineTransform;
@@ -429,6 +426,7 @@ public class X11GraphicsConfig extends GraphicsConfiguration
*/
public void flip(X11ComponentPeer peer,
Component target, VolatileImage xBackBuffer,
+ int x1, int y1, int x2, int y2,
BufferCapabilities.FlipContents flipAction)
{
long window = peer.getContentWindow();
diff --git a/jdk/src/solaris/classes/sun/awt/X11GraphicsDevice.java b/jdk/src/solaris/classes/sun/awt/X11GraphicsDevice.java
index 347302682cf..ed495b177ec 100644
--- a/jdk/src/solaris/classes/sun/awt/X11GraphicsDevice.java
+++ b/jdk/src/solaris/classes/sun/awt/X11GraphicsDevice.java
@@ -385,6 +385,9 @@ public class X11GraphicsDevice
throw new IllegalStateException("Must be in fullscreen mode " +
"in order to set display mode");
}
+ if (getDisplayMode().equals(dm)) {
+ return;
+ }
if (dm == null ||
(dm = getMatchingDisplayMode(dm)) == null)
{
diff --git a/jdk/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java b/jdk/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java
index fffa5c9eece..dadfe84d71f 100644
--- a/jdk/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java
+++ b/jdk/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java
@@ -44,6 +44,7 @@ import java.util.*;
import java.util.logging.*;
import sun.awt.motif.MFontConfiguration;
+import sun.font.FcFontConfiguration;
import sun.font.Font2D;
import sun.font.FontManager;
import sun.font.NativeFont;
@@ -350,6 +351,14 @@ public class X11GraphicsEnvironment
* only to get called for these fonts.
*/
public String getFileNameFromPlatformName(String platName) {
+
+ /* If the FontConfig file doesn't use xlfds, or its
+ * FcFontConfiguration, this may be already a file name.
+ */
+ if (platName.startsWith("/")) {
+ return platName;
+ }
+
String fileName = null;
String fontID = specificFontIDForName(platName);
@@ -905,12 +914,50 @@ public class X11GraphicsEnvironment
// Implements SunGraphicsEnvironment.createFontConfiguration.
protected FontConfiguration createFontConfiguration() {
- return new MFontConfiguration(this);
+
+ /* The logic here decides whether to use a preconfigured
+ * fontconfig.properties file, or synthesise one using platform APIs.
+ * On Solaris (as opposed to OpenSolaris) we try to use the
+ * pre-configured ones, but if the files it specifies are missing
+ * we fail-safe to synthesising one. This might happen if Solaris
+ * changes its fonts.
+ * For OpenSolaris I don't expect us to ever create fontconfig files,
+ * so it will always synthesise. Note that if we misidentify
+ * OpenSolaris as Solaris, then the test for the presence of
+ * Solaris-only font files will correct this.
+ * For Linux we require an exact match of distro and version to
+ * use the preconfigured file, and also that it points to
+ * existent fonts.
+ * If synthesising fails, we fall back to any preconfigured file
+ * and do the best we can. For the commercial JDK this will be
+ * fine as it includes the Lucida fonts. OpenJDK should not hit
+ * this as the synthesis should always work on its platforms.
+ */
+ FontConfiguration mFontConfig = new MFontConfiguration(this);
+ if (isOpenSolaris ||
+ (isLinux &&
+ (!mFontConfig.foundOsSpecificFile() ||
+ !mFontConfig.fontFilesArePresent()) ||
+ (isSolaris && !mFontConfig.fontFilesArePresent()))) {
+ FcFontConfiguration fcFontConfig =
+ new FcFontConfiguration(this);
+ if (fcFontConfig.init()) {
+ return fcFontConfig;
+ }
+ }
+ mFontConfig.init();
+ return mFontConfig;
}
public FontConfiguration
createFontConfiguration(boolean preferLocaleFonts,
boolean preferPropFonts) {
+ FontConfiguration config = getFontConfiguration();
+ if (config instanceof FcFontConfiguration) {
+ // Doesn't need to implement the alternate support.
+ return config;
+ }
+
return new MFontConfiguration(this,
preferLocaleFonts, preferPropFonts);
}
@@ -921,6 +968,7 @@ public class X11GraphicsEnvironment
* for this platform.
*/
public String getDefaultFontFaceName() {
+
return null;
}
diff --git a/jdk/src/solaris/classes/sun/awt/motif/MComponentPeer.java b/jdk/src/solaris/classes/sun/awt/motif/MComponentPeer.java
index 23c06c363c4..87f556c93d1 100644
--- a/jdk/src/solaris/classes/sun/awt/motif/MComponentPeer.java
+++ b/jdk/src/solaris/classes/sun/awt/motif/MComponentPeer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1995-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1995-2008 Sun Microsystems, Inc. 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
@@ -1138,11 +1138,14 @@ abstract class MComponentPeer implements ComponentPeer, DropTargetPeer, X11Compo
backBuffer);
}
- public void flip(BufferCapabilities.FlipContents flipAction) {
+ public void flip(int x1, int y1, int x2, int y2,
+ BufferCapabilities.FlipContents flipAction)
+ {
if (backBuffer == 0) {
throw new IllegalStateException("Buffers have not been created");
}
- graphicsConfig.flip(this, target, xBackBuffer, flipAction);
+ graphicsConfig.flip(this, target, xBackBuffer,
+ x1, y1, x2, y2, flipAction);
}
public Image getBackBuffer() {
diff --git a/jdk/src/solaris/classes/sun/font/FcFontConfiguration.java b/jdk/src/solaris/classes/sun/font/FcFontConfiguration.java
new file mode 100644
index 00000000000..fe2e5dbf836
--- /dev/null
+++ b/jdk/src/solaris/classes/sun/font/FcFontConfiguration.java
@@ -0,0 +1,517 @@
+/*
+ * Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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
+ * 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.font;
+
+import java.awt.Font;
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
+import java.io.IOException;
+import java.net.InetAddress;
+import java.net.UnknownHostException;
+import java.nio.charset.Charset;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.logging.Logger;
+import java.util.Properties;
+import java.util.Scanner;
+import sun.awt.FontConfiguration;
+import sun.awt.FontDescriptor;
+import sun.awt.SunToolkit;
+import sun.font.CompositeFontDescriptor;
+import sun.font.FontManager;
+import sun.font.FontManager.FontConfigInfo;
+import sun.font.FontManager.FcCompFont;
+import sun.font.FontManager.FontConfigFont;
+import sun.java2d.SunGraphicsEnvironment;
+
+public class FcFontConfiguration extends FontConfiguration {
+
+ /** Version of the cache file format understood by this code.
+ * Its part of the file name so that we can rev this at
+ * any time, even in a minor JDK update.
+ * It is stored as the value of the "version" property.
+ * This is distinct from the version of "libfontconfig" that generated
+ * the cached results, and which is the "fcversion" property in the file.
+ * {@code FontConfiguration.getVersion()} also returns a version string,
+ * and has meant the version of the fontconfiguration.properties file
+ * that was read. Since this class doesn't use such files, then what
+ * that really means is whether the methods on this class return
+ * values that are compatible with the classes that do directly read
+ * from such files. It is a compatible subset of version "1".
+ */
+ private static final String fileVersion = "1";
+ private String fcInfoFileName = null;
+
+ private FcCompFont[] fcCompFonts = null;
+
+ public FcFontConfiguration(SunGraphicsEnvironment environment) {
+ super(environment);
+ init();
+ }
+
+ /* This isn't called but is needed to satisfy super-class contract. */
+ public FcFontConfiguration(SunGraphicsEnvironment environment,
+ boolean preferLocaleFonts,
+ boolean preferPropFonts) {
+ super(environment, preferLocaleFonts, preferPropFonts);
+ init();
+ }
+
+ @Override
+ public synchronized boolean init() {
+ if (fcCompFonts != null) {
+ return true;
+ }
+
+ readFcInfo();
+ if (fcCompFonts == null) {
+ fcCompFonts = FontManager.loadFontConfig();
+ if (fcCompFonts != null) {
+ try {
+ writeFcInfo();
+ } catch (Exception e) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger =
+ Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Exception writing fcInfo " + e);
+ }
+ }
+ } else if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Failed to get info from libfontconfig");
+ }
+ } else {
+ FontManager.populateFontConfig(fcCompFonts);
+ }
+
+ if (fcCompFonts == null) {
+ return false; // couldn't load fontconfig.
+ }
+
+ // NB already in a privileged block from SGE
+ String javaHome = System.getProperty("java.home");
+ if (javaHome == null) {
+ throw new Error("java.home property not set");
+ }
+ String javaLib = javaHome + File.separator + "lib";
+ getInstalledFallbackFonts(javaLib);
+
+ return true;
+ }
+
+ @Override
+ public String getFallbackFamilyName(String fontName,
+ String defaultFallback) {
+ // maintain compatibility with old font.properties files, which either
+ // had aliases for TimesRoman & Co. or defined mappings for them.
+ String compatibilityName = getCompatibilityFamilyName(fontName);
+ if (compatibilityName != null) {
+ return compatibilityName;
+ }
+ return defaultFallback;
+ }
+
+ @Override
+ protected String
+ getFaceNameFromComponentFontName(String componentFontName) {
+ return null;
+ }
+
+ @Override
+ protected String
+ getFileNameFromComponentFontName(String componentFontName) {
+ return null;
+ }
+
+ @Override
+ public String getFileNameFromPlatformName(String platformName) {
+ /* Platform name is the file name, but rather than returning
+ * the arg, return null*/
+ return null;
+ }
+
+ @Override
+ protected Charset getDefaultFontCharset(String fontName) {
+ return Charset.forName("ISO8859_1");
+ }
+
+ @Override
+ protected String getEncoding(String awtFontName,
+ String characterSubsetName) {
+ return "default";
+ }
+
+ @Override
+ protected void initReorderMap() {
+ reorderMap = new HashMap();
+ }
+
+ @Override
+ public FontDescriptor[] getFontDescriptors(String fontName, int style) {
+ throw new InternalError("Not implemented");
+ }
+
+ @Override
+ public int getNumberCoreFonts() {
+ return 1;
+ }
+
+ @Override
+ public String[] getPlatformFontNames() {
+ HashSet nameSet = new HashSet();
+ FcCompFont[] fcCompFonts = FontManager.loadFontConfig();
+ for (int i=0; i lastModified) {
+ return;
+ }
+ cacheDirIndex++;
+ }
+
+ String[] names = { "sansserif", "serif", "monospaced" };
+ String[] fcnames = { "sans", "serif", "monospace" };
+ int namesLen = names.length;
+ int numStyles = 4;
+ FcCompFont[] fci = new FcCompFont[namesLen*numStyles];
+
+ try {
+ for (int i=0; i
+#include
+#include
#include
+#if defined(__GNUC__) && (__GNUC__ >= 4)
+#define FC_ATTRIBUTE_SENTINEL(x) __attribute__((__sentinel__(0)))
+#else
+#define FC_ATTRIBUTE_SENTINEL(x)
+#endif
+
+#ifndef FcPublic
+#define FcPublic
+#endif
+
typedef unsigned char FcChar8;
typedef unsigned short FcChar16;
typedef unsigned int FcChar32;
@@ -43,7 +52,7 @@ typedef int FcBool;
*/
#define FC_MAJOR 2
-#define FC_MINOR 2
+#define FC_MINOR 5
#define FC_REVISION 0
#define FC_VERSION ((FC_MAJOR * 10000) + (FC_MINOR * 100) + (FC_REVISION))
@@ -58,7 +67,7 @@ typedef int FcBool;
* it means multiple copies of the font information.
*/
-#define FC_CACHE_VERSION "1"
+#define FC_CACHE_VERSION "2"
#define FcTrue 1
#define FcFalse 0
@@ -74,6 +83,7 @@ typedef int FcBool;
#define FC_FOUNDRY "foundry" /* String */
#define FC_ANTIALIAS "antialias" /* Bool (depends) */
#define FC_HINTING "hinting" /* Bool (true) */
+#define FC_HINT_STYLE "hintstyle" /* Int */
#define FC_VERTICAL_LAYOUT "verticallayout" /* Bool (false) */
#define FC_AUTOHINT "autohint" /* Bool (false) */
#define FC_GLOBAL_ADVANCE "globaladvance" /* Bool (true) */
@@ -88,11 +98,21 @@ typedef int FcBool;
#define FC_DPI "dpi" /* double */
#define FC_RGBA "rgba" /* Int */
#define FC_MINSPACE "minspace" /* Bool use minimum line spacing */
-#define FC_SOURCE "source" /* String (X11, freetype) */
+#define FC_SOURCE "source" /* String (deprecated) */
#define FC_CHARSET "charset" /* CharSet */
#define FC_LANG "lang" /* String RFC 3066 langs */
#define FC_FONTVERSION "fontversion" /* Int from 'head' table */
+#define FC_FULLNAME "fullname" /* String */
+#define FC_FAMILYLANG "familylang" /* String RFC 3066 langs */
+#define FC_STYLELANG "stylelang" /* String RFC 3066 langs */
+#define FC_FULLNAMELANG "fullnamelang" /* String RFC 3066 langs */
+#define FC_CAPABILITY "capability" /* String */
+#define FC_FONTFORMAT "fontformat" /* String */
+#define FC_EMBOLDEN "embolden" /* Bool - true if emboldening needed*/
+#define FC_EMBEDDED_BITMAP "embeddedbitmap" /* Bool - true to enable embedded bitmaps */
+#define FC_DECORATIVE "decorative" /* Bool - true if style is a decorative variant */
+#define FC_CACHE_SUFFIX ".cache-"FC_CACHE_VERSION
#define FC_DIR_CACHE_FILE "fonts.cache-"FC_CACHE_VERSION
#define FC_USER_CACHE_FILE ".fonts.cache-"FC_CACHE_VERSION
@@ -105,6 +125,7 @@ typedef int FcBool;
#define FC_WEIGHT_EXTRALIGHT 40
#define FC_WEIGHT_ULTRALIGHT FC_WEIGHT_EXTRALIGHT
#define FC_WEIGHT_LIGHT 50
+#define FC_WEIGHT_BOOK 75
#define FC_WEIGHT_REGULAR 80
#define FC_WEIGHT_NORMAL FC_WEIGHT_REGULAR
#define FC_WEIGHT_MEDIUM 100
@@ -115,6 +136,8 @@ typedef int FcBool;
#define FC_WEIGHT_ULTRABOLD FC_WEIGHT_EXTRABOLD
#define FC_WEIGHT_BLACK 210
#define FC_WEIGHT_HEAVY FC_WEIGHT_BLACK
+#define FC_WEIGHT_EXTRABLACK 215
+#define FC_WEIGHT_ULTRABLACK FC_WEIGHT_EXTRABLACK
#define FC_SLANT_ROMAN 0
#define FC_SLANT_ITALIC 100
@@ -131,6 +154,7 @@ typedef int FcBool;
#define FC_WIDTH_ULTRAEXPANDED 200
#define FC_PROPORTIONAL 0
+#define FC_DUAL 90
#define FC_MONO 100
#define FC_CHARCELL 110
@@ -142,6 +166,12 @@ typedef int FcBool;
#define FC_RGBA_VBGR 4
#define FC_RGBA_NONE 5
+/* hinting style */
+#define FC_HINT_NONE 0
+#define FC_HINT_SLIGHT 1
+#define FC_HINT_MEDIUM 2
+#define FC_HINT_FULL 3
+
typedef enum _FcType {
FcTypeVoid,
FcTypeInteger,
@@ -180,7 +210,8 @@ typedef struct _FcConstant {
} FcConstant;
typedef enum _FcResult {
- FcResultMatch, FcResultNoMatch, FcResultTypeMismatch, FcResultNoId
+ FcResultMatch, FcResultNoMatch, FcResultTypeMismatch, FcResultNoId,
+ FcResultOutOfMemory
} FcResult;
typedef struct _FcPattern FcPattern;
@@ -197,7 +228,6 @@ typedef struct _FcValue {
const FcMatrix *m;
const FcCharSet *c;
void *f;
- const FcPattern *p;
const FcLangSet *l;
} u;
} FcValue;
@@ -215,11 +245,14 @@ typedef struct _FcObjectSet {
} FcObjectSet;
typedef enum _FcMatchKind {
- FcMatchPattern, FcMatchFont
+ FcMatchPattern, FcMatchFont, FcMatchScan
} FcMatchKind;
typedef enum _FcLangResult {
- FcLangEqual, FcLangDifferentCountry, FcLangDifferentLang
+ FcLangEqual = 0,
+ FcLangDifferentCountry = 1,
+ FcLangDifferentTerritory = 1,
+ FcLangDifferentLang = 2
} FcLangResult;
typedef enum _FcSetName {
@@ -249,169 +282,207 @@ typedef struct _FcStrList FcStrList;
typedef struct _FcStrSet FcStrSet;
+typedef struct _FcCache FcCache;
+
_FCFUNCPROTOBEGIN
-FcBool
-FcDirCacheValid (const FcChar8 *cache_file);
-
/* fcblanks.c */
-FcBlanks *
+FcPublic FcBlanks *
FcBlanksCreate (void);
-void
+FcPublic void
FcBlanksDestroy (FcBlanks *b);
-FcBool
+FcPublic FcBool
FcBlanksAdd (FcBlanks *b, FcChar32 ucs4);
-FcBool
+FcPublic FcBool
FcBlanksIsMember (FcBlanks *b, FcChar32 ucs4);
+/* fccache.c */
+
+FcPublic const FcChar8 *
+FcCacheDir(const FcCache *c);
+
+FcPublic FcFontSet *
+FcCacheCopySet(const FcCache *c);
+
+FcPublic const FcChar8 *
+FcCacheSubdir (const FcCache *c, int i);
+
+FcPublic int
+FcCacheNumSubdir (const FcCache *c);
+
+FcPublic int
+FcCacheNumFont (const FcCache *c);
+
+FcPublic FcBool
+FcDirCacheUnlink (const FcChar8 *dir, FcConfig *config);
+
+FcPublic FcBool
+FcDirCacheValid (const FcChar8 *cache_file);
+
/* fccfg.c */
-FcChar8 *
+FcPublic FcChar8 *
FcConfigHome (void);
-FcBool
+FcPublic FcBool
FcConfigEnableHome (FcBool enable);
-FcChar8 *
+FcPublic FcChar8 *
FcConfigFilename (const FcChar8 *url);
-FcConfig *
+FcPublic FcConfig *
FcConfigCreate (void);
-void
+FcPublic void
FcConfigDestroy (FcConfig *config);
-FcBool
+FcPublic FcBool
FcConfigSetCurrent (FcConfig *config);
-FcConfig *
+FcPublic FcConfig *
FcConfigGetCurrent (void);
-FcBool
+FcPublic FcBool
FcConfigUptoDate (FcConfig *config);
-FcBool
+FcPublic FcBool
FcConfigBuildFonts (FcConfig *config);
-FcStrList *
+FcPublic FcStrList *
FcConfigGetFontDirs (FcConfig *config);
-FcStrList *
+FcPublic FcStrList *
FcConfigGetConfigDirs (FcConfig *config);
-FcStrList *
+FcPublic FcStrList *
FcConfigGetConfigFiles (FcConfig *config);
-FcChar8 *
+FcPublic FcChar8 *
FcConfigGetCache (FcConfig *config);
-FcBlanks *
+FcPublic FcBlanks *
FcConfigGetBlanks (FcConfig *config);
-int
-FcConfigGetRescanInverval (FcConfig *config);
+FcPublic FcStrList *
+FcConfigGetCacheDirs (FcConfig *config);
-FcBool
-FcConfigSetRescanInverval (FcConfig *config, int rescanInterval);
+FcPublic int
+FcConfigGetRescanInterval (FcConfig *config);
-FcFontSet *
+FcPublic FcBool
+FcConfigSetRescanInterval (FcConfig *config, int rescanInterval);
+
+FcPublic FcFontSet *
FcConfigGetFonts (FcConfig *config,
FcSetName set);
-FcBool
+FcPublic FcBool
FcConfigAppFontAddFile (FcConfig *config,
const FcChar8 *file);
-FcBool
+FcPublic FcBool
FcConfigAppFontAddDir (FcConfig *config,
const FcChar8 *dir);
-void
+FcPublic void
FcConfigAppFontClear (FcConfig *config);
-FcBool
+FcPublic FcBool
FcConfigSubstituteWithPat (FcConfig *config,
FcPattern *p,
FcPattern *p_pat,
FcMatchKind kind);
-FcBool
+FcPublic FcBool
FcConfigSubstitute (FcConfig *config,
FcPattern *p,
FcMatchKind kind);
/* fccharset.c */
-FcCharSet *
+FcPublic FcCharSet*
FcCharSetCreate (void);
-void
+/* deprecated alias for FcCharSetCreate */
+FcPublic FcCharSet *
+FcCharSetNew (void);
+
+FcPublic void
FcCharSetDestroy (FcCharSet *fcs);
-FcBool
+FcPublic FcBool
FcCharSetAddChar (FcCharSet *fcs, FcChar32 ucs4);
-FcCharSet *
+FcPublic FcCharSet*
FcCharSetCopy (FcCharSet *src);
-FcBool
+FcPublic FcBool
FcCharSetEqual (const FcCharSet *a, const FcCharSet *b);
-FcCharSet *
+FcPublic FcCharSet*
FcCharSetIntersect (const FcCharSet *a, const FcCharSet *b);
-FcCharSet *
+FcPublic FcCharSet*
FcCharSetUnion (const FcCharSet *a, const FcCharSet *b);
-FcCharSet *
+FcPublic FcCharSet*
FcCharSetSubtract (const FcCharSet *a, const FcCharSet *b);
-FcBool
+FcPublic FcBool
FcCharSetHasChar (const FcCharSet *fcs, FcChar32 ucs4);
-FcChar32
+FcPublic FcChar32
FcCharSetCount (const FcCharSet *a);
-FcChar32
+FcPublic FcChar32
FcCharSetIntersectCount (const FcCharSet *a, const FcCharSet *b);
-FcChar32
+FcPublic FcChar32
FcCharSetSubtractCount (const FcCharSet *a, const FcCharSet *b);
-FcBool
+FcPublic FcBool
FcCharSetIsSubset (const FcCharSet *a, const FcCharSet *b);
#define FC_CHARSET_MAP_SIZE (256/32)
#define FC_CHARSET_DONE ((FcChar32) -1)
-FcChar32
+FcPublic FcChar32
FcCharSetFirstPage (const FcCharSet *a,
FcChar32 map[FC_CHARSET_MAP_SIZE],
FcChar32 *next);
-FcChar32
+FcPublic FcChar32
FcCharSetNextPage (const FcCharSet *a,
FcChar32 map[FC_CHARSET_MAP_SIZE],
FcChar32 *next);
+/*
+ * old coverage API, rather hard to use correctly
+ */
+
+FcPublic FcChar32
+FcCharSetCoverage (const FcCharSet *a, FcChar32 page, FcChar32 *result);
/* fcdbg.c */
-void
+FcPublic void
FcValuePrint (const FcValue v);
-void
+FcPublic void
FcPatternPrint (const FcPattern *p);
-void
+FcPublic void
FcFontSetPrint (const FcFontSet *s);
/* fcdefault.c */
-void
+FcPublic void
FcDefaultSubstitute (FcPattern *pattern);
/* fcdir.c */
-FcBool
+FcPublic FcBool
+FcFileIsDir (const FcChar8 *file);
+
+FcPublic FcBool
FcFileScan (FcFontSet *set,
FcStrSet *dirs,
FcFileCache *cache,
@@ -419,7 +490,7 @@ FcFileScan (FcFontSet *set,
const FcChar8 *file,
FcBool force);
-FcBool
+FcPublic FcBool
FcDirScan (FcFontSet *set,
FcStrSet *dirs,
FcFileCache *cache,
@@ -427,144 +498,165 @@ FcDirScan (FcFontSet *set,
const FcChar8 *dir,
FcBool force);
-FcBool
+FcPublic FcBool
FcDirSave (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir);
+FcPublic FcCache *
+FcDirCacheLoad (const FcChar8 *dir, FcConfig *config, FcChar8 **cache_file);
+
+FcPublic FcCache *
+FcDirCacheRead (const FcChar8 *dir, FcBool force, FcConfig *config);
+
+FcPublic FcCache *
+FcDirCacheLoadFile (const FcChar8 *cache_file, struct stat *file_stat);
+
+FcPublic void
+FcDirCacheUnload (FcCache *cache);
+
/* fcfreetype.c */
-FcPattern *
+FcPublic FcPattern *
FcFreeTypeQuery (const FcChar8 *file, int id, FcBlanks *blanks, int *count);
/* fcfs.c */
-FcFontSet *
+FcPublic FcFontSet *
FcFontSetCreate (void);
-void
+FcPublic void
FcFontSetDestroy (FcFontSet *s);
-FcBool
+FcPublic FcBool
FcFontSetAdd (FcFontSet *s, FcPattern *font);
/* fcinit.c */
-FcConfig *
+FcPublic FcConfig *
FcInitLoadConfig (void);
-FcConfig *
+FcPublic FcConfig *
FcInitLoadConfigAndFonts (void);
-FcBool
+FcPublic FcBool
FcInit (void);
-int
+FcPublic void
+FcFini (void);
+
+FcPublic int
FcGetVersion (void);
-FcBool
+FcPublic FcBool
FcInitReinitialize (void);
-FcBool
+FcPublic FcBool
FcInitBringUptoDate (void);
/* fclang.c */
-FcLangSet *
+FcPublic FcStrSet *
+FcGetLangs (void);
+
+FcPublic const FcCharSet *
+FcLangGetCharSet (const FcChar8 *lang);
+
+FcPublic FcLangSet*
FcLangSetCreate (void);
-void
+FcPublic void
FcLangSetDestroy (FcLangSet *ls);
-FcLangSet *
+FcPublic FcLangSet*
FcLangSetCopy (const FcLangSet *ls);
-FcBool
+FcPublic FcBool
FcLangSetAdd (FcLangSet *ls, const FcChar8 *lang);
-FcLangResult
+FcPublic FcLangResult
FcLangSetHasLang (const FcLangSet *ls, const FcChar8 *lang);
-FcLangResult
+FcPublic FcLangResult
FcLangSetCompare (const FcLangSet *lsa, const FcLangSet *lsb);
-FcBool
+FcPublic FcBool
FcLangSetContains (const FcLangSet *lsa, const FcLangSet *lsb);
-FcBool
+FcPublic FcBool
FcLangSetEqual (const FcLangSet *lsa, const FcLangSet *lsb);
-FcChar32
+FcPublic FcChar32
FcLangSetHash (const FcLangSet *ls);
/* fclist.c */
-FcObjectSet *
+FcPublic FcObjectSet *
FcObjectSetCreate (void);
-FcBool
+FcPublic FcBool
FcObjectSetAdd (FcObjectSet *os, const char *object);
-void
+FcPublic void
FcObjectSetDestroy (FcObjectSet *os);
-FcObjectSet *
+FcPublic FcObjectSet *
FcObjectSetVaBuild (const char *first, va_list va);
-FcObjectSet *
-FcObjectSetBuild (const char *first, ...);
+FcPublic FcObjectSet *
+FcObjectSetBuild (const char *first, ...) FC_ATTRIBUTE_SENTINEL(0);
-FcFontSet *
+FcPublic FcFontSet *
FcFontSetList (FcConfig *config,
FcFontSet **sets,
int nsets,
FcPattern *p,
FcObjectSet *os);
-FcFontSet *
+FcPublic FcFontSet *
FcFontList (FcConfig *config,
FcPattern *p,
FcObjectSet *os);
/* fcatomic.c */
-FcAtomic *
+FcPublic FcAtomic *
FcAtomicCreate (const FcChar8 *file);
-FcBool
+FcPublic FcBool
FcAtomicLock (FcAtomic *atomic);
-FcChar8 *
+FcPublic FcChar8 *
FcAtomicNewFile (FcAtomic *atomic);
-FcChar8 *
+FcPublic FcChar8 *
FcAtomicOrigFile (FcAtomic *atomic);
-FcBool
+FcPublic FcBool
FcAtomicReplaceOrig (FcAtomic *atomic);
-void
+FcPublic void
FcAtomicDeleteNew (FcAtomic *atomic);
-void
+FcPublic void
FcAtomicUnlock (FcAtomic *atomic);
-void
+FcPublic void
FcAtomicDestroy (FcAtomic *atomic);
/* fcmatch.c */
-FcPattern *
+FcPublic FcPattern *
FcFontSetMatch (FcConfig *config,
FcFontSet **sets,
int nsets,
FcPattern *p,
FcResult *result);
-FcPattern *
+FcPublic FcPattern *
FcFontMatch (FcConfig *config,
FcPattern *p,
FcResult *result);
-FcPattern *
+FcPublic FcPattern *
FcFontRenderPrepare (FcConfig *config,
FcPattern *pat,
FcPattern *font);
-FcFontSet *
+FcPublic FcFontSet *
FcFontSetSort (FcConfig *config,
FcFontSet **sets,
int nsets,
@@ -573,179 +665,198 @@ FcFontSetSort (FcConfig *config,
FcCharSet **csp,
FcResult *result);
-FcFontSet *
+FcPublic FcFontSet *
FcFontSort (FcConfig *config,
FcPattern *p,
FcBool trim,
FcCharSet **csp,
FcResult *result);
-void
+FcPublic void
FcFontSetSortDestroy (FcFontSet *fs);
/* fcmatrix.c */
-FcMatrix *
+FcPublic FcMatrix *
FcMatrixCopy (const FcMatrix *mat);
-FcBool
+FcPublic FcBool
FcMatrixEqual (const FcMatrix *mat1, const FcMatrix *mat2);
-void
+FcPublic void
FcMatrixMultiply (FcMatrix *result, const FcMatrix *a, const FcMatrix *b);
-void
+FcPublic void
FcMatrixRotate (FcMatrix *m, double c, double s);
-void
+FcPublic void
FcMatrixScale (FcMatrix *m, double sx, double sy);
-void
+FcPublic void
FcMatrixShear (FcMatrix *m, double sh, double sv);
/* fcname.c */
-FcBool
+FcPublic FcBool
FcNameRegisterObjectTypes (const FcObjectType *types, int ntype);
-FcBool
+FcPublic FcBool
FcNameUnregisterObjectTypes (const FcObjectType *types, int ntype);
-const FcObjectType *
+FcPublic const FcObjectType *
FcNameGetObjectType (const char *object);
-FcBool
+FcPublic FcBool
FcNameRegisterConstants (const FcConstant *consts, int nconsts);
-FcBool
+FcPublic FcBool
FcNameUnregisterConstants (const FcConstant *consts, int nconsts);
-const FcConstant *
+FcPublic const FcConstant *
FcNameGetConstant (FcChar8 *string);
-FcBool
+FcPublic FcBool
FcNameConstant (FcChar8 *string, int *result);
-FcPattern *
+FcPublic FcPattern *
FcNameParse (const FcChar8 *name);
-FcChar8 *
+FcPublic FcChar8 *
FcNameUnparse (FcPattern *pat);
/* fcpat.c */
-FcPattern *
+FcPublic FcPattern *
FcPatternCreate (void);
-FcPattern *
+FcPublic FcPattern *
FcPatternDuplicate (const FcPattern *p);
-void
+FcPublic void
FcPatternReference (FcPattern *p);
-void
+FcPublic void
FcValueDestroy (FcValue v);
-FcBool
+FcPublic FcBool
FcValueEqual (FcValue va, FcValue vb);
-FcValue
+FcPublic FcValue
FcValueSave (FcValue v);
-void
+FcPublic void
FcPatternDestroy (FcPattern *p);
-FcBool
+FcPublic FcBool
FcPatternEqual (const FcPattern *pa, const FcPattern *pb);
-FcBool
+FcPublic FcBool
FcPatternEqualSubset (const FcPattern *pa, const FcPattern *pb, const FcObjectSet *os);
-FcChar32
+FcPublic FcChar32
FcPatternHash (const FcPattern *p);
-FcBool
+FcPublic FcBool
FcPatternAdd (FcPattern *p, const char *object, FcValue value, FcBool append);
-FcBool
+FcPublic FcBool
FcPatternAddWeak (FcPattern *p, const char *object, FcValue value, FcBool append);
-FcResult
+FcPublic FcResult
FcPatternGet (const FcPattern *p, const char *object, int id, FcValue *v);
-FcBool
+FcPublic FcBool
FcPatternDel (FcPattern *p, const char *object);
-FcBool
+FcPublic FcBool
+FcPatternRemove (FcPattern *p, const char *object, int id);
+
+FcPublic FcBool
FcPatternAddInteger (FcPattern *p, const char *object, int i);
-FcBool
+FcPublic FcBool
FcPatternAddDouble (FcPattern *p, const char *object, double d);
-FcBool
+FcPublic FcBool
FcPatternAddString (FcPattern *p, const char *object, const FcChar8 *s);
-FcBool
+FcPublic FcBool
FcPatternAddMatrix (FcPattern *p, const char *object, const FcMatrix *s);
-FcBool
+FcPublic FcBool
FcPatternAddCharSet (FcPattern *p, const char *object, const FcCharSet *c);
-FcBool
+FcPublic FcBool
FcPatternAddBool (FcPattern *p, const char *object, FcBool b);
-FcBool
+FcPublic FcBool
FcPatternAddLangSet (FcPattern *p, const char *object, const FcLangSet *ls);
-FcResult
+FcPublic FcResult
FcPatternGetInteger (const FcPattern *p, const char *object, int n, int *i);
-FcResult
+FcPublic FcResult
FcPatternGetDouble (const FcPattern *p, const char *object, int n, double *d);
-FcResult
+FcPublic FcResult
FcPatternGetString (const FcPattern *p, const char *object, int n, FcChar8 ** s);
-FcResult
+FcPublic FcResult
FcPatternGetMatrix (const FcPattern *p, const char *object, int n, FcMatrix **s);
-FcResult
+FcPublic FcResult
FcPatternGetCharSet (const FcPattern *p, const char *object, int n, FcCharSet **c);
-FcResult
+FcPublic FcResult
FcPatternGetBool (const FcPattern *p, const char *object, int n, FcBool *b);
-FcResult
+FcPublic FcResult
FcPatternGetLangSet (const FcPattern *p, const char *object, int n, FcLangSet **ls);
-FcPattern *
+FcPublic FcPattern *
FcPatternVaBuild (FcPattern *orig, va_list va);
-FcPattern *
-FcPatternBuild (FcPattern *orig, ...);
+FcPublic FcPattern *
+FcPatternBuild (FcPattern *orig, ...) FC_ATTRIBUTE_SENTINEL(0);
/* fcstr.c */
-FcChar8 *
+FcPublic FcChar8 *
FcStrCopy (const FcChar8 *s);
-FcChar8 *
+FcPublic FcChar8 *
FcStrCopyFilename (const FcChar8 *s);
-#define FcIsUpper(c) (('A' <= (c) && (c) <= 'Z'))
-#define FcIsLower(c) (('a' <= (c) && (c) <= 'z'))
-#define FcToLower(c) (FcIsUpper(c) ? (c) - 'A' + 'a' : (c))
+FcPublic FcChar8 *
+FcStrPlus (const FcChar8 *s1, const FcChar8 *s2);
-int
+FcPublic void
+FcStrFree (FcChar8 *s);
+
+/* These are ASCII only, suitable only for pattern element names */
+#define FcIsUpper(c) ((0101 <= (c) && (c) <= 0132))
+#define FcIsLower(c) ((0141 <= (c) && (c) <= 0172))
+#define FcToLower(c) (FcIsUpper(c) ? (c) - 0101 + 0141 : (c))
+
+FcPublic FcChar8 *
+FcStrDowncase (const FcChar8 *s);
+
+FcPublic int
FcStrCmpIgnoreCase (const FcChar8 *s1, const FcChar8 *s2);
-int
+FcPublic int
FcStrCmp (const FcChar8 *s1, const FcChar8 *s2);
-int
+FcPublic const FcChar8 *
+FcStrStrIgnoreCase (const FcChar8 *s1, const FcChar8 *s2);
+
+FcPublic const FcChar8 *
+FcStrStr (const FcChar8 *s1, const FcChar8 *s2);
+
+FcPublic int
FcUtf8ToUcs4 (const FcChar8 *src_orig,
FcChar32 *dst,
int len);
-FcBool
+FcPublic FcBool
FcUtf8Len (const FcChar8 *string,
int len,
int *nchar,
@@ -753,63 +864,78 @@ FcUtf8Len (const FcChar8 *string,
#define FC_UTF8_MAX_LEN 6
-int
+FcPublic int
FcUcs4ToUtf8 (FcChar32 ucs4,
FcChar8 dest[FC_UTF8_MAX_LEN]);
-int
+FcPublic int
FcUtf16ToUcs4 (const FcChar8 *src_orig,
FcEndian endian,
FcChar32 *dst,
int len); /* in bytes */
-FcBool
+FcPublic FcBool
FcUtf16Len (const FcChar8 *string,
FcEndian endian,
int len, /* in bytes */
int *nchar,
int *wchar);
-FcChar8 *
+FcPublic FcChar8 *
FcStrDirname (const FcChar8 *file);
-FcChar8 *
+FcPublic FcChar8 *
FcStrBasename (const FcChar8 *file);
-FcStrSet *
+FcPublic FcStrSet *
FcStrSetCreate (void);
-FcBool
+FcPublic FcBool
FcStrSetMember (FcStrSet *set, const FcChar8 *s);
-FcBool
+FcPublic FcBool
FcStrSetEqual (FcStrSet *sa, FcStrSet *sb);
-FcBool
+FcPublic FcBool
FcStrSetAdd (FcStrSet *set, const FcChar8 *s);
-FcBool
+FcPublic FcBool
FcStrSetAddFilename (FcStrSet *set, const FcChar8 *s);
-FcBool
+FcPublic FcBool
FcStrSetDel (FcStrSet *set, const FcChar8 *s);
-void
+FcPublic void
FcStrSetDestroy (FcStrSet *set);
-FcStrList *
+FcPublic FcStrList *
FcStrListCreate (FcStrSet *set);
-FcChar8 *
+FcPublic FcChar8 *
FcStrListNext (FcStrList *list);
-void
+FcPublic void
FcStrListDone (FcStrList *list);
/* fcxml.c */
-FcBool
+FcPublic FcBool
FcConfigParseAndLoad (FcConfig *config, const FcChar8 *file, FcBool complain);
_FCFUNCPROTOEND
+#undef FC_ATTRIBUTE_SENTINEL
+
+
+#ifndef _FCINT_H_
+
+/*
+ * Deprecated functions are placed here to help users fix their code without
+ * digging through documentation
+ */
+
+#define FcConfigGetRescanInverval FcConfigGetRescanInverval_REPLACE_BY_FcConfigGetRescanInterval
+#define FcConfigSetRescanInverval FcConfigSetRescanInverval_REPLACE_BY_FcConfigSetRescanInterval
+
+#endif
+
#endif /* _FONTCONFIG_H_ */
diff --git a/jdk/src/solaris/native/sun/awt/fontpath.c b/jdk/src/solaris/native/sun/awt/fontpath.c
index d863d265b58..8f9f3d9a894 100644
--- a/jdk/src/solaris/native/sun/awt/fontpath.c
+++ b/jdk/src/solaris/native/sun/awt/fontpath.c
@@ -735,6 +735,25 @@ typedef FcPattern* (*FcFontMatchFuncType)(FcConfig *config,
typedef FcFontSet* (*FcFontSetCreateFuncType)();
typedef FcBool (*FcFontSetAddFuncType)(FcFontSet *s, FcPattern *font);
+typedef FcResult (*FcPatternGetCharSetFuncType)(FcPattern *p,
+ const char *object,
+ int n,
+ FcCharSet **c);
+typedef FcFontSet* (*FcFontSortFuncType)(FcConfig *config,
+ FcPattern *p,
+ FcBool trim,
+ FcCharSet **csp,
+ FcResult *result);
+typedef FcCharSet* (*FcCharSetUnionFuncType)(const FcCharSet *a,
+ const FcCharSet *b);
+typedef FcChar32 (*FcCharSetSubtractCountFuncType)(const FcCharSet *a,
+ const FcCharSet *b);
+
+typedef int (*FcGetVersionFuncType)();
+
+typedef FcStrList* (*FcConfigGetCacheDirsFuncType)(FcConfig *config);
+typedef FcChar8* (*FcStrListNextFuncType)(FcStrList *list);
+typedef FcChar8* (*FcStrListDoneFuncType)(FcStrList *list);
static char **getFontConfigLocations() {
@@ -955,10 +974,35 @@ Java_sun_font_FontManager_getFontConfigAASettings
}
}
+JNIEXPORT jint JNICALL
+Java_sun_font_FontManager_getFontConfigVersion
+ (JNIEnv *env, jclass obj) {
+
+ void* libfontconfig;
+ FcGetVersionFuncType FcGetVersion;
+ int version = 0;
+
+ if ((libfontconfig = openFontConfig()) == NULL) {
+ return 0;
+ }
+
+ FcGetVersion = (FcGetVersionFuncType)dlsym(libfontconfig, "FcGetVersion");
+
+ if (FcGetVersion == NULL) {
+ closeFontConfig(libfontconfig, JNI_FALSE);
+ return 0;
+ }
+ version = (*FcGetVersion)();
+ closeFontConfig(libfontconfig, JNI_FALSE);
+
+ return version;
+}
+
JNIEXPORT void JNICALL
Java_sun_font_FontManager_getFontConfig
-(JNIEnv *env, jclass obj, jstring localeStr, jobjectArray fontInfoArray) {
+(JNIEnv *env, jclass obj, jstring localeStr, jobject fcInfoObj,
+ jobjectArray fcCompFontArray, jboolean includeFallbacks) {
FcNameParseFuncType FcNameParse;
FcPatternAddStringFuncType FcPatternAddString;
@@ -967,33 +1011,70 @@ Java_sun_font_FontManager_getFontConfig
FcFontMatchFuncType FcFontMatch;
FcPatternGetStringFuncType FcPatternGetString;
FcPatternDestroyFuncType FcPatternDestroy;
+ FcPatternGetCharSetFuncType FcPatternGetCharSet;
+ FcFontSortFuncType FcFontSort;
+ FcFontSetDestroyFuncType FcFontSetDestroy;
+ FcCharSetUnionFuncType FcCharSetUnion;
+ FcCharSetSubtractCountFuncType FcCharSetSubtractCount;
+ FcGetVersionFuncType FcGetVersion;
+ FcConfigGetCacheDirsFuncType FcConfigGetCacheDirs;
+ FcStrListNextFuncType FcStrListNext;
+ FcStrListDoneFuncType FcStrListDone;
int i, arrlen;
- jobject fontInfoObj;
+ jobject fcCompFontObj;
jstring fcNameStr, jstr;
const char *locale, *fcName;
- FcPattern *pattern, *matchPattern;
+ FcPattern *pattern;
FcResult result;
void* libfontconfig;
- jfieldID fcNameID, familyNameID, fontFileID;
+ jfieldID fcNameID, fcFirstFontID, fcAllFontsID, fcVersionID, fcCacheDirsID;
+ jfieldID familyNameID, styleNameID, fullNameID, fontFileID;
+ jmethodID fcFontCons;
+ char* debugMinGlyphsStr = getenv("J2D_DEBUG_MIN_GLYPHS");
- jclass fontInfoArrayClass =
- (*env)->FindClass(env, "[Lsun/font/FontManager$FontConfigInfo;");
- jclass fontInfoClass =
+ jclass fcInfoClass =
(*env)->FindClass(env, "sun/font/FontManager$FontConfigInfo");
+ jclass fcCompFontClass =
+ (*env)->FindClass(env, "sun/font/FontManager$FcCompFont");
+ jclass fcFontClass =
+ (*env)->FindClass(env, "sun/font/FontManager$FontConfigFont");
- if (fontInfoArray == NULL || fontInfoClass == NULL) {
+ if (fcInfoObj == NULL || fcCompFontArray == NULL || fcInfoClass == NULL ||
+ fcCompFontClass == NULL || fcFontClass == NULL) {
return;
}
- fcNameID = (*env)->GetFieldID(env, fontInfoClass,
+ fcVersionID = (*env)->GetFieldID(env, fcInfoClass, "fcVersion", "I");
+
+ fcCacheDirsID = (*env)->GetFieldID(env, fcInfoClass, "cacheDirs",
+ "[Ljava/lang/String;");
+
+ fcNameID = (*env)->GetFieldID(env, fcCompFontClass,
"fcName", "Ljava/lang/String;");
- familyNameID = (*env)->GetFieldID(env, fontInfoClass,
+ fcFirstFontID =
+ (*env)->GetFieldID(env, fcCompFontClass, "firstFont",
+ "Lsun/font/FontManager$FontConfigFont;");
+
+ fcAllFontsID =
+ (*env)->GetFieldID(env, fcCompFontClass, "allFonts",
+ "[Lsun/font/FontManager$FontConfigFont;");
+
+ fcFontCons = (*env)->GetMethodID(env, fcFontClass, "", "()V");
+
+ familyNameID = (*env)->GetFieldID(env, fcFontClass,
"familyName", "Ljava/lang/String;");
- fontFileID = (*env)->GetFieldID(env, fontInfoClass,
+ styleNameID = (*env)->GetFieldID(env, fcFontClass,
+ "styleStr", "Ljava/lang/String;");
+ fullNameID = (*env)->GetFieldID(env, fcFontClass,
+ "fullName", "Ljava/lang/String;");
+ fontFileID = (*env)->GetFieldID(env, fcFontClass,
"fontFile", "Ljava/lang/String;");
- if (fcNameID == NULL || familyNameID == NULL || fontFileID == NULL) {
+ if (fcVersionID == NULL || fcCacheDirsID == NULL || fcNameID == NULL ||
+ fcFirstFontID == NULL || fcAllFontsID == NULL || fcFontCons == NULL ||
+ familyNameID == NULL || styleNameID == NULL || fullNameID == NULL ||
+ fontFileID == NULL) {
return;
}
@@ -1013,6 +1094,19 @@ Java_sun_font_FontManager_getFontConfig
(FcPatternGetStringFuncType)dlsym(libfontconfig, "FcPatternGetString");
FcPatternDestroy =
(FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy");
+ FcPatternGetCharSet =
+ (FcPatternGetCharSetFuncType)dlsym(libfontconfig,
+ "FcPatternGetCharSet");
+ FcFontSort =
+ (FcFontSortFuncType)dlsym(libfontconfig, "FcFontSort");
+ FcFontSetDestroy =
+ (FcFontSetDestroyFuncType)dlsym(libfontconfig, "FcFontSetDestroy");
+ FcCharSetUnion =
+ (FcCharSetUnionFuncType)dlsym(libfontconfig, "FcCharSetUnion");
+ FcCharSetSubtractCount =
+ (FcCharSetSubtractCountFuncType)dlsym(libfontconfig,
+ "FcCharSetSubtractCount");
+ FcGetVersion = (FcGetVersionFuncType)dlsym(libfontconfig, "FcGetVersion");
if (FcNameParse == NULL ||
FcPatternAddString == NULL ||
@@ -1020,23 +1114,77 @@ Java_sun_font_FontManager_getFontConfig
FcDefaultSubstitute == NULL ||
FcFontMatch == NULL ||
FcPatternGetString == NULL ||
- FcPatternDestroy == NULL) { /* problem with the library: return. */
+ FcPatternDestroy == NULL ||
+ FcPatternGetCharSet == NULL ||
+ FcFontSetDestroy == NULL ||
+ FcCharSetUnion == NULL ||
+ FcGetVersion == NULL ||
+ FcCharSetSubtractCount == NULL) {/* problem with the library: return.*/
closeFontConfig(libfontconfig, JNI_FALSE);
return;
}
+ (*env)->SetIntField(env, fcInfoObj, fcVersionID, (*FcGetVersion)());
+
+ /* Optionally get the cache dir locations. This isn't
+ * available until v 2.4.x, but this is OK since on those later versions
+ * we can check the time stamps on the cache dirs to see if we
+ * are out of date. There are a couple of assumptions here. First
+ * that the time stamp on the directory changes when the contents are
+ * updated. Secondly that the locations don't change. The latter is
+ * most likely if a new version of fontconfig is installed, but we also
+ * invalidate the cache if we detect that. Arguably even that is "rare",
+ * and most likely is tied to an OS upgrade which gets a new file anyway.
+ */
+ FcConfigGetCacheDirs =
+ (FcConfigGetCacheDirsFuncType)dlsym(libfontconfig,
+ "FcConfigGetCacheDirs");
+ FcStrListNext =
+ (FcStrListNextFuncType)dlsym(libfontconfig, "FcStrListNext");
+ FcStrListDone =
+ (FcStrListDoneFuncType)dlsym(libfontconfig, "FcStrListDone");
+ if (FcStrListNext != NULL && FcStrListDone != NULL &&
+ FcConfigGetCacheDirs != NULL) {
+
+ FcStrList* cacheDirs;
+ FcChar8* cacheDir;
+ int cnt = 0;
+ jobject cacheDirArray =
+ (*env)->GetObjectField(env, fcInfoObj, fcCacheDirsID);
+ int max = (*env)->GetArrayLength(env, cacheDirArray);
+
+ cacheDirs = (*FcConfigGetCacheDirs)(NULL);
+ if (cacheDirs != NULL) {
+ while ((cnt < max) && (cacheDir = (*FcStrListNext)(cacheDirs))) {
+ jstr = (*env)->NewStringUTF(env, (const char*)cacheDir);
+ (*env)->SetObjectArrayElement(env, cacheDirArray, cnt++, jstr);
+ }
+ (*FcStrListDone)(cacheDirs);
+ }
+ }
+
locale = (*env)->GetStringUTFChars(env, localeStr, 0);
- arrlen = (*env)->GetArrayLength(env, fontInfoArray);
+ arrlen = (*env)->GetArrayLength(env, fcCompFontArray);
for (i=0; iGetObjectArrayElement(env, fontInfoArray, i);
+ FcFontSet* fontset;
+ int fn, j, fontCount, nfonts, minGlyphs;
+ FcChar8 **family, **styleStr, **fullname, **file;
+ jarray fcFontArr;
+
+ fcCompFontObj = (*env)->GetObjectArrayElement(env, fcCompFontArray, i);
fcNameStr =
- (jstring)((*env)->GetObjectField(env, fontInfoObj, fcNameID));
+ (jstring)((*env)->GetObjectField(env, fcCompFontObj, fcNameID));
fcName = (*env)->GetStringUTFChars(env, fcNameStr, 0);
if (fcName == NULL) {
continue;
}
pattern = (*FcNameParse)((FcChar8 *)fcName);
+ if (pattern == NULL) {
+ closeFontConfig(libfontconfig, JNI_FALSE);
+ return;
+ }
+
/* locale may not usually be necessary as fontconfig appears to apply
* this anyway based on the user's environment. However we want
* to use the value of the JDK startup locale so this should take
@@ -1047,25 +1195,134 @@ Java_sun_font_FontManager_getFontConfig
}
(*FcConfigSubstitute)(NULL, pattern, FcMatchPattern);
(*FcDefaultSubstitute)(pattern);
- matchPattern = (*FcFontMatch)(NULL, pattern, &result);
- if (matchPattern) {
- FcChar8 *file, *family;
+ fontset = (*FcFontSort)(NULL, pattern, FcTrue, NULL, &result);
+ if (fontset == NULL) {
+ closeFontConfig(libfontconfig, JNI_FALSE);
+ return;
+ }
- (*FcPatternGetString)(matchPattern, FC_FILE, 0, &file);
- (*FcPatternGetString)(matchPattern, FC_FAMILY, 0, &family);
+ /* fontconfig returned us "nfonts". If we are just getting the
+ * first font, we set nfont to zero. Otherwise we use "nfonts".
+ * Next create separate C arrrays of length nfonts for family file etc.
+ * Inspect the returned fonts and the ones we like (adds enough glyphs)
+ * are added to the arrays and we increment 'fontCount'.
+ */
+ if (includeFallbacks) {
+ nfonts = fontset->nfont;
+ } else {
+ nfonts = 1;
+ }
+ family = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
+ styleStr = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
+ fullname = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
+ file = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
+ if (family == NULL || styleStr == NULL ||
+ fullname == NULL || file == NULL) {
+ closeFontConfig(libfontconfig, JNI_FALSE);
+ return;
+ }
+ fontCount = 0;
+ minGlyphs = 20;
+ if (debugMinGlyphsStr != NULL) {
+ int val = minGlyphs;
+ sscanf(debugMinGlyphsStr, "%5d", &val);
+ if (val >= 0 && val <= 65536) {
+ minGlyphs = val;
+ }
+ }
+ for (j=0; jfonts[j];
+ FcChar8 *fontformat;
+ FcCharSet *unionCharset, *charset;
- if (file != NULL) {
- jstr = (*env)->NewStringUTF(env, (const char*)file);
- ((*env)->SetObjectField(env, fontInfoObj, fontFileID, jstr));
+ fontformat = NULL;
+ (*FcPatternGetString)(fontPattern, FC_FONTFORMAT, 0, &fontformat);
+ if (fontformat != NULL && strcmp((char*)fontformat, "TrueType")
+ != 0) {
+ continue;
}
- if (family != NULL) {
- jstr = (*env)->NewStringUTF(env, (const char*)family);
- ((*env)->SetObjectField(env, fontInfoObj, familyNameID, jstr));
+ result = (*FcPatternGetCharSet)(fontPattern,
+ FC_CHARSET, 0, &charset);
+ if (result != FcResultMatch) {
+ closeFontConfig(libfontconfig, JNI_FALSE);
+ return;
+ }
+
+ /* We don't want 20 or 30 fonts, so once we hit 10 fonts,
+ * then require that they really be adding value. Too many
+ * adversely affects load time for minimal value-add.
+ * This is still likely far more than we've had in the past.
+ */
+ if (nfonts==10) {
+ minGlyphs = 50;
+ }
+ if (j == 0) {
+ unionCharset = charset;
+ } else {
+ if ((*FcCharSetSubtractCount)(charset, unionCharset)
+ > minGlyphs) {
+ unionCharset = (* FcCharSetUnion)(unionCharset, charset);
+ } else {
+ continue;
+ }
+ }
+
+ fontCount++; // found a font we will use.
+ (*FcPatternGetString)(fontPattern, FC_FILE, 0, &file[j]);
+ (*FcPatternGetString)(fontPattern, FC_FAMILY, 0, &family[j]);
+ (*FcPatternGetString)(fontPattern, FC_STYLE, 0, &styleStr[j]);
+ (*FcPatternGetString)(fontPattern, FC_FULLNAME, 0, &fullname[j]);
+ }
+
+ /* Once we get here 'fontCount' is the number of returned fonts
+ * we actually want to use, so we create 'fcFontArr' of that length.
+ * The non-null entries of "family[]" etc are those fonts.
+ * Then loop again over all nfonts adding just those non-null ones
+ * to 'fcFontArr'. If its null (we didn't want the font)
+ * then we don't enter the main body.
+ * So we should never get more than 'fontCount' entries.
+ */
+ if (includeFallbacks) {
+ fcFontArr =
+ (*env)->NewObjectArray(env, fontCount, fcFontClass, NULL);
+ (*env)->SetObjectField(env,fcCompFontObj, fcAllFontsID, fcFontArr);
+ }
+ fn=0;
+
+ for (j=0;jNewObject(env, fcFontClass, fcFontCons);
+ jstr = (*env)->NewStringUTF(env, (const char*)family[j]);
+ (*env)->SetObjectField(env, fcFont, familyNameID, jstr);
+ if (file[j] != NULL) {
+ jstr = (*env)->NewStringUTF(env, (const char*)file[j]);
+ (*env)->SetObjectField(env, fcFont, fontFileID, jstr);
+ }
+ if (styleStr[j] != NULL) {
+ jstr = (*env)->NewStringUTF(env, (const char*)styleStr[j]);
+ (*env)->SetObjectField(env, fcFont, styleNameID, jstr);
+ }
+ if (fullname[j] != NULL) {
+ jstr = (*env)->NewStringUTF(env, (const char*)fullname[j]);
+ (*env)->SetObjectField(env, fcFont, fullNameID, jstr);
+ }
+ if (fn==0) {
+ (*env)->SetObjectField(env, fcCompFontObj,
+ fcFirstFontID, fcFont);
+ }
+ if (includeFallbacks) {
+ (*env)->SetObjectArrayElement(env, fcFontArr, fn++,fcFont);
+ }
}
- (*FcPatternDestroy)(matchPattern);
}
(*env)->ReleaseStringUTFChars (env, fcNameStr, (const char*)fcName);
+ (*FcFontSetDestroy)(fontset);
(*FcPatternDestroy)(pattern);
+ free(family);
+ free(styleStr);
+ free(fullname);
+ free(file);
}
/* release resources and close the ".so" */
diff --git a/jdk/src/solaris/native/sun/java2d/opengl/GLXGraphicsConfig.c b/jdk/src/solaris/native/sun/java2d/opengl/GLXGraphicsConfig.c
index 6339582941b..6e4376fb051 100644
--- a/jdk/src/solaris/native/sun/java2d/opengl/GLXGraphicsConfig.c
+++ b/jdk/src/solaris/native/sun/java2d/opengl/GLXGraphicsConfig.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -471,7 +471,7 @@ Java_sun_java2d_opengl_GLXGraphicsConfig_getGLXConfigInfo(JNIEnv *env,
GLXContext context;
GLXPbuffer scratch;
GLXGraphicsConfigInfo *glxinfo;
- jint caps = sun_java2d_opengl_OGLContext_CAPS_EMPTY;
+ jint caps = CAPS_EMPTY;
int db, alpha;
const unsigned char *versionstr;
@@ -581,11 +581,11 @@ Java_sun_java2d_opengl_GLXGraphicsConfig_getGLXConfigInfo(JNIEnv *env,
// get config-specific capabilities
j2d_glXGetFBConfigAttrib(awt_display, fbconfig, GLX_DOUBLEBUFFER, &db);
if (db) {
- caps |= sun_java2d_opengl_OGLContext_CAPS_DOUBLEBUFFERED;
+ caps |= CAPS_DOUBLEBUFFERED;
}
j2d_glXGetFBConfigAttrib(awt_display, fbconfig, GLX_ALPHA_SIZE, &alpha);
if (alpha > 0) {
- caps |= sun_java2d_opengl_OGLContext_CAPS_STORED_ALPHA;
+ caps |= CAPS_STORED_ALPHA;
}
// initialize the OGLContext, which wraps the GLXFBConfig and GLXContext
@@ -662,11 +662,11 @@ Java_sun_java2d_opengl_GLXGraphicsConfig_getOGLCapabilities(JNIEnv *env,
J2dTraceLn(J2D_TRACE_INFO, "GLXGraphicsConfig_getOGLCapabilities");
if (glxinfo == NULL || glxinfo->context == NULL) {
- return sun_java2d_opengl_OGLContext_CAPS_EMPTY;
+ return CAPS_EMPTY;
}
return glxinfo->context->caps;
#else
- return sun_java2d_opengl_OGLContext_CAPS_EMPTY;
+ return CAPS_EMPTY;
#endif /* !HEADLESS */
}
diff --git a/jdk/src/solaris/native/sun/java2d/opengl/GLXSurfaceData.c b/jdk/src/solaris/native/sun/java2d/opengl/GLXSurfaceData.c
index 83e548017dd..a8c303f94bb 100644
--- a/jdk/src/solaris/native/sun/java2d/opengl/GLXSurfaceData.c
+++ b/jdk/src/solaris/native/sun/java2d/opengl/GLXSurfaceData.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -47,6 +47,9 @@ extern DisposeFunc OGLSD_Dispose;
extern struct MComponentPeerIDs mComponentPeerIDs;
+extern void
+ OGLSD_SetNativeDimensions(JNIEnv *env, OGLSDOps *oglsdo, jint w, jint h);
+
jboolean surfaceCreationFailed = JNI_FALSE;
#endif /* !HEADLESS */
@@ -460,6 +463,8 @@ Java_sun_java2d_opengl_GLXSurfaceData_initPbuffer
glxsdo->drawable = pbuffer;
glxsdo->xdrawable = 0;
+ OGLSD_SetNativeDimensions(env, oglsdo, width, height);
+
return JNI_TRUE;
}
diff --git a/jdk/src/windows/classes/sun/awt/Win32GraphicsConfig.java b/jdk/src/windows/classes/sun/awt/Win32GraphicsConfig.java
index fbd5adac5e3..d24ce0eb1bb 100644
--- a/jdk/src/windows/classes/sun/awt/Win32GraphicsConfig.java
+++ b/jdk/src/windows/classes/sun/awt/Win32GraphicsConfig.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1997-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1997-2008 Sun Microsystems, Inc. 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
@@ -55,9 +55,7 @@ import sun.java2d.InvalidPipeException;
import sun.java2d.loops.RenderLoops;
import sun.java2d.loops.SurfaceType;
import sun.java2d.loops.CompositeType;
-import sun.java2d.windows.Win32SurfaceData;
-import sun.java2d.windows.WinBackBuffer;
-import sun.java2d.windows.WindowsFlags;
+import sun.java2d.windows.GDIWindowSurfaceData;
/**
* This is an implementation of a GraphicsConfiguration object for a
@@ -72,8 +70,6 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
protected Win32GraphicsDevice screen;
protected int visual; //PixelFormatID
protected RenderLoops solidloops;
- private static BufferCapabilities bufferCaps;
- private static ImageCapabilities imageCaps;
private static native void initIDs();
@@ -170,32 +166,12 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
case Transparency.BITMASK:
return new DirectColorModel(25, 0xff0000, 0xff00, 0xff, 0x1000000);
case Transparency.TRANSLUCENT:
- return getTranslucentColorModel();
+ return ColorModel.getRGBdefault();
default:
return null;
}
}
- private static final int DCM_4444_RED_MASK = 0x0f00;
- private static final int DCM_4444_GRN_MASK = 0x00f0;
- private static final int DCM_4444_BLU_MASK = 0x000f;
- private static final int DCM_4444_ALP_MASK = 0xf000;
- static ColorModel translucentCM = null;
- public static ColorModel getTranslucentColorModel() {
- if (WindowsFlags.getD3DTexBpp() == 16) {
- if (translucentCM == null) {
- translucentCM = new DirectColorModel(16,
- DCM_4444_RED_MASK,
- DCM_4444_GRN_MASK,
- DCM_4444_BLU_MASK,
- DCM_4444_ALP_MASK);
- }
- return translucentCM;
- } else {
- return ColorModel.getRGBdefault();
- }
- }
-
/**
* Returns the default Transform for this configuration. This
* Transform is typically the Identity transform for most normal
@@ -246,44 +222,6 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
return getBounds(screen.getScreen());
}
- private static class DDrawBufferCapabilities extends BufferCapabilities {
- public DDrawBufferCapabilities(ImageCapabilities imageCaps) {
- super(imageCaps, imageCaps, FlipContents.PRIOR);
- }
- public boolean isFullScreenRequired() { return true; }
- public boolean isMultiBufferAvailable() { return true; }
- }
-
- private static class DDrawImageCapabilities extends ImageCapabilities {
- public DDrawImageCapabilities() {
- super(true);
- }
- public boolean isTrueVolatile() { return true; }
- }
-
- public BufferCapabilities getBufferCapabilities() {
- if (bufferCaps == null) {
- if (WindowsFlags.isDDEnabled()) {
- bufferCaps = new DDrawBufferCapabilities(
- getImageCapabilities());
- } else {
- bufferCaps = super.getBufferCapabilities();
- }
- }
- return bufferCaps;
- }
-
- public ImageCapabilities getImageCapabilities() {
- if (imageCaps == null) {
- if (WindowsFlags.isDDEnabled()) {
- imageCaps = new DDrawImageCapabilities();
- } else {
- imageCaps = super.getImageCapabilities();
- }
- }
- return imageCaps;
- }
-
public synchronized void displayChanged() {
solidloops = null;
}
@@ -305,11 +243,11 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
public SurfaceData createSurfaceData(WComponentPeer peer,
int numBackBuffers)
{
- return Win32SurfaceData.createData(peer, numBackBuffers);
+ return GDIWindowSurfaceData.createData(peer);
}
/**
- * Creates a new hidden-acceleration image of the given width and height
+ * Creates a new managed image of the given width and height
* that is associated with the target Component.
*/
public Image createAcceleratedImage(Component target,
@@ -327,15 +265,6 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
* WComponentPeer.java...
*/
- private boolean isFullScreenExclusive(Component target) {
- Win32GraphicsDevice gd = (Win32GraphicsDevice)getDevice();
- while (target != null && !(target instanceof Window)) {
- target = target.getParent();
- }
- return (target == gd.getFullScreenWindow() &&
- gd.isDDEnabledOnDevice());
- }
-
/**
* Checks that the requested configuration is natively supported; if not,
* an AWTException is thrown.
@@ -345,51 +274,61 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
BufferCapabilities caps)
throws AWTException
{
- if (!isFullScreenExclusive(target)) {
- throw new AWTException(
- "The operation requested is only supported on a full-screen" +
- " exclusive window");
- }
+ // the default pipeline doesn't support flip buffer strategy
+ throw new AWTException(
+ "The operation requested is not supported");
}
/**
- * Creates a backbuffer for the given peer and returns the image wrapper.
+ * This method is called from WComponentPeer when a surface data is replaced
+ * REMIND: while the default pipeline doesn't support flipping, it may
+ * happen that the accelerated device may have this graphics config
+ * (like if the device restoration failed when one device exits fs mode
+ * while others remain).
*/
public VolatileImage createBackBuffer(WComponentPeer peer) {
- // Create the back buffer object
- return new WinBackBuffer((Component)peer.getTarget(),
- (Win32SurfaceData)peer.getSurfaceData());
+ Component target = (Component)peer.getTarget();
+ return new SunVolatileImage(target,
+ target.getWidth(), target.getHeight(),
+ Boolean.TRUE);
}
/**
* Performs the native flip operation for the given target Component.
+ *
+ * REMIND: we should really not get here because that would mean that
+ * a FLIP BufferStrategy has been created, and one could only be created
+ * if accelerated pipeline is present but in some rare (and transitional)
+ * cases it may happen that the accelerated graphics device may have a
+ * default graphics configuraiton, so this is just a precaution.
*/
public void flip(WComponentPeer peer,
Component target, VolatileImage backBuffer,
+ int x1, int y1, int x2, int y2,
BufferCapabilities.FlipContents flipAction)
{
- int width = target.getWidth();
- int height = target.getHeight();
- if (flipAction == BufferCapabilities.FlipContents.COPIED) {
- Graphics g = target.getGraphics();
- g.drawImage(backBuffer, 0, 0, width, height, null);
- g.dispose();
- return;
- }
- Win32SurfaceData sd = (Win32SurfaceData)peer.getSurfaceData();
- try {
- sd.flip(((WinBackBuffer)backBuffer).getHWSurfaceData());
- } catch (sun.java2d.InvalidPipeException e) {
- // copy software surface to the screen via gdi blit
- Graphics g = target.getGraphics();
- g.drawImage(backBuffer, 0, 0, width, height, null);
- g.dispose();
- }
- if (flipAction == BufferCapabilities.FlipContents.BACKGROUND) {
+ if (flipAction == BufferCapabilities.FlipContents.COPIED ||
+ flipAction == BufferCapabilities.FlipContents.UNDEFINED) {
+ Graphics g = peer.getGraphics();
+ try {
+ g.drawImage(backBuffer,
+ x1, y1, x2, y2,
+ x1, y1, x2, y2,
+ null);
+ } finally {
+ g.dispose();
+ }
+ } else if (flipAction == BufferCapabilities.FlipContents.BACKGROUND) {
Graphics g = backBuffer.getGraphics();
- g.setColor(target.getBackground());
- g.fillRect(0, 0, width, height);
- g.dispose();
+ try {
+ g.setColor(target.getBackground());
+ g.fillRect(0, 0,
+ backBuffer.getWidth(),
+ backBuffer.getHeight());
+ } finally {
+ g.dispose();
+ }
}
+ // the rest of the flip actions are not supported
}
}
diff --git a/jdk/src/windows/classes/sun/awt/Win32GraphicsDevice.java b/jdk/src/windows/classes/sun/awt/Win32GraphicsDevice.java
index 098124296ae..ba4769813d8 100644
--- a/jdk/src/windows/classes/sun/awt/Win32GraphicsDevice.java
+++ b/jdk/src/windows/classes/sun/awt/Win32GraphicsDevice.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1997-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1997-2008 Sun Microsystems, Inc. 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
@@ -33,12 +33,14 @@ import java.awt.DisplayMode;
import java.awt.Frame;
import java.awt.Rectangle;
import java.awt.Window;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.event.WindowListener;
import java.awt.image.ColorModel;
import java.util.ArrayList;
import java.util.Vector;
import java.awt.peer.WindowPeer;
import sun.awt.windows.WWindowPeer;
-import sun.java2d.d3d.D3DContext;
import sun.java2d.opengl.WGLGraphicsConfig;
import sun.java2d.windows.WindowsFlags;
@@ -54,13 +56,11 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
int screen;
ColorModel dynamicColorModel; // updated with dev changes
ColorModel colorModel; // static for device
- GraphicsConfiguration[] configs;
- GraphicsConfiguration defaultConfig;
- boolean offscreenAccelerationEnabled = true;
- private D3DContext d3dContext;
+ protected GraphicsConfiguration[] configs;
+ protected GraphicsConfiguration defaultConfig;
private final String idString;
- private final String descString;
+ protected String descString;
// Note that we do not synchronize access to this variable - it doesn't
// really matter if a thread does an operation on graphics device which is
// about to become invalid (or already become) - we are prepared to deal
@@ -69,12 +69,16 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
// keep track of top-level windows on this display
private SunDisplayChanger topLevels = new SunDisplayChanger();
- private static boolean pfDisabled;
+ // REMIND: we may disable the use of pixel formats for some accelerated
+ // pipelines which are mutually exclusive with opengl, for which
+ // pixel formats were added in the first place
+ protected static boolean pfDisabled;
private static AWTPermission fullScreenExclusivePermission;
- private Rectangle ownerWindowedModeBounds = null;
// the original display mode we had before entering the fullscreen
// mode
private DisplayMode defaultDisplayMode;
+ // activation/deactivation listener for the full-screen window
+ private WindowListener fsWindowListener;
static {
@@ -91,16 +95,6 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
private static native void initIDs();
- /**
- * Acceleration can be disabled due to capabilities of the display
- * device discovered during ddraw initialization. This is not the
- * same as isDDEnabledOnDevice(), which returns false when ddraw
- * was disabled by the user or had problems initializing.
- */
- public boolean isOffscreenAccelerationEnabled() {
- return offscreenAccelerationEnabled;
- }
-
native void initDevice(int screen);
public Win32GraphicsDevice(int screennum) {
@@ -109,6 +103,7 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
// to reflect the original screen number (which may change if the
// device is removed)
idString = "\\Display"+screen;
+ // REMIND: may be should use class name?
descString = "Win32GraphicsDevice[screen=" + screen;
valid = true;
@@ -136,7 +131,7 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
* Returns whether this is a valid devicie. Device can become
* invalid as a result of device removal event.
*/
- boolean isValid() {
+ public boolean isValid() {
return valid;
}
@@ -145,7 +140,7 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
*
* @param defaultScreen the current default screen
*/
- void invalidate(int defaultScreen) {
+ protected void invalidate(int defaultScreen) {
valid = false;
screen = defaultScreen;
}
@@ -178,8 +173,7 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
int defaultPixID = getDefaultPixID(screen);
Vector v = new Vector( max );
if (defaultPixID == 0) {
- // Workaround for failing GDI calls, or if DirectDraw
- // is disabled
+ // Workaround for failing GDI calls
defaultConfig = Win32GraphicsConfig.getConfig(this,
defaultPixID);
v.addElement(defaultConfig);
@@ -309,32 +303,6 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
getLocalGraphicsEnvironment().getDefaultScreenDevice());
}
- private native boolean isDDEnabledOnDeviceNative(int screen);
-
- public D3DContext getD3DContext() {
- if (d3dContext == null) {
- d3dContext = new D3DContext(this);
- }
- return d3dContext;
- }
-
-
- public boolean isDDEnabledOnDevice() {
- return (WindowsFlags.isDDEnabled() && isValid() &&
- isDDEnabledOnDeviceNative(screen));
- }
-
- public boolean isD3DEnabledOnDevice() {
- // The conditions under which we enable the D3D pipeline for the device:
- // - d3d is not disabled via a flag
- // - either d3d is forced via property or we're in fullscreen mode
- // - the hardware/drivers meet our requirements
- return (WindowsFlags.isD3DEnabled() && isValid() &&
- (WindowsFlags.isD3DSet() || getFullScreenWindow() != null) &&
- ((getD3DContext().getDeviceCaps() &
- D3DContext.J2D_D3D_ENABLED_OK) != 0));
- }
-
private static boolean isFSExclusiveModeAllowed() {
SecurityManager security = System.getSecurityManager();
if (security != null) {
@@ -352,31 +320,14 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
}
/**
- * We support the exclusive fullscreen mode in both ddraw and
- * noddraw modes, so we return true unless we're not allowed to use
- * fullscreen mode.
+ * returns true unless we're not allowed to use fullscreen mode.
*/
+ @Override
public boolean isFullScreenSupported() {
return isFSExclusiveModeAllowed();
}
- /**
- * Return the owning Frame for a given Window. Used in setFSWindow below
- * to set the properties of the owning Frame when a Window goes
- * into fullscreen mode.
- */
- private Frame getToplevelOwner(Window w) {
- Window owner = w;
- while (owner != null) {
- owner = owner.getOwner();
- if (owner instanceof Frame) {
- return (Frame) owner;
- }
- }
- // Should not get here, but return something intelligent just in case
- return null;
- }
-
+ @Override
public synchronized void setFullScreenWindow(Window w) {
Window old = getFullScreenWindow();
if (w == old) {
@@ -402,51 +353,25 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
}
WWindowPeer peer = (WWindowPeer)old.getPeer();
if (peer != null) {
+ // we used to destroy the buffers on exiting fs mode, this
+ // is no longer needed since fs change will cause a surface
+ // data replacement
synchronized(peer) {
- peer.destroyBuffers();
- exitFullScreenExclusive(isDDEnabledOnDevice(),
- screen, peer);
+ exitFullScreenExclusive(screen, peer);
}
}
- /**
- * Bug 4933099: There is some funny-business to deal with when this
- * method is called with a Window instead of a Frame. See 4836744
- * for more information on this. One side-effect of our workaround
- * for the problem is that the owning Frame of a Window may end
- * up getting resized during the fullscreen process. When we
- * return from fullscreen mode, we should resize the Frame to
- * its original size (just like the Window is being resized
- * to its original size in GraphicsDevice).
- */
- if (!(old instanceof Frame)) {
- Frame owner = getToplevelOwner(old);
- if (owner != null && ownerWindowedModeBounds != null) {
- owner.setBounds(ownerWindowedModeBounds);
- }
- ownerWindowedModeBounds = null;
- }
+ removeFSWindowListener(old);
}
super.setFullScreenWindow(w);
if (w != null) {
// always record the default display mode prior to going
// fullscreen
defaultDisplayMode = getDisplayMode();
- // Bug 4933099
- if (!(w instanceof Frame)) {
- Frame owner = getToplevelOwner(w);
- if (owner != null) {
- ownerWindowedModeBounds = owner.getBounds();
- // These will get set for the native window in
- // any case. Set them here so that resetting them
- // later actually does the right thing
- owner.setBounds(w.getBounds());
- }
- }
+ addFSWindowListener(w);
// Enter full screen exclusive mode.
WWindowPeer peer = (WWindowPeer)w.getPeer();
synchronized(peer) {
- enterFullScreenExclusive(isDDEnabledOnDevice(),
- screen, peer);
+ enterFullScreenExclusive(screen, peer);
// Note: removed replaceSurfaceData() call because
// changing the window size or making it visible
// will cause this anyway, and both of these events happen
@@ -463,15 +388,18 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
// tree-lock and should never lock on any resources which are
// required by other threads which may have them and may require
// the tree-lock.
- private native void enterFullScreenExclusive(boolean useDD,
- int screen, WindowPeer w);
- private native void exitFullScreenExclusive(boolean useDD,
- int screen, WindowPeer w);
+ // REMIND: in the future these methods may need to become protected so that
+ // subclasses could override them and use appropriate api other than GDI
+ // for implementing these functions.
+ protected native void enterFullScreenExclusive(int screen, WindowPeer w);
+ protected native void exitFullScreenExclusive(int screen, WindowPeer w);
+ @Override
public boolean isDisplayChangeSupported() {
return (isFullScreenSupported() && getFullScreenWindow() != null);
}
+ @Override
public synchronized void setDisplayMode(DisplayMode dm) {
if (!isDisplayChangeSupported()) {
super.setDisplayMode(dm);
@@ -501,22 +429,19 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
}
}
- private native DisplayMode getCurrentDisplayMode(int screen);
- private native void configDisplayMode(int screen, WindowPeer w, int width,
+ protected native DisplayMode getCurrentDisplayMode(int screen);
+ protected native void configDisplayMode(int screen, WindowPeer w, int width,
int height, int bitDepth,
int refreshRate);
- private native void enumDisplayModes(int screen, ArrayList modes);
- // This function is only available if DirectDraw is enabled, otherwise we
- // have to do the work the hard way (enumerating all of the display modes
- // and checking each one)
- private native boolean isDisplayModeAvailable(int screen, int width, int height,
- int bitDepth, int refreshRate);
+ protected native void enumDisplayModes(int screen, ArrayList modes);
+ @Override
public synchronized DisplayMode getDisplayMode() {
DisplayMode res = getCurrentDisplayMode(screen);
return res;
}
+ @Override
public synchronized DisplayMode[] getDisplayModes() {
ArrayList modes = new ArrayList();
enumDisplayModes(screen, modes);
@@ -528,33 +453,22 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
return retArray;
}
- private synchronized DisplayMode getMatchingDisplayMode(DisplayMode dm) {
+ protected synchronized DisplayMode getMatchingDisplayMode(DisplayMode dm) {
if (!isDisplayChangeSupported()) {
return null;
}
- if (isDDEnabledOnDevice()) {
- return
- isDisplayModeAvailable(screen, dm.getWidth(), dm.getHeight(),
- dm.getBitDepth(), dm.getRefreshRate())
- ? dm : null;
- } else {
- // The function isDisplayModeAvailable is only available if
- // DirectDraw is enabled, otherwise we have to do the work the
- // hard way (enumerating all of the display modes
- // and checking each one)
- DisplayMode[] modes = getDisplayModes();
- for (DisplayMode mode : modes) {
- if (dm.equals(mode) ||
- (dm.getRefreshRate() == DisplayMode.REFRESH_RATE_UNKNOWN &&
- dm.getWidth() == mode.getWidth() &&
- dm.getHeight() == mode.getHeight() &&
- dm.getBitDepth() == mode.getBitDepth()))
- {
- return mode;
- }
+ DisplayMode[] modes = getDisplayModes();
+ for (DisplayMode mode : modes) {
+ if (dm.equals(mode) ||
+ (dm.getRefreshRate() == DisplayMode.REFRESH_RATE_UNKNOWN &&
+ dm.getWidth() == mode.getWidth() &&
+ dm.getHeight() == mode.getHeight() &&
+ dm.getBitDepth() == mode.getBitDepth()))
+ {
+ return mode;
}
- return null;
}
+ return null;
}
/*
@@ -563,7 +477,6 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
* changed.
*/
public void displayChanged() {
- d3dContext = null;
dynamicColorModel = null;
defaultConfig = null;
configs = null;
@@ -621,17 +534,93 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
return colorModel;
}
- private native int getDeviceMemoryNative(int screen);
+ /**
+ * WindowAdapter class responsible for de/iconifying full-screen window
+ * of this device.
+ *
+ * The listener restores the default display mode when window is iconified
+ * and sets it back to the one set by the user on de-iconification.
+ */
+ private static class Win32FSWindowAdapter extends WindowAdapter {
+ private Win32GraphicsDevice device;
+ private DisplayMode dm;
+
+ Win32FSWindowAdapter(Win32GraphicsDevice device) {
+ this.device = device;
+ }
+
+ private void setFSWindowsState(Window other, int state) {
+ GraphicsDevice gds[] =
+ GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getScreenDevices();
+ // check if the de/activation was caused by other
+ // fs window and ignore the event if that's the case
+ if (other != null) {
+ for (GraphicsDevice gd : gds) {
+ if (other == gd.getFullScreenWindow()) {
+ return;
+ }
+ }
+ }
+ // otherwise apply state to all fullscreen windows
+ for (GraphicsDevice gd : gds) {
+ Window fsw = gd.getFullScreenWindow();
+ if (fsw instanceof Frame) {
+ ((Frame)fsw).setExtendedState(state);
+ }
+ }
+ }
+
+ @Override
+ public void windowDeactivated(WindowEvent e) {
+ setFSWindowsState(e.getOppositeWindow(), Frame.ICONIFIED);
+ }
+
+ @Override
+ public void windowActivated(WindowEvent e) {
+ setFSWindowsState(e.getOppositeWindow(), Frame.NORMAL);
+ }
+
+ @Override
+ public void windowIconified(WindowEvent e) {
+ // restore the default display mode for this device
+ DisplayMode ddm = device.defaultDisplayMode;
+ if (ddm != null) {
+ dm = device.getDisplayMode();
+ device.setDisplayMode(ddm);
+ }
+ }
+
+ @Override
+ public void windowDeiconified(WindowEvent e) {
+ // restore the user-set display mode for this device
+ if (dm != null) {
+ device.setDisplayMode(dm);
+ dm = null;
+ }
+ }
+ }
/**
- * Returns number of bytes available in VRAM on this device.
+ * Adds a WindowListener to be used as
+ * activation/deactivation listener for the current full-screen window.
+ *
+ * @param w full-screen window
*/
- public int getAvailableAcceleratedMemory() {
- if (getDefaultConfiguration() instanceof WGLGraphicsConfig) {
- // when OGL is enabled, there is no way to determine the amount
- // of accelerated memory, so just return the default value
- return super.getAvailableAcceleratedMemory();
- }
- return getDeviceMemoryNative(screen);
+ protected void addFSWindowListener(Window w) {
+ // Note: even though we create a listener for Window instances of
+ // fs windows they will not receive window events.
+ fsWindowListener = new Win32FSWindowAdapter(this);
+ w.addWindowListener(fsWindowListener);
+ }
+
+ /**
+ * Removes the fs window listener.
+ *
+ * @param w full-screen window
+ */
+ protected void removeFSWindowListener(Window w) {
+ w.removeWindowListener(fsWindowListener);
+ fsWindowListener = null;
}
}
diff --git a/jdk/src/windows/classes/sun/awt/Win32GraphicsEnvironment.java b/jdk/src/windows/classes/sun/awt/Win32GraphicsEnvironment.java
index d17b8b00aea..80ee7747004 100644
--- a/jdk/src/windows/classes/sun/awt/Win32GraphicsEnvironment.java
+++ b/jdk/src/windows/classes/sun/awt/Win32GraphicsEnvironment.java
@@ -25,9 +25,11 @@
package sun.awt;
+import java.awt.GraphicsConfiguration;
import java.awt.GraphicsDevice;
import java.awt.GraphicsEnvironment;
import java.awt.Toolkit;
+import java.awt.peer.ComponentPeer;
import java.io.File;
import java.io.IOException;
import java.lang.ref.WeakReference;
@@ -44,6 +46,7 @@ import sun.font.FontManager;
import sun.java2d.SunGraphicsEnvironment;
import sun.java2d.SurfaceManagerFactory;
import sun.java2d.WindowsSurfaceManagerFactory;
+import sun.java2d.d3d.D3DGraphicsDevice;
import sun.java2d.windows.WindowsFlags;
/**
@@ -333,12 +336,21 @@ public class Win32GraphicsEnvironment
protected static native void deRegisterFontWithPlatform(String fontName);
protected GraphicsDevice makeScreenDevice(int screennum) {
- return new Win32GraphicsDevice(screennum);
+ GraphicsDevice device = null;
+ if (WindowsFlags.isD3DEnabled()) {
+ device = D3DGraphicsDevice.createDevice(screennum);
+ }
+ if (device == null) {
+ device = new Win32GraphicsDevice(screennum);
+ }
+ return device;
}
// Implements SunGraphicsEnvironment.createFontConfiguration.
protected FontConfiguration createFontConfiguration() {
- return new WFontConfiguration(this);
+ FontConfiguration fc = new WFontConfiguration(this);
+ fc.init();
+ return fc;
}
public FontConfiguration createFontConfiguration(boolean preferLocaleFonts,
@@ -346,4 +358,39 @@ public class Win32GraphicsEnvironment
return new WFontConfiguration(this, preferLocaleFonts,preferPropFonts);
}
+
+ @Override
+ public boolean isFlipStrategyPreferred(ComponentPeer peer) {
+ GraphicsConfiguration gc;
+ if (peer != null && (gc = peer.getGraphicsConfiguration()) != null) {
+ GraphicsDevice gd = gc.getDevice();
+ if (gd instanceof D3DGraphicsDevice) {
+ return ((D3DGraphicsDevice)gd).isD3DEnabledOnDevice();
+ }
+ }
+ return false;
+ }
+
+ private static volatile boolean isDWMCompositionEnabled;
+ /**
+ * Returns true if dwm composition is currently enabled, false otherwise.
+ *
+ * @return true if dwm composition is enabled, false otherwise
+ */
+ public static boolean isDWMCompositionEnabled() {
+ return isDWMCompositionEnabled;
+ }
+
+ /**
+ * Called from the native code when DWM composition state changed.
+ * May be called multiple times during the lifetime of the application.
+ * REMIND: we may want to create a listener mechanism for this.
+ *
+ * Note: called on the Toolkit thread, no user code or locks are allowed.
+ *
+ * @param enabled indicates the state of dwm composition
+ */
+ private static void dwmCompositionChanged(boolean enabled) {
+ isDWMCompositionEnabled = enabled;
+ }
}
diff --git a/jdk/src/windows/classes/sun/awt/windows/WComponentPeer.java b/jdk/src/windows/classes/sun/awt/windows/WComponentPeer.java
index 5b35c1e7210..0f71eda29de 100644
--- a/jdk/src/windows/classes/sun/awt/windows/WComponentPeer.java
+++ b/jdk/src/windows/classes/sun/awt/windows/WComponentPeer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1996-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1996-2008 Sun Microsystems, Inc. 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
@@ -40,8 +40,10 @@ import java.awt.event.InvocationEvent;
import java.awt.event.KeyEvent;
import sun.awt.Win32GraphicsConfig;
import sun.java2d.InvalidPipeException;
-import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
+import sun.java2d.d3d.D3DScreenUpdateManager;
+import static sun.java2d.d3d.D3DSurfaceData.*;
+import sun.java2d.ScreenUpdateManager;
import sun.java2d.opengl.OGLSurfaceData;
import sun.awt.DisplayChangedListener;
import sun.awt.PaintEventDispatcher;
@@ -51,7 +53,6 @@ import java.awt.dnd.DropTarget;
import java.awt.dnd.peer.DropTargetPeer;
import sun.awt.ComponentAccessor;
-import java.util.logging.*;
import java.util.logging.*;
@@ -87,6 +88,7 @@ public abstract class WComponentPeer extends WObjectPeer
int oldHeight = -1;
private int numBackBuffers = 0;
private VolatileImage backBuffer = null;
+ private BufferCapabilities backBufferCaps = null;
// foreground, background and color are cached to avoid calling back
// into the Component.
@@ -191,10 +193,12 @@ public abstract class WComponentPeer extends WObjectPeer
cont.invalidate();
cont.validate();
- if (surfaceData instanceof OGLSurfaceData) {
- // 6290245: When OGL is enabled, it is necessary to
+ if (surfaceData instanceof D3DWindowSurfaceData ||
+ surfaceData instanceof OGLSurfaceData)
+ {
+ // When OGL or D3D is enabled, it is necessary to
// replace the SurfaceData for each dynamic layout
- // request so that the OGL viewport stays in sync
+ // request so that the viewport stays in sync
// with the window bounds.
try {
replaceSurfaceData();
@@ -377,7 +381,7 @@ public abstract class WComponentPeer extends WObjectPeer
* just call that version with our current numBackBuffers.
*/
public void replaceSurfaceData() {
- replaceSurfaceData(this.numBackBuffers);
+ replaceSurfaceData(this.numBackBuffers, this.backBufferCaps);
}
/**
@@ -386,35 +390,51 @@ public abstract class WComponentPeer extends WObjectPeer
* order, but also needs to perform additional functions inside the
* locks.
*/
- public void replaceSurfaceData(int newNumBackBuffers) {
+ public void replaceSurfaceData(int newNumBackBuffers,
+ BufferCapabilities caps)
+ {
+ SurfaceData oldData = null;
+ VolatileImage oldBB = null;
synchronized(((Component)target).getTreeLock()) {
synchronized(this) {
if (pData == 0) {
return;
}
numBackBuffers = newNumBackBuffers;
- SurfaceData oldData = surfaceData;
Win32GraphicsConfig gc =
- (Win32GraphicsConfig)getGraphicsConfiguration();
- surfaceData = gc.createSurfaceData(this, numBackBuffers);
+ (Win32GraphicsConfig)getGraphicsConfiguration();
+ ScreenUpdateManager mgr = ScreenUpdateManager.getInstance();
+ oldData = surfaceData;
+ mgr.dropScreenSurface(oldData);
+ surfaceData =
+ mgr.createScreenSurface(gc, this, numBackBuffers, true);
if (oldData != null) {
oldData.invalidate();
- // null out the old data to make it collected faster
- oldData = null;
}
- if (backBuffer != null) {
- // this will remove the back buffer from the
- // display change notification list
- backBuffer.flush();
+ oldBB = backBuffer;
+ if (numBackBuffers > 0) {
+ // set the caps first, they're used when creating the bb
+ backBufferCaps = caps;
+ backBuffer = gc.createBackBuffer(this);
+ } else if (backBuffer != null) {
+ backBufferCaps = null;
backBuffer = null;
}
-
- if (numBackBuffers > 0) {
- backBuffer = gc.createBackBuffer(this);
- }
}
}
+ // it would be better to do this before we create new ones,
+ // but then we'd run into deadlock issues
+ if (oldData != null) {
+ oldData.flush();
+ // null out the old data to make it collected faster
+ oldData = null;
+ }
+ if (oldBB != null) {
+ oldBB.flush();
+ // null out the old data to make it collected faster
+ oldData = null;
+ }
}
public void replaceSurfaceDataLater() {
@@ -518,7 +538,10 @@ public abstract class WComponentPeer extends WObjectPeer
if (font == null) {
font = defaultFont;
}
- return new SunGraphics2D(surfaceData, fgColor, bgColor, font);
+ ScreenUpdateManager mgr =
+ ScreenUpdateManager.getInstance();
+ return mgr.createGraphics(surfaceData, this, fgColor,
+ bgColor, font);
}
return null;
}
@@ -530,6 +553,7 @@ public abstract class WComponentPeer extends WObjectPeer
protected void disposeImpl() {
SurfaceData oldData = surfaceData;
surfaceData = null;
+ ScreenUpdateManager.getInstance().dropScreenSurface(oldData);
oldData.invalidate();
// remove from updater before calling targetDisposedPeer
WToolkit.targetDisposedPeer(target, this);
@@ -546,6 +570,16 @@ public abstract class WComponentPeer extends WObjectPeer
_setBackground(c.getRGB());
}
+ /**
+ * This method is intentionally not synchronized as it is called while
+ * holding other locks.
+ *
+ * @see sun.java2d.d3d.D3DScreenUpdateManager#validate(D3DWindowSurfaceData)
+ */
+ public Color getBackgroundNoSync() {
+ return background;
+ }
+
public native void _setForeground(int rgb);
public native void _setBackground(int rgb);
@@ -617,8 +651,9 @@ public abstract class WComponentPeer extends WObjectPeer
checkCreation();
this.winGraphicsConfig =
(Win32GraphicsConfig)getGraphicsConfiguration();
- this.surfaceData =
- winGraphicsConfig.createSurfaceData(this, numBackBuffers);
+ ScreenUpdateManager mgr = ScreenUpdateManager.getInstance();
+ this.surfaceData = mgr.createScreenSurface(winGraphicsConfig, this,
+ numBackBuffers, false);
initialize();
start(); // Initialize enable/disable state, turn on callbacks
}
@@ -793,6 +828,7 @@ public abstract class WComponentPeer extends WObjectPeer
* native windowing system specific actions.
*/
+ @Override
public void createBuffers(int numBuffers, BufferCapabilities caps)
throws AWTException
{
@@ -802,39 +838,44 @@ public abstract class WComponentPeer extends WObjectPeer
// Re-create the primary surface with the new number of back buffers
try {
- replaceSurfaceData(numBuffers - 1);
+ replaceSurfaceData(numBuffers - 1, caps);
} catch (InvalidPipeException e) {
throw new AWTException(e.getMessage());
}
}
- public synchronized void destroyBuffers() {
- disposeBackBuffer();
- numBackBuffers = 0;
+ @Override
+ public void destroyBuffers() {
+ replaceSurfaceData(0, null);
}
- private synchronized void disposeBackBuffer() {
- if (backBuffer == null) {
- return;
- }
- backBuffer = null;
- }
-
- public synchronized void flip(BufferCapabilities.FlipContents flipAction) {
+ @Override
+ public void flip(int x1, int y1, int x2, int y2,
+ BufferCapabilities.FlipContents flipAction)
+ {
+ VolatileImage backBuffer = this.backBuffer;
if (backBuffer == null) {
throw new IllegalStateException("Buffers have not been created");
}
Win32GraphicsConfig gc =
(Win32GraphicsConfig)getGraphicsConfiguration();
- gc.flip(this, (Component)target, backBuffer, flipAction);
+ gc.flip(this, (Component)target, backBuffer, x1, y1, x2, y2, flipAction);
}
+ @Override
public synchronized Image getBackBuffer() {
+ Image backBuffer = this.backBuffer;
if (backBuffer == null) {
throw new IllegalStateException("Buffers have not been created");
}
return backBuffer;
}
+ public BufferCapabilities getBackBufferCaps() {
+ return backBufferCaps;
+ }
+ public int getBackBuffersNum() {
+ return numBackBuffers;
+ }
/* override and return false on components that DO NOT require
a clearRect() before painting (i.e. native components) */
@@ -866,6 +907,51 @@ public abstract class WComponentPeer extends WObjectPeer
sun.java2d.pipe.Region region);
+ // REMIND: Temp workaround for issues with using HW acceleration
+ // in the browser on Vista when DWM is enabled.
+ // @return true if the toplevel container is not an EmbeddedFrame or
+ // if this EmbeddedFrame is acceleration capable, false otherwise
+ private static final boolean isContainingTopLevelAccelCapable(Component c) {
+ while (c != null && !(c instanceof WEmbeddedFrame)) {
+ c = c.getParent();
+ }
+ if (c == null) {
+ return true;
+ }
+ return ((WEmbeddedFramePeer)c.getPeer()).isAccelCapable();
+ }
+
+ /**
+ * Returns whether this component is capable of being hw accelerated.
+ * More specifically, whether rendering to this component or a
+ * BufferStrategy's back-buffer for this component can be hw accelerated.
+ *
+ * Conditions which could prevent hw acceleration include the toplevel
+ * window containing this component being
+ * {@link com.sun.awt.AWTUtilities.Translucency#TRANSLUCENT TRANSLUCENT}.
+ *
+ * @return {@code true} if this component is capable of being hw
+ * accelerated, {@code false} otherwise
+ * @see com.sun.awt.AWTUtilities.Translucency#TRANSLUCENT
+ */
+ public boolean isAccelCapable() {
+ // REMIND: Temp workaround for issues with using HW acceleration
+ // in the browser on Vista when DWM is enabled
+ if (!isContainingTopLevelAccelCapable((Component)target)) {
+ return false;
+ }
+
+ // REMIND: translucent windows support-related
+/*
+ boolean isTranslucent =
+ SunToolkit.isContainingTopLevelTranslucent((Component)target);
+ // D3D/OGL and translucent windows interacted poorly in Windows XP;
+ // these problems are no longer present in Vista
+ return !isTranslucent || Win32GraphicsEnvironment.isVistaOS();
+*/
+ return true;
+ }
+
/**
* Applies the shape to the native component window.
* @since 1.7
diff --git a/jdk/src/windows/classes/sun/awt/windows/WEmbeddedFramePeer.java b/jdk/src/windows/classes/sun/awt/windows/WEmbeddedFramePeer.java
index c1d40b868fb..2e1e6e618dc 100644
--- a/jdk/src/windows/classes/sun/awt/windows/WEmbeddedFramePeer.java
+++ b/jdk/src/windows/classes/sun/awt/windows/WEmbeddedFramePeer.java
@@ -30,6 +30,7 @@ import java.awt.Graphics;
import java.awt.Rectangle;
import sun.awt.EmbeddedFrame;
+import sun.awt.Win32GraphicsEnvironment;
public class WEmbeddedFramePeer extends WFramePeer {
@@ -71,4 +72,13 @@ public class WEmbeddedFramePeer extends WFramePeer {
// We don't constrain the bounds of the EmbeddedFrames
return new Rectangle(x, y, width, height);
}
+
+ @Override
+ public boolean isAccelCapable() {
+ // REMIND: Temp workaround for issues with using HW acceleration
+ // in the browser on Vista when DWM is enabled
+ // Note: isDWMCompositionEnabled is only relevant on Vista, returns
+ // false on other systems.
+ return !Win32GraphicsEnvironment.isDWMCompositionEnabled();
+ }
}
diff --git a/jdk/src/windows/classes/sun/awt/windows/WToolkit.java b/jdk/src/windows/classes/sun/awt/windows/WToolkit.java
index 607cff8d774..5ea8ca3acc5 100644
--- a/jdk/src/windows/classes/sun/awt/windows/WToolkit.java
+++ b/jdk/src/windows/classes/sun/awt/windows/WToolkit.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1996-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1996-2008 Sun Microsystems, Inc. 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
@@ -40,6 +40,7 @@ import sun.awt.AWTAutoShutdown;
import sun.awt.SunToolkit;
import sun.awt.Win32GraphicsDevice;
import sun.awt.Win32GraphicsEnvironment;
+import sun.java2d.d3d.D3DRenderQueue;
import sun.java2d.opengl.OGLRenderQueue;
import sun.print.PrintJob2D;
@@ -591,6 +592,8 @@ public class WToolkit extends SunToolkit implements Runnable {
nativeSync();
// now flush the OGL pipeline (this is a no-op if OGL is not enabled)
OGLRenderQueue.sync();
+ // now flush the D3D pipeline (this is a no-op if D3D is not enabled)
+ D3DRenderQueue.sync();
}
public PrintJob getPrintJob(Frame frame, String doctitle,
@@ -910,8 +913,31 @@ public class WToolkit extends SunToolkit implements Runnable {
return toolkit;
}
+ /**
+ * There are two reasons why we don't use buffer per window when
+ * Vista's DWM (aka Aero) is enabled:
+ * - since with DWM all windows are already double-buffered, the application
+ * doesn't get expose events so we don't get to use our true back-buffer,
+ * wasting memory and performance (this is valid for both d3d and gdi
+ * pipelines)
+ * - in some cases with buffer per window enabled it is possible for the
+ * paint manager to redirect rendering to the screen for some operations
+ * (like copyArea), and since bpw uses its own BufferStrategy the
+ * d3d onscreen rendering support is disabled and rendering goes through
+ * GDI. This doesn't work well with Vista's DWM since one
+ * can not perform GDI and D3D operations on the same surface
+ * (see 6630702 for more info)
+ *
+ * Note: even though DWM composition state can change during the lifetime
+ * of the application it is a rare event, and it is more often that it
+ * is temporarily disabled (because of some app) than it is getting
+ * permanently enabled so we can live with this approach without the
+ * complexity of dwm state listeners and such. This can be revisited if
+ * proved otherwise.
+ */
+ @Override
public boolean useBufferPerWindow() {
- return true;
+ return !Win32GraphicsEnvironment.isDWMCompositionEnabled();
}
public void grab(Window w) {
diff --git a/jdk/src/windows/classes/sun/java2d/ScreenUpdateManager.java b/jdk/src/windows/classes/sun/java2d/ScreenUpdateManager.java
new file mode 100644
index 00000000000..ab09a972fa0
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/ScreenUpdateManager.java
@@ -0,0 +1,131 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d;
+
+import java.awt.Color;
+import java.awt.Font;
+import java.awt.Graphics2D;
+import sun.awt.Win32GraphicsConfig;
+import sun.awt.windows.WComponentPeer;
+import sun.java2d.d3d.D3DScreenUpdateManager;
+import sun.java2d.windows.WindowsFlags;
+
+/**
+ * This class handles the creation of on-screen surfaces and
+ * corresponding graphics objects.
+ *
+ * By default it delegates the surface creation to the
+ * particular GraphicsConfiguration classes.
+ */
+public class ScreenUpdateManager {
+ private static ScreenUpdateManager theInstance;
+
+ protected ScreenUpdateManager() {
+ }
+
+ /**
+ * Creates a SunGraphics2D object for the surface,
+ * given the parameters.
+ *
+ * @param sd surface data for which a graphics is to be created
+ * @param peer peer which owns the surface
+ * @param fgColor fg color to be used in the graphics
+ * @param bgColor bg color to be used in the graphics
+ * @param font font to be used in the graphics
+ * @return a SunGraphics2D object for rendering to the passed surface
+ */
+ public synchronized Graphics2D createGraphics(SurfaceData sd,
+ WComponentPeer peer, Color fgColor, Color bgColor, Font font)
+ {
+ return new SunGraphics2D(sd, fgColor, bgColor, font);
+ }
+
+ /**
+ * Creates and returns the surface for the peer. This surface becomes
+ * managed by this manager. To remove the surface from the managed list
+ * {@code}dropScreenSurface(SurfaceData){@code} will need to be called.
+ *
+ * The default implementation delegates surface creation
+ * to the passed in GraphicsConfiguration object.
+ *
+ * @param gc graphics configuration for which the surface is to be created
+ * @param peer peer for which the onscreen surface is to be created
+ * @param bbNum number of back-buffers requested for this peer
+ * @param isResize whether this surface is being created in response to
+ * a component resize event
+ * @return a SurfaceData to be used for on-screen rendering for this peer.
+ * @see #dropScreenSurface(SurfaceData)
+ */
+ public SurfaceData createScreenSurface(Win32GraphicsConfig gc,
+ WComponentPeer peer, int bbNum,
+ boolean isResize)
+ {
+ return gc.createSurfaceData(peer, bbNum);
+ }
+
+ /**
+ * Drops the passed surface from the list of managed surfaces.
+ *
+ * Nothing happens if the surface wasn't managed by this manager.
+ *
+ * @param sd SurfaceData to be removed from the list of managed surfaces
+ */
+ public void dropScreenSurface(SurfaceData sd) {}
+
+ /**
+ * Returns a replacement SurfaceData for the invalid passed one.
+ *
+ * This method should be used by SurfaceData's created by
+ * the ScreenUpdateManager for providing replacement surfaces.
+ *
+ * @param peer to which the old surface belongs
+ * @param oldsd the old (invalid) surface to get replaced
+ * @return a replacement surface
+ * @see sun.java2d.d3d.D3DSurfaceData.D3DWindowSurfaceData#getReplacement()
+ * @see sun.java2d.windows.GDIWindowSurfaceData#getReplacement()
+ */
+ public SurfaceData getReplacementScreenSurface(WComponentPeer peer,
+ SurfaceData oldsd)
+ {
+ return peer.getSurfaceData();
+ }
+
+ /**
+ * Returns an (singleton) instance of the screen surfaces
+ * manager class.
+ * @return instance of onscreen surfaces manager
+ */
+ public static synchronized ScreenUpdateManager getInstance() {
+ if (theInstance == null) {
+ if (WindowsFlags.isD3DEnabled()) {
+ theInstance = new D3DScreenUpdateManager();
+ } else {
+ theInstance = new ScreenUpdateManager();
+ }
+ }
+ return theInstance;
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/WindowsSurfaceManagerFactory.java b/jdk/src/windows/classes/sun/java2d/WindowsSurfaceManagerFactory.java
index af0fd7bd400..b8e7207535c 100644
--- a/jdk/src/windows/classes/sun/java2d/WindowsSurfaceManagerFactory.java
+++ b/jdk/src/windows/classes/sun/java2d/WindowsSurfaceManagerFactory.java
@@ -26,14 +26,13 @@
package sun.java2d;
import java.awt.GraphicsConfiguration;
-import java.awt.image.BufferedImage;
+import sun.awt.image.BufImgVolatileSurfaceManager;
import sun.awt.image.SunVolatileImage;
-import sun.awt.image.SurfaceManager;
import sun.awt.image.VolatileSurfaceManager;
+import sun.java2d.d3d.D3DGraphicsConfig;
+import sun.java2d.d3d.D3DVolatileSurfaceManager;
import sun.java2d.opengl.WGLGraphicsConfig;
import sun.java2d.opengl.WGLVolatileSurfaceManager;
-import sun.java2d.windows.WindowsFlags;
-import sun.java2d.windows.WinVolatileSurfaceManager;
/**
* The SurfaceManagerFactory that creates VolatileSurfaceManager
@@ -54,10 +53,12 @@ public class WindowsSurfaceManagerFactory extends SurfaceManagerFactory {
Object context)
{
GraphicsConfiguration gc = vImg.getGraphicsConfig();
- if (gc instanceof WGLGraphicsConfig) {
+ if (gc instanceof D3DGraphicsConfig) {
+ return new D3DVolatileSurfaceManager(vImg, context);
+ } else if (gc instanceof WGLGraphicsConfig) {
return new WGLVolatileSurfaceManager(vImg, context);
} else {
- return new WinVolatileSurfaceManager(vImg, context);
+ return new BufImgVolatileSurfaceManager(vImg, context);
}
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DBackBufferSurfaceData.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DBackBufferSurfaceData.java
deleted file mode 100644
index 79914431dc7..00000000000
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DBackBufferSurfaceData.java
+++ /dev/null
@@ -1,87 +0,0 @@
-/*
- * Copyright 2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.d3d;
-
-import java.awt.GraphicsConfiguration;
-import java.awt.Image;
-import java.awt.Transparency;
-import java.awt.image.ColorModel;
-import sun.awt.Win32GraphicsDevice;
-import sun.java2d.loops.SurfaceType;
-import sun.java2d.windows.Win32SurfaceData;
-
-public class D3DBackBufferSurfaceData extends D3DSurfaceData {
-
- private Win32SurfaceData parentData;
-
- /**
- * Private constructor. Use createData() to create an object.
- */
- private D3DBackBufferSurfaceData(int width, int height,
- SurfaceType sType, ColorModel cm,
- GraphicsConfiguration gc,
- Image image, int screen,
- Win32SurfaceData parentData)
- {
- super(width, height, D3DSurfaceData.D3D_ATTACHED_SURFACE,
- sType, cm, gc, image, Transparency.OPAQUE);
- this.parentData = parentData;
- initSurface(width, height, screen, parentData);
- }
-
- private native void restoreDepthBuffer();
-
- @Override
- public void restoreSurface() {
- parentData.restoreSurface();
- // The above call restores the primary surface
- // to which this backbuffer is attached. But
- // we need to explicitly restore the depth buffer
- // associated with this backbuffer surface, because it's not
- // part of a 'complex' primary surface, and thus will not be
- // restored as part of the primary surface restoration.
- restoreDepthBuffer();
- }
-
- public static D3DBackBufferSurfaceData
- createData(int width, int height,
- ColorModel cm, GraphicsConfiguration gc,
- Image image,
- Win32SurfaceData parentData)
- {
- Win32GraphicsDevice gd = (Win32GraphicsDevice)gc.getDevice();
- if (!gd.isD3DEnabledOnDevice()) {
- return null;
- }
- SurfaceType sType = getSurfaceType(cm, Transparency.OPAQUE);
- return new
- D3DBackBufferSurfaceData(width, height,
- getSurfaceType(gc, cm,
- D3DSurfaceData.D3D_ATTACHED_SURFACE),
- cm, gc, image,
- gd.getScreen(), parentData);
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DBlitLoops.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DBlitLoops.java
index 23aafd25f4f..459a197fab6 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DBlitLoops.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DBlitLoops.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -29,130 +29,777 @@ import java.awt.Composite;
import java.awt.Transparency;
import java.awt.geom.AffineTransform;
import java.awt.image.AffineTransformOp;
+import java.awt.image.BufferedImage;
+import java.awt.image.BufferedImageOp;
+import java.lang.ref.WeakReference;
+import sun.java2d.ScreenUpdateManager;
+import sun.java2d.SurfaceData;
+import sun.java2d.loops.Blit;
+import sun.java2d.loops.CompositeType;
import sun.java2d.loops.GraphicsPrimitive;
import sun.java2d.loops.GraphicsPrimitiveMgr;
-import sun.java2d.loops.CompositeType;
-import sun.java2d.loops.SurfaceType;
-import sun.java2d.loops.Blit;
import sun.java2d.loops.ScaledBlit;
+import sun.java2d.loops.SurfaceType;
import sun.java2d.loops.TransformBlit;
import sun.java2d.pipe.Region;
-import sun.java2d.SurfaceData;
+import sun.java2d.pipe.RenderBuffer;
+import sun.java2d.pipe.RenderQueue;
+import static sun.java2d.pipe.BufferedOpCodes.*;
+import sun.java2d.windows.GDIWindowSurfaceData;
-import static sun.java2d.d3d.D3DSurfaceData.*;
+class D3DBlitLoops {
-/**
- * This class contains accelerated blits/scales/transforms
- * between textures and DD surfaces.
- */
-public class D3DBlitLoops {
+ static void register() {
+ Blit blitIntArgbPreToSurface =
+ new D3DSwToSurfaceBlit(SurfaceType.IntArgbPre,
+ D3DSurfaceData.ST_INT_ARGB_PRE);
+ Blit blitIntArgbPreToTexture =
+ new D3DSwToTextureBlit(SurfaceType.IntArgbPre,
+ D3DSurfaceData.ST_INT_ARGB_PRE);
- static void register()
- {
GraphicsPrimitive[] primitives = {
- new D3DTextureToSurfaceBlit(IntRgbD3D),
- new D3DTextureToSurfaceBlit(Ushort565RgbD3D),
- new D3DTextureToSurfaceBlit(IntRgbxD3D),
- new D3DTextureToSurfaceBlit(Ushort555RgbD3D),
- new D3DTextureToSurfaceBlit(ThreeByteBgrD3D),
+ // prevent D3DSurface -> Screen blits
+ new D3DSurfaceToGDIWindowSurfaceBlit(),
+ new D3DSurfaceToGDIWindowSurfaceScale(),
+ new D3DSurfaceToGDIWindowSurfaceTransform(),
- new D3DTextureToSurfaceScale(IntRgbD3D),
- new D3DTextureToSurfaceScale(Ushort565RgbD3D),
- new D3DTextureToSurfaceScale(IntRgbxD3D),
- new D3DTextureToSurfaceScale(Ushort555RgbD3D),
- new D3DTextureToSurfaceScale(ThreeByteBgrD3D),
+ // surface->surface ops
+ new D3DSurfaceToSurfaceBlit(),
+ new D3DSurfaceToSurfaceScale(),
+ new D3DSurfaceToSurfaceTransform(),
- new D3DTextureToSurfaceTransform(D3DTexture, IntRgbD3D),
- new D3DTextureToSurfaceTransform(D3DTexture, Ushort565RgbD3D),
- new D3DTextureToSurfaceTransform(D3DTexture, IntRgbxD3D),
- new D3DTextureToSurfaceTransform(D3DTexture, Ushort555RgbD3D),
- new D3DTextureToSurfaceTransform(D3DTexture, ThreeByteBgrD3D),
+ // render-to-texture surface->surface ops
+ new D3DRTTSurfaceToSurfaceBlit(),
+ new D3DRTTSurfaceToSurfaceScale(),
+ new D3DRTTSurfaceToSurfaceTransform(),
- new DelegateSwToTextureLoop(),
+ // surface->sw ops
+ new D3DSurfaceToSwBlit(SurfaceType.IntArgb,
+ D3DSurfaceData.ST_INT_ARGB),
+ // sw->surface ops
+ blitIntArgbPreToSurface,
+ new D3DSwToSurfaceBlit(SurfaceType.IntArgb,
+ D3DSurfaceData.ST_INT_ARGB),
+ new D3DSwToSurfaceBlit(SurfaceType.IntRgb,
+ D3DSurfaceData.ST_INT_RGB),
+ new D3DSwToSurfaceBlit(SurfaceType.IntBgr,
+ D3DSurfaceData.ST_INT_BGR),
+ new D3DSwToSurfaceBlit(SurfaceType.Ushort565Rgb,
+ D3DSurfaceData.ST_USHORT_565_RGB),
+ new D3DSwToSurfaceBlit(SurfaceType.Ushort555Rgb,
+ D3DSurfaceData.ST_USHORT_555_RGB),
+ new D3DSwToSurfaceBlit(SurfaceType.ByteIndexed,
+ D3DSurfaceData.ST_BYTE_INDEXED),
+ // REMIND: we don't have a native sw loop to back this loop up
+// new D3DSwToSurfaceBlit(SurfaceType.ByteIndexedBm,
+// D3DSurfaceData.ST_BYTE_INDEXED_BM),
+ new D3DGeneralBlit(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ blitIntArgbPreToSurface),
+
+ new D3DSwToSurfaceScale(SurfaceType.IntArgb,
+ D3DSurfaceData.ST_INT_ARGB),
+ new D3DSwToSurfaceScale(SurfaceType.IntArgbPre,
+ D3DSurfaceData.ST_INT_ARGB_PRE),
+ new D3DSwToSurfaceScale(SurfaceType.IntRgb,
+ D3DSurfaceData.ST_INT_RGB),
+ new D3DSwToSurfaceScale(SurfaceType.IntBgr,
+ D3DSurfaceData.ST_INT_BGR),
+ new D3DSwToSurfaceScale(SurfaceType.Ushort565Rgb,
+ D3DSurfaceData.ST_USHORT_565_RGB),
+ new D3DSwToSurfaceScale(SurfaceType.Ushort555Rgb,
+ D3DSurfaceData.ST_USHORT_555_RGB),
+ new D3DSwToSurfaceScale(SurfaceType.ByteIndexed,
+ D3DSurfaceData.ST_BYTE_INDEXED),
+ // REMIND: we don't have a native sw loop to back this loop up
+// new D3DSwToSurfaceScale(SurfaceType.ByteIndexedBm,
+// D3DSurfaceData.ST_BYTE_INDEXED_BM),
+
+ new D3DSwToSurfaceTransform(SurfaceType.IntArgb,
+ D3DSurfaceData.ST_INT_ARGB),
+ new D3DSwToSurfaceTransform(SurfaceType.IntArgbPre,
+ D3DSurfaceData.ST_INT_ARGB_PRE),
+ new D3DSwToSurfaceTransform(SurfaceType.IntRgb,
+ D3DSurfaceData.ST_INT_RGB),
+ new D3DSwToSurfaceTransform(SurfaceType.IntBgr,
+ D3DSurfaceData.ST_INT_BGR),
+ new D3DSwToSurfaceTransform(SurfaceType.Ushort565Rgb,
+ D3DSurfaceData.ST_USHORT_565_RGB),
+ new D3DSwToSurfaceTransform(SurfaceType.Ushort555Rgb,
+ D3DSurfaceData.ST_USHORT_555_RGB),
+ new D3DSwToSurfaceTransform(SurfaceType.ByteIndexed,
+ D3DSurfaceData.ST_BYTE_INDEXED),
+ // REMIND: we don't have a native sw loop to back this loop up
+// new D3DSwToSurfaceTransform(SurfaceType.ByteIndexedBm,
+// D3DSurfaceData.ST_BYTE_INDEXED_BM),
+
+ // texture->surface ops
+ new D3DTextureToSurfaceBlit(),
+ new D3DTextureToSurfaceScale(),
+ new D3DTextureToSurfaceTransform(),
+
+ // sw->texture ops
+ blitIntArgbPreToTexture,
+ new D3DSwToTextureBlit(SurfaceType.IntRgb,
+ D3DSurfaceData.ST_INT_RGB),
+ new D3DSwToTextureBlit(SurfaceType.IntArgb,
+ D3DSurfaceData.ST_INT_ARGB),
+ new D3DSwToTextureBlit(SurfaceType.IntBgr,
+ D3DSurfaceData.ST_INT_BGR),
+ new D3DSwToTextureBlit(SurfaceType.Ushort565Rgb,
+ D3DSurfaceData.ST_USHORT_565_RGB),
+ new D3DSwToTextureBlit(SurfaceType.Ushort555Rgb,
+ D3DSurfaceData.ST_USHORT_555_RGB),
+ new D3DSwToTextureBlit(SurfaceType.ByteIndexed,
+ D3DSurfaceData.ST_BYTE_INDEXED),
+ // REMIND: we don't have a native sw loop to back this loop up
+// new D3DSwToTextureBlit(SurfaceType.ByteIndexedBm,
+// D3DSurfaceData.ST_BYTE_INDEXED_BM),
+ new D3DGeneralBlit(D3DSurfaceData.D3DTexture,
+ CompositeType.SrcNoEa,
+ blitIntArgbPreToTexture),
};
GraphicsPrimitiveMgr.register(primitives);
}
- static native void doTransform(long pSrc, long pDst, long pCtx,
- int hint,
- int sx1, int sy1, int sx2, int sy2,
- float dx1, float dy1,
- float dx2, float dy2);
- static long getContext(SurfaceData src, SurfaceData dst,
- Region clip, Composite comp, AffineTransform at)
+ /**
+ * The following offsets are used to pack the parameters in
+ * createPackedParams(). (They are also used at the native level when
+ * unpacking the params.)
+ */
+ private static final int OFFSET_SRCTYPE = 16;
+ private static final int OFFSET_HINT = 8;
+ private static final int OFFSET_TEXTURE = 3;
+ private static final int OFFSET_RTT = 2;
+ private static final int OFFSET_XFORM = 1;
+ private static final int OFFSET_ISOBLIT = 0;
+
+ /**
+ * Packs the given parameters into a single int value in order to save
+ * space on the rendering queue.
+ */
+ private static int createPackedParams(boolean isoblit, boolean texture,
+ boolean rtt, boolean xform,
+ int hint, int srctype)
{
- int ctxFlags;
- if (src.getTransparency() == Transparency.OPAQUE) {
- ctxFlags = D3DContext.SRC_IS_OPAQUE;
- } else {
- ctxFlags = D3DContext.NO_CONTEXT_FLAGS;
- }
-
- return D3DContext.getContext(src, dst, clip, comp, at,
- 0xffffffff /* rgb */, ctxFlags);
- }
-}
-
-class D3DTextureToSurfaceBlit extends Blit {
- D3DTextureToSurfaceBlit(SurfaceType dstType) {
- super(D3DTexture, CompositeType.AnyAlpha , dstType);
+ return
+ ((srctype << OFFSET_SRCTYPE) |
+ (hint << OFFSET_HINT ) |
+ ((texture ? 1 : 0) << OFFSET_TEXTURE) |
+ ((rtt ? 1 : 0) << OFFSET_RTT ) |
+ ((xform ? 1 : 0) << OFFSET_XFORM ) |
+ ((isoblit ? 1 : 0) << OFFSET_ISOBLIT));
}
/**
- * Blit
- * This native method is where all of the work happens in the
- * accelerated Blit.
+ * Enqueues a BLIT operation with the given parameters. Note that the
+ * RenderQueue lock must be held before calling this method.
*/
- @Override
+ private static void enqueueBlit(RenderQueue rq,
+ SurfaceData src, SurfaceData dst,
+ int packedParams,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ // assert rq.lock.isHeldByCurrentThread();
+ RenderBuffer buf = rq.getBuffer();
+ rq.ensureCapacityAndAlignment(72, 24);
+ buf.putInt(BLIT);
+ buf.putInt(packedParams);
+ buf.putInt(sx1).putInt(sy1);
+ buf.putInt(sx2).putInt(sy2);
+ buf.putDouble(dx1).putDouble(dy1);
+ buf.putDouble(dx2).putDouble(dy2);
+ buf.putLong(src.getNativeOps());
+ buf.putLong(dst.getNativeOps());
+ }
+
+ static void Blit(SurfaceData srcData, SurfaceData dstData,
+ Composite comp, Region clip,
+ AffineTransform xform, int hint,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ int srctype, boolean texture)
+ {
+ int ctxflags = 0;
+ if (srcData.getTransparency() == Transparency.OPAQUE) {
+ ctxflags |= D3DContext.SRC_IS_OPAQUE;
+ }
+
+ D3DSurfaceData d3dDst = (D3DSurfaceData)dstData;
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ // make sure the RenderQueue keeps a hard reference to the
+ // source (sysmem) SurfaceData to prevent it from being
+ // disposed while the operation is processed on the QFT
+ rq.addReference(srcData);
+
+ if (texture) {
+ // make sure we have a current context before uploading
+ // the sysmem data to the texture object
+ D3DContext.setScratchSurface(d3dDst.getContext());
+ } else {
+ D3DContext.validateContext(d3dDst, d3dDst,
+ clip, comp, xform, null, null,
+ ctxflags);
+ }
+
+ int packedParams = createPackedParams(false, texture,
+ false, xform != null,
+ hint, srctype);
+ enqueueBlit(rq, srcData, dstData,
+ packedParams,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2);
+
+ // always flush immediately, since we (currently) have no means
+ // of tracking changes to the system memory surface
+ rq.flushNow();
+ } finally {
+ rq.unlock();
+ }
+
+ if (d3dDst.getType() == D3DSurfaceData.WINDOW) {
+ // flush immediately when copying to the screen to improve
+ // responsiveness of applications using VI or BI backbuffers
+ D3DScreenUpdateManager mgr =
+ (D3DScreenUpdateManager)ScreenUpdateManager.getInstance();
+ mgr.runUpdateNow();
+ }
+ }
+
+ /**
+ * Note: The srcImg and biop parameters are only used when invoked
+ * from the D3DBufImgOps.renderImageWithOp() method; in all other cases,
+ * this method can be called with null values for those two parameters,
+ * and they will be effectively ignored.
+ */
+ static void IsoBlit(SurfaceData srcData, SurfaceData dstData,
+ BufferedImage srcImg, BufferedImageOp biop,
+ Composite comp, Region clip,
+ AffineTransform xform, int hint,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ boolean texture)
+ {
+ int ctxflags = 0;
+ if (srcData.getTransparency() == Transparency.OPAQUE) {
+ ctxflags |= D3DContext.SRC_IS_OPAQUE;
+ }
+
+ D3DSurfaceData d3dDst = (D3DSurfaceData)dstData;
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ boolean rtt = false;
+ rq.lock();
+ try {
+ D3DSurfaceData d3dSrc = (D3DSurfaceData)srcData;
+ int srctype = d3dSrc.getType();
+ D3DSurfaceData srcCtxData = d3dSrc;
+ if (srctype == D3DSurfaceData.TEXTURE) {
+ rtt = false;
+ } else {
+ // the source is a backbuffer, or render-to-texture
+ // surface; we set rtt to true to differentiate this kind
+ // of surface from a regular texture object
+ rtt = true;
+ }
+
+ D3DContext.validateContext(srcCtxData, d3dDst,
+ clip, comp, xform, null, null,
+ ctxflags);
+
+ if (biop != null) {
+ D3DBufImgOps.enableBufImgOp(rq, d3dSrc, srcImg, biop);
+ }
+
+ int packedParams = createPackedParams(true, texture,
+ rtt, xform != null,
+ hint, 0 /*unused*/);
+ enqueueBlit(rq, srcData, dstData,
+ packedParams,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2);
+
+ if (biop != null) {
+ D3DBufImgOps.disableBufImgOp(rq, biop);
+ }
+ } finally {
+ rq.unlock();
+ }
+
+ if (rtt && (d3dDst.getType() == D3DSurfaceData.WINDOW)) {
+ // we only have to flush immediately when copying from a
+ // (non-texture) surface to the screen; otherwise Swing apps
+ // might appear unresponsive until the auto-flush completes
+ D3DScreenUpdateManager mgr =
+ (D3DScreenUpdateManager)ScreenUpdateManager.getInstance();
+ mgr.runUpdateNow();
+ }
+ }
+}
+
+class D3DSurfaceToSurfaceBlit extends Blit {
+
+ D3DSurfaceToSurfaceBlit() {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
public void Blit(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
int sx, int sy, int dx, int dy, int w, int h)
{
- synchronized (D3DContext.LOCK) {
- long pCtx = D3DBlitLoops.getContext(src, dst, clip, comp, null);
- D3DBlitLoops.doTransform(src.getNativeOps(), dst.getNativeOps(),
- pCtx,
- AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
- sx, sy, sx+w, sy+h,
- (float)dx, (float)dy,
- (float)(dx+w), (float)(dy+h));
- }
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ false);
}
}
-class D3DTextureToSurfaceTransform extends TransformBlit {
+class D3DSurfaceToSurfaceScale extends ScaledBlit {
- D3DTextureToSurfaceTransform(SurfaceType srcType,
- SurfaceType dstType)
- {
- super(srcType, CompositeType.AnyAlpha, dstType);
+ D3DSurfaceToSurfaceScale() {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
+ public void Scale(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2,
+ false);
+ }
+}
+
+class D3DSurfaceToSurfaceTransform extends TransformBlit {
+
+ D3DSurfaceToSurfaceTransform() {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
}
- @Override
public void Transform(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
AffineTransform at, int hint,
int sx, int sy, int dx, int dy,
int w, int h)
{
- synchronized (D3DContext.LOCK) {
- long pCtx = D3DBlitLoops.getContext(src, dst, clip, comp, at);
- D3DBlitLoops.doTransform(src.getNativeOps(), dst.getNativeOps(),
- pCtx, hint,
- sx, sy, sx+w, sy+h,
- (float)dx, (float)dy,
- (float)(dx+w), (float)(dy+h));
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, at, hint,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ false);
+ }
+}
+
+class D3DRTTSurfaceToSurfaceBlit extends Blit {
+
+ D3DRTTSurfaceToSurfaceBlit() {
+ super(D3DSurfaceData.D3DSurfaceRTT,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
+ public void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ true);
+ }
+}
+
+class D3DRTTSurfaceToSurfaceScale extends ScaledBlit {
+
+ D3DRTTSurfaceToSurfaceScale() {
+ super(D3DSurfaceData.D3DSurfaceRTT,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
+ public void Scale(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2,
+ true);
+ }
+}
+
+class D3DRTTSurfaceToSurfaceTransform extends TransformBlit {
+
+ D3DRTTSurfaceToSurfaceTransform() {
+ super(D3DSurfaceData.D3DSurfaceRTT,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
+ public void Transform(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ AffineTransform at, int hint,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, at, hint,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ true);
+ }
+}
+
+class D3DSurfaceToSwBlit extends Blit {
+
+ private int typeval;
+
+ // REMIND: destination will actually be opaque/premultiplied...
+ D3DSurfaceToSwBlit(SurfaceType dstType, int typeval) {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.SrcNoEa,
+ dstType);
+ this.typeval = typeval;
+ }
+
+ public void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy,
+ int w, int h)
+ {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ // make sure the RenderQueue keeps a hard reference to the
+ // destination (sysmem) SurfaceData to prevent it from being
+ // disposed while the operation is processed on the QFT
+ rq.addReference(dst);
+
+ RenderBuffer buf = rq.getBuffer();
+ D3DContext.setScratchSurface(((D3DSurfaceData)src).getContext());
+
+ rq.ensureCapacityAndAlignment(48, 32);
+ buf.putInt(SURFACE_TO_SW_BLIT);
+ buf.putInt(sx).putInt(sy);
+ buf.putInt(dx).putInt(dy);
+ buf.putInt(w).putInt(h);
+ buf.putInt(typeval);
+ buf.putLong(src.getNativeOps());
+ buf.putLong(dst.getNativeOps());
+
+ // always flush immediately
+ rq.flushNow();
+ } finally {
+ rq.unlock();
}
}
}
+class D3DSwToSurfaceBlit extends Blit {
+
+ private int typeval;
+
+ D3DSwToSurfaceBlit(SurfaceType srcType, int typeval) {
+ super(srcType,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ this.typeval = typeval;
+ }
+
+ public void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ D3DBlitLoops.Blit(src, dst,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ typeval, false);
+ }
+}
+
+class D3DSwToSurfaceScale extends ScaledBlit {
+
+ private int typeval;
+
+ D3DSwToSurfaceScale(SurfaceType srcType, int typeval) {
+ super(srcType,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ this.typeval = typeval;
+ }
+
+ public void Scale(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ D3DBlitLoops.Blit(src, dst,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2,
+ typeval, false);
+ }
+}
+
+class D3DSwToSurfaceTransform extends TransformBlit {
+
+ private int typeval;
+
+ D3DSwToSurfaceTransform(SurfaceType srcType, int typeval) {
+ super(srcType,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ this.typeval = typeval;
+ }
+
+ public void Transform(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ AffineTransform at, int hint,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ D3DBlitLoops.Blit(src, dst,
+ comp, clip, at, hint,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ typeval, false);
+ }
+}
+
+class D3DSwToTextureBlit extends Blit {
+
+ private int typeval;
+
+ D3DSwToTextureBlit(SurfaceType srcType, int typeval) {
+ super(srcType,
+ CompositeType.SrcNoEa,
+ D3DSurfaceData.D3DTexture);
+ this.typeval = typeval;
+ }
+
+ public void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ D3DBlitLoops.Blit(src, dst,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ typeval, true);
+ }
+}
+
+class D3DTextureToSurfaceBlit extends Blit {
+
+ D3DTextureToSurfaceBlit() {
+ super(D3DSurfaceData.D3DTexture,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
+ public void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ true);
+ }
+}
+
class D3DTextureToSurfaceScale extends ScaledBlit {
- D3DTextureToSurfaceScale(SurfaceType dstType) {
- super(D3DTexture, CompositeType.AnyAlpha, dstType);
+ D3DTextureToSurfaceScale() {
+ super(D3DSurfaceData.D3DTexture,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
}
+ public void Scale(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx1, int sy1,
+ int sx2, int sy2,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, null,
+ AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2,
+ true);
+ }
+}
+
+class D3DTextureToSurfaceTransform extends TransformBlit {
+
+ D3DTextureToSurfaceTransform() {
+ super(D3DSurfaceData.D3DTexture,
+ CompositeType.AnyAlpha,
+ D3DSurfaceData.D3DSurface);
+ }
+
+ public void Transform(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ AffineTransform at, int hint,
+ int sx, int sy, int dx, int dy,
+ int w, int h)
+ {
+ D3DBlitLoops.IsoBlit(src, dst,
+ null, null,
+ comp, clip, at, hint,
+ sx, sy, sx+w, sy+h,
+ dx, dy, dx+w, dy+h,
+ true);
+ }
+}
+
+/**
+ * This general Blit implemenation converts any source surface to an
+ * intermediate IntArgbPre surface, and then uses the more specific
+ * IntArgbPre->D3DSurface/Texture loop to get the intermediate
+ * (premultiplied) surface down to D3D.
+ */
+class D3DGeneralBlit extends Blit {
+
+ private Blit performop;
+ private WeakReference srcTmp;
+
+ D3DGeneralBlit(SurfaceType dstType,
+ CompositeType compType,
+ Blit performop)
+ {
+ super(SurfaceType.Any, compType, dstType);
+ this.performop = performop;
+ }
+
+ public synchronized void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy,
+ int w, int h)
+ {
+ Blit convertsrc = Blit.getFromCache(src.getSurfaceType(),
+ CompositeType.SrcNoEa,
+ SurfaceType.IntArgbPre);
+
+ SurfaceData cachedSrc = null;
+ if (srcTmp != null) {
+ // use cached intermediate surface, if available
+ cachedSrc = (SurfaceData)srcTmp.get();
+ }
+
+ // convert source to IntArgbPre
+ src = convertFrom(convertsrc, src, sx, sy, w, h,
+ cachedSrc, BufferedImage.TYPE_INT_ARGB_PRE);
+
+ // copy IntArgbPre intermediate surface to D3D surface
+ performop.Blit(src, dst, comp, clip,
+ 0, 0, dx, dy, w, h);
+
+ if (src != cachedSrc) {
+ // cache the intermediate surface
+ srcTmp = new WeakReference(src);
+ }
+ }
+}
+
+/*
+ * The following classes prohibit copying D3DSurfaces to the screen
+ * (the D3D->sysmem->GDI path is known to be very very slow).
+ *
+ * Note: we used to disable hw acceleration for the surafce manager associated
+ * with the source surface in these loops but it proved to be too cautious.
+ *
+ * In most cases d3d->screen copy happens only during some transitional
+ * period where the accelerated destination surface is being recreated or
+ * restored (for example, when Swing's backbuffer VI is copied to the screen
+ * but the D3DScreenSurfaceManager couldn't restore its surface).
+ *
+ * An exception is if for some reason we could not enable accelerated on-screen
+ * rendering for this window for some permanent reason (like window being too
+ * small, or a present BufferStrategy).
+ *
+ * This meant that we'd disable hw acceleration after the first failure
+ * completely (at least until the src image is recreated which in case of
+ * Swing back-buffer happens only after resize).
+ *
+ * Now we delegate to the VISM to figure out if the acceleration needs to
+ * be disabled or if we can wait for a while until the onscreen accelerated
+ * can resume (by marking the source surface lost and making sure the
+ * VISM has a chance to use the backup surface).
+ *
+ */
+
+class D3DSurfaceToGDIWindowSurfaceBlit extends Blit {
+
+ D3DSurfaceToGDIWindowSurfaceBlit() {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ GDIWindowSurfaceData.AnyGdi);
+ }
+ @Override
+ public void Blit(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ int sx, int sy, int dx, int dy, int w, int h)
+ {
+ // see comment above
+ D3DVolatileSurfaceManager.handleVItoScreenOp(src, dst);
+ }
+
+}
+
+class D3DSurfaceToGDIWindowSurfaceScale extends ScaledBlit {
+
+ D3DSurfaceToGDIWindowSurfaceScale() {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ GDIWindowSurfaceData.AnyGdi);
+ }
@Override
public void Scale(SurfaceData src, SurfaceData dst,
Composite comp, Region clip,
@@ -161,65 +808,26 @@ class D3DTextureToSurfaceScale extends ScaledBlit {
double dx1, double dy1,
double dx2, double dy2)
{
- synchronized (D3DContext.LOCK) {
- long pCtx = D3DBlitLoops.getContext(src, dst, clip, comp, null);
- D3DBlitLoops.doTransform(src.getNativeOps(), dst.getNativeOps(),
- pCtx,
- AffineTransformOp.TYPE_NEAREST_NEIGHBOR,
- sx1, sy1, sx2, sy2,
- (float)dx1, (float)dy1,
- (float)dx2, (float)dy2);
- }
+ // see comment above
+ D3DVolatileSurfaceManager.handleVItoScreenOp(src, dst);
}
}
-class DelegateSwToTextureLoop extends Blit {
+class D3DSurfaceToGDIWindowSurfaceTransform extends TransformBlit {
- DelegateSwToTextureLoop() {
- super(SurfaceType.Any, CompositeType.SrcNoEa, D3DTexture);
+ D3DSurfaceToGDIWindowSurfaceTransform() {
+ super(D3DSurfaceData.D3DSurface,
+ CompositeType.AnyAlpha,
+ GDIWindowSurfaceData.AnyGdi);
}
-
@Override
- public void Blit(SurfaceData src, SurfaceData dst,
- Composite comp, Region clip,
- int sx, int sy, int dx, int dy, int w, int h)
+ public void Transform(SurfaceData src, SurfaceData dst,
+ Composite comp, Region clip,
+ AffineTransform at, int hint,
+ int sx, int sy, int dx, int dy,
+ int w, int h)
{
- Blit realBlit = null;
- int pf = ((D3DSurfaceData)dst).getPixelFormat();
- switch (pf) {
- case PF_INT_ARGB:
- realBlit = Blit.getFromCache(src.getSurfaceType(),
- CompositeType.SrcNoEa,
- SurfaceType.IntArgbPre);
- break;
- case PF_INT_RGB:
- realBlit = Blit.getFromCache(src.getSurfaceType(),
- CompositeType.SrcNoEa,
- SurfaceType.IntRgb);
- break;
- case PF_USHORT_565_RGB:
- realBlit = Blit.getFromCache(src.getSurfaceType(),
- CompositeType.SrcNoEa,
- SurfaceType.Ushort565Rgb);
- break;
- case PF_USHORT_555_RGB:
- realBlit = Blit.getFromCache(src.getSurfaceType(),
- CompositeType.SrcNoEa,
- SurfaceType.Ushort555Rgb);
- break;
- case PF_USHORT_4444_ARGB:
- // REMIND: this should really be premultiplied!
- realBlit = Blit.getFromCache(src.getSurfaceType(),
- CompositeType.SrcNoEa,
- SurfaceType.Ushort4444Argb);
- break;
- default:
- throw
- new InternalError("Can't yet handle dest pixel format: "+pf);
- }
-
- if (realBlit != null) {
- realBlit.Blit(src, dst, comp, clip, sx, sy, dx, dy, w, h);
- }
+ // see comment above
+ D3DVolatileSurfaceManager.handleVItoScreenOp(src, dst);
}
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DBufImgOps.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DBufImgOps.java
new file mode 100644
index 00000000000..2906272b8a3
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DBufImgOps.java
@@ -0,0 +1,116 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.image.AffineTransformOp;
+import java.awt.image.BufferedImage;
+import java.awt.image.BufferedImageOp;
+import java.awt.image.ConvolveOp;
+import java.awt.image.LookupOp;
+import java.awt.image.RescaleOp;
+import sun.java2d.SunGraphics2D;
+import sun.java2d.SurfaceData;
+import sun.java2d.loops.CompositeType;
+import sun.java2d.pipe.BufferedBufImgOps;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
+
+class D3DBufImgOps extends BufferedBufImgOps {
+
+ /**
+ * This method is called from D3DDrawImage.transformImage() only. It
+ * validates the provided BufferedImageOp to determine whether the op
+ * is one that can be accelerated by the D3D pipeline. If the operation
+ * cannot be completed for any reason, this method returns false;
+ * otherwise, the given BufferedImage is rendered to the destination
+ * using the provided BufferedImageOp and this method returns true.
+ */
+ static boolean renderImageWithOp(SunGraphics2D sg, BufferedImage img,
+ BufferedImageOp biop, int x, int y)
+ {
+ // Validate the provided BufferedImage (make sure it is one that
+ // is supported, and that its properties are acceleratable)
+ if (biop instanceof ConvolveOp) {
+ if (!isConvolveOpValid((ConvolveOp)biop)) {
+ return false;
+ }
+ } else if (biop instanceof RescaleOp) {
+ if (!isRescaleOpValid((RescaleOp)biop, img)) {
+ return false;
+ }
+ } else if (biop instanceof LookupOp) {
+ if (!isLookupOpValid((LookupOp)biop, img)) {
+ return false;
+ }
+ } else {
+ // No acceleration for other BufferedImageOps (yet)
+ return false;
+ }
+
+ SurfaceData dstData = sg.surfaceData;
+ if (!(dstData instanceof D3DSurfaceData) ||
+ (sg.interpolationType == AffineTransformOp.TYPE_BICUBIC) ||
+ (sg.compositeState > SunGraphics2D.COMP_ALPHA))
+ {
+ return false;
+ }
+
+ SurfaceData srcData =
+ dstData.getSourceSurfaceData(img, sg.TRANSFORM_ISIDENT,
+ CompositeType.SrcOver, null);
+ if (!(srcData instanceof D3DSurfaceData)) {
+ // REMIND: this hack tries to ensure that we have a cached texture
+ srcData =
+ dstData.getSourceSurfaceData(img, sg.TRANSFORM_ISIDENT,
+ CompositeType.SrcOver, null);
+ if (!(srcData instanceof D3DSurfaceData)) {
+ return false;
+ }
+ }
+
+ // Verify that the source surface is actually a texture and that
+ // shaders are supported
+ D3DSurfaceData d3dSrc = (D3DSurfaceData)srcData;
+ D3DGraphicsDevice gd =
+ (D3DGraphicsDevice)d3dSrc.getDeviceConfiguration().getDevice();
+ if (d3dSrc.getType() != D3DSurfaceData.TEXTURE ||
+ !gd.isCapPresent(CAPS_LCD_SHADER))
+ {
+ return false;
+ }
+
+ int sw = img.getWidth();
+ int sh = img.getHeight();
+ D3DBlitLoops.IsoBlit(srcData, dstData,
+ img, biop,
+ sg.composite, sg.getCompClip(),
+ sg.transform, sg.interpolationType,
+ 0, 0, sw, sh,
+ x, y, x+sw, y+sh,
+ true);
+
+ return true;
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DContext.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DContext.java
index b41de7b34e9..53aa36a73d1 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DContext.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DContext.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -25,392 +25,165 @@
package sun.java2d.d3d;
-import java.awt.AlphaComposite;
-import java.awt.Composite;
-import java.awt.GraphicsEnvironment;
-import java.awt.geom.AffineTransform;
-import sun.awt.Win32GraphicsDevice;
-import sun.java2d.InvalidPipeException;
-import sun.java2d.SurfaceData;
-import sun.java2d.pipe.Region;
-import sun.java2d.windows.WindowsFlags;
+import sun.java2d.pipe.BufferedContext;
+import sun.java2d.pipe.RenderBuffer;
+import sun.java2d.pipe.RenderQueue;
+import sun.java2d.pipe.hw.ContextCapabilities;
+import static sun.java2d.pipe.BufferedOpCodes.*;
+import static sun.java2d.pipe.hw.ContextCapabilities.*;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
-public class D3DContext {
+/**
+ * Note that the RenderQueue lock must be acquired before calling any of
+ * the methods in this class.
+ */
+class D3DContext extends BufferedContext {
- public static final int NO_CONTEXT_FLAGS = 0;
- /**
- * Used in D3DBlitLoops: if the source surface is opaque
- * alpha blending can be turned off on the native level
- * (if there's no ea), thus improving performance.
- */
- public static final int SRC_IS_OPAQUE = 1;
+ private final D3DGraphicsDevice device;
+
+ D3DContext(RenderQueue rq, D3DGraphicsDevice device) {
+ super(rq);
+ this.device = device;
+ }
/**
- * This is a list of capabilities supported by the device this
- * context is associated with.
- * @see getDeviceCaps
+ * Invalidates the currentContext field to ensure that we properly
+ * revalidate the D3DContext (make it current, etc.) next time through
+ * the validate() method. This is typically invoked from methods
+ * that affect the current context state (e.g. disposing a context or
+ * surface).
*/
- public static final int J2D_D3D_FAILURE = (0 << 0);
- /**
- * Device supports depth buffer for d3d render targets
- */
- public static final int J2D_D3D_DEPTH_SURFACE_OK = (1 << 0);
- /**
- * Device supports creation of plain d3d surfaces
- */
- public static final int J2D_D3D_PLAIN_SURFACE_OK = (1 << 1);
- /**
- * Device supports creation of opaque textures
- */
- public static final int J2D_D3D_OP_TEXTURE_SURFACE_OK = (1 << 2);
- /**
- * Device supports creation of bitmask textures
- */
- public static final int J2D_D3D_BM_TEXTURE_SURFACE_OK = (1 << 3);
- /**
- * Device supports creation of translucent textures
- */
- public static final int J2D_D3D_TR_TEXTURE_SURFACE_OK = (1 << 4);
- /**
- * Device supports creation of opaque render-to-textures
- */
- public static final int J2D_D3D_OP_RTT_SURFACE_OK = (1 << 5);
- /**
- * Device can render lines correctly (no pixelization issues)
- */
- public static final int J2D_D3D_LINES_OK = (1 << 6);
- /**
- * Device supports texture mapping (no pixelization issues)
- */
- public static final int J2D_D3D_TEXTURE_BLIT_OK = (1 << 7);
- /**
- * Device supports texture mapping with transforms (no pixelization issues)
- */
- public static final int J2D_D3D_TEXTURE_TRANSFORM_OK = (1 << 8);
- /**
- * Device can render clipped lines correctly.
- */
- public static final int J2D_D3D_LINE_CLIPPING_OK = (1 << 9);
- /**
- * Device has all hw capabilities the d3d pipeline requires
- */
- public static final int J2D_D3D_DEVICE_OK = (1 <<10);
- /**
- * Device supports all necessary texture formats required by d3d pipeline
- */
- public static final int J2D_D3D_PIXEL_FORMATS_OK = (1 <<11);
- /**
- * Device supports geometry transformations
- */
- public static final int J2D_D3D_SET_TRANSFORM_OK = (1 <<12);
- /**
- * The device is not from a list of known bad devices
- * (see D3DRuntimeTest.cpp)
- */
- public static final int J2D_D3D_HW_OK = (1 <<13);
- /**
- * Direct3D pipeline is enabled on this device
- */
- public static final int J2D_D3D_ENABLED_OK = (1 <<14);
+ static void invalidateCurrentContext() {
+ // assert D3DRenderQueue.getInstance().lock.isHeldByCurrentThread();
- /**
- * The lock object used to synchronize access to the native windowing
- * system layer. Note that rendering methods should always synchronize on
- * D3DContext.LOCK before calling the D3DContext.getContext() method,
- * or any other method that invokes native D3d commands.
- * REMIND: in D3D case we should really be synchronizing on per-device
- * basis.
- */
- static Object LOCK;
-
- private Win32GraphicsDevice gd;
- private boolean valid;
-
- protected long nativeContext;
- private SurfaceData validatedDstData;
- private Region validatedClip;
- private Composite validatedComp;
- private int validatedPixel;
- private int validatedFlags;
- private boolean xformInUse;
- // validated transform's data
- private double vScaleX, vScaleY, vShearX, vShearY, vTransX, vTransY;
-
- private int deviceCaps;
-
- private native void setRenderTarget(long pCtx, long pDst);
- private native void setClip(long pCtx, long pDst, Region clip, boolean isRect,
- int x1, int y1, int x2, int y2);
- private native void resetClip(long pCtx, long pDst);
- private native void resetComposite(long pCtx);
- private native void setAlphaComposite(long pCtx, int rule,
- float extraAlpha, int flags);
- private native void setTransform(long pCtx, long pDst,
- AffineTransform xform,
- double m00, double m10, double m01,
- double m11, double m02, double m12);
- private native void resetTransform(long pCtx, long pDst);
- private native void setColor(long pCtx, int pixel, int flags);
- private native long initNativeContext(int screen);
- private native int getNativeDeviceCaps(long pCtx);
-
- static {
- if (!GraphicsEnvironment.isHeadless()) {
- LOCK = D3DContext.class;
+ // invalidate the current Java-level context so that we
+ // revalidate everything the next time around
+ if (currentContext != null) {
+ currentContext.invalidateContext();
+ currentContext = null;
}
- }
- public D3DContext(Win32GraphicsDevice gd) {
- this.gd = gd;
- reinitNativeContext();
+ // invalidate the context reference at the native level, and
+ // then flush the queue so that we have no pending operations
+ // dependent on the current context
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.ensureCapacity(4);
+ rq.getBuffer().putInt(INVALIDATE_CONTEXT);
+ rq.flushNow();
}
/**
- * Reinitializes the context by retrieving a pointer to the native
- * D3DContext object, and resetting the device caps.
- */
- void reinitNativeContext() {
- nativeContext = initNativeContext(gd.getScreen());
- deviceCaps = nativeContext != 0L ?
- getNativeDeviceCaps(nativeContext) : J2D_D3D_FAILURE;
- valid = ((deviceCaps & J2D_D3D_ENABLED_OK) != 0);
- if (WindowsFlags.isD3DVerbose()) {
- if (valid) {
- System.out.println("Direct3D pipeline enabled on screen " +
- gd.getScreen());
- } else {
- System.out.println("Could not enable Direct3D pipeline on " +
- "screen " + gd.getScreen() +
- ". Device Caps: " +
- Integer.toHexString(deviceCaps));
- }
- }
- }
-
- /**
- * Invalidates this context by resetting its status: the validated
- * destination surface, and a pointer to the native context.
- * This method is called in the following cases:
- * - if a surface loss situation is detected at the native level
- * during any of the validation methods (setClip, setRenderTarget etc)
- * and an InvalidPipeException is thrown.
- * This situation happens when there was a surface loss, but
- * there were no display change event (like in case of command prompt
- * going fullscreen).
- * - as part of surface restoration when a surface is the current
- * target surface for this context. Since surface restoration
- * resets the depth buffer contents, we need to make sure the clip
- * is reset, and since the target surface is reset, we'll set a new
- * clip the next time we attempt to render to the target surface.
- * - when a display change occurs, the native D3DContext object is
- * released and recreated as part of primary surface recreation.
- * At the time of the release, the java D3DContext object need to be
- * invalidated because a new D3D device is created and the target
- * surface will need to be reset.
+ * Sets the current context on the native level to be the one passed as
+ * the argument.
+ * If the context is not the same as the defaultContext the latter
+ * will be reset to null.
*
- * Invalidation of the context causes its revalidation the next time
- * someone tries to get the D3DContext for rendering or creating a new
- * surface.
+ * This call is needed when copying from a SW surface to a Texture
+ * (the upload test) or copying from d3d to SW surface to make sure we
+ * have the correct current context.
*
- * @see #reinitNativeContext
+ * @param d3dc the context to be made current on the native level
*/
- private void invalidateContext() {
- valid = false;
- nativeContext = 0L;
- validatedDstData = null;
- // We don't set deviceCaps to J2D_D3D_FAILURE here because
- // it will prevent from creating d3d surfaces, which means that
- // we'll never get a chance to continue using d3d after a single
- // invalidation event (for example, a display change).
- }
+ static void setScratchSurface(D3DContext d3dc) {
+ // assert D3DRenderQueue.getInstance().lock.isHeldByCurrentThread();
- /**
- * Fetches the D3DContext associated with the current
- * thread/GraphicsConfig pair, validates the context using the given
- * parameters, then returns the handle to the native context object.
- * Most rendering operations will call this method first in order to
- * prepare the native D3d layer before issuing rendering commands.
- */
- static long getContext(SurfaceData srcData,
- SurfaceData dstData,
- Region clip, Composite comp,
- AffineTransform xform,
- int pixel, int flags)
- {
- if (dstData instanceof D3DSurfaceData == false) {
- throw new InvalidPipeException("Incorrect destination surface");
+ // invalidate the current context
+ if (d3dc != currentContext) {
+ currentContext = null;
}
- D3DContext d3dc = ((D3DSurfaceData)dstData).getContext();
- try {
- d3dc.validate(srcData, dstData, clip, comp, xform, pixel, flags);
- } catch (InvalidPipeException e) {
- d3dc.invalidateContext();
- // note that we do not propagate the exception. Once the context
- // is invalidated, any d3d rendering operations are noops, and
- // we are waiting for the primary surface restoration, which
- // happens when VolatileImage is validated. At this point
- // the native D3DContext will be reinitialized, and the next
- // time around validation of the context will succeed.
- // Throwing the exception here will do no good, since the
- // destination surface (which is associated with a VolatileImage
- // or a BufferStrategy) will not be restored until VI.validate()
- // is called by the rendering thread.
- }
- return d3dc.getNativeContext();
+ // set the scratch context
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ RenderBuffer buf = rq.getBuffer();
+ rq.ensureCapacity(8);
+ buf.putInt(SET_SCRATCH_SURFACE);
+ buf.putInt(d3dc.getDevice().getScreen());
}
- public int getDeviceCaps() {
- return deviceCaps;
+ public RenderQueue getRenderQueue() {
+ return D3DRenderQueue.getInstance();
}
- boolean isRTTSupported() {
- return ((deviceCaps & J2D_D3D_OP_RTT_SURFACE_OK) != 0);
+ @Override
+ public void saveState() {
+ // assert rq.lock.isHeldByCurrentThread();
+
+ // reset all attributes of this and current contexts
+ invalidateContext();
+ invalidateCurrentContext();
+
+ setScratchSurface(this);
+
+ // save the state on the native level
+ rq.ensureCapacity(4);
+ buf.putInt(SAVE_STATE);
+ rq.flushNow();
}
- /**
- * Returns a handle to the native D3DContext structure associated with
- * this object.
- */
- long getNativeContext() {
- return nativeContext;
+ @Override
+ public void restoreState() {
+ // assert rq.lock.isHeldByCurrentThread();
+
+ // reset all attributes of this and current contexts
+ invalidateContext();
+ invalidateCurrentContext();
+
+ setScratchSurface(this);
+
+ // restore the state on the native level
+ rq.ensureCapacity(4);
+ buf.putInt(RESTORE_STATE);
+ rq.flushNow();
}
- /**
- * Validates the given parameters against the current state for this
- * context. If this context is not current, it will be made current
- * for the given source and destination surfaces, and the viewport will
- * be updated. Then each part of the context state (clip, composite,
- * etc.) is checked against the previous value. If the value has changed
- * since the last call to validate(), it will be updated accordingly.
- */
- private void validate(SurfaceData srcData, SurfaceData dstData,
- Region clip, Composite comp, AffineTransform xform,
- int pixel, int flags)
- {
- boolean updateClip = false;
+ D3DGraphicsDevice getDevice() {
+ return device;
+ }
- if ((srcData != null && !srcData.isValid()) || !dstData.isValid() ||
- dstData.getNativeOps() == 0L || dstData.isSurfaceLost())
- {
- throw new InvalidPipeException("Invalid surface");
+ static class D3DContextCaps extends ContextCapabilities {
+ /**
+ * Indicates the presence of pixel shaders (v2.0 or greater).
+ * This cap will only be set if the hardware supports the minimum number
+ * of texture units.
+ */
+ static final int CAPS_LCD_SHADER = (FIRST_PRIVATE_CAP << 0);
+ /**
+ * Indicates the presence of pixel shaders (v2.0 or greater).
+ * This cap will only be set if the hardware meets our
+ * minimum requirements.
+ */
+ static final int CAPS_BIOP_SHADER = (FIRST_PRIVATE_CAP << 1);
+ /**
+ * Indicates that the device was successfully initialized and can
+ * be safely used.
+ */
+ static final int CAPS_DEVICE_OK = (FIRST_PRIVATE_CAP << 2);
+ /**
+ * Indicates that the device has all of the necessary capabilities
+ * to support the Antialiasing Pixel Shader program.
+ */
+ static final int CAPS_AA_SHADER = (FIRST_PRIVATE_CAP << 3);
+
+ D3DContextCaps(int caps, String adapterId) {
+ super(caps, adapterId);
}
- if (!valid) {
- // attempt to reinitialize the context. If the device has been
- // reset, the following calls to setRenderTarget/setClip will
- // succeed and not throw InvalidPipeException.
- reinitNativeContext();
- }
-
- if (dstData != validatedDstData) {
- // invalidate pixel and clip (so they will be updated below)
- validatedPixel = ~pixel;
- updateClip = true;
-
- // update the viewport
- long pDst = dstData.getNativeOps();
- setRenderTarget(nativeContext, pDst);
-
- // keep the reference to the old data until we set the
- // new one on the native level, preventing it from being disposed
- SurfaceData tmpData = dstData;
- validatedDstData = dstData;
- tmpData = null;
- }
- // it's better to use dstData instead of validatedDstData because
- // the latter may be set to null via invalidateContext at any moment.
- long pDest = dstData.getNativeOps();
-
- // validate clip
- if ((clip != validatedClip) || updateClip) {
- if (clip != null) {
- /**
- * It's cheaper to make this check than set clip every time.
- *
- * Set the new clip only if:
- * - we were asked to do it (updateClip == true)
- * - no clip was set before
- * - if both the old and the new clip are shapes
- * - if they're both rectangular but don't represent
- * the same rectangle
- */
- if (updateClip ||
- validatedClip == null ||
- !(validatedClip.isRectangular() && clip.isRectangular()) ||
- ((clip.getLoX() != validatedClip.getLoX() ||
- clip.getLoY() != validatedClip.getLoY() ||
- clip.getHiX() != validatedClip.getHiX() ||
- clip.getHiY() != validatedClip.getHiY())))
- {
- setClip(nativeContext, pDest,
- clip, clip.isRectangular(),
- clip.getLoX(), clip.getLoY(),
- clip.getHiX(), clip.getHiY());
- }
- } else {
- resetClip(nativeContext, pDest);
+ @Override
+ public String toString() {
+ StringBuffer buf = new StringBuffer(super.toString());
+ if ((caps & CAPS_LCD_SHADER) != 0) {
+ buf.append("CAPS_LCD_SHADER|");
}
- validatedClip = clip;
- }
-
- if ((comp != validatedComp) || (flags != validatedFlags)) {
- // invalidate pixel
- validatedPixel = ~pixel;
- validatedComp = comp;
- if (comp != null) {
- AlphaComposite ac = (AlphaComposite)comp;
- setAlphaComposite(nativeContext, ac.getRule(),
- ac.getAlpha(), flags);
- } else {
- resetComposite(nativeContext);
+ if ((caps & CAPS_BIOP_SHADER) != 0) {
+ buf.append("CAPS_BIOP_SHADER|");
}
- }
-
- // validate transform
- if (xform == null) {
- if (xformInUse) {
- resetTransform(nativeContext, pDest);
- xformInUse = false;
- vScaleX = vScaleY = 1.0;
- vShearX = vShearY = vTransX = vTransY = 0.0;
+ if ((caps & CAPS_AA_SHADER) != 0) {
+ buf.append("CAPS_AA_SHADER|");
}
- } else {
- double nScaleX = xform.getScaleX();
- double nScaleY = xform.getScaleY();
- double nShearX = xform.getShearX();
- double nShearY = xform.getShearY();
- double nTransX = xform.getTranslateX();
- double nTransY = xform.getTranslateY();
-
- if (nTransX != vTransX || nTransY != vTransY ||
- nScaleX != vScaleX || nScaleY != vScaleY ||
- nShearX != vShearX || nShearY != vShearY)
- {
- setTransform(nativeContext, pDest,
- xform,
- nScaleX, nShearY, nShearX, nScaleY,
- nTransX, nTransY);
- vScaleX = nScaleX;
- vScaleY = nScaleY;
- vShearX = nShearX;
- vShearY = nShearY;
- vTransX = nTransY;
- vTransY = nTransY;
- xformInUse = true;
+ if ((caps & CAPS_DEVICE_OK) != 0) {
+ buf.append("CAPS_DEVICE_OK|");
}
+ return buf.toString();
}
-
- // validate pixel
- if (pixel != validatedPixel) {
- validatedPixel = pixel;
- setColor(nativeContext, pixel, flags);
- }
-
- // save flags for later comparison
- validatedFlags = flags;
-
- // mark dstData dirty
- dstData.markDirty();
}
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DDrawImage.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DDrawImage.java
index df7be228633..774294fe4ca 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DDrawImage.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DDrawImage.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -29,6 +29,8 @@ import java.awt.Color;
import java.awt.Image;
import java.awt.geom.AffineTransform;
import java.awt.image.AffineTransformOp;
+import java.awt.image.BufferedImage;
+import java.awt.image.BufferedImageOp;
import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
import sun.java2d.loops.SurfaceType;
@@ -36,6 +38,8 @@ import sun.java2d.loops.TransformBlit;
import sun.java2d.pipe.DrawImage;
public class D3DDrawImage extends DrawImage {
+
+ @Override
protected void renderImageXform(SunGraphics2D sg, Image img,
AffineTransform tx, int interpType,
int sx1, int sy1, int sx2, int sy2,
@@ -44,7 +48,6 @@ public class D3DDrawImage extends DrawImage {
// punt to the MediaLib-based transformImage() in the superclass if:
// - bicubic interpolation is specified
// - a background color is specified and will be used
- // - the source surface is not a texture
// - an appropriate TransformBlit primitive could not be found
if (interpType != AffineTransformOp.TYPE_BICUBIC) {
SurfaceData dstData = sg.surfaceData;
@@ -54,10 +57,7 @@ public class D3DDrawImage extends DrawImage {
sg.imageComp,
bgColor);
- if (srcData != null &&
- !isBgOperation(srcData, bgColor) &&
- srcData.getSurfaceType() == D3DSurfaceData.D3DTexture)
- {
+ if (srcData != null && !isBgOperation(srcData, bgColor)) {
SurfaceType srcType = srcData.getSurfaceType();
SurfaceType dstType = dstData.getSurfaceType();
TransformBlit blit = TransformBlit.getFromCache(srcType,
@@ -77,4 +77,25 @@ public class D3DDrawImage extends DrawImage {
super.renderImageXform(sg, img, tx, interpType,
sx1, sy1, sx2, sy2, bgColor);
}
+
+ @Override
+ public void transformImage(SunGraphics2D sg, BufferedImage img,
+ BufferedImageOp op, int x, int y)
+ {
+ if (op != null) {
+ if (op instanceof AffineTransformOp) {
+ AffineTransformOp atop = (AffineTransformOp) op;
+ transformImage(sg, img, x, y,
+ atop.getTransform(),
+ atop.getInterpolationType());
+ return;
+ } else {
+ if (D3DBufImgOps.renderImageWithOp(sg, img, op, x, y)) {
+ return;
+ }
+ }
+ img = op.filter(img, null);
+ }
+ copyImage(sg, img, x, y, null);
+ }
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DGraphicsConfig.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DGraphicsConfig.java
new file mode 100644
index 00000000000..4bc7e085f12
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DGraphicsConfig.java
@@ -0,0 +1,327 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.AWTException;
+import java.awt.BufferCapabilities;
+import java.awt.BufferCapabilities.FlipContents;
+import java.awt.Component;
+import java.awt.Graphics;
+import java.awt.ImageCapabilities;
+import java.awt.Transparency;
+import java.awt.color.ColorSpace;
+import java.awt.image.ColorModel;
+import java.awt.image.DataBuffer;
+import java.awt.image.DirectColorModel;
+import java.awt.image.VolatileImage;
+import sun.awt.Win32GraphicsConfig;
+import sun.awt.image.SunVolatileImage;
+import sun.awt.image.SurfaceManager;
+import sun.awt.windows.WComponentPeer;
+import sun.java2d.Surface;
+import sun.java2d.SurfaceData;
+import sun.java2d.pipe.hw.AccelDeviceEventNotifier;
+import sun.java2d.pipe.hw.AccelTypedVolatileImage;
+import sun.java2d.pipe.hw.AccelGraphicsConfig;
+import sun.java2d.pipe.hw.AccelSurface;
+import sun.java2d.pipe.hw.ContextCapabilities;
+import static sun.java2d.pipe.hw.AccelSurface.*;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
+import sun.java2d.pipe.hw.AccelDeviceEventListener;
+
+public class D3DGraphicsConfig
+ extends Win32GraphicsConfig
+ implements AccelGraphicsConfig
+{
+ private static ImageCapabilities imageCaps = new D3DImageCaps();
+
+ private BufferCapabilities bufferCaps;
+ private D3DGraphicsDevice device;
+
+ protected D3DGraphicsConfig(D3DGraphicsDevice device) {
+ super(device, 0);
+ this.device = device;
+ }
+
+ public SurfaceData createManagedSurface(int w, int h, int transparency) {
+ return D3DSurfaceData.createData(this, w, h,
+ getColorModel(transparency),
+ null,
+ D3DSurfaceData.TEXTURE);
+ }
+
+ @Override
+ public synchronized void displayChanged() {
+ super.displayChanged();
+ // the context could hold a reference to a D3DSurfaceData, which in
+ // turn has a reference back to this D3DGraphicsConfig, so in order
+ // for this instance to be disposed we need to break the connection
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ D3DContext.invalidateCurrentContext();
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ @Override
+ public ColorModel getColorModel(int transparency) {
+ switch (transparency) {
+ case Transparency.OPAQUE:
+ // REMIND: once the ColorModel spec is changed, this should be
+ // an opaque premultiplied DCM...
+ return new DirectColorModel(24, 0xff0000, 0xff00, 0xff);
+ case Transparency.BITMASK:
+ return new DirectColorModel(25, 0xff0000, 0xff00, 0xff, 0x1000000);
+ case Transparency.TRANSLUCENT:
+ ColorSpace cs = ColorSpace.getInstance(ColorSpace.CS_sRGB);
+ return new DirectColorModel(cs, 32,
+ 0xff0000, 0xff00, 0xff, 0xff000000,
+ true, DataBuffer.TYPE_INT);
+ default:
+ return null;
+ }
+ }
+
+ @Override
+ public String toString() {
+ return ("D3DGraphicsConfig[dev="+screen+",pixfmt="+visual+"]");
+ }
+
+ /**
+ * The following methods are invoked from WComponentPeer.java rather
+ * than having the Win32-dependent implementations hardcoded in that
+ * class. This way the appropriate actions are taken based on the peer's
+ * GraphicsConfig, whether it is a Win32GraphicsConfig or a
+ * D3DGraphicsConfig.
+ */
+
+ /**
+ * Creates a new SurfaceData that will be associated with the given
+ * WComponentPeer. D3D9 doesn't allow rendering to the screen,
+ * so a GDI surface will be returned.
+ */
+ @Override
+ public SurfaceData createSurfaceData(WComponentPeer peer,
+ int numBackBuffers)
+ {
+ return super.createSurfaceData(peer, numBackBuffers);
+ }
+
+ /**
+ * The following methods correspond to the multibuffering methods in
+ * WComponentPeer.java...
+ */
+
+ /**
+ * Checks that the requested configuration is natively supported; if not,
+ * an AWTException is thrown.
+ */
+ @Override
+ public void assertOperationSupported(Component target,
+ int numBuffers,
+ BufferCapabilities caps)
+ throws AWTException
+ {
+ if (numBuffers < 2 || numBuffers > 4) {
+ throw new AWTException("Only 2-4 buffers supported");
+ }
+ if (caps.getFlipContents() == BufferCapabilities.FlipContents.COPIED &&
+ numBuffers != 2)
+ {
+ throw new AWTException("FlipContents.COPIED is only" +
+ "supported for 2 buffers");
+ }
+ }
+
+ /**
+ * Creates a D3D-based backbuffer for the given peer and returns the
+ * image wrapper.
+ */
+ @Override
+ public VolatileImage createBackBuffer(WComponentPeer peer) {
+ Component target = (Component)peer.getTarget();
+ // it is possible for the component to have size 0x0, adjust it to
+ // be at least 1x1 to avoid IAE
+ int w = Math.max(1, target.getWidth());
+ int h = Math.max(1, target.getHeight());
+ return new SunVolatileImage(target, w, h, Boolean.TRUE);
+ }
+
+ /**
+ * Performs the native D3D flip operation for the given target Component.
+ */
+ @Override
+ public void flip(WComponentPeer peer,
+ Component target, VolatileImage backBuffer,
+ int x1, int y1, int x2, int y2,
+ BufferCapabilities.FlipContents flipAction)
+ {
+ // REMIND: we should actually get a surface data for the
+ // backBuffer's VI
+ SurfaceManager d3dvsm =
+ SurfaceManager.getManager(backBuffer);
+ SurfaceData sd = d3dvsm.getPrimarySurfaceData();
+ if (sd instanceof D3DSurfaceData) {
+ D3DSurfaceData d3dsd = (D3DSurfaceData)sd;
+ D3DSurfaceData.swapBuffers(d3dsd, x1, y1, x2, y2);
+ } else {
+ // the surface was likely lost could not have been restored
+ Graphics g = peer.getGraphics();
+ try {
+ g.drawImage(backBuffer,
+ x1, y1, x2, y2,
+ x1, y1, x2, y2,
+ null);
+ } finally {
+ g.dispose();
+ }
+ }
+
+ if (flipAction == BufferCapabilities.FlipContents.BACKGROUND) {
+ Graphics g = backBuffer.getGraphics();
+ try {
+ g.setColor(target.getBackground());
+ g.fillRect(0, 0,
+ backBuffer.getWidth(),
+ backBuffer.getHeight());
+ } finally {
+ g.dispose();
+ }
+ }
+ }
+
+ private static class D3DBufferCaps extends BufferCapabilities {
+ public D3DBufferCaps() {
+ // REMIND: should we indicate that the front-buffer
+ // (the on-screen rendering) is not accelerated?
+ super(imageCaps, imageCaps, FlipContents.UNDEFINED);
+ }
+ @Override
+ public boolean isMultiBufferAvailable() {
+ return true;
+ }
+
+ }
+
+ @Override
+ public BufferCapabilities getBufferCapabilities() {
+ if (bufferCaps == null) {
+ bufferCaps = new D3DBufferCaps();
+ }
+ return bufferCaps;
+ }
+
+ private static class D3DImageCaps extends ImageCapabilities {
+ private D3DImageCaps() {
+ super(true);
+ }
+ @Override
+ public boolean isTrueVolatile() {
+ return true;
+ }
+ }
+
+ @Override
+ public ImageCapabilities getImageCapabilities() {
+ return imageCaps;
+ }
+
+ D3DGraphicsDevice getD3DDevice() {
+ return device;
+ }
+
+ /**
+ * {@inheritDoc}
+ *
+ * @see sun.java2d.pipe.hw.BufferedContextProvider#getContext
+ */
+ @Override
+ public D3DContext getContext() {
+ return device.getContext();
+ }
+
+ /**
+ * {@inheritDoc}
+ *
+ * @see sun.java2d.pipe.hw.AccelGraphicsConfig#createCompatibleVolatileImage
+ */
+ @Override
+ public VolatileImage
+ createCompatibleVolatileImage(int width, int height,
+ int transparency, int type)
+ {
+ if (type == FLIP_BACKBUFFER || type == WINDOW || type == UNDEFINED ||
+ transparency == Transparency.BITMASK)
+ {
+ return null;
+ }
+ boolean isOpaque = transparency == Transparency.OPAQUE;
+ if (type == RT_TEXTURE) {
+ int cap = isOpaque ? CAPS_RT_TEXTURE_OPAQUE : CAPS_RT_TEXTURE_ALPHA;
+ if (!device.isCapPresent(cap)) {
+ return null;
+ }
+ } else if (type == RT_PLAIN) {
+ if (!isOpaque && !device.isCapPresent(CAPS_RT_PLAIN_ALPHA)) {
+ return null;
+ }
+ }
+
+ SunVolatileImage vi = new AccelTypedVolatileImage(this, width, height,
+ transparency, type);
+ Surface sd = vi.getDestSurface();
+ if (!(sd instanceof AccelSurface) ||
+ ((AccelSurface)sd).getType() != type)
+ {
+ vi.flush();
+ vi = null;
+ }
+
+ return vi;
+ }
+
+ /**
+ * {@inheritDoc}
+ *
+ * @see sun.java2d.pipe.hw.AccelGraphicsConfig#getContextCapabilities
+ */
+ @Override
+ public ContextCapabilities getContextCapabilities() {
+ return device.getContextCapabilities();
+ }
+
+ @Override
+ public void addDeviceEventListener(AccelDeviceEventListener l) {
+ AccelDeviceEventNotifier.addListener(l, device.getScreen());
+ }
+
+ @Override
+ public void removeDeviceEventListener(AccelDeviceEventListener l) {
+ AccelDeviceEventNotifier.removeListener(l);
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DGraphicsDevice.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DGraphicsDevice.java
new file mode 100644
index 00000000000..a4334b42f12
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DGraphicsDevice.java
@@ -0,0 +1,500 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.Dialog;
+import java.awt.DisplayMode;
+import java.awt.Frame;
+import java.awt.GraphicsConfiguration;
+import java.awt.Rectangle;
+import java.awt.Toolkit;
+import java.awt.Window;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.event.WindowListener;
+import java.awt.peer.WindowPeer;
+import java.util.ArrayList;
+import sun.awt.Win32GraphicsDevice;
+import sun.awt.windows.WWindowPeer;
+import sun.java2d.pipe.hw.ContextCapabilities;
+import sun.java2d.windows.WindowsFlags;
+import static sun.java2d.pipe.BufferedOpCodes.*;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
+import sun.java2d.d3d.D3DContext.D3DContextCaps;
+
+/**
+ * This class implements D3D-specific functionality, such as fullscreen
+ * exclusive mode and display changes. It is kept separate from
+ * Win32GraphicsDevice to help avoid overburdening the parent class.
+ */
+public class D3DGraphicsDevice extends Win32GraphicsDevice {
+ private D3DContext context;
+
+ private static boolean d3dAvailable;
+
+ private ContextCapabilities d3dCaps;
+
+ private static native boolean initD3D();
+
+ static {
+ // loading the library doesn't help because we need the
+ // toolkit thread running, so we have to call getDefaultToolkit()
+ Toolkit.getDefaultToolkit();
+ d3dAvailable = initD3D();
+ if (d3dAvailable) {
+ // we don't use pixel formats for the d3d pipeline
+ pfDisabled = true;
+ }
+ }
+
+ /**
+ * Used to construct a Direct3D-enabled GraphicsDevice.
+ *
+ * @return a D3DGraphicsDevice if it could be created
+ * successfully, null otherwise.
+ */
+ public static D3DGraphicsDevice createDevice(int screen) {
+ if (!d3dAvailable) {
+ return null;
+ }
+
+ ContextCapabilities d3dCaps = getDeviceCaps(screen);
+ // could not initialize the device successfully
+ if ((d3dCaps.getCaps() & CAPS_DEVICE_OK) == 0) {
+ if (WindowsFlags.isD3DVerbose()) {
+ System.out.println("Could not enable Direct3D pipeline on " +
+ "screen " + screen);
+ }
+ return null;
+ }
+ if (WindowsFlags.isD3DVerbose()) {
+ System.out.println("Direct3D pipeline enabled on screen " + screen);
+ }
+
+ D3DGraphicsDevice gd = new D3DGraphicsDevice(screen, d3dCaps);
+ return gd;
+ }
+
+ private static native int getDeviceCapsNative(int screen);
+ private static native String getDeviceIdNative(int screen);
+ private static ContextCapabilities getDeviceCaps(final int screen) {
+ ContextCapabilities d3dCaps = null;
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ class Result {
+ int caps;
+ String id;
+ };
+ final Result res = new Result();
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ res.caps = getDeviceCapsNative(screen);
+ res.id = getDeviceIdNative(screen);
+ }
+ });
+ d3dCaps = new D3DContextCaps(res.caps, res.id);
+ } finally {
+ rq.unlock();
+ }
+
+ return d3dCaps != null ? d3dCaps : new D3DContextCaps(CAPS_EMPTY, null);
+ }
+
+ public final boolean isCapPresent(int cap) {
+ return ((d3dCaps.getCaps() & cap) != 0);
+ }
+
+ private D3DGraphicsDevice(int screennum, ContextCapabilities d3dCaps) {
+ super(screennum);
+ descString = "D3DGraphicsDevice[screen="+screennum;
+ this.d3dCaps = d3dCaps;
+ context = new D3DContext(D3DRenderQueue.getInstance(), this);
+ }
+
+ public boolean isD3DEnabledOnDevice() {
+ return isValid() && isCapPresent(CAPS_DEVICE_OK);
+ }
+
+ /**
+ * Returns true if d3d pipeline has been successfully initialized.
+ * @return true if d3d pipeline is initialized, false otherwise
+ */
+ public static boolean isD3DAvailable() {
+ return d3dAvailable;
+ }
+
+ /**
+ * Return the owning Frame for a given Window. Used in setFSWindow below
+ * to set the properties of the owning Frame when a Window goes
+ * into fullscreen mode.
+ */
+ private Frame getToplevelOwner(Window w) {
+ Window owner = w;
+ while (owner != null) {
+ owner = owner.getOwner();
+ if (owner instanceof Frame) {
+ return (Frame) owner;
+ }
+ }
+ // could get here if passed Window is an owner-less Dialog
+ return null;
+ }
+
+ private boolean fsStatus;
+ private Rectangle ownerOrigBounds = null;
+ private boolean ownerWasVisible;
+ private Window realFSWindow;
+ private WindowListener fsWindowListener;
+ private boolean fsWindowWasAlwaysOnTop;
+ private static native boolean enterFullScreenExclusiveNative(int screen,
+ long hwnd);
+
+ @Override
+ protected void enterFullScreenExclusive(final int screen, WindowPeer wp)
+ {
+ final WWindowPeer wpeer = (WWindowPeer)realFSWindow.getPeer();
+
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ long hwnd = wpeer.getHWnd();
+ if (hwnd == 0l) {
+ // window is disposed
+ fsStatus = false;
+ return;
+ }
+ fsStatus = enterFullScreenExclusiveNative(screen, hwnd);
+ }
+ });
+ } finally {
+ rq.unlock();
+ }
+ if (!fsStatus) {
+ super.enterFullScreenExclusive(screen, wp);
+ }
+ }
+
+ private static native boolean exitFullScreenExclusiveNative(int screen);
+ @Override
+ protected void exitFullScreenExclusive(final int screen, WindowPeer w) {
+ if (fsStatus) {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ exitFullScreenExclusiveNative(screen);
+ }
+ });
+ } finally {
+ rq.unlock();
+ }
+ } else {
+ super.exitFullScreenExclusive(screen, w);
+ }
+ }
+
+ /**
+ * WindowAdapter class for the full-screen frame, responsible for
+ * restoring the devices. This is important to do because unless the device
+ * is restored it will not go back into the FS mode once alt+tabbed out.
+ * This is a problem for windows for which we do not do any d3d-related
+ * operations (like when we disabled on-screen rendering).
+ *
+ * REMIND: we create an instance per each full-screen device while a single
+ * instance would suffice (but requires more management).
+ */
+ private static class D3DFSWindowAdapter extends WindowAdapter {
+ @Override
+ public void windowDeactivated(WindowEvent e) {
+ D3DRenderQueue.getInstance().restoreDevices();
+ }
+ @Override
+ public void windowActivated(WindowEvent e) {
+ D3DRenderQueue.getInstance().restoreDevices();
+ }
+ }
+
+ @Override
+ protected void addFSWindowListener(Window w) {
+ // if the window is not a toplevel (has an owner) we have to use the
+ // real toplevel to enter the full-screen mode with (4933099).
+ if (!(w instanceof Frame) && !(w instanceof Dialog) &&
+ (realFSWindow = getToplevelOwner(w)) != null)
+ {
+ ownerOrigBounds = realFSWindow.getBounds();
+ WWindowPeer fp = (WWindowPeer)realFSWindow.getPeer();
+
+ ownerWasVisible = realFSWindow.isVisible();
+ Rectangle r = w.getBounds();
+ // we use operations on peer instead of component because calling
+ // them on component will take the tree lock
+ fp.reshape(r.x, r.y, r.width, r.height);
+ fp.setVisible(true);
+ } else {
+ realFSWindow = w;
+ }
+
+ fsWindowWasAlwaysOnTop = realFSWindow.isAlwaysOnTop();
+ ((WWindowPeer)realFSWindow.getPeer()).setAlwaysOnTop(true);
+
+ fsWindowListener = new D3DFSWindowAdapter();
+ realFSWindow.addWindowListener(fsWindowListener);
+ }
+
+ @Override
+ protected void removeFSWindowListener(Window w) {
+ realFSWindow.removeWindowListener(fsWindowListener);
+ fsWindowListener = null;
+
+ /**
+ * Bug 4933099: There is some funny-business to deal with when this
+ * method is called with a Window instead of a Frame. See 4836744
+ * for more information on this. One side-effect of our workaround
+ * for the problem is that the owning Frame of a Window may end
+ * up getting resized during the fullscreen process. When we
+ * return from fullscreen mode, we should resize the Frame to
+ * its original size (just like the Window is being resized
+ * to its original size in GraphicsDevice).
+ */
+ WWindowPeer wpeer = (WWindowPeer)realFSWindow.getPeer();
+ if (wpeer != null) {
+ if (ownerOrigBounds != null) {
+ // if the window went into fs mode before it was realized it
+ // could have (0,0) dimensions
+ if (ownerOrigBounds.width == 0) ownerOrigBounds.width = 1;
+ if (ownerOrigBounds.height == 0) ownerOrigBounds.height = 1;
+ wpeer.reshape(ownerOrigBounds.x, ownerOrigBounds.y,
+ ownerOrigBounds.width, ownerOrigBounds.height);
+ if (!ownerWasVisible) {
+ wpeer.setVisible(false);
+ }
+ ownerOrigBounds = null;
+ }
+ if (!fsWindowWasAlwaysOnTop) {
+ wpeer.setAlwaysOnTop(false);
+ }
+ }
+
+ realFSWindow = null;
+ }
+
+ private static native DisplayMode getCurrentDisplayModeNative(int screen);
+ @Override
+ protected DisplayMode getCurrentDisplayMode(final int screen) {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ class Result {
+ DisplayMode dm = null;
+ };
+ final Result res = new Result();
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ res.dm = getCurrentDisplayModeNative(screen);
+ }
+ });
+ if (res.dm == null) {
+ return super.getCurrentDisplayMode(screen);
+ }
+ return res.dm;
+ } finally {
+ rq.unlock();
+ }
+ }
+ private static native void configDisplayModeNative(int screen, long hwnd,
+ int width, int height,
+ int bitDepth,
+ int refreshRate);
+ @Override
+ protected void configDisplayMode(final int screen, final WindowPeer w,
+ final int width, final int height,
+ final int bitDepth, final int refreshRate)
+ {
+ // we entered fs mode via gdi
+ if (!fsStatus) {
+ super.configDisplayMode(screen, w, width, height, bitDepth,
+ refreshRate);
+ return;
+ }
+
+ final WWindowPeer wpeer = (WWindowPeer)realFSWindow.getPeer();
+
+ // REMIND: we do this before we switch the display mode, so
+ // the dimensions may be exceeding the dimensions of the screen,
+ // is this a problem?
+
+ // update the bounds of the owner frame
+ if (getFullScreenWindow() != realFSWindow) {
+ Rectangle screenBounds = getDefaultConfiguration().getBounds();
+ wpeer.reshape(screenBounds.x, screenBounds.y, width, height);
+ }
+
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ long hwnd = wpeer.getHWnd();
+ if (hwnd == 0l) {
+ // window is disposed
+ return;
+ }
+ // REMIND: do we really need a window here?
+ // we should probably just use the current one
+ configDisplayModeNative(screen, hwnd, width, height,
+ bitDepth, refreshRate);
+ }
+ });
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ private static native void enumDisplayModesNative(int screen,
+ ArrayList modes);
+ @Override
+ protected void enumDisplayModes(final int screen, final ArrayList modes) {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ enumDisplayModesNative(screen, modes);
+ }
+ });
+ if (modes.size() == 0) {
+ modes.add(getCurrentDisplayModeNative(screen));
+ }
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ private static native long getAvailableAcceleratedMemoryNative(int screen);
+ @Override
+ public int getAvailableAcceleratedMemory() {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ class Result {
+ long mem = 0L;
+ };
+ final Result res = new Result();
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ res.mem = getAvailableAcceleratedMemoryNative(getScreen());
+ }
+ });
+ return (int)res.mem;
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ @Override
+ public GraphicsConfiguration[] getConfigurations() {
+ if (configs == null) {
+ if (isD3DEnabledOnDevice()) {
+ defaultConfig = getDefaultConfiguration();
+ if (defaultConfig != null) {
+ configs = new GraphicsConfiguration[1];
+ configs[0] = defaultConfig;
+ return configs;
+ }
+ }
+ }
+ return super.getConfigurations();
+ }
+
+ @Override
+ public GraphicsConfiguration getDefaultConfiguration() {
+ if (defaultConfig == null) {
+ if (isD3DEnabledOnDevice()) {
+ defaultConfig = new D3DGraphicsConfig(this);
+ } else {
+ defaultConfig = super.getDefaultConfiguration();
+ }
+ }
+ return defaultConfig;
+ }
+
+ private static native boolean isD3DAvailableOnDeviceNative(int screen);
+ // REMIND: this method is not used now, we use caps instead
+ public static boolean isD3DAvailableOnDevice(final int screen) {
+ if (!d3dAvailable) {
+ return false;
+ }
+
+ // REMIND: should we cache the result per device somehow,
+ // and then reset and retry it on display change?
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ class Result {
+ boolean avail = false;
+ };
+ final Result res = new Result();
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ res.avail = isD3DAvailableOnDeviceNative(screen);
+ }
+ });
+ return res.avail;
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ D3DContext getContext() {
+ return context;
+ }
+
+ ContextCapabilities getContextCapabilities() {
+ return d3dCaps;
+ }
+
+ @Override
+ public void displayChanged() {
+ super.displayChanged();
+ // REMIND: make sure this works when the device is lost and we don't
+ // disable d3d too eagerly
+ if (d3dAvailable) {
+ d3dCaps = getDeviceCaps(getScreen());
+ }
+ }
+
+ @Override
+ protected void invalidate(int defaultScreen) {
+ super.invalidate(defaultScreen);
+ // REMIND: this is a bit excessive, isD3DEnabledOnDevice will return
+ // false anyway because the device is invalid
+ d3dCaps = new D3DContextCaps(CAPS_EMPTY, null);
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskBlit.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskBlit.java
new file mode 100644
index 00000000000..dbdc345e2ec
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskBlit.java
@@ -0,0 +1,69 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.Composite;
+import sun.java2d.SurfaceData;
+import sun.java2d.loops.CompositeType;
+import sun.java2d.loops.GraphicsPrimitive;
+import sun.java2d.loops.GraphicsPrimitiveMgr;
+import sun.java2d.loops.SurfaceType;
+import sun.java2d.pipe.Region;
+import sun.java2d.pipe.BufferedMaskBlit;
+import static sun.java2d.loops.CompositeType.*;
+import static sun.java2d.loops.SurfaceType.*;
+
+class D3DMaskBlit extends BufferedMaskBlit {
+
+ static void register() {
+ GraphicsPrimitive[] primitives = {
+ new D3DMaskBlit(IntArgb, SrcOver),
+ new D3DMaskBlit(IntArgbPre, SrcOver),
+ new D3DMaskBlit(IntRgb, SrcOver),
+ new D3DMaskBlit(IntRgb, SrcNoEa),
+ new D3DMaskBlit(IntBgr, SrcOver),
+ new D3DMaskBlit(IntBgr, SrcNoEa),
+ };
+ GraphicsPrimitiveMgr.register(primitives);
+ }
+
+ private D3DMaskBlit(SurfaceType srcType,
+ CompositeType compType)
+ {
+ super(D3DRenderQueue.getInstance(),
+ srcType, compType, D3DSurfaceData.D3DSurface);
+ }
+
+ @Override
+ protected void validateContext(SurfaceData dstData,
+ Composite comp, Region clip)
+ {
+ D3DSurfaceData d3dDst = (D3DSurfaceData)dstData;
+ D3DContext.validateContext(d3dDst, d3dDst,
+ clip, comp, null, null, null,
+ D3DContext.NO_CONTEXT_FLAGS);
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskFill.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskFill.java
index 1f0abcbbf32..51a54cd23e4 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskFill.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DMaskFill.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -25,115 +25,51 @@
package sun.java2d.d3d;
-import java.awt.AlphaComposite;
import java.awt.Composite;
import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
import sun.java2d.loops.GraphicsPrimitive;
import sun.java2d.loops.GraphicsPrimitiveMgr;
import sun.java2d.loops.CompositeType;
import sun.java2d.loops.SurfaceType;
-import sun.java2d.loops.MaskFill;
-import static sun.java2d.d3d.D3DSurfaceData.*;
+import sun.java2d.pipe.BufferedMaskFill;
+import static sun.java2d.loops.CompositeType.*;
+import static sun.java2d.loops.SurfaceType.*;
-/**
- * The MaskFill operation is expressed as:
- * dst = ((src dst) * pathA) + (dst * (1 - pathA))
- *
- * The D3D implementation of the MaskFill operation differs from the above
- * equation because it is not possible to perform such a complex operation in
- * D3d (without the use of advanced techniques like fragment shaders and
- * multitexturing). Therefore, the D3DMaskFill operation is expressed as:
- * dst = (src * pathA) dst
- *
- * This simplified formula is only equivalent to the "true" MaskFill equation
- * in the following situations:
- * - is SrcOver
- * - is Src, extra alpha == 1.0, and the source color is opaque
- *
- * Therefore, we register D3DMaskFill primitives for only the SurfaceType and
- * CompositeType restrictions mentioned above. In addition for the Src
- * case, we must override the composite with a SrcOver (no extra alpha)
- * instance, so that we set up the D3d blending mode to match the
- * D3DMaskFill equation.
- */
-public class D3DMaskFill extends MaskFill {
+class D3DMaskFill extends BufferedMaskFill {
- public static void register() {
+ static void register() {
GraphicsPrimitive[] primitives = {
- new D3DMaskFill(SurfaceType.AnyColor,
- CompositeType.SrcOver,
- IntRgbD3D),
- new D3DMaskFill(SurfaceType.OpaqueColor,
- CompositeType.SrcNoEa,
- IntRgbD3D),
-
- new D3DMaskFill(SurfaceType.AnyColor,
- CompositeType.SrcOver,
- Ushort565RgbD3D),
- new D3DMaskFill(SurfaceType.OpaqueColor,
- CompositeType.SrcNoEa,
- Ushort565RgbD3D),
-
- new D3DMaskFill(SurfaceType.AnyColor,
- CompositeType.SrcOver,
- IntRgbxD3D),
- new D3DMaskFill(SurfaceType.OpaqueColor,
- CompositeType.SrcNoEa,
- IntRgbxD3D),
-
- new D3DMaskFill(SurfaceType.AnyColor,
- CompositeType.SrcOver,
- Ushort555RgbD3D),
- new D3DMaskFill(SurfaceType.OpaqueColor,
- CompositeType.SrcNoEa,
- Ushort555RgbD3D),
-
- new D3DMaskFill(SurfaceType.AnyColor,
- CompositeType.SrcOver,
- Ushort555RgbxD3D),
- new D3DMaskFill(SurfaceType.OpaqueColor,
- CompositeType.SrcNoEa,
- Ushort555RgbxD3D),
-
- new D3DMaskFill(SurfaceType.AnyColor,
- CompositeType.SrcOver,
- ThreeByteBgrD3D),
- new D3DMaskFill(SurfaceType.OpaqueColor,
- CompositeType.SrcNoEa,
- ThreeByteBgrD3D),
+ new D3DMaskFill(AnyColor, SrcOver),
+ new D3DMaskFill(OpaqueColor, SrcNoEa),
+ new D3DMaskFill(GradientPaint, SrcOver),
+ new D3DMaskFill(OpaqueGradientPaint, SrcNoEa),
+ new D3DMaskFill(LinearGradientPaint, SrcOver),
+ new D3DMaskFill(OpaqueLinearGradientPaint, SrcNoEa),
+ new D3DMaskFill(RadialGradientPaint, SrcOver),
+ new D3DMaskFill(OpaqueRadialGradientPaint, SrcNoEa),
+ new D3DMaskFill(TexturePaint, SrcOver),
+ new D3DMaskFill(OpaqueTexturePaint, SrcNoEa),
};
GraphicsPrimitiveMgr.register(primitives);
}
- D3DMaskFill(SurfaceType srcType, CompositeType compType,
- SurfaceType dstType) {
- super(srcType, compType, dstType);
+ protected D3DMaskFill(SurfaceType srcType, CompositeType compType) {
+ super(D3DRenderQueue.getInstance(),
+ srcType, compType, D3DSurfaceData.D3DSurface);
}
- private native void MaskFill(long pData, long pCtx,
- int x, int y, int w, int h,
- byte[] mask, int maskoff, int maskscan);
-
@Override
- public void MaskFill(SunGraphics2D sg2d, SurfaceData sData,
- Composite comp,
- int x, int y, int w, int h,
- byte[] mask, int maskoff, int maskscan)
+ protected native void maskFill(int x, int y, int w, int h,
+ int maskoff, int maskscan, int masklen,
+ byte[] mask);
+
+ @Override
+ protected void validateContext(SunGraphics2D sg2d,
+ Composite comp, int ctxflags)
{
- AlphaComposite acomp = (AlphaComposite)comp;
- if (acomp.getRule() != AlphaComposite.SRC_OVER) {
- comp = AlphaComposite.SrcOver;
- }
-
- synchronized (D3DContext.LOCK) {
- long pCtx = D3DContext.getContext(sData, sData,
- sg2d.getCompClip(), comp,
- null, sg2d.eargb,
- D3DContext.NO_CONTEXT_FLAGS);
-
- MaskFill(sData.getNativeOps(), pCtx, x, y, w, h,
- mask, maskoff, maskscan);
- }
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DContext.validateContext(dstData, dstData,
+ sg2d.getCompClip(), comp,
+ null, sg2d.paint, sg2d, ctxflags);
}
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DPaints.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DPaints.java
new file mode 100644
index 00000000000..a112ffcc49a
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DPaints.java
@@ -0,0 +1,233 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.LinearGradientPaint;
+import java.awt.MultipleGradientPaint;
+import java.awt.MultipleGradientPaint.ColorSpaceType;
+import java.awt.MultipleGradientPaint.CycleMethod;
+import java.awt.TexturePaint;
+import java.awt.image.BufferedImage;
+import java.util.HashMap;
+import java.util.Map;
+import sun.java2d.SunGraphics2D;
+import sun.java2d.SurfaceData;
+import sun.java2d.loops.CompositeType;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
+
+abstract class D3DPaints {
+
+ /**
+ * Holds all registered implementations, using the corresponding
+ * SunGraphics2D.PAINT_* constant as the hash key.
+ */
+ private static Map impls =
+ new HashMap(4, 1.0f);
+
+ static {
+ impls.put(SunGraphics2D.PAINT_GRADIENT, new Gradient());
+ impls.put(SunGraphics2D.PAINT_LIN_GRADIENT, new LinearGradient());
+ impls.put(SunGraphics2D.PAINT_RAD_GRADIENT, new RadialGradient());
+ impls.put(SunGraphics2D.PAINT_TEXTURE, new Texture());
+ }
+
+ /**
+ * Attempts to locate an implementation corresponding to the paint state
+ * of the provided SunGraphics2D object. If no implementation can be
+ * found, or if the paint cannot be accelerated under the conditions
+ * of the SunGraphics2D, this method returns false; otherwise, returns
+ * true.
+ */
+ static boolean isValid(SunGraphics2D sg2d) {
+ D3DPaints impl = impls.get(sg2d.paintState);
+ return (impl != null && impl.isPaintValid(sg2d));
+ }
+
+ /**
+ * Returns true if this implementation is able to accelerate the
+ * Paint object associated with, and under the conditions of, the
+ * provided SunGraphics2D instance; otherwise returns false.
+ */
+ abstract boolean isPaintValid(SunGraphics2D sg2d);
+
+/************************* GradientPaint support ****************************/
+
+ private static class Gradient extends D3DPaints {
+ private Gradient() {}
+
+ /**
+ * Returns true if the given GradientPaint instance can be
+ * used by the accelerated D3DPaints.Gradient implementation.
+ * A GradientPaint is considered valid only if the destination
+ * has support for fragment shaders.
+ */
+ @Override
+ boolean isPaintValid(SunGraphics2D sg2d) {
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DGraphicsDevice gd = (D3DGraphicsDevice)
+ dstData.getDeviceConfiguration().getDevice();
+ return gd.isCapPresent(CAPS_LCD_SHADER);
+ }
+ }
+
+/************************** TexturePaint support ****************************/
+
+ private static class Texture extends D3DPaints {
+ private Texture() {}
+
+ /**
+ * Returns true if the given TexturePaint instance can be used by the
+ * accelerated BufferedPaints.Texture implementation.
+ *
+ * A TexturePaint is considered valid if the following conditions
+ * are met:
+ * - the texture image dimensions are power-of-two
+ * - the texture image can be (or is already) cached in a D3D
+ * texture object
+ */
+ @Override
+ public boolean isPaintValid(SunGraphics2D sg2d) {
+ TexturePaint paint = (TexturePaint)sg2d.paint;
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ BufferedImage bi = paint.getImage();
+
+ // verify that the texture image dimensions are pow2
+ D3DGraphicsDevice gd =
+ (D3DGraphicsDevice)dstData.getDeviceConfiguration().getDevice();
+ int imgw = bi.getWidth();
+ int imgh = bi.getHeight();
+ if (!gd.isCapPresent(CAPS_TEXNONPOW2)) {
+ if ((imgw & (imgw - 1)) != 0 || (imgh & (imgh - 1)) != 0) {
+ return false;
+ }
+ }
+ // verify that the texture image is square if it has to be
+ if (!gd.isCapPresent(CAPS_TEXNONSQUARE) && imgw != imgh)
+ {
+ return false;
+ }
+
+ SurfaceData srcData =
+ dstData.getSourceSurfaceData(bi, sg2d.TRANSFORM_ISIDENT,
+ CompositeType.SrcOver, null);
+ if (!(srcData instanceof D3DSurfaceData)) {
+ // REMIND: this is a hack that attempts to cache the system
+ // memory image from the TexturePaint instance into a
+ // D3D texture...
+ srcData =
+ dstData.getSourceSurfaceData(bi, sg2d.TRANSFORM_ISIDENT,
+ CompositeType.SrcOver, null);
+ if (!(srcData instanceof D3DSurfaceData)) {
+ return false;
+ }
+ }
+
+ // verify that the source surface is actually a texture
+ D3DSurfaceData d3dData = (D3DSurfaceData)srcData;
+ if (d3dData.getType() != D3DSurfaceData.TEXTURE) {
+ return false;
+ }
+
+ return true;
+ }
+ }
+
+/****************** Shared MultipleGradientPaint support ********************/
+
+ private static abstract class MultiGradient extends D3DPaints {
+
+ /**
+ * Note that this number is lower than the MULTI_MAX_FRACTIONS
+ * defined in the superclass. The D3D pipeline now uses a
+ * slightly more complicated shader (to avoid the gradient banding
+ * issues), which has a higher instruction count. To ensure that
+ * all versions of the shader can be compiled for PS 2.0 hardware,
+ * we need to cap this maximum value at 8.
+ */
+ public static final int MULTI_MAX_FRACTIONS_D3D = 8;
+
+ protected MultiGradient() {}
+
+ /**
+ * Returns true if the given MultipleGradientPaint instance can be
+ * used by the accelerated D3DPaints.MultiGradient implementation.
+ * A MultipleGradientPaint is considered valid if the following
+ * conditions are met:
+ * - the number of gradient "stops" is <= MAX_FRACTIONS
+ * - the destination has support for fragment shaders
+ */
+ @Override
+ boolean isPaintValid(SunGraphics2D sg2d) {
+ MultipleGradientPaint paint = (MultipleGradientPaint)sg2d.paint;
+ // REMIND: ugh, this creates garbage; would be nicer if
+ // we had a MultipleGradientPaint.getNumStops() method...
+ if (paint.getFractions().length > MULTI_MAX_FRACTIONS_D3D) {
+ return false;
+ }
+
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DGraphicsDevice gd = (D3DGraphicsDevice)
+ dstData.getDeviceConfiguration().getDevice();
+ if (!gd.isCapPresent(CAPS_LCD_SHADER)) {
+ return false;
+ }
+ return true;
+ }
+ }
+
+/********************** LinearGradientPaint support *************************/
+
+ private static class LinearGradient extends MultiGradient {
+ private LinearGradient() {}
+
+ @Override
+ boolean isPaintValid(SunGraphics2D sg2d) {
+ LinearGradientPaint paint = (LinearGradientPaint)sg2d.paint;
+
+ if (paint.getFractions().length == 2 &&
+ paint.getCycleMethod() != CycleMethod.REPEAT &&
+ paint.getColorSpace() != ColorSpaceType.LINEAR_RGB)
+ {
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DGraphicsDevice gd = (D3DGraphicsDevice)
+ dstData.getDeviceConfiguration().getDevice();
+ if (gd.isCapPresent(CAPS_LCD_SHADER)) {
+ // we can delegate to the optimized two-color gradient
+ // codepath, which should be faster
+ return true;
+ }
+ }
+
+ return super.isPaintValid(sg2d);
+ }
+ }
+
+/********************** RadialGradientPaint support *************************/
+
+ private static class RadialGradient extends MultiGradient {
+ private RadialGradient() {}
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderQueue.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderQueue.java
new file mode 100644
index 00000000000..2cdcc70faab
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderQueue.java
@@ -0,0 +1,159 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import sun.java2d.ScreenUpdateManager;
+import sun.java2d.pipe.RenderBuffer;
+import sun.java2d.pipe.RenderQueue;
+import static sun.java2d.pipe.BufferedOpCodes.*;
+
+/**
+ * D3D-specific implementation of RenderQueue.
+ */
+public class D3DRenderQueue extends RenderQueue {
+
+ private static D3DRenderQueue theInstance;
+ private static Thread rqThread;
+
+ private D3DRenderQueue() {
+ }
+
+ /**
+ * Returns the single D3DRenderQueue instance. If it has not yet been
+ * initialized, this method will first construct the single instance
+ * before returning it.
+ */
+ public static synchronized D3DRenderQueue getInstance() {
+ if (theInstance == null) {
+ theInstance = new D3DRenderQueue();
+ // no need to lock, noone has reference to this instance yet
+ theInstance.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ rqThread = Thread.currentThread();
+ }
+ });
+ }
+ return theInstance;
+ }
+
+ /**
+ * Flushes the single D3DRenderQueue instance synchronously. If an
+ * D3DRenderQueue has not yet been instantiated, this method is a no-op.
+ * This method is useful in the case of Toolkit.sync(), in which we want
+ * to flush the D3D pipeline, but only if the D3D pipeline is currently
+ * enabled. Since this class has few external dependencies, callers need
+ * not be concerned that calling this method will trigger initialization
+ * of the D3D pipeline and related classes.
+ */
+ public static void sync() {
+ if (theInstance != null) {
+ // need to make sure any/all screen surfaces are presented prior
+ // to completing the sync operation
+ D3DScreenUpdateManager mgr =
+ (D3DScreenUpdateManager)ScreenUpdateManager.getInstance();
+ mgr.runUpdateNow();
+
+ theInstance.lock();
+ try {
+ theInstance.ensureCapacity(4);
+ theInstance.getBuffer().putInt(SYNC);
+ theInstance.flushNow();
+ } finally {
+ theInstance.unlock();
+ }
+ }
+ }
+
+ /**
+ * Attempt to restore the devices if they're in the lost state.
+ * (used when a full-screen window is activated/deactivated)
+ */
+ public static void restoreDevices() {
+ D3DRenderQueue rq = getInstance();
+ rq.lock();
+ try {
+ rq.ensureCapacity(4);
+ rq.getBuffer().putInt(RESTORE_DEVICES);
+ rq.flushNow();
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ /**
+ * @return true if current thread is the render queue thread,
+ * false otherwise
+ */
+ public static boolean isRenderQueueThread() {
+ return (Thread.currentThread() == rqThread);
+ }
+
+ /**
+ * Disposes the native memory associated with the given native
+ * graphics config info pointer on the single queue flushing thread.
+ */
+ public static void disposeGraphicsConfig(long pConfigInfo) {
+ D3DRenderQueue rq = getInstance();
+ rq.lock();
+ try {
+
+ RenderBuffer buf = rq.getBuffer();
+ rq.ensureCapacityAndAlignment(12, 4);
+ buf.putInt(DISPOSE_CONFIG);
+ buf.putLong(pConfigInfo);
+
+ // this call is expected to complete synchronously, so flush now
+ rq.flushNow();
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ public void flushNow() {
+ // assert lock.isHeldByCurrentThread();
+ flushBuffer(null);
+ }
+
+ public void flushAndInvokeNow(Runnable r) {
+ // assert lock.isHeldByCurrentThread();
+ flushBuffer(r);
+ }
+
+ private native void flushBuffer(long buf, int limit, Runnable task);
+
+ private void flushBuffer(Runnable task) {
+ // assert lock.isHeldByCurrentThread();
+ int limit = buf.position();
+ if (limit > 0 || task != null) {
+ // process the queue
+ flushBuffer(buf.getAddress(), limit, task);
+ }
+ // reset the buffer position
+ buf.clear();
+ // clear the set of references, since we no longer need them
+ refSet.clear();
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderer.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderer.java
index c56cd7b0802..e93ebdabe27 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderer.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DRenderer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -25,341 +25,169 @@
package sun.java2d.d3d;
-import java.awt.Composite;
-import java.awt.Polygon;
-import java.awt.Shape;
-import java.awt.geom.AffineTransform;
-import java.awt.geom.Arc2D;
-import java.awt.geom.Ellipse2D;
-import java.awt.geom.IllegalPathStateException;
-import java.awt.geom.PathIterator;
-import java.awt.geom.RoundRectangle2D;
+import java.awt.Transparency;
+import java.awt.geom.Path2D;
import sun.java2d.SunGraphics2D;
-import sun.java2d.pipe.Region;
-import sun.java2d.SurfaceData;
import sun.java2d.loops.GraphicsPrimitive;
-import sun.java2d.pipe.LoopPipe;
-import sun.java2d.pipe.ShapeSpanIterator;
+import sun.java2d.pipe.BufferedPaints;
+import sun.java2d.pipe.BufferedRenderPipe;
+import sun.java2d.pipe.RenderQueue;
import sun.java2d.pipe.SpanIterator;
+import sun.java2d.pipe.ParallelogramPipe;
+import static sun.java2d.pipe.BufferedOpCodes.*;
-import static sun.java2d.d3d.D3DContext.*;
-import sun.java2d.windows.DDRenderer;
+class D3DRenderer extends BufferedRenderPipe {
-public class D3DRenderer extends DDRenderer {
-
- native boolean doDrawLineD3D(long pData, long pCtx,
- int x1, int y1, int x2, int y2);
- native boolean doDrawRectD3D(long pData, long pCtx,
- int x, int y, int w, int h);
- native boolean doFillRectD3D(long pData, long pCtx, int x, int y,
- int width, int height);
- native void doDrawPoly(long pData, long pCtx, int transx, int transy,
- int[] xpoints, int[] ypoints,
- int npoints, boolean isclosed);
- native void devFillSpans(long pData, long pCtx, SpanIterator si,
- long iterator, int transx, int transy);
-
-
- private long getContext(SunGraphics2D sg2d) {
- AffineTransform at =
- sg2d.transformState < sg2d.TRANSFORM_TRANSLATESCALE ?
- null : sg2d.transform;
- int ctxflags = (sg2d.eargb >>> 24) == 0xff ?
- SRC_IS_OPAQUE : NO_CONTEXT_FLAGS;
- return D3DContext.getContext(null, sg2d.surfaceData,
- sg2d.getCompClip(),
- sg2d.getComposite(),
- at,
- sg2d.eargb,
- ctxflags);
+ D3DRenderer(RenderQueue rq) {
+ super(rq);
}
@Override
- public void drawLine(SunGraphics2D sg2d,
- int x1, int y1, int x2, int y2)
+ protected void validateContext(SunGraphics2D sg2d) {
+ int ctxflags =
+ sg2d.paint.getTransparency() == Transparency.OPAQUE ?
+ D3DContext.SRC_IS_OPAQUE : D3DContext.NO_CONTEXT_FLAGS;
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DContext.validateContext(dstData, dstData,
+ sg2d.getCompClip(), sg2d.composite,
+ null, sg2d.paint, sg2d, ctxflags);
+ }
+
+ @Override
+ protected void validateContextAA(SunGraphics2D sg2d) {
+ int ctxflags = D3DContext.NO_CONTEXT_FLAGS;
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DContext.validateContext(dstData, dstData,
+ sg2d.getCompClip(), sg2d.composite,
+ null, sg2d.paint, sg2d, ctxflags);
+ }
+
+ void copyArea(SunGraphics2D sg2d,
+ int x, int y, int w, int h, int dx, int dy)
{
- synchronized (D3DContext.LOCK) {
- doDrawLineD3D(sg2d.surfaceData.getNativeOps(),
- getContext(sg2d),
- x1 + sg2d.transX, y1 + sg2d.transY,
- x2 + sg2d.transX, y2 + sg2d.transY);
- }
- }
-
- @Override
- public void fillRect(SunGraphics2D sg2d,
- int x, int y, int width, int height)
- {
- synchronized (D3DContext.LOCK) {
- doFillRectD3D(sg2d.surfaceData.getNativeOps(),
- getContext(sg2d),
- sg2d.transX + x, sg2d.transY + y, width, height);
- }
- }
-
- @Override
- public void drawRect(SunGraphics2D sg2d,
- int x, int y, int width, int height)
- {
- synchronized (D3DContext.LOCK) {
- doDrawRectD3D(sg2d.surfaceData.getNativeOps(),
- getContext(sg2d),
- x + sg2d.transX, sg2d.transY + y, width, height);
- }
- }
-
- @Override
- public void drawPolyline(SunGraphics2D sg2d,
- int xpoints[], int ypoints[], int npoints)
- {
- synchronized (D3DContext.LOCK) {
- doDrawPoly(sg2d.surfaceData.getNativeOps(),
- getContext(sg2d),
- sg2d.transX, sg2d.transY,
- xpoints, ypoints, npoints, false);
- }
- }
-
- @Override
- public void drawPolygon(SunGraphics2D sg2d,
- int xpoints[], int ypoints[], int npoints)
- {
- synchronized (D3DContext.LOCK) {
- doDrawPoly(sg2d.surfaceData.getNativeOps(),
- getContext(sg2d),
- sg2d.transX, sg2d.transY,
- xpoints, ypoints, npoints, true);
- }
- }
-
- @Override
- public void drawRoundRect(SunGraphics2D sg2d,
- int x, int y, int width, int height,
- int arcWidth, int arcHeight)
- {
- draw(sg2d, new RoundRectangle2D.Float(x, y, width, height,
- arcWidth, arcHeight));
- }
-
- @Override
- public void drawOval(SunGraphics2D sg2d,
- int x, int y, int width, int height)
- {
- draw(sg2d, new Ellipse2D.Float(x, y, width, height));
- }
-
- @Override
- public void drawArc(SunGraphics2D sg2d,
- int x, int y, int width, int height,
- int startAngle, int arcAngle)
- {
- draw(sg2d, new Arc2D.Float(x, y, width, height,
- startAngle, arcAngle,
- Arc2D.OPEN));
- }
-
- @Override
- public void fillRoundRect(SunGraphics2D sg2d,
- int x, int y, int width, int height,
- int arcWidth, int arcHeight)
- {
- fill(sg2d, new RoundRectangle2D.Float(x, y, width, height,
- arcWidth, arcHeight));
- }
-
- @Override
- public void fillOval(SunGraphics2D sg2d,
- int x, int y, int width, int height)
- {
- fill(sg2d, new Ellipse2D.Float(x, y, width, height));
- }
-
- @Override
- public void fillArc(SunGraphics2D sg2d,
- int x, int y, int width, int height,
- int startAngle, int arcAngle)
- {
- fill(sg2d, new Arc2D.Float(x, y, width, height,
- startAngle, arcAngle, Arc2D.PIE));
- }
-
- @Override
- public void fillPolygon(SunGraphics2D sg2d,
- int xpoints[], int ypoints[],
- int npoints)
- {
- fill(sg2d, new Polygon(xpoints, ypoints, npoints));
- }
-
- @Override
- public void draw(SunGraphics2D sg2d, Shape s)
- {
- if (sg2d.strokeState == sg2d.STROKE_THIN) {
- Polygon p;
- if (s instanceof Polygon) {
- p = (Polygon) s;
- drawPolygon(sg2d, p.xpoints, p.ypoints, p.npoints);
- return;
- }
- // we're letting d3d handle the transforms
- PathIterator pi = s.getPathIterator(null, 0.5f);
- p = new Polygon();
- float coords[] = new float[2];
- while (!pi.isDone()) {
- switch (pi.currentSegment(coords)) {
- case PathIterator.SEG_MOVETO:
- if (p.npoints > 1) {
- drawPolyline(sg2d, p.xpoints, p.ypoints, p.npoints);
- }
- p.reset();
- p.addPoint((int) Math.floor(coords[0]),
- (int) Math.floor(coords[1]));
- break;
- case PathIterator.SEG_LINETO:
- if (p.npoints == 0) {
- throw new IllegalPathStateException
- ("missing initial moveto in path definition");
- }
- p.addPoint((int) Math.floor(coords[0]),
- (int) Math.floor(coords[1]));
- break;
- case PathIterator.SEG_CLOSE:
- if (p.npoints > 0) {
- p.addPoint(p.xpoints[0], p.ypoints[0]);
- }
- break;
- default:
- throw new
- IllegalPathStateException("path not flattened");
- }
- pi.next();
- }
- if (p.npoints > 1) {
- drawPolyline(sg2d, p.xpoints, p.ypoints, p.npoints);
- }
- } else if (sg2d.strokeState < sg2d.STROKE_CUSTOM) {
- ShapeSpanIterator si = LoopPipe.getStrokeSpans(sg2d, s);
- try {
- synchronized (D3DContext.LOCK) {
- int ctxflags = (sg2d.eargb >>> 24) == 0xff ?
- SRC_IS_OPAQUE : NO_CONTEXT_FLAGS;
- // in this case the spans will be pre-transformed, so we
- // pass null transform to getContext
- long pCtx = D3DContext.getContext(null, sg2d.surfaceData,
- sg2d.getCompClip(),
- sg2d.getComposite(),
- null /*transform*/,
- sg2d.eargb/*pixel*/,
- ctxflags);
- devFillSpans(sg2d.surfaceData.getNativeOps(), pCtx, si,
- si.getNativeIterator(), 0, 0);
- }
- } finally {
- si.dispose();
- }
- } else {
- fill(sg2d, sg2d.stroke.createStrokedShape(s));
- }
- }
-
- @Override
- public void fill(SunGraphics2D sg2d, Shape s) {
- AffineTransform at;
- int transx, transy;
-
- if ( sg2d.transformState < sg2d.TRANSFORM_TRANSLATESCALE) {
- // Transform (translation) will be done by devFillSpans
- at = null;
- transx = sg2d.transX;
- transy = sg2d.transY;
- } else {
- // Transform will be done by the PathIterator
- at = sg2d.transform;
- transx = transy = 0;
- }
-
- ShapeSpanIterator ssi = LoopPipe.getFillSSI(sg2d);
+ rq.lock();
try {
- // Subtract transx/y from the SSI clip to match the
- // (potentially untranslated) geometry fed to it
- Region clip = sg2d.getCompClip();
- ssi.setOutputAreaXYXY(clip.getLoX() - transx,
- clip.getLoY() - transy,
- clip.getHiX() - transx,
- clip.getHiY() - transy);
- ssi.appendPath(s.getPathIterator(at));
- synchronized (D3DContext.LOCK) {
- int ctxflags = (sg2d.eargb >>> 24) == 0xff ?
- SRC_IS_OPAQUE : NO_CONTEXT_FLAGS;
- long pCtx = D3DContext.getContext(null, sg2d.surfaceData,
- sg2d.getCompClip(),
- sg2d.getComposite(),
- null/*transform*/,
- sg2d.eargb/*pixel*/,
- ctxflags);
- devFillSpans(sg2d.surfaceData.getNativeOps(), pCtx, ssi,
- ssi.getNativeIterator(),
- transx, transy);
- }
+ int ctxflags =
+ sg2d.surfaceData.getTransparency() == Transparency.OPAQUE ?
+ D3DContext.SRC_IS_OPAQUE : D3DContext.NO_CONTEXT_FLAGS;
+ D3DSurfaceData dstData = (D3DSurfaceData)sg2d.surfaceData;
+ D3DContext.validateContext(dstData, dstData,
+ sg2d.getCompClip(), sg2d.composite,
+ null, null, null, ctxflags);
+
+ rq.ensureCapacity(28);
+ buf.putInt(COPY_AREA);
+ buf.putInt(x).putInt(y).putInt(w).putInt(h);
+ buf.putInt(dx).putInt(dy);
} finally {
- ssi.dispose();
+ rq.unlock();
}
}
- D3DRenderer traceWrapD3D() {
- return new Tracer();
+ protected native void drawPoly(int[] xPoints, int[] yPoints,
+ int nPoints, boolean isClosed,
+ int transX, int transY);
+
+ D3DRenderer traceWrap() {
+ return new Tracer(this);
}
private class Tracer extends D3DRenderer {
- @Override
+ private D3DRenderer d3dr;
+ Tracer(D3DRenderer d3dr) {
+ super(d3dr.rq);
+ this.d3dr = d3dr;
+ }
+ public ParallelogramPipe getAAParallelogramPipe() {
+ final ParallelogramPipe realpipe = d3dr.getAAParallelogramPipe();
+ return new ParallelogramPipe() {
+ public void fillParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2)
+ {
+ GraphicsPrimitive.tracePrimitive("D3DFillAAParallelogram");
+ realpipe.fillParallelogram(sg2d,
+ x, y, dx1, dy1, dx2, dy2);
+ }
+ public void drawParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2)
+ {
+ GraphicsPrimitive.tracePrimitive("D3DDrawAAParallelogram");
+ realpipe.drawParallelogram(sg2d,
+ x, y, dx1, dy1, dx2, dy2,
+ lw1, lw2);
+ }
+ };
+ }
+
+ protected void validateContext(SunGraphics2D sg2d) {
+ d3dr.validateContext(sg2d);
+ }
public void drawLine(SunGraphics2D sg2d,
int x1, int y1, int x2, int y2)
{
GraphicsPrimitive.tracePrimitive("D3DDrawLine");
- super.drawLine(sg2d, x1, y1, x2, y2);
+ d3dr.drawLine(sg2d, x1, y1, x2, y2);
}
- @Override
public void drawRect(SunGraphics2D sg2d, int x, int y, int w, int h) {
GraphicsPrimitive.tracePrimitive("D3DDrawRect");
- super.drawRect(sg2d, x, y, w, h);
+ d3dr.drawRect(sg2d, x, y, w, h);
}
- @Override
- public void drawPolyline(SunGraphics2D sg2d,
- int[] xPoints, int[] yPoints,
- int nPoints)
- {
- GraphicsPrimitive.tracePrimitive("D3DDrawPolyline");
- super.drawPolyline(sg2d, xPoints, yPoints, nPoints);
- }
- @Override
- public void drawPolygon(SunGraphics2D sg2d,
+ protected void drawPoly(SunGraphics2D sg2d,
int[] xPoints, int[] yPoints,
- int nPoints)
+ int nPoints, boolean isClosed)
{
- GraphicsPrimitive.tracePrimitive("D3DDrawPolygon");
- super.drawPolygon(sg2d, xPoints, yPoints, nPoints);
+ GraphicsPrimitive.tracePrimitive("D3DDrawPoly");
+ d3dr.drawPoly(sg2d, xPoints, yPoints, nPoints, isClosed);
}
- @Override
public void fillRect(SunGraphics2D sg2d, int x, int y, int w, int h) {
GraphicsPrimitive.tracePrimitive("D3DFillRect");
- super.fillRect(sg2d, x, y, w, h);
+ d3dr.fillRect(sg2d, x, y, w, h);
}
- @Override
- void devFillSpans(long pData, long pCtx, SpanIterator si, long iterator,
- int transx, int transy)
+ protected void drawPath(SunGraphics2D sg2d,
+ Path2D.Float p2df, int transx, int transy)
+ {
+ GraphicsPrimitive.tracePrimitive("D3DDrawPath");
+ d3dr.drawPath(sg2d, p2df, transx, transy);
+ }
+ protected void fillPath(SunGraphics2D sg2d,
+ Path2D.Float p2df, int transx, int transy)
+ {
+ GraphicsPrimitive.tracePrimitive("D3DFillPath");
+ d3dr.fillPath(sg2d, p2df, transx, transy);
+ }
+ protected void fillSpans(SunGraphics2D sg2d, SpanIterator si,
+ int transx, int transy)
{
GraphicsPrimitive.tracePrimitive("D3DFillSpans");
- super.devFillSpans(pData, pCtx, si, iterator, transx, transy);
+ d3dr.fillSpans(sg2d, si, transx, transy);
}
- @Override
- public void devCopyArea(SurfaceData sData,
- int srcx, int srcy,
- int dx, int dy,
- int w, int h)
+ public void fillParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2)
{
- GraphicsPrimitive.tracePrimitive("DXCopyArea");
- super.devCopyArea(sData, srcx, srcy, dx, dy, w, h);
+ GraphicsPrimitive.tracePrimitive("D3DFillParallelogram");
+ d3dr.fillParallelogram(sg2d, x, y, dx1, dy1, dx2, dy2);
+ }
+ public void drawParallelogram(SunGraphics2D sg2d,
+ double x, double y,
+ double dx1, double dy1,
+ double dx2, double dy2,
+ double lw1, double lw2)
+ {
+ GraphicsPrimitive.tracePrimitive("D3DDrawParallelogram");
+ d3dr.drawParallelogram(sg2d, x, y, dx1, dy1, dx2, dy2, lw1, lw2);
+ }
+ public void copyArea(SunGraphics2D sg2d,
+ int x, int y, int w, int h, int dx, int dy)
+ {
+ GraphicsPrimitive.tracePrimitive("D3DCopyArea");
+ d3dr.copyArea(sg2d, x, y, w, h, dx, dy);
}
-
}
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DScreenUpdateManager.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DScreenUpdateManager.java
new file mode 100644
index 00000000000..b3b0fa8d5fc
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DScreenUpdateManager.java
@@ -0,0 +1,552 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.Color;
+import java.awt.Component;
+import java.awt.Container;
+import java.awt.Font;
+import java.awt.Graphics2D;
+import java.awt.Rectangle;
+import java.awt.Window;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
+import java.util.ArrayList;
+import java.util.HashMap;
+import sun.awt.SunToolkit;
+import sun.awt.Win32GraphicsConfig;
+import sun.awt.windows.WComponentPeer;
+import sun.java2d.InvalidPipeException;
+import sun.java2d.ScreenUpdateManager;
+import sun.java2d.SunGraphics2D;
+import sun.java2d.SurfaceData;
+import sun.java2d.windows.GDIWindowSurfaceData;
+import sun.java2d.d3d.D3DSurfaceData.D3DWindowSurfaceData;
+import sun.java2d.windows.WindowsFlags;
+
+/**
+ * This class handles rendering to the screen with the D3D pipeline.
+ *
+ * Since it is not possible to render directly to the front buffer
+ * with D3D9, we create a swap chain surface (with COPY effect) in place of the
+ * GDIWindowSurfaceData. A background thread handles the swap chain flips.
+ *
+ * There are some restrictions to which windows we would use this for.
+ * @see #createScreenSurface()
+ */
+public class D3DScreenUpdateManager extends ScreenUpdateManager
+ implements Runnable
+{
+ /**
+ * A window must be at least MIN_WIN_SIZE in one or both dimensions
+ * to be considered for the update manager.
+ */
+ private static final int MIN_WIN_SIZE = 150;
+
+ private volatile boolean done;
+ private volatile Thread screenUpdater;
+ private boolean needsUpdateNow;
+
+ /**
+ * Object used by the screen updater thread for waiting
+ */
+ private Object runLock = new Object();
+ /**
+ * List of D3DWindowSurfaceData surfaces. Surfaces are added to the
+ * list when a graphics object is created, and removed when the surface
+ * is invalidated.
+ */
+ private ArrayList d3dwSurfaces;
+ /**
+ * Cache of GDIWindowSurfaceData surfaces corresponding to the
+ * D3DWindowSurfaceData surfaces. Surfaces are added to the list when
+ * a d3dw surface is lost and could not be restored (due to lack of vram,
+ * for example), and removed then the d3dw surface is invalidated.
+ */
+ private HashMap gdiSurfaces;
+
+ public D3DScreenUpdateManager() {
+ done = false;
+ AccessController.doPrivileged(
+ new PrivilegedAction() {
+ public Object run() {
+ ThreadGroup currentTG =
+ Thread.currentThread().getThreadGroup();
+ ThreadGroup parentTG = currentTG.getParent();
+ while (parentTG != null) {
+ currentTG = parentTG;
+ parentTG = currentTG.getParent();
+ }
+ try {
+ Runtime.getRuntime().addShutdownHook(
+ new Thread(currentTG,
+ new Runnable() {
+ public void run() {
+ done = true;
+ wakeUpUpdateThread();
+ }
+ }
+ )
+ );
+ } catch (Exception e) {
+ done = true;
+ }
+ return null;
+ }
+ }
+ );
+ }
+
+ /**
+ * If possible, creates a D3DWindowSurfaceData (which is actually
+ * a back-buffer surface). If the creation fails, returns GDI
+ * onscreen surface instead.
+ *
+ * Note that the created D3D surface does not initialize the native
+ * resources (and is marked lost) to avoid wasting video memory. It is
+ * restored when a graphics object is requested from the peer.
+ *
+ * Note that this method is called from a synchronized block in
+ * WComponentPeer, so we don't need to synchronize
+ *
+ * Note that we only create a substibute d3dw surface if certain conditions
+ * are met
+ *
+ * the fake d3d rendering on screen is not disabled via flag
+ * d3d on the device is enabled
+ * surface is larger than MIN_WIN_SIZE (don't bother for smaller ones)
+ * it doesn't have a backBuffer for a BufferStrategy already
+ * the peer is either Canvas, Panel, Window, Frame,
+ * Dialog or EmbeddedFrame
+ *
+ *
+ * @param gc GraphicsConfiguration on associated with the surface
+ * @param peer peer for which the surface is to be created
+ * @param bbNum number of back-buffers requested. if this number is >0,
+ * method returns GDI surface (we don't want to have two swap chains)
+ * @param isResize whether this surface is being created in response to
+ * a component resize event. This determines whether a repaint event will
+ * be issued after a surface is created: it will be if isResize
+ * is true
.
+ * @return surface data to be use for onscreen rendering
+ */
+ @Override
+ public SurfaceData createScreenSurface(Win32GraphicsConfig gc,
+ WComponentPeer peer,
+ int bbNum, boolean isResize)
+ {
+ if (done || !(gc instanceof D3DGraphicsConfig)) {
+ return super.createScreenSurface(gc, peer, bbNum, isResize);
+ }
+
+ SurfaceData sd = null;
+
+ if (canUseD3DOnScreen(peer, gc, bbNum)) {
+ try {
+ // note that the created surface will be in the "lost"
+ // state, it will be restored prior to rendering to it
+ // for the first time. This is done so that vram is not
+ // wasted for surfaces never rendered to
+ sd = D3DSurfaceData.createData(peer);
+ } catch (InvalidPipeException ipe) {
+ sd = null;
+ }
+ }
+ if (sd == null) {
+ sd = GDIWindowSurfaceData.createData(peer);
+ // note that we do not add this surface to the list of cached gdi
+ // surfaces as there's no d3dw surface to associate it with;
+ // this peer will have a gdi surface until next time a surface
+ // will need to be replaced
+ }
+
+ if (isResize) {
+ // since we'd potentially replaced the back-buffer surface
+ // (either with another bb, or a gdi one), the
+ // component will need to be completely repainted;
+ // this only need to be done when the surface is created in
+ // response to a resize event since when a component is created it
+ // will be repainted anyway
+ repaintPeerTarget(peer);
+ }
+
+ return sd;
+ }
+
+ /**
+ * Determines if we can use a d3d surface for onscreen rendering for this
+ * peer.
+ * We only create onscreen d3d surfaces if the following conditions are met:
+ * - d3d is enabled on this device and onscreen emulation is enabled
+ * - window is big enough to bother (either dimension > MIN_WIN_SIZE)
+ * - this heavyweight doesn't have a BufferStrategy
+ * - if we are in full-screen mode then it must be the peer of the
+ * full-screen window (since there could be only one SwapChain in fs)
+ * and it must not have any heavyweight children
+ * (as Present() doesn't respect component clipping in fullscreen mode)
+ * - it's one of the classes likely to have custom rendering worth
+ * accelerating
+ *
+ * @returns true if we can use a d3d surface for this peer's onscreen
+ * rendering
+ */
+ public static boolean canUseD3DOnScreen(final WComponentPeer peer,
+ final Win32GraphicsConfig gc,
+ final int bbNum)
+ {
+ if (!(gc instanceof D3DGraphicsConfig)) {
+ return false;
+ }
+ D3DGraphicsConfig d3dgc = (D3DGraphicsConfig)gc;
+ D3DGraphicsDevice d3dgd = d3dgc.getD3DDevice();
+ String peerName = peer.getClass().getName();
+ Rectangle r = peer.getBounds();
+ Component target = (Component)peer.getTarget();
+ Window fsw = d3dgd.getFullScreenWindow();
+
+ return
+ WindowsFlags.isD3DOnScreenEnabled() &&
+ d3dgd.isD3DEnabledOnDevice() &&
+ peer.isAccelCapable() &&
+ (r.width > MIN_WIN_SIZE || r.height > MIN_WIN_SIZE) &&
+ bbNum == 0 &&
+ (fsw == null || (fsw == target && !hasHWChildren(target))) &&
+ (peerName.equals("sun.awt.windows.WCanvasPeer") ||
+ peerName.equals("sun.awt.windows.WDialogPeer") ||
+ peerName.equals("sun.awt.windows.WPanelPeer") ||
+ peerName.equals("sun.awt.windows.WWindowPeer") ||
+ peerName.equals("sun.awt.windows.WFramePeer") ||
+ peerName.equals("sun.awt.windows.WEmbeddedFramePeer"));
+ }
+
+ /**
+ * Creates a graphics object for the passed in surface data. If
+ * the surface is lost, it is restored.
+ * If the surface wasn't lost or the restoration was successful
+ * the surface is added to the list of maintained surfaces
+ * (if it hasn't been already).
+ *
+ * If the updater thread hasn't been created yet , it will be created and
+ * started.
+ *
+ * @param sd surface data for which to create SunGraphics2D
+ * @param peer peer associated with the surface data
+ * @param fgColor fg color to be used in graphics
+ * @param bgColor bg color to be used in graphics
+ * @param font font to be used in graphics
+ * @return a SunGraphics2D object for the surface (or for temp GDI
+ * surface data)
+ */
+ @Override
+ public Graphics2D createGraphics(SurfaceData sd,
+ WComponentPeer peer, Color fgColor, Color bgColor, Font font)
+ {
+ if (!done && sd instanceof D3DWindowSurfaceData) {
+ D3DWindowSurfaceData d3dw = (D3DWindowSurfaceData)sd;
+ if (!d3dw.isSurfaceLost() || validate(d3dw)) {
+ trackScreenSurface(d3dw);
+ return new SunGraphics2D(sd, fgColor, bgColor, font);
+ }
+ // could not restore the d3dw surface, use the cached gdi surface
+ // instead for this graphics object; note that we do not track
+ // this new gdi surface, it is only used for this graphics
+ // object
+ sd = getGdiSurface(d3dw);
+ }
+ return super.createGraphics(sd, peer, fgColor, bgColor, font);
+ }
+
+ /**
+ * Posts a repaint event for the peer's target to the EDT
+ * @param peer for which target's the repaint should be issued
+ */
+ private void repaintPeerTarget(WComponentPeer peer) {
+ // we don't want to call user code on our priveleged
+ // thread, delegate to EDT
+ final Component target = (Component)peer.getTarget();
+ SunToolkit.executeOnEventHandlerThread(target, new Runnable() {
+ public void run() {
+ target.repaint();
+ }
+ });
+ }
+
+ /**
+ * Adds a surface to the list of tracked surfaces.
+ *
+ * @param d3dw the surface to be added
+ */
+ private void trackScreenSurface(SurfaceData sd) {
+ if (!done && sd instanceof D3DWindowSurfaceData) {
+ synchronized (this) {
+ if (d3dwSurfaces == null) {
+ d3dwSurfaces = new ArrayList();
+ }
+ D3DWindowSurfaceData d3dw = (D3DWindowSurfaceData)sd;
+ if (!d3dwSurfaces.contains(d3dw)) {
+ d3dwSurfaces.add(d3dw);
+ }
+ }
+ startUpdateThread();
+ }
+ }
+
+ @Override
+ public synchronized void dropScreenSurface(SurfaceData sd) {
+ if (d3dwSurfaces != null && sd instanceof D3DWindowSurfaceData) {
+ D3DWindowSurfaceData d3dw = (D3DWindowSurfaceData)sd;
+ removeGdiSurface(d3dw);
+ d3dwSurfaces.remove(d3dw);
+ }
+ }
+
+ @Override
+ public SurfaceData getReplacementScreenSurface(WComponentPeer peer,
+ SurfaceData sd)
+ {
+ SurfaceData newSurface = super.getReplacementScreenSurface(peer, sd);
+ // if some outstanding graphics context wants to get a replacement we
+ // need to make sure that the new surface (if it is accelerated) is
+ // being tracked
+ trackScreenSurface(newSurface);
+ return newSurface;
+ }
+
+ /**
+ * Remove the gdi surface corresponding to the passed d3dw surface
+ * from list of the cached gdi surfaces.
+ *
+ * @param d3dw surface for which associated gdi surface is to be removed
+ */
+ private void removeGdiSurface(final D3DWindowSurfaceData d3dw) {
+ if (gdiSurfaces != null) {
+ GDIWindowSurfaceData gdisd = gdiSurfaces.get(d3dw);
+ if (gdisd != null) {
+ gdisd.invalidate();
+ gdiSurfaces.remove(d3dw);
+ }
+ }
+ }
+
+ /**
+ * If the update thread hasn't yet been created, it will be;
+ * otherwise it is awaken
+ */
+ private synchronized void startUpdateThread() {
+ if (screenUpdater == null) {
+ screenUpdater = (Thread)java.security.AccessController.doPrivileged(
+ new java.security.PrivilegedAction() {
+ public Object run() {
+ ThreadGroup tg =
+ Thread.currentThread().getThreadGroup();
+ for (ThreadGroup tgn = tg;
+ tgn != null; tg = tgn, tgn = tg.getParent());
+ Thread t = new Thread(tg, D3DScreenUpdateManager.this,
+ "D3D Screen Updater");
+ // REMIND: should it be higher?
+ t.setPriority(Thread.NORM_PRIORITY + 2);
+ t.setDaemon(true);
+ return t;
+ }
+ });
+ screenUpdater.start();
+ } else {
+ wakeUpUpdateThread();
+ }
+ }
+
+ /**
+ * Wakes up the screen updater thread.
+ *
+ * This method is not synchronous, it doesn't wait
+ * for the updater thread to complete the updates.
+ *
+ * It should be used when it is not necessary to wait for the
+ * completion, for example, when a new surface had been added
+ * to the list of tracked surfaces (which means that it's about
+ * to be rendered to).
+ */
+ public void wakeUpUpdateThread() {
+ synchronized (runLock) {
+ runLock.notifyAll();
+ }
+ }
+
+ /**
+ * Wakes up the screen updater thread and waits for the completion
+ * of the update.
+ *
+ * This method is called from Toolkit.sync() or
+ * when there was a copy from a VI to the screen
+ * so that swing applications would not appear to be
+ * sluggish.
+ */
+ public void runUpdateNow() {
+ synchronized (this) {
+ // nothing to do if the updater thread hadn't been started or if
+ // there are no tracked surfaces
+ if (done || screenUpdater == null ||
+ d3dwSurfaces == null || d3dwSurfaces.size() == 0)
+ {
+ return;
+ }
+ }
+ synchronized (runLock) {
+ needsUpdateNow = true;
+ runLock.notifyAll();
+ while (needsUpdateNow) {
+ try {
+ runLock.wait();
+ } catch (InterruptedException e) {}
+ }
+ }
+ }
+
+ public void run() {
+ while (!done) {
+ synchronized (runLock) {
+ // If the list is empty, suspend the thread until a
+ // new surface is added. Note that we have to check before
+ // wait() (and inside the runLock), otherwise we could miss a
+ // notify() when a new surface is added and sleep forever.
+ long timeout = d3dwSurfaces.size() > 0 ? 100 : 0;
+
+ // don't go to sleep if there's a thread waiting for an update
+ if (!needsUpdateNow) {
+ try { runLock.wait(timeout); }
+ catch (InterruptedException e) {}
+ }
+ // if we were woken up, there are probably surfaces in the list,
+ // no need to check if the list is empty
+ }
+
+ // make a copy to avoid synchronization during the loop
+ D3DWindowSurfaceData surfaces[] = new D3DWindowSurfaceData[] {};
+ synchronized (this) {
+ surfaces = d3dwSurfaces.toArray(surfaces);
+ }
+ for (D3DWindowSurfaceData sd : surfaces) {
+ // skip invalid surfaces (they could have become invalid
+ // after we made a copy of the list) - just a precaution
+ if (sd.isValid() && (sd.isDirty() || sd.isSurfaceLost())) {
+ if (!sd.isSurfaceLost()) {
+ // the flip and the clearing of the dirty state
+ // must be done under the lock, otherwise it's
+ // possible to miss an update to the surface
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ Rectangle r = sd.getBounds();
+ D3DSurfaceData.swapBuffers(sd, 0, 0,
+ r.width, r.height);
+ sd.markClean();
+ } finally {
+ rq.unlock();
+ }
+ } else if (!validate(sd)) {
+ // it is possible that the validation may never
+ // succeed, we need to detect this and replace
+ // the d3dw surface with gdi; the replacement of
+ // the surface will also trigger a repaint
+ sd.getPeer().replaceSurfaceDataLater();
+ }
+ }
+ }
+ synchronized (runLock) {
+ needsUpdateNow = false;
+ runLock.notifyAll();
+ }
+ }
+ }
+
+ /**
+ * Restores the passed surface if it was lost, resets the lost status.
+ * @param sd surface to be validated
+ * @return true if surface wasn't lost or if restoration was successful,
+ * false otherwise
+ */
+ private boolean validate(D3DWindowSurfaceData sd) {
+ if (sd.isSurfaceLost()) {
+ try {
+ sd.restoreSurface();
+ // if succeeded, first fill the surface with bg color
+ // note: use the non-synch method to avoid incorrect lock order
+ Color bg = sd.getPeer().getBackgroundNoSync();
+ SunGraphics2D sg2d = new SunGraphics2D(sd, bg, bg, null);
+ sg2d.fillRect(0, 0, sd.getBounds().width, sd.getBounds().height);
+ sg2d.dispose();
+ // now clean the dirty status so that we don't flip it
+ // next time before it gets repainted; it is safe
+ // to do without the lock because we will issue a
+ // repaint anyway so we will not lose any rendering
+ sd.markClean();
+ // since the surface was successfully restored we need to
+ // repaint whole window to repopulate the back-buffer
+ repaintPeerTarget(sd.getPeer());
+ } catch (InvalidPipeException ipe) {
+ return false;
+ }
+ }
+ return true;
+ }
+
+ /**
+ * Creates (or returns a cached one) gdi surface for the same peer as
+ * the passed d3dw surface has.
+ *
+ * @param d3dw surface used as key into the cache
+ * @return gdi window surface associated with the d3d window surfaces' peer
+ */
+ private synchronized SurfaceData getGdiSurface(D3DWindowSurfaceData d3dw) {
+ if (gdiSurfaces == null) {
+ gdiSurfaces =
+ new HashMap();
+ }
+ GDIWindowSurfaceData gdisd = gdiSurfaces.get(d3dw);
+ if (gdisd == null) {
+ gdisd = GDIWindowSurfaceData.createData(d3dw.getPeer());
+ gdiSurfaces.put(d3dw, gdisd);
+ }
+ return gdisd;
+ }
+
+ /**
+ * Returns true if the component has heavyweight children.
+ *
+ * @param comp component to check for hw children
+ * @return true if Component has heavyweight children
+ */
+ private static boolean hasHWChildren(Component comp) {
+ if (comp instanceof Container) {
+ for (Component c : ((Container)comp).getComponents()) {
+ if (c.getPeer() instanceof WComponentPeer || hasHWChildren(c)) {
+ return true;
+ }
+ }
+ }
+ return false;
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceData.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceData.java
index 2dd7a183b95..3d1457c721f 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceData.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceData.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -26,432 +26,919 @@
package sun.java2d.d3d;
import java.awt.AlphaComposite;
+import java.awt.BufferCapabilities;
+import java.awt.Component;
import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
import java.awt.Image;
+import java.awt.Rectangle;
import java.awt.Transparency;
import java.awt.image.ColorModel;
+import java.awt.image.DataBuffer;
import java.awt.image.DirectColorModel;
+import java.awt.image.Raster;
+import java.awt.image.SampleModel;
+import java.awt.image.SinglePixelPackedSampleModel;
import sun.awt.SunHints;
-import sun.awt.Win32GraphicsConfig;
-import sun.awt.Win32GraphicsDevice;
+import sun.awt.image.DataBufferNative;
+import sun.awt.image.PixelConverter;
import sun.awt.image.SurfaceManager;
+import sun.awt.image.WritableRasterNative;
+import sun.awt.windows.WComponentPeer;
+import sun.java2d.pipe.hw.AccelSurface;
import sun.java2d.InvalidPipeException;
import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
-import sun.java2d.SurfaceDataProxy;
import sun.java2d.loops.GraphicsPrimitive;
+import sun.java2d.loops.MaskFill;
import sun.java2d.loops.SurfaceType;
-import sun.java2d.pipe.PixelToShapeConverter;
+import sun.java2d.loops.CompositeType;
+import sun.java2d.pipe.ParallelogramPipe;
+import sun.java2d.pipe.PixelToParallelogramConverter;
+import sun.java2d.pipe.RenderBuffer;
import sun.java2d.pipe.TextPipe;
-import sun.java2d.windows.Win32OffScreenSurfaceData;
-import sun.java2d.windows.Win32SurfaceData;
-import sun.java2d.windows.WinVolatileSurfaceManager;
-import sun.java2d.windows.WindowsFlags;
+import static sun.java2d.pipe.BufferedOpCodes.*;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
+import static sun.java2d.pipe.hw.ExtendedBufferCapabilities.VSyncType.*;
+import sun.java2d.pipe.hw.ExtendedBufferCapabilities.VSyncType;
+import java.awt.BufferCapabilities.FlipContents;
+import java.awt.Window;
+import sun.awt.SunToolkit;
+import sun.awt.image.SunVolatileImage;
+import sun.java2d.ScreenUpdateManager;
+import sun.java2d.StateTracker;
+import sun.java2d.SurfaceDataProxy;
+import sun.java2d.pipe.hw.ExtendedBufferCapabilities;
-import static sun.java2d.windows.Win32SurfaceData.*;
-
-public class D3DSurfaceData extends Win32OffScreenSurfaceData {
-
- // properties of a surface
- /**
- * This property is used for a back-buffer surface
- */
- public static final int D3D_ATTACHED_SURFACE = (1 << 15);
- /**
- * A surface with this property can be used as a Direct3D rendering
- * destination.
- */
- public static final int D3D_RENDER_TARGET = (1 << 16);
-
- public static final int
- D3D_INVALID_SURFACE = 0;
- /**
- * Surface is a Direct3D plain surface (not a texture).
- * Plain surface can be used as render target.
- * VolatileImages typically use plain surfaces as their hardware
- * accelerated surfaces.
- */
- public static final int
- D3D_PLAIN_SURFACE = (1 << 0) | D3D_RENDER_TARGET;
- /**
- * Direct3D texture. Mostly used for cached accelerated surfaces.
- * Surfaces of this type can be copied from using hardware acceleration
- * by using texture mapping.
- */
- public static final int
- D3D_TEXTURE_SURFACE = (1 << 1);
- /**
- * Direct3D Backbuffer surface - an attached surface. Used for
- * multibuffered BufferStrategies.
- */
- public static final int
- D3D_BACKBUFFER_SURFACE = D3D_PLAIN_SURFACE | D3D_ATTACHED_SURFACE;
- /**
- * Render-to-texture. A texture which can also be a render target.
- * Combines the benefits of textures (fast copies-from) and
- * backbuffers or plain surfaces (hw-accelerated rendering to the surface)
- */
- public static final int
- D3D_RTT_SURFACE = D3D_TEXTURE_SURFACE | D3D_RENDER_TARGET;
-
- // supported texture pixel formats
- public static final int PF_INVALID = 0;
- public static final int PF_INT_ARGB = 1;
- public static final int PF_INT_RGB = 2;
- public static final int PF_INT_RGBX = 3;
- public static final int PF_INT_BGR = 4;
- public static final int PF_USHORT_565_RGB = 5;
- public static final int PF_USHORT_555_RGB = 6;
- public static final int PF_USHORT_555_RGBX = 7;
- public static final int PF_INT_ARGB_PRE = 8;
- public static final int PF_USHORT_4444_ARGB= 9;
-
- public static final String
- DESC_INT_ARGB_D3D = "Integer ARGB D3D with translucency";
- public static final String
- DESC_USHORT_4444_ARGB_D3D = "UShort 4444 ARGB D3D with translucency";
+/**
+ * This class describes a D3D "surface", that is, a region of pixels
+ * managed via D3D. An D3DSurfaceData can be tagged with one of three
+ * different SurfaceType objects for the purpose of registering loops, etc.
+ * This diagram shows the hierarchy of D3D SurfaceTypes:
+ *
+ * Any
+ * / \
+ * D3DSurface D3DTexture
+ * |
+ * D3DSurfaceRTT
+ *
+ * D3DSurface
+ * This kind of surface can be rendered to using D3D APIs. It is also
+ * possible to copy a D3DSurface to another D3DSurface (or to itself).
+ *
+ * D3DTexture
+ * This kind of surface cannot be rendered to using D3D (in the same sense
+ * as in D3DSurface). However, it is possible to upload a region of pixels
+ * to a D3DTexture object via Lock/UnlockRect(). One can also copy a
+ * surface of type D3DTexture to a D3DSurface by binding the texture
+ * to a quad and then rendering it to the destination surface (this process
+ * is known as "texture mapping").
+ *
+ * D3DSurfaceRTT
+ * This kind of surface can be thought of as a sort of hybrid between
+ * D3DSurface and D3DTexture, in that one can render to this kind of
+ * surface as if it were of type D3DSurface, but the process of copying
+ * this kind of surface to another is more like a D3DTexture. (Note that
+ * "RTT" stands for "render-to-texture".)
+ *
+ * In addition to these SurfaceType variants, we have also defined some
+ * constants that describe in more detail the type of underlying D3D
+ * surface. This table helps explain the relationships between those
+ * "type" constants and their corresponding SurfaceType:
+ *
+ * D3D Type Corresponding SurfaceType
+ * -------- -------------------------
+ * RT_PLAIN D3DSurface
+ * TEXTURE D3DTexture
+ * FLIP_BACKBUFFER D3DSurface
+ * RT_TEXTURE D3DSurfaceRTT
+ */
+public class D3DSurfaceData extends SurfaceData implements AccelSurface {
/**
- * Surface type for texture destination. We cannot render textures to
- * the screen because Direct3D is not clipped by the window's clip list,
- * so we only enable the texture blit loops for copies to offscreen
- * accelerated surfaces.
+ * To be used with getNativeResource() only.
+ * @see #getNativeResource()
*/
- public static final String
- DESC_DEST_D3D = "D3D render target";
-
- public static final SurfaceType D3DSurface =
- SurfaceType.Any.deriveSubType("Direct3D Surface");
- public static final SurfaceType D3DTexture =
- D3DSurface.deriveSubType("Direct3D Texture");
-
- /**
- * D3D destination surface types (derive from offscreen dd surfaces).
- * Note that all of these surfaces have the same surface description;
- * we do not care about the depth of the surface since texture ops
- * support multiple depths.
+ public static final int D3D_DEVICE_RESOURCE= 100;
+ /*
+ * Surface types.
+ * We use these surface types when copying from a sw surface
+ * to a surface or texture.
*/
- public static final SurfaceType IntRgbD3D =
- IntRgbDD.deriveSubType(DESC_DEST_D3D);
+ public static final int ST_INT_ARGB = 0;
+ public static final int ST_INT_ARGB_PRE = 1;
+ public static final int ST_INT_ARGB_BM = 2;
+ public static final int ST_INT_RGB = 3;
+ public static final int ST_INT_BGR = 4;
+ public static final int ST_USHORT_565_RGB = 5;
+ public static final int ST_USHORT_555_RGB = 6;
+ public static final int ST_BYTE_INDEXED = 7;
+ public static final int ST_BYTE_INDEXED_BM = 8;
- public static final SurfaceType IntRgbxD3D =
- IntRgbxDD.deriveSubType(DESC_DEST_D3D);
+ /** Equals to D3DSWAPEFFECT_DISCARD */
+ public static final int SWAP_DISCARD = 1;
+ /** Equals to D3DSWAPEFFECT_FLIP */
+ public static final int SWAP_FLIP = 2;
+ /** Equals to D3DSWAPEFFECT_COPY */
+ public static final int SWAP_COPY = 3;
+ /*
+ * SurfaceTypes
+ */
+ private static final String DESC_D3D_SURFACE = "D3D Surface";
+ private static final String DESC_D3D_SURFACE_RTT =
+ "D3D Surface (render-to-texture)";
+ private static final String DESC_D3D_TEXTURE = "D3D Texture";
- public static final SurfaceType Ushort565RgbD3D =
- Ushort565RgbDD.deriveSubType(DESC_DEST_D3D);
+ // REMIND: regarding ArgbPre??
+ static final SurfaceType D3DSurface =
+ SurfaceType.Any.deriveSubType(DESC_D3D_SURFACE,
+ PixelConverter.ArgbPre.instance);
+ static final SurfaceType D3DSurfaceRTT =
+ D3DSurface.deriveSubType(DESC_D3D_SURFACE_RTT);
+ static final SurfaceType D3DTexture =
+ SurfaceType.Any.deriveSubType(DESC_D3D_TEXTURE);
- public static final SurfaceType Ushort555RgbxD3D =
- Ushort555RgbxDD.deriveSubType(DESC_DEST_D3D);
+ private int type;
+ private int width, height;
+ // these fields are set from the native code when the surface is
+ // initialized
+ private int nativeWidth, nativeHeight;
+ protected WComponentPeer peer;
+ private Image offscreenImage;
+ protected D3DGraphicsDevice graphicsDevice;
- public static final SurfaceType Ushort555RgbD3D =
- Ushort555RgbDD.deriveSubType(DESC_DEST_D3D);
+ private int swapEffect;
+ private VSyncType syncType;
+ private int backBuffersNum;
- // REMIND: Is it possible to have d3d accelerated on this type of surface?
- public static final SurfaceType ThreeByteBgrD3D =
- ThreeByteBgrDD.deriveSubType(DESC_DEST_D3D);
+ private WritableRasterNative wrn;
- public static final SurfaceType IntArgbD3D =
- SurfaceType.IntArgb.deriveSubType(DESC_INT_ARGB_D3D);
-
- public static final SurfaceType Ushort4444ArgbD3D =
- SurfaceType.Ushort4444Argb.deriveSubType(DESC_USHORT_4444_ARGB_D3D);
-
- // Textures we can render to using d3d
- public static final SurfaceType IntRgbD3D_RTT =
- IntRgbD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType IntRgbxD3D_RTT =
- IntRgbxD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType Ushort565RgbD3D_RTT =
- Ushort565RgbD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType Ushort555RgbxD3D_RTT =
- Ushort555RgbxD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType Ushort555RgbD3D_RTT =
- Ushort555RgbD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType Ushort4444ArgbD3D_RTT =
- Ushort4444ArgbD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType IntArgbD3D_RTT =
- IntArgbD3D.deriveSubType(DESC_DEST_D3D);
-
- public static final SurfaceType ThreeByteBgrD3D_RTT =
- ThreeByteBgrD3D.deriveSubType(DESC_DEST_D3D);
-
- // the type of this surface - texture, plain, back-buffer
- protected int type;
- protected int pixelFormat;
-
- private D3DContext d3dContext;
-
- protected static D3DRenderer d3dPipe;
- protected static PixelToShapeConverter d3dTxPipe;
+ protected static D3DRenderer d3dRenderPipe;
+ protected static PixelToParallelogramConverter d3dTxRenderPipe;
+ protected static ParallelogramPipe d3dAAPgramPipe;
protected static D3DTextRenderer d3dTextPipe;
- protected static D3DDrawImage d3dDrawImagePipe;
+ protected static D3DDrawImage d3dImagePipe;
- private native void initOps(int depth, int transparency);
+ private native boolean initTexture(long pData, boolean isRTT,
+ boolean isOpaque);
+ private native boolean initFlipBackbuffer(long pData, long pPeerData,
+ int numbuffers,
+ int swapEffect, int syncType);
+ private native boolean initRTSurface(long pData, boolean isOpaque);
+ private native void initOps(int screen, int width, int height);
static {
- if (WindowsFlags.isD3DEnabled()) {
- D3DBlitLoops.register();
- D3DMaskFill.register();
- }
-
- d3dPipe = new D3DRenderer();
- d3dTxPipe = new PixelToShapeConverter(d3dPipe);
- d3dTextPipe = new D3DTextRenderer();
- d3dDrawImagePipe = new D3DDrawImage();
-
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ d3dImagePipe = new D3DDrawImage();
+ d3dTextPipe = new D3DTextRenderer(rq);
+ d3dRenderPipe = new D3DRenderer(rq);
if (GraphicsPrimitive.tracingEnabled()) {
- d3dPipe = d3dPipe.traceWrapD3D();
d3dTextPipe = d3dTextPipe.traceWrap();
+ d3dRenderPipe = d3dRenderPipe.traceWrap();
+ //The wrapped d3dRenderPipe will wrap the AA pipe as well...
+ //d3dAAPgramPipe = d3dRenderPipe.traceWrap();
}
+ d3dAAPgramPipe = d3dRenderPipe.getAAParallelogramPipe();
+ d3dTxRenderPipe =
+ new PixelToParallelogramConverter(d3dRenderPipe, d3dRenderPipe,
+ 1.0, 0.25, true);
+
+ D3DBlitLoops.register();
+ D3DMaskFill.register();
+ D3DMaskBlit.register();
+ }
+
+ protected D3DSurfaceData(WComponentPeer peer, D3DGraphicsConfig gc,
+ int width, int height, Image image,
+ ColorModel cm, int numBackBuffers,
+ int swapEffect, VSyncType vSyncType,
+ int type)
+ {
+ super(getCustomSurfaceType(type), cm);
+ this.graphicsDevice = gc.getD3DDevice();
+ this.peer = peer;
+ this.type = type;
+ this.width = width;
+ this.height = height;
+ this.offscreenImage = image;
+ this.backBuffersNum = numBackBuffers;
+ this.swapEffect = swapEffect;
+ this.syncType = vSyncType;
+
+ initOps(graphicsDevice.getScreen(), width, height);
+ if (type == WINDOW) {
+ // we put the surface into the "lost"
+ // state; it will be restored by the D3DScreenUpdateManager
+ // prior to rendering to it for the first time. This is done
+ // so that vram is not wasted for surfaces never rendered to
+ setSurfaceLost(true);
+ } else {
+ initSurface();
+ }
+ setBlitProxyKey(gc.getProxyKey());
}
@Override
public SurfaceDataProxy makeProxyFor(SurfaceData srcData) {
- //D3D may be eliminated soon so no Proxy was created for it...
- //return D3DSurfaceDataProxy.createProxy(srcData, graphicsConfig);
- return SurfaceDataProxy.UNCACHED;
+ return D3DSurfaceDataProxy.
+ createProxy(srcData,
+ (D3DGraphicsConfig)graphicsDevice.getDefaultConfiguration());
}
/**
- * Non-public constructor. Use createData() to create an object.
- *
- * This constructor is used to house the common construction
- * code shared between the creation of D3DSurfaceData objects
- * and subclasses of D3DSurfaceData (such as D3DBackBufferSD).
- *
- * It calls the common constructor in the parent, and then
- * initializes other shared D3D data.
+ * Creates a SurfaceData object representing the back buffer of a
+ * double-buffered on-screen Window.
*/
- protected D3DSurfaceData(int width, int height,
- int d3dSurfaceType,
- SurfaceType sType, ColorModel cm,
- GraphicsConfiguration gc,
- Image image, int transparency)
- {
- super(width, height, sType, cm, gc, image, transparency);
- this.type = d3dSurfaceType;
- }
-
- /**
- * Private constructor. Use createData() to create an object.
- *
- * This constructor calls the common constructor above and then
- * performs the specific initialization of the D3DSurface.
- */
- private D3DSurfaceData(int width, int height,
- int d3dSurfaceType,
- SurfaceType sType, ColorModel cm,
- GraphicsConfiguration gc,
- Image image, int transparency,
- int screen)
- {
- this(width, height, d3dSurfaceType, sType, cm, gc, image, transparency);
- pixelFormat = initSurface(width, height, screen,
- null /*parent SurfaceData*/);
- }
-
- public static D3DSurfaceData createData(int width, int height,
- int d3dSurfaceType,
- ColorModel cm,
- GraphicsConfiguration gc,
- Image image)
- {
- Win32GraphicsDevice gd = (Win32GraphicsDevice)gc.getDevice();
- // After a display change ddInstance may not be
- // recreated yet, and in this case isD3DEnabledOnDevice will
- // return false, until someone attempted to recreate the
- // primary.
- if (!gd.isD3DEnabledOnDevice()) {
+ public static D3DSurfaceData createData(WComponentPeer peer, Image image) {
+ D3DGraphicsConfig gc = getGC(peer);
+ if (gc == null || !peer.isAccelCapable()) {
return null;
}
-
- return new D3DSurfaceData(width, height,
- d3dSurfaceType,
- getSurfaceType(gc, cm, d3dSurfaceType),
- cm, gc, image,
- cm.getTransparency(), gd.getScreen());
- }
-
- int getPixelFormat() {
- return pixelFormat;
- }
-
- static SurfaceType getSurfaceType(GraphicsConfiguration gc,
- ColorModel cm,
- int d3dSurfaceType)
- {
- if (d3dSurfaceType == D3D_TEXTURE_SURFACE) {
- // for non-rtt textures we have only one surface type
- return D3DTexture;
- } else {
- int pixelSize = cm.getPixelSize();
- Win32GraphicsDevice gd = (Win32GraphicsDevice)gc.getDevice();
- int transparency = cm.getTransparency();
-
- // We'll attempt to use render-to-texture if render target is
- // requested, but it's not a back-buffer and we support RTT
- // for this configuration.
- boolean useRTT =
- ((d3dSurfaceType & D3D_RENDER_TARGET) != 0) &&
- ((d3dSurfaceType & D3D_BACKBUFFER_SURFACE) == 0) &&
- gd.getD3DContext().isRTTSupported();
-
- // if there's no RTT available, we can't accelerate non-opaque
- // surfaces, so we return null.
- if (transparency == Transparency.TRANSLUCENT ||
- transparency == Transparency.BITMASK)
- {
- if (pixelSize == 16) {
- return useRTT ? Ushort4444ArgbD3D_RTT :
- null/*Ushort4444ArgbD3D*/;
- } else {
- return useRTT ? IntArgbD3D_RTT : null/*IntArgbD3D*/;
- }
- } else {
- // it's an opaque surface, either a VI or a back-buffer
- switch (pixelSize) {
- case 32:
- case 24:
- if (cm instanceof DirectColorModel) {
- if (((DirectColorModel)cm).getRedMask() == 0xff0000) {
- return useRTT ? IntRgbD3D_RTT : IntRgbD3D;
- } else {
- return useRTT ? IntRgbxD3D_RTT : IntRgbxD3D;
- }
- } else {
- return useRTT ? ThreeByteBgrD3D_RTT : ThreeByteBgrD3D;
- }
- case 15:
- return useRTT ? Ushort555RgbD3D_RTT : Ushort555RgbD3D;
- case 16:
- if ((cm instanceof DirectColorModel) &&
- (((DirectColorModel)cm).getBlueMask() == 0x3e))
- {
- return useRTT ? Ushort555RgbxD3D_RTT : Ushort555RgbxD3D;
- } else {
- return useRTT ? Ushort565RgbD3D_RTT : Ushort565RgbD3D;
- }
- case 8: // not supported
- default:
- throw new sun.java2d.InvalidPipeException("Unsupported bit " +
- "depth: " +
- cm.getPixelSize());
- }
- }
+ BufferCapabilities caps = peer.getBackBufferCaps();
+ VSyncType vSyncType = VSYNC_DEFAULT;
+ if (caps instanceof ExtendedBufferCapabilities) {
+ vSyncType = ((ExtendedBufferCapabilities)caps).getVSync();
}
- }
-
- private native int initOffScreenSurface(long pCtx,
- long pData, long parentPdata,
- int width, int height,
- int type, int screen);
-
- protected int initSurface(int width, int height, int screen,
- Win32SurfaceData parentData)
- {
- int pFormat = PF_INVALID;
-
- synchronized (D3DContext.LOCK) {
- long pData = getNativeOps();
- long pDataParent = 0L;
- if (parentData != null) {
- pDataParent = parentData.getNativeOps();
- }
- D3DContext d3dContext = getContext();
- long pCtx = d3dContext.getNativeContext();
- // native context could be 0 if the context is currently invalid,
- // so attempt to revalidate
- if (pCtx == 0) {
- d3dContext.reinitNativeContext();
- pCtx = d3dContext.getNativeContext();
- }
- if (pData != 0 && pCtx != 0) {
- pFormat = initOffScreenSurface(pCtx,
- pData, pDataParent,
- width, height, type, screen);
- } else {
- // if the context can't be restored, give up for now.
- throw new InvalidPipeException("D3DSD.initSurface: pData " +
- "or pCtx is null");
- }
+ Rectangle r = peer.getBounds();
+ BufferCapabilities.FlipContents flip = caps.getFlipContents();
+ int swapEffect;
+ if (flip == FlipContents.COPIED) {
+ swapEffect = SWAP_COPY;
+ } else if (flip == FlipContents.PRIOR) {
+ swapEffect = SWAP_FLIP;
+ } else { // flip == FlipContents.UNDEFINED || .BACKGROUND
+ swapEffect = SWAP_DISCARD;
}
- return pFormat;
- }
-
- @Override
- public void validatePipe(SunGraphics2D sg2d) {
- // we don't support COMP_XOR yet..
- if (sg2d.compositeState < sg2d.COMP_XOR) {
- TextPipe textpipe;
- boolean validated = false;
-
- if (((sg2d.compositeState <= sg2d.COMP_ISCOPY &&
- sg2d.paintState <= sg2d.PAINT_ALPHACOLOR) ||
- (sg2d.compositeState == sg2d.COMP_ALPHA &&
- sg2d.paintState <= sg2d.PAINT_ALPHACOLOR &&
- (((AlphaComposite)sg2d.composite).getRule() ==
- AlphaComposite.SRC_OVER))) &&
- sg2d.textAntialiasHint <= SunHints.INTVAL_TEXT_ANTIALIAS_GASP)
- {
- // D3DTextRenderer handles both AA and non-AA text, but
- // only works if composite is SrcNoEa or SrcOver
- textpipe = d3dTextPipe;
- } else {
- // do this to initialize textpipe correctly; we will attempt
- // to override the non-text pipes below
- super.validatePipe(sg2d);
- textpipe = sg2d.textpipe;
- validated = true;
- }
-
- if (sg2d.antialiasHint != SunHints.INTVAL_ANTIALIAS_ON &&
- sg2d.paintState <= sg2d.PAINT_ALPHACOLOR)
- {
- sg2d.drawpipe =
- sg2d.strokeState == sg2d.STROKE_THIN ? d3dPipe : d3dTxPipe;
- sg2d.fillpipe = d3dPipe;
- sg2d.shapepipe = d3dPipe;
- } else if (!validated) {
- super.validatePipe(sg2d);
- }
- // install the text pipe based on our earlier decision
- sg2d.textpipe = textpipe;
- } else {
- super.validatePipe(sg2d);
- }
-
- // always override the image pipe with the specialized D3D pipe
- sg2d.imagepipe = d3dDrawImagePipe;
+ return new D3DSurfaceData(peer, gc, r.width, r.height,
+ image, peer.getColorModel(),
+ peer.getBackBuffersNum(),
+ swapEffect, vSyncType, FLIP_BACKBUFFER);
}
/**
- * Disables D3D acceleration on the surface manager of this surfaceData
- * object. This can happen when we encounter a hard error in rendering a D3D
- * primitive (for example, if we were unable to set a surface as D3D target
- * surface).
- * Upon next validation the SurfaceManager will create a non-D3D surface.
+ * Returns a WINDOW type of surface - a
+ * swap chain which serves as an on-screen surface,
+ * handled by the D3DScreenUpdateManager.
+ *
+ * Note that the native surface is not initialized
+ * when the surface is created to avoid using excessive
+ * resources, and the surface is placed into the lost
+ * state. It will be restored prior to any rendering
+ * to it.
+ *
+ * @param peer peer for which the onscreen surface is to be created
+ * @return a D3DWindowSurfaceData (flip chain) surface
*/
- public void disableD3D() {
- markSurfaceLost();
- SurfaceManager sMgr = SurfaceManager.getManager(image);
- if (sMgr instanceof WinVolatileSurfaceManager) {
- ((WinVolatileSurfaceManager)sMgr).setD3DAccelerationEnabled(false);
+ public static D3DSurfaceData createData(WComponentPeer peer) {
+ D3DGraphicsConfig gc = getGC(peer);
+ if (gc == null || !peer.isAccelCapable()) {
+ return null;
+ }
+ return new D3DWindowSurfaceData(peer, gc);
+ }
+
+ /**
+ * Creates a SurfaceData object representing an off-screen buffer (either
+ * a plain surface or Texture).
+ */
+ public static D3DSurfaceData createData(D3DGraphicsConfig gc,
+ int width, int height,
+ ColorModel cm,
+ Image image, int type)
+ {
+ if (type == RT_TEXTURE) {
+ boolean isOpaque = cm.getTransparency() == Transparency.OPAQUE;
+ int cap = isOpaque ? CAPS_RT_TEXTURE_OPAQUE : CAPS_RT_TEXTURE_ALPHA;
+ if (!gc.getD3DDevice().isCapPresent(cap)) {
+ type = RT_PLAIN;
+ }
+ }
+ D3DSurfaceData ret = null;
+ try {
+ ret = new D3DSurfaceData(null, gc, width, height,
+ image, cm, 0, SWAP_DISCARD, VSYNC_DEFAULT,
+ type);
+ } catch (InvalidPipeException ipe) {
+ // try again - we might have ran out of vram, and rt textures
+ // could take up more than a plain surface, so it might succeed
+ if (type == RT_TEXTURE) {
+ // If a RT_TEXTURE was requested do not attempt to create a
+ // plain surface. (note that RT_TEXTURE can only be requested
+ // from a VI so the cast is safe)
+ if (((SunVolatileImage)image).getForcedAccelSurfaceType() !=
+ RT_TEXTURE)
+ {
+ type = RT_PLAIN;
+ ret = new D3DSurfaceData(null, gc, width, height,
+ image, cm, 0, SWAP_DISCARD,
+ VSYNC_DEFAULT, type);
+ }
+ }
+ }
+ return ret;
+ }
+
+ /**
+ * Returns the appropriate SurfaceType corresponding to the given D3D
+ * surface type constant (e.g. TEXTURE -> D3DTexture).
+ */
+ private static SurfaceType getCustomSurfaceType(int d3dType) {
+ switch (d3dType) {
+ case TEXTURE:
+ return D3DTexture;
+ case RT_TEXTURE:
+ return D3DSurfaceRTT;
+ default:
+ return D3DSurface;
}
}
+ private boolean initSurfaceNow() {
+ boolean isOpaque = (getTransparency() == Transparency.OPAQUE);
+ switch (type) {
+ case RT_PLAIN:
+ return initRTSurface(getNativeOps(), isOpaque);
+ case TEXTURE:
+ return initTexture(getNativeOps(), false/*isRTT*/, isOpaque);
+ case RT_TEXTURE:
+ return initTexture(getNativeOps(), true/*isRTT*/, isOpaque);
+ // REMIND: we may want to pass the exact type to the native
+ // level here so that we could choose the right presentation
+ // interval for the frontbuffer (immediate vs v-synced)
+ case WINDOW:
+ case FLIP_BACKBUFFER:
+ return initFlipBackbuffer(getNativeOps(), peer.getData(),
+ backBuffersNum, swapEffect,
+ syncType.id());
+ default:
+ return false;
+ }
+ }
+
+ /**
+ * Initializes the appropriate D3D offscreen surface based on the value
+ * of the type parameter. If the surface creation fails for any reason,
+ * an OutOfMemoryError will be thrown.
+ */
+ protected void initSurface() {
+ // any time we create or restore the surface, recreate the raster
+ synchronized (this) {
+ wrn = null;
+ }
+ // REMIND: somewhere a puppy died
+ class Status {
+ boolean success = false;
+ };
+ final Status status = new Status();
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ status.success = initSurfaceNow();
+ }
+ });
+ if (!status.success) {
+ throw new InvalidPipeException("Error creating D3DSurface");
+ }
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ /**
+ * Returns the D3DContext for the GraphicsConfig associated with this
+ * surface.
+ */
+ public final D3DContext getContext() {
+ return graphicsDevice.getContext();
+ }
+
+ /**
+ * Returns one of the surface type constants defined above.
+ */
+ public final int getType() {
+ return type;
+ }
+
+ private static native int dbGetPixelNative(long pData, int x, int y);
+ private static native void dbSetPixelNative(long pData, int x, int y,
+ int pixel);
+ static class D3DDataBufferNative extends DataBufferNative {
+ int pixel;
+ protected D3DDataBufferNative(SurfaceData sData,
+ int type, int w, int h)
+ {
+ super(sData, type, w, h);
+ }
+
+ protected int getElem(final int x, final int y,
+ final SurfaceData sData)
+ {
+ int retPixel;
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ pixel = dbGetPixelNative(sData.getNativeOps(), x, y);
+ }
+ });
+ } finally {
+ retPixel = pixel;
+ rq.unlock();
+ }
+ return retPixel;
+ }
+
+ protected void setElem(final int x, final int y, final int pixel,
+ final SurfaceData sData)
+ {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ dbSetPixelNative(sData.getNativeOps(), x, y, pixel);
+ }
+ });
+ sData.markDirty();
+ } finally {
+ rq.unlock();
+ }
+ }
+ }
+
+ public synchronized Raster getRaster(int x, int y, int w, int h) {
+ if (wrn == null) {
+ DirectColorModel dcm = (DirectColorModel)getColorModel();
+ SampleModel smHw;
+ int dataType = 0;
+ int scanStride = width;
+
+ if (dcm.getPixelSize() == 24 || dcm.getPixelSize() == 32) {
+ dataType = DataBuffer.TYPE_INT;
+ } else {
+ // 15, 16
+ dataType = DataBuffer.TYPE_USHORT;
+ }
+
+ // note that we have to use the surface width and height here,
+ // not the passed w,h
+ smHw = new SinglePixelPackedSampleModel(dataType, width, height,
+ scanStride, dcm.getMasks());
+ DataBuffer dbn = new D3DDataBufferNative(this, dataType,
+ width, height);
+ wrn = WritableRasterNative.createNativeRaster(smHw, dbn);
+ }
+
+ return wrn;
+ }
+
+ /**
+ * For now, we can only render LCD text if:
+ * - the pixel shaders are available, and
+ * - blending is disabled, and
+ * - the source color is opaque
+ */
+ public boolean canRenderLCDText(SunGraphics2D sg2d) {
+ return
+ graphicsDevice.isCapPresent(CAPS_LCD_SHADER) &&
+ sg2d.compositeState <= SunGraphics2D.COMP_ISCOPY &&
+ sg2d.paintState <= SunGraphics2D.PAINT_OPAQUECOLOR;
+ }
+
+ public void validatePipe(SunGraphics2D sg2d) {
+ TextPipe textpipe;
+ boolean validated = false;
+
+ // REMIND: the D3D pipeline doesn't support XOR!, more
+ // fixes will be needed below
+ if (sg2d.compositeState >= sg2d.COMP_XOR) {
+ super.validatePipe(sg2d);
+ sg2d.imagepipe = d3dImagePipe;
+ return;
+ }
+
+ // D3DTextRenderer handles both AA and non-AA text, but
+ // only works with the following modes:
+ // (Note: For LCD text we only enter this code path if
+ // canRenderLCDText() has already validated that the mode is
+ // CompositeType.SrcNoEa (opaque color), which will be subsumed
+ // by the CompositeType.SrcNoEa (any color) test below.)
+
+ if (/* CompositeType.SrcNoEa (any color) */
+ (sg2d.compositeState <= sg2d.COMP_ISCOPY &&
+ sg2d.paintState <= sg2d.PAINT_ALPHACOLOR) ||
+
+ /* CompositeType.SrcOver (any color) */
+ (sg2d.compositeState == sg2d.COMP_ALPHA &&
+ sg2d.paintState <= sg2d.PAINT_ALPHACOLOR &&
+ (((AlphaComposite)sg2d.composite).getRule() ==
+ AlphaComposite.SRC_OVER)) ||
+
+ /* CompositeType.Xor (any color) */
+ (sg2d.compositeState == sg2d.COMP_XOR &&
+ sg2d.paintState <= sg2d.PAINT_ALPHACOLOR))
+ {
+ textpipe = d3dTextPipe;
+ } else {
+ // do this to initialize textpipe correctly; we will attempt
+ // to override the non-text pipes below
+ super.validatePipe(sg2d);
+ textpipe = sg2d.textpipe;
+ validated = true;
+ }
+
+ PixelToParallelogramConverter txPipe = null;
+ D3DRenderer nonTxPipe = null;
+
+ if (sg2d.antialiasHint != SunHints.INTVAL_ANTIALIAS_ON) {
+ if (sg2d.paintState <= sg2d.PAINT_ALPHACOLOR) {
+ if (sg2d.compositeState <= sg2d.COMP_XOR) {
+ txPipe = d3dTxRenderPipe;
+ nonTxPipe = d3dRenderPipe;
+ }
+ } else if (sg2d.compositeState <= sg2d.COMP_ALPHA) {
+ if (D3DPaints.isValid(sg2d)) {
+ txPipe = d3dTxRenderPipe;
+ nonTxPipe = d3dRenderPipe;
+ }
+ // custom paints handled by super.validatePipe() below
+ }
+ } else {
+ if (sg2d.paintState <= sg2d.PAINT_ALPHACOLOR) {
+ if (graphicsDevice.isCapPresent(CAPS_AA_SHADER) &&
+ (sg2d.imageComp == CompositeType.SrcOverNoEa ||
+ sg2d.imageComp == CompositeType.SrcOver))
+ {
+ if (!validated) {
+ super.validatePipe(sg2d);
+ validated = true;
+ }
+ PixelToParallelogramConverter aaConverter =
+ new PixelToParallelogramConverter(sg2d.shapepipe,
+ d3dAAPgramPipe,
+ 1.0/8.0, 0.499,
+ false);
+ sg2d.drawpipe = aaConverter;
+ sg2d.fillpipe = aaConverter;
+ sg2d.shapepipe = aaConverter;
+ } else if (sg2d.compositeState == sg2d.COMP_XOR) {
+ // install the solid pipes when AA and XOR are both enabled
+ txPipe = d3dTxRenderPipe;
+ nonTxPipe = d3dRenderPipe;
+ }
+ }
+ // other cases handled by super.validatePipe() below
+ }
+
+ if (txPipe != null) {
+ if (sg2d.transformState >= sg2d.TRANSFORM_TRANSLATESCALE) {
+ sg2d.drawpipe = txPipe;
+ sg2d.fillpipe = txPipe;
+ } else if (sg2d.strokeState != sg2d.STROKE_THIN) {
+ sg2d.drawpipe = txPipe;
+ sg2d.fillpipe = nonTxPipe;
+ } else {
+ sg2d.drawpipe = nonTxPipe;
+ sg2d.fillpipe = nonTxPipe;
+ }
+ // Note that we use the transforming pipe here because it
+ // will examine the shape and possibly perform an optimized
+ // operation if it can be simplified. The simplifications
+ // will be valid for all STROKE and TRANSFORM types.
+ sg2d.shapepipe = txPipe;
+ } else {
+ if (!validated) {
+ super.validatePipe(sg2d);
+ }
+ }
+
+ // install the text pipe based on our earlier decision
+ sg2d.textpipe = textpipe;
+
+ // always override the image pipe with the specialized D3D pipe
+ sg2d.imagepipe = d3dImagePipe;
+ }
+
@Override
- public boolean surfacePunted() {
- // Punting is disabled for D3D surfaces
+ protected MaskFill getMaskFill(SunGraphics2D sg2d) {
+ if (sg2d.paintState > sg2d.PAINT_ALPHACOLOR) {
+ /*
+ * We can only accelerate non-Color MaskFill operations if
+ * all of the following conditions hold true:
+ * - there is an implementation for the given paintState
+ * - the current Paint can be accelerated for this destination
+ * - multitexturing is available (since we need to modulate
+ * the alpha mask texture with the paint texture)
+ *
+ * In all other cases, we return null, in which case the
+ * validation code will choose a more general software-based loop.
+ */
+ if (!D3DPaints.isValid(sg2d) ||
+ !graphicsDevice.isCapPresent(CAPS_MULTITEXTURE))
+ {
+ return null;
+ }
+ }
+ return super.getMaskFill(sg2d);
+ }
+
+ @Override
+ public boolean copyArea(SunGraphics2D sg2d,
+ int x, int y, int w, int h, int dx, int dy)
+ {
+ if (sg2d.transformState < sg2d.TRANSFORM_TRANSLATESCALE &&
+ sg2d.compositeState < sg2d.COMP_XOR)
+ {
+ x += sg2d.transX;
+ y += sg2d.transY;
+
+ d3dRenderPipe.copyArea(sg2d, x, y, w, h, dx, dy);
+
+ return true;
+ }
return false;
}
- D3DContext getContext() {
- return ((Win32GraphicsDevice)graphicsConfig.getDevice()).getD3DContext();
+ @Override
+ public void flush() {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ RenderBuffer buf = rq.getBuffer();
+ rq.ensureCapacityAndAlignment(12, 4);
+ buf.putInt(FLUSH_SURFACE);
+ buf.putLong(getNativeOps());
+
+ // this call is expected to complete synchronously, so flush now
+ rq.flushNow();
+ } finally {
+ rq.unlock();
+ }
}
+
+ /**
+ * Disposes the native resources associated with the given D3DSurfaceData
+ * (referenced by the pData parameter). This method is invoked from
+ * the native Dispose() method from the Disposer thread when the
+ * Java-level D3DSurfaceData object is about to go away.
+ */
+ static void dispose(long pData) {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ RenderBuffer buf = rq.getBuffer();
+ rq.ensureCapacityAndAlignment(12, 4);
+ buf.putInt(DISPOSE_SURFACE);
+ buf.putLong(pData);
+
+ // this call is expected to complete synchronously, so flush now
+ rq.flushNow();
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ static void swapBuffers(D3DSurfaceData sd,
+ final int x1, final int y1,
+ final int x2, final int y2)
+ {
+ long pData = sd.getNativeOps();
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ // swapBuffers can be called from the toolkit thread by swing, we
+ // should detect this and prevent the deadlocks
+ if (rq.isRenderQueueThread()) {
+ if (!rq.tryLock()) {
+ // if we could not obtain the lock, repaint the area
+ // that was supposed to be swapped, and no-op this swap
+ final Component target = (Component)sd.getPeer().getTarget();
+ SunToolkit.executeOnEventHandlerThread(target, new Runnable() {
+ public void run() {
+ target.repaint(x1, y1, x2, y2);
+ }
+ });
+ return;
+ }
+ } else {
+ rq.lock();
+ }
+ try {
+ RenderBuffer buf = rq.getBuffer();
+ rq.ensureCapacityAndAlignment(28, 4);
+ buf.putInt(SWAP_BUFFERS);
+ buf.putLong(pData);
+ buf.putInt(x1);
+ buf.putInt(y1);
+ buf.putInt(x2);
+ buf.putInt(y2);
+ rq.flushNow();
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ /**
+ * Returns destination Image associated with this SurfaceData.
+ */
+ public Object getDestination() {
+ return offscreenImage;
+ }
+
+ public Rectangle getBounds() {
+ if (type == FLIP_BACKBUFFER || type == WINDOW) {
+ Rectangle r = peer.getBounds();
+ r.x = r.y = 0;
+ return r;
+ } else {
+ return new Rectangle(width, height);
+ }
+ }
+
+ public Rectangle getNativeBounds() {
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ // need to lock to make sure nativeWidth and Height are consistent
+ // since they are set from the render thread from the native
+ // level
+ rq.lock();
+ try {
+ // REMIND: use xyoffsets?
+ return new Rectangle(nativeWidth, nativeHeight);
+ } finally {
+ rq.unlock();
+ }
+ }
+
+
+ public GraphicsConfiguration getDeviceConfiguration() {
+ return graphicsDevice.getDefaultConfiguration();
+ }
+
+ public SurfaceData getReplacement() {
+ return restoreContents(offscreenImage);
+ }
+
+ private static D3DGraphicsConfig getGC(WComponentPeer peer) {
+ GraphicsConfiguration gc;
+ if (peer != null) {
+ gc = peer.getGraphicsConfiguration();
+ } else {
+ GraphicsEnvironment env =
+ GraphicsEnvironment.getLocalGraphicsEnvironment();
+ GraphicsDevice gd = env.getDefaultScreenDevice();
+ gc = gd.getDefaultConfiguration();
+ }
+ return (gc instanceof D3DGraphicsConfig) ? (D3DGraphicsConfig)gc : null;
+ }
+
+ /**
+ * Attempts to restore the surface by initializing the native data
+ */
+ void restoreSurface() {
+ initSurface();
+ }
+
+ WComponentPeer getPeer() {
+ return peer;
+ }
+
+ /**
+ * We need to let the surface manager know that the surface is lost so
+ * that for example BufferStrategy.contentsLost() returns correct result.
+ * Normally the status of contentsLost is set in validate(), but in some
+ * cases (like Swing's buffer per window) we intentionally don't call
+ * validate from the toolkit thread but only check for the BS status.
+ */
+ @Override
+ public void setSurfaceLost(boolean lost) {
+ super.setSurfaceLost(lost);
+ if (lost && offscreenImage != null) {
+ SurfaceManager sm = SurfaceManager.getManager(offscreenImage);
+ sm.acceleratedSurfaceLost();
+ }
+ }
+
+ private static native long getNativeResourceNative(long sdops, int resType);
+ /**
+ * Returns a pointer to the native resource of specified {@code resType}
+ * associated with this surface.
+ *
+ * Specifically, for {@code D3DSurfaceData} this method returns pointers of
+ * the following:
+ *
+ * TEXTURE - (IDirect3DTexture9*)
+ * RT_TEXTURE, RT_PLAIN - (IDirect3DSurface9*)
+ * FLIP_BACKBUFFER - (IDirect3DSwapChain9*)
+ * D3D_DEVICE_RESOURCE - (IDirect3DDevice9*)
+ *
+ *
+ * Multiple resources may be available for some types (i.e. for render to
+ * texture one could retrieve both a destination surface by specifying
+ * RT_TEXTURE, and a texture by using TEXTURE).
+ *
+ * Note: the pointer returned by this method is only valid on the rendering
+ * thread.
+ *
+ * @return pointer to the native resource of specified type or 0L if
+ * such resource doesn't exist or can not be retrieved.
+ * @see sun.java2d.pipe.hw.AccelSurface#getNativeResource
+ */
+ public long getNativeResource(int resType) {
+ return getNativeResourceNative(getNativeOps(), resType);
+ }
+
+ /**
+ * Class representing an on-screen d3d surface. Since d3d can't
+ * render to the screen directly, it is implemented as a swap chain,
+ * controlled by D3DScreenUpdateManager.
+ *
+ * @see D3DScreenUpdateManager
+ */
+ public static class D3DWindowSurfaceData extends D3DSurfaceData {
+ StateTracker dirtyTracker;
+
+ public D3DWindowSurfaceData(WComponentPeer peer,
+ D3DGraphicsConfig gc)
+ {
+ super(peer, gc,
+ peer.getBounds().width, peer.getBounds().height,
+ null, peer.getColorModel(), 1, SWAP_COPY, VSYNC_DEFAULT,
+ WINDOW);
+ dirtyTracker = getStateTracker();
+ }
+
+ /**
+ * {@inheritDoc}
+ *
+ * Overridden to use ScreenUpdateManager to obtain the replacement
+ * surface.
+ *
+ * @see sun.java2d.ScreenUpdateManager#getReplacementScreenSurface
+ */
+ @Override
+ public SurfaceData getReplacement() {
+ ScreenUpdateManager mgr = ScreenUpdateManager.getInstance();
+ return mgr.getReplacementScreenSurface(peer, this);
+ }
+
+ /**
+ * Returns destination Component associated with this SurfaceData.
+ */
+ @Override
+ public Object getDestination() {
+ return peer.getTarget();
+ }
+
+ @Override
+ void restoreSurface() {
+ Window fsw = graphicsDevice.getFullScreenWindow();
+ if (fsw != null && fsw != peer.getTarget()) {
+ throw new InvalidPipeException("Can't restore onscreen surface"+
+ " when in full-screen mode");
+ }
+ super.restoreSurface();
+ // if initialization was unsuccessful, an IPE will be thrown
+ // and the surface will remain lost
+ setSurfaceLost(false);
+
+ // This is to make sure the render target is reset after this
+ // surface is restored. The reason for this is that sometimes this
+ // surface can be restored from multiple threads (the screen update
+ // manager's thread and app's rendering thread) at the same time,
+ // and when that happens the second restoration will create the
+ // native resource which will not be set as render target because
+ // the BufferedContext's validate method will think that since the
+ // surface data object didn't change then the current render target
+ // is correct and no rendering will appear on the screen.
+ D3DRenderQueue rq = D3DRenderQueue.getInstance();
+ rq.lock();
+ try {
+ getContext().invalidateContext();
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ public boolean isDirty() {
+ return !dirtyTracker.isCurrent();
+ }
+
+ public void markClean() {
+ dirtyTracker = getStateTracker();
+ }
+ }
+
+ /**
+ * Updates the layered window with the contents of the surface.
+ *
+ * @param pd3dsd pointer to the D3DSDOps structure
+ * @param pData pointer to the AwtWindow peer data
+ * @param w width of the window
+ * @param h height of the window
+ * @see sun.awt.windows.TranslucentWindowPainter
+ */
+ public static native boolean updateWindowAccelImpl(long pd3dsd, long pData,
+ int w, int h);
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceDataProxy.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceDataProxy.java
new file mode 100644
index 00000000000..373636d9a4b
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DSurfaceDataProxy.java
@@ -0,0 +1,93 @@
+/*
+ * Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.Color;
+import java.awt.Transparency;
+
+import sun.java2d.InvalidPipeException;
+import sun.java2d.SurfaceData;
+import sun.java2d.SurfaceDataProxy;
+import sun.java2d.loops.CompositeType;
+
+/**
+ * The proxy class contains the logic for when to replace a
+ * SurfaceData with a cached D3D Texture and the code to create
+ * the accelerated surfaces.
+ */
+public class D3DSurfaceDataProxy extends SurfaceDataProxy {
+
+ public static SurfaceDataProxy createProxy(SurfaceData srcData,
+ D3DGraphicsConfig dstConfig)
+ {
+ if (srcData instanceof D3DSurfaceData) {
+ // srcData must be a VolatileImage which either matches
+ // our pixel format or not - either way we do not cache it...
+ return UNCACHED;
+ }
+
+ return new D3DSurfaceDataProxy(dstConfig, srcData.getTransparency());
+ }
+
+ D3DGraphicsConfig d3dgc;
+ int transparency;
+
+ public D3DSurfaceDataProxy(D3DGraphicsConfig d3dgc, int transparency) {
+ this.d3dgc = d3dgc;
+ this.transparency = transparency;
+ // REMIND: we may want to change this for the d3d pipeline, it's not
+ // necessary to invalidate them all at once on display change
+ activateDisplayListener();
+ }
+
+ @Override
+ public SurfaceData validateSurfaceData(SurfaceData srcData,
+ SurfaceData cachedData,
+ int w, int h)
+ {
+ if (cachedData == null || cachedData.isSurfaceLost()) {
+ try {
+ cachedData = d3dgc.createManagedSurface(w, h, transparency);
+ } catch (InvalidPipeException e) {
+ if (!d3dgc.getD3DDevice().isD3DAvailable()) {
+ invalidate();
+ flush();
+ return null;
+ }
+ }
+ }
+ return cachedData;
+ }
+
+ @Override
+ public boolean isSupportedOperation(SurfaceData srcData,
+ int txtype,
+ CompositeType comp,
+ Color bgColor)
+ {
+ return (bgColor == null || transparency == Transparency.OPAQUE);
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DTextRenderer.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DTextRenderer.java
index 4c4cc182d75..fa7b746d1e7 100644
--- a/jdk/src/windows/classes/sun/java2d/d3d/D3DTextRenderer.java
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DTextRenderer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -25,51 +25,47 @@
package sun.java2d.d3d;
-import java.awt.AlphaComposite;
+import java.awt.Composite;
import sun.font.GlyphList;
import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
import sun.java2d.loops.GraphicsPrimitive;
-import sun.java2d.pipe.GlyphListPipe;
-import sun.java2d.pipe.Region;
+import sun.java2d.pipe.BufferedTextPipe;
+import sun.java2d.pipe.RenderQueue;
-public class D3DTextRenderer extends GlyphListPipe {
+class D3DTextRenderer extends BufferedTextPipe {
- protected native void doDrawGlyphList(long pData, long pCtx,
- Region clip, GlyphList gl);
-
- protected void drawGlyphList(SunGraphics2D sg2d, GlyphList gl) {
- AlphaComposite comp = (AlphaComposite)sg2d.composite;
- // We can only get here if the comp is Src or SrcOver (see
- // pipeline validation in D3DSurfaceData, so we force
- // it to be SrcOver.
- if (comp.getRule() != AlphaComposite.SRC_OVER) {
- comp = AlphaComposite.SrcOver;
- }
-
- synchronized (D3DContext.LOCK) {
- SurfaceData dstData = sg2d.surfaceData;
- long pCtx = D3DContext.getContext(dstData, dstData,
- sg2d.getCompClip(), comp,
- null, sg2d.eargb,
- D3DContext.NO_CONTEXT_FLAGS);
-
- doDrawGlyphList(dstData.getNativeOps(), pCtx,
- sg2d.getCompClip(), gl);
- }
+ D3DTextRenderer(RenderQueue rq) {
+ super(rq);
}
- public D3DTextRenderer traceWrap() {
- return new Tracer();
+ @Override
+ protected native void drawGlyphList(int numGlyphs, boolean usePositions,
+ boolean subPixPos, boolean rgbOrder,
+ int lcdContrast,
+ float glOrigX, float glOrigY,
+ long[] images, float[] positions);
+
+ @Override
+ protected void validateContext(SunGraphics2D sg2d, Composite comp) {
+ // assert rq.lock.isHeldByCurrentThread();
+ D3DSurfaceData d3dDst = (D3DSurfaceData)sg2d.surfaceData;
+ D3DContext.validateContext(d3dDst, d3dDst,
+ sg2d.getCompClip(), comp,
+ null, sg2d.paint, sg2d,
+ D3DContext.NO_CONTEXT_FLAGS);
}
- public static class Tracer extends D3DTextRenderer {
- @Override
- protected void doDrawGlyphList(long pData, long pCtx,
- Region clip, GlyphList gl)
- {
+ D3DTextRenderer traceWrap() {
+ return new Tracer(this);
+ }
+
+ private static class Tracer extends D3DTextRenderer {
+ Tracer(D3DTextRenderer d3dtr) {
+ super(d3dtr.rq);
+ }
+ protected void drawGlyphList(SunGraphics2D sg2d, GlyphList gl) {
GraphicsPrimitive.tracePrimitive("D3DDrawGlyphs");
- super.doDrawGlyphList(pData, pCtx, clip, gl);
+ super.drawGlyphList(sg2d, gl);
}
}
}
diff --git a/jdk/src/windows/classes/sun/java2d/d3d/D3DVolatileSurfaceManager.java b/jdk/src/windows/classes/sun/java2d/d3d/D3DVolatileSurfaceManager.java
new file mode 100644
index 00000000000..3dcf99b4cc0
--- /dev/null
+++ b/jdk/src/windows/classes/sun/java2d/d3d/D3DVolatileSurfaceManager.java
@@ -0,0 +1,230 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+package sun.java2d.d3d;
+
+import java.awt.Component;
+import java.awt.GraphicsConfiguration;
+import java.awt.Image;
+import java.awt.Transparency;
+import java.awt.image.ColorModel;
+import sun.awt.Win32GraphicsConfig;
+import sun.awt.image.SunVolatileImage;
+import sun.awt.image.SurfaceManager;
+import sun.awt.image.VolatileSurfaceManager;
+import sun.awt.windows.WComponentPeer;
+import sun.java2d.InvalidPipeException;
+import sun.java2d.SurfaceData;
+import static sun.java2d.pipe.hw.AccelSurface.*;
+import static sun.java2d.d3d.D3DContext.D3DContextCaps.*;
+import sun.java2d.windows.GDIWindowSurfaceData;
+
+public class D3DVolatileSurfaceManager
+ extends VolatileSurfaceManager
+{
+ private boolean accelerationEnabled;
+ private int restoreCountdown;
+
+ public D3DVolatileSurfaceManager(SunVolatileImage vImg, Object context) {
+ super(vImg, context);
+
+ /*
+ * We will attempt to accelerate this image only under the
+ * following conditions:
+ * - the image is opaque OR
+ * - the image is translucent AND
+ * - the GraphicsConfig supports the FBO extension OR
+ * - the GraphicsConfig has a stored alpha channel
+ */
+ int transparency = vImg.getTransparency();
+ D3DGraphicsDevice gd = (D3DGraphicsDevice)
+ vImg.getGraphicsConfig().getDevice();
+ accelerationEnabled =
+ (transparency == Transparency.OPAQUE) ||
+ (transparency == Transparency.TRANSLUCENT &&
+ (gd.isCapPresent(CAPS_RT_PLAIN_ALPHA) ||
+ gd.isCapPresent(CAPS_RT_TEXTURE_ALPHA)));
+ }
+
+ protected boolean isAccelerationEnabled() {
+ return accelerationEnabled;
+ }
+ public void setAccelerationEnabled(boolean accelerationEnabled) {
+ this.accelerationEnabled = accelerationEnabled;
+ }
+
+ /**
+ * Create a pbuffer-based SurfaceData object (or init the backbuffer
+ * of an existing window if this is a double buffered GraphicsConfig).
+ */
+ protected SurfaceData initAcceleratedSurface() {
+ SurfaceData sData;
+ Component comp = vImg.getComponent();
+ WComponentPeer peer =
+ (comp != null) ? (WComponentPeer)comp.getPeer() : null;
+
+ try {
+ boolean forceback = false;
+ if (context instanceof Boolean) {
+ forceback = ((Boolean)context).booleanValue();
+ }
+
+ if (forceback) {
+ // peer must be non-null in this case
+ sData = D3DSurfaceData.createData(peer, vImg);
+ } else {
+ D3DGraphicsConfig gc =
+ (D3DGraphicsConfig)vImg.getGraphicsConfig();
+ ColorModel cm = gc.getColorModel(vImg.getTransparency());
+ int type = vImg.getForcedAccelSurfaceType();
+ // if acceleration type is forced (type != UNDEFINED) then
+ // use the forced type, otherwise use RT_TEXTURE
+ if (type == UNDEFINED) {
+ type = RT_TEXTURE;
+ }
+ sData = D3DSurfaceData.createData(gc,
+ vImg.getWidth(),
+ vImg.getHeight(),
+ cm, vImg,
+ type);
+ }
+ } catch (NullPointerException ex) {
+ sData = null;
+ } catch (OutOfMemoryError er) {
+ sData = null;
+ } catch (InvalidPipeException ipe) {
+ sData = null;
+ }
+
+ return sData;
+ }
+
+ protected boolean isConfigValid(GraphicsConfiguration gc) {
+ return ((gc == null) || (gc == vImg.getGraphicsConfig()));
+ }
+
+ /**
+ * Set the number of iterations for restoreAcceleratedSurface to fail
+ * before attempting to restore the accelerated surface.
+ *
+ * @see #restoreAcceleratedSurface
+ * @see #handleVItoScreenOp
+ */
+ private synchronized void setRestoreCountdown(int count) {
+ restoreCountdown = count;
+ }
+
+ /**
+ * Note that we create a new surface instead of restoring
+ * an old one. This will help with D3DContext revalidation.
+ */
+ @Override
+ protected void restoreAcceleratedSurface() {
+ synchronized (this) {
+ if (restoreCountdown > 0) {
+ restoreCountdown--;
+ throw new
+ InvalidPipeException("Will attempt to restore surface " +
+ " in " + restoreCountdown);
+ }
+ }
+
+ SurfaceData sData = initAcceleratedSurface();
+ if (sData != null) {
+ sdAccel = sData;
+ } else {
+ throw new InvalidPipeException("could not restore surface");
+ // REMIND: alternatively, we could try this:
+// ((D3DSurfaceData)sdAccel).restoreSurface();
+ }
+ }
+
+ /**
+ * We're asked to restore contents by the accelerated surface, which means
+ * that it had been lost.
+ */
+ @Override
+ public SurfaceData restoreContents() {
+ acceleratedSurfaceLost();
+ return super.restoreContents();
+ }
+
+ /**
+ * If the destination surface's peer can potentially handle accelerated
+ * on-screen rendering then it is likely that the condition which resulted
+ * in VI to Screen operation is temporary, so this method sets the
+ * restore countdown in hope that the on-screen accelerated rendering will
+ * resume. In the meantime the backup surface of the VISM will be used.
+ *
+ * The countdown is needed because otherwise we may never break out
+ * of "do { vi.validate()..} while(vi.lost)" loop since validate() could
+ * restore the source surface every time and it will get lost again on the
+ * next copy attempt, and we would never get a chance to use the backup
+ * surface. By using the countdown we allow the backup surface to be used
+ * while the screen surface gets sorted out, or if it for some reason can
+ * never be restored.
+ *
+ * If the destination surface's peer could never do accelerated onscreen
+ * rendering then the acceleration for the SurfaceManager associated with
+ * the source surface is disabled forever.
+ */
+ static void handleVItoScreenOp(SurfaceData src, SurfaceData dst) {
+ if (src instanceof D3DSurfaceData &&
+ dst instanceof GDIWindowSurfaceData)
+ {
+ D3DSurfaceData d3dsd = (D3DSurfaceData)src;
+ SurfaceManager mgr =
+ SurfaceManager.getManager((Image)d3dsd.getDestination());
+ if (mgr instanceof D3DVolatileSurfaceManager) {
+ D3DVolatileSurfaceManager vsm = (D3DVolatileSurfaceManager)mgr;
+ if (vsm != null) {
+ d3dsd.setSurfaceLost(true);
+
+ GDIWindowSurfaceData wsd = (GDIWindowSurfaceData)dst;
+ WComponentPeer p = wsd.getPeer();
+ if (D3DScreenUpdateManager.canUseD3DOnScreen(p,
+ (Win32GraphicsConfig)p.getGraphicsConfiguration(),
+ p.getBackBuffersNum()))
+ {
+ // 10 is only chosen to be greater than the number of
+ // times a sane person would call validate() inside
+ // a validation loop, and to reduce thrashing between
+ // accelerated and backup surfaces
+ vsm.setRestoreCountdown(10);
+ } else {
+ vsm.setAccelerationEnabled(false);
+ }
+ }
+ }
+ }
+ }
+
+ @Override
+ public void initContents() {
+ if (vImg.getForcedAccelSurfaceType() != TEXTURE) {
+ super.initContents();
+ }
+ }
+}
diff --git a/jdk/src/windows/classes/sun/java2d/opengl/WGLGraphicsConfig.java b/jdk/src/windows/classes/sun/java2d/opengl/WGLGraphicsConfig.java
index 38af32de544..7ff7e1e8d09 100644
--- a/jdk/src/windows/classes/sun/java2d/opengl/WGLGraphicsConfig.java
+++ b/jdk/src/windows/classes/sun/java2d/opengl/WGLGraphicsConfig.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -27,8 +27,11 @@ package sun.java2d.opengl;
import java.awt.AWTException;
import java.awt.BufferCapabilities;
+import java.awt.BufferCapabilities.FlipContents;
+import java.awt.Color;
import java.awt.Component;
import java.awt.Graphics;
+import java.awt.Graphics2D;
import java.awt.ImageCapabilities;
import java.awt.Transparency;
import java.awt.color.ColorSpace;
@@ -39,10 +42,22 @@ import java.awt.image.VolatileImage;
import sun.awt.Win32GraphicsConfig;
import sun.awt.Win32GraphicsDevice;
import sun.awt.image.SunVolatileImage;
+import sun.awt.image.SurfaceManager;
import sun.awt.windows.WComponentPeer;
import sun.java2d.Disposer;
import sun.java2d.DisposerRecord;
+import sun.java2d.SunGraphics2D;
+import sun.java2d.Surface;
import sun.java2d.SurfaceData;
+import sun.java2d.pipe.hw.AccelSurface;
+import sun.java2d.pipe.hw.AccelTypedVolatileImage;
+import sun.java2d.pipe.hw.ContextCapabilities;
+import static sun.java2d.opengl.OGLContext.OGLContextCaps.*;
+import static sun.java2d.opengl.WGLSurfaceData.*;
+import sun.java2d.opengl.OGLContext.OGLContextCaps;
+import sun.java2d.pipe.hw.AccelDeviceEventListener;
+import sun.java2d.pipe.hw.AccelDeviceEventNotifier;
+import sun.java2d.windows.GDIWindowSurfaceData;
public class WGLGraphicsConfig
extends Win32GraphicsConfig
@@ -53,7 +68,7 @@ public class WGLGraphicsConfig
private BufferCapabilities bufferCaps;
private long pConfigInfo;
- private int oglCaps;
+ private ContextCapabilities oglCaps;
private OGLContext context;
private Object disposerReferent = new Object();
@@ -67,17 +82,18 @@ public class WGLGraphicsConfig
}
protected WGLGraphicsConfig(Win32GraphicsDevice device, int visualnum,
- long configInfo, int oglCaps)
+ long configInfo, ContextCapabilities oglCaps)
{
super(device, visualnum);
this.pConfigInfo = configInfo;
this.oglCaps = oglCaps;
- context = new OGLContext(OGLRenderQueue.getInstance());
+ context = new OGLContext(OGLRenderQueue.getInstance(), this);
// add a record to the Disposer so that we destroy the native
// WGLGraphicsConfigInfo data when this object goes away
Disposer.addRecord(disposerReferent,
- new WGLGCDisposerRecord(pConfigInfo));
+ new WGLGCDisposerRecord(pConfigInfo,
+ device.getScreen()));
}
public Object getProxyKey() {
@@ -99,6 +115,7 @@ public class WGLGraphicsConfig
}
long cfginfo = 0;
+ final String ids[] = new String[1];
OGLRenderQueue rq = OGLRenderQueue.getInstance();
rq.lock();
try {
@@ -110,6 +127,12 @@ public class WGLGraphicsConfig
new WGLGetConfigInfo(device.getScreen(), pixfmt);
rq.flushAndInvokeNow(action);
cfginfo = action.getConfigInfo();
+ OGLContext.setScratchSurface(cfginfo);
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ ids[0] = OGLContext.getOGLIdString();
+ }
+ });
} finally {
rq.unlock();
}
@@ -118,8 +141,9 @@ public class WGLGraphicsConfig
}
int oglCaps = getOGLCapabilities(cfginfo);
+ ContextCapabilities caps = new OGLContextCaps(oglCaps, ids[0]);
- return new WGLGraphicsConfig(device, pixfmt, cfginfo, oglCaps);
+ return new WGLGraphicsConfig(device, pixfmt, cfginfo, caps);
}
/**
@@ -150,24 +174,49 @@ public class WGLGraphicsConfig
* Returns true if the provided capability bit is present for this config.
* See OGLContext.java for a list of supported capabilities.
*/
+ @Override
public final boolean isCapPresent(int cap) {
- return ((oglCaps & cap) != 0);
+ return ((oglCaps.getCaps() & cap) != 0);
}
+ @Override
public final long getNativeConfigInfo() {
return pConfigInfo;
}
+ /**
+ * {@inheritDoc}
+ *
+ * @see sun.java2d.pipe.hw.BufferedContextProvider#getContext
+ */
+ @Override
public final OGLContext getContext() {
return context;
}
private static class WGLGCDisposerRecord implements DisposerRecord {
private long pCfgInfo;
- public WGLGCDisposerRecord(long pCfgInfo) {
+ private int screen;
+ public WGLGCDisposerRecord(long pCfgInfo, int screen) {
this.pCfgInfo = pCfgInfo;
}
public void dispose() {
+ OGLRenderQueue rq = OGLRenderQueue.getInstance();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ AccelDeviceEventNotifier.
+ eventOccured(screen,
+ AccelDeviceEventNotifier.DEVICE_RESET);
+ AccelDeviceEventNotifier.
+ eventOccured(screen,
+ AccelDeviceEventNotifier.DEVICE_DISPOSED);
+ }
+ });
+ } finally {
+ rq.unlock();
+ }
if (pCfgInfo != 0) {
OGLRenderQueue.disposeGraphicsConfig(pCfgInfo);
pCfgInfo = 0;
@@ -230,7 +279,11 @@ public class WGLGraphicsConfig
public SurfaceData createSurfaceData(WComponentPeer peer,
int numBackBuffers)
{
- return WGLSurfaceData.createData(peer);
+ SurfaceData sd = WGLSurfaceData.createData(peer);
+ if (sd == null) {
+ sd = GDIWindowSurfaceData.createData(peer);
+ }
+ return sd;
}
/**
@@ -279,16 +332,36 @@ public class WGLGraphicsConfig
@Override
public void flip(WComponentPeer peer,
Component target, VolatileImage backBuffer,
+ int x1, int y1, int x2, int y2,
BufferCapabilities.FlipContents flipAction)
{
if (flipAction == BufferCapabilities.FlipContents.COPIED) {
- Graphics g = peer.getGraphics();
- try {
- g.drawImage(backBuffer, 0, 0, null);
- } finally {
- g.dispose();
+ SurfaceManager vsm = SurfaceManager.getManager(backBuffer);
+ SurfaceData sd = vsm.getPrimarySurfaceData();
+
+ if (sd instanceof WGLVSyncOffScreenSurfaceData) {
+ WGLVSyncOffScreenSurfaceData vsd =
+ (WGLVSyncOffScreenSurfaceData)sd;
+ SurfaceData bbsd = vsd.getFlipSurface();
+ Graphics2D bbg =
+ new SunGraphics2D(bbsd, Color.black, Color.white, null);
+ try {
+ bbg.drawImage(backBuffer, 0, 0, null);
+ } finally {
+ bbg.dispose();
+ }
+ } else {
+ Graphics g = peer.getGraphics();
+ try {
+ g.drawImage(backBuffer,
+ x1, y1, x2, y2,
+ x1, y1, x2, y2,
+ null);
+ } finally {
+ g.dispose();
+ }
+ return;
}
- return;
} else if (flipAction == BufferCapabilities.FlipContents.PRIOR) {
// not supported by WGL...
return;
@@ -319,7 +392,7 @@ public class WGLGraphicsConfig
@Override
public BufferCapabilities getBufferCapabilities() {
if (bufferCaps == null) {
- boolean dblBuf = isCapPresent(OGLContext.CAPS_DOUBLEBUFFERED);
+ boolean dblBuf = isCapPresent(CAPS_DOUBLEBUFFERED);
bufferCaps = new WGLBufferCaps(dblBuf);
}
return bufferCaps;
@@ -338,4 +411,64 @@ public class WGLGraphicsConfig
public ImageCapabilities getImageCapabilities() {
return imageCaps;
}
+
+ /**
+ * {@inheritDoc}
+ *
+ * @see sun.java2d.pipe.hw.AccelGraphicsConfig#createCompatibleVolatileImage
+ */
+ @Override
+ public VolatileImage
+ createCompatibleVolatileImage(int width, int height,
+ int transparency, int type)
+ {
+ if (type == FLIP_BACKBUFFER || type == WINDOW || type == UNDEFINED ||
+ transparency == Transparency.BITMASK)
+ {
+ return null;
+ }
+
+ if (type == FBOBJECT) {
+ if (!isCapPresent(CAPS_EXT_FBOBJECT)) {
+ return null;
+ }
+ } else if (type == PBUFFER) {
+ boolean isOpaque = transparency == Transparency.OPAQUE;
+ if (!isOpaque && !isCapPresent(CAPS_STORED_ALPHA)) {
+ return null;
+ }
+ }
+
+ SunVolatileImage vi = new AccelTypedVolatileImage(this, width, height,
+ transparency, type);
+ Surface sd = vi.getDestSurface();
+ if (!(sd instanceof AccelSurface) ||
+ ((AccelSurface)sd).getType() != type)
+ {
+ vi.flush();
+ vi = null;
+ }
+
+ return vi;
+ }
+
+ /**
+ * {@inheritDoc}
+ *
+ * @see sun.java2d.pipe.hw.AccelGraphicsConfig#getContextCapabilities
+ */
+ @Override
+ public ContextCapabilities getContextCapabilities() {
+ return oglCaps;
+ }
+
+ @Override
+ public void addDeviceEventListener(AccelDeviceEventListener l) {
+ AccelDeviceEventNotifier.addListener(l, screen.getScreen());
+ }
+
+ @Override
+ public void removeDeviceEventListener(AccelDeviceEventListener l) {
+ AccelDeviceEventNotifier.removeListener(l);
+ }
}
diff --git a/jdk/src/windows/classes/sun/java2d/opengl/WGLSurfaceData.java b/jdk/src/windows/classes/sun/java2d/opengl/WGLSurfaceData.java
index ffb6ba89a71..27997aa8c59 100644
--- a/jdk/src/windows/classes/sun/java2d/opengl/WGLSurfaceData.java
+++ b/jdk/src/windows/classes/sun/java2d/opengl/WGLSurfaceData.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -26,26 +26,23 @@
package sun.java2d.opengl;
import java.awt.Component;
-import java.awt.Container;
import java.awt.GraphicsConfiguration;
import java.awt.GraphicsDevice;
import java.awt.GraphicsEnvironment;
import java.awt.Image;
-import java.awt.Insets;
import java.awt.Rectangle;
-import java.awt.Transparency;
import java.awt.image.ColorModel;
+import sun.awt.SunToolkit;
import sun.awt.windows.WComponentPeer;
import sun.java2d.SurfaceData;
-import sun.java2d.loops.SurfaceType;
public abstract class WGLSurfaceData extends OGLSurfaceData {
protected WComponentPeer peer;
private WGLGraphicsConfig graphicsConfig;
- private native void initOps(long pConfigInfo, long pPeerData,
- int xoff, int yoff);
+ private native void initOps(long pConfigInfo, WComponentPeer peer,
+ long hwnd);
protected native boolean initPbuffer(long pData, long pConfigInfo,
boolean isOpaque,
int width, int height);
@@ -58,19 +55,9 @@ public abstract class WGLSurfaceData extends OGLSurfaceData {
this.graphicsConfig = gc;
long pConfigInfo = gc.getNativeConfigInfo();
- long pPeerData = 0L;
- int xoff = 0, yoff = 0;
- if (peer != null) {
- Component c = (Component)peer.getTarget();
- if (c instanceof Container) {
- Insets insets = ((Container)c).getInsets();
- xoff = -insets.left;
- yoff = -insets.bottom;
- }
- pPeerData = peer.getData();
- }
+ long hwnd = peer != null ? peer.getHWnd() : 0L;
- initOps(pConfigInfo, pPeerData, xoff, yoff);
+ initOps(pConfigInfo, peer, hwnd);
}
public GraphicsConfiguration getDeviceConfiguration() {
@@ -82,6 +69,15 @@ public abstract class WGLSurfaceData extends OGLSurfaceData {
* of an on-screen Window.
*/
public static WGLWindowSurfaceData createData(WComponentPeer peer) {
+ // the OGL pipeline can render directly to the screen and interfere
+ // with layered windows, which is why we don't allow accelerated
+ // surfaces in this case
+ if (!peer.isAccelCapable())
+ // REMIND: commented until toplevel translucency is implemented
+// || !SunToolkit.isContainingTopLevelOpaque((Component)peer.getTarget()))
+ {
+ return null;
+ }
WGLGraphicsConfig gc = getGC(peer);
return new WGLWindowSurfaceData(peer, gc);
}
@@ -91,13 +87,29 @@ public abstract class WGLSurfaceData extends OGLSurfaceData {
* double-buffered on-screen Window.
*/
public static WGLOffScreenSurfaceData createData(WComponentPeer peer,
- Image image)
+ Image image,
+ int type)
{
+ // the OGL pipeline can render directly to the screen and interfere
+ // with layered windows, which is why we don't allow accelerated
+ // surfaces in this case
+ if (!peer.isAccelCapable())
+ // REMIND: commented until toplevel translucency is implemented
+// || !SunToolkit.isContainingTopLevelOpaque((Component)peer.getTarget()))
+ {
+ return null;
+ }
WGLGraphicsConfig gc = getGC(peer);
Rectangle r = peer.getBounds();
- return new WGLOffScreenSurfaceData(peer, gc, r.width, r.height,
- image, peer.getColorModel(),
- FLIP_BACKBUFFER);
+ if (type == FLIP_BACKBUFFER) {
+ return new WGLOffScreenSurfaceData(peer, gc, r.width, r.height,
+ image, peer.getColorModel(),
+ type);
+ } else {
+ return new WGLVSyncOffScreenSurfaceData(peer, gc, r.width, r.height,
+ image, peer.getColorModel(),
+ type);
+ }
}
/**
@@ -152,6 +164,42 @@ public abstract class WGLSurfaceData extends OGLSurfaceData {
}
}
+ /**
+ * A surface which implements a v-synced flip back-buffer with COPIED
+ * FlipContents.
+ *
+ * This surface serves as a back-buffer to the outside world, while
+ * it is actually an offscreen surface. When the BufferStrategy this surface
+ * belongs to is showed, it is first copied to the real private
+ * FLIP_BACKBUFFER, which is then flipped.
+ */
+ public static class WGLVSyncOffScreenSurfaceData extends
+ WGLOffScreenSurfaceData
+ {
+ private WGLOffScreenSurfaceData flipSurface;
+
+ public WGLVSyncOffScreenSurfaceData(WComponentPeer peer,
+ WGLGraphicsConfig gc,
+ int width, int height,
+ Image image, ColorModel cm,
+ int type)
+ {
+ super(peer, gc, width, height, image, cm, type);
+ flipSurface = WGLSurfaceData.createData(peer, image, FLIP_BACKBUFFER);
+ }
+
+ public SurfaceData getFlipSurface() {
+ return flipSurface;
+ }
+
+ @Override
+ public void flush() {
+ flipSurface.flush();
+ super.flush();
+ }
+
+ }
+
public static class WGLOffScreenSurfaceData extends WGLSurfaceData {
private Image offscreenImage;
@@ -193,4 +241,17 @@ public abstract class WGLSurfaceData extends OGLSurfaceData {
return offscreenImage;
}
}
+
+ /**
+ * Updates the layered window with the contents of the surface.
+ *
+ * @param psdops pointer to the native ogl sd structure
+ * @param pData pointer to the AwtWindow peer data
+ * @param w width of the window
+ * @param h height of the window
+ * @see sun.awt.windows.TranslucentWindowPainter
+ */
+ public static native boolean updateWindowAccelImpl(long psdops,
+ WComponentPeer peer,
+ int w, int h);
}
diff --git a/jdk/src/windows/classes/sun/java2d/opengl/WGLVolatileSurfaceManager.java b/jdk/src/windows/classes/sun/java2d/opengl/WGLVolatileSurfaceManager.java
index b4642158bec..c43e3b1c1ce 100644
--- a/jdk/src/windows/classes/sun/java2d/opengl/WGLVolatileSurfaceManager.java
+++ b/jdk/src/windows/classes/sun/java2d/opengl/WGLVolatileSurfaceManager.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -25,16 +25,20 @@
package sun.java2d.opengl;
+import java.awt.BufferCapabilities;
+import static java.awt.BufferCapabilities.FlipContents.*;
import java.awt.Component;
import java.awt.GraphicsConfiguration;
-import java.awt.ImageCapabilities;
-import java.awt.Rectangle;
import java.awt.Transparency;
import java.awt.image.ColorModel;
import sun.awt.image.SunVolatileImage;
import sun.awt.image.VolatileSurfaceManager;
import sun.awt.windows.WComponentPeer;
import sun.java2d.SurfaceData;
+import static sun.java2d.opengl.OGLContext.OGLContextCaps.*;
+import static sun.java2d.pipe.hw.AccelSurface.*;
+import sun.java2d.pipe.hw.ExtendedBufferCapabilities;
+import static sun.java2d.pipe.hw.ExtendedBufferCapabilities.VSyncType.*;
public class WGLVolatileSurfaceManager
extends VolatileSurfaceManager
@@ -57,8 +61,8 @@ public class WGLVolatileSurfaceManager
accelerationEnabled =
(transparency == Transparency.OPAQUE) ||
((transparency == Transparency.TRANSLUCENT) &&
- (gc.isCapPresent(OGLContext.CAPS_EXT_FBOBJECT) ||
- gc.isCapPresent(OGLContext.CAPS_STORED_ALPHA)));
+ (gc.isCapPresent(CAPS_EXT_FBOBJECT) ||
+ gc.isCapPresent(CAPS_STORED_ALPHA)));
}
protected boolean isAccelerationEnabled() {
@@ -76,24 +80,47 @@ public class WGLVolatileSurfaceManager
(comp != null) ? (WComponentPeer)comp.getPeer() : null;
try {
+ boolean createVSynced = false;
boolean forceback = false;
if (context instanceof Boolean) {
forceback = ((Boolean)context).booleanValue();
+ if (forceback) {
+ BufferCapabilities caps = peer.getBackBufferCaps();
+ if (caps instanceof ExtendedBufferCapabilities) {
+ ExtendedBufferCapabilities ebc =
+ (ExtendedBufferCapabilities)caps;
+ if (ebc.getVSync() == VSYNC_ON &&
+ ebc.getFlipContents() == COPIED)
+ {
+ createVSynced = true;
+ forceback = false;
+ }
+ }
+ }
}
if (forceback) {
// peer must be non-null in this case
- sData = WGLSurfaceData.createData(peer, vImg);
+ sData = WGLSurfaceData.createData(peer, vImg, FLIP_BACKBUFFER);
} else {
WGLGraphicsConfig gc =
(WGLGraphicsConfig)vImg.getGraphicsConfig();
ColorModel cm = gc.getColorModel(vImg.getTransparency());
- int type = gc.isCapPresent(OGLContext.CAPS_EXT_FBOBJECT) ?
- OGLSurfaceData.FBOBJECT : OGLSurfaceData.PBUFFER;
- sData = WGLSurfaceData.createData(gc,
- vImg.getWidth(),
- vImg.getHeight(),
- cm, vImg, type);
+ int type = vImg.getForcedAccelSurfaceType();
+ // if acceleration type is forced (type != UNDEFINED) then
+ // use the forced type, otherwise choose one based on caps
+ if (type == OGLSurfaceData.UNDEFINED) {
+ type = gc.isCapPresent(CAPS_EXT_FBOBJECT) ?
+ OGLSurfaceData.FBOBJECT : OGLSurfaceData.PBUFFER;
+ }
+ if (createVSynced) {
+ sData = WGLSurfaceData.createData(peer, vImg, type);
+ } else {
+ sData = WGLSurfaceData.createData(gc,
+ vImg.getWidth(),
+ vImg.getHeight(),
+ cm, vImg, type);
+ }
}
} catch (NullPointerException ex) {
sData = null;
@@ -104,7 +131,15 @@ public class WGLVolatileSurfaceManager
return sData;
}
+ @Override
protected boolean isConfigValid(GraphicsConfiguration gc) {
return ((gc == null) || (gc == vImg.getGraphicsConfig()));
}
+
+ @Override
+ public void initContents() {
+ if (vImg.getForcedAccelSurfaceType() != OGLSurfaceData.TEXTURE) {
+ super.initContents();
+ }
+ }
}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/DDBlitLoops.java b/jdk/src/windows/classes/sun/java2d/windows/DDBlitLoops.java
deleted file mode 100644
index 3bed85e72a6..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/DDBlitLoops.java
+++ /dev/null
@@ -1,186 +0,0 @@
-/*
- * Copyright 2000-2004 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import java.awt.AlphaComposite;
-import java.awt.Color;
-import java.awt.Composite;
-import java.awt.Font;
-import java.awt.image.ColorModel;
-import java.awt.image.WritableRaster;
-import java.awt.image.BufferedImage;
-import sun.java2d.loops.GraphicsPrimitive;
-import sun.java2d.loops.GraphicsPrimitiveMgr;
-import sun.java2d.loops.CompositeType;
-import sun.java2d.loops.SurfaceType;
-import sun.java2d.loops.Blit;
-import sun.java2d.loops.BlitBg;
-import sun.java2d.loops.FillRect;
-import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
-import sun.java2d.pipe.Region;
-import sun.awt.image.BufImgSurfaceData;
-
-/**
- * DDBlitLoops
- *
- * This class accelerates Blits between two DD surfaces of the same depth,
- * using DirectDraw commands in native code to dispatch Blt calls to
- * the video accelerator.
- */
-public class DDBlitLoops extends Blit {
-
- public static void register()
- {
- GraphicsPrimitive[] primitives = {
- // opaque to opaque loops
-
- // Note that we use the offscreen surfaceType for
- // this registry, but that both the onscreen and offscreen
- // DD types use the same DESC strings, so these loops will
- // be used whether the src/dest surfaces are onscreen or not
- new DDBlitLoops(Win32SurfaceData.IntRgbDD,
- Win32SurfaceData.IntRgbDD, false),
- new DDBlitLoops(Win32SurfaceData.Ushort565RgbDD,
- Win32SurfaceData.Ushort565RgbDD, false),
- new DDBlitLoops(Win32SurfaceData.IntRgbxDD,
- Win32SurfaceData.IntRgbxDD, false),
- new DDBlitLoops(Win32SurfaceData.Ushort555RgbxDD,
- Win32SurfaceData.Ushort555RgbxDD, false),
- new DDBlitLoops(Win32SurfaceData.Ushort555RgbDD,
- Win32SurfaceData.Ushort555RgbDD, false),
- new DDBlitLoops(Win32SurfaceData.ByteIndexedOpaqueDD,
- Win32SurfaceData.ByteIndexedOpaqueDD, false),
- new DDBlitLoops(Win32SurfaceData.ByteGrayDD,
- Win32SurfaceData.ByteGrayDD, false),
- new DDBlitLoops(Win32SurfaceData.Index8GrayDD,
- Win32SurfaceData.Index8GrayDD, false),
- new DDBlitLoops(Win32SurfaceData.ThreeByteBgrDD,
- Win32SurfaceData.ThreeByteBgrDD, false),
-
- // 1-bit transparent to opaque loops
- new DDBlitLoops(Win32SurfaceData.IntRgbDD_BM,
- Win32SurfaceData.IntRgbDD, true),
- new DDBlitLoops(Win32SurfaceData.Ushort565RgbDD_BM,
- Win32SurfaceData.Ushort565RgbDD, true),
- new DDBlitLoops(Win32SurfaceData.IntRgbxDD_BM,
- Win32SurfaceData.IntRgbxDD, true),
- new DDBlitLoops(Win32SurfaceData.Ushort555RgbxDD_BM,
- Win32SurfaceData.Ushort555RgbxDD, true),
- new DDBlitLoops(Win32SurfaceData.Ushort555RgbDD_BM,
- Win32SurfaceData.Ushort555RgbDD, true),
- new DDBlitLoops(Win32SurfaceData.ByteIndexedDD_BM,
- Win32SurfaceData.ByteIndexedOpaqueDD, true),
- new DDBlitLoops(Win32SurfaceData.ByteGrayDD_BM,
- Win32SurfaceData.ByteGrayDD, true),
- new DDBlitLoops(Win32SurfaceData.Index8GrayDD_BM,
- Win32SurfaceData.Index8GrayDD, true),
- new DDBlitLoops(Win32SurfaceData.ThreeByteBgrDD_BM,
- Win32SurfaceData.ThreeByteBgrDD, true),
-
- // any to 1-bit transparent bg loops
- new DelegateBlitBgLoop(Win32SurfaceData.IntRgbDD_BM,
- Win32SurfaceData.IntRgbDD),
- new DelegateBlitBgLoop(Win32SurfaceData.Ushort565RgbDD_BM,
- Win32SurfaceData.Ushort565RgbDD),
- new DelegateBlitBgLoop(Win32SurfaceData.IntRgbxDD_BM,
- Win32SurfaceData.IntRgbxDD),
- new DelegateBlitBgLoop(Win32SurfaceData.Ushort555RgbxDD_BM,
- Win32SurfaceData.Ushort555RgbxDD),
- new DelegateBlitBgLoop(Win32SurfaceData.Ushort555RgbDD_BM,
- Win32SurfaceData.Ushort555RgbDD),
- new DelegateBlitBgLoop(Win32SurfaceData.ByteIndexedDD_BM,
- Win32SurfaceData.ByteIndexedOpaqueDD),
- new DelegateBlitBgLoop(Win32SurfaceData.ByteGrayDD_BM,
- Win32SurfaceData.ByteGrayDD),
- new DelegateBlitBgLoop(Win32SurfaceData.Index8GrayDD_BM,
- Win32SurfaceData.Index8GrayDD),
- new DelegateBlitBgLoop(Win32SurfaceData.ThreeByteBgrDD_BM,
- Win32SurfaceData.ThreeByteBgrDD),
-
- };
- GraphicsPrimitiveMgr.register(primitives);
- }
-
- public DDBlitLoops(SurfaceType srcType, SurfaceType dstType, boolean over) {
- super(srcType,
- over ? CompositeType.SrcOverNoEa : CompositeType.SrcNoEa,
- dstType);
- }
-
- /**
- * Blit
- * This native method is where all of the work happens in the
- * accelerated Blit.
- */
- public native void Blit(SurfaceData src, SurfaceData dst,
- Composite comp, Region clip,
- int sx, int sy, int dx, int dy, int w, int h);
-
-
- /**
- * BlitBg
- * This loop is used to render from Sw surface data
- * to the Hw one in AOSI.copyBackupToAccelerated.
- */
- static class DelegateBlitBgLoop extends BlitBg {
- SurfaceType dstType;
- private static final Font defaultFont = new Font(Font.DIALOG, Font.PLAIN, 12);
-
- public DelegateBlitBgLoop(SurfaceType realDstType, SurfaceType delegateDstType) {
- super(SurfaceType.Any, CompositeType.SrcNoEa, realDstType);
- this.dstType = delegateDstType;
- }
-
- public void BlitBg(SurfaceData srcData, SurfaceData dstData,
- Composite comp, Region clip, Color bgColor,
- int srcx, int srcy, int dstx, int dsty, int width, int height)
- {
- ColorModel dstModel = dstData.getColorModel();
- WritableRaster wr =
- dstModel.createCompatibleWritableRaster(width, height);
- boolean isPremult = dstModel.isAlphaPremultiplied();
- BufferedImage bimg =
- new BufferedImage(dstModel, wr, isPremult, null);
- SurfaceData tmpData = BufImgSurfaceData.createData(bimg);
- SunGraphics2D sg2d = new SunGraphics2D(tmpData, bgColor, bgColor,
- defaultFont);
- FillRect fillop = FillRect.locate(SurfaceType.AnyColor,
- CompositeType.SrcNoEa,
- tmpData.getSurfaceType());
- Blit combineop = Blit.getFromCache(srcData.getSurfaceType(),
- CompositeType.SrcOverNoEa,
- tmpData.getSurfaceType());
- Blit blitop = Blit.getFromCache(tmpData.getSurfaceType(),
- CompositeType.SrcNoEa, dstType);
- fillop.FillRect(sg2d, tmpData, 0, 0, width, height);
- combineop.Blit(srcData, tmpData, AlphaComposite.SrcOver, null,
- srcx, srcy, 0, 0, width, height);
- blitop.Blit(tmpData, dstData, comp, clip,
- 0, 0, dstx, dsty, width, height);
- }
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/DDRenderer.java b/jdk/src/windows/classes/sun/java2d/windows/DDRenderer.java
deleted file mode 100644
index 4a376bf786b..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/DDRenderer.java
+++ /dev/null
@@ -1,368 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import java.awt.Composite;
-import java.awt.Shape;
-import java.awt.geom.Path2D;
-import java.awt.geom.PathIterator;
-import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
-import sun.java2d.pipe.Region;
-import sun.java2d.pipe.PixelDrawPipe;
-import sun.java2d.pipe.PixelFillPipe;
-import sun.java2d.pipe.ShapeDrawPipe;
-import sun.java2d.pipe.SpanIterator;
-import sun.java2d.loops.GraphicsPrimitive;
-
-/**
- * DDRenderer
- *
- * This class accelerates rendering to a surface of type
- * WinOffScreenSurfaceData. The renderers in here are simply java wrappers
- * around native methods that do the real work.
- */
-public class DDRenderer extends GDIRenderer {
-
- //
- // Native implementations
- //
- native void doDrawLineDD(SurfaceData sData,
- int color,
- int x1, int y1, int x2, int y2);
- native void doFillRectDD(SurfaceData sData,
- int color,
- int left, int top,
- int right, int bottom);
- native void doDrawRectDD(SurfaceData sData,
- int color,
- int x, int y, int w, int h);
-
- //
- // Internal Java methods for rendering
- //
-
- /**
- * Clip the given line to the clip bounds in sg2d.
- * Assume that the line passed in is given in the order of
- * x1 <= x2, y1 <= y2
- */
- private void clipAndDrawLine(SunGraphics2D sg2d,
- int x1, int y1, int x2, int y2)
- {
- // If any of these are true, the line lies outside of the
- // clip bounds
- Region clip = sg2d.getCompClip();
- int cx1 = clip.getLoX();
- int cy1 = clip.getLoY();
- int cx2 = clip.getHiX();
- int cy2 = clip.getHiY();
- // For each edge, clip the appropriate coordinate against
- // that edge. We are only dealing with horizontal or vertical lines
- // for now, so there is no interpolation between points to
- // the proper clip coordinate.
- if (x1 < cx1) x1 = cx1;
- if (y1 < cy1) y1 = cy1;
- if (x2 >= cx2) x2 = cx2 - 1;
- if (y2 >= cy2) y2 = cy2 - 1;
- // If the start moved past the end (or vice versa),
- // then we are outside the clip.
- if (x1 <= x2 && y1 <= y2) {
- doDrawLineDD(sg2d.surfaceData, sg2d.pixel, x1, y1, x2, y2);
- }
- }
-
- // REMIND: This is just a hack to get WIDE lines to honor the
- // necessary hinted pixelization rules. This should be replaced
- // by a native FillSpans method or a getHintedStrokeGeneralPath()
- // method that could be filled by the doShape method more quickly.
- public void doFillSpans(SunGraphics2D sg2d, SpanIterator si) {
- int box[] = new int[4];
- SurfaceData sd = sg2d.surfaceData;
- while (si.nextSpan(box)) {
- doFillRectDD(sd, sg2d.pixel, box[0], box[1], box[2], box[3]);
- }
- }
-
-
- //
- // Java wrappers for the primitive renderers
- //
-
- /**
- * drawLine draws a line between the pixel at x1, y1 and the
- * pixel at x2, y2 (including the last pixel).
- */
- public void drawLine(SunGraphics2D sg2d,
- int x1, int y1, int x2, int y2)
- {
- // Note that we only handle horizontal or vertical lines through
- // this renderer. This is because the implementation uses a fill
- // Blt through DirectDraw, which only works for rectangle shapes.
- if (x1 == x2 || y1 == y2) {
-
- int transx1 = x1 + sg2d.transX;
- int transy1 = y1 + sg2d.transY;
- int transx2 = x2 + sg2d.transX;
- int transy2 = y2 + sg2d.transY;
- int t;
- // First, set the ordering of the line coordinates;
- // clipAndDrawLine() expects x1 < x2 and y1 < y2
- if (transx1 > transx2) {
- t = transx1;
- transx1 = transx2;
- transx2 = t;
- }
- if (transy1 > transy2) {
- t = transy1;
- transy1 = transy2;
- transy2 = t;
- }
- clipAndDrawLine(sg2d, transx1, transy1, transx2, transy2);
- }
- else {
- // Punt to our superclass renderer to render diagonal lines
- super.drawLine(sg2d, x1, y1, x2, y2);
- }
- }
-
-
- /**
- * fillRect filles a rect from the pixel at x, y to (but not including)
- * the pixel at (x + width), (y + height)
- */
- public void fillRect(SunGraphics2D sg2d,
- int x, int y, int width, int height)
- {
- int clipLeft, clipTop, clipRight, clipBottom;
-
- // REMIND: This check should probably go in SunGraphics2D instead.
- if (width <= 0 || height <= 0) {
- return;
- }
-
- // Here we clip the fill rect to the size of the clip bounds in sg2d.
- // The native code can then assume that it receives a non-empty, post-
- // clipped primitive.
- clipLeft = x + sg2d.transX;
- clipTop = y + sg2d.transY;
- clipRight = clipLeft + width;
- clipBottom = clipTop + height;
-
- Region clip = sg2d.getCompClip();
-
- // Clip each edge of the rect to the appropriate edge of the clip
- // bounds.
- if (clipLeft < clip.getLoX()) clipLeft = clip.getLoX();
- if (clipTop < clip.getLoY()) clipTop = clip.getLoY();
- if (clipRight > clip.getHiX()) clipRight = clip.getHiX();
- if (clipBottom > clip.getHiY()) clipBottom = clip.getHiY();
-
- if (clipRight > clipLeft && clipBottom > clipTop) {
- doFillRectDD(sg2d.surfaceData, sg2d.pixel, clipLeft, clipTop,
- clipRight, clipBottom);
- }
- }
-
-
- /**
- * draw a rectangle outline starting at x, y and going to the pixel
- * at (x + width), (y + width) (including the lower right pixel)
- */
- public void drawRect(SunGraphics2D sg2d,
- int x, int y, int width, int height)
- {
- if (width < 2 || height < 2) {
- fillRect(sg2d, x, y, width+1, height+1);
- return;
- }
- int transx = x + sg2d.transX;
- int transy = y + sg2d.transY;
- Region clip = sg2d.getCompClip();
- if (!clip.encompassesXYWH(transx, transy, width+1, height+1)) {
- // Rect needs clipping - draw each edge separately, clipping
- // as we go.
- // Prefer longer horizontal lines if possible.
- clipAndDrawLine(sg2d, transx, transy,
- transx + width, transy);
- clipAndDrawLine(sg2d, transx, transy + 1,
- transx, transy + height - 1);
- clipAndDrawLine(sg2d, transx + width, transy + 1,
- transx + width, transy + height - 1);
- clipAndDrawLine(sg2d, transx, transy + height,
- transx + width, transy + height);
- } else {
- // No clipping needed - just call native method which draws
- // all edges in one method
- doDrawRectDD(sg2d.surfaceData, sg2d.pixel, transx, transy,
- width, height);
- }
- }
-
- @Override
- public native void devCopyArea(SurfaceData sData,
- int srcx, int srcy, int dx, int dy,
- int w, int h);
-
- public DDRenderer traceWrapDD() {
- return new Tracer();
- }
-
- public static class Tracer extends DDRenderer {
- void doDrawLine(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x1, int y1, int x2, int y2)
- {
- GraphicsPrimitive.tracePrimitive("GDIDrawLine");
- super.doDrawLine(sData, clip, comp, color, x1, y1, x2, y2);
- }
- void doDrawRect(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h)
- {
- GraphicsPrimitive.tracePrimitive("GDIDrawRect");
- super.doDrawRect(sData, clip, comp, color, x, y, w, h);
- }
- void doDrawRoundRect(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h,
- int arcW, int arcH)
- {
- GraphicsPrimitive.tracePrimitive("GDIDrawRoundRect");
- super.doDrawRoundRect(sData, clip, comp, color,
- x, y, w, h, arcW, arcH);
- }
- void doDrawOval(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h)
- {
- GraphicsPrimitive.tracePrimitive("GDIDrawOval");
- super.doDrawOval(sData, clip, comp, color, x, y, w, h);
- }
- void doDrawArc(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h,
- int angleStart, int angleExtent)
- {
- GraphicsPrimitive.tracePrimitive("GDIDrawArc");
- super.doDrawArc(sData, clip, comp, color, x, y, w, h,
- angleStart, angleExtent);
- }
- void doDrawPoly(SurfaceData sData,
- Region clip, Composite comp, int color,
- int transx, int transy,
- int[] xpoints, int[] ypoints,
- int npoints, boolean isclosed)
- {
- GraphicsPrimitive.tracePrimitive("GDIDrawPoly");
- super.doDrawPoly(sData, clip, comp, color, transx, transy,
- xpoints, ypoints, npoints, isclosed);
- }
- void doFillRect(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h)
- {
- GraphicsPrimitive.tracePrimitive("GDIFillRect");
- super.doFillRect(sData, clip, comp, color, x, y, w, h);
- }
- void doFillRoundRect(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h,
- int arcW, int arcH)
- {
- GraphicsPrimitive.tracePrimitive("GDIFillRoundRect");
- super.doFillRoundRect(sData, clip, comp, color,
- x, y, w, h, arcW, arcH);
- }
- void doFillOval(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h)
- {
- GraphicsPrimitive.tracePrimitive("GDIFillOval");
- super.doFillOval(sData, clip, comp, color, x, y, w, h);
- }
- void doFillArc(SurfaceData sData,
- Region clip, Composite comp, int color,
- int x, int y, int w, int h,
- int angleStart, int angleExtent)
- {
- GraphicsPrimitive.tracePrimitive("GDIFillArc");
- super.doFillArc(sData, clip, comp, color, x, y, w, h,
- angleStart, angleExtent);
- }
- void doFillPoly(SurfaceData sData,
- Region clip, Composite comp, int color,
- int transx, int transy,
- int[] xpoints, int[] ypoints,
- int npoints)
- {
- GraphicsPrimitive.tracePrimitive("GDIFillPoly");
- super.doFillPoly(sData, clip, comp, color, transx, transy,
- xpoints, ypoints, npoints);
- }
- void doShape(SurfaceData sData,
- Region clip, Composite comp, int color,
- int transX, int transY,
- Path2D.Float p2df, boolean isfill)
- {
- GraphicsPrimitive.tracePrimitive(isfill
- ? "GDIFillShape"
- : "GDIDrawShape");
- super.doShape(sData, clip, comp, color,
- transX, transY, p2df, isfill);
- }
- public void devCopyArea(SurfaceData sData,
- int srcx, int srcy,
- int dx, int dy,
- int w, int h)
- {
- GraphicsPrimitive.tracePrimitive("DXCopyArea");
- super.devCopyArea(sData, srcx, srcy, dx, dy, w, h);
- }
- void doDrawLineDD(SurfaceData sData,
- int color,
- int x1, int y1, int x2, int y2)
- {
- GraphicsPrimitive.tracePrimitive("DXDrawLine");
- super.doDrawLineDD(sData, color, x1, y1, x2, y2);
- }
- void doFillRectDD(SurfaceData sData,
- int color,
- int left, int top,
- int right, int bottom)
- {
- GraphicsPrimitive.tracePrimitive("DXFillRect");
- super.doFillRectDD(sData, color, left, top, right, bottom);
- }
- void doDrawRectDD(SurfaceData sData,
- int color,
- int x, int y, int w, int h)
- {
- GraphicsPrimitive.tracePrimitive("DXDrawRect");
- super.doDrawRectDD(sData, color, x, y, w, h);
- }
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/DDScaleLoops.java b/jdk/src/windows/classes/sun/java2d/windows/DDScaleLoops.java
deleted file mode 100644
index ccad3167e3f..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/DDScaleLoops.java
+++ /dev/null
@@ -1,114 +0,0 @@
-/*
- * Copyright 2001-2007 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import sun.java2d.loops.GraphicsPrimitive;
-import sun.java2d.loops.GraphicsPrimitiveMgr;
-import sun.java2d.loops.GraphicsPrimitiveProxy;
-import sun.java2d.loops.CompositeType;
-import sun.java2d.loops.SurfaceType;
-import sun.java2d.loops.ScaledBlit;
-import sun.java2d.pipe.Region;
-import sun.java2d.SurfaceData;
-import java.awt.Composite;
-
-/**
- * DDScaleLoops
- *
- * This class accelerates ScaledBlits between two DirectDraw surfaces. Since
- * the onscreen surface is of that type and some of the offscreen surfaces
- * may be of that type (if they were created in a SurfaceDataProxy), then
- * this type of ScaledBlit will accelerated double-buffer copies between those
- * two surfaces.
-*/
-public class DDScaleLoops extends ScaledBlit {
- private ScaledBlit swblit;
-
- public static void register()
- {
- GraphicsPrimitive[] primitives = {
- new DDScaleLoops(Win32SurfaceData.IntRgbDD),
- new DDScaleLoops(Win32SurfaceData.Ushort565RgbDD),
- new DDScaleLoops(Win32SurfaceData.IntRgbxDD),
- new DDScaleLoops(Win32SurfaceData.Ushort555RgbxDD),
- new DDScaleLoops(Win32SurfaceData.Ushort555RgbDD),
- new DDScaleLoops(Win32SurfaceData.ByteIndexedOpaqueDD),
- new DDScaleLoops(Win32SurfaceData.ThreeByteBgrDD)
- };
- GraphicsPrimitiveMgr.register(primitives);
- }
-
- public DDScaleLoops(SurfaceType surfType) {
- super(surfType, CompositeType.SrcNoEa, surfType);
- }
-
- /**
- * Scale
- * This native method is where all of the work happens in the
- * accelerated ScaledBlit for the scaling case.
- */
- public native void Scale(SurfaceData src, SurfaceData dst,
- Composite comp, int sx, int sy,
- int dx, int dy, int sw, int sh,
- int dw, int dh);
-
- public void Scale(SurfaceData src, SurfaceData dst,
- Composite comp, Region clip,
- int sx1, int sy1,
- int sx2, int sy2,
- double dx1, double dy1,
- double dx2, double dy2)
- {
- // REMIND: We can still do it if the clip equals the device
- // bounds for a destination window, but this logic rejects
- // that case...
- int dx = (int) Math.round(dx1);
- int dy = (int) Math.round(dy1);
- int dw = (int) Math.round(dx2) - dx;
- int dh = (int) Math.round(dy2) - dy;
- if (clip.encompassesXYWH(dx, dy, dw, dh)) {
- // Note that this rounding creates inaccuracies, but these
- // loops are disabled by default until a user specifically
- // enables them so this rounding behavior can be one of
- // the drawbacks that the user accepts when enabling this
- // non-standard feature.
- // If we should ever want to turn them on by default then
- // we will need to decide what better handling to put here.
- Scale(src, dst, comp, sx1, sy1, dx, dy, sx2-sx1, sy2-sy1, dw, dh);
- } else {
- if (swblit == null) {
- // REMIND: This assumes that the DD surface types are
- // directly derived from a non-DD type that has a loop.
- swblit = ScaledBlit.getFromCache(getSourceType().getSuperType(),
- getCompositeType(),
- getDestType().getSuperType());
- }
- swblit.Scale(src, dst, comp, clip,
- sx1, sy1, sx2, sy2,
- dx1, dy1, dx2, dy2);
- }
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/GDIBlitLoops.java b/jdk/src/windows/classes/sun/java2d/windows/GDIBlitLoops.java
index 023ebc0f46a..541a5a38bb9 100644
--- a/jdk/src/windows/classes/sun/java2d/windows/GDIBlitLoops.java
+++ b/jdk/src/windows/classes/sun/java2d/windows/GDIBlitLoops.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2004 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2002-2008 Sun Microsystems, Inc. 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
@@ -60,23 +60,23 @@ public class GDIBlitLoops extends Blit {
{
GraphicsPrimitive[] primitives = {
new GDIBlitLoops(SurfaceType.IntRgb,
- Win32SurfaceData.AnyGdi),
+ GDIWindowSurfaceData.AnyGdi),
new GDIBlitLoops(SurfaceType.Ushort555Rgb,
- Win32SurfaceData.AnyGdi,
+ GDIWindowSurfaceData.AnyGdi,
0x7C00, 0x03E0, 0x001F),
new GDIBlitLoops(SurfaceType.Ushort565Rgb,
- Win32SurfaceData.AnyGdi,
+ GDIWindowSurfaceData.AnyGdi,
0xF800, 0x07E0, 0x001F),
new GDIBlitLoops(SurfaceType.ThreeByteBgr,
- Win32SurfaceData.AnyGdi),
+ GDIWindowSurfaceData.AnyGdi),
new GDIBlitLoops(SurfaceType.ByteIndexedOpaque,
- Win32SurfaceData.AnyGdi,
+ GDIWindowSurfaceData.AnyGdi,
true),
new GDIBlitLoops(SurfaceType.Index8Gray,
- Win32SurfaceData.AnyGdi,
+ GDIWindowSurfaceData.AnyGdi,
true),
new GDIBlitLoops(SurfaceType.ByteGray,
- Win32SurfaceData.AnyGdi),
+ GDIWindowSurfaceData.AnyGdi),
};
GraphicsPrimitiveMgr.register(primitives);
}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/Win32SurfaceData.java b/jdk/src/windows/classes/sun/java2d/windows/GDIWindowSurfaceData.java
similarity index 59%
rename from jdk/src/windows/classes/sun/java2d/windows/Win32SurfaceData.java
rename to jdk/src/windows/classes/sun/java2d/windows/GDIWindowSurfaceData.java
index f62ad07870d..4d12fe680fc 100644
--- a/jdk/src/windows/classes/sun/java2d/windows/Win32SurfaceData.java
+++ b/jdk/src/windows/classes/sun/java2d/windows/GDIWindowSurfaceData.java
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -37,10 +37,10 @@ import java.awt.image.Raster;
import sun.awt.SunHints;
import sun.awt.Win32GraphicsConfig;
import sun.awt.Win32GraphicsDevice;
-import sun.awt.image.PixelConverter;
import sun.awt.windows.WComponentPeer;
import sun.awt.windows.WFileDialogPeer;
import sun.awt.windows.WPrintDialogPeer;
+import sun.java2d.ScreenUpdateManager;
import sun.java2d.SunGraphics2D;
import sun.java2d.SurfaceData;
import sun.java2d.SurfaceDataProxy;
@@ -52,8 +52,8 @@ import sun.java2d.loops.CompositeType;
import sun.java2d.loops.RenderLoops;
import sun.java2d.loops.XORComposite;
-public class Win32SurfaceData extends SurfaceData {
- WComponentPeer peer;
+public class GDIWindowSurfaceData extends SurfaceData {
+ private WComponentPeer peer;
private Win32GraphicsConfig graphicsConfig;
private RenderLoops solidloops;
@@ -61,69 +61,6 @@ public class Win32SurfaceData extends SurfaceData {
public static final String
DESC_GDI = "GDI";
- // DDraw offscreen surface type names
- public static final String
- DESC_INT_RGB_DD = "Integer RGB DirectDraw";
-
- public static final String
- DESC_INT_RGBx_DD = "Integer RGBx DirectDraw";
-
- public static final String
- DESC_USHORT_565_RGB_DD = "Short 565 RGB DirectDraw";
-
- public static final String
- DESC_USHORT_555_RGBx_DD = "Short 555 RGBx DirectDraw";
-
- public static final String
- DESC_USHORT_555_RGB_DD = "Short 555 RGB DirectDraw";
-
- public static final String
- DESC_BYTE_INDEXED_OPAQUE_DD
- = "8-bit Indexed (Opaque) DirectDraw";
-
- public static final String
- DESC_BYTE_GRAY_DD = "Byte Gray DirectDraw";
-
- public static final String
- DESC_INDEX8_GRAY_DD = "Index8 Gray DirectDraw";
-
- public static final String
- DESC_3BYTE_BGR_DD = "3 Byte BGR DirectDraw";
-
- // Surface types with 1-bit transparency
- public static final String
- DESC_INT_RGB_DD_BM = "Integer RGB DirectDraw with 1 bit transp";
-
- public static final String
- DESC_INT_RGBx_DD_BM = "Integer RGBx DirectDraw with 1 bit transp";
-
- public static final String
- DESC_USHORT_565_RGB_DD_BM
- = "Short 565 RGB DirectDraw with 1 bit transp";
-
- public static final String
- DESC_USHORT_555_RGBx_DD_BM
- = "Short 555 RGBx DirectDraw with 1 bit transp";
-
- public static final String
- DESC_USHORT_555_RGB_DD_BM
- = "Short 555 RGB DirectDraw with 1 bit transp";
-
- public static final String
- DESC_3BYTE_BGR_DD_BM = "3 Byte BGR DirectDraw with 1 bit transp";
-
- public static final String
- DESC_BYTE_INDEXED_DD_BM = "8-bit Indexed DirectDraw with 1 bit transp";
-
- public static final String
- DESC_BYTE_GRAY_DD_BM = "Byte Gray DirectDraw with 1 bit transp";
-
- public static final String
- DESC_INDEX8_GRAY_DD_BM = "Index8 Gray DirectDraw with 1 bit transp";
-
-
- // Gdi (screen) surface types
-
// Generic GDI surface type - used for registering all loops
public static final SurfaceType AnyGdi =
SurfaceType.IntRgb.deriveSubType(DESC_GDI);
@@ -140,97 +77,6 @@ public class Win32SurfaceData extends SurfaceData {
public static final SurfaceType ThreeByteBgrGdi =
SurfaceType.ThreeByteBgr.deriveSubType(DESC_GDI);
- // DDraw offscreen surface types
- public static final SurfaceType IntRgbDD =
- SurfaceType.IntRgb.deriveSubType(DESC_INT_RGB_DD);
-
- public static final SurfaceType IntRgbxDD =
- SurfaceType.IntRgbx.deriveSubType(DESC_INT_RGBx_DD);
-
- public static final SurfaceType Ushort565RgbDD =
- SurfaceType.Ushort565Rgb.deriveSubType(DESC_USHORT_565_RGB_DD);
-
- public static final SurfaceType Ushort555RgbxDD =
- SurfaceType.Ushort555Rgbx.deriveSubType(DESC_USHORT_555_RGBx_DD);
-
- public static final SurfaceType Ushort555RgbDD =
- SurfaceType.Ushort555Rgb.deriveSubType(DESC_USHORT_555_RGB_DD);
-
- public static final SurfaceType ByteIndexedOpaqueDD =
- SurfaceType.ByteIndexedOpaque.deriveSubType(DESC_BYTE_INDEXED_OPAQUE_DD);
-
- public static final SurfaceType ByteGrayDD =
- SurfaceType.ByteGray.deriveSubType(DESC_BYTE_GRAY_DD);
-
- public static final SurfaceType Index8GrayDD =
- SurfaceType.Index8Gray.deriveSubType(DESC_INDEX8_GRAY_DD);
-
- public static final SurfaceType ThreeByteBgrDD =
- SurfaceType.ThreeByteBgr.deriveSubType(DESC_3BYTE_BGR_DD);
-
- // DDraw onscreen surface types (derive from Gdi surfaces)
- public static final SurfaceType IntRgbDDscreen =
- IntRgbGdi.deriveSubType(DESC_INT_RGB_DD);
-
- public static final SurfaceType Ushort565RgbDDscreen =
- Ushort565RgbGdi.deriveSubType(DESC_USHORT_565_RGB_DD);
-
- public static final SurfaceType Ushort555RgbDDscreen =
- Ushort555RgbGdi.deriveSubType(DESC_USHORT_555_RGB_DD);
-
- public static final SurfaceType ThreeByteBgrDDscreen =
- ThreeByteBgrGdi.deriveSubType(DESC_3BYTE_BGR_DD);
-
- // These screen types will not be handled as GDI surfaces
- // (we can do dithering to 8-bit surfaces faster than
- // GDI, so do not use GDI Blits to indexed surfaces.
- // And Rgbx surfaces are documented to not work with
- // GDI, so do not use GDI for that surface type either)
- public static final SurfaceType IntRgbxDDscreen = IntRgbxDD;
-
- public static final SurfaceType Ushort555RgbxDDscreen = Ushort555RgbxDD;
-
- public static final SurfaceType ByteIndexedOpaqueDDscreen =
- ByteIndexedOpaqueDD;
-
- public static final SurfaceType ByteGrayDDscreen = ByteGrayDD;
-
- public static final SurfaceType Index8GrayDDscreen = Index8GrayDD;
-
- // Surface types with 1-bit transparency
- public static final SurfaceType IntRgbDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_INT_RGB_DD_BM,
- PixelConverter.Xrgb.instance);
-
- public static final SurfaceType IntRgbxDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_INT_RGBx_DD_BM,
- PixelConverter.Rgbx.instance);
-
- public static final SurfaceType Ushort565RgbDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_USHORT_565_RGB_DD_BM,
- PixelConverter.Ushort565Rgb.instance);
-
- public static final SurfaceType Ushort555RgbxDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_USHORT_555_RGBx_DD_BM,
- PixelConverter.Ushort555Rgbx.instance);
-
- public static final SurfaceType Ushort555RgbDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_USHORT_555_RGB_DD_BM,
- PixelConverter.Ushort555Rgb.instance);
-
- public static final SurfaceType ByteIndexedDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_BYTE_INDEXED_DD_BM);
-
- public static final SurfaceType ByteGrayDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_BYTE_GRAY_DD_BM);
-
- public static final SurfaceType Index8GrayDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_INDEX8_GRAY_DD_BM);
-
- public static final SurfaceType ThreeByteBgrDD_BM =
- SurfaceType.Custom.deriveSubType(DESC_3BYTE_BGR_DD_BM,
- PixelConverter.Xrgb.instance);
-
private static native void initIDs(Class xorComp);
static {
@@ -242,38 +88,37 @@ public class Win32SurfaceData extends SurfaceData {
}
public static SurfaceType getSurfaceType(ColorModel cm) {
- // REMIND: If ddraw not available, set sType to non-ddraw surface type
switch (cm.getPixelSize()) {
case 32:
case 24:
if (cm instanceof DirectColorModel) {
if (((DirectColorModel)cm).getRedMask() == 0xff0000) {
- return IntRgbDDscreen;
+ return IntRgbGdi;
} else {
- return IntRgbxDDscreen;
+ return SurfaceType.IntRgbx;
}
} else {
- return ThreeByteBgrDDscreen;
+ return ThreeByteBgrGdi;
}
case 15:
- return Ushort555RgbDDscreen;
+ return Ushort555RgbGdi;
case 16:
if ((cm instanceof DirectColorModel) &&
(((DirectColorModel)cm).getBlueMask() == 0x3e))
{
- return Ushort555RgbxDDscreen;
+ return SurfaceType.Ushort555Rgbx;
} else {
- return Ushort565RgbDDscreen;
+ return Ushort565RgbGdi;
}
case 8:
if (cm.getColorSpace().getType() == ColorSpace.TYPE_GRAY &&
cm instanceof ComponentColorModel) {
- return ByteGrayDDscreen;
+ return SurfaceType.ByteGray;
} else if (cm instanceof IndexColorModel &&
isOpaqueGray((IndexColorModel)cm)) {
- return Index8GrayDDscreen;
+ return SurfaceType.Index8Gray;
} else {
- return ByteIndexedOpaqueDDscreen;
+ return SurfaceType.ByteIndexedOpaque;
}
default:
throw new sun.java2d.InvalidPipeException("Unsupported bit " +
@@ -282,23 +127,16 @@ public class Win32SurfaceData extends SurfaceData {
}
}
+ public static GDIWindowSurfaceData createData(WComponentPeer peer) {
+ SurfaceType sType = getSurfaceType(peer.getDeviceColorModel());
+ return new GDIWindowSurfaceData(peer, sType);
+ }
+
@Override
public SurfaceDataProxy makeProxyFor(SurfaceData srcData) {
- // If D3D is enabled then we might have D3D capabilities, but
- // that pipeline is going away soon so we will not bother
- // creating the D3DProxy needed to manage those. For now we
- // will just use DDraw cached surfaces in all cases.
- return Win32SurfaceDataProxy.createProxy(srcData, graphicsConfig);
+ return SurfaceDataProxy.UNCACHED;
}
- public static Win32SurfaceData createData(WComponentPeer peer,
- int numBuffers)
- {
- SurfaceType sType = getSurfaceType(peer.getDeviceColorModel());
- return new Win32SurfaceData(peer, sType, numBuffers);
- }
-
-
public Raster getRaster(int x, int y, int w, int h) {
throw new InternalError("not implemented yet");
}
@@ -312,6 +150,7 @@ public class Win32SurfaceData extends SurfaceData {
gdiPipe = gdiPipe.traceWrap();
}
gdiTxPipe = new PixelToShapeConverter(gdiPipe);
+
}
public void validatePipe(SunGraphics2D sg2d) {
@@ -396,12 +235,9 @@ public class Win32SurfaceData extends SurfaceData {
* Initializes the native Ops pointer.
*/
private native void initOps(WComponentPeer peer, int depth, int redMask,
- int greenMask, int blueMask, int numBuffers,
- int screen);
+ int greenMask, int blueMask, int screen);
- public Win32SurfaceData(WComponentPeer peer, SurfaceType sType,
- int numBuffers)
- {
+ private GDIWindowSurfaceData(WComponentPeer peer, SurfaceType sType) {
super(sType, peer.getDeviceColorModel());
ColorModel cm = peer.getDeviceColorModel();
this.peer = peer;
@@ -433,7 +269,7 @@ public class Win32SurfaceData extends SurfaceData {
{
// REMIND: Awful hack. The right fix for this problem
// would be for these type of Peers to not even use a
- // Win32SurfaceData object since they never do any
+ // GDIWindowSurfaceData object since they never do any
// rendering. Or they could actually implement the
// functionality needed in initOps. But this seems
// to work for now. See bug 4391928 for more info.
@@ -441,12 +277,21 @@ public class Win32SurfaceData extends SurfaceData {
}
Win32GraphicsDevice gd =
(Win32GraphicsDevice)graphicsConfig.getDevice();
- initOps(peer, depth, rMask, gMask, bMask, numBuffers, gd.getScreen());
+ initOps(peer, depth, rMask, gMask, bMask, gd.getScreen());
setBlitProxyKey(graphicsConfig.getProxyKey());
}
+ /**
+ * {@inheritDoc}
+ *
+ * Overridden to use ScreenUpdateManager to obtain the replacement surface.
+ *
+ * @see sun.java2d.ScreenUpdateManager#getReplacementScreenSurface
+ */
+ @Override
public SurfaceData getReplacement() {
- return peer.getSurfaceData();
+ ScreenUpdateManager mgr = ScreenUpdateManager.getInstance();
+ return mgr.getReplacementScreenSurface(peer, this);
}
public Rectangle getBounds() {
@@ -486,6 +331,7 @@ public class Win32SurfaceData extends SurfaceData {
}
private native void invalidateSD();
+ @Override
public void invalidate() {
if (isValid()) {
invalidateSD();
@@ -494,14 +340,15 @@ public class Win32SurfaceData extends SurfaceData {
}
}
- // This gets called when restoring the back buffer
- public native void restoreSurface();
- public native void flip(SurfaceData data);
-
/**
* Returns destination Component associated with this SurfaceData.
*/
+ @Override
public Object getDestination() {
return peer.getTarget();
}
+
+ public WComponentPeer getPeer() {
+ return peer;
+ }
}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/Win32OffScreenSurfaceData.java b/jdk/src/windows/classes/sun/java2d/windows/Win32OffScreenSurfaceData.java
deleted file mode 100644
index a5e34ac34cc..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/Win32OffScreenSurfaceData.java
+++ /dev/null
@@ -1,459 +0,0 @@
-/*
- * Copyright 2000-2007 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-import java.awt.GraphicsConfiguration;
-import java.awt.Image;
-import java.awt.Rectangle;
-import java.awt.Transparency;
-import java.awt.color.ColorSpace;
-import java.awt.image.ColorModel;
-import java.awt.image.ComponentColorModel;
-import java.awt.image.DirectColorModel;
-import java.awt.image.IndexColorModel;
-import java.awt.image.Raster;
-
-import sun.awt.SunHints;
-import sun.awt.Win32GraphicsConfig;
-import sun.awt.Win32GraphicsDevice;
-import sun.awt.image.SurfaceManager;
-import sun.awt.image.SunVolatileImage;
-import sun.awt.image.WritableRasterNative;
-import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
-import sun.java2d.SurfaceDataProxy;
-import sun.java2d.loops.CompositeType;
-import sun.java2d.pipe.PixelToShapeConverter;
-import sun.java2d.loops.GraphicsPrimitive;
-import sun.java2d.loops.SurfaceType;
-import sun.java2d.loops.RenderLoops;
-import sun.java2d.pipe.Region;
-
-/**
- * Win32OffScreenSurfaceData
- *
- * This class implements a hardware-accelerated video memory surface. It uses
- * a custom renderer (DDRenderer) to render via DirectDraw into the
- * surface and uses a custom Blit loop (DDBlitLoops) to copy between
- * two hardware-accelerated surfaces (including the screen).
- */
-public class Win32OffScreenSurfaceData extends SurfaceData {
-
- protected int width;
- protected int height;
- protected int transparency;
-
- protected GraphicsConfiguration graphicsConfig;
- protected Image image;
- protected RenderLoops solidloops;
- private boolean ddSurfacePunted = false;
-
- private static native void initIDs();
-
- static {
- initIDs();
- // REMIND: This isn't really thought-out; if the user doesn't have or
- // doesn't want ddraw then we should not even have this surface type
- // in the loop
- if (WindowsFlags.isDDEnabled() && WindowsFlags.isDDOffscreenEnabled()) {
- if (WindowsFlags.isDDBlitEnabled()) {
- // Register out hardware-accelerated Blit loops
- DDBlitLoops.register();
- }
- if (WindowsFlags.isDDScaleEnabled()) {
- DDScaleLoops.register();
- }
- }
- }
-
- public static SurfaceType getSurfaceType(ColorModel cm, int transparency) {
- boolean transparent = (transparency == Transparency.BITMASK);
- switch (cm.getPixelSize()) {
- case 32:
- case 24:
- if (cm instanceof DirectColorModel) {
- if (((DirectColorModel)cm).getRedMask() == 0xff0000) {
- return transparent ? Win32SurfaceData.IntRgbDD_BM :
- Win32SurfaceData.IntRgbDD;
- } else {
- return transparent ? Win32SurfaceData.IntRgbxDD_BM :
- Win32SurfaceData.IntRgbxDD;
- }
- } else {
- return transparent ? Win32SurfaceData.ThreeByteBgrDD_BM :
- Win32SurfaceData.ThreeByteBgrDD;
- }
- case 15:
- return transparent ? Win32SurfaceData.Ushort555RgbDD_BM :
- Win32SurfaceData.Ushort555RgbDD;
- case 16:
- if ((cm instanceof DirectColorModel) &&
- (((DirectColorModel)cm).getBlueMask() == 0x3e))
- {
- return transparent ? Win32SurfaceData.Ushort555RgbxDD_BM :
- Win32SurfaceData.Ushort555RgbxDD;
- } else {
- return transparent ? Win32SurfaceData.Ushort565RgbDD_BM :
- Win32SurfaceData.Ushort565RgbDD;
- }
- case 8:
- if (cm.getColorSpace().getType() == ColorSpace.TYPE_GRAY &&
- cm instanceof ComponentColorModel) {
- return transparent ? Win32SurfaceData.ByteGrayDD_BM :
- Win32SurfaceData.ByteGrayDD;
- } else if (cm instanceof IndexColorModel &&
- isOpaqueGray((IndexColorModel)cm)) {
- return transparent ? Win32SurfaceData.Index8GrayDD_BM :
- Win32SurfaceData.Index8GrayDD;
- } else {
- return transparent ? Win32SurfaceData.ByteIndexedDD_BM :
- Win32SurfaceData.ByteIndexedOpaqueDD;
- }
- default:
- throw new sun.java2d.InvalidPipeException("Unsupported bit " +
- "depth: " +
- cm.getPixelSize());
- }
- }
-
- @Override
- public SurfaceDataProxy makeProxyFor(SurfaceData srcData) {
- Win32GraphicsConfig wgc = (Win32GraphicsConfig) graphicsConfig;
- return Win32SurfaceDataProxy.createProxy(srcData, wgc);
- }
-
- public static Win32OffScreenSurfaceData
- createData(int width, int height,
- ColorModel cm, Win32GraphicsConfig gc,
- Image image, int transparency)
- {
- // Win32OSD doesn't support acceleration of translucent images
- if (transparency == Transparency.TRANSLUCENT) {
- return null;
- }
-
-
- Win32GraphicsDevice gd = (Win32GraphicsDevice)gc.getDevice();
- if (!gd.isOffscreenAccelerationEnabled())
- {
- // If acceleration for this type of image is disabled on this
- // device, do not create an accelerated surface type
- return null;
- }
-
- return new Win32OffScreenSurfaceData(width, height,
- getSurfaceType(cm, transparency),
- cm, gc, image, transparency,
- gd.getScreen());
- }
-
- protected static DDRenderer ddPipe;
- protected static PixelToShapeConverter ddTxPipe;
-
- static {
- ddPipe = new DDRenderer();
- if (GraphicsPrimitive.tracingEnabled()) {
- ddPipe = ddPipe.traceWrapDD();
- }
- ddTxPipe = new PixelToShapeConverter(ddPipe);
- }
-
- public void validatePipe(SunGraphics2D sg2d) {
- if (sg2d.antialiasHint != SunHints.INTVAL_ANTIALIAS_ON &&
- sg2d.paintState <= sg2d.PAINT_ALPHACOLOR &&
- sg2d.compositeState <= sg2d.COMP_ISCOPY &&
- sg2d.clipState != sg2d.CLIP_SHAPE &&
- transparency != Transparency.TRANSLUCENT)
- {
- PixelToShapeConverter txPipe;
- DDRenderer nontxPipe;
- txPipe = ddTxPipe;
- nontxPipe = ddPipe;
- sg2d.imagepipe = imagepipe;
- if (sg2d.transformState >= sg2d.TRANSFORM_TRANSLATESCALE) {
- sg2d.drawpipe = txPipe;
- sg2d.fillpipe = txPipe;
- } else if (sg2d.strokeState != sg2d.STROKE_THIN){
- sg2d.drawpipe = txPipe;
- sg2d.fillpipe = nontxPipe;
- } else {
- sg2d.drawpipe = nontxPipe;
- sg2d.fillpipe = nontxPipe;
- }
- sg2d.shapepipe = nontxPipe;
- switch (sg2d.textAntialiasHint) {
-
- case SunHints.INTVAL_TEXT_ANTIALIAS_DEFAULT:
- /* equate DEFAULT to OFF which it is for us */
- case SunHints.INTVAL_TEXT_ANTIALIAS_OFF:
- sg2d.textpipe = solidTextRenderer;
- break;
-
- case SunHints.INTVAL_TEXT_ANTIALIAS_ON:
- sg2d.textpipe = aaTextRenderer;
- break;
-
- default:
- switch (sg2d.getFontInfo().aaHint) {
-
- case SunHints.INTVAL_TEXT_ANTIALIAS_LCD_HRGB:
- case SunHints.INTVAL_TEXT_ANTIALIAS_LCD_VRGB:
- sg2d.textpipe = lcdTextRenderer;
- break;
-
- case SunHints.INTVAL_TEXT_ANTIALIAS_ON:
- sg2d.textpipe = aaTextRenderer;
- break;
-
- default:
- sg2d.textpipe = solidTextRenderer;
- }
- }
- // This is needed for AA text.
- // Note that even a SolidTextRenderer can dispatch AA text
- // if a GlyphVector overrides the AA setting.
- sg2d.loops = solidloops;
- } else {
- super.validatePipe(sg2d);
- }
- }
-
- public static boolean isDDScaleEnabled() {
- return WindowsFlags.isDDScaleEnabled();
- }
-
- private WritableRasterNative wrn = null;
- public synchronized Raster getRaster(int x, int y, int w, int h) {
- if (wrn == null) {
- wrn = WritableRasterNative.createNativeRaster(getColorModel(),
- this,
- width, height);
- if (wrn == null) {
- throw new InternalError("Unable to create native raster");
- }
- }
-
- return wrn;
- }
-
- public RenderLoops getRenderLoops(SunGraphics2D sg2d) {
- if (sg2d.paintState <= sg2d.PAINT_ALPHACOLOR &&
- sg2d.compositeState <= sg2d.COMP_ISCOPY)
- {
- return solidloops;
- }
- return super.getRenderLoops(sg2d);
- }
-
- public GraphicsConfiguration getDeviceConfiguration() {
- return graphicsConfig;
- }
-
- /**
- * Initializes the native Ops pointer.
- */
- private native void initOps(int depth, int transparency);
-
- /**
- * This native method creates the offscreen surface in video memory and
- * (if necessary) initializes DirectDraw
- */
- private native void initSurface(int depth, int width, int height,
- int screen,
- boolean isVolatile,
- int transparency);
-
- public native void restoreSurface();
-
- /**
- * Non-public constructor. Use createData() to create an object.
- *
- * This constructor is used to house the common construction
- * code shared between the creation of Win32OSSD objects
- * and subclasses of Win32OSSD (such as D3DSurfaceData
- * and WinBackBufferSurfaceData).
- *
- * It calls the common constructor in the parent, and then
- * initializes other shared Win32 data.
- */
- protected Win32OffScreenSurfaceData(int width, int height,
- SurfaceType sType, ColorModel cm,
- GraphicsConfiguration gc,
- Image image, int transparency)
- {
- super(sType, cm);
- this.width = width;
- this.height = height;
- this.graphicsConfig = gc;
- this.image = image;
- this.transparency = transparency;
- this.solidloops =
- ((Win32GraphicsConfig)graphicsConfig).getSolidLoops(sType);
- initOps(cm.getPixelSize(), transparency);
- }
-
- /**
- * Private constructor. Use createData() to create an object.
- *
- * This constructor calls the common constructor above and then
- * performs the specific initialization of the Win32Surface.
- */
- private Win32OffScreenSurfaceData(int width, int height,
- SurfaceType sType, ColorModel cm,
- Win32GraphicsConfig gc,
- Image image, int transparency,
- int screen)
- {
- this(width, height, sType, cm, gc, image, transparency);
- initSurface(cm.getPixelSize(), width, height, screen,
- (image instanceof SunVolatileImage), transparency);
- setBlitProxyKey(gc.getProxyKey());
- }
-
- /**
- * Need this since the surface data is created with
- * the color model of the target GC, which is always
- * opaque. But in SunGraphics2D.blitSD we choose loops
- * based on the transparency on the source SD, so
- * we could choose wrong loop (blit instead of blitbg,
- * for example, which will cause problems in transparent
- * case).
- */
- public int getTransparency() {
- return transparency;
- }
-
- /**
- * When someone asks for a new surface data, we punt to our
- * container image which will attempt to restore the contents
- * of this surface or, failing that, will return null.
- */
- public SurfaceData getReplacement() {
- return restoreContents(image);
- }
-
- public Rectangle getBounds() {
- return new Rectangle(width, height);
- }
-
- protected native void nativeInvalidate();
-
- public void invalidate() {
- if (isValid()) {
- synchronized (this) {
- wrn = null;
- }
- nativeInvalidate();
- super.invalidate();
- }
- }
-
- public native void setTransparentPixel(int pixel);
-
- public native void flush();
-
- /**
- * Returns true if the native representation of this image has been
- * moved into ddraw system memory. This happens when many reads
- * or read-modify-write operations are requested of that surface.
- * If we have moved that surface into system memory, we should note that
- * here so that someone wanting to copy something to this surface will
- * take that into account during that copy.
- */
- public boolean surfacePunted() {
- return ddSurfacePunted;
- }
-
- protected void markSurfaceLost() {
- synchronized (this) {
- wrn = null;
- }
- setSurfaceLost(true);
- if (image != null) {
- // Inform the Volatile that it lost its accelerated surface
- SurfaceManager sMgr = SurfaceManager.getManager(image);
- sMgr.acceleratedSurfaceLost();
- }
- }
-
- /**
- * This method is called from the native code if an unrecoverable
- * error has been detected.
- *
- * Marks the surface lost, and notifies the surface manager
- * that the DirectDraw acceleration for the corresponding image
- * should be disabled.
- */
- protected void disableDD() {
- markSurfaceLost();
- if (image != null) {
- SurfaceManager sMgr = SurfaceManager.getManager(image);
- // REMIND: yes, this is not pretty; the accelerationEnabled property
- // should be pulled up to SurfaceManager some day.
- if (sMgr instanceof WinVolatileSurfaceManager) {
- ((WinVolatileSurfaceManager)sMgr).setAccelerationEnabled(false);
- }
- }
- setBlitProxyKey(null);
- }
-
- /**
- * Returns destination Image associated with this SurfaceData.
- */
- public Object getDestination() {
- return image;
- }
-
- @Override
- public boolean copyArea(SunGraphics2D sg2d,
- int x, int y, int w, int h, int dx, int dy)
- {
- CompositeType comptype = sg2d.imageComp;
- if (sg2d.transformState < sg2d.TRANSFORM_TRANSLATESCALE &&
- sg2d.clipState != sg2d.CLIP_SHAPE &&
- (CompositeType.SrcOverNoEa.equals(comptype) ||
- CompositeType.SrcNoEa.equals(comptype)))
- {
- x += sg2d.transX;
- y += sg2d.transY;
- int dstx1 = x + dx;
- int dsty1 = y + dy;
- int dstx2 = dstx1 + w;
- int dsty2 = dsty1 + h;
- Region clip = sg2d.getCompClip();
- if (dstx1 < clip.getLoX()) dstx1 = clip.getLoX();
- if (dsty1 < clip.getLoY()) dsty1 = clip.getLoY();
- if (dstx2 > clip.getHiX()) dstx2 = clip.getHiX();
- if (dsty2 > clip.getHiY()) dsty2 = clip.getHiY();
- if (dstx1 < dstx2 && dsty1 < dsty2) {
- ddPipe.devCopyArea(this, dstx1 - dx, dsty1 - dy,
- dx, dy,
- dstx2 - dstx1, dsty2 - dsty1);
- }
- return true;
- }
- return false;
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/Win32SurfaceDataProxy.java b/jdk/src/windows/classes/sun/java2d/windows/Win32SurfaceDataProxy.java
deleted file mode 100644
index d631584cdcc..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/Win32SurfaceDataProxy.java
+++ /dev/null
@@ -1,434 +0,0 @@
-/*
- * Copyright 2007 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import java.awt.Color;
-import java.awt.Transparency;
-import java.awt.Rectangle;
-import java.awt.image.ColorModel;
-import java.awt.image.IndexColorModel;
-import java.awt.image.DirectColorModel;
-import java.awt.image.BufferedImage;
-import java.awt.image.DataBuffer;
-import java.awt.image.DataBufferInt;
-
-import sun.awt.Win32GraphicsConfig;
-import sun.awt.Win32GraphicsDevice;
-import sun.awt.image.BufImgSurfaceData;
-import sun.awt.image.SunWritableRaster;
-import sun.java2d.SurfaceData;
-import sun.java2d.SurfaceDataProxy;
-import sun.java2d.SunGraphics2D;
-import sun.java2d.StateTracker;
-import sun.java2d.InvalidPipeException;
-import sun.java2d.loops.CompositeType;
-
-/**
- * The proxy class contains the logic for when to replace a
- * SurfaceData with a cached X11 Pixmap and the code to create
- * the accelerated surfaces.
- */
-public abstract class Win32SurfaceDataProxy extends SurfaceDataProxy {
- /**
- * Represents the maximum size (width * height) of an image that we should
- * scan for an unused color. Any image larger than this would probably
- * require too much computation time.
- */
- private static final int MAX_SIZE = 65536;
-
- public static SurfaceDataProxy createProxy(SurfaceData srcData,
- Win32GraphicsConfig dstConfig)
- {
- Win32GraphicsDevice wgd =
- (Win32GraphicsDevice) dstConfig.getDevice();
- if (!wgd.isDDEnabledOnDevice() ||
- srcData instanceof Win32SurfaceData ||
- srcData instanceof Win32OffScreenSurfaceData)
- {
- // If they are not on the same screen then we could cache the
- // blit by returning an instance of Opaque below, but this
- // only happens for VolatileImage blits to the wrong screen
- // which we make no promises on so we just punt to UNCACHED...
- return UNCACHED;
- }
-
- ColorModel srcCM = srcData.getColorModel();
- int srcTransparency = srcCM.getTransparency();
-
- if (srcTransparency == Transparency.OPAQUE) {
- return new Opaque(dstConfig);
- } else if (srcTransparency == Transparency.BITMASK) {
- if (Bitmask.isCompatible(srcCM, srcData)) {
- return new Bitmask(dstConfig);
- }
- }
-
- return UNCACHED;
- }
-
- int srcTransparency;
- Win32GraphicsConfig wgc;
-
- public Win32SurfaceDataProxy(Win32GraphicsConfig wgc,
- int srcTransparency)
- {
- this.wgc = wgc;
- this.srcTransparency = srcTransparency;
- activateDisplayListener();
- }
-
- @Override
- public SurfaceData validateSurfaceData(SurfaceData srcData,
- SurfaceData cachedData,
- int w, int h)
- {
- if (cachedData == null ||
- !cachedData.isValid() ||
- cachedData.isSurfaceLost())
- {
- // use the device's color model for ddraw surfaces
- ColorModel dstScreenCM = wgc.getDeviceColorModel();
- try {
- cachedData =
- Win32OffScreenSurfaceData.createData(w, h,
- dstScreenCM,
- wgc, null,
- srcTransparency);
- } catch (InvalidPipeException e) {
- Win32GraphicsDevice wgd = (Win32GraphicsDevice) wgc.getDevice();
- if (!wgd.isDDEnabledOnDevice()) {
- invalidate();
- flush();
- return null;
- }
- }
- }
- return cachedData;
- }
-
- /**
- * Proxy for opaque source images.
- */
- public static class Opaque extends Win32SurfaceDataProxy {
- static int TXMAX =
- (WindowsFlags.isDDScaleEnabled()
- ? SunGraphics2D.TRANSFORM_TRANSLATESCALE
- : SunGraphics2D.TRANSFORM_ANY_TRANSLATE);
-
- public Opaque(Win32GraphicsConfig wgc) {
- super(wgc, Transparency.OPAQUE);
- }
-
- @Override
- public boolean isSupportedOperation(SurfaceData srcData,
- int txtype,
- CompositeType comp,
- Color bgColor)
- {
- // we save a read from video memory for compositing
- // operations by copying from the buffered image sd
- return (txtype <= TXMAX &&
- (CompositeType.SrcOverNoEa.equals(comp) ||
- CompositeType.SrcNoEa.equals(comp)));
- }
- }
-
- /**
- * Proxy for bitmask transparent source images.
- * This proxy can accelerate unscaled SrcOver copies with no bgColor.
- *
- * Note that this proxy plays some games with returning the srcData
- * from the validate method. It needs to do this since the conditions
- * for caching an accelerated copy depend on many factors that can
- * change over time, including:
- *
- * - the depth of the display
- * - the availability of a transparent pixel
- */
- public static class Bitmask extends Win32SurfaceDataProxy {
- /**
- * Tests a source image ColorModel and SurfaceData to
- * see if they are of an appropriate size and type to
- * perform our transparent pixel searches.
- *
- * Note that some dynamic factors may occur which prevent
- * us from finding or using a transparent pixel. These
- * are detailed above in the class comments. We do not
- * test those conditions here, but rely on the Bitmask
- * proxy to verify those conditions on the fly.
- */
- public static boolean isCompatible(ColorModel srcCM,
- SurfaceData srcData)
- {
- if (srcCM instanceof IndexColorModel) {
- return true;
- } else if (srcCM instanceof DirectColorModel) {
- return isCompatibleDCM((DirectColorModel) srcCM, srcData);
- }
-
- return false;
- }
-
- /**
- * Tests a given DirectColorModel to make sure it is
- * compatible with the assumptions we make when scanning
- * a DCM image for a transparent pixel.
- */
- public static boolean isCompatibleDCM(DirectColorModel dcm,
- SurfaceData srcData)
- {
- // The BISD restriction is because we need to
- // examine the pixels to find a tranparent color
- if (!(srcData instanceof BufImgSurfaceData)) {
- return false;
- }
-
- // The size restriction prevents us from wasting too
- // much time scanning large images for unused pixel values.
- Rectangle bounds = srcData.getBounds();
- // Using division instead of multiplication avoids overflow
- if (bounds.width <= 0 ||
- MAX_SIZE / bounds.width < bounds.height)
- {
- return false;
- }
-
- // Below we use the pixels from the data buffer to map
- // directly to pixel values using the dstData.pixelFor()
- // method so the pixel format must be compatible with
- // ARGB or we could end up with bad results. We assume
- // here that the destination is opaque and so only the
- // red, green, and blue masks matter.
- // These new checks for RGB masks are more correct,
- // but potentially reject the acceleration of some images
- // that we used to allow just because we cannot prove
- // that they will work OK. If we ever had an INT_BGR
- // image for instance, would that have really failed here?
- // 565 and 555 screens will both keep equal numbers of
- // bits of red and blue, but will differ in the amount of
- // green they keep so INT_BGR might be safe, but if anyone
- // ever created an INT_RBG image then 555 and 565 might
- // differ in whether they thought a transparent pixel
- // was available. Also, are there any other strange
- // screen formats where bizarre orderings of the RGB
- // would cause the tests below to make mistakes?
- return ((dcm.getPixelSize() == 25) &&
- (dcm.getTransferType() == DataBuffer.TYPE_INT) &&
- (dcm.getRedMask() == 0x00ff0000) &&
- (dcm.getGreenMask() == 0x0000ff00) &&
- (dcm.getBlueMask() == 0x000000ff));
- }
-
- int transPixel;
- Color transColor;
-
- // The real accelerated surface - only used when we can find
- // a transparent color.
- SurfaceData accelData;
-
- public Bitmask(Win32GraphicsConfig wgc) {
- super(wgc, Transparency.BITMASK);
- }
-
- @Override
- public boolean isSupportedOperation(SurfaceData srcData,
- int txtype,
- CompositeType comp,
- Color bgColor)
- {
- // We have accelerated loops only for blits with SrcOverNoEa
- // (no blit bg loops or blit loops with SrcNoEa)
- return (CompositeType.SrcOverNoEa.equals(comp) &&
- bgColor == null &&
- txtype < SunGraphics2D.TRANSFORM_TRANSLATESCALE);
- }
-
- /**
- * Note that every time we update the surface we may or may
- * not find a transparent pixel depending on what was modified
- * in the source image since the last time we looked.
- * Our validation method saves the accelerated surface aside
- * in a different field so we can switch back and forth between
- * the accelerated version and null depending on whether we
- * find a transparent pixel.
- * Note that we also override getRetryTracker() and return a
- * tracker that tracks the source pixels so that we do not
- * try to revalidate until there are new pixels to be scanned.
- */
- @Override
- public SurfaceData validateSurfaceData(SurfaceData srcData,
- SurfaceData cachedData,
- int w, int h)
- {
- // Evaluate the dest screen pixel size every time
- ColorModel dstScreenCM = wgc.getDeviceColorModel();
- if (dstScreenCM.getPixelSize() <= 8) {
- return null;
- }
- accelData = super.validateSurfaceData(srcData, accelData, w, h);
- return (accelData != null &&
- findTransparentPixel(srcData, accelData))
- ? accelData
- : null;
- }
-
- @Override
- public StateTracker getRetryTracker(SurfaceData srcData) {
- // If we failed to validate, it is permanent until the
- // next change to srcData...
- return srcData.getStateTracker();
- }
-
- @Override
- public void updateSurfaceData(SurfaceData srcData,
- SurfaceData dstData,
- int w, int h)
- {
- updateSurfaceDataBg(srcData, dstData, w, h, transColor);
- }
-
- /**
- * Invoked when the cached surface should be dropped.
- * Overrides the base class implementation so we can invalidate
- * the accelData field instead of the cachedSD field.
- */
- @Override
- public synchronized void flush() {
- SurfaceData accelData = this.accelData;
- if (accelData != null) {
- this.accelData = null;
- accelData.flush();
- }
- super.flush();
- }
-
- /**
- * The following constants determine the size of the histograms
- * used when searching for an unused color
- */
- private static final int ICM_HISTOGRAM_SIZE = 256;
- private static final int ICM_HISTOGRAM_MASK = ICM_HISTOGRAM_SIZE - 1;
- private static final int DCM_HISTOGRAM_SIZE = 1024;
- private static final int DCM_HISTOGRAM_MASK = DCM_HISTOGRAM_SIZE - 1;
-
- /**
- * Attempts to find an unused pixel value in the image and if
- * successful, sets up the DirectDraw surface so that it uses
- * this value as its color key.
- */
- public boolean findTransparentPixel(SurfaceData srcData,
- SurfaceData accelData)
- {
- ColorModel srcCM = srcData.getColorModel();
- boolean success = false;
-
- if (srcCM instanceof IndexColorModel) {
- success = findUnusedPixelICM((IndexColorModel) srcCM,
- accelData);
- } else if (srcCM instanceof DirectColorModel) {
- success = findUnusedPixelDCM((BufImgSurfaceData) srcData,
- accelData);
- }
-
- if (success) {
- int rgb = accelData.rgbFor(transPixel);
- transColor = new Color(rgb);
- Win32OffScreenSurfaceData wossd =
- (Win32OffScreenSurfaceData) accelData;
- wossd.setTransparentPixel(transPixel);
- } else {
- transColor = null;
- }
- return success;
- }
-
- /**
- * Attempts to find an unused pixel value in the color map of an
- * IndexColorModel. If successful, it returns that value (in the
- * ColorModel of the destination surface) or null otherwise.
- */
- private boolean findUnusedPixelICM(IndexColorModel icm,
- SurfaceData accelData) {
- int mapsize = icm.getMapSize();
- int[] histogram = new int[ICM_HISTOGRAM_SIZE];
- int[] cmap = new int[mapsize];
- icm.getRGBs(cmap);
-
- // load up the histogram
- for (int i = 0; i < mapsize; i++) {
- int pixel = accelData.pixelFor(cmap[i]);
- histogram[pixel & ICM_HISTOGRAM_MASK]++;
- }
-
- // find an empty histo-bucket
- for (int j = 0; j < histogram.length; j++) {
- if (histogram[j] == 0) {
- transPixel = j;
- return true;
- }
- }
-
- return false;
- }
-
- /**
- * Attempts to find an unused pixel value in an image with a
- * 25-bit DirectColorModel and a DataBuffer of TYPE_INT.
- * If successful, it returns that value (in the ColorModel
- * of the destination surface) or null otherwise.
- */
- private boolean findUnusedPixelDCM(BufImgSurfaceData bisd,
- SurfaceData accelData)
- {
- BufferedImage bimg = (BufferedImage) bisd.getDestination();
- DataBufferInt db =
- (DataBufferInt) bimg.getRaster().getDataBuffer();
- int[] pixels = SunWritableRaster.stealData(db, 0);
- int[] histogram = new int[DCM_HISTOGRAM_SIZE];
-
- // load up the histogram
- // REMIND: we could possibly make this faster by keeping track
- // of the unique colors found, and only doing a pixelFor()
- // when we come across a new unique color
- // REMIND: We are assuming pixels are in ARGB format. Is that
- // a safe assumption here?
- for (int i = 0; i < pixels.length; i++) {
- int pixel = accelData.pixelFor(pixels[i]);
- histogram[pixel & DCM_HISTOGRAM_MASK]++;
- }
-
- // find an empty histo-bucket
- for (int j = 0; j < histogram.length; j++) {
- if (histogram[j] == 0) {
- transPixel = j;
- return true;
- }
- }
-
- return false;
- }
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/WinBackBuffer.java b/jdk/src/windows/classes/sun/java2d/windows/WinBackBuffer.java
deleted file mode 100644
index dac4eb5b5f7..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/WinBackBuffer.java
+++ /dev/null
@@ -1,101 +0,0 @@
-/*
- * Copyright 2000-2007 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import java.awt.Component;
-import java.awt.GraphicsConfiguration;
-import java.awt.ImageCapabilities;
-import java.awt.image.ColorModel;
-import java.awt.image.VolatileImage;
-import sun.awt.image.SurfaceManager;
-import sun.awt.image.SunVolatileImage;
-import sun.awt.image.VolatileSurfaceManager;
-import sun.java2d.SurfaceData;
-
-import sun.java2d.d3d.D3DBackBufferSurfaceData;
-
-public class WinBackBuffer extends SunVolatileImage {
-
- /**
- * Create an image for an attached surface
- */
- public WinBackBuffer(Component c, Win32SurfaceData parentData) {
- super(c, c.getWidth(), c.getHeight(), parentData);
- }
-
- @Override
- protected VolatileSurfaceManager createSurfaceManager(Object context,
- ImageCapabilities caps)
- {
- return new WinBackBufferSurfaceManager(this, context);
- }
-
- public Win32OffScreenSurfaceData getHWSurfaceData() {
- SurfaceData sd = SurfaceData.getPrimarySurfaceData(this);
- return (sd instanceof Win32OffScreenSurfaceData) ?
- (Win32OffScreenSurfaceData)sd : null;
- }
-
- private class WinBackBufferSurfaceManager
- extends WinVolatileSurfaceManager
- {
- public WinBackBufferSurfaceManager(SunVolatileImage vImg,
- Object context)
- {
- super(vImg, context);
- }
-
- protected Win32OffScreenSurfaceData createAccelSurface() {
- GraphicsConfiguration gc = vImg.getGraphicsConfig();
- ColorModel cm = getDeviceColorModel();
- Win32SurfaceData parent = (Win32SurfaceData)context;
-
- Win32OffScreenSurfaceData ret =
- D3DBackBufferSurfaceData.createData(vImg.getWidth(),
- vImg.getHeight(),
- cm, gc, vImg, parent);
- if (ret == null) {
- ret = WinBackBufferSurfaceData.createData(vImg.getWidth(),
- vImg.getHeight(),
- cm, gc, vImg, parent);
- }
- return ret;
- }
-
- /**
- * Removes this surface manager from the display change listeners.
- * Since the user don't have access to the VolatileImage
- * representing the backbuffer, we know that nobody but us
- * can call it. And we do it when the backbuffer is replaced.
- */
- public void flush() {
- sun.awt.Win32GraphicsEnvironment ge =
- (sun.awt.Win32GraphicsEnvironment)
- java.awt.GraphicsEnvironment.getLocalGraphicsEnvironment();
- ge.removeDisplayChangedListener(this);
- }
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/WinBackBufferSurfaceData.java b/jdk/src/windows/classes/sun/java2d/windows/WinBackBufferSurfaceData.java
deleted file mode 100644
index 9220c96e100..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/WinBackBufferSurfaceData.java
+++ /dev/null
@@ -1,71 +0,0 @@
-/*
- * Copyright 2000-2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import java.awt.GraphicsConfiguration;
-import java.awt.Image;
-import java.awt.Transparency;
-import java.awt.image.ColorModel;
-import sun.awt.Win32GraphicsDevice;
-import sun.java2d.loops.SurfaceType;
-
-public class WinBackBufferSurfaceData extends Win32OffScreenSurfaceData {
-
- private Win32SurfaceData parentData;
-
- /**
- * Private constructor. Use createData() to create an object.
- */
- private WinBackBufferSurfaceData(int width, int height,
- SurfaceType sType, ColorModel cm,
- GraphicsConfiguration gc,
- Image image, int screen,
- Win32SurfaceData parentData)
- {
- super(width, height, sType, cm, gc, image, Transparency.OPAQUE);
- this.parentData = parentData;
- initSurface(cm.getPixelSize(), width, height, screen, parentData);
- }
-
- private native void initSurface(int depth, int width, int height,
- int screen, Win32SurfaceData parentData);
-
- public void restoreSurface() {
- parentData.restoreSurface();
- }
-
- public static WinBackBufferSurfaceData
- createData(int width, int height,
- ColorModel cm, GraphicsConfiguration gc, Image image,
- Win32SurfaceData parentData)
- {
- Win32GraphicsDevice gd = (Win32GraphicsDevice)gc.getDevice();
- SurfaceType sType = getSurfaceType(cm, Transparency.OPAQUE);
- return new WinBackBufferSurfaceData(width, height, sType,
- cm, gc, image,
- gd.getScreen(), parentData);
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/WinVolatileSurfaceManager.java b/jdk/src/windows/classes/sun/java2d/windows/WinVolatileSurfaceManager.java
deleted file mode 100644
index 2ad49d7f5b9..00000000000
--- a/jdk/src/windows/classes/sun/java2d/windows/WinVolatileSurfaceManager.java
+++ /dev/null
@@ -1,218 +0,0 @@
-/*
- * Copyright 2000-2007 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-package sun.java2d.windows;
-
-import java.awt.Color;
-import java.awt.Component;
-import java.awt.GraphicsConfiguration;
-import java.awt.GraphicsEnvironment;
-import java.awt.ImageCapabilities;
-import java.awt.Transparency;
-import java.awt.image.ColorModel;
-import sun.awt.DisplayChangedListener;
-import sun.awt.Win32GraphicsConfig;
-import sun.awt.Win32GraphicsDevice;
-import sun.awt.Win32GraphicsEnvironment;
-import sun.awt.image.SunVolatileImage;
-import sun.awt.image.VolatileSurfaceManager;
-import sun.java2d.SunGraphics2D;
-import sun.java2d.SurfaceData;
-import sun.java2d.d3d.D3DSurfaceData;
-
-/**
- * Windows platform implementation of the VolatileSurfaceManager class.
- * Th superclass implementation handles the case of surface loss due
- * to displayChange or other events. This class attempts to create
- * and use a hardware-based SurfaceData object (Win32OffScreenSurfaceData).
- * If this object cannot be created or re-created as necessary, the
- * class falls back to a software-based SurfaceData object
- * (BufImgSurfaceData) that will be used until the hardware-based
- * SurfaceData can be restored.
- */
-public class WinVolatileSurfaceManager
- extends VolatileSurfaceManager
-{
- private boolean accelerationEnabled;
-
- /**
- * Controls whether the manager should attempt to create a
- * D3DSurfaceData to accelerate the image.
- *
- * The default is the value of accelerationEnabled, but the value could
- * change during the life of this SurfaceManager.
- */
- private boolean d3dAccelerationEnabled;
-
- public WinVolatileSurfaceManager(SunVolatileImage vImg, Object context) {
- super(vImg, context);
-
- /* We enable acceleration only if all of the following are true:
- * - ddraw is enabled
- * - ddraw offscreen surfaces are enabled
- * - Either:
- * - the image is opaque OR
- * - the image is translucent and translucency acceleration
- * is enabled on this device
- * There is no acceleration for bitmask images yet because the
- * process to convert transparent pixels into ddraw colorkey
- * values is not worth the effort and time. We should eventually
- * accelerate transparent images the same way we do translucent
- * ones; through translucent textures (transparent pixels would
- * simply have an alpha of 0).
- */
- Win32GraphicsDevice gd =
- (Win32GraphicsDevice)vImg.getGraphicsConfig().getDevice();
- accelerationEnabled =
- WindowsFlags.isDDEnabled() &&
- WindowsFlags.isDDOffscreenEnabled() &&
- (vImg.getTransparency() == Transparency.OPAQUE);
- // REMIND: we don't really accelerate non-opaque VIs yet,
- // since we'll need RTT for that
-// ||
-// ((vImg.getTransparency() == Transparency.TRANSLUCENT) &&
-// WindowsFlags.isTranslucentAccelerationEnabled() &&
-// gd.isD3DEnabledOnDevice()));
-
- d3dAccelerationEnabled = accelerationEnabled;
- }
-
- protected SurfaceData createAccelSurface() {
- int transparency = vImg.getTransparency();
- ColorModel cm;
- Win32GraphicsConfig gc = (Win32GraphicsConfig) vImg.getGraphicsConfig();
- if (transparency != Transparency.TRANSLUCENT) {
- // REMIND: This will change when we accelerate bitmask VImages.
- // Currently, we can only reach here if the image is either
- // opaque or translucent
- cm = getDeviceColorModel();
- } else {
- cm = gc.getColorModel(Transparency.TRANSLUCENT);
- }
-
- // createData will return null if the device doesnt support d3d surfaces
- SurfaceData ret = null;
- // avoid pulling in D3D classes unless d3d is enabled on the device
- if (d3dAccelerationEnabled &&
- ((Win32GraphicsDevice)gc.getDevice()).isD3DEnabledOnDevice())
- {
- try {
- ret =
- D3DSurfaceData.createData(vImg.getWidth(), vImg.getHeight(),
- D3DSurfaceData.D3D_PLAIN_SURFACE,
- cm, gc, vImg);
- } catch (sun.java2d.InvalidPipeException e) {
- // exception is ignored, ret will be null so code
- // below will create a non-d3d surface
- }
- }
-
- if (ret == null) {
- ret = Win32OffScreenSurfaceData.createData(vImg.getWidth(),
- vImg.getHeight(),
- cm, gc, vImg,
- transparency);
- }
- return ret;
- }
-
- public boolean isAccelerationEnabled() {
- return accelerationEnabled;
- }
-
- /**
- *
- * @param enabled if true, enable both DirectDraw and Direct3D
- * acceleration for this surface manager, disable both if false
- */
- public void setAccelerationEnabled(boolean enabled) {
- if (enabled != accelerationEnabled) {
- sdCurrent = getBackupSurface();
- sdAccel = null;
- accelerationEnabled = enabled;
- }
- d3dAccelerationEnabled = enabled;
- }
-
- /**
- * Controls whether this surface manager should attempt to accelerate
- * the image using the Direct3D pipeline.
- *
- * If the state changes, sdCurrent will be reset to a backup surface,
- * and sdAccel will be nulled out so that a new surface is created
- * during the following validation.
- *
- * @param enabled if true, enable d3d acceleration for this SM,
- * disable otherwise.
- */
- public void setD3DAccelerationEnabled(boolean enabled) {
- if (enabled != d3dAccelerationEnabled) {
- sdCurrent = getBackupSurface();
- sdAccel = null;
- d3dAccelerationEnabled = enabled;
- }
- }
-
-
- /**
- * Create a vram-based SurfaceData object
- */
- public sun.java2d.SurfaceData initAcceleratedSurface() {
- SurfaceData sData;
-
- try {
- sData = createAccelSurface();
- } catch (sun.java2d.InvalidPipeException e) {
- // Problems during creation. Don't propagate the exception, just
- // set the hardware surface data to null; the software surface
- // data will be used in the meantime
- sData = null;
- }
- return sData;
- }
-
- /**
- * Called from Win32OffScreenSurfaceData to notify us that our
- * accelerated surface has been lost.
- */
- public SurfaceData restoreContents() {
- acceleratedSurfaceLost();
- return super.restoreContents();
- }
-
- protected ColorModel getDeviceColorModel() {
- Win32GraphicsConfig gc = (Win32GraphicsConfig)vImg.getGraphicsConfig();
- return gc.getDeviceColorModel();
- }
-
- /**
- * Called from superclass to force restoration of this surface
- * during the validation process. The method calls into the
- * hardware SurfaceData object to force the restore.
- */
- protected void restoreAcceleratedSurface() {
- ((Win32OffScreenSurfaceData)sdAccel).restoreSurface();
- }
-}
diff --git a/jdk/src/windows/classes/sun/java2d/windows/WindowsFlags.java b/jdk/src/windows/classes/sun/java2d/windows/WindowsFlags.java
index 8a6162286d7..05dad9c377d 100644
--- a/jdk/src/windows/classes/sun/java2d/windows/WindowsFlags.java
+++ b/jdk/src/windows/classes/sun/java2d/windows/WindowsFlags.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -39,71 +39,24 @@ public class WindowsFlags {
* has forced it on with d3d=true. These associated variables have
* the same base (eg, d3d) but end in "Set" (eg, d3dEnabled and
* d3dSet).
- * ddEnabled: usage: "-Dsun.java2d.noddraw[=true]"
- * turns off all usage of ddraw, including surface -> surface
- * Blts (including onscreen->onscreen copyarea), offscreen
- * surface creation, and surface locking via DDraw.
- * ddOffscreenEnabled: usage: "-Dsun.java2d.ddoffscreen=false"
- * disables the use of ddraw surfaces for offscreen
- * images. Effectively disables use of ddraw for most
- * operations except onscreen scrolling and some locking.
- * ddVramForced: usage: "-Dsun.java2d.ddforcevram=true"
- * Disables punts of offscreen ddraw surfaces to ddraw
- * system memory. We use a punting mechanism when we detect
- * a high proportion of expensive read operations on a ddraw
- * surface; this flag disables that mechanism and forces the
- * surfaces to remain in VRAM regardless.
- * ddLockEnabled: usage: "-Dsun.java2d.ddlock=[true|false]"
- * forces on|off usage of DirectDraw for locking the
- * screen. This feature is usually enabled by default
- * for pre-Win2k OS's and disabled by default for
- * Win2k and future OS's (as of jdk1.4.1).
+ * ddEnabled: usage: "-Dsun.java2d.noddraw[=false|true]"
+ * turns on/off all usage of Direct3D
+ * ddOffscreenEnabled: equivalent of sun.java2d.noddraw
* gdiBlitEnabled: usage: "-Dsun.java2d.gdiblit=false"
* turns off Blit loops that use GDI for copying to
* the screen from certain image types. Copies will,
* instead, happen via ddraw locking or temporary GDI DIB
* creation/copying (depending on OS and other flags)
- * ddBlitEnabled: usage: "-Dsun.java2d.ddblit=false"
- * turns off Blit loops that use DDraw for copying to
- * the screen from other ddraw surfaces. Copies will use
- * fallback mechanisms of GDI blits or ddraw locks, as
- * appropriate. This flag is primarily for debugging
- * purposes, to force our copies through a different code
- * path.
- * ddScaleEnabled: usage: "-Dsun.java2d.ddscale=true"
- * Turns on hardware-accelerated iamge scaling via ddraw.
- * This is off by default because we cannot guarantee the
- * quality of the scaling; hardware may choose to do
- * filtered or unfiltered scales, resulting in inconsistent
- * scaling between ddraw-accelerated and java2D-rendered
- * operations. This flag and capability should go away
- * someday as we eventually should use Direct3D for any
- * scaling operation (where we can control the filtering
- * used).
* d3dEnabled: usage: "-Dsun.java2d.d3d=[true|false]"
* Forces our use of Direct3D on or off. Direct3D is on
* by default, but may be disabled in some situations, such
- * as when running on Itanium, or on a card with bad d3d line
- * quality, or on a video card that we have had bad experience
- * with (e.g., Trident). This flag can force us to use d3d
+ * as on a card with bad d3d line quality, or on a video card
+ * that we have had bad experience with (e.g., Trident).
+ * This flag can force us to use d3d
* anyway in these situations. Or, this flag can force us to
* not use d3d in a situation where we would use it otherwise.
* translAccelEnabled: usage: "-Dsun.java2d.translaccel=true"
- * Turns on hardware acceleration for some translucent
- * image copying via Direct3D. Images that are created with
- * GraphicsConfiguration.createCompatibleImage(w, h, trans)
- * may be acceleratable by use a Direct3D texture and
- * performing copying operations to other DirectX-based
- * image destinations via a textured quad. This capability
- * is disabled by default pending further testing and fixing
- * some minor bugs (such as the ability to render Direct3D
- * to the screen which completely ignores the desktop
- * clip list). When this capability is turned on by default,
- * this flag should go away. Note: currently, enabling this
- * flag also enables the ddVramForced flag. This is because
- * d3d translucency acceleration can only happen to offscreen
- * surfaces which have not been punted through the means that
- * ddVramForced disables.
+ * equivalent to sun.java2d.d3d=true
* offscreenSharingEnabled: usage: "-Dsun.java2d.offscreenSharing=true"
* Turns on the ability to share a hardware-accelerated
* offscreen surface through the JAWT interface. See
@@ -114,15 +67,6 @@ public class WindowsFlags {
* without being very sure that we will be willing to support
* that API in the future regardless of other native
* rendering pipeline changes.
- * d3dTexBpp: usage: "-Dsun.java2d.d3dtexbpp=[16|32]
- * When translucent image acceleration is enabled (see
- * translAccelEnabled above), this flag specifies the bit
- * depth of the software (BufferedImage) and hardware
- * (textures) that we should use. The default is to use
- * 32 bit images and textures, but specifying a value of 16
- * for this flag will force us to use a color model of 4444
- * and 16-bit textures. This can be useful for applications
- * with heavy requirements on constrained VRAM resources.
* accelReset: usage: "-Dsun.java2d.accelReset"
* This flag tells us to reset any persistent information
* the display device acceleration characteristics so that
@@ -169,28 +113,19 @@ public class WindowsFlags {
*
*/
- private static boolean ddEnabled;
- private static boolean ddSet;
- private static boolean ddOffscreenEnabled;
- private static boolean ddVramForced;
- private static boolean ddLockEnabled;
- private static boolean ddLockSet;
private static boolean gdiBlitEnabled;
- private static boolean ddBlitEnabled;
- private static boolean ddScaleEnabled;
private static boolean d3dEnabled;
private static boolean d3dVerbose;
private static boolean d3dSet;
+ private static boolean d3dOnScreenEnabled;
private static boolean oglEnabled;
private static boolean oglVerbose;
- private static boolean translAccelEnabled;
private static boolean offscreenSharingEnabled;
private static boolean accelReset;
private static boolean checkRegistry;
private static boolean disableRegistry;
private static boolean magPresent;
private static boolean setHighDPIAware;
- private static int d3dTexBpp;
private static String javaVersion;
// TODO: other flags, including nopixfmt
@@ -271,20 +206,18 @@ public class WindowsFlags {
public Object run() {
magPresent = getBooleanProp(
"javax.accessibility.screen_magnifier_present", false);
- ddEnabled = !getBooleanProp("sun.java2d.noddraw", magPresent);
- ddSet = getPropertySet("sun.java2d.noddraw");
- ddOffscreenEnabled = getBooleanProp("sun.java2d.ddoffscreen",
- !magPresent);
- ddVramForced = getBooleanProp("sun.java2d.ddforcevram", false);
- ddLockEnabled = getBooleanProp("sun.java2d.ddlock", false);
- ddBlitEnabled = getBooleanProp("sun.java2d.ddblit", !magPresent);
- ddScaleEnabled = getBooleanProp("sun.java2d.ddscale", false);
- d3dEnabled = getBooleanProp("sun.java2d.d3d", !magPresent);
+ boolean ddEnabled =
+ !getBooleanProp("sun.java2d.noddraw", magPresent);
+ boolean ddOffscreenEnabled =
+ getBooleanProp("sun.java2d.ddoffscreen", ddEnabled);
+ d3dEnabled = getBooleanProp("sun.java2d.d3d",
+ ddEnabled && ddOffscreenEnabled);
+ d3dOnScreenEnabled =
+ getBooleanProp("sun.java2d.d3d.onscreen", d3dEnabled);
oglEnabled = getBooleanProp("sun.java2d.opengl", false);
if (oglEnabled) {
oglVerbose = isBooleanPropTrueVerbose("sun.java2d.opengl");
if (WGLGraphicsConfig.isWGLAvailable()) {
- ddEnabled = false;
d3dEnabled = false;
} else {
if (oglVerbose) {
@@ -300,18 +233,6 @@ public class WindowsFlags {
if (d3dSet) {
d3dVerbose = isBooleanPropTrueVerbose("sun.java2d.d3d");
}
- translAccelEnabled =
- getBooleanProp("sun.java2d.translaccel", false);
- if (translAccelEnabled) {
- // translucency only accelerated to un-punted buffers
- ddVramForced = true;
- // since they've requested translucency acceleration,
- // we assume they'll be happy with d3d quality
- if (!d3dSet && !magPresent) {
- d3dEnabled = true;
- d3dSet = true;
- }
- }
offscreenSharingEnabled =
getBooleanProp("sun.java2d.offscreenSharing", false);
accelReset = getBooleanProp("sun.java2d.accelReset", false);
@@ -330,9 +251,6 @@ public class WindowsFlags {
javaVersion = javaVersion.substring(0, dashIndex);
}
}
- d3dTexBpp = getIntProp("sun.java2d.d3dtexbpp", 32);
- ddLockSet = getPropertySet("sun.java2d.ddlock");
-
String dpiOverride = System.getProperty("sun.java2d.dpiaware");
if (dpiOverride != null) {
setHighDPIAware = dpiOverride.equalsIgnoreCase("true");
@@ -351,16 +269,6 @@ public class WindowsFlags {
"releases and applications that depend on it " +
"may not work correctly");
}
- if (translAccelEnabled) {
- System.out.println(
- "Acceleration for translucent images is enabled.");
- }
- if (!ddBlitEnabled) {
- System.out.println("DirectDraw Blits disabled");
- }
- if (ddScaleEnabled) {
- System.out.println("DirectDraw Scaling enabled");
- }
*/
return null;
}
@@ -388,34 +296,6 @@ public class WindowsFlags {
*/
}
- public static boolean isDDEnabled() {
- return ddEnabled;
- }
-
- public static boolean isDDOffscreenEnabled() {
- return ddOffscreenEnabled;
- }
-
- public static boolean isDDVramForced() {
- return ddVramForced;
- }
-
- public static boolean isDDLockEnabled() {
- return ddLockEnabled;
- }
-
- public static boolean isDDLockSet() {
- return ddLockSet;
- }
-
- public static boolean isDDBlitEnabled() {
- return ddBlitEnabled;
- }
-
- public static boolean isDDScaleEnabled() {
- return ddScaleEnabled;
- }
-
public static boolean isD3DEnabled() {
return d3dEnabled;
}
@@ -424,12 +304,12 @@ public class WindowsFlags {
return d3dSet;
}
- public static boolean isD3DVerbose() {
- return d3dVerbose;
+ public static boolean isD3DOnScreenEnabled() {
+ return d3dOnScreenEnabled;
}
- public static int getD3DTexBpp() {
- return d3dTexBpp;
+ public static boolean isD3DVerbose() {
+ return d3dVerbose;
}
public static boolean isGdiBlitEnabled() {
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DBadHardware.h b/jdk/src/windows/native/sun/java2d/d3d/D3DBadHardware.h
new file mode 100644
index 00000000000..52629d1a5d0
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DBadHardware.h
@@ -0,0 +1,122 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DBADHARDWARE_H
+#define D3DBADHARDWARE_H
+
+#include "D3DPipeline.h"
+#include "D3DPipelineManager.h"
+
+typedef struct ADAPTER_INFO {
+ DWORD VendorId;
+ DWORD DeviceId;
+ LONGLONG DriverVersion; // minimum driver version to pass, or NO_VERSION
+ USHORT OsInfo; // OSes where the DriverVersion is relevant or, OS_ALL
+} ADAPTER_INFO;
+
+// this DeviceId means that all vendor boards are to be excluded
+#define ALL_DEVICEIDS (0xffffffff)
+
+#define D_VERSION(H1, H2, L1, L2) \
+ (((LONGLONG)((H1 << 16) | H2) << 32) | ((L1 << 16) | (L2)))
+
+// this driver version is used to pass the driver version check
+// as it is always greater than any driver version
+#define MAX_VERSION D_VERSION(0x7fff, 0x7fff, 0x7fff, 0x7fff)
+// this DriverVersion means that the version of the driver doesn't matter,
+// all versions must fail ("there's no version of the driver that passes")
+#define NO_VERSION D_VERSION(0xffff, 0xffff, 0xffff, 0xffff)
+
+static const ADAPTER_INFO badHardware[] = {
+
+ // any Intel chip
+ // Reason: workaround for 6620073, 6612195, 6620073
+ { 0x8086, ALL_DEVICEIDS, NO_VERSION, OS_ALL },
+
+ // ATI Mobility Radeon X1600, X1400, X1450, X1300, X1350
+ // Reason: workaround for 6613066, 6687166
+ // X1300 (four sub ids)
+ { 0x1002, 0x714A, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x714A, D_VERSION(7,14,10,0567), OS_VISTA },
+ { 0x1002, 0x7149, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x7149, D_VERSION(7,14,10,0567), OS_VISTA },
+ { 0x1002, 0x714B, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x714B, D_VERSION(7,14,10,0567), OS_VISTA },
+ { 0x1002, 0x714C, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x714C, D_VERSION(7,14,10,0567), OS_VISTA },
+ // X1350 (three sub ids)
+ { 0x1002, 0x718B, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x718B, D_VERSION(7,14,10,0567), OS_VISTA },
+ { 0x1002, 0x718C, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x718C, D_VERSION(7,14,10,0567), OS_VISTA },
+ { 0x1002, 0x7196, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x7196, D_VERSION(7,14,10,0567), OS_VISTA },
+ // X1400
+ { 0x1002, 0x7145, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x7145, D_VERSION(7,14,10,0567), OS_VISTA },
+ // X1450 (two sub ids)
+ { 0x1002, 0x7186, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x7186, D_VERSION(7,14,10,0567), OS_VISTA },
+ { 0x1002, 0x718D, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x718D, D_VERSION(7,14,10,0567), OS_VISTA },
+ // X1600
+ { 0x1002, 0x71C5, D_VERSION(6,14,10,6706), OS_WINXP },
+ { 0x1002, 0x71C5, D_VERSION(7,14,10,0567), OS_VISTA },
+
+ // Nvidia Quadro NVS 110M
+ // Reason: workaround for 6629891
+ { 0x10DE, 0x01D7, D_VERSION(6,14,11,5665), OS_WINXP },
+
+ // Nvidia Quadro PCI-E series
+ // Reason: workaround for 6653860
+ { 0x10DE, 0x00FD, D_VERSION(6,14,10,6573), OS_WINXP },
+
+ // Nvidia GeForce 6200 TurboCache(TM)
+ // Reason: workaround for 6588384
+ { 0x10DE, 0x0161, NO_VERSION, OS_VISTA },
+
+ // any Matrox board
+ // Reason: there are no known Matrox boards with proper Direct3D support
+ { 0x102B, ALL_DEVICEIDS, NO_VERSION, OS_ALL },
+
+ // any SiS board
+ // Reason: there aren't many PS2.0-capable SiS boards and they weren't
+ // tested
+ { 0x1039, ALL_DEVICEIDS, NO_VERSION, OS_ALL },
+
+ // any S3 board
+ // Reason: no available S3 Chrome (the only S3 boards with PS2.0 support)
+ // for testing
+ { 0x5333, ALL_DEVICEIDS, NO_VERSION, OS_ALL },
+
+ // any S3 board (in VIA motherboards)
+ // Reason: These are S3 chips in VIA motherboards
+ { 0x1106, ALL_DEVICEIDS, NO_VERSION, OS_ALL },
+
+ // last record must be empty
+ { 0x0000, 0x0000, NO_VERSION, OS_ALL }
+};
+
+#endif // D3DBADHARDWARE_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.cpp
index e3e208a77df..6792a444a7d 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.cpp
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -23,116 +23,1092 @@
* have any questions.
*/
-#include
#include
#include "jlong.h"
-#include
-#include "ddrawUtils.h"
-#include "GraphicsPrimitiveMgr.h"
-#include "Region.h"
-#include "D3DUtils.h"
-#include "D3DContext.h"
+#include "D3DPipeline.h"
+
+#include "SurfaceData.h"
+#include "D3DBlitLoops.h"
+#include "D3DRenderQueue.h"
#include "D3DSurfaceData.h"
+#include "GraphicsPrimitiveMgr.h"
-extern CriticalSection windowMoveLock;
+#include "IntArgb.h"
+#include "IntArgbPre.h"
+#include "IntRgb.h"
+#include "IntBgr.h"
+#include "Ushort555Rgb.h"
+#include "Ushort565Rgb.h"
+#include "ByteIndexed.h"
-extern "C" {
+
+extern "C" BlitFunc IntArgbToIntArgbPreConvert;
+extern "C" BlitFunc IntArgbPreToIntArgbConvert;
+extern "C" BlitFunc IntArgbBmToIntArgbConvert;
+extern "C" BlitFunc IntRgbToIntArgbConvert;
+extern "C" BlitFunc Ushort565RgbToIntArgbConvert;
+extern "C" BlitFunc Ushort555RgbToIntArgbConvert;
+extern "C" BlitFunc IntBgrToIntArgbConvert;
+extern "C" BlitFunc AnyIntIsomorphicCopy;
+extern "C" BlitFunc ByteIndexedToIntArgbConvert;
+extern "C" BlitFunc ByteIndexedToIntArgbPreConvert;
+
+#define GETMIN(v1, v2) (((v1) > (t=(v2))) && ((v1) = t))
+#define GETMAX(v1, v2) (((v1) < (t=(v2))) && ((v1) = t))
+
+#ifdef D3D_PPL_DLL
JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DBlitLoops_doTransform
- (JNIEnv *env, jclass d3dbl,
- jlong pSrcData, jlong pDstData,
- jlong pCtx,
- jint hint,
- jint sx1, jint sy1, jint sx2, jint sy2,
- jfloat dx1, jfloat dy1, jfloat dx2, jfloat dy2)
+SurfaceData_IntersectBounds(SurfaceDataBounds *dst, SurfaceDataBounds *src)
{
- static J2DLVERTEX quadVerts[4] = {
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f }
- };
+ int t;
+ GETMAX(dst->x1, src->x1);
+ GETMAX(dst->y1, src->y1);
+ GETMIN(dst->x2, src->x2);
+ GETMIN(dst->y2, src->y2);
+}
- J2dTraceLn(J2D_TRACE_INFO, "D3DBlitLoops_doTransform");
- J2dTraceLn4(J2D_TRACE_VERBOSE, " sx1=%-4d sy1=%-4d sx2=%-4d sy2=%-4d ",
- sx1, sy1, sx2, sy2);
+JNIEXPORT void JNICALL
+SurfaceData_IntersectBoundsXYXY(SurfaceDataBounds *bounds,
+ jint x1, jint y1, jint x2, jint y2)
+{
+ int t;
+ GETMAX(bounds->x1, x1);
+ GETMAX(bounds->y1, y1);
+ GETMIN(bounds->x2, x2);
+ GETMIN(bounds->y2, y2);
+}
+
+JNIEXPORT void JNICALL
+SurfaceData_IntersectBoundsXYWH(SurfaceDataBounds *bounds,
+ jint x, jint y, jint w, jint h)
+{
+ w = (w <= 0) ? x : x+w;
+ if (w < x) {
+ w = 0x7fffffff;
+ }
+ if (bounds->x1 < x) {
+ bounds->x1 = x;
+ }
+ if (bounds->x2 > w) {
+ bounds->x2 = w;
+ }
+ h = (h <= 0) ? y : y+h;
+ if (h < y) {
+ h = 0x7fffffff;
+ }
+ if (bounds->y1 < y) {
+ bounds->y1 = y;
+ }
+ if (bounds->y2 > h) {
+ bounds->y2 = h;
+ }
+}
+
+JNIEXPORT void JNICALL
+SurfaceData_IntersectBlitBounds(SurfaceDataBounds *src,
+ SurfaceDataBounds *dst,
+ jint dx, jint dy)
+{
+ int t;
+ GETMAX(dst->x1, src->x1 + dx);
+ GETMAX(dst->y1, src->y1 + dy);
+ GETMIN(dst->x2, src->x2 + dx);
+ GETMIN(dst->y2, src->y2 + dy);
+ GETMAX(src->x1, dst->x1 - dx);
+ GETMAX(src->y1, dst->y1 - dy);
+ GETMIN(src->x2, dst->x2 - dx);
+ GETMIN(src->y2, dst->y2 - dy);
+}
+
+#endif /* D3D_PPL_DLL */
+
+D3DPIPELINE_API HRESULT
+D3DBL_CopySurfaceToIntArgbImage(IDirect3DSurface9 *pSurface,
+ SurfaceDataRasInfo *pDstInfo,
+ jint srcx, jint srcy,
+ jint srcWidth, jint srcHeight,
+ jint dstx, jint dsty)
+{
+ HRESULT res = S_OK;
+ D3DLOCKED_RECT lockedRect;
+ RECT r = { srcx, srcy, srcx+srcWidth, srcy+srcHeight };
+ D3DSURFACE_DESC desc;
+ SurfaceDataRasInfo srcInfo;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBL_CopySurfaceToIntArgbImage");
J2dTraceLn4(J2D_TRACE_VERBOSE,
- " dx1=%4f dy1=%4f dx2=%4f dy2=%4f", dx1, dy1, dx2, dy2);
+ " rect={%-4d, %-4d, %-4d, %-4d}",
+ r.left, r.top, r.right, r.bottom);
- if (sx2 <= sx1 || sy2 <= sy1 || dx2 <= dx1 || dy2 <= dy1) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "D3DBlitLoops_doTransform: invalid dimensions");
- return;
+ res = pSurface->LockRect(&lockedRect, &r, D3DLOCK_NOSYSLOCK);
+ RETURN_STATUS_IF_FAILED(res);
+ pSurface->GetDesc(&desc);
+
+ ZeroMemory(&srcInfo, sizeof(SurfaceDataRasInfo));
+ // srcInfo.bounds.x1 = 0;
+ // srcInfo.bounds.y1 = 0;
+ srcInfo.bounds.x2 = srcWidth;
+ srcInfo.bounds.y2 = srcHeight;
+ srcInfo.scanStride = lockedRect.Pitch;
+
+ void *pSrcBase = lockedRect.pBits;
+ void *pDstBase = PtrCoord(pDstInfo->rasBase,
+ dstx, pDstInfo->pixelStride,
+ dsty, pDstInfo->scanStride);
+
+ switch (desc.Format) {
+ case D3DFMT_A8R8G8B8:
+ srcInfo.pixelStride = 4;
+ IntArgbPreToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ &srcInfo, pDstInfo, NULL, NULL);
+ break;
+ case D3DFMT_X8R8G8B8:
+ srcInfo.pixelStride = 4;
+ IntRgbToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ &srcInfo, pDstInfo, NULL, NULL);
+ break;
+ case D3DFMT_X8B8G8R8:
+ srcInfo.pixelStride = 4;
+ IntBgrToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ &srcInfo, pDstInfo, NULL, NULL);
+ break;
+ case D3DFMT_X1R5G5B5:
+ srcInfo.pixelStride = 2;
+ Ushort555RgbToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ &srcInfo, pDstInfo, NULL, NULL);
+ break;
+ case D3DFMT_R5G6B5:
+ srcInfo.pixelStride = 2;
+ Ushort565RgbToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ &srcInfo, pDstInfo, NULL, NULL);
+ break;
+ default:
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DBL_CopySurfaceToIntArgbImage: unknown format %d",
+ desc.Format);
}
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
- if (d3dc == NULL) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "D3DBlitLoops_doTransform: null device context");
- return;
- }
- Win32SDOps *srcOps = (Win32SDOps *)jlong_to_ptr(pSrcData);
- Win32SDOps *dstOps = (Win32SDOps *)jlong_to_ptr(pDstData);
+ return pSurface->UnlockRect();
+}
- if (!srcOps->ddInstance || !dstOps->ddInstance) {
- // Some situations can cause us to fail on primary
- // creation, resulting in null lpSurface and null ddInstance
- // for a Win32Surface object.. Just noop this call in that case.
- return;
+D3DPIPELINE_API HRESULT
+D3DBL_CopyImageToIntXrgbSurface(SurfaceDataRasInfo *pSrcInfo,
+ int srctype,
+ D3DResource *pDstSurfaceRes,
+ jint srcx, jint srcy,
+ jint srcWidth, jint srcHeight,
+ jint dstx, jint dsty)
+{
+ HRESULT res = S_OK;
+ D3DLOCKED_RECT lockedRect;
+ RECT r = { dstx, dsty, dstx+srcWidth, dsty+srcHeight };
+ RECT *pR = &r;
+ SurfaceDataRasInfo dstInfo;
+ IDirect3DSurface9 *pDstSurface = pDstSurfaceRes->GetSurface();
+ D3DSURFACE_DESC *pDesc = pDstSurfaceRes->GetDesc();
+ DWORD dwLockFlags = D3DLOCK_NOSYSLOCK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBL_CopyImageToIntXrgbSurface");
+ J2dTraceLn5(J2D_TRACE_VERBOSE,
+ " srctype=%d rect={%-4d, %-4d, %-4d, %-4d}",
+ srctype, r.left, r.top, r.right, r.bottom);
+
+ if (pDesc->Usage == D3DUSAGE_DYNAMIC &&
+ dstx == 0 && dstx == 0 &&
+ srcWidth == pDesc->Width && srcHeight == pDesc->Height)
+ {
+ dwLockFlags |= D3DLOCK_DISCARD;
+ pR = NULL;
}
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- DDrawSurface *ddSrcSurface = srcOps->lpSurface;
- if (ddTargetSurface == NULL || ddSrcSurface == NULL) {
- return;
- }
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
+ res = pDstSurface->LockRect(&lockedRect, pR, dwLockFlags);
+ RETURN_STATUS_IF_FAILED(res);
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return;
+ ZeroMemory(&dstInfo, sizeof(SurfaceDataRasInfo));
+ // dstInfo.bounds.x1 = 0;
+ // dstInfo.bounds.y1 = 0;
+ dstInfo.bounds.x2 = srcWidth;
+ dstInfo.bounds.y2 = srcHeight;
+ dstInfo.scanStride = lockedRect.Pitch;
+ dstInfo.pixelStride = 4;
+
+ void *pSrcBase = PtrCoord(pSrcInfo->rasBase,
+ srcx, pSrcInfo->pixelStride,
+ srcy, pSrcInfo->scanStride);
+ void *pDstBase = lockedRect.pBits;
+
+ switch (srctype) {
+ case ST_INT_ARGB:
+ IntArgbToIntArgbPreConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_INT_ARGB_PRE:
+ case ST_INT_RGB:
+ AnyIntIsomorphicCopy(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_INT_ARGB_BM:
+ // REMIND: we don't have such sw loop
+ // so this path is disabled for now on java level
+// IntArgbBmToIntArgbPreConvert(pSrcBase, pDstBase,
+// srcWidth, srcHeight,
+// pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_INT_BGR:
+ IntBgrToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_USHORT_555_RGB:
+ Ushort555RgbToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_USHORT_565_RGB:
+ Ushort565RgbToIntArgbConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_BYTE_INDEXED:
+ ByteIndexedToIntArgbPreConvert(pSrcBase, pDstBase,
+ srcWidth, srcHeight,
+ pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ case ST_BYTE_INDEXED_BM:
+ // REMIND: we don't have such sw loop
+ // so this path is disabled for now on java level
+// ByteIndexedBmToIntArgbPreConvert(pSrcBase, pDstBase,
+// srcWidth, srcHeight,
+// pSrcInfo, &dstInfo, NULL, NULL);
+ break;
+ default:
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DBL_CopyImageToIntXrgbSurface: unknown type %d",
+ srctype);
}
- float tw = (float)ddSrcSurface->GetDXSurface()->GetWidth();
- float th = (float)ddSrcSurface->GetDXSurface()->GetHeight();
- float tx1 = ((float)sx1) / tw;
- float ty1 = ((float)sy1) / th;
- float tx2 = ((float)sx2) / tw;
- float ty2 = ((float)sy2) / th;
+ return pDstSurface->UnlockRect();
+}
- D3DU_INIT_VERTEX_QUAD(quadVerts, dx1, dy1, dx2, dy2,
- d3dc->blitPolygonPixel,
- tx1, ty1, tx2, ty2);
+/**
+ * Inner loop used for copying a source "render-to" D3D "Surface" to a
+ * destination D3D "Surface". Note that the same surface can
+ * not be used as both the source and destination, as is the case in a copyArea()
+ * operation. This method is invoked from D3DBlitLoops_IsoBlit().
+ *
+ * The standard StretchRect() mechanism is used to copy the source region
+ * into the destination region. If the regions have different dimensions,
+ * the source will be scaled into the destination as appropriate (only
+ * nearest neighbor filtering will be applied for simple scale operations).
+ */
+HRESULT
+D3DBlitSurfaceToSurface(D3DContext *d3dc, D3DSDOps *srcOps, D3DSDOps *dstOps,
+ D3DTEXTUREFILTERTYPE hint,
+ jint sx1, jint sy1, jint sx2, jint sy2,
+ jint dx1, jint dy1, jint dx2, jint dy2)
+{
+ IDirect3DSurface9 *pSrc, *pDst;
- if (hint == D3DSD_XFORM_BILINEAR) {
- d3dDevice->SetTextureStageState(0, D3DTSS_MAGFILTER, D3DTFG_LINEAR);
- d3dDevice->SetTextureStageState(0, D3DTSS_MINFILTER, D3DTFG_LINEAR);
- } else if (hint == D3DSD_XFORM_NEAREST_NEIGHBOR) {
- d3dDevice->SetTextureStageState(0, D3DTSS_MAGFILTER, D3DTFG_POINT);
- d3dDevice->SetTextureStageState(0, D3DTSS_MINFILTER, D3DTFG_POINT);
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBlitSurfaceToSurface");
+
+ RETURN_STATUS_IF_NULL(srcOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(pSrc = srcOps->pResource->GetSurface(), E_FAIL);
+ RETURN_STATUS_IF_NULL(pDst = dstOps->pResource->GetSurface(), E_FAIL);
+
+ d3dc->UpdateState(STATE_OTHEROP);
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+
+ // need to clip the destination bounds,
+ // otherwise StretchRect could fail
+ jint sw = sx2 - sx1;
+ jint sh = sy2 - sy1;
+ jdouble dw = dx2 - dx1;
+ jdouble dh = dy2 - dy1;
+
+ SurfaceDataBounds dstBounds;
+ dstBounds.x1 = dx1;
+ dstBounds.y1 = dy1;
+ dstBounds.x2 = dx2;
+ dstBounds.y2 = dy2;
+ SurfaceData_IntersectBoundsXYXY(&dstBounds, 0, 0,
+ dstOps->width, dstOps->height);
+ if (d3dc->GetClipType() == CLIP_RECT) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " rect clip, clip dest manually");
+ RECT clipRect;
+ pd3dDevice->GetScissorRect(&clipRect);
+ SurfaceData_IntersectBoundsXYXY(&dstBounds,
+ clipRect.left, clipRect.top,
+ clipRect.right, clipRect.bottom);
}
+ if (dstBounds.x1 != dx1) {
+ sx1 += (int)((dstBounds.x1 - dx1) * (sw / dw));
+ }
+ if (dstBounds.y1 != dy1) {
+ sy1 += (int)((dstBounds.y1 - dy1) * (sh / dh));
+ }
+ if (dstBounds.x2 != dx2) {
+ sx2 += (int)((dstBounds.x2 - dx2) * (sw / dw));
+ }
+ if (dstBounds.y2 != dy2) {
+ sy2 += (int)((dstBounds.y2 - dy2) * (sh / dh));
+ }
+
+ // check if the rects are empty (StretchRect will fail if so)
+ if (dstBounds.x1 >= dstBounds.x2 || dstBounds.y1 >= dstBounds.y2 ||
+ sx1 >= sx2 || sy1 >= sy2)
+ {
+ return S_OK;
+ }
+
+ RECT srcRect = { sx1, sy1, sx2, sy2 };
+ RECT dstRect = { dstBounds.x1, dstBounds.y1, dstBounds.x2, dstBounds.y2 };
+
+ return pd3dDevice->StretchRect(pSrc, &srcRect, pDst, &dstRect, hint);
+}
+
+/**
+ * A convenience method for issuing DrawTexture calls depending on the
+ * hint. See detailed explanation below.
+ */
+static inline HRESULT
+D3DDrawTextureWithHint(D3DContext *d3dc, D3DTEXTUREFILTERTYPE hint,
+ jint srcWidth, jint srcHeight,
+ float tw, float th,
+ jint sx1, jint sy1, jint sx2, jint sy2,
+ float dx1, float dy1, float dx2, float dy2,
+ float tx1, float ty1, float tx2, float ty2)
+{
HRESULT res;
- D3DU_PRIM2_LOOP_BEGIN(res, srcOps, dstOps);
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_BLITOP))) {
- DXSurface *dxSurface = ddSrcSurface->GetDXSurface();
- if (SUCCEEDED(res = d3dc->SetTexture(dxSurface)))
- {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN, D3DFVF_J2DLVERTEX,
- quadVerts, 4, 0);
- }
- d3dc->EndScene(res);
+
+ if (hint == D3DTEXF_LINEAR &&
+ (srcWidth != tw || srcHeight != th ||
+ srcWidth != sx2 || srcHeight != sy2 ))
+ {
+ /*
+ * When the image bounds are smaller than the bounds of the
+ * texture that the image resides in, D3DTEXF_LINEAR will use pixels
+ * from outside the valid image bounds, which could result in garbage
+ * pixels showing up at the edges of the transformed result. We set
+ * the texture wrap mode to D3DTADDRESS_CLAMP, which solves the problem
+ * for the top and left edges. But when the source bounds do not
+ * match the texture bounds, we need to perform this as a four-part
+ * operation in order to prevent the filter used by D3D from using
+ * invalid pixels at the bottom and right edges.
+ *
+ * Note that we only need to apply this technique when the source
+ * bounds are equal to the actual image bounds. If the source bounds
+ * fall within the image bounds there is no need to apply this hack
+ * because the filter used by D3D will access valid pixels.
+ * Likewise, if the image bounds are equal to the texture bounds,
+ * then the edge conditions are handled properly by D3DTADDRESS_CLAMP.
+ */
+
+ // These values represent the bottom-right corner of source texture
+ // region pulled in by 1/2 of a source texel.
+ float tx2adj = tx2 - (1.0f / (2.0f * tw));
+ float ty2adj = ty2 - (1.0f / (2.0f * th));
+
+ // These values represent the above coordinates pulled in by a
+ // tiny fraction. As an example, if we sample the tiny area from
+ // tx2adj2 to tx2adj, the result should be the solid color at the
+ // texel center corresponding to tx2adj.
+ float tx2adj2 = tx2adj - 0.0001f;
+ float ty2adj2 = ty2adj - 0.0001f;
+
+ // These values represent the bottom-right corner of the destination
+ // region pulled in by 1/2 of a destination pixel.
+ float dx2adj = dx2 - 0.5f;
+ float dy2adj = dy2 - 0.5f;
+
+ // First, render a majority of the source texture, from the top-left
+ // corner to the bottom-right, but not including the right or bottom
+ // edges.
+ d3dc->pVCacher->DrawTexture(dx1, dy1, dx2adj, dy2adj,
+ tx1, ty1, tx2adj, ty2adj);
+
+ // Second, render the remaining sliver on the right edge.
+ d3dc->pVCacher->DrawTexture(dx2adj, dy1, dx2, dy2adj,
+ tx2adj2, ty1, tx2adj, ty2adj);
+
+ // Third, render the remaining sliver on the bottom edge.
+ d3dc->pVCacher->DrawTexture(dx1, dy2adj, dx2adj, dy2,
+ tx1, ty2adj2, tx2adj, ty2adj);
+
+ // Finally, render the remaining speck at the bottom-right corner.
+ res = d3dc->pVCacher->DrawTexture(dx2adj, dy2adj, dx2, dy2,
+ tx2adj2, ty2adj2, tx2adj, ty2adj);
+ } else {
+ /*
+ * As mentioned above, we can issue a simple textured quad if:
+ * - the hint is D3DTEXF_POINT or
+ * - the source bounds are sufficiently inside the texture bounds or
+ * - the image bounds are equal to the texture bounds (as is the
+ * case when the image has power-of-two dimensions, or when the
+ * device supports non-pow2 textures)
+ */
+ res = d3dc->pVCacher->DrawTexture(dx1, dy1, dx2, dy2,
+ tx1, ty1, tx2, ty2);
}
- D3DU_PRIM2_LOOP_END(env, res, srcOps, dstOps,
- "DrawPrimitive(D3DPT_TRIANGLEFAN)");
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
+ return res;
}
+/**
+ * Inner loop used for copying a source D3D "Texture" to a destination
+ * D3D "Surface". This method is invoked from D3DBlitLoops_IsoBlit().
+ *
+ * This method will copy, scale, or transform the source texture into the
+ * destination depending on the transform state, as established in
+ * and D3DContext::SetTransform(). If the source texture is
+ * transformed in any way when rendered into the destination, the filtering
+ * method applied is determined by the hint parameter.
+ */
+static HRESULT
+D3DBlitTextureToSurface(D3DContext *d3dc,
+ D3DSDOps *srcOps, D3DSDOps *dstOps,
+ jboolean rtt, D3DTEXTUREFILTERTYPE hint,
+ jint sx1, jint sy1, jint sx2, jint sy2,
+ float dx1, float dy1, float dx2, float dy2)
+{
+ HRESULT res;
+ IDirect3DTexture9 *pSrc;
+ IDirect3DDevice9 *pd3dDevice;
+ float tx1, ty1, tx2, ty2;
+ float tw, th;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBlitTextureToSurface");
+
+ RETURN_STATUS_IF_NULL(srcOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+
+ if ((pSrc = srcOps->pResource->GetTexture()) == NULL ||
+ FAILED(res = d3dc->BeginScene(STATE_TEXTUREOP) ||
+ FAILED(res = d3dc->SetTexture(pSrc))))
+ {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DBlitTextureToSurface: BeginScene or SetTexture failed");
+ return res;
+ }
+
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, hint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, hint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP);
+
+ tw = (float)srcOps->pResource->GetDesc()->Width;
+ th = (float)srcOps->pResource->GetDesc()->Height;
+
+ // convert the source bounds into the range [0,1]
+ tx1 = ((float)sx1) / tw;
+ ty1 = ((float)sy1) / th;
+ tx2 = ((float)sx2) / tw;
+ ty2 = ((float)sy2) / th;
+
+ return D3DDrawTextureWithHint(d3dc, hint,
+ srcOps->width, srcOps->height,
+ tw, th,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2,
+ tx1, ty1, tx2, ty2);
+}
+
+/**
+ * Inner loop used for copying a source system memory ("Sw") surface or
+ * D3D "Surface" to a destination D3D "Surface", using an D3D texture
+ * tile as an intermediate surface. This method is invoked from
+ * D3DBlitLoops_Blit() for "Sw" surfaces and D3DBlitLoops_IsoBlit() for
+ * "Surface" surfaces.
+ *
+ * This method is used to transform the source surface into the destination.
+ * Pixel rectangles cannot be arbitrarily transformed. However, texture
+ * mapped quads do respect the modelview transform matrix, so we use
+ * textures here to perform the transform operation. This method uses a
+ * tile-based approach in which a small subregion of the source surface is
+ * copied into a cached texture tile. The texture tile is then mapped
+ * into the appropriate location in the destination surface.
+ *
+ */
+D3DPIPELINE_API HRESULT
+D3DBlitToSurfaceViaTexture(D3DContext *d3dc, SurfaceDataRasInfo *srcInfo,
+ int srctype, D3DSDOps *srcOps,
+ jboolean swsurface, jint hint,
+ jint sx1, jint sy1, jint sx2, jint sy2,
+ jdouble dx1, jdouble dy1, jdouble dx2, jdouble dy2)
+{
+ double tx1, ty1, tx2, ty2;
+ double dx, dy, dw, dh, cdw, cdh;
+ jint tw, th;
+ jint sx, sy, sw, sh;
+ HRESULT res = S_OK;
+ D3DResource *pBlitTextureRes = NULL;
+ IDirect3DTexture9 *pBlitTexture = NULL;
+ IDirect3DSurface9 *pBlitSurface = NULL, *pSrc = NULL;
+ D3DTEXTUREFILTERTYPE fhint =
+ (hint == D3DSD_XFORM_BILINEAR) ? D3DTEXF_LINEAR : D3DTEXF_POINT;
+ fhint = d3dc->IsTextureFilteringSupported(fhint) ? fhint : D3DTEXF_NONE;
+
+ if (swsurface) {
+ res = d3dc->GetResourceManager()->GetBlitTexture(&pBlitTextureRes);
+ } else {
+ RETURN_STATUS_IF_NULL(srcOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(pSrc = srcOps->pResource->GetSurface(), E_FAIL);
+
+ res = d3dc->GetResourceManager()->
+ GetBlitRTTexture(D3DC_BLIT_TILE_SIZE, D3DC_BLIT_TILE_SIZE,
+ srcOps->pResource->GetDesc()->Format,
+ &pBlitTextureRes);
+ }
+ if (FAILED(res)) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DBlitToSurfaceViaTexture: could not init blit tile");
+ return res;
+ }
+ pBlitSurface = pBlitTextureRes->GetSurface();
+ pBlitTexture = pBlitTextureRes->GetTexture();
+
+ D3DSURFACE_DESC *pDesc = pBlitTextureRes->GetDesc();
+
+ tx1 = 0.0f;
+ ty1 = 0.0f;
+ tw = pDesc->Width;
+ th = pDesc->Height;
+ cdw = (dx2-dx1) / (((double)(sx2-sx1)) / tw);
+ cdh = (dy2-dy1) / (((double)(sy2-sy1)) / th);
+
+ res = d3dc->BeginScene(STATE_TEXTUREOP);
+ RETURN_STATUS_IF_FAILED(res);
+ res = d3dc->SetTexture(pBlitTexture);
+ RETURN_STATUS_IF_FAILED(res);
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP);
+
+ for (sy = sy1, dy = dy1; sy < sy2; sy += th, dy += cdh) {
+ sh = ((sy + th) > sy2) ? (sy2 - sy) : th;
+ dh = ((dy + cdh) > dy2) ? (dy2 - dy) : cdh;
+
+ for (sx = sx1, dx = dx1; sx < sx2; sx += tw, dx += cdw) {
+ sw = ((sx + tw) > sx2) ? (sx2 - sx) : tw;
+ dw = ((dx + cdw) > dx2) ? (dx2 - dx) : cdw;
+
+ tx2 = ((double)sw) / tw;
+ ty2 = ((double)sh) / th;
+
+ if (swsurface) {
+ D3DBL_CopyImageToIntXrgbSurface(srcInfo,
+ srctype, pBlitTextureRes,
+ sx, sy, sw, sh,
+ 0, 0);
+ } else {
+ RECT srcRect = { (LONG)sx, (LONG)sy,
+ (LONG)(sx+dw), (LONG)(sy+dh) };
+ RECT dstRect = { 0l, 0l, (LONG)dw, (LONG)dh };
+ pd3dDevice->StretchRect(pSrc,
+ &srcRect, pBlitSurface, &dstRect,
+ D3DTEXF_NONE);
+ }
+ D3DDrawTextureWithHint(d3dc, fhint,
+ tw, th,
+ (float)tw, (float)th,
+ sx, sy, sw, sh,
+ (float)dx, (float)dy, (float)(dx+dw), (float)(dy+dh),
+ (float)tx1, (float)ty1, (float)tx2, (float)ty2);
+ res = d3dc->pVCacher->Render();
+ }
+ }
+ return res;
+}
+
+/**
+ * Inner loop used for copying a source system memory ("Sw") surface to a
+ * destination D3D "Texture". This method is invoked from
+ * D3DBlitLoops_Blit().
+ *
+ * The source surface is effectively loaded into the D3D texture object,
+ * which must have already been initialized by D3DSD_initTexture(). Note
+ * that this method is only capable of copying the source surface into the
+ * destination surface (i.e. no scaling or general transform is allowed).
+ * This restriction should not be an issue as this method is only used
+ * currently to cache a static system memory image into an D3D texture in
+ * a hidden-acceleration situation.
+ */
+static HRESULT
+D3DBlitSwToTexture(D3DContext *d3dc,
+ SurfaceDataRasInfo *srcInfo, int srctype,
+ D3DSDOps *dstOps,
+ jint sx1, jint sy1, jint sx2, jint sy2)
+{
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps->pResource->GetSurface(), E_FAIL);
+
+ return D3DBL_CopyImageToIntXrgbSurface(srcInfo, srctype,
+ dstOps->pResource,
+ sx1, sy1, sx2-sx1, sy2-sy1,
+ 0, 0);
+}
+
+/**
+ * General blit method for copying a native D3D surface (of type "Surface"
+ * or "Texture") to another D3D "Surface". If texture is JNI_TRUE, this
+ * method will invoke the Texture->Surface inner loop; otherwise, one of the
+ * Surface->Surface inner loops will be invoked, depending on the transform
+ * state.
+ */
+D3DPIPELINE_API HRESULT
+D3DBlitLoops_IsoBlit(JNIEnv *env,
+ D3DContext *d3dc, jlong pSrcOps, jlong pDstOps,
+ jboolean xform, jint hint,
+ jboolean texture, jboolean rtt,
+ jint sx1, jint sy1, jint sx2, jint sy2,
+ jdouble dx1, jdouble dy1, jdouble dx2, jdouble dy2)
+{
+ D3DSDOps *srcOps = (D3DSDOps *)jlong_to_ptr(pSrcOps);
+ D3DSDOps *dstOps = (D3DSDOps *)jlong_to_ptr(pDstOps);
+ SurfaceDataRasInfo srcInfo;
+ jint sw = sx2 - sx1;
+ jint sh = sy2 - sy1;
+ jdouble dw = dx2 - dx1;
+ jdouble dh = dy2 - dy1;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBlitLoops_IsoBlit");
+
+ if (sw <= 0 || sh <= 0 || dw <= 0 || dh <= 0) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DBlitLoops_IsoBlit: invalid dimensions");
+ return E_FAIL;
+ }
+
+ RETURN_STATUS_IF_NULL(srcOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc->Get3DDevice(), E_FAIL);
+
+ srcInfo.bounds.x1 = sx1;
+ srcInfo.bounds.y1 = sy1;
+ srcInfo.bounds.x2 = sx2;
+ srcInfo.bounds.y2 = sy2;
+
+ SurfaceData_IntersectBoundsXYXY(&srcInfo.bounds,
+ 0, 0, srcOps->width, srcOps->height);
+
+
+ HRESULT res = S_OK;
+ if (srcInfo.bounds.x2 > srcInfo.bounds.x1 &&
+ srcInfo.bounds.y2 > srcInfo.bounds.y1)
+ {
+ if (srcInfo.bounds.x1 != sx1) {
+ dx1 += (srcInfo.bounds.x1 - sx1) * (dw / sw);
+ sx1 = srcInfo.bounds.x1;
+ }
+ if (srcInfo.bounds.y1 != sy1) {
+ dy1 += (srcInfo.bounds.y1 - sy1) * (dh / sh);
+ sy1 = srcInfo.bounds.y1;
+ }
+ if (srcInfo.bounds.x2 != sx2) {
+ dx2 += (srcInfo.bounds.x2 - sx2) * (dw / sw);
+ sx2 = srcInfo.bounds.x2;
+ }
+ if (srcInfo.bounds.y2 != sy2) {
+ dy2 += (srcInfo.bounds.y2 - sy2) * (dh / sh);
+ sy2 = srcInfo.bounds.y2;
+ }
+
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " texture=%d hint=%d", texture, hint);
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " sx1=%d sy1=%d sx2=%d sy2=%d",
+ sx1, sy1, sx2, sy2);
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " dx1=%f dy1=%f dx2=%f dy2=%f",
+ dx1, dy1, dx2, dy2);
+
+ D3DTEXTUREFILTERTYPE fhint =
+ (hint == D3DSD_XFORM_BILINEAR) ? D3DTEXF_LINEAR : D3DTEXF_POINT;
+ if (texture) {
+ fhint = d3dc->IsTextureFilteringSupported(fhint) ?
+ fhint : D3DTEXF_NONE;
+ res = D3DBlitTextureToSurface(d3dc, srcOps, dstOps, rtt, fhint,
+ sx1, sy1, sx2, sy2,
+ (float)dx1, (float)dy1,
+ (float)dx2, (float)dy2);
+ } else {
+ // StretchRect does not do compositing or clipping
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ DWORD abEnabled = 0;
+
+ pd3dDevice->GetRenderState(D3DRS_ALPHABLENDENABLE, &abEnabled);
+ J2dTraceLn3(J2D_TRACE_VERBOSE, " xform=%d clip=%d abEnabled=%d",
+ xform, d3dc->GetClipType(), abEnabled);
+ if (!xform && d3dc->GetClipType() != CLIP_SHAPE && !abEnabled) {
+ fhint = d3dc->IsStretchRectFilteringSupported(fhint) ?
+ fhint : D3DTEXF_NONE;
+
+ res = D3DBlitSurfaceToSurface(d3dc, srcOps, dstOps, fhint,
+ sx1, sy1, sx2, sy2,
+ (int)dx1, (int)dy1,
+ (int)dx2, (int)dy2);
+ } else {
+ res = D3DBlitToSurfaceViaTexture(d3dc, &srcInfo,
+ // surface type is unused here
+ ST_INT_ARGB_PRE,
+ srcOps,
+ JNI_FALSE, hint,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2);
+ }
+ }
+ }
+ return res;
+}
+
+/**
+ * General blit method for copying a system memory ("Sw") surface to a native
+ * D3D surface (of type "Surface" or "Texture"). If texture is JNI_TRUE,
+ * this method will invoke the Sw->Texture inner loop; otherwise, one of the
+ * Sw->Surface inner loops will be invoked, depending on the transform state.
+ */
+HRESULT
+D3DBlitLoops_Blit(JNIEnv *env,
+ D3DContext *d3dc, jlong pSrcOps, jlong pDstOps,
+ jboolean xform, jint hint,
+ jint srctype, jboolean texture,
+ jint sx1, jint sy1, jint sx2, jint sy2,
+ jdouble dx1, jdouble dy1, jdouble dx2, jdouble dy2)
+{
+ SurfaceDataOps *srcOps = (SurfaceDataOps *)jlong_to_ptr(pSrcOps);
+ D3DSDOps *dstOps = (D3DSDOps *)jlong_to_ptr(pDstOps);
+ SurfaceDataRasInfo srcInfo;
+ HRESULT res = S_OK;
+ jint sw = sx2 - sx1;
+ jint sh = sy2 - sy1;
+ jdouble dw = dx2 - dx1;
+ jdouble dh = dy2 - dy1;
+ jint lockFlags = SD_LOCK_READ;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBlitLoops_Blit");
+
+ if (sw <= 0 || sh <= 0 || dw <= 0 || dh <= 0 || srctype < 0) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DBlitLoops_Blit: invalid dimensions or srctype");
+ return E_FAIL;
+ }
+
+ RETURN_STATUS_IF_NULL(srcOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc->Get3DDevice(), E_FAIL);
+
+ srcInfo.bounds.x1 = sx1;
+ srcInfo.bounds.y1 = sy1;
+ srcInfo.bounds.x2 = sx2;
+ srcInfo.bounds.y2 = sy2;
+
+ if (srctype == ST_BYTE_INDEXED || srctype == ST_BYTE_INDEXED_BM) {
+ lockFlags |= SD_LOCK_LUT;
+ }
+ if (srcOps->Lock(env, srcOps, &srcInfo, lockFlags) != SD_SUCCESS) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DBlitLoops_Blit: could not acquire lock");
+ return E_FAIL;
+ }
+
+ if (srcInfo.bounds.x2 > srcInfo.bounds.x1 &&
+ srcInfo.bounds.y2 > srcInfo.bounds.y1)
+ {
+ srcOps->GetRasInfo(env, srcOps, &srcInfo);
+ if (srcInfo.rasBase) {
+ if (srcInfo.bounds.x1 != sx1) {
+ dx1 += (srcInfo.bounds.x1 - sx1) * (dw / sw);
+ sx1 = srcInfo.bounds.x1;
+ }
+ if (srcInfo.bounds.y1 != sy1) {
+ dy1 += (srcInfo.bounds.y1 - sy1) * (dh / sh);
+ sy1 = srcInfo.bounds.y1;
+ }
+ if (srcInfo.bounds.x2 != sx2) {
+ dx2 += (srcInfo.bounds.x2 - sx2) * (dw / sw);
+ sx2 = srcInfo.bounds.x2;
+ }
+ if (srcInfo.bounds.y2 != sy2) {
+ dy2 += (srcInfo.bounds.y2 - sy2) * (dh / sh);
+ sy2 = srcInfo.bounds.y2;
+ }
+
+ J2dTraceLn3(J2D_TRACE_VERBOSE, " texture=%d srctype=%d hint=%d",
+ texture, srctype, hint);
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " sx1=%d sy1=%d sx2=%d sy2=%d",
+ sx1, sy1, sx2, sy2);
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " dx1=%f dy1=%f dx2=%f dy2=%f",
+ dx1, dy1, dx2, dy2);
+
+ if (texture) {
+ // These coordinates will always be integers since we
+ // only ever do a straight copy from sw to texture.
+ // Thus these casts are "safe" - no loss of precision.
+ res = D3DBlitSwToTexture(d3dc, &srcInfo, srctype, dstOps,
+ (jint)dx1, (jint)dy1,
+ (jint)dx2, (jint)dy2);
+ } else {
+ res = D3DBlitToSurfaceViaTexture(d3dc, &srcInfo, srctype, NULL,
+ JNI_TRUE, hint,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2);
+ }
+ }
+ SurfaceData_InvokeRelease(env, srcOps, &srcInfo);
+ }
+ SurfaceData_InvokeUnlock(env, srcOps, &srcInfo);
+ return res;
+}
+
+/**
+ * Specialized blit method for copying a native D3D "Surface" (pbuffer,
+ * window, etc.) to a system memory ("Sw") surface.
+ */
+HRESULT
+D3DBlitLoops_SurfaceToSwBlit(JNIEnv *env, D3DContext *d3dc,
+ jlong pSrcOps, jlong pDstOps, jint dsttype,
+ jint srcx, jint srcy, jint dstx, jint dsty,
+ jint width, jint height)
+{
+ D3DSDOps *srcOps = (D3DSDOps *)jlong_to_ptr(pSrcOps);
+ SurfaceDataOps *dstOps = (SurfaceDataOps *)jlong_to_ptr(pDstOps);
+ SurfaceDataRasInfo srcInfo, dstInfo;
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBlitLoops_SurfaceToSwBlit");
+
+ if (width <= 0 || height <= 0) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DBlitLoops_SurfaceToSwBlit: dimensions are non-positive");
+ return S_OK;
+ }
+
+ RETURN_STATUS_IF_NULL(srcOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(srcOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc->Get3DDevice(), E_FAIL);
+
+ srcInfo.bounds.x1 = srcx;
+ srcInfo.bounds.y1 = srcy;
+ srcInfo.bounds.x2 = srcx + width;
+ srcInfo.bounds.y2 = srcy + height;
+ dstInfo.bounds.x1 = dstx;
+ dstInfo.bounds.y1 = dsty;
+ dstInfo.bounds.x2 = dstx + width;
+ dstInfo.bounds.y2 = dsty + height;
+
+ if (dstOps->Lock(env, dstOps, &dstInfo, SD_LOCK_WRITE) != SD_SUCCESS) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DBlitLoops_SurfaceToSwBlit: could not acquire dst lock");
+ return S_OK;
+ }
+
+ SurfaceData_IntersectBoundsXYXY(&srcInfo.bounds,
+ 0, 0, srcOps->width, srcOps->height);
+ SurfaceData_IntersectBlitBounds(&dstInfo.bounds, &srcInfo.bounds,
+ srcx - dstx, srcy - dsty);
+
+ if (srcInfo.bounds.x2 > srcInfo.bounds.x1 &&
+ srcInfo.bounds.y2 > srcInfo.bounds.y1)
+ {
+ dstOps->GetRasInfo(env, dstOps, &dstInfo);
+ if (dstInfo.rasBase) {
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ IDirect3DSurface9 *pSrc = srcOps->pResource->GetSurface();
+ D3DFORMAT srcFmt = srcOps->pResource->GetDesc()->Format;
+ UINT srcw = srcOps->pResource->GetDesc()->Width;
+ UINT srch = srcOps->pResource->GetDesc()->Height;
+ D3DResource *pLockableRes;
+
+ srcx = srcInfo.bounds.x1;
+ srcy = srcInfo.bounds.y1;
+ dstx = dstInfo.bounds.x1;
+ dsty = dstInfo.bounds.y1;
+ width = srcInfo.bounds.x2 - srcInfo.bounds.x1;
+ height = srcInfo.bounds.y2 - srcInfo.bounds.y1;
+
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " sx=%d sy=%d w=%d h=%d",
+ srcx, srcy, width, height);
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " dx=%d dy=%d",
+ dstx, dsty);
+
+ d3dc->UpdateState(STATE_OTHEROP);
+
+ // if we read more than 50% of the image it is faster
+ // to get the whole thing (50% is pulled out of a hat)
+ BOOL fullRead = ((width * height) >= (srcw * srch * 0.5f));
+ UINT lockSrcX = 0, lockSrcY = 0;
+
+ if (fullRead) {
+ // read whole surface into a sysmem surface
+ lockSrcX = srcx;
+ lockSrcY = srcy;
+ // the dest surface must have the same dimensions and format as
+ // the source, GetBlitOSPSurface ensures that
+ res = d3dc->GetResourceManager()->
+ GetBlitOSPSurface(srcw, srch, srcFmt, &pLockableRes);
+ } else {
+ // we first copy the source region to a temp
+ // render target surface of the same format as the
+ // source, then copy the pixels to the
+ // target buffered image surface
+ res = d3dc->GetResourceManager()->
+ GetLockableRTSurface(width, height, srcFmt, &pLockableRes);
+ }
+ if (SUCCEEDED(res)) {
+ IDirect3DSurface9 *pTmpSurface = pLockableRes->GetSurface();
+
+ if (fullRead) {
+ res = pd3dDevice->GetRenderTargetData(pSrc, pTmpSurface);
+ } else {
+ RECT srcRect = { srcx, srcy, srcx+width, srcy+height};
+ RECT dstRect = { 0l, 0l, width, height };
+
+ res = pd3dDevice->StretchRect(pSrc,
+ &srcRect, pTmpSurface,
+ &dstRect, D3DTEXF_NONE);
+ }
+
+ if (SUCCEEDED(res)) {
+ res = D3DBL_CopySurfaceToIntArgbImage(
+ pTmpSurface, /* src surface */
+ &dstInfo, /* dst info */
+ lockSrcX, lockSrcY, width, height, /* src rect */
+ dstx, dsty); /* dst coords */
+ }
+ }
+ }
+ SurfaceData_InvokeRelease(env, dstOps, &dstInfo);
+ }
+ SurfaceData_InvokeUnlock(env, dstOps, &dstInfo);
+ return res;
+}
+
+HRESULT
+D3DBlitLoops_CopyArea(JNIEnv *env,
+ D3DContext *d3dc, D3DSDOps *dstOps,
+ jint x, jint y, jint width, jint height,
+ jint dx, jint dy)
+{
+ SurfaceDataBounds srcBounds, dstBounds;
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBlitLoops_CopyArea");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " x=%d y=%d w=%d h=%d",
+ x, y, width, height);
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " dx=%d dy=%d",
+ dx, dy);
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ RETURN_STATUS_IF_NULL(pd3dDevice, E_FAIL);
+ ClipType clipType = d3dc->GetClipType();
+
+ srcBounds.x1 = x;
+ srcBounds.y1 = y;
+ srcBounds.x2 = srcBounds.x1 + width;
+ srcBounds.y2 = srcBounds.y1 + height;
+ dstBounds.x1 = x + dx;
+ dstBounds.y1 = y + dy;
+ dstBounds.x2 = dstBounds.x1 + width;
+ dstBounds.y2 = dstBounds.y1 + height;
+
+ SurfaceData_IntersectBoundsXYXY(&srcBounds,
+ 0, 0, dstOps->width, dstOps->height);
+ if (clipType == CLIP_RECT) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " rect clip, clip dest manually");
+ RECT clipRect;
+ pd3dDevice->GetScissorRect(&clipRect);
+ SurfaceData_IntersectBoundsXYXY(&dstBounds,
+ clipRect.left, clipRect.top,
+ clipRect.right, clipRect.bottom);
+ }
+ SurfaceData_IntersectBoundsXYXY(&dstBounds,
+ 0, 0, dstOps->width, dstOps->height);
+ SurfaceData_IntersectBlitBounds(&dstBounds, &srcBounds, -dx, -dy);
+
+ if (dstBounds.x1 < dstBounds.x2 && dstBounds.y1 < dstBounds.y2) {
+ jint sx1 = srcBounds.x1, sy1 = srcBounds.y1,
+ sx2 = srcBounds.x2, sy2 = srcBounds.y2;
+ jint dx1 = dstBounds.x1, dy1 = dstBounds.y1,
+ dx2 = dstBounds.x2, dy2 = dstBounds.y2;
+ jint dw = dx2 - dx1, dh = dy2 - dy1;
+
+ IDirect3DTexture9 *pBlitTexture = NULL;
+ IDirect3DSurface9 *pBlitSurface = NULL;
+ D3DResource *pBlitTextureRes;
+
+ res = d3dc->GetResourceManager()->
+ GetBlitRTTexture(dw, dh,
+ dstOps->pResource->GetDesc()->Format,
+ &pBlitTextureRes);
+ if (SUCCEEDED(res)) {
+ pBlitSurface = pBlitTextureRes->GetSurface();
+ pBlitTexture = pBlitTextureRes->GetTexture();
+ }
+ if (!pBlitTexture || !pBlitSurface) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DBlitLoops_CopyArea: could not init blit tile");
+ return E_FAIL;
+ }
+
+ // flush the rendering first
+ d3dc->UpdateState(STATE_OTHEROP);
+
+ // REMIND: see if we could always use texture mapping;
+ // the assumption here is that StretchRect is faster,
+ // if it's not, then we should always use texture mapping
+
+ // from src surface to the temp texture
+ RECT srcRect = { sx1, sy1, sx2, sy2 };
+ RECT tmpDstRect = { 0l, 0l, 0+dw, 0+dh };
+ res = pd3dDevice->StretchRect(dstOps->pResource->GetSurface(), &srcRect,
+ pBlitSurface, &tmpDstRect,
+ D3DTEXF_NONE);
+ if (clipType != CLIP_SHAPE) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " rect or no clip, use StretchRect");
+ // just do stretch rect to the destination
+ RECT dstRect = { dx1, dy1, dx2, dy2 };
+ // from temp surface to the destination
+ res = pd3dDevice->StretchRect(pBlitSurface, &tmpDstRect,
+ dstOps->pResource->GetSurface(),
+ &dstRect,
+ D3DTEXF_NONE);
+ } else {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " shape clip, use texture mapping");
+ // shape clip - have to use texture mapping
+ D3DTEXTUREFILTERTYPE fhint =
+ d3dc->IsTextureFilteringSupported(D3DTEXF_NONE) ?
+ D3DTEXF_NONE: D3DTEXF_POINT;
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, fhint);
+ res = d3dc->BeginScene(STATE_TEXTUREOP);
+ RETURN_STATUS_IF_FAILED(res);
+ res = d3dc->SetTexture(pBlitTexture);
+
+ float tx2 = (float)dw/(float)pBlitTextureRes->GetDesc()->Width;
+ float ty2 = (float)dh/(float)pBlitTextureRes->GetDesc()->Height;
+ res = d3dc->pVCacher->DrawTexture(
+ (float)dx1, (float)dy1, (float)dx2, (float)dy2,
+ 0.0f, 0.0f, tx2, ty2);
+ }
+ }
+ return res;
}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.h b/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.h
new file mode 100644
index 00000000000..9f549d2bbb9
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DBlitLoops.h
@@ -0,0 +1,89 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DBlitLoops_h_Included
+#define D3DBlitLoops_h_Included
+
+#include "sun_java2d_d3d_D3DBlitLoops.h"
+#include "D3DSurfaceData.h"
+#include "D3DContext.h"
+
+#define OFFSET_SRCTYPE sun_java2d_d3d_D3DBlitLoops_OFFSET_SRCTYPE
+#define OFFSET_HINT sun_java2d_d3d_D3DBlitLoops_OFFSET_HINT
+#define OFFSET_TEXTURE sun_java2d_d3d_D3DBlitLoops_OFFSET_TEXTURE
+#define OFFSET_RTT sun_java2d_d3d_D3DBlitLoops_OFFSET_RTT
+#define OFFSET_XFORM sun_java2d_d3d_D3DBlitLoops_OFFSET_XFORM
+#define OFFSET_ISOBLIT sun_java2d_d3d_D3DBlitLoops_OFFSET_ISOBLIT
+
+D3DPIPELINE_API HRESULT
+D3DBlitLoops_IsoBlit(JNIEnv *env,
+ D3DContext *d3dc, jlong pSrcOps, jlong pDstOps,
+ jboolean xform, jint hint,
+ jboolean texture, jboolean rtt,
+ jint sx1, jint sy1,
+ jint sx2, jint sy2,
+ jdouble dx1, jdouble dy1,
+ jdouble dx2, jdouble dy2);
+
+D3DPIPELINE_API HRESULT
+D3DBL_CopySurfaceToIntArgbImage(IDirect3DSurface9 *pSurface,
+ SurfaceDataRasInfo *pDstInfo,
+ jint srcx, jint srcy,
+ jint srcWidth, jint srcHeight,
+ jint dstx, jint dsty);
+
+D3DPIPELINE_API HRESULT
+D3DBL_CopyImageToIntXrgbSurface(SurfaceDataRasInfo *pSrcInfo,
+ int srctype,
+ D3DResource *pDstSurfaceRes,
+ jint srcx, jint srcy,
+ jint srcWidth, jint srcHeight,
+ jint dstx, jint dsty);
+
+HRESULT
+D3DBlitLoops_Blit(JNIEnv *env,
+ D3DContext *d3dc, jlong pSrcOps, jlong pDstOps,
+ jboolean xform, jint hint,
+ jint srctype, jboolean texture,
+ jint sx1, jint sy1,
+ jint sx2, jint sy2,
+ jdouble dx1, jdouble dy1,
+ jdouble dx2, jdouble dy2);
+
+HRESULT
+D3DBlitLoops_SurfaceToSwBlit(JNIEnv *env, D3DContext *d3dc,
+ jlong pSrcOps, jlong pDstOps, jint dsttype,
+ jint srcx, jint srcy,
+ jint dstx, jint dsty,
+ jint width, jint height);
+
+HRESULT
+D3DBlitLoops_CopyArea(JNIEnv *env,
+ D3DContext *d3dc, D3DSDOps *dstOps,
+ jint x, jint y,
+ jint width, jint height,
+ jint dx, jint dy);
+
+#endif /* D3DBlitLoops_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DBufImgOps.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DBufImgOps.cpp
new file mode 100644
index 00000000000..dcf2b632e8e
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DBufImgOps.cpp
@@ -0,0 +1,306 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include
+
+#include "D3DBufImgOps.h"
+#include "D3DContext.h"
+#include "D3DRenderQueue.h"
+#include "D3DSurfaceData.h"
+#include "GraphicsPrimitiveMgr.h"
+
+/**************************** ConvolveOp support ****************************/
+
+/**
+ * The maximum kernel size supported by the ConvolveOp shader.
+ */
+#define MAX_KERNEL_SIZE 25
+
+HRESULT
+D3DBufImgOps_EnableConvolveOp(D3DContext *d3dc, jlong pSrcOps,
+ jboolean edgeZeroFill,
+ jint kernelWidth, jint kernelHeight,
+ unsigned char *kernel)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ D3DSDOps *srcOps = (D3DSDOps *)jlong_to_ptr(pSrcOps);
+ jint kernelSize = kernelWidth * kernelHeight;
+ jint texW, texH;
+ jfloat xoff, yoff;
+ jfloat edgeX, edgeY;
+ jfloat imgEdge[4];
+ jfloat kernelVals[MAX_KERNEL_SIZE*4];
+ jint i, j, kIndex;
+ jint flags = 0;
+
+ J2dTraceLn2(J2D_TRACE_INFO,
+ "D3DBufImgOps_EnableConvolveOp: kernelW=%d kernelH=%d",
+ kernelWidth, kernelHeight);
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(srcOps, E_FAIL);
+
+ d3dc->UpdateState(STATE_CHANGE);
+
+ // texcoords are specified in the range [0,1], so to achieve an
+ // x/y offset of approximately one pixel we have to normalize
+ // to that range here
+ texW = srcOps->pResource->GetDesc()->Width;
+ texH = srcOps->pResource->GetDesc()->Height;
+ xoff = 1.0f / texW;
+ yoff = 1.0f / texH;
+
+ if (edgeZeroFill) {
+ flags |= CONVOLVE_EDGE_ZERO_FILL;
+ }
+ if (kernelWidth == 5 && kernelHeight == 5) {
+ flags |= CONVOLVE_5X5;
+ }
+
+ // locate/enable the shader program for the given flags
+ res = d3dc->EnableConvolveProgram(flags);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the "uniform" image min/max values
+ // (texcoords are in the range [0,1])
+ // imgEdge[0] = imgMin.x
+ // imgEdge[1] = imgMin.y
+ // imgEdge[2] = imgMax.x
+ // imgEdge[3] = imgMax.y
+ edgeX = (kernelWidth/2) * xoff;
+ edgeY = (kernelHeight/2) * yoff;
+ imgEdge[0] = edgeX;
+ imgEdge[1] = edgeY;
+ imgEdge[2] = (((jfloat)srcOps->width) / texW) - edgeX;
+ imgEdge[3] = (((jfloat)srcOps->height) / texH) - edgeY;
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetPixelShaderConstantF(0, imgEdge, 1);
+
+ // update the "uniform" kernel offsets and values
+ kIndex = 0;
+ for (i = -kernelHeight/2; i < kernelHeight/2+1; i++) {
+ for (j = -kernelWidth/2; j < kernelWidth/2+1; j++) {
+ kernelVals[kIndex+0] = j*xoff;
+ kernelVals[kIndex+1] = i*yoff;
+ kernelVals[kIndex+2] = NEXT_FLOAT(kernel);
+ kernelVals[kIndex+3] = 0.0f; // unused
+ kIndex += 4;
+ }
+ }
+ return pd3dDevice->SetPixelShaderConstantF(1, kernelVals, kernelSize);
+}
+
+HRESULT
+D3DBufImgOps_DisableConvolveOp(D3DContext *d3dc)
+{
+ IDirect3DDevice9 *pd3dDevice;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBufImgOps_DisableConvolveOp");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ d3dc->UpdateState(STATE_CHANGE);
+
+ // disable the ConvolveOp shader
+ pd3dDevice = d3dc->Get3DDevice();
+ return pd3dDevice->SetPixelShader(NULL);
+}
+
+/**************************** RescaleOp support *****************************/
+
+HRESULT
+D3DBufImgOps_EnableRescaleOp(D3DContext *d3dc,
+ jboolean nonPremult,
+ unsigned char *scaleFactors,
+ unsigned char *offsets)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ jint flags = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBufImgOps_EnableRescaleOp");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ d3dc->UpdateState(STATE_CHANGE);
+
+ // choose the appropriate shader, depending on the source image
+ if (nonPremult) {
+ flags |= RESCALE_NON_PREMULT;
+ }
+
+ // locate/enable the shader program for the given flags
+ res = d3dc->EnableRescaleProgram(flags);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the "uniform" scale factor values (note that the Java-level
+ // dispatching code always passes down 4 values here, regardless of
+ // the original source image type)
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetPixelShaderConstantF(0, (float *)scaleFactors, 1);
+
+ // update the "uniform" offset values (note that the Java-level
+ // dispatching code always passes down 4 values here, and that the
+ // offsets will have already been normalized to the range [0,1])
+ return pd3dDevice->SetPixelShaderConstantF(1, (float *)offsets, 1);
+}
+
+HRESULT
+D3DBufImgOps_DisableRescaleOp(D3DContext *d3dc)
+{
+ IDirect3DDevice9 *pd3dDevice;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBufImgOps_DisableRescaleOp");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ d3dc->UpdateState(STATE_CHANGE);
+
+ // disable the RescaleOp shader
+ pd3dDevice = d3dc->Get3DDevice();
+ return pd3dDevice->SetPixelShader(NULL);
+}
+
+/**************************** LookupOp support ******************************/
+
+HRESULT
+D3DBufImgOps_EnableLookupOp(D3DContext *d3dc,
+ jboolean nonPremult, jboolean shortData,
+ jint numBands, jint bandLength, jint offset,
+ void *tableValues)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ D3DResource *pLutTexRes;
+ IDirect3DTexture9 *pLutTex;
+ int bytesPerElem = (shortData ? 2 : 1);
+ jfloat foffsets[4];
+ void *bands[4];
+ int i;
+ jint flags = 0;
+
+ J2dTraceLn4(J2D_TRACE_INFO,
+ "D3DBufImgOps_EnableLookupOp: short=%d num=%d len=%d off=%d",
+ shortData, numBands, bandLength, offset);
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ d3dc->UpdateState(STATE_CHANGE);
+
+ // choose the appropriate shader, depending on the source image
+ // and the number of bands involved
+ if (numBands != 4) {
+ flags |= LOOKUP_USE_SRC_ALPHA;
+ }
+ if (nonPremult) {
+ flags |= LOOKUP_NON_PREMULT;
+ }
+
+ // locate/enable the shader program for the given flags
+ res = d3dc->EnableLookupProgram(flags);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the "uniform" offset value
+ for (i = 0; i < 4; i++) {
+ foffsets[i] = offset / 255.0f;
+ }
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetPixelShaderConstantF(0, foffsets, 1);
+
+ res = d3dc->GetResourceManager()->GetLookupOpLutTexture(&pLutTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+ pLutTex = pLutTexRes->GetTexture();
+
+ // update the lookup table with the user-provided values
+ if (numBands == 1) {
+ // replicate the single band for R/G/B; alpha band is unused
+ for (i = 0; i < 3; i++) {
+ bands[i] = tableValues;
+ }
+ bands[3] = NULL;
+ } else if (numBands == 3) {
+ // user supplied band for each of R/G/B; alpha band is unused
+ for (i = 0; i < 3; i++) {
+ bands[i] = PtrAddBytes(tableValues, i*bandLength*bytesPerElem);
+ }
+ bands[3] = NULL;
+ } else if (numBands == 4) {
+ // user supplied band for each of R/G/B/A
+ for (i = 0; i < 4; i++) {
+ bands[i] = PtrAddBytes(tableValues, i*bandLength*bytesPerElem);
+ }
+ }
+
+ // upload the bands one row at a time into our lookup table texture
+ D3DLOCKED_RECT lockedRect;
+ res = pLutTex->LockRect(0, &lockedRect, NULL, D3DLOCK_NOSYSLOCK);
+ RETURN_STATUS_IF_FAILED(res);
+
+ jushort *pBase = (jushort*)lockedRect.pBits;
+ for (i = 0; i < 4; i++) {
+ jushort *pDst;
+ if (bands[i] == NULL) {
+ continue;
+ }
+ pDst = pBase + (i * 256);
+ if (shortData) {
+ memcpy(pDst, bands[i], bandLength*sizeof(jushort));
+ } else {
+ int j;
+ jubyte *pSrc = (jubyte *)bands[i];
+ for (j = 0; j < bandLength; j++) {
+ pDst[j] = (jushort)(pSrc[j] << 8);
+ }
+ }
+ }
+ pLutTex->UnlockRect(0);
+
+ // bind the lookup table to texture unit 1 and enable texturing
+ res = d3dc->SetTexture(pLutTex, 1);
+ pd3dDevice->SetSamplerState(1, D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(1, D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(1, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR);
+ pd3dDevice->SetSamplerState(1, D3DSAMP_MINFILTER, D3DTEXF_LINEAR);
+ return res;
+}
+
+HRESULT
+D3DBufImgOps_DisableLookupOp(D3DContext *d3dc)
+{
+ IDirect3DDevice9 *pd3dDevice;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DBufImgOps_DisableLookupOp");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ d3dc->UpdateState(STATE_CHANGE);
+
+ // disable the LookupOp shader
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetPixelShader(NULL);
+
+ // disable the lookup table on texture unit 1
+ return d3dc->SetTexture(NULL, 1);
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DBufImgOps.h b/jdk/src/windows/native/sun/java2d/d3d/D3DBufImgOps.h
new file mode 100644
index 00000000000..55ee250862b
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DBufImgOps.h
@@ -0,0 +1,78 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DBufImgOps_h_Included
+#define D3DBufImgOps_h_Included
+
+#include "D3DContext.h"
+
+/**************************** ConvolveOp support ****************************/
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define CONVOLVE_EDGE_ZERO_FILL (1 << 0)
+#define CONVOLVE_5X5 (1 << 1)
+#define MAX_CONVOLVE (1 << 2)
+
+HRESULT D3DBufImgOps_EnableConvolveOp(D3DContext *oglc, jlong pSrcOps,
+ jboolean edgeZeroFill,
+ jint kernelWidth, jint KernelHeight,
+ unsigned char *kernelVals);
+HRESULT D3DBufImgOps_DisableConvolveOp(D3DContext *oglc);
+
+/**************************** RescaleOp support *****************************/
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define RESCALE_NON_PREMULT (1 << 0)
+#define MAX_RESCALE (1 << 1)
+
+HRESULT D3DBufImgOps_EnableRescaleOp(D3DContext *oglc,
+ jboolean nonPremult,
+ unsigned char *scaleFactors,
+ unsigned char *offsets);
+HRESULT D3DBufImgOps_DisableRescaleOp(D3DContext *oglc);
+
+/**************************** LookupOp support ******************************/
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define LOOKUP_USE_SRC_ALPHA (1 << 0)
+#define LOOKUP_NON_PREMULT (1 << 1)
+#define MAX_LOOKUP (1 << 2)
+
+HRESULT D3DBufImgOps_EnableLookupOp(D3DContext *oglc,
+ jboolean nonPremult, jboolean shortData,
+ jint numBands, jint bandLength, jint offset,
+ void *tableValues);
+HRESULT D3DBufImgOps_DisableLookupOp(D3DContext *oglc);
+
+#endif /* D3DBufImgOps_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DContext.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DContext.cpp
index a8b4d8f4bd1..e3937b8d3e9 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DContext.cpp
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DContext.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -23,29 +23,19 @@
* have any questions.
*/
-#include "sun_java2d_d3d_D3DContext.h"
+#include "D3DPipeline.h"
#include "jlong.h"
-#include "jni_util.h"
-#include "Trace.h"
-
-#include "ddrawUtils.h"
-#include "awt_Win32GraphicsDevice.h"
-#include "sun_java2d_SunGraphics2D.h"
#include "GraphicsPrimitiveMgr.h"
-
-#include "RegistryKey.h"
-#include "WindowsFlags.h"
-
-#include "Win32SurfaceData.h"
-#include "D3DSurfaceData.h"
-#include "D3DUtils.h"
#include "D3DContext.h"
-#include "D3DRuntimeTest.h"
-
-#include "IntDcm.h"
-#include "IntArgb.h"
-#include "Region.h"
+#include "D3DSurfaceData.h"
+#include "D3DBufImgOps.h"
+#include "D3DPaints.h"
+#include "D3DRenderQueue.h"
+#include "D3DShaders.h"
+#include "D3DTextRenderer.h"
+#include "D3DPipelineManager.h"
+#include "D3DGlyphCache.h"
typedef struct {
D3DBLEND src;
@@ -73,472 +63,907 @@ D3DBlendRule StdBlendRules[] = {
{ D3DBLEND_INVDESTALPHA, D3DBLEND_INVSRCALPHA }, /*12 - RULE_AlphaXor*/
};
-/**
- * D3DContext
- */
-D3DContext* D3DContext::CreateD3DContext(DDraw *ddObject, DXObject* dxObject)
+void
+D3DUtils_SetOrthoMatrixOffCenterLH(D3DMATRIX *m,
+ float width, float height)
{
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::CreateD3DContext");
- // create and test the d3d context
- D3DContext *d3dContext = new D3DContext(ddObject, dxObject);
- // if there was a failure while creating or testing the device,
- // dispose of it and return NULL
- if (!(d3dContext->GetDeviceCaps() & J2D_D3D_ENABLED_OK)) {
- delete d3dContext;
- d3dContext = NULL;
- }
- return d3dContext;
+ ZeroMemory(m, sizeof(D3DMATRIX));
+ m->_11 = 2.0f/width;
+ m->_22 = -2.0f/height;
+ m->_33 = 0.5f;
+ m->_44 = 1.0f;
+
+ m->_41 = -1.0f;
+ m->_42 = 1.0f;
+ m->_43 = 0.5f;
}
-D3DContext::D3DContext(DDraw *ddObject, DXObject* dxObject)
+void
+D3DUtils_SetIdentityMatrix(D3DMATRIX *m)
{
- GetExclusiveAccess();
- J2dRlsTraceLn(J2D_TRACE_INFO, "D3DContext::D3DContext");
- J2dTraceLn2(J2D_TRACE_VERBOSE, " ddObject=0x%d dxObject=0x%x",
- ddObject, dxObject);
- d3dDevice = NULL;
- d3dObject = NULL;
- ddTargetSurface = NULL;
- lpMaskTexture = NULL;
- lpGlyphCacheTexture = NULL;
- glyphCache = NULL;
- glyphCacheAvailable = TRUE;
- deviceCaps = J2D_D3D_FAILURE;
- bBeginScenePending = FALSE;
- jD3DContext = NULL;
+ m->_12 = m->_13 = m->_14 = m->_21 = m->_23 = m->_24 = 0.0f;
+ m->_31 = m->_32 = m->_34 = m->_41 = m->_42 = m->_43 = 0.0f;
+ m->_11 = m->_22 = m->_33 = m->_44 = 1.0f;
+}
- this->dxObject = dxObject;
- this->ddObject = ddObject;
+// the following methods are copies of the AffineTransform's class
+// corresponding methods, with these changes to the indexes:
+// 00 -> 11
+// 11 -> 22
+// 01 -> 21
+// 10 -> 12
+// 02 -> 41
+// 12 -> 42
- if (SUCCEEDED(dxObject->CreateD3DObject(&d3dObject))) {
+void
+D3DUtils_2DConcatenateM(D3DMATRIX *m, D3DMATRIX *m1)
+{
+ float M0, M1;
+ float T00, T10, T01, T11;
+ float T02, T12;
- // The device type we choose to use doesn't change over time
- pDeviceGUID = D3DUtils_SelectDeviceGUID(d3dObject);
- if (pDeviceGUID) {
- bIsHWRasterizer = (*pDeviceGUID == IID_IDirect3DHALDevice ||
- *pDeviceGUID == IID_IDirect3DTnLHalDevice);
- CreateD3DDevice();
- } else {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3CCoD3DContext::D3DContext: Can't find "\
- "suitable D3D device");
- }
+ T00 = m1->_11; T01 = m1->_21; T02 = m1->_41;
+ T10 = m1->_12; T11 = m1->_22; T12 = m1->_42;
+
+ M0 = m->_11;
+ M1 = m->_21;
+ m->_11 = T00 * M0 + T10 * M1;
+ m->_21 = T01 * M0 + T11 * M1;
+ m->_41 += T02 * M0 + T12 * M1;
+
+ M0 = m->_12;
+ M1 = m->_22;
+ m->_12 = T00 * M0 + T10 * M1;
+ m->_22 = T01 * M0 + T11 * M1;
+ m->_42 += T02 * M0 + T12 * M1;
+}
+
+#ifdef UPDATE_TX
+
+void
+D3DUtils_2DScaleM(D3DMATRIX *m, float sx, float sy)
+{
+ m->_11 *= sx;
+ m->_22 *= sy;
+}
+
+void
+D3DUtils_2DInvertM(D3DMATRIX *m)
+{
+ float M11, M21, M41;
+ float M12, M22, M42;
+ float det;
+
+ M11 = m->_11; M21 = m->_21; M41 = m->_41;
+ M12 = m->_12; M22 = m->_22; M42 = m->_42;
+ det = M11 * M22 - M21 * M12;
+ if (fabs(det) <= 0.0000000001f) {
+ memset(m, 0, sizeof(D3DMATRIX));
+ return;
+ }
+ m->_11 = M22 / det;
+ m->_12 = -M12 / det;
+ m->_21 = -M21 / det;
+ m->_22 = M11 / det;
+ m->_41 = (M21 * M42 - M22 * M41) / det;
+ m->_42 = (M12 * M41 - M11 * M42) / det;
+}
+
+void
+D3DUtils_2DTranslateM(D3DMATRIX *m, float tx, float ty)
+{
+ m->_41 = tx * m->_11 + ty * m->_21 + m->_41;
+ m->_42 = tx * m->_12 + ty * m->_22 + m->_42;
+}
+
+void
+D3DUtils_2DTransformXY(D3DMATRIX *m, float *px, float *py)
+{
+ float x = *px;
+ float y = *py;
+
+ *px = x * m->_11 + y * m->_21 + m->_41;
+ *py = x * m->_12 + y * m->_22 + m->_42;
+}
+
+void
+D3DUtils_2DInverseTransformXY(D3DMATRIX *m, float *px, float *py)
+{
+ float x = *px, y = *py;
+
+ x -= m->_41;
+ y -= m->_42;
+
+ float det = m->_11 * m->_22 - m->_21 * m->_12;
+ if (fabs(det) < 0.0000000001f) {
+ *px = 0.0f;
+ *py = 0.0f;
} else {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3DContext::D3DContext: Can't "\
- "create IDirect3D7 interface");
+ *px = (x * m->_22 - y * m->_21) / det;
+ *py = (y * m->_11 - x * m->_12) / det;
}
+}
+
+#endif // UPDATE_TX
+
+static void
+D3DContext_DisposeShader(jlong programID)
+{
+ IDirect3DPixelShader9 *shader =
+ (IDirect3DPixelShader9 *)jlong_to_ptr(programID);
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext_DisposeShader");
+
+ SAFE_RELEASE(shader);
+}
+
+// static
+HRESULT
+D3DContext::CreateInstance(IDirect3D9 *pd3d9, UINT adapter, D3DContext **ppCtx)
+{
+ HRESULT res;
+ *ppCtx = new D3DContext(pd3d9, adapter);
+ if (FAILED(res = (*ppCtx)->InitContext())) {
+ delete *ppCtx;
+ *ppCtx = NULL;
+ }
+ return res;
+}
+
+D3DContext::D3DContext(IDirect3D9 *pd3d, UINT adapter)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::D3DContext");
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " pd3d=0x%x", pd3d);
+ pd3dObject = pd3d;
+ pd3dDevice = NULL;
+ adapterOrdinal = adapter;
+
+ pResourceMgr = NULL;
+ pMaskCache = NULL;
+ pVCacher = NULL;
+
+ pSyncQuery = NULL;
+ pSyncRTRes = NULL;
+ pStateBlock = NULL;
+
+ D3DC_INIT_SHADER_LIST(convolvePrograms, MAX_CONVOLVE);
+ D3DC_INIT_SHADER_LIST(rescalePrograms, MAX_RESCALE);
+ D3DC_INIT_SHADER_LIST(lookupPrograms, MAX_LOOKUP);
+ D3DC_INIT_SHADER_LIST(basicGradPrograms, 4);
+ D3DC_INIT_SHADER_LIST(linearGradPrograms, 8);
+ D3DC_INIT_SHADER_LIST(radialGradPrograms, 8);
+
+ pLCDGlyphCache= NULL;
+ pGrayscaleGlyphCache= NULL;
+ lcdTextProgram = NULL;
+ aaPgramProgram = NULL;
+
+ contextCaps = CAPS_EMPTY;
+ bBeginScenePending = FALSE;
+
+ ZeroMemory(&devCaps, sizeof(D3DCAPS9));
+ ZeroMemory(&curParams, sizeof(curParams));
- compState = sun_java2d_SunGraphics2D_COMP_ISCOPY;
extraAlpha = 1.0f;
- colorPixel = 0xffffffff;
-
- ReleaseExclusiveAccess();
}
-void D3DContext::SetJavaContext(JNIEnv *env, jobject newD3Dc) {
- GetExclusiveAccess();
+void D3DContext::ReleaseDefPoolResources()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::ReleaseDefPoolResources");
- // Only bother if the new D3DContext object is different
- // from the one we already have reference to.
- if (env->IsSameObject(newD3Dc, jD3DContext) == FALSE) {
- J2dTraceLn(J2D_TRACE_VERBOSE, "D3DContext:SetJavaContext: "\
- "setting new java context object");
- // invalidate the old context, since we've got a new one
- InvalidateIfTarget(env, ddTargetSurface);
+ EndScene();
- if (jD3DContext != NULL) {
- env->DeleteWeakGlobalRef(jD3DContext);
- }
- // set the new java-level context object
- jD3DContext = env->NewWeakGlobalRef(newD3Dc);
+ D3DPipelineManager::NotifyAdapterEventListeners(devCaps.AdapterOrdinal,
+ DEVICE_RESET);
+
+ contextCaps = CAPS_EMPTY;
+
+ SAFE_RELEASE(pSyncQuery);
+ SAFE_RELEASE(pStateBlock);
+
+ if (pVCacher != NULL) {
+ pVCacher->ReleaseDefPoolResources();
}
- ReleaseExclusiveAccess();
+ if (pMaskCache != NULL) {
+ pMaskCache->ReleaseDefPoolResources();
+ }
+ if (pLCDGlyphCache != NULL) {
+ pLCDGlyphCache->ReleaseDefPoolResources();
+ }
+ if (pGrayscaleGlyphCache != NULL) {
+ pGrayscaleGlyphCache->ReleaseDefPoolResources();
+ }
+ if (pResourceMgr != NULL) {
+ if (pSyncRTRes != NULL) {
+ pResourceMgr->ReleaseResource(pSyncRTRes);
+ pSyncRTRes = NULL;
+ }
+ pResourceMgr->ReleaseDefPoolResources();
+ }
+ ZeroMemory(lastTexture, sizeof(lastTexture));
+ ZeroMemory(lastTextureColorState, sizeof(lastTextureColorState));
}
-void D3DContext::Release3DDevice() {
- GetExclusiveAccess();
+void D3DContext::ReleaseContextResources()
+{
J2dTraceLn1(J2D_TRACE_INFO,
- "D3DContext::Release3DDevice: d3dDevice = 0x%x",
- d3dDevice);
+ "D3DContext::ReleaseContextResources: pd3dDevice = 0x%x",
+ pd3dDevice);
- // make sure we do EndScene if one is pending
- FlushD3DQueueForTarget(ddTargetSurface);
+ ReleaseDefPoolResources();
- // Let the java-level object know that the context
- // state is no longer valid, forcing it to be reinitialized
- // later.
- JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
- InvalidateIfTarget(env, ddTargetSurface);
+ D3DPipelineManager::NotifyAdapterEventListeners(devCaps.AdapterOrdinal,
+ DEVICE_DISPOSED);
- // We don't need to release it since we didn't create it
- ddTargetSurface = NULL;
+ // dispose shader lists
+ ShaderList_Dispose(&convolvePrograms);
+ ShaderList_Dispose(&rescalePrograms);
+ ShaderList_Dispose(&lookupPrograms);
+ ShaderList_Dispose(&basicGradPrograms);
+ ShaderList_Dispose(&linearGradPrograms);
+ ShaderList_Dispose(&radialGradPrograms);
- // disable the use of this context until we ensure the capabilities
- // of the new device and run the tests
- deviceCaps = J2D_D3D_FAILURE;
+ SAFE_DELETE(pLCDGlyphCache);
+ SAFE_DELETE(pGrayscaleGlyphCache);
- if (lpMaskTexture != NULL) {
- lpMaskTexture->Release();
- delete lpMaskTexture;
- lpMaskTexture = NULL;
- }
+ SAFE_RELEASE(lcdTextProgram);
+ SAFE_RELEASE(aaPgramProgram);
- // reset the depth buffer format
- memset(&depthBufferFormat, 0, sizeof(depthBufferFormat));
-
- if (d3dDevice) {
- // setting the texture increases its reference number, so
- // we should reset the textures for all stages to make sure
- // they're released
- for (int stage = 0; stage <= MAX_USED_TEXTURE_STAGE; stage++) {
- d3dDevice->SetTexture(stage, NULL);
- lastTexture[stage] = NULL;
- }
- d3dDevice->Release();
- d3dDevice = NULL;
- }
- ReleaseExclusiveAccess();
+ SAFE_DELETE(pVCacher);
+ SAFE_DELETE(pMaskCache);
+ SAFE_DELETE(pResourceMgr);
}
D3DContext::~D3DContext() {
J2dTraceLn2(J2D_TRACE_INFO,
- "~D3DContext: d3dDevice=0x%x, d3dObject =0x%x",
- d3dDevice, d3dObject);
- GetExclusiveAccess();
- if (lpGlyphCacheTexture != NULL) {
- lpGlyphCacheTexture->Release();
- delete lpGlyphCacheTexture;
- lpGlyphCacheTexture = NULL;
- }
- Release3DDevice();
- if (d3dObject != NULL) {
- d3dObject->Release();
- d3dObject = NULL;
- }
- ReleaseExclusiveAccess();
+ "~D3DContext: pd3dDevice=0x%x, pd3dObject =0x%x",
+ pd3dDevice, pd3dObject);
+ ReleaseContextResources();
+ SAFE_RELEASE(pd3dDevice);
}
HRESULT
-D3DContext::InitD3DDevice(IDirect3DDevice7 *d3dDevice)
+D3DContext::InitDevice(IDirect3DDevice9 *pd3dDevice)
{
- HRESULT res = D3D_OK;
+ HRESULT res = S_OK;
+
+ pd3dDevice->GetDeviceCaps(&devCaps);
+
J2dRlsTraceLn1(J2D_TRACE_INFO,
- "D3DContext::InitD3DDevice: d3dDevice=Ox%x", d3dDevice);
+ "D3DContext::InitDevice: device %d", adapterOrdinal);
- d3dDevice->GetCaps(&d3dDevDesc);
// disable some of the unneeded and costly d3d functionality
- d3dDevice->SetRenderState(D3DRENDERSTATE_CULLMODE, D3DCULL_NONE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_TEXTUREPERSPECTIVE, FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_SPECULARENABLE, FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_LIGHTING, FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_CLIPPING, FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_ZENABLE, D3DZB_FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_COLORVERTEX, FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_STENCILENABLE, FALSE);
+ pd3dDevice->SetRenderState(D3DRS_CULLMODE, D3DCULL_NONE);
+ pd3dDevice->SetRenderState(D3DRS_SPECULARENABLE, FALSE);
+ pd3dDevice->SetRenderState(D3DRS_LIGHTING, FALSE);
+ pd3dDevice->SetRenderState(D3DRS_CLIPPING, FALSE);
+ pd3dDevice->SetRenderState(D3DRS_ZENABLE, D3DZB_FALSE);
+ pd3dDevice->SetRenderState(D3DRS_ZWRITEENABLE, D3DZB_FALSE);
+ pd3dDevice->SetRenderState(D3DRS_COLORVERTEX, FALSE);
+ pd3dDevice->SetRenderState(D3DRS_STENCILENABLE, FALSE);
- d3dDevice->SetTextureStageState(0, D3DTSS_MAGFILTER, D3DTFG_POINT);
- d3dDevice->SetTextureStageState(0, D3DTSS_MINFILTER, D3DTFG_POINT);
+ // set the default texture addressing mode
+ pd3dDevice->SetSamplerState(0, D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP);
+
+ // REMIND: check supported filters with
+ // IDirect3D9::CheckDeviceFormat with D3DUSAGE_QUERY_FILTER
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, D3DTEXF_POINT);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, D3DTEXF_POINT);
// these states never change
- d3dDevice->SetTextureStageState(0, D3DTSS_ALPHAOP, D3DTOP_MODULATE);
- d3dDevice->SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_MODULATE);
- d3dDevice->SetTextureStageState(0, D3DTSS_ALPHAARG2, D3DTA_DIFFUSE);
- d3dDevice->SetTextureStageState(0, D3DTSS_COLORARG2, D3DTA_DIFFUSE);
+ pd3dDevice->SetTextureStageState(0, D3DTSS_ALPHAOP, D3DTOP_MODULATE);
+ pd3dDevice->SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_MODULATE);
+ pd3dDevice->SetTextureStageState(0, D3DTSS_ALPHAARG2, D3DTA_DIFFUSE);
+ pd3dDevice->SetTextureStageState(0, D3DTSS_COLORARG2, D3DTA_DIFFUSE);
+ pd3dDevice->SetTextureStageState(1, D3DTSS_ALPHAOP, D3DTOP_MODULATE);
+ pd3dDevice->SetTextureStageState(1, D3DTSS_COLOROP, D3DTOP_MODULATE);
+ pd3dDevice->SetTextureStageState(1, D3DTSS_ALPHAARG2, D3DTA_CURRENT);
+ pd3dDevice->SetTextureStageState(1, D3DTSS_COLORARG2, D3DTA_CURRENT);
+
// init the array of latest textures
- memset(&lastTexture, 0, sizeof(lastTexture));
- // this will force the state initialization on first UpdateState
- opState = STATE_UNDEFINED;
+ ZeroMemory(lastTexture, sizeof(lastTexture));
+ ZeroMemory(lastTextureColorState, sizeof(lastTextureColorState));
+
+ opState = STATE_CHANGE;
+
+ if (pResourceMgr == NULL) {
+ res = D3DResourceManager::CreateInstance(this, &pResourceMgr);
+ } else {
+ res = pResourceMgr->Init(this);
+ }
+ RETURN_STATUS_IF_FAILED(res);
+
+ if (pVCacher == NULL) {
+ res = D3DVertexCacher::CreateInstance(this, &pVCacher);
+ } else {
+ res = pVCacher->Init(this);
+ }
+ RETURN_STATUS_IF_FAILED(res);
+
+ if (pMaskCache == NULL) {
+ res = D3DMaskCache::CreateInstance(this, &pMaskCache);
+ } else{
+ res = pMaskCache->Init(this);
+ }
+ RETURN_STATUS_IF_FAILED(res);
+
+ if (pLCDGlyphCache != NULL) {
+ if (FAILED(res = pLCDGlyphCache->Init(this))) {
+ // we can live without the cache
+ SAFE_DELETE(pLCDGlyphCache);
+ res = S_OK;
+ }
+ }
+
+ if (pGrayscaleGlyphCache != NULL) {
+ if (FAILED(res = pGrayscaleGlyphCache->Init(this))) {
+ // we can live without the cache
+ SAFE_DELETE(pGrayscaleGlyphCache);
+ res = S_OK;
+ }
+ }
D3DMATRIX tx;
D3DUtils_SetIdentityMatrix(&tx);
- d3dDevice->SetTransform(D3DTRANSFORMSTATE_WORLD, &tx);
+ pd3dDevice->SetTransform(D3DTS_WORLD, &tx);
+ bIsIdentityTx = TRUE;
+
+ if (pSyncQuery == NULL) {
+ // this is allowed to fail, do not propagate the error
+ if (FAILED(pd3dDevice->CreateQuery(D3DQUERYTYPE_EVENT, &pSyncQuery))) {
+ J2dRlsTraceLn(J2D_TRACE_WARNING,
+ "D3DContext::InitDevice: sync query not available");
+ pSyncQuery = NULL;
+ }
+ }
+ if (pSyncRTRes == NULL) {
+ D3DFORMAT format;
+ if (FAILED(GetResourceManager()->
+ CreateRTSurface(32, 32, TRUE, TRUE, &format, &pSyncRTRes))) {
+ J2dRlsTraceLn(J2D_TRACE_WARNING,
+ "D3DContext::InitDevice: "
+ "error creating sync surface");
+ }
+ }
bBeginScenePending = FALSE;
- D3DUtils_SetupTextureFormats(d3dDevice, textureTable);
+ J2dRlsTraceLn1(J2D_TRACE_INFO,
+ "D3DContext::InitDefice: successfully initialized device %d",
+ adapterOrdinal);
- // REMIND: debugging: allows testing the argb path in
- // UploadImageToTexture on devices with alpha texture support
- if ((getenv("J2D_D3D_NOALPHATEXTURE") != NULL) ||
- FAILED(res = D3DUtils_FindMaskTileTextureFormat(d3dDevice,
- &maskTileTexFormat)))
- {
- // use ARGB if can't find alpha texture (or in case argb
- // was specifically requested)
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "D3DContext::InitD3DDevice: "\
- "Using IntARBG instead of Alpha texture");
- if (textureTable[TR_TRANSLUCENT_IDX][DEPTH32_IDX].pfType != PF_INVALID)
- {
- memcpy(&maskTileTexFormat,
- &textureTable[TR_TRANSLUCENT_IDX][DEPTH32_IDX].pddpf,
- sizeof(maskTileTexFormat));
- res = D3D_OK;
- }
- } else {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "D3DContext::InitD3DDevice: Found Alpha-texture format");
- }
return res;
}
HRESULT
-D3DContext::CreateAndTestD3DDevice(DxCapabilities *dxCaps)
+D3DContext::CheckAndResetDevice()
{
- J2dRlsTraceLn(J2D_TRACE_INFO, "D3DContext::CreateAndTestD3DDevice");
- HRESULT res;
- if (pDeviceGUID == NULL) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3DContext::CreateAndTestD3DDevice: "\
- "No usable d3d device");
- deviceCaps = J2D_D3D_FAILURE;
- return DDERR_GENERIC;
- }
+ HRESULT res = E_FAIL;
- Release3DDevice();
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::CheckAndResetDevice");
- // Create a temp surface so we can use it when creating a device
- DXSurface *target = NULL;
- if (FAILED(res = CreateSurface(NULL, 10, 10, 32, TR_OPAQUE,
- D3D_PLAIN_SURFACE|D3D_RENDER_TARGET,
- &target, NULL)))
- {
- DebugPrintDirectDrawError(res,
- "D3DContext::CreateAndTestD3DDevice: "\
- "can't create scratch surface");
- return res;
- }
-
- if (FAILED(res = d3dObject->CreateDevice(*pDeviceGUID,
- target->GetDDSurface(),
- &d3dDevice)))
- {
- DebugPrintDirectDrawError(res,
- "D3DContext::CreateAndTestD3DDevice: "\
- "error creating d3d device");
- } else if (FAILED(res = InitD3DDevice(d3dDevice))) {
- DebugPrintDirectDrawError(res,
- "D3DContext::CreateAndTestD3DDevice: "\
- "error initializing D3D device");
- } else {
- J2dRlsTraceLn(J2D_TRACE_VERBOSE,
- "D3DContext::CreateAndTestD3DDevice: "\
- "D3D device creation/initialization successful");
- // the device is successfully created and initialized,
- // now run some tests on it
- deviceCaps = TestD3DDevice(ddObject, this, dxCaps);
- }
-
- // We can safely dispose the scratch surface here
- if (target != NULL) {
- target->Release();
- delete target;
- }
-
- return res;
-}
-
-void
-D3DContext::CreateD3DDevice()
-{
- GetExclusiveAccess();
- J2dRlsTraceLn(J2D_TRACE_INFO, "D3DContext::CreateD3DDevice");
- // this is a weird way of getting a handle on the ddInstance
- HMONITOR hMonitor = dxObject->GetHMonitor();
- DxCapabilities *dxCaps =
- AwtWin32GraphicsDevice::GetDxCapsForDevice(hMonitor);
-
- int d3dCapsValidity = dxCaps->GetD3dCapsValidity();
- // Always run the test unless we crashed doing so the last time.
- // The reasons:
- // - the user may have disabled d3d acceleration in the display panel
- // since the last run
- // - the user may have installed the new drivers, which may cause BSODs
- // - if the test had failed previously because of quality issues, the
- // new driver may have fixed the problem, but we'd never know since we
- // never try again
- // - user (or developer, rather) may have specified a
- // different rasterizer via env. variable
- if (d3dCapsValidity != J2D_ACCEL_TESTING) {
- dxCaps->SetD3dCapsValidity(J2D_ACCEL_TESTING);
-
- // this will create the device, test it and set the
- // deviceCaps
- CreateAndTestD3DDevice(dxCaps);
-
- dxCaps->SetD3dDeviceCaps(deviceCaps);
- dxCaps->SetD3dCapsValidity(J2D_ACCEL_SUCCESS);
- }
- int requiredResults = forceD3DUsage ?
- J2D_D3D_REQUIRED_RESULTS : J2D_D3D_DESIRED_RESULTS;
-
-#ifdef DEBUG
- J2dTraceLn(J2D_TRACE_VERBOSE, "CreateD3DDevice: requested caps:");
- PrintD3DCaps(requiredResults);
- J2dTraceLn(J2D_TRACE_VERBOSE, " caps supported by the device:");
- PrintD3DCaps(deviceCaps);
- J2dTraceLn(J2D_TRACE_VERBOSE, " missing caps:");
- PrintD3DCaps(requiredResults & ~deviceCaps);
-#endif // DEBUG
-
- if ((deviceCaps & requiredResults) != requiredResults) {
- if (!(deviceCaps & J2D_D3D_HW_OK)) {
- // disable d3d for all devices, because we've encountered
- // known bad hardware. See comment in TestForBadHardware().
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CreateD3DDevice: bad hardware found,"\
- " disabling d3d for all devices.");
- SetD3DEnabledFlag(NULL, FALSE, FALSE);
+ if (pd3dDevice != NULL) {
+ if (FAILED(res = pd3dDevice->TestCooperativeLevel())) {
+ if (res == D3DERR_DEVICELOST) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " device %d is still lost",
+ adapterOrdinal);
+ // nothing to be done here, wait for D3DERR_DEVICENOTRESET
+ return res;
+ } else if (res == D3DERR_DEVICENOTRESET) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " device %d needs to be reset",
+ adapterOrdinal);
+ res = ResetContext();
+ } else {
+ // some unexpected error
+ DebugPrintD3DError(res, "D3DContext::CheckAndResetDevice: "\
+ "unknown error %x from TestCooperativeLevel");
+ }
} else {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CreateD3DDevice: tests FAILED, d3d disabled.");
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " device %d is not lost",
+ adapterOrdinal);
}
- // REMIND: the first time the context initialization fails,
- // deviceUseD3D is set to FALSE in DDrawObjectStruct, and because of
- // this we never attempt to initialize it again later.
- // For example, if the app switches to a display mode where
- // d3d is not supported, we disable d3d, but it stays disabled
- // even when the display mode is switched back to a supported one.
- // May be we should disable it only in case of a hard error.
- ddObject->DisableD3D();
- Release3DDevice();
} else {
- deviceCaps |= J2D_D3D_ENABLED_OK;
- J2dRlsTraceLn1(J2D_TRACE_INFO,
- "CreateD3DDevice: tests PASSED, "\
- "d3d enabled (forced: %s).",
- forceD3DUsage ? "yes" : "no");
+ J2dTraceLn(J2D_TRACE_VERBOSE, " null device");
}
-
- ReleaseExclusiveAccess();
-}
-
-HRESULT
-D3DContext::SetRenderTarget(DDrawSurface *ddSurface)
-{
- static D3DVIEWPORT7 vp = { 0, 0, 0, 0, 0.0f, 1.0f };
- static D3DMATRIX tx;
- BOOL bSetProjectionMatrix = FALSE;
- HRESULT res = DDERR_GENERIC;
- GetExclusiveAccess();
-
- J2dTraceLn2(J2D_TRACE_INFO,
- "D3DContext::SetRenderTarget: old=0x%x new=0x%x",
- ddTargetSurface, ddSurface);
-
- ddTargetSurface = NULL;
-
- DXSurface *dxSurface = NULL;
- if (d3dDevice == NULL || ddSurface == NULL ||
- (dxSurface = ddSurface->GetDXSurface()) == NULL)
- {
- ReleaseExclusiveAccess();
- J2dTraceLn3(J2D_TRACE_WARNING,
- "D3DContext::SetRenderTarget invalid state:"\
- "d3dDevice=0x%x ddSurface=0x%x dxSurface=0x%x",
- d3dDevice, ddSurface, dxSurface);
- return res;
- }
-
- if (FAILED(res = ddSurface->IsLost())) {
- ReleaseExclusiveAccess();
- DebugPrintDirectDrawError(res, "D3DContext::SetRenderTarget: "\
- "target surface (and/or depth buffer) lost");
- return res;
- }
-
- ForceEndScene();
-
- if (FAILED(res = d3dDevice->SetRenderTarget(dxSurface->GetDDSurface(), 0)))
- {
- ReleaseExclusiveAccess();
- DebugPrintDirectDrawError(res, "D3DContext::SetRenderTarget: "\
- "error setting render target");
- return res;
- }
-
- int width = dxSurface->GetWidth();
- int height = dxSurface->GetHeight();
- // set the projection matrix if the the dimensions of the new
- // rendertarget are different from the old one.
- if (FAILED(d3dDevice->GetViewport(&vp)) ||
- (int)vp.dwWidth != width || (int)vp.dwHeight != height)
- {
- bSetProjectionMatrix = TRUE;
- }
-
- vp.dwX = vp.dwY = 0;
- vp.dwWidth = width;
- vp.dwHeight = height;
- vp.dvMinZ = 0.0f;
- vp.dvMaxZ = 1.0f;
-
- if (FAILED(res = d3dDevice->SetViewport(&vp))) {
- DebugPrintDirectDrawError(res, "D3DContext::SetRenderTarget: "\
- "error setting viewport");
- ReleaseExclusiveAccess();
- return res;
- }
-
- if (bSetProjectionMatrix) {
- D3DUtils_SetOrthoMatrixOffCenterLH(&tx, (float)width, (float)height);
- res = d3dDevice->SetTransform(D3DTRANSFORMSTATE_PROJECTION, &tx);
- }
-
- if (SUCCEEDED(res)) {
- ddTargetSurface = ddSurface;
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "D3DContext::SetRenderTarget: succeeded, "\
- "new target=0x%x", ddTargetSurface);
- } else {
- DebugPrintDirectDrawError(res, "D3DContext::SetRenderTarget: failed");
- }
-
- ReleaseExclusiveAccess();
return res;
}
HRESULT
-D3DContext::SetTransform(jobject xform,
- jdouble m00, jdouble m10,
+D3DContext::ResetContext()
+{
+ HRESULT res = E_FAIL;
+
+ J2dRlsTraceLn(J2D_TRACE_INFO, "D3DContext::ResetContext");
+ if (pd3dDevice != NULL) {
+ D3DPRESENT_PARAMETERS newParams;
+
+ newParams = curParams;
+
+ if (newParams.Windowed) {
+ // reset to the current display mode if we're windowed,
+ // otherwise to the display mode we were in when the device
+ // was lost
+ newParams.BackBufferFormat = D3DFMT_UNKNOWN;
+ newParams.FullScreen_RefreshRateInHz = 0;
+ newParams.BackBufferWidth = 0;
+ newParams.BackBufferHeight = 0;
+ }
+ res = ConfigureContext(&newParams);
+ }
+ return res;
+}
+
+HRESULT
+D3DContext::ConfigureContext(D3DPRESENT_PARAMETERS *pNewParams)
+{
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "D3DContext::ConfigureContext device %d",
+ adapterOrdinal);
+ HRESULT res = S_OK;
+ D3DFORMAT stencilFormat;
+ HWND focusHWND = D3DPipelineManager::GetInstance()->GetCurrentFocusWindow();
+ D3DDEVTYPE devType = D3DPipelineManager::GetInstance()->GetDeviceType();
+ // this is needed so that we can find the stencil buffer format
+ if (pNewParams->BackBufferFormat == D3DFMT_UNKNOWN) {
+ D3DDISPLAYMODE dm;
+
+ pd3dObject->GetAdapterDisplayMode(adapterOrdinal, &dm);
+ pNewParams->BackBufferFormat = dm.Format;
+ }
+
+ stencilFormat =
+ D3DPipelineManager::GetInstance()->GetMatchingDepthStencilFormat(
+ adapterOrdinal,
+ pNewParams->BackBufferFormat, pNewParams->BackBufferFormat);
+
+ pNewParams->EnableAutoDepthStencil = TRUE;
+ pNewParams->AutoDepthStencilFormat = stencilFormat;
+
+ // do not set device window in the windowed mode, we use additional
+ // swap chains for rendering, the default chain is not used. otherwise
+ // our scratch focus window will be made visible
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " windowed=%d",pNewParams->Windowed);
+ if (pNewParams->Windowed) {
+ pNewParams->hDeviceWindow = (HWND)0;
+ }
+
+ // The focus window may change when we're entering/exiting the full-screen
+ // mode. It may either be set to the default focus window (when there are
+ // no more devices in fs mode), or to fs window for another device
+ // in fs mode. See D3DPipelineManager::GetCurrentFocusWindow.
+ if (pd3dDevice != NULL) {
+ D3DDEVICE_CREATION_PARAMETERS cParams;
+ pd3dDevice->GetCreationParameters(&cParams);
+ if (cParams.hFocusWindow != focusHWND) {
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " focus window changed, need to recreate the device");
+
+ // if fs -> windowed, first exit fs, then recreate, otherwise
+ // the screen might be left in a different display mode
+ if (pNewParams->Windowed && !curParams.Windowed) {
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " exiting full-screen mode, reset the device");
+ curParams.Windowed = FALSE;
+ ReleaseDefPoolResources();
+ res = pd3dDevice->Reset(&curParams);
+
+ if (FAILED(res)) {
+ DebugPrintD3DError(res, "D3DContext::ConfigureContext: "\
+ "cound not reset the device");
+ }
+ }
+
+ // note that here we should release all device resources, not only
+ // thos in the default pool since the device is released
+ ReleaseContextResources();
+ SAFE_RELEASE(pd3dDevice);
+ }
+ }
+
+ if (pd3dDevice != NULL) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " resetting the device");
+
+ ReleaseDefPoolResources();
+
+ if (pNewParams->PresentationInterval == D3DPRESENT_INTERVAL_IMMEDIATE &&
+ !IsImmediateIntervalSupported())
+ {
+ pNewParams->PresentationInterval = D3DPRESENT_INTERVAL_DEFAULT;
+ }
+
+ res = pd3dDevice->Reset(pNewParams);
+ if (FAILED(res)) {
+ DebugPrintD3DError(res,
+ "D3DContext::ConfigureContext: cound not reset the device");
+ return res;
+ }
+ J2dRlsTraceLn1(J2D_TRACE_INFO,
+ "D3DContext::ConfigureContext: successfully reset device: %d",
+ adapterOrdinal);
+ } else {
+ D3DCAPS9 d3dCaps;
+ DWORD dwBehaviorFlags;
+
+ J2dTraceLn(J2D_TRACE_VERBOSE, " creating a new device");
+
+ if (FAILED(res = pd3dObject->GetDeviceCaps(adapterOrdinal,
+ devType, &d3dCaps)))
+ {
+ DebugPrintD3DError(res,
+ "D3DContext::ConfigureContext: failed to get caps");
+ return res;
+ }
+
+ if (pNewParams->PresentationInterval == D3DPRESENT_INTERVAL_IMMEDIATE &&
+ !(d3dCaps.PresentationIntervals & D3DPRESENT_INTERVAL_IMMEDIATE))
+ {
+ pNewParams->PresentationInterval = D3DPRESENT_INTERVAL_DEFAULT;
+ }
+
+ // not preserving fpu control word could cause issues (4860749)
+ dwBehaviorFlags = D3DCREATE_FPU_PRESERVE;
+
+ J2dRlsTrace(J2D_TRACE_VERBOSE,
+ "[V] dwBehaviorFlags=D3DCREATE_FPU_PRESERVE|");
+ if (d3dCaps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT) {
+ J2dRlsTrace(J2D_TRACE_VERBOSE,
+ "D3DCREATE_HARDWARE_VERTEXPROCESSING");
+ dwBehaviorFlags |= D3DCREATE_HARDWARE_VERTEXPROCESSING;
+ } else {
+ J2dRlsTrace(J2D_TRACE_VERBOSE,
+ "D3DCREATE_SOFTWARE_VERTEXPROCESSING");
+ dwBehaviorFlags |= D3DCREATE_SOFTWARE_VERTEXPROCESSING;
+ }
+ // Handling focus changes by ourselves proved to be problematic,
+ // so we're reverting back to D3D handling
+ // dwBehaviorFlags |= D3DCREATE_NOWINDOWCHANGES;
+ J2dRlsTrace(J2D_TRACE_VERBOSE,"\n");
+
+ if (FAILED(res = pd3dObject->CreateDevice(adapterOrdinal, devType,
+ focusHWND,
+ dwBehaviorFlags,
+ pNewParams, &pd3dDevice)))
+ {
+ DebugPrintD3DError(res,
+ "D3DContext::ConfigureContext: error creating d3d device");
+ return res;
+ }
+ J2dRlsTraceLn1(J2D_TRACE_INFO,
+ "D3DContext::ConfigureContext: successfully created device: %d",
+ adapterOrdinal);
+ bIsHWRasterizer = (devType == D3DDEVTYPE_HAL);
+ }
+
+ curParams = *pNewParams;
+ // during the creation of the device d3d modifies this field, we reset
+ // it back to 0
+ curParams.Flags = 0;
+
+ if (FAILED(res = InitDevice(pd3dDevice))) {
+ ReleaseContextResources();
+ return res;
+ }
+
+ res = InitContextCaps();
+
+ return res;
+}
+
+HRESULT
+D3DContext::InitContext()
+{
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "D3DContext::InitContext device %d",
+ adapterOrdinal);
+
+ D3DPRESENT_PARAMETERS params;
+ ZeroMemory(¶ms, sizeof(D3DPRESENT_PARAMETERS));
+
+ params.hDeviceWindow = 0;
+ params.Windowed = TRUE;
+ params.BackBufferCount = 1;
+ params.BackBufferFormat = D3DFMT_UNKNOWN;
+ params.SwapEffect = D3DSWAPEFFECT_DISCARD;
+ params.PresentationInterval = D3DPRESENT_INTERVAL_DEFAULT;
+
+ return ConfigureContext(¶ms);
+}
+
+HRESULT
+D3DContext::Sync()
+{
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::Sync");
+
+ if (pSyncQuery != NULL) {
+ J2dTrace(J2D_TRACE_VERBOSE, " flushing the device queue..");
+ while (S_FALSE ==
+ (res = pSyncQuery->GetData(NULL, 0, D3DGETDATA_FLUSH))) ;
+ J2dTrace(J2D_TRACE_VERBOSE, ".. done\n");
+ }
+ if (pSyncRTRes != NULL) {
+ D3DLOCKED_RECT lr;
+ IDirect3DSurface9 *pSurface = pSyncRTRes->GetSurface();
+ if (SUCCEEDED(pSurface->LockRect(&lr, NULL, D3DLOCK_NOSYSLOCK))) {
+ pSurface->UnlockRect();
+ }
+ }
+ return res;
+}
+
+HRESULT
+D3DContext::SaveState()
+{
+ HRESULT res;
+
+ RETURN_STATUS_IF_NULL(pd3dDevice, S_OK);
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::SaveState");
+
+ FlushVertexQueue();
+ UpdateState(STATE_CHANGE);
+
+ if (pStateBlock != NULL) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DContext::SaveState: existing state block!");
+ SAFE_RELEASE(pStateBlock);
+ }
+
+ if (SUCCEEDED(res =
+ pd3dDevice->CreateStateBlock(D3DSBT_ALL, &pStateBlock)))
+ {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " created state block");
+ } else {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DContext::SaveState: failed to create state block");
+ }
+ ZeroMemory(lastTexture, sizeof(lastTexture));
+
+ return res;
+}
+
+HRESULT
+D3DContext::RestoreState()
+{
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::RestoreState");
+
+ FlushVertexQueue();
+ UpdateState(STATE_CHANGE);
+
+ if (pStateBlock != NULL) {
+ if (SUCCEEDED(res = pStateBlock->Apply())) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " restored device state");
+ } else {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DContext::RestoreState: failed to restore state");
+ }
+ SAFE_RELEASE(pStateBlock);
+ } else {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DContext::RestoreState: empty state block!");
+ }
+ ZeroMemory(lastTexture, sizeof(lastTexture));
+
+ return res;
+}
+
+#define POINT_FILTER_CAP (D3DPTFILTERCAPS_MAGFPOINT|D3DPTFILTERCAPS_MINFPOINT)
+#define LINEAR_FILTER_CAP (D3DPTFILTERCAPS_MAGFLINEAR|D3DPTFILTERCAPS_MINFLINEAR)
+
+BOOL
+D3DContext::IsStretchRectFilteringSupported(D3DTEXTUREFILTERTYPE fType)
+{
+ if (fType == D3DTEXF_POINT) {
+ return ((devCaps.StretchRectFilterCaps & POINT_FILTER_CAP) != 0);
+ }
+ if (fType == D3DTEXF_LINEAR) {
+ return ((devCaps.StretchRectFilterCaps & LINEAR_FILTER_CAP) != 0);
+ }
+ return FALSE;
+}
+
+BOOL
+D3DContext::IsTextureFilteringSupported(D3DTEXTUREFILTERTYPE fType)
+{
+ if (fType == D3DTEXF_POINT) {
+ return ((devCaps.TextureFilterCaps & POINT_FILTER_CAP) != 0);
+ }
+ if (fType == D3DTEXF_LINEAR) {
+ return ((devCaps.TextureFilterCaps & LINEAR_FILTER_CAP) != 0);
+ }
+ return FALSE;
+}
+
+BOOL
+D3DContext::IsTextureFormatSupported(D3DFORMAT format, DWORD usage)
+{
+ HRESULT hr = pd3dObject->CheckDeviceFormat(adapterOrdinal,
+ devCaps.DeviceType,
+ curParams.BackBufferFormat,
+ usage,
+ D3DRTYPE_TEXTURE,
+ format);
+ return SUCCEEDED( hr );
+}
+
+BOOL
+D3DContext::IsDepthStencilBufferOk(D3DSURFACE_DESC *pTargetDesc)
+{
+ IDirect3DSurface9 *pStencil;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::IsDepthStencilBufferOk");
+
+ if (SUCCEEDED(pd3dDevice->GetDepthStencilSurface(&pStencil))) {
+ D3DSURFACE_DESC descStencil;
+ pStencil->GetDesc(&descStencil);
+ pStencil->Release();
+
+ D3DDISPLAYMODE dm;
+ return
+ (SUCCEEDED(pd3dDevice->GetDisplayMode(0, &dm)) &&
+ pTargetDesc->Width <= descStencil.Width &&
+ pTargetDesc->Height <= descStencil.Height &&
+ SUCCEEDED(pd3dObject->CheckDepthStencilMatch(
+ adapterOrdinal,
+ devCaps.DeviceType,
+ dm.Format, pTargetDesc->Format,
+ descStencil.Format)));
+ }
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " current stencil buffer is not compatible with new Render Target");
+
+ return false;
+}
+
+
+
+HRESULT
+D3DContext::InitDepthStencilBuffer(D3DSURFACE_DESC *pTargetDesc)
+{
+ HRESULT res;
+ IDirect3DSurface9 *pBB;
+ D3DDISPLAYMODE dm;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::InitDepthStencilBuffer");
+
+ if (FAILED(res = pd3dDevice->GetDisplayMode(0, &dm))) {
+ return res;
+ }
+
+ D3DFORMAT newFormat =
+ D3DPipelineManager::GetInstance()->GetMatchingDepthStencilFormat(
+ adapterOrdinal, dm.Format, pTargetDesc->Format);
+
+ res = pd3dDevice->CreateDepthStencilSurface(
+ pTargetDesc->Width, pTargetDesc->Height,
+ newFormat, D3DMULTISAMPLE_NONE, 0, false, &pBB, 0);
+ if (SUCCEEDED(res)) {
+ res = pd3dDevice->SetDepthStencilSurface(pBB);
+ pBB->Release();
+ }
+
+ return res;
+}
+
+
+HRESULT
+D3DContext::SetRenderTarget(IDirect3DSurface9 *pSurface)
+{
+ static D3DMATRIX tx;
+ HRESULT res;
+ D3DSURFACE_DESC descNew;
+ IDirect3DSurface9 *pCurrentTarget;
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DContext::SetRenderTarget: pSurface=0x%x",
+ pSurface);
+
+ RETURN_STATUS_IF_NULL(pd3dDevice, E_FAIL);
+ RETURN_STATUS_IF_NULL(pSurface, E_FAIL);
+
+ pSurface->GetDesc(&descNew);
+
+ if (SUCCEEDED(res = pd3dDevice->GetRenderTarget(0, &pCurrentTarget))) {
+ if (pCurrentTarget != pSurface) {
+ FlushVertexQueue();
+ if (FAILED(res = pd3dDevice->SetRenderTarget(0, pSurface))) {
+ DebugPrintD3DError(res, "D3DContext::SetRenderTarget: "\
+ "error setting render target");
+ SAFE_RELEASE(pCurrentTarget);
+ return res;
+ }
+
+ if (!IsDepthStencilBufferOk(&descNew)) {
+ if (FAILED(res = InitDepthStencilBuffer(&descNew))) {
+ SAFE_RELEASE(pCurrentTarget);
+ return res;
+ }
+ }
+ }
+ SAFE_RELEASE(pCurrentTarget);
+ }
+ // we set the transform even if the render target didn't change;
+ // this is because in some cases (fs mode) we use the default SwapChain of
+ // the device, and its render target will be the same as the device's, and
+ // we have to set the matrix correctly. This shouldn't be a performance
+ // issue as render target changes are relatively rare
+ D3DUtils_SetOrthoMatrixOffCenterLH(&tx,
+ (float)descNew.Width,
+ (float)descNew.Height);
+ pd3dDevice->SetTransform(D3DTS_PROJECTION, &tx);
+
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " current render target=0x%x", pSurface);
+ return res;
+}
+
+HRESULT
+D3DContext::ResetTransform()
+{
+ HRESULT res = S_OK;
+ D3DMATRIX tx;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::ResetTransform");
+ if (pd3dDevice == NULL) {
+ return E_FAIL;
+ }
+
+ // no need for state change, just flush the queue
+ FlushVertexQueue();
+
+ D3DUtils_SetIdentityMatrix(&tx);
+ if (FAILED(res = pd3dDevice->SetTransform(D3DTS_WORLD, &tx))) {
+ DebugPrintD3DError(res, "D3DContext::SetTransform failed");
+ }
+ bIsIdentityTx = TRUE;
+ return res;
+}
+
+HRESULT
+D3DContext::SetTransform(jdouble m00, jdouble m10,
jdouble m01, jdouble m11,
jdouble m02, jdouble m12)
{
- GetExclusiveAccess();
+ HRESULT res = S_OK;
+ D3DMATRIX tx, tx1;
J2dTraceLn(J2D_TRACE_INFO, "D3DContext::SetTransform");
- if (d3dDevice == NULL) {
- ReleaseExclusiveAccess();
- return DDERR_GENERIC;
- }
- HRESULT res = D3D_OK;
- D3DMATRIX tx;
-
- if (xform == NULL) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " disabling transform");
- D3DUtils_SetIdentityMatrix(&tx);
- } else {
- J2dTraceLn(J2D_TRACE_VERBOSE, " enabling transform");
-
- // copy values from AffineTransform object into native matrix array
- memset(&tx, 0, sizeof(D3DMATRIX));
- tx._11 = (float)m00;
- tx._12 = (float)m10;
- tx._21 = (float)m01;
- tx._22 = (float)m11;
- // The -0.5 adjustment is needed to correctly align texels to
- // pixels with orgthogonal projection matrix.
- // Note that we readjust vertex coordinates for cases
- // when we don't do texture mapping or use D3DPT_LINESTRIP.
- tx._41 = (float)m02-0.5f;
- tx._42 = (float)m12-0.5f;
-
- tx._33 = 1.0f;
- tx._44 = 1.0f;
+ if (pd3dDevice == NULL) {
+ return E_FAIL;
}
- J2dTraceLn(J2D_TRACE_VERBOSE, " setting new tx matrix");
+ // no need for state change, just flush the queue
+ FlushVertexQueue();
+
+ // In order to correctly map texels to pixels we need to
+ // adjust geometry by -0.5f in the transformed space.
+ // In order to do that we first create a translated matrix
+ // and then concatenate it with the world transform.
+ //
+ // Note that we only use non-id transform with DrawTexture,
+ // the rest is rendered pre-transformed.
+ //
+ // The identity transform for textures is handled in
+ // D3DVertexCacher::DrawTexture() because shifting by -0.5 for id
+ // transform breaks lines rendering.
+
+ ZeroMemory(&tx1, sizeof(D3DMATRIX));
+
+ tx1._11 = (float)m00;
+ tx1._12 = (float)m10;
+ tx1._21 = (float)m01;
+ tx1._22 = (float)m11;
+ tx1._41 = (float)m02;
+ tx1._42 = (float)m12;
+
+ tx1._33 = 1.0f;
+ tx1._44 = 1.0f;
+
+ D3DUtils_SetIdentityMatrix(&tx);
+ tx._41 = -0.5f;
+ tx._42 = -0.5f;
+ D3DUtils_2DConcatenateM(&tx, &tx1);
+
J2dTraceLn4(J2D_TRACE_VERBOSE,
" %5f %5f %5f %5f", tx._11, tx._12, tx._13, tx._14);
J2dTraceLn4(J2D_TRACE_VERBOSE,
@@ -547,103 +972,134 @@ D3DContext::SetTransform(jobject xform,
" %5f %5f %5f %5f", tx._31, tx._32, tx._33, tx._34);
J2dTraceLn4(J2D_TRACE_VERBOSE,
" %5f %5f %5f %5f", tx._41, tx._42, tx._43, tx._44);
- if (FAILED(res = d3dDevice->SetTransform(D3DTRANSFORMSTATE_WORLD, &tx))) {
- DebugPrintDirectDrawError(res, "D3DContext::SetTransform failed");
+ if (FAILED(res = pd3dDevice->SetTransform(D3DTS_WORLD, &tx))) {
+ DebugPrintD3DError(res, "D3DContext::SetTransform failed");
}
+ bIsIdentityTx = FALSE;
- ReleaseExclusiveAccess();
return res;
}
+HRESULT
+D3DContext::SetRectClip(int x1, int y1, int x2, int y2)
+{
+ HRESULT res = S_OK;
+ D3DSURFACE_DESC desc;
+ IDirect3DSurface9 *pCurrentTarget;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::SetRectClip");
+ J2dTraceLn4(J2D_TRACE_VERBOSE,
+ " x1=%-4d y1=%-4d x2=%-4d y2=%-4d",
+ x1, y1, x2, y2);
+
+ RETURN_STATUS_IF_NULL(pd3dDevice, E_FAIL);
+
+ // no need for state change, just flush the queue
+ FlushVertexQueue();
+
+ pd3dDevice->SetRenderState(D3DRS_ZENABLE, D3DZB_FALSE);
+
+ res = pd3dDevice->GetRenderTarget(0, &pCurrentTarget);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pCurrentTarget->GetDesc(&desc);
+ SAFE_RELEASE(pCurrentTarget);
+
+ if (x1 <= 0 && y1 <= 0 &&
+ (UINT)x2 >= desc.Width && (UINT)y2 >= desc.Height)
+ {
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " disabling clip (== render target dimensions)");
+ return pd3dDevice->SetRenderState(D3DRS_SCISSORTESTENABLE, FALSE);
+ }
+
+ // clip to the dimensions of the target surface, otherwise
+ // SetScissorRect will fail
+ if (x1 < 0) x1 = 0;
+ if (y1 < 0) y1 = 0;
+ if ((UINT)x2 > desc.Width) x2 = desc.Width;
+ if ((UINT)y2 > desc.Height) y2 = desc.Height;
+ if (x1 > x2) x2 = x1 = 0;
+ if (y1 > y2) y2 = y1 = 0;
+ RECT newRect = { x1, y1, x2, y2 };
+ if (SUCCEEDED(res = pd3dDevice->SetScissorRect(&newRect))) {
+ res = pd3dDevice->SetRenderState(D3DRS_SCISSORTESTENABLE, TRUE);
+ } else {
+ DebugPrintD3DError(res, "Error setting scissor rect");
+ J2dRlsTraceLn4(J2D_TRACE_ERROR,
+ " x1=%-4d y1=%-4d x2=%-4d y2=%-4d",
+ x1, y1, x2, y2);
+ }
+
+ return res;
+}
+
+HRESULT
+D3DContext::ResetClip()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::ResetClip");
+ // no need for state change, just flush the queue
+ FlushVertexQueue();
+ pd3dDevice->SetRenderState(D3DRS_SCISSORTESTENABLE, FALSE);
+ return pd3dDevice->SetRenderState(D3DRS_ZENABLE, D3DZB_FALSE);
+}
+
+ClipType
+D3DContext::GetClipType()
+{
+ // REMIND: this method could be optimized: we could keep the
+ // clip state around when re/setting the clip instead of asking
+ // every time.
+ DWORD zEnabled = 0;
+ DWORD stEnabled = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::GetClipType");
+ pd3dDevice->GetRenderState(D3DRS_SCISSORTESTENABLE, &stEnabled);
+ if (stEnabled) {
+ return CLIP_RECT;
+ }
+ pd3dDevice->GetRenderState(D3DRS_ZENABLE, &zEnabled);
+ if (zEnabled) {
+ return CLIP_SHAPE;
+ }
+ return CLIP_NONE;
+}
+
+
/**
* This method assumes that ::SetRenderTarget has already
* been called. SetRenderTarget creates and attaches a
* depth buffer to the target surface prior to setting it
* as target surface to the device.
*/
+DWORD dwAlphaSt, dwSrcBlendSt, dwDestBlendSt;
+D3DMATRIX tx, idTx;
+
HRESULT
-D3DContext::SetClip(JNIEnv *env, jobject clip,
- jboolean isRect,
- int x1, int y1,
- int x2, int y2)
+D3DContext::BeginShapeClip()
{
- HRESULT res;
- static J2D_XY_VERTEX clipRect[] = {
-#ifdef USE_SINGLE_VERTEX_FORMAT
- { 0.0f, 0.0f, 1.0f, 0xffffffff, 0.0f, 0.0f },
- { 0.0f, 0.0f, 1.0f, 0xffffffff, 0.0f, 0.0f },
- { 0.0f, 0.0f, 1.0f, 0xffffffff, 0.0f, 0.0f },
- { 0.0f, 0.0f, 1.0f, 0xffffffff, 0.0f, 0.0f }
-#else
- // Note that we use D3DFVF_XYZ vertex format
- // implies 0xffffffff diffuse color, so we don't
- // have to specify it.
- { 0.0f, 0.0f, 1.0f },
- { 0.0f, 0.0f, 1.0f },
- { 0.0f, 0.0f, 1.0f },
- { 0.0f, 0.0f, 1.0f },
-#endif // USE_SINGLE_VERTEX_FORMAT
- };
- static J2DXY_HEXA spanVx[MAX_CACHED_SPAN_VX_NUM];
+ HRESULT res = S_OK;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::BeginShapeClip");
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::SetClip");
- J2dTraceLn5(J2D_TRACE_VERBOSE,
- " x1=%-4d y1=%-4d x2=%-4d y2=%-4d isRect=%-2d",
- x1, y1, x2, y2, isRect);
- GetExclusiveAccess();
- // the target surface must already be set
- if (d3dDevice == NULL || ddTargetSurface == NULL) {
- ReleaseExclusiveAccess();
- return DDERR_GENERIC;
- }
+ UpdateState(STATE_CHANGE);
+ pd3dDevice->SetRenderState(D3DRS_SCISSORTESTENABLE, FALSE);
- // Must do EndScene prior to setting a new clip, otherwise the
- // primitives which are already in the pipeline will be rendered with
- // the new clip when we do EndScene.
- ForceEndScene();
+ // save alpha blending state
+ pd3dDevice->GetRenderState(D3DRS_ALPHABLENDENABLE, &dwAlphaSt);
+ pd3dDevice->GetRenderState(D3DRS_SRCBLEND, &dwSrcBlendSt);
+ pd3dDevice->GetRenderState(D3DRS_DESTBLEND, &dwDestBlendSt);
- if (clip == NULL) {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "D3DContext::SetClip: disabling clip (== NULL)");
- res = d3dDevice->SetRenderState(D3DRENDERSTATE_ZENABLE, D3DZB_FALSE);
- ReleaseExclusiveAccess();
- return res;
- } else if (isRect) {
- // optimization: disable depth buffer if the clip is equal to
- // the size of the viewport
- int w = ddTargetSurface->GetDXSurface()->GetWidth();
- int h = ddTargetSurface->GetDXSurface()->GetHeight();
- if (x1 == 0 && y1 == 0 && x2 == w && y2 == h) {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "D3DContext::SetClip: disabling clip (== viewport)");
- res = d3dDevice->SetRenderState(D3DRENDERSTATE_ZENABLE, D3DZB_FALSE);
- ReleaseExclusiveAccess();
- return res;
- }
- }
+ pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE, TRUE);
+ pd3dDevice->SetRenderState(D3DRS_SRCBLEND, D3DBLEND_ZERO);
+ pd3dDevice->SetRenderState(D3DRS_DESTBLEND, D3DBLEND_ONE);
- // save the old settings
- DWORD dwAlphaSt, dwSrcBlendSt, dwDestBlendSt;
- d3dDevice->GetRenderState(D3DRENDERSTATE_ALPHABLENDENABLE, &dwAlphaSt);
- d3dDevice->GetRenderState(D3DRENDERSTATE_SRCBLEND, &dwSrcBlendSt);
- d3dDevice->GetRenderState(D3DRENDERSTATE_DESTBLEND, &dwDestBlendSt);
-
- d3dDevice->SetRenderState(D3DRENDERSTATE_ALPHABLENDENABLE, TRUE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_SRCBLEND, D3DBLEND_ZERO);
- d3dDevice->SetRenderState(D3DRENDERSTATE_DESTBLEND, D3DBLEND_ONE);
-
- // disable texturing
- if (lastTexture[0] != NULL) {
- // note that we do not restore the texture after we set the clip,
- // it will be reset the next time a texturing operation is performed
- SetTexture(NULL);
- }
-
- D3DMATRIX tx, idTx;
- d3dDevice->GetTransform(D3DTRANSFORMSTATE_WORLD, &tx);
+ pd3dDevice->GetTransform(D3DTS_WORLD, &tx);
D3DUtils_SetIdentityMatrix(&idTx);
- d3dDevice->SetTransform(D3DTRANSFORMSTATE_WORLD, &idTx);
+ // translate the clip spans by 1.0f in z direction so that the
+ // clip spans are rendered to the z buffer
+ idTx._43 = 1.0f;
+ pd3dDevice->SetTransform(D3DTS_WORLD, &idTx);
// The depth buffer is first cleared with zeroes, which is the farthest
// plane from the viewer (our projection matrix is an inversed orthogonal
@@ -653,545 +1109,381 @@ D3DContext::SetClip(JNIEnv *env, jobject clip,
// have their vertices' Z coordinate set to 0.0, they will effectively be
// clipped because the Z depth test for them will fail (vertex with 1.0
// depth is closer than the one with 0.0f)
- d3dDevice->SetRenderState(D3DRENDERSTATE_ZENABLE, D3DZB_TRUE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_ZWRITEENABLE, TRUE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_ZFUNC, D3DCMP_ALWAYS);
- d3dDevice->Clear(0, NULL, D3DCLEAR_ZBUFFER, 0L, 0.0f, 0x0L);
+ pd3dDevice->SetRenderState(D3DRS_ZENABLE, D3DZB_TRUE);
+ pd3dDevice->SetRenderState(D3DRS_ZWRITEENABLE, TRUE);
+ pd3dDevice->SetRenderState(D3DRS_ZFUNC, D3DCMP_ALWAYS);
+ pd3dDevice->Clear(0, NULL, D3DCLEAR_ZBUFFER, 0L, 0.0f, 0x0L);
- float fx1, fy1, fx2, fy2;
- if (SUCCEEDED(d3dDevice->BeginScene())) {
- if (isRect) {
- fx1 = (float)x1; fy1 = (float)y1;
- fx2 = (float)x2; fy2 = (float)y2;
- D3DU_INIT_VERTEX_QUAD_XY(clipRect, fx1, fy1, fx2, fy2);
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN, D3DFVF_XY_VERTEX,
- clipRect, 4, NULL);
- } else {
- RegionData clipInfo;
- Region_GetInfo(env, clip, &clipInfo);
- SurfaceDataBounds span;
- J2DXY_HEXA *pHexa = (J2DXY_HEXA*)spanVx;
- jint numOfCachedSpans = 0;
+ //res = BeginScene(STATE_SHAPE_CLIPOP);
- Region_StartIteration(env, &clipInfo);
- while (Region_NextIteration(&clipInfo, &span)) {
- fx1 = (float)(span.x1); fy1 = (float)(span.y1);
- fx2 = (float)(span.x2); fy2 = (float)(span.y2);
- D3DU_INIT_VERTEX_XYZ_6(*pHexa, fx1, fy1, fx2, fy2, 1.0f);
- numOfCachedSpans++;
- pHexa = (J2DXY_HEXA*)PtrAddBytes(pHexa, sizeof(J2DXY_HEXA));
- if (numOfCachedSpans >= MAX_CACHED_SPAN_VX_NUM) {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLELIST,
- D3DFVF_XY_VERTEX,
- (void*)spanVx,
- 6*numOfCachedSpans, NULL);
- numOfCachedSpans = 0;
- pHexa = (J2DXY_HEXA*)spanVx;
- if (FAILED(res)) {
- break;
- }
- }
- }
- if (numOfCachedSpans > 0) {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLELIST,
- D3DFVF_XY_VERTEX,
- (void*)spanVx,
- 6*numOfCachedSpans, NULL);
- }
- Region_EndIteration(env, &clipInfo);
- }
- res = d3dDevice->EndScene();
- }
-
- // reset the transform
- d3dDevice->SetTransform(D3DTRANSFORMSTATE_WORLD, &tx);
-
- // reset the alpha compositing
- d3dDevice->SetRenderState(D3DRENDERSTATE_ALPHABLENDENABLE, dwAlphaSt);
- d3dDevice->SetRenderState(D3DRENDERSTATE_SRCBLEND, dwSrcBlendSt);
- d3dDevice->SetRenderState(D3DRENDERSTATE_DESTBLEND, dwDestBlendSt);
-
- // Setup the depth buffer.
- // We disable further updates to the depth buffer: it should only
- // be updated in SetClip method.
- d3dDevice->SetRenderState(D3DRENDERSTATE_ZWRITEENABLE, FALSE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_ZFUNC, D3DCMP_LESS);
-
- ReleaseExclusiveAccess();
return res;
}
-DXSurface *
-D3DContext::GetMaskTexture()
-{
- if (lpMaskTexture != NULL) {
- // This in theory should never happen since
- // we're using managed textures, but in case
- // we switch to using something else.
- if (FAILED(lpMaskTexture->IsLost())) {
- lpMaskTexture->Restore();
- }
- return lpMaskTexture;
- }
- InitMaskTileTexture();
- return lpMaskTexture;
-}
-
-
HRESULT
-D3DContext::InitMaskTileTexture()
+D3DContext::EndShapeClip()
{
HRESULT res;
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::InitMaskTileTexture");
- if (lpMaskTexture != NULL) {
- lpMaskTexture->Release();
- }
- lpMaskTexture = NULL;
+ // no need for state change, just flush the queue
+ res = FlushVertexQueue();
- DWORD caps2 = 0, caps = DDSCAPS_TEXTURE;
- if (bIsHWRasterizer) {
- caps2 = DDSCAPS2_TEXTUREMANAGE;
- } else {
- caps |= DDSCAPS_SYSTEMMEMORY;
- }
+ // restore alpha blending state
+ pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE, dwAlphaSt);
+ pd3dDevice->SetRenderState(D3DRS_SRCBLEND, dwSrcBlendSt);
+ pd3dDevice->SetRenderState(D3DRS_DESTBLEND, dwDestBlendSt);
+
+ // resore the transform
+ pd3dDevice->SetTransform(D3DTS_WORLD, &tx);
+
+ // Enable the depth buffer.
+ // We disable further updates to the depth buffer: it should only
+ // be updated in SetClip method.
+ pd3dDevice->SetRenderState(D3DRS_ZWRITEENABLE, FALSE);
+ pd3dDevice->SetRenderState(D3DRS_ZFUNC, D3DCMP_LESS);
- if (FAILED(res =
- dxObject->CreateSurface(DDSD_WIDTH|DDSD_HEIGHT|DDSD_CAPS|
- DDSD_PIXELFORMAT|DDSD_TEXTURESTAGE,
- caps,
- caps2,
- &maskTileTexFormat,
- D3DSD_MASK_TILE_SIZE, D3DSD_MASK_TILE_SIZE,
- (DXSurface **)&lpMaskTexture, 0)))
- {
- // in case we want to do something here later..
- DebugPrintDirectDrawError(res,
- "D3DContext::InitMaskTileTexture: "\
- "failed to create mask tile texture");
- }
return res;
}
HRESULT
-D3DContext::UploadImageToTexture(DXSurface *texture, jubyte *pixels,
- jint dstx, jint dsty,
- jint srcx, jint srcy,
- jint srcWidth, jint srcHeight,
- jint srcStride)
+D3DContext::UploadTileToTexture(D3DResource *pTextureRes, void *pixels,
+ jint dstx, jint dsty,
+ jint srcx, jint srcy,
+ jint srcWidth, jint srcHeight,
+ jint srcStride,
+ TileFormat srcFormat,
+ jint *pPixelsTouchedL,
+ jint* pPixelsTouchedR)
{
- HRESULT res = D3D_OK;
- SurfaceDataRasInfo rasInfo;
-
+#ifndef PtrAddBytes
+#define PtrAddBytes(p, b) ((void *) (((intptr_t) (p)) + (b)))
+#define PtrCoord(p, x, xinc, y, yinc) PtrAddBytes(p, (y)*(yinc) + (x)*(xinc))
+#endif // PtrAddBytes
+ HRESULT res = S_OK;
+ IDirect3DTexture9 *pTexture = pTextureRes->GetTexture();
+ D3DSURFACE_DESC *pDesc = pTextureRes->GetDesc();
RECT r = { dstx, dsty, dstx+srcWidth, dsty+srcHeight };
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::UploadImageToTexture");
+ RECT *pR = &r;
+ D3DLOCKED_RECT lockedRect;
+ DWORD dwLockFlags = D3DLOCK_NOSYSLOCK;
+ // these are only counted for LCD glyph uploads
+ jint pixelsTouchedL = 0, pixelsTouchedR = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::UploadTileToTexture");
J2dTraceLn4(J2D_TRACE_VERBOSE,
- " rect={%-4d, %-4d, %-4d, %-4d}",
- r.left, r.top, r.right, r.bottom);
- // REMIND: it may be faster to lock for NULL instead of
- // rect, need to test later.
- if (FAILED(res = texture->Lock(&r, &rasInfo,
- DDLOCK_WAIT|DDLOCK_NOSYSLOCK, NULL)))
+ " rect={%-4d, %-4d, %-4d, %-4d}",
+ r.left, r.top, r.right, r.bottom);
+
+ // REMIND: we should also check for dstx, dsty being 0 here,
+ // but they're always 0 in dynamic texture case
+ if (pDesc->Usage == D3DUSAGE_DYNAMIC &&
+ srcWidth == pDesc->Width && srcHeight == pDesc->Height)
{
- DebugPrintDirectDrawError(res,
- "D3DContext::UploadImageToTexture: could "\
- "not lock texture");
+ dwLockFlags |= D3DLOCK_DISCARD;
+ pR = NULL;
+ }
+
+ if (FAILED(res = pTexture->LockRect(0, &lockedRect, pR, dwLockFlags))) {
+ DebugPrintD3DError(res,
+ "D3DContext::UploadImageToTexture: could "\
+ "not lock texture");
return res;
}
- if (rasInfo.pixelStride == 1) {
- // 8bpp alpha texture
- void *pSrcPixels = PtrCoord(pixels, srcx, 1, srcy, srcStride);
- void *pDstPixels = rasInfo.rasBase;
- do {
- memcpy(pDstPixels, pSrcPixels, srcWidth);
- pSrcPixels = PtrAddBytes(pSrcPixels, srcStride);
- pDstPixels = PtrAddBytes(pDstPixels, rasInfo.scanStride);
- } while (--srcHeight > 0);
- } else {
- // ARGB texture
- jubyte *pSrcPixels = (jubyte*)PtrCoord(pixels, srcx, 1, srcy, srcStride);
- jint *pDstPixels = (jint*)rasInfo.rasBase;
- for (int yy = 0; yy < srcHeight; yy++) {
- for (int xx = 0; xx < srcWidth; xx++) {
- jubyte pix = pSrcPixels[xx];
- StoreIntArgbFrom4ByteArgb(pDstPixels, 0, xx,
- pix, pix, pix, pix);
- }
- pSrcPixels = (jubyte*)PtrAddBytes(pSrcPixels, srcStride);
- pDstPixels = (jint*)PtrAddBytes(pDstPixels, rasInfo.scanStride);
+ if (srcFormat == TILEFMT_1BYTE_ALPHA) {
+ // either a MaskFill tile, or a grayscale glyph
+ if (pDesc->Format == D3DFMT_A8) {
+ void *pSrcPixels = PtrCoord(pixels, srcx, 1, srcy, srcStride);
+ void *pDstPixels = lockedRect.pBits;
+ do {
+ memcpy(pDstPixels, pSrcPixels, srcWidth);
+ pSrcPixels = PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ } while (--srcHeight > 0);
}
+ else if (pDesc->Format == D3DFMT_A8R8G8B8) {
+ jubyte *pSrcPixels = (jubyte*)
+ PtrCoord(pixels, srcx, 1, srcy, srcStride);
+ jint *pDstPixels = (jint*)lockedRect.pBits;
+ for (int yy = 0; yy < srcHeight; yy++) {
+ for (int xx = 0; xx < srcWidth; xx++) {
+ // only need to set the alpha channel (the D3D texture
+ // state will be setup in this case to replicate the
+ // alpha channel as needed)
+ pDstPixels[xx] = pSrcPixels[xx] << 24;
+ }
+ pSrcPixels = (jubyte*)PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = (jint*)PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ }
+ }
+ } else if (srcFormat == TILEFMT_3BYTE_RGB) {
+ // LCD glyph with RGB order
+ if (pDesc->Format == D3DFMT_R8G8B8) {
+ jubyte *pSrcPixels = (jubyte*)
+ PtrCoord(pixels, srcx, 3, srcy, srcStride);
+ jubyte *pDstPixels = (jubyte*)lockedRect.pBits;
+ for (int yy = 0; yy < srcHeight; yy++) {
+ for (int xx = 0; xx < srcWidth*3; xx+=3) {
+ // alpha channel is ignored in this case
+ // (note that this is backwards from what one might
+ // expect; it appears that D3DFMT_R8G8B8 is actually
+ // laid out in BGR order in memory)
+ pDstPixels[xx+0] = pSrcPixels[xx+2];
+ pDstPixels[xx+1] = pSrcPixels[xx+1];
+ pDstPixels[xx+2] = pSrcPixels[xx+0];
+ }
+ pixelsTouchedL +=
+ (pDstPixels[0+0]|pDstPixels[0+1]|pDstPixels[0+2]) ? 1 : 0;
+ jint i = 3*(srcWidth-1);
+ pixelsTouchedR +=
+ (pDstPixels[i+0]|pDstPixels[i+1]|pDstPixels[i+2]) ? 1 : 0;
+
+ pSrcPixels = (jubyte*)PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = (jubyte*)PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ }
+ }
+ else if (pDesc->Format == D3DFMT_A8R8G8B8) {
+ jubyte *pSrcPixels = (jubyte*)
+ PtrCoord(pixels, srcx, 3, srcy, srcStride);
+ jint *pDstPixels = (jint*)lockedRect.pBits;
+ for (int yy = 0; yy < srcHeight; yy++) {
+ for (int dx = 0, sx = 0; dx < srcWidth; dx++, sx+=3) {
+ // alpha channel is ignored in this case
+ jubyte r = pSrcPixels[sx+0];
+ jubyte g = pSrcPixels[sx+1];
+ jubyte b = pSrcPixels[sx+2];
+ pDstPixels[dx] = (r << 16) | (g << 8) | (b);
+ }
+ pixelsTouchedL += (pDstPixels[0] ? 1 : 0);
+ pixelsTouchedR += (pDstPixels[srcWidth-1] ? 1 : 0);
+
+ pSrcPixels = (jubyte*)PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = (jint*)PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ }
+ }
+ } else if (srcFormat == TILEFMT_3BYTE_BGR) {
+ // LCD glyph with BGR order
+ if (pDesc->Format == D3DFMT_R8G8B8) {
+ void *pSrcPixels = PtrCoord(pixels, srcx, 3, srcy, srcStride);
+ void *pDstPixels = lockedRect.pBits;
+ jubyte *pbDst;
+ do {
+ // alpha channel is ignored in this case
+ // (note that this is backwards from what one might
+ // expect; it appears that D3DFMT_R8G8B8 is actually
+ // laid out in BGR order in memory)
+ memcpy(pDstPixels, pSrcPixels, srcWidth * 3);
+
+ pbDst = (jubyte*)pDstPixels;
+ pixelsTouchedL +=(pbDst[0+0]|pbDst[0+1]|pbDst[0+2]) ? 1 : 0;
+ jint i = 3*(srcWidth-1);
+ pixelsTouchedR +=(pbDst[i+0]|pbDst[i+1]|pbDst[i+2]) ? 1 : 0;
+
+ pSrcPixels = PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ } while (--srcHeight > 0);
+ }
+ else if (pDesc->Format == D3DFMT_A8R8G8B8) {
+ jubyte *pSrcPixels = (jubyte*)
+ PtrCoord(pixels, srcx, 3, srcy, srcStride);
+ jint *pDstPixels = (jint*)lockedRect.pBits;
+ for (int yy = 0; yy < srcHeight; yy++) {
+ for (int dx = 0, sx = 0; dx < srcWidth; dx++, sx+=3) {
+ // alpha channel is ignored in this case
+ jubyte b = pSrcPixels[sx+0];
+ jubyte g = pSrcPixels[sx+1];
+ jubyte r = pSrcPixels[sx+2];
+ pDstPixels[dx] = (r << 16) | (g << 8) | (b);
+ }
+ pixelsTouchedL += (pDstPixels[0] ? 1 : 0);
+ pixelsTouchedR += (pDstPixels[srcWidth-1] ? 1 : 0);
+
+ pSrcPixels = (jubyte*)PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = (jint*)PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ }
+ }
+ } else if (srcFormat == TILEFMT_4BYTE_ARGB_PRE) {
+ // MaskBlit tile
+ if (pDesc->Format == D3DFMT_A8R8G8B8) {
+ void *pSrcPixels = PtrCoord(pixels, srcx, 4, srcy, srcStride);
+ void *pDstPixels = lockedRect.pBits;
+ do {
+ memcpy(pDstPixels, pSrcPixels, srcWidth * 4);
+ pSrcPixels = PtrAddBytes(pSrcPixels, srcStride);
+ pDstPixels = PtrAddBytes(pDstPixels, lockedRect.Pitch);
+ } while (--srcHeight > 0);
+ }
+ } else {
+ // should not happen, no-op just in case...
}
- return texture->Unlock(&r);
+
+ if (pPixelsTouchedL) {
+ *pPixelsTouchedL = pixelsTouchedL;
+ }
+ if (pPixelsTouchedR) {
+ *pPixelsTouchedR = pixelsTouchedR;
+ }
+
+ return pTexture->UnlockRect(0);
}
HRESULT
-D3DContext::InitGlyphCache()
+D3DContext::InitLCDGlyphCache()
{
- HRESULT res = D3D_OK;
-
- if (glyphCache != NULL) {
- return D3D_OK;
+ if (pLCDGlyphCache == NULL) {
+ return D3DGlyphCache::CreateInstance(this, CACHE_LCD, &pLCDGlyphCache);
}
-
- if (!glyphCacheAvailable) {
- return DDERR_GENERIC;
- }
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::InitGlyphCache");
-
- // init glyph cache data structure
- glyphCache = AccelGlyphCache_Init(D3D_GCACHE_WIDTH,
- D3D_GCACHE_HEIGHT,
- D3D_GCACHE_CELL_WIDTH,
- D3D_GCACHE_CELL_HEIGHT,
- NULL);
- if (glyphCache == NULL) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3DContext::InitGlyphCache: "\
- "could not init D3D glyph cache");
- glyphCacheAvailable = FALSE;
- return DDERR_GENERIC;
- }
-
- DWORD caps2 = 0, caps = DDSCAPS_TEXTURE;
- if (bIsHWRasterizer) {
- caps2 = DDSCAPS2_TEXTUREMANAGE;
- } else {
- caps |= DDSCAPS_SYSTEMMEMORY;
- }
- if (FAILED(res =
- dxObject->CreateSurface(DDSD_WIDTH|DDSD_HEIGHT|DDSD_CAPS|
- DDSD_PIXELFORMAT|DDSD_TEXTURESTAGE,
- caps,
- caps2,
- &maskTileTexFormat,
- D3D_GCACHE_WIDTH, D3D_GCACHE_HEIGHT,
- (DXSurface **)&lpGlyphCacheTexture, 0)))
- {
- DebugPrintDirectDrawError(res,
- "D3DContext::InitGlyphCache: glyph cache "\
- "texture creation failed");
- glyphCacheAvailable = FALSE;
- return res;
- }
- return res;
+ return S_OK;
}
HRESULT
-D3DContext::GlyphCacheAdd(JNIEnv *env, GlyphInfo *glyph)
+D3DContext::InitGrayscaleGlyphCache()
{
- HRESULT res = D3D_OK;
- if (!glyphCacheAvailable || glyph->image == NULL) {
- return DDERR_GENERIC;
+ if (pGrayscaleGlyphCache == NULL) {
+ return D3DGlyphCache::CreateInstance(this, CACHE_GRAY,
+ &pGrayscaleGlyphCache);
}
-
- AccelGlyphCache_AddGlyph(glyphCache, glyph);
-
- if (glyph->cellInfo != NULL) {
- // store glyph image in texture cell
- res = UploadImageToTexture(lpGlyphCacheTexture, (jubyte*)glyph->image,
- glyph->cellInfo->x, glyph->cellInfo->y,
- 0, 0,
- glyph->width, glyph->height,
- glyph->width);
- }
-
- return res;
+ return S_OK;
}
-void
-D3DContext::SetColor(jint eargb, jint flags)
-{
- J2dTraceLn2(J2D_TRACE_INFO,
- "D3DContext::SetColor: eargb=%08x flags=%d", eargb, flags);
-
- /*
- * The colorPixel field is a 32-bit ARGB premultiplied color
- * value. The incoming eargb field is a 32-bit ARGB value
- * that is not premultiplied. If the alpha is not 1.0 (255)
- * then we need to premultiply the color components before
- * storing it in the colorPixel field.
- */
- jint a = (eargb >> 24) & 0xff;
-
- if (a == 0xff) {
- colorPixel = eargb;
- } else {
- jint a2 = a + (a >> 7);
- jint r = (((eargb >> 16) & 0xff) * a2) >> 8;
- jint g = (((eargb >> 8) & 0xff) * a2) >> 8;
- jint b = (((eargb ) & 0xff) * a2) >> 8;
- colorPixel = (a << 24) | (r << 16) | (g << 8) | (b << 0);
- }
- J2dTraceLn1(J2D_TRACE_VERBOSE, " updated color: colorPixel=%08x",
- colorPixel);
-}
-
-void
+HRESULT
D3DContext::ResetComposite()
{
J2dTraceLn(J2D_TRACE_INFO, "D3DContext::ResetComposite");
- GetExclusiveAccess();
- if (d3dDevice == NULL) {
- ReleaseExclusiveAccess();
- return;
- }
- d3dDevice->SetRenderState(D3DRENDERSTATE_ALPHABLENDENABLE, TRUE);
- compState = sun_java2d_SunGraphics2D_COMP_ISCOPY;
+
+ RETURN_STATUS_IF_NULL(pd3dDevice, E_FAIL);
+
+ HRESULT res = UpdateState(STATE_CHANGE);
+ pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE, FALSE);
extraAlpha = 1.0f;
- ReleaseExclusiveAccess();
+ return res;
}
-void
+HRESULT
D3DContext::SetAlphaComposite(jint rule, jfloat ea, jint flags)
{
+ HRESULT res;
J2dTraceLn3(J2D_TRACE_INFO,
"D3DContext::SetAlphaComposite: rule=%-1d ea=%f flags=%d",
rule, ea, flags);
- GetExclusiveAccess();
- if (d3dDevice == NULL) {
- ReleaseExclusiveAccess();
- return;
- }
+ RETURN_STATUS_IF_NULL(pd3dDevice, E_FAIL);
+
+ res = UpdateState(STATE_CHANGE);
// we can safely disable blending when:
// - comp is SrcNoEa or SrcOverNoEa, and
// - the source is opaque
- // (turning off blending can have a large positive impact on
- // performance);
+ // (turning off blending can have a large positive impact on performance)
if ((rule == RULE_Src || rule == RULE_SrcOver) &&
(ea == 1.0f) &&
(flags & D3DC_SRC_IS_OPAQUE))
- {
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- " disabling alpha comp rule=%-1d ea=1.0 src=opq)", rule);
- d3dDevice->SetRenderState(D3DRENDERSTATE_ALPHABLENDENABLE, FALSE);
- } else {
+ {
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ " disabling alpha comp rule=%-1d ea=1.0 src=opq)", rule);
+ pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE, FALSE);
+ } else {
J2dTraceLn2(J2D_TRACE_VERBOSE,
" enabling alpha comp (rule=%-1d ea=%f)", rule, ea);
- d3dDevice->SetRenderState(D3DRENDERSTATE_ALPHABLENDENABLE, TRUE);
+ pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE, TRUE);
- d3dDevice->SetRenderState(D3DRENDERSTATE_SRCBLEND,
- StdBlendRules[rule].src);
- d3dDevice->SetRenderState(D3DRENDERSTATE_DESTBLEND,
- StdBlendRules[rule].dst);
- }
+ pd3dDevice->SetRenderState(D3DRS_SRCBLEND,
+ StdBlendRules[rule].src);
+ pd3dDevice->SetRenderState(D3DRS_DESTBLEND,
+ StdBlendRules[rule].dst);
+ }
- // update state
- compState = sun_java2d_SunGraphics2D_COMP_ALPHA;
extraAlpha = ea;
-
- if (extraAlpha == 1.0f) {
- blitPolygonPixel = 0xffffffff;
- } else {
- // the 0xffffffff pixel needs to be premultiplied by extraAlpha
- jint ea = (jint)(extraAlpha * 255.0f + 0.5f) & 0xff;
- blitPolygonPixel = (ea << 24) | (ea << 16) | (ea << 8) | (ea << 0);
- }
-
- ReleaseExclusiveAccess();
-}
-
-HRESULT D3DContext::CreateSurface(JNIEnv *env, jint width, jint height,
- jint depth, jint transparency,
- jint d3dSurfaceType,
- DXSurface **dxSurface, jint* pType)
-{
- DWORD dwFlags = 0, ddsCaps = 0, ddsCaps2 = 0;
- D3DTextureTableCell *cell = NULL;
- DXSurface *lpRetSurface = NULL;
- HRESULT res;
-
- GetExclusiveAccess();
-
- dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH;
-
- if (d3dSurfaceType & D3D_TEXTURE_SURFACE) {
- ddsCaps |= DDSCAPS_TEXTURE;
- dwFlags |= DDSD_PIXELFORMAT | DDSD_TEXTURESTAGE;
-
- jint trIdx = D3D_TR_IDX(transparency);
- jint depthIdx = D3D_DEPTH_IDX(depth);
- cell = &textureTable[trIdx][depthIdx];
- if (cell->pfType == PF_INVALID) {
- ReleaseExclusiveAccess();
- J2dTraceLn2(J2D_TRACE_ERROR,
- "D3DContext::CreateSurface: no texture "\
- "pixel format for depth: %d transparency=%d",
- depth, transparency);
- return DDERR_NOTFOUND;
- }
- if (pType != NULL) *pType = cell->pfType;
- if (d3dSurfaceType & D3D_RENDER_TARGET) {
- // RTT is requested => must be allocated non-managed and
- // non-systemmemory pool.
- // REMIND: must check if this is supported by
- // the device, as it may not have a local video memory, only AGP
- // may be we should just use VIDEOMEMORY
- // NOTE: this will likely fail if the device is not accelerated
- ddsCaps |= DDSCAPS_LOCALVIDMEM;
- } else {
- // This is a normal texture, allocate in managed pool if the device
- // is accelerated, otherwise must use system memory.
- if (bIsHWRasterizer) {
- ddsCaps2 |= DDSCAPS2_TEXTUREMANAGE;
- } else {
- ddsCaps |= DDSCAPS_SYSTEMMEMORY;
- }
- }
-
- if (IsPow2TexturesOnly()) {
- jint w, h;
- for (w = 1; width > w; w <<= 1);
- for (h = 1; height > h; h <<= 1);
- width = w;
- height = h;
- }
- if (IsSquareTexturesOnly()) {
- if (width > height) {
- height = width;
- } else {
- width = height;
- }
- }
-
- DWORD dwRatio = GetMaxTextureAspectRatio();
- // Note: Reference rasterizer returns ratio '0',
- // which presumably means 'any'.
- if ((DWORD)width > GetMaxTextureWidth() ||
- (DWORD)height > GetMaxTextureHeight() ||
- (DWORD)width < GetMinTextureWidth() ||
- (DWORD)height < GetMinTextureHeight() ||
- ((dwRatio > 0) && ((DWORD)(width/height) > dwRatio ||
- (DWORD)(height/width) > dwRatio)))
- {
- ReleaseExclusiveAccess();
- J2dRlsTraceLn2(J2D_TRACE_ERROR,
- "D3DContext::CreateSurface: failed to create"\
- " texture: dimensions %dx%d not supported.",
- width, height);
- J2dRlsTraceLn5(J2D_TRACE_ERROR,
- " Supported texture dimensions: %dx%d-%dxd% "\
- " with max ratio %f.",
- GetMinTextureWidth(), GetMinTextureHeight(),
- GetMaxTextureWidth(), GetMaxTextureHeight(),
- GetMaxTextureAspectRatio());
- return D3DERR_TEXTURE_BADSIZE;
- }
- } else if (d3dSurfaceType & D3D_PLAIN_SURFACE) {
- ddsCaps |= DDSCAPS_OFFSCREENPLAIN |
- (bIsHWRasterizer ? DDSCAPS_VIDEOMEMORY : DDSCAPS_SYSTEMMEMORY);
- } else if (d3dSurfaceType & D3D_ATTACHED_SURFACE) {
- // can't handle this for now
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3DContext::CreateSurface: Can't create attached"\
- " surfaces using this code path yet");
- ReleaseExclusiveAccess();
- return DDERR_GENERIC;
- }
- if (d3dSurfaceType & D3D_RENDER_TARGET) {
- ddsCaps |= DDSCAPS_3DDEVICE;
- }
-
- if (SUCCEEDED(res = dxObject->CreateSurface(dwFlags, ddsCaps, ddsCaps2,
- (cell != NULL) ?
- &cell->pddpf : NULL,
- width, height,
- &lpRetSurface,
- 0/*backbuffers*/)))
- {
- if (d3dSurfaceType & D3D_RENDER_TARGET) {
- if (FAILED(res = AttachDepthBuffer(lpRetSurface))) {
- lpRetSurface->Release();
- delete lpRetSurface;
- ReleaseExclusiveAccess();
- return res;
- }
- // Attempt to set the new surface as a temporary render target;
- // in some cases this may fail. For example, if undocumented maximum
- // Direct3D target surface dimensions were exceeded (2048 in some
- // cases).
- if (d3dDevice != NULL) {
- FlushD3DQueueForTarget(NULL);
-
- IDirectDrawSurface7 *lpDDSurface = NULL;
- HRESULT res1 = d3dDevice->GetRenderTarget(&lpDDSurface);
-
- // we are holding a lock for the context, so we can
- // change/restore the current render target safely
- res = d3dDevice->SetRenderTarget(lpRetSurface->GetDDSurface(), 0);
- if (SUCCEEDED(res1) && lpDDSurface != NULL) {
- d3dDevice->SetRenderTarget(lpDDSurface, 0);
- }
- if (FAILED(res)) {
- DebugPrintDirectDrawError(res,
- "D3DContext::CreateSurface: cannot set new surface as "\
- "temp. render target");
- lpRetSurface->Release();
- delete lpRetSurface;
- ReleaseExclusiveAccess();
- return res;
- }
- }
- }
- *dxSurface = lpRetSurface;
- } else {
- DebugPrintDirectDrawError(res,
- "D3DContext::CreateSurface: error"\
- " creating surface");
- }
-
- ReleaseExclusiveAccess();
return res;
}
+#ifdef UPDATE_TX
+
+// Note: this method of adjusting pixel to texel mapping proved to be
+// difficult to perfect. The current variation works great for id,
+// scale (including all kinds of flips) transforms, but not still not
+// for generic transforms.
+//
+// Since we currently only do DrawTexture with non-id transform we instead
+// adjust the geometry (see D3DVertexCacher::DrawTexture(), SetTransform())
+//
+// In order to enable this code path UpdateTextureTransforms needs to
+// be called in SetTexture(), SetTransform() and ResetTranform().
HRESULT
-D3DContext::AttachDepthBuffer(DXSurface *dxSurface)
+D3DContext::UpdateTextureTransforms(DWORD dwSamplerToUpdate)
{
- HRESULT res;
+ HRESULT res = S_OK;
+ DWORD dwSampler, dwMaxSampler;
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::AttachDepthBuffer");
-
- if (dxSurface == NULL) {
- return DDERR_GENERIC;
+ if (dwSamplerToUpdate == -1) {
+ // update all used samplers, dwMaxSampler will be set to max
+ dwSampler = 0;
+ dwSampler = MAX_USED_TEXTURE_SAMPLER;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::UpdateTextureTransforms: "\
+ "updating all samplers");
+ } else {
+ // update only given sampler, dwMaxSampler will be set to it as well
+ dwSampler = dwSamplerToUpdate;
+ dwMaxSampler = dwSamplerToUpdate;
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DContext::UpdateTextureTransforms: "\
+ "updating sampler %d", dwSampler);
}
- GetExclusiveAccess();
+ do {
+ D3DTRANSFORMSTATETYPE state =
+ (D3DTRANSFORMSTATETYPE)(D3DTS_TEXTURE0 + dwSampler);
+ IDirect3DTexture9 *pTexture = lastTexture[dwSampler];
- // initialize the depth buffer format it needed
- if (depthBufferFormat.dwSize == 0) {
- // Some hardware has a restriction that the target surface and the
- // attached depth buffer must have the same bit depth, so we should
- // attempt to find a depth pixel format with the same depth as
- // the target.
- DWORD prefDepth = dxSurface->ddsd.ddpfPixelFormat.dwRGBBitCount;
- if (FAILED(res = D3DUtils_FindDepthBufferFormat(d3dObject,
- prefDepth,
- &depthBufferFormat,
- pDeviceGUID)))
- {
- DebugPrintDirectDrawError(res,
- "D3DContext::AttachDepthBuffer: "\
- "can't find depth buffer format");
- ReleaseExclusiveAccess();
- return res;
+ if (pTexture != NULL) {
+ D3DMATRIX mt, tx;
+ D3DSURFACE_DESC texDesc;
+
+ pd3dDevice->GetTransform(D3DTS_WORLD, &tx);
+ J2dTraceLn4(10,
+ " %5f %5f %5f %5f", tx._11, tx._12, tx._13, tx._14);
+ J2dTraceLn4(10,
+ " %5f %5f %5f %5f", tx._21, tx._22, tx._23, tx._24);
+ J2dTraceLn4(10,
+ " %5f %5f %5f %5f", tx._31, tx._32, tx._33, tx._34);
+ J2dTraceLn4(10,
+ " %5f %5f %5f %5f", tx._41, tx._42, tx._43, tx._44);
+
+ // this formula works for scales and flips
+ if (tx._11 == 0.0f) {
+ tx._11 = tx._12;
+ }
+ if (tx._22 == 0.0f) {
+ tx._22 = tx._21;
+ }
+
+ pTexture->GetLevelDesc(0, &texDesc);
+
+ // shift by .5 texel, but take into account
+ // the scale factor of the device transform
+
+ // REMIND: this approach is not entirely correct,
+ // as it only takes into account the scale of the device
+ // transform.
+ mt._31 = (1.0f / (2.0f * texDesc.Width * tx._11));
+ mt._32 = (1.0f / (2.0f * texDesc.Height * tx._22));
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " offsets: tx=%f ty=%f",
+ mt._31, mt._32);
+
+ pd3dDevice->SetTextureStageState(dwSampler,
+ D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_COUNT2);
+ res = pd3dDevice->SetTransform(state, &mt);
+ } else {
+ res = pd3dDevice->SetTextureStageState(dwSampler,
+ D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_DISABLE);
}
- }
- if (FAILED(res = dxSurface->AttachDepthBuffer(dxObject,
- bIsHWRasterizer,
- &depthBufferFormat)))
- {
- DebugPrintDirectDrawError(res,
- "D3DContext::AttachDepthBuffer: "\
- "can't attach depth buffer or it is lost");
- }
+ dwSampler++;
+ } while (dwSampler <= dwMaxSampler);
- ReleaseExclusiveAccess();
return res;
}
+#endif // UPDATE_TX
/**
* We go into the pains of maintaining the list of set textures
@@ -1200,399 +1492,412 @@ D3DContext::AttachDepthBuffer(DXSurface *dxSurface)
* GetTexture() (note that we'd have to then call Release() on the
* texture since GetTexture() increases texture's ref. count).
*/
-HRESULT /*NOLOCK*/
-D3DContext::SetTexture(DXSurface *dxSurface, DWORD dwStage)
+HRESULT
+D3DContext::SetTexture(IDirect3DTexture9 *pTexture, DWORD dwSampler)
{
- HRESULT res = D3D_OK;
- IDirectDrawSurface7 *newTexture =
- dxSurface == NULL ? NULL : dxSurface->GetDDSurface();
+ HRESULT res = S_OK;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::SetTexture");
- if (dwStage < 0 || dwStage > MAX_USED_TEXTURE_STAGE) {
+ if (dwSampler < 0 || dwSampler > MAX_USED_TEXTURE_SAMPLER) {
J2dTraceLn1(J2D_TRACE_ERROR,
- "D3DContext::SetTexture: incorrect stage: %d", dwStage);
- return DDERR_GENERIC;
+ "D3DContext::SetTexture: incorrect sampler: %d", dwSampler);
+ return E_FAIL;
}
- if (lastTexture[dwStage] != newTexture) {
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "D3DContext::SetTexture: new texture=0x%x", newTexture);
- res = d3dDevice->SetTexture(dwStage, newTexture);
- lastTexture[dwStage] = SUCCEEDED(res) ? newTexture : NULL;
+ if (lastTexture[dwSampler] != pTexture) {
+ if (FAILED(res = FlushVertexQueue())) {
+ return res;
+ }
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ " new texture=0x%x on sampler %d", pTexture, dwSampler);
+ res = pd3dDevice->SetTexture(dwSampler, pTexture);
+ if (SUCCEEDED(res)) {
+ lastTexture[dwSampler] = pTexture;
+ // REMIND: see comment at UpdateTextureTransforms
+#ifdef UPDATE_TX
+ res = UpdateTextureTransforms(dwSampler);
+#endif
+ } else {
+ lastTexture[dwSampler] = NULL;
+ }
}
return res;
}
-void
-D3DContext::FlushD3DQueueForTarget(DDrawSurface *ddSurface)
+HRESULT
+D3DContext::UpdateTextureColorState(DWORD dwState, DWORD dwSampler)
{
- GetExclusiveAccess();
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- "D3DContext::FlushD3DQueueForTarget surface=0x%x target=0x%x",
- ddSurface, ddTargetSurface);
+ HRESULT res = S_OK;
- if ((ddSurface == ddTargetSurface || ddSurface == NULL) &&
- d3dDevice != NULL)
- {
- ForceEndScene();
+ if (dwState != lastTextureColorState[dwSampler]) {
+ res = pd3dDevice->SetTextureStageState(dwSampler,
+ D3DTSS_ALPHAARG1, dwState);
+ res = pd3dDevice->SetTextureStageState(dwSampler,
+ D3DTSS_COLORARG1, dwState);
+ lastTextureColorState[dwSampler] = dwState;
}
- ReleaseExclusiveAccess();
+
+ return res;
}
-void
-D3DContext::InvalidateIfTarget(JNIEnv *env, DDrawSurface *ddSurface)
-{
- GetExclusiveAccess();
- if ((ddSurface == ddTargetSurface) && d3dDevice != NULL &&
- jD3DContext != NULL)
- {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "D3DContext:InvalidateIfTarget: invalidating java context");
-
- jobject jD3DContext_tmp = env->NewLocalRef(jD3DContext);
- if (jD3DContext_tmp != NULL) {
- JNU_CallMethodByName(env, NULL, jD3DContext_tmp,
- "invalidateContext", "()V");
- env->DeleteLocalRef(jD3DContext_tmp);
- }
- }
- ReleaseExclusiveAccess();
-}
-
-void /*NOLOCK*/
+HRESULT /*NOLOCK*/
D3DContext::UpdateState(jbyte newState)
{
- // Try to minimize context switching by only changing
- // attributes when necessary.
- if (newState != opState) {
- // if the new context is texture rendering
- if (newState & STATE_TEXTURE) {
- // we can be here because of two reasons:
- // old context wasn't STATE_TEXTURE or
- // the new STATE_TEXTURE_STAGE is different
+ HRESULT res = S_OK;
- // do the appropriate texture stage setup if needed
- DWORD dwAA1, dwCA1;
- BOOL bUpdateStateNeeded = FALSE;
- if ((newState & STATE_TEXTURE_STAGE_MASK) &&
- !(opState & STATE_TEXTURE_STAGE_MASK))
- {
- // setup mask rendering
- dwAA1 = (D3DTA_TEXTURE|D3DTA_ALPHAREPLICATE);
- dwCA1 = (D3DTA_TEXTURE|D3DTA_ALPHAREPLICATE);
- bUpdateStateNeeded = TRUE;
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "UpdateState: STATE_TEXTURE_STAGE_MASK");
- } else if ((newState & STATE_TEXTURE_STAGE_BLIT) &&
- !(opState & STATE_TEXTURE_STAGE_BLIT))
- {
- // setup blit rendering
- dwAA1 = D3DTA_TEXTURE;
- dwCA1 = D3DTA_TEXTURE;
- bUpdateStateNeeded = TRUE;
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "UpdateState: STATE_TEXTURE_STAGE_BLIT");
- }
-
- // this optimization makes sense because if the state
- // is changing from non-texture to texture, we don't necessarily
- // need to update the texture stage state
- if (bUpdateStateNeeded) {
- d3dDevice->SetTextureStageState(0, D3DTSS_ALPHAARG1, dwAA1);
- d3dDevice->SetTextureStageState(0, D3DTSS_COLORARG1, dwCA1);
- } else {
- J2dTraceLn2(J2D_TRACE_WARNING,
- "UpdateState: no context changes were made! "\
- "current=0x%x new=0x%x", opState, newState);
- }
- } else {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "UpdateState: STATE_RENDEROP");
- // if switching from a texture rendering state
- if (opState & STATE_TEXTURE) {
- // disable texture rendering
- // we don't need to change texture stage states
- // because they're irrelevant if the texture
- // is not set
- // REMIND: another possible optimiziation: instead of
- // setting texture to NULL, change the texture stage state
- SetTexture(NULL);
- }
- }
- opState = newState;
+ if (opState == newState) {
+ // The op is the same as last time, so we can return immediately.
+ return res;
+ } else if (opState != STATE_CHANGE) {
+ res = FlushVertexQueue();
}
+
+ switch (opState) {
+ case STATE_MASKOP:
+ pMaskCache->Disable();
+ break;
+ case STATE_GLYPHOP:
+ D3DTR_DisableGlyphVertexCache(this);
+ break;
+ case STATE_TEXTUREOP:
+ // optimization: certain state changes (those marked STATE_CHANGE)
+ // are allowed while texturing is enabled.
+ // In this case, we can allow previousOp to remain as it is and
+ // then return early.
+ if (newState == STATE_CHANGE) {
+ return res;
+ }
+ // REMIND: not necessary if we are switching to MASKOP or GLYPHOP
+ // (or a complex paint, for that matter), but would that be a
+ // worthwhile optimization?
+ SetTexture(NULL);
+ break;
+ case STATE_AAPGRAMOP:
+ res = DisableAAParallelogramProgram();
+ break;
+ default:
+ break;
+ }
+
+ switch (newState) {
+ case STATE_MASKOP:
+ pMaskCache->Enable();
+ UpdateTextureColorState(D3DTA_TEXTURE | D3DTA_ALPHAREPLICATE);
+ break;
+ case STATE_GLYPHOP:
+ D3DTR_EnableGlyphVertexCache(this);
+ UpdateTextureColorState(D3DTA_TEXTURE | D3DTA_ALPHAREPLICATE);
+ break;
+ case STATE_TEXTUREOP:
+ UpdateTextureColorState(D3DTA_TEXTURE);
+ break;
+ case STATE_AAPGRAMOP:
+ res = EnableAAParallelogramProgram();
+ break;
+ default:
+ break;
+ }
+
+ opState = newState;
+
+ return res;
+}
+
+HRESULT D3DContext::FlushVertexQueue()
+{
+ if (pVCacher != NULL) {
+ return pVCacher->Render();
+ }
+ return E_FAIL;
}
HRESULT D3DContext::BeginScene(jbyte newState)
{
- if (!d3dDevice) {
- return DDERR_GENERIC;
+ if (!pd3dDevice) {
+ return E_FAIL;
} else {
UpdateState(newState);
if (!bBeginScenePending) {
bBeginScenePending = TRUE;
-#ifdef DEBUG
- endSceneQueueDepth = 0;
-#endif /* DEBUG */
- HRESULT res = d3dDevice->BeginScene();
+ HRESULT res = pd3dDevice->BeginScene();
J2dTraceLn(J2D_TRACE_INFO, "D3DContext::BeginScene");
if (FAILED(res)) {
// this will cause context reinitialization
- opState = STATE_UNDEFINED;
+ opState = STATE_CHANGE;
}
return res;
}
- return D3D_OK;
+ return S_OK;
}
}
-HRESULT D3DContext::EndScene(HRESULT ddResult) {
- if (FAILED(ddResult)) {
- return ForceEndScene();
- }
-#ifdef DEBUG
- endSceneQueueDepth++;
-#endif /* DEBUG */
- return D3D_OK;
-}
-
-HRESULT D3DContext::ForceEndScene() {
+HRESULT D3DContext::EndScene() {
if (bBeginScenePending) {
+ FlushVertexQueue();
bBeginScenePending = FALSE;
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext::ForceEndScene");
-#ifdef DEBUG
- J2dTraceLn1(J2D_TRACE_VERBOSE, " queue depth=%d",
- endSceneQueueDepth);
- endSceneQueueDepth = 0;
-#endif /* DEBUG */
- return d3dDevice->EndScene();
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::EndScene");
+ return pd3dDevice->EndScene();
}
- return D3D_OK;
+ return S_OK;
}
/**
- * Utility function: checks the result, calls RestoreSurface
- * on the destination surface, and throws InvalidPipeException.
+ * Compiles and links the given fragment shader program. If
+ * successful, this function returns a handle to the newly created shader
+ * program; otherwise returns 0.
*/
-static void
-D3DContext_CheckResult(JNIEnv *env, HRESULT res, jlong pDest) {
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_CheckResult");
- if (FAILED(res)) {
- J2dTraceLn(J2D_TRACE_ERROR,
- "D3DContext_CheckResult: failed, restoring dest surface");
- Win32SDOps *dstOps = (Win32SDOps *)jlong_to_ptr(pDest);
- if (dstOps != NULL) {
- // RestoreSurface for surfaces associated
- // with VolatileImages only marks them lost, not
- // attempting to restore. This is done later
- // when VolatileImage.validate() is called.
- dstOps->RestoreSurface(env, dstOps);
+IDirect3DPixelShader9 *D3DContext::CreateFragmentProgram(DWORD **shaders,
+ ShaderList *programs,
+ jint flags)
+{
+ DWORD *sourceCode;
+ IDirect3DPixelShader9 *pProgram;
- // if this is an "unexpected" error, disable acceleration
- // of this image to avoid an infinite recreate/render/error loop
- if (res != DDERR_SURFACELOST && res != DDERR_INVALIDMODE &&
- res != DDERR_GENERIC && res != DDERR_WASSTILLDRAWING &&
- res != DDERR_SURFACEBUSY)
- {
- jobject sdObject = env->NewLocalRef(dstOps->sdOps.sdObject);
- if (sdObject != NULL) {
- JNU_CallMethodByName(env, NULL, sdObject,
- "disableD3D", "()V");
- env->DeleteLocalRef(sdObject);
- }
- }
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DContext::CreateFragmentProgram: flags=%d",
+ flags);
+ sourceCode = shaders[flags];
+ if (FAILED(pd3dDevice->CreatePixelShader(sourceCode, &pProgram))) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DContext::CreateFragmentProgram: error creating program");
+ return NULL;
+ }
+
+ // add it to the cache
+ ShaderList_AddProgram(programs, ptr_to_jlong(pProgram),
+ 0 /*unused*/, 0 /*unused*/, flags);
+
+ return pProgram;
+}
+
+/**
+ * Locates and enables a fragment program given a list of shader programs
+ * (ShaderInfos), using this context's state and flags as search
+ * parameters. The "flags" parameter is a bitwise-or'd value that helps
+ * differentiate one program for another; the interpretation of this value
+ * varies depending on the type of shader (BufImgOp, Paint, etc) but here
+ * it is only used to find another ShaderInfo with that same "flags" value.
+ */
+HRESULT D3DContext::EnableFragmentProgram(DWORD **shaders,
+ ShaderList *programList,
+ jint flags)
+{
+ HRESULT res;
+ jlong programID;
+ IDirect3DPixelShader9 *pProgram;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::EnableFragmentProgram");
+
+ programID =
+ ShaderList_FindProgram(programList,
+ 0 /*unused*/, 0 /*unused*/, flags);
+
+ pProgram = (IDirect3DPixelShader9 *)jlong_to_ptr(programID);
+ if (pProgram == NULL) {
+ pProgram = CreateFragmentProgram(shaders, programList, flags);
+ if (pProgram == NULL) {
+ return E_FAIL;
}
- SurfaceData_ThrowInvalidPipeException(env, "Surface Lost");
}
+
+ if (FAILED(res = pd3dDevice->SetPixelShader(pProgram))) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DContext::EnableFragmentProgram: error setting pixel shader");
+ return res;
+ }
+
+ return S_OK;
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: setTransform
- * Signature: (JLLjava/awt/geom/AffineTransform;DDDDDD)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DContext_setTransform
- (JNIEnv *env, jobject d3dc, jlong pCtx, jlong pDest, jobject xform,
- jdouble m00, jdouble m10,
- jdouble m01, jdouble m11,
- jdouble m02, jdouble m12)
+HRESULT D3DContext::EnableBasicGradientProgram(jint flags)
{
- D3DContext *pd3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_setTransform");
- if (pd3dc != NULL) {
- HRESULT res = pd3dc->SetTransform(xform,
- m00, m10,
- m01, m11,
- m02, m12);
- D3DContext_CheckResult(env, res, pDest);
- }
+ return EnableFragmentProgram((DWORD **)gradShaders,
+ &basicGradPrograms, flags);
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: resetTransform
- * Signature: (JLL)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DContext_resetTransform
- (JNIEnv *env, jobject d3dc, jlong pCtx, jlong pDest)
+HRESULT D3DContext::EnableLinearGradientProgram(jint flags)
{
- D3DContext *pd3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_resetTransform");
- if (pd3dc != NULL) {
- HRESULT res = pd3dc->SetTransform(NULL, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0);
- D3DContext_CheckResult(env, res, pDest);
- }
+ return EnableFragmentProgram((DWORD **)linearShaders,
+ &linearGradPrograms, flags);
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: setClip
- * Signature: (JLLsun/java2d/pipe/Region;ZIIII)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DContext_setClip
- (JNIEnv *env, jobject d3dc, jlong pCtx, jlong pDest,
- jobject clip, jboolean isRect,
- jint x1, jint y1, jint x2, jint y2)
+HRESULT D3DContext::EnableRadialGradientProgram(jint flags)
{
- D3DContext *pd3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_setClip");
- if (pd3dc != NULL) {
- HRESULT res = pd3dc->SetClip(env, clip, isRect, x1, y1, x2, y2);
- D3DContext_CheckResult(env, res, pDest);
- }
+ return EnableFragmentProgram((DWORD **)radialShaders,
+ &radialGradPrograms, flags);
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: resetClip
- * Signature: (JLLsun/java2d/pipe/Region;Z)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DContext_resetClip
- (JNIEnv *env, jobject d3dc, jlong pCtx, jlong pDest)
+HRESULT D3DContext::EnableConvolveProgram(jint flags)
{
- D3DContext *pd3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_resetClip");
- if (pd3dc != NULL) {
- HRESULT res = pd3dc->SetClip(env, NULL, JNI_FALSE, 0, 0, 0, 0);
- D3DContext_CheckResult(env, res, pDest);
- }
+ return EnableFragmentProgram((DWORD **)convolveShaders,
+ &convolvePrograms, flags);
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: setRenderTarget
- * Signature: (JJ)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DContext_setRenderTarget
- (JNIEnv *env, jobject d3dc, jlong pCtx, jlong pDest)
+HRESULT D3DContext::EnableRescaleProgram(jint flags)
{
- D3DContext *pd3dc = (D3DContext *)jlong_to_ptr(pCtx);
- Win32SDOps *dstOps = (Win32SDOps *)jlong_to_ptr(pDest);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_setRenderTarget");
- if (pd3dc != NULL && dstOps != NULL) {
- HRESULT res = pd3dc->SetRenderTarget(dstOps->lpSurface);
- D3DContext_CheckResult(env, res, pDest);
- }
+ return EnableFragmentProgram((DWORD **)rescaleShaders,
+ &rescalePrograms, flags);
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: setColor
- * Signature: (JII)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DContext_setColor(JNIEnv *env, jobject oc,
- jlong pCtx,
- jint pixel, jint flags)
+HRESULT D3DContext::EnableLookupProgram(jint flags)
{
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_setColor");
- if (d3dc != NULL) {
- d3dc->SetColor(pixel, flags);
- }
+ return EnableFragmentProgram((DWORD **)lookupShaders,
+ &lookupPrograms, flags);
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: setAlphaComposite
- * Signature: (JIFI)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DContext_setAlphaComposite(JNIEnv *env, jobject oc,
- jlong pCtx,
- jint rule,
- jfloat extraAlpha,
- jint flags)
+HRESULT D3DContext::EnableLCDTextProgram()
{
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
+ HRESULT res;
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_setAlphaComposite");
- if (d3dc != NULL) {
- d3dc->SetAlphaComposite(rule, extraAlpha, flags);
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::EnableLCDTextProgram");
+
+ if (lcdTextProgram == NULL) {
+ if (FAILED(res = pd3dDevice->CreatePixelShader(lcdtext0,
+ &lcdTextProgram)))
+ {
+ return res;
+ }
}
+ if (FAILED(res = pd3dDevice->SetPixelShader(lcdTextProgram))) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DContext::EnableLCDTextProgram: error setting pixel shader");
+ return res;
+ }
+
+ return S_OK;
}
-JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DContext_resetComposite(JNIEnv *env, jobject oc,
- jlong pCtx)
+HRESULT D3DContext::EnableAAParallelogramProgram()
{
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
+ HRESULT res;
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_resetComposite");
- if (d3dc != NULL) {
- d3dc->ResetComposite();
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::EnableAAParallelogramProgram");
+
+ if (aaPgramProgram == NULL) {
+ if (FAILED(res = pd3dDevice->CreatePixelShader(aapgram0,
+ &aaPgramProgram))) {
+ DebugPrintD3DError(res, "D3DContext::EnableAAParallelogramProgram: "
+ "error creating pixel shader");
+ return res;
+ }
}
+
+ if (FAILED(res = pd3dDevice->SetPixelShader(aaPgramProgram))) {
+ DebugPrintD3DError(res, "D3DContext::EnableAAParallelogramProgram: "
+ "error setting pixel shader");
+ return res;
+ }
+
+ return S_OK;
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: initNativeContext
- * Signature: (I)J
- */
-JNIEXPORT jlong JNICALL
-Java_sun_java2d_d3d_D3DContext_initNativeContext
- (JNIEnv *env, jobject d3dc, jint screen)
+HRESULT D3DContext::DisableAAParallelogramProgram()
{
- J2dTraceLn1(J2D_TRACE_INFO, "D3DContext_initNativeContext screen=%d",
- screen);
+ HRESULT res;
- HMONITOR hMon = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- D3DContext *d3dContext = NULL;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::DisableAAParallelogramProgram");
- if (tmpDdInstance != NULL && tmpDdInstance->ddObject != NULL) {
- AwtToolkit::GetInstance().SendMessage(WM_AWT_D3D_CREATE_DEVICE,
- (WPARAM)tmpDdInstance->ddObject,
- NULL);
- d3dContext = tmpDdInstance->ddObject->GetD3dContext();
+ if (aaPgramProgram != NULL) {
+ if (FAILED(res = pd3dDevice->SetPixelShader(NULL))) {
+ DebugPrintD3DError(res,
+ "D3DContext::DisableAAParallelogramProgram: "
+ "error clearing pixel shader");
+ return res;
+ }
}
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "D3DContext_initNativeContext created d3dContext=0x%x",
- d3dContext);
- return ptr_to_jlong(d3dContext);
+ return S_OK;
}
-/*
- * Class: sun_java2d_d3d_D3DContext
- * Method: getNativeDeviceCaps
- * Signature: (J)I
- */
-JNIEXPORT jint JNICALL Java_sun_java2d_d3d_D3DContext_getNativeDeviceCaps
- (JNIEnv *env, jobject d3dc, jlong pCtx)
+BOOL D3DContext::IsAlphaRTSurfaceSupported()
{
- D3DContext *d3dContext = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DContext_getNativeDeviceCaps");
- if (d3dContext != NULL) {
- d3dContext->SetJavaContext(env, d3dc);
- return (jint)d3dContext->GetDeviceCaps();
- }
- return J2D_D3D_FAILURE;
+ HRESULT res = pd3dObject->CheckDeviceFormat(adapterOrdinal,
+ devCaps.DeviceType,
+ curParams.BackBufferFormat,
+ D3DUSAGE_RENDERTARGET,
+ D3DRTYPE_SURFACE,
+ D3DFMT_A8R8G8B8);
+ return SUCCEEDED(res);
+}
+
+BOOL D3DContext::IsAlphaRTTSupported()
+{
+ HRESULT res = pd3dObject->CheckDeviceFormat(adapterOrdinal,
+ devCaps.DeviceType,
+ curParams.BackBufferFormat,
+ D3DUSAGE_RENDERTARGET,
+ D3DRTYPE_TEXTURE,
+ D3DFMT_A8R8G8B8);
+ return SUCCEEDED(res);
+}
+
+BOOL D3DContext::IsOpaqueRTTSupported()
+{
+ HRESULT res = pd3dObject->CheckDeviceFormat(adapterOrdinal,
+ devCaps.DeviceType,
+ curParams.BackBufferFormat,
+ D3DUSAGE_RENDERTARGET,
+ D3DRTYPE_TEXTURE,
+ curParams.BackBufferFormat);
+ return SUCCEEDED(res);
+}
+
+HRESULT D3DContext::InitContextCaps() {
+ J2dTraceLn(J2D_TRACE_INFO, "D3DContext::InitContextCaps");
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " caps for adapter %d :", adapterOrdinal);
+
+ if (pd3dDevice == NULL || pd3dObject == NULL) {
+ contextCaps = CAPS_EMPTY;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_EMPTY");
+ return E_FAIL;
+ }
+
+ contextCaps = CAPS_DEVICE_OK;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_DEVICE_OK");
+
+ if (IsAlphaRTSurfaceSupported()) {
+ contextCaps |= CAPS_RT_PLAIN_ALPHA;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_RT_PLAIN_ALPHA");
+ }
+ if (IsAlphaRTTSupported()) {
+ contextCaps |= CAPS_RT_TEXTURE_ALPHA;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_RT_TEXTURE_ALPHA");
+ }
+ if (IsOpaqueRTTSupported()) {
+ contextCaps |= CAPS_RT_TEXTURE_OPAQUE;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_RT_TEXTURE_OPAQUE");
+ }
+ if (IsPixelShader20Supported()) {
+ contextCaps |= CAPS_LCD_SHADER | CAPS_BIOP_SHADER | CAPS_PS20;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE,
+ " | CAPS_LCD_SHADER | CAPS_BIOP_SHADER | CAPS_PS20");
+ // Pre-PS3.0 video boards are very slow with the AA shader, so
+ // we will require PS30 hw even though the shader is compiled for 2.0a
+// if (IsGradientInstructionExtensionSupported()) {
+// contextCaps |= CAPS_AA_SHADER;
+// J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_AA_SHADER");
+// }
+ }
+ if (IsPixelShader30Supported()) {
+ if ((contextCaps & CAPS_AA_SHADER) == 0) {
+ // This flag was not already mentioned above...
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_AA_SHADER");
+ }
+ contextCaps |= CAPS_PS30 | CAPS_AA_SHADER;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_PS30");
+ }
+ if (IsMultiTexturingSupported()) {
+ contextCaps |= CAPS_MULTITEXTURE;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_MULTITEXTURE");
+ }
+ if (!IsPow2TexturesOnly()) {
+ contextCaps |= CAPS_TEXNONPOW2;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_TEXNONPOW2");
+ }
+ if (!IsSquareTexturesOnly()) {
+ contextCaps |= CAPS_TEXNONSQUARE;
+ J2dRlsTraceLn(J2D_TRACE_VERBOSE, " | CAPS_TEXNONSQUARE");
+ }
+ return S_OK;
}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DContext.h b/jdk/src/windows/native/sun/java2d/d3d/D3DContext.h
index c4313c101c2..2dba0918e8f 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DContext.h
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DContext.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -27,14 +27,33 @@
#define D3DCONTEXT_H
#include "java_awt_Transparency.h"
+#include "sun_java2d_pipe_BufferedContext.h"
#include "sun_java2d_d3d_D3DContext.h"
-#include "ddrawObject.h"
-extern "C" {
-#include "glyphblitting.h"
-#include "AccelGlyphCache.h"
-}
+#include "sun_java2d_d3d_D3DContext_D3DContextCaps.h"
+#include "sun_java2d_d3d_D3DSurfaceData.h"
+#include "sun_java2d_pipe_hw_AccelDeviceEventNotifier.h"
+
+#include "ShaderList.h"
+#include "D3DPipeline.h"
+#include "D3DMaskCache.h"
+#include "D3DVertexCacher.h"
+#include "D3DResourceManager.h"
+
#include "j2d_md.h"
+typedef enum {
+ TILEFMT_UNKNOWN,
+ TILEFMT_1BYTE_ALPHA,
+ TILEFMT_3BYTE_RGB,
+ TILEFMT_3BYTE_BGR,
+ TILEFMT_4BYTE_ARGB_PRE,
+} TileFormat;
+
+typedef enum {
+ CLIP_NONE,
+ CLIP_RECT,
+ CLIP_SHAPE,
+} ClipType;
// - State switching optimizations -----------------------------------
@@ -52,58 +71,28 @@ extern "C" {
* See D3DContext::UpdateState() and D3DContext::BeginScene() for
* more information.
*/
-
-// The state is undefined, assume that complete initialization is
-// needed.
-#define STATE_UNDEFINED (0 << 0)
-// Current state uses texture mapping
-#define STATE_TEXTURE (1 << 0)
-// Texture stage state which is used when mask is involved
-// (text rendering, maskfill)
-#define STATE_TEXTURE_STAGE_MASK (1 << 1)
-// Texture stage state which is used when doing texture
-// mapping in blits
-#define STATE_TEXTURE_STAGE_BLIT (1 << 2)
-// Texture stage state which is used when not doing
-// texture mapping, only use diffuse color
-#define STATE_TEXTURE_STAGE_POLY (1 << 3)
-// Texture mapping operation which involves mask texture
-#define STATE_MASKOP (STATE_TEXTURE|STATE_TEXTURE_STAGE_MASK)
-// Texture mapping operation which involves image texture
-#define STATE_BLITOP (STATE_TEXTURE|STATE_TEXTURE_STAGE_BLIT)
-// Rendering operation which doesn't use texture mapping
-#define STATE_RENDEROP (STATE_TEXTURE_STAGE_POLY)
+#define STATE_CHANGE (0 << 0)
+#define STATE_RENDEROP (1 << 0)
+#define STATE_MASKOP (1 << 1)
+#define STATE_GLYPHOP (1 << 2)
+#define STATE_TEXTUREOP (1 << 3)
+#define STATE_AAPGRAMOP (1 << 4)
+#define STATE_OTHEROP (1 << 5)
// The max. stage number we currently use (could not be
// larger than 7)
-#define MAX_USED_TEXTURE_STAGE 0
+#define MAX_USED_TEXTURE_SAMPLER 1
// - Texture pixel format table -------------------------------------
#define TR_OPAQUE java_awt_Transparency_OPAQUE
#define TR_BITMASK java_awt_Transparency_BITMASK
#define TR_TRANSLUCENT java_awt_Transparency_TRANSLUCENT
-// depth indices for the D3DTextureTable type
-#define DEPTH16_IDX 0
-#define DEPTH24_IDX 1
-#define DEPTH32_IDX 2
-#define DEPTH_MAX_IDX 3
-
-// corresponding transparency indices for the D3DTextureTable type
-#define TR_OPAQUE_IDX 0
-#define TR_BITMASK_IDX 1
-#define TR_TRANSLUCENT_IDX 2
-#define TR_MAX_IDX 3
-
-typedef struct
-{
- DDPIXELFORMAT pddpf;
- jint pfType;
-} D3DTextureTableCell;
-
-// texture table:
-// [transparency={OPAQUE,BITMASK,TRANCLUCENT},depth={16,24,32}]
-typedef D3DTextureTableCell D3DTextureTable[TR_MAX_IDX][DEPTH_MAX_IDX];
+class D3DResource;
+class D3DResourceManager;
+class D3DMaskCache;
+class D3DVertexCacher;
+class D3DGlyphCache;
// - D3DContext class -----------------------------------------------
@@ -111,7 +100,7 @@ typedef D3DTextureTableCell D3DTextureTable[TR_MAX_IDX][DEPTH_MAX_IDX];
* This class provides the following functionality:
* - holds the state of D3DContext java class (current pixel color,
* alpha compositing mode, extra alpha)
- * - provides access to IDirect3DDevice7 interface (creation,
+ * - provides access to IDirect3DDevice9 interface (creation,
* disposal, exclusive access)
* - handles state changes of the direct3d device (transform,
* compositing mode, current texture)
@@ -119,14 +108,8 @@ typedef D3DTextureTableCell D3DTextureTable[TR_MAX_IDX][DEPTH_MAX_IDX];
* - holds a glyph cache texture for the associated device
* - implements primitives batching mechanism
*/
-class D3DContext {
+class D3DPIPELINE_API D3DContext {
public:
- /**
- * Creates and returns D3DContext instance. If created context was
- * unable to initialize d3d device or if the device tests failed,
- * returns NULL.
- */
- static D3DContext* CreateD3DContext(DDraw *ddObject, DXObject* dxObject);
/**
* Releases the old device (if there was one) and all associated
* resources, re-creates, initializes and tests the new device.
@@ -141,140 +124,153 @@ public:
* to initialize and test the device last time, it doesn't attempt
* to create/init/test the device.
*/
- void CreateD3DDevice();
- void Release3DDevice();
+ static
+ HRESULT CreateInstance(IDirect3D9 *pd3d9, UINT adapter, D3DContext **ppCtx);
+ // creates a new D3D windowed device with swap copy effect and default
+ // present interval
+ HRESULT InitContext();
+ // creates or resets a D3D device given the parameters
+ HRESULT ConfigureContext(D3DPRESENT_PARAMETERS *pNewParams);
+ // resets existing D3D device with the current presentation parameters
+ HRESULT ResetContext();
+ HRESULT CheckAndResetDevice();
+
+ // saves the state of the D3D device in a state block, resets
+ // context's state to STATE_CHANGE
+ HRESULT SaveState();
+ // restores the state of the D3D device from existing state block,
+ // resets context's state to STATE_CHANGE
+ HRESULT RestoreState();
+
+ void ReleaseContextResources();
+ void ReleaseDefPoolResources();
virtual ~D3DContext();
- /**
- * Stores a weak reference of passed D3DContext object.
- * This method is called from _getNativeDeviceCaps method, and does the
- * association of the native D3DContext with the corresponding java object.
- * We need a reference to the java object so it can be notified when
- * the native context is released or recreated.
- *
- * See jobject jD3DContext field
- */
- void SetJavaContext(JNIEnv *env, jobject jd3dc);
-
- /**
- * Methods to get/release exclusive access to the direct3d device
- * interface. Note that some methods of this class assume that the
- * lock is already taken. They're marked with 'NOLOCK' comment.
- * Those methods not dealing with the d3d device interface are not
- * required to obtain the lock (and not marked with NOLOCK)
- */
- void GetExclusiveAccess() { CRITICAL_SECTION_ENTER(deviceLock);}
- void ReleaseExclusiveAccess() { CRITICAL_SECTION_LEAVE(deviceLock);}
-
// methods replicating java-level D3DContext objext
- void SetColor(jint eargb, jint flags);
- void SetAlphaComposite(jint rule, jfloat extraAlpha, jint flags);
- void ResetComposite();
-
- // Glyph cache-related methods
- HRESULT /*NOLOCK*/ InitGlyphCache();
- HRESULT /*NOLOCK*/ GlyphCacheAdd(JNIEnv *env, GlyphInfo *glyph);
- HRESULT /*NOLOCK*/ UploadImageToTexture(DXSurface *texture, jubyte *pixels,
- jint dstx, jint dsty,
- jint srcx, jint srcy,
- jint srcWidth, jint srcHeight,
- jint srcStride);
- DXSurface /*NOLOCK*/ *GetGlyphCacheTexture() { return lpGlyphCacheTexture; }
- DXSurface /*NOLOCK*/ *GetMaskTexture();
- GlyphCacheInfo *GetGlyphCache() { return glyphCache; }
-
- HRESULT CreateSurface(JNIEnv *env,
- jint width, jint height, jint depth,
- jint transparency, jint d3dSurfaceType,
- DXSurface** dxSurface, jint* pType);
+ HRESULT SetAlphaComposite(jint rule, jfloat extraAlpha, jint flags);
+ HRESULT ResetComposite();
/**
- * Attaches a depth buffer to the specified dxSurface.
- * If depthBufferFormat is not initialized (depthBufferFormat.dwSize == 0),
- * it will be initialized at the time of the call.
- *
- * If the buffer for this surface already exists, a "lost" status of the
- * depth buffer is returned.
+ * Glyph cache-related methods
*/
- HRESULT AttachDepthBuffer(DXSurface *dxSurface);
+ HRESULT InitGrayscaleGlyphCache();
+ HRESULT InitLCDGlyphCache();
+ D3DGlyphCache* GetGrayscaleGlyphCache() { return pGrayscaleGlyphCache; }
+ D3DGlyphCache* GetLCDGlyphCache() { return pLCDGlyphCache; }
- // methods for dealing with device capabilities as determined by
- // methods in D3DRuntimeTest
- int GetDeviceCaps() { return deviceCaps; }
- void SetDeviceCaps(int caps) { deviceCaps = caps; }
+ D3DResourceManager *GetResourceManager() { return pResourceMgr; }
+ D3DMaskCache *GetMaskCache() { return pMaskCache; }
- // Returns the texture pixel format table
- D3DTextureTable &GetTextureTable() { return textureTable; }
+ HRESULT UploadTileToTexture(D3DResource *pTextureRes, void *pixels,
+ jint dstx, jint dsty,
+ jint srcx, jint srcy,
+ jint srcWidth, jint srcHeight,
+ jint srcStride,
+ TileFormat srcFormat,
+ // out: num of pixels in first and last
+ // columns, only counted for LCD glyph uploads
+ jint *pPixelsTouchedL = NULL,
+ jint *pPixelsTouchedR = NULL);
- DDrawSurface *GetTargetSurface() { return ddTargetSurface; }
- IDirect3DDevice7 *Get3DDevice() { return d3dDevice; }
+ // returns capabilities of the Direct3D device
+ D3DCAPS9 *GetDeviceCaps() { return &devCaps; }
+ // returns caps in terms of the D3DContext
+ int GetContextCaps() { return contextCaps; }
+ D3DPRESENT_PARAMETERS *GetPresentationParams() { return &curParams; }
- // IDirect3DDevice7-delegation methods
+ IDirect3DDevice9 *Get3DDevice() { return pd3dDevice; }
+ IDirect3D9 *Get3DObject() { return pd3dObject; }
/**
* This method only sets the texture if it's not already set.
*/
- HRESULT /*NOLOCK*/ SetTexture(DXSurface *dxSurface, DWORD dwStage = 0);
- HRESULT SetRenderTarget(DDrawSurface *lpSurface);
- HRESULT SetTransform(jobject xform,
- jdouble m00, jdouble m10,
- jdouble m01, jdouble m11,
- jdouble m02, jdouble m12);
- HRESULT SetClip(JNIEnv *env, jobject clip,
- jboolean isRect,
- int x1, int y1, int x2, int y2);
-
- DWORD GetMinTextureWidth() { return d3dDevDesc.dwMinTextureWidth; }
- DWORD GetMinTextureHeight() { return d3dDevDesc.dwMinTextureHeight; }
- DWORD GetMaxTextureWidth() { return d3dDevDesc.dwMaxTextureWidth; }
- DWORD GetMaxTextureHeight() { return d3dDevDesc.dwMaxTextureHeight; }
- DWORD GetMaxTextureAspectRatio()
- { return d3dDevDesc.dwMaxTextureAspectRatio; };
- BOOL IsPow2TexturesOnly()
- { return d3dDevDesc.dpcTriCaps.dwTextureCaps & D3DPTEXTURECAPS_POW2; };
- BOOL IsSquareTexturesOnly()
- { return d3dDevDesc.dpcTriCaps.dwTextureCaps &
- D3DPTEXTURECAPS_SQUAREONLY; }
+ HRESULT SetTexture(IDirect3DTexture9 *pTexture, DWORD dwSampler = 0);
/**
- * This method invalidates the java-level D3DContext object if
- * the passed DDrawSurface is the current render target.
- * The invalidation needs to be done so that the D3DContext object
- * resets itself in case the native d3d device has been recreated, or
- * the target surface has been lost (in which case this method is called
- * from D3DSD_RestoreSurface function, see D3DSD_RestoreSurface for
- * more info).
+ * This method only updates the texture color state if it hasn't changed.
*/
- void InvalidateIfTarget(JNIEnv *env, DDrawSurface *lpSurface);
+ HRESULT UpdateTextureColorState(DWORD dwState, DWORD dwSampler = 0);
+
+ HRESULT SetRenderTarget(IDirect3DSurface9 *pSurface);
+ HRESULT SetTransform(jdouble m00, jdouble m10,
+ jdouble m01, jdouble m11,
+ jdouble m02, jdouble m12);
+ HRESULT ResetTransform();
+
+ // clipping-related methods
+ HRESULT SetRectClip(int x1, int y1, int x2, int y2);
+ HRESULT BeginShapeClip();
+ HRESULT EndShapeClip();
+ HRESULT ResetClip();
+ ClipType GetClipType();
+
+ /**
+ * Shader-related methods
+ */
+ HRESULT EnableBasicGradientProgram(jint flags);
+ HRESULT EnableLinearGradientProgram(jint flags);
+ HRESULT EnableRadialGradientProgram(jint flags);
+ HRESULT EnableConvolveProgram(jint flags);
+ HRESULT EnableRescaleProgram(jint flags);
+ HRESULT EnableLookupProgram(jint flags);
+ HRESULT EnableLCDTextProgram();
+ HRESULT EnableAAParallelogramProgram();
+ HRESULT DisableAAParallelogramProgram();
+
+ BOOL IsTextureFilteringSupported(D3DTEXTUREFILTERTYPE fType);
+ BOOL IsStretchRectFilteringSupported(D3DTEXTUREFILTERTYPE fType);
+ BOOL IsPow2TexturesOnly()
+ { return devCaps.TextureCaps & D3DPTEXTURECAPS_POW2; };
+ BOOL IsSquareTexturesOnly()
+ { return devCaps.TextureCaps & D3DPTEXTURECAPS_SQUAREONLY; }
+ BOOL IsHWRasterizer() { return bIsHWRasterizer; }
+ BOOL IsTextureFormatSupported(D3DFORMAT format, DWORD usage = 0);
+ BOOL IsDynamicTextureSupported()
+ { return devCaps.Caps2 & D3DCAPS2_DYNAMICTEXTURES; }
+// REMIND: for now for performance testing
+// { return (getenv("J2D_D3D_USE_DYNAMIC_TEX") != NULL); }
+ BOOL IsImmediateIntervalSupported()
+ { return devCaps.PresentationIntervals & D3DPRESENT_INTERVAL_IMMEDIATE;}
+ BOOL IsPixelShader20Supported()
+ { return (devCaps.PixelShaderVersion >= D3DPS_VERSION(2,0)); }
+ BOOL IsGradientInstructionExtensionSupported()
+ { return devCaps.PS20Caps.Caps & D3DPS20CAPS_GRADIENTINSTRUCTIONS; }
+ BOOL IsPixelShader30Supported()
+ { return (devCaps.PixelShaderVersion >= D3DPS_VERSION(3,0)); }
+ BOOL IsMultiTexturingSupported()
+ { return (devCaps.MaxSimultaneousTextures > 1); }
+ BOOL IsAlphaRTSurfaceSupported();
+ BOOL IsAlphaRTTSupported();
+ BOOL IsOpaqueRTTSupported();
+
+ jint GetPaintState() { return paintState; }
+ void SetPaintState(jint state) { this->paintState = state; }
+ BOOL IsIdentityTx() { return bIsIdentityTx; }
+
+ HRESULT FlushVertexQueue();
+ D3DVertexCacher *pVCacher;
+ HRESULT UpdateState(jbyte newState);
+
+ HRESULT Sync();
// primitives batching-related methods
/**
* Calls devices's BeginScene if there weren't one already pending,
* sets the pending flag.
*/
- HRESULT /*NOLOCK*/ BeginScene(jbyte newState);
+ HRESULT BeginScene(jbyte newState);
/**
- * Only calls device's EndScene if ddResult is an error.
+ * Flushes the vertex queue and does end scene if
+ * a BeginScene is pending
*/
- HRESULT /*NOLOCK*/ EndScene(HRESULT ddResult);
- /**
- * forces the end of batching by calling EndScene if
- * there was BeginScene pending.
- */
- HRESULT /*NOLOCK*/ ForceEndScene();
- /**
- * flushes the queue if the argument is this device's render target
- */
- void FlushD3DQueueForTarget(DDrawSurface *ddSurface);
+ HRESULT EndScene();
- // fields replicating D3DContext class' fields
- jint compState;
+ /**
+ * Fields that track native-specific state.
+ */
+ jint paintState;
+ jboolean useMask;
jfloat extraAlpha;
- jint colorPixel;
-
- // pixel for vertices used in blits via texture mapping,
- // set in SetAlphaComposite()
- jint blitPolygonPixel;
/**
* Current operation state.
@@ -283,164 +279,144 @@ public:
jbyte opState;
private:
- D3DContext(DDraw *ddObject, DXObject* dxObject);
- HRESULT InitD3DDevice(IDirect3DDevice7 *d3dDevice);
- /**
- * This method releases an old device, creates a new one,
- * runs d3d caps tests on it and sets the device caps according
- * to the results.
- */
- HRESULT /*NOLOCK*/ CreateAndTestD3DDevice(DxCapabilities *dxCaps);
- HRESULT /*NOLOCK*/ InitMaskTileTexture();
- void /*NOLOCK*/ UpdateState(jbyte newState);
- IDirect3DDevice7 *d3dDevice;
- IDirect3D7 *d3dObject;
- DDraw *ddObject;
- DXObject *dxObject;
- const GUID *pDeviceGUID;
- DDrawSurface *ddTargetSurface;
- DXSurface *lpMaskTexture;
- DXSurface *lpGlyphCacheTexture;
- D3DTextureTable textureTable;
- DDPIXELFORMAT depthBufferFormat;
- DDPIXELFORMAT maskTileTexFormat;
- GlyphCacheInfo *glyphCache;
- BOOL glyphCacheAvailable;
+ /**
+ * Glyph cache-related methods/fields...
+ */
+ D3DGlyphCache *pGrayscaleGlyphCache;
+ D3DGlyphCache *pLCDGlyphCache;
+
+ /**
+ * The handle to the LCD text pixel shader program.
+ */
+ IDirect3DPixelShader9 *lcdTextProgram;
+
+ /**
+ * The handle to the AA pixel and vertex shader programs.
+ */
+ IDirect3DPixelShader9 *aaPgramProgram;
+
+ IDirect3DPixelShader9 *CreateFragmentProgram(DWORD **shaders,
+ ShaderList *programs,
+ jint flags);
+ HRESULT EnableFragmentProgram(DWORD **shaders,
+ ShaderList *programList,
+ jint flags);
+
+ // finds appropriate to the target surface depth format,
+ // creates the depth buffer and installs it onto the device
+ HRESULT InitDepthStencilBuffer(D3DSURFACE_DESC *pTargetDesc);
+ // returns true if the current depth buffer is compatible
+ // with the new target, and the dimensions fit, false otherwise
+ BOOL IsDepthStencilBufferOk(D3DSURFACE_DESC *pTargetDesc);
+
+ D3DContext(IDirect3D9 *pd3dObject, UINT adapter);
+ HRESULT InitDevice(IDirect3DDevice9 *d3dDevice);
+ HRESULT InitContextCaps();
+ // updates the texture transform(s) used for better texel to pixel mapping
+ // for the passed in sampler;
+ // if -1 is passed as the sampler, texture transforms for
+ // samplers [0..MAX_USED_TEXTURE_SAMPLER] are updated
+ // REMIND: see the comment in the method implementation before enabling.
+#undef UPDATE_TX
+#ifdef UPDATE_TX
+ HRESULT UpdateTextureTransforms(DWORD dwSamplerToUpdate);
+#endif // UPDATE_TX
+ IDirect3DDevice9 *pd3dDevice;
+ IDirect3D9 *pd3dObject;
+
+ D3DResourceManager *pResourceMgr;
+ D3DMaskCache *pMaskCache;
+
+ ShaderList convolvePrograms;
+ ShaderList rescalePrograms;
+ ShaderList lookupPrograms;
+ ShaderList basicGradPrograms;
+ ShaderList linearGradPrograms;
+ ShaderList radialGradPrograms;
+
// array of the textures currently set to the device
- IDirectDrawSurface7 *lastTexture[MAX_USED_TEXTURE_STAGE+1];
+ IDirect3DTexture9 *lastTexture[MAX_USED_TEXTURE_SAMPLER+1];
- /**
- * A weak reference to the java-level D3DContext object.
- * Used to invalidate the java D3DContext object if the device has been
- * recreated.
- * See SetJavaContext() method.
- */
- jobject jD3DContext;
+ DWORD lastTextureColorState[MAX_USED_TEXTURE_SAMPLER+1];
- D3DDEVICEDESC7 d3dDevDesc;
- int deviceCaps;
+ UINT adapterOrdinal;
+ D3DPRESENT_PARAMETERS curParams;
+ D3DCAPS9 devCaps;
+ int contextCaps;
BOOL bIsHWRasterizer;
+ BOOL bIsIdentityTx;
+
+ IDirect3DQuery9* pSyncQuery;
+ D3DResource* pSyncRTRes;
+
+ IDirect3DStateBlock9* pStateBlock;
+
/**
* Used to implement simple primitive batching.
* See BeginScene/EndScene/ForceEndScene.
*/
BOOL bBeginScenePending;
-#ifdef DEBUG
- int endSceneQueueDepth;
-#endif /* DEBUG */
-
- CriticalSection deviceLock;
};
-// - Various vertex formats -------------------------------------------
-
-#define D3DFVF_J2DLVERTEX (D3DFVF_XYZ | D3DFVF_DIFFUSE | D3DFVF_TEX1)
-typedef struct _J2DLVERTEX {
- float x, y, z;
- DWORD color;
- float tu, tv;
-} J2DLVERTEX;
-
-/**
- * We're still debating whether to use a single vertex format
- * for all primitives or specific per-primitive formats.
- * Using different vertex formats reduces the amount of
- * data being sent to the video board, and this shows
- * benetits when running Java2D benchmarks.
- *
- * However, in a typical Swing application the number
- * of primitives of the same type rendered in a row is
- * relatively small, which means that the driver has
- * to spend more time state switching to account for different
- * vertex formats (and according to MSDN, switching vertex format
- * is a very expensive operation). So for this kind of application
- * it's better to stick with a single vertex format.
- */
-#define USE_SINGLE_VERTEX_FORMAT
-
-#ifndef USE_SINGLE_VERTEX_FORMAT
-
-#define D3DFVF_J2D_XY_C (D3DFVF_XYZ | D3DFVF_DIFFUSE)
-#define D3DFVF_XY_VERTEX D3DFVF_XYZ
-
-typedef struct _J2D_XY_C_VERTEX {
- float x, y, z;
- DWORD color;
-} J2D_XY_C_VERTEX;
-typedef struct _J2D_XY_VERTEX {
- float x, y, z;
-} J2D_XY_VERTEX;
-
-#else // USE_SINGLE_VERTEX_FORMAT
-
-// When using a single vertex format, define
-// every format as J2DLVERTEX
-
-#define D3DFVF_J2D_XY_C D3DFVF_J2DLVERTEX
-#define D3DFVF_XY_VERTEX D3DFVF_J2DLVERTEX
-typedef J2DLVERTEX J2D_XY_C_VERTEX;
-typedef J2DLVERTEX J2D_XY_VERTEX;
-
-#endif // USE_SINGLE_VERTEX_FORMAT
-
-typedef J2DLVERTEX J2DLV_QUAD[4];
-typedef J2DLVERTEX J2DLV_HEXA[6];
-typedef J2D_XY_C_VERTEX J2DXYC_HEXA[6];
-typedef J2D_XY_VERTEX J2DXY_HEXA[6];
-#define MAX_CACHED_SPAN_VX_NUM 100
-
// - Helper Macros ---------------------------------------------------
-#define D3D_DEPTH_IDX(DEPTH) \
- (((DEPTH) <= 16) ? DEPTH16_IDX : \
- (((DEPTH) <= 24) ? DEPTH24_IDX : DEPTH32_IDX))
+#define D3DC_INIT_SHADER_LIST(list, max) \
+ do { \
+ (list).head = NULL; \
+ (list).maxItems = (max); \
+ (list).dispose = D3DContext_DisposeShader; \
+ } while (0)
-#define D3D_TR_IDX(TRAN) ((TRAN) - 1)
-
-#define D3DSD_MASK_TILE_SIZE 32
-#define D3D_GCACHE_WIDTH 512
-#define D3D_GCACHE_HEIGHT 512
-#define D3D_GCACHE_CELL_WIDTH 16
-#define D3D_GCACHE_CELL_HEIGHT 16
+/**
+ * This constant determines the size of the shared tile texture used
+ * by a number of image rendering methods. For example, the blit tile texture
+ * will have dimensions with width D3DC_BLIT_TILE_SIZE and height
+ * D3DC_BLIT_TILE_SIZE (the tile will always be square).
+ */
+#define D3DC_BLIT_TILE_SIZE 256
+/**
+ * See BufferedContext.java for more on these flags...
+ */
#define D3DC_NO_CONTEXT_FLAGS \
- sun_java2d_d3d_D3DContext_NO_CONTEXT_FLAGS
+ sun_java2d_pipe_BufferedContext_NO_CONTEXT_FLAGS
#define D3DC_SRC_IS_OPAQUE \
- sun_java2d_d3d_D3DContext_SRC_IS_OPAQUE
+ sun_java2d_pipe_BufferedContext_SRC_IS_OPAQUE
+#define D3DC_USE_MASK \
+ sun_java2d_pipe_BufferedContext_USE_MASK
-#define J2D_D3D_FAILURE \
- sun_java2d_d3d_D3DContext_J2D_D3D_FAILURE
-#define J2D_D3D_PLAIN_SURFACE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_PLAIN_SURFACE_OK
-#define J2D_D3D_OP_TEXTURE_SURFACE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_OP_TEXTURE_SURFACE_OK
-#define J2D_D3D_BM_TEXTURE_SURFACE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_BM_TEXTURE_SURFACE_OK
-#define J2D_D3D_TR_TEXTURE_SURFACE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_TR_TEXTURE_SURFACE_OK
-#define J2D_D3D_DEPTH_SURFACE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_DEPTH_SURFACE_OK
-#define J2D_D3D_OP_RTT_SURFACE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_OP_RTT_SURFACE_OK
-#define J2D_D3D_LINES_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_LINES_OK
-#define J2D_D3D_TEXTURE_BLIT_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_TEXTURE_BLIT_OK
-#define J2D_D3D_TEXTURE_TRANSFORM_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_TEXTURE_TRANSFORM_OK
-#define J2D_D3D_LINE_CLIPPING_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_LINE_CLIPPING_OK
-#define J2D_D3D_DEVICE_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_DEVICE_OK
-#define J2D_D3D_PIXEL_FORMATS_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_PIXEL_FORMATS_OK
-#define J2D_D3D_SET_TRANSFORM_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_SET_TRANSFORM_OK
-#define J2D_D3D_HW_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_HW_OK
-#define J2D_D3D_ENABLED_OK \
- sun_java2d_d3d_D3DContext_J2D_D3D_ENABLED_OK
+#define CAPS_EMPTY \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_EMPTY
+#define CAPS_RT_PLAIN_ALPHA \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_RT_PLAIN_ALPHA
+#define CAPS_RT_TEXTURE_ALPHA \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_RT_TEXTURE_ALPHA
+#define CAPS_RT_TEXTURE_OPAQUE \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_RT_TEXTURE_OPAQUE
+#define CAPS_MULTITEXTURE \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_MULTITEXTURE
+#define CAPS_TEXNONPOW2 \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_TEXNONPOW2
+#define CAPS_TEXNONSQUARE \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_TEXNONSQUARE
+#define CAPS_LCD_SHADER \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_LCD_SHADER
+#define CAPS_BIOP_SHADER \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_BIOP_SHADER
+#define CAPS_AA_SHADER \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_AA_SHADER
+#define CAPS_DEVICE_OK \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_DEVICE_OK
+#define CAPS_PS20 \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_PS20
+#define CAPS_PS30 \
+ sun_java2d_d3d_D3DContext_D3DContextCaps_CAPS_PS30
-#endif D3DCONTEXT_H
+#define DEVICE_RESET \
+ sun_java2d_pipe_hw_AccelDeviceEventNotifier_DEVICE_RESET
+#define DEVICE_DISPOSED \
+ sun_java2d_pipe_hw_AccelDeviceEventNotifier_DEVICE_DISPOSED
+
+#endif // D3DCONTEXT_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DGlyphCache.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DGlyphCache.cpp
new file mode 100644
index 00000000000..09cb1f5953b
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DGlyphCache.cpp
@@ -0,0 +1,207 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "D3DGlyphCache.h"
+#include "D3DTextRenderer.h"
+#include "D3DRenderQueue.h"
+
+void D3DGlyphCache_FlushGlyphVertexCache();
+
+// static
+HRESULT
+D3DGlyphCache::CreateInstance(D3DContext *pCtx, GlyphCacheType gcType,
+ D3DGlyphCache **ppGlyphCache)
+{
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGlyphCache::CreateInstance");
+
+ *ppGlyphCache = new D3DGlyphCache(gcType);
+ if (FAILED(res = (*ppGlyphCache)->Init(pCtx))) {
+ delete *ppGlyphCache;
+ *ppGlyphCache = NULL;
+ }
+ return res;
+}
+
+D3DGlyphCache::D3DGlyphCache(GlyphCacheType type)
+{
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DGlyphCache::D3DGlyphCache gcType=%d", type);
+
+ pCtx = NULL;
+ gcType = type;
+ pGlyphCacheRes = NULL;
+ pGlyphCache = NULL;
+ tileFormat = (gcType == CACHE_GRAY) ? TILEFMT_1BYTE_ALPHA : TILEFMT_UNKNOWN;
+ lastRGBOrder = JNI_FALSE;
+}
+
+D3DGlyphCache::~D3DGlyphCache()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGlyphCache::~D3DGlyphCache");
+
+ ReleaseDefPoolResources();
+
+ pCtx = NULL;
+ if (pGlyphCache != NULL) {
+ AccelGlyphCache_Free(pGlyphCache);
+ pGlyphCache = NULL;
+ }
+}
+
+void
+D3DGlyphCache::ReleaseDefPoolResources()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGlyphCache::ReleaseDefPoolResources");
+
+ AccelGlyphCache_Invalidate(pGlyphCache);
+ // REMIND: the glyph cache texture is not in the default pool, so
+ // this can be optimized not to release the texture
+ pCtx->GetResourceManager()->ReleaseResource(pGlyphCacheRes);
+ pGlyphCacheRes = NULL;
+}
+
+HRESULT
+D3DGlyphCache::Init(D3DContext *pCtx)
+{
+ D3DFORMAT format;
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DGlyphCache::Init pCtx=%x", pCtx);
+
+ this->pCtx = pCtx;
+
+ if (pGlyphCache == NULL) {
+ // init glyph cache data structure
+ pGlyphCache = AccelGlyphCache_Init(D3DTR_CACHE_WIDTH,
+ D3DTR_CACHE_HEIGHT,
+ D3DTR_CACHE_CELL_WIDTH,
+ D3DTR_CACHE_CELL_HEIGHT,
+ D3DGlyphCache_FlushGlyphVertexCache);
+ if (pGlyphCache == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DGlyphCache::Init: "\
+ "could not init D3D glyph cache");
+ return E_FAIL;
+ }
+ }
+
+ if (gcType == CACHE_GRAY) {
+ format = pCtx->IsTextureFormatSupported(D3DFMT_A8) ?
+ D3DFMT_A8 : D3DFMT_A8R8G8B8;
+ } else { // gcType == CACHE_LCD
+ format = pCtx->IsTextureFormatSupported(D3DFMT_R8G8B8) ?
+ D3DFMT_R8G8B8 : D3DFMT_A8R8G8B8;
+ }
+
+ HRESULT res = pCtx->GetResourceManager()->
+ CreateTexture(D3DTR_CACHE_WIDTH, D3DTR_CACHE_HEIGHT,
+ FALSE/*isRTT*/, FALSE/*isOpaque*/, &format, 0/*usage*/,
+ &pGlyphCacheRes);
+ if (FAILED(res)) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DGlyphCache::Init: "\
+ "could not create glyph cache texture");
+ }
+
+ return res;
+}
+
+HRESULT
+D3DGlyphCache::AddGlyph(GlyphInfo *glyph)
+{
+ HRESULT res = S_OK;
+
+ RETURN_STATUS_IF_NULL(pGlyphCacheRes, E_FAIL);
+
+ CacheCellInfo *cellInfo = AccelGlyphCache_AddGlyph(pGlyphCache, glyph);
+ if (cellInfo != NULL) {
+ jint pixelsTouchedL = 0, pixelsTouchedR = 0;
+ // store glyph image in texture cell
+ res = pCtx->UploadTileToTexture(pGlyphCacheRes,
+ glyph->image,
+ cellInfo->x, cellInfo->y,
+ 0, 0,
+ glyph->width, glyph->height,
+ glyph->rowBytes, tileFormat,
+ &pixelsTouchedL,
+ &pixelsTouchedR);
+ // LCD text rendering optimization: if the number of pixels touched on
+ // the first or last column of the glyph image is less than 1/3 of the
+ // height of the glyph we do not consider them touched.
+ // See D3DTextRenderer.cpp:UpdateCachedDestination for more information.
+ // The leftOff/rightOff are only used in LCD cache case.
+ if (gcType == CACHE_LCD) {
+ jint threshold = glyph->height/3;
+
+ cellInfo->leftOff = pixelsTouchedL < threshold ? 1 : 0;
+ cellInfo->rightOff = pixelsTouchedR < threshold ? -1 : 0;
+ } else {
+ cellInfo->leftOff = 0;
+ cellInfo->rightOff = 0;
+ }
+ }
+
+ return res;
+}
+
+HRESULT
+D3DGlyphCache::CheckGlyphCacheByteOrder(jboolean rgbOrder)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGlyphCache::CheckGlyphCacheByteOrder");
+
+ if (gcType != CACHE_LCD) {
+ J2dTraceLn(J2D_TRACE_ERROR, "D3DGlyphCache::CheckGlyphCacheByteOrder"\
+ " invoked on CACHE_GRAY cache type instance!");
+ return E_FAIL;
+ }
+
+ if (rgbOrder != lastRGBOrder) {
+ // need to invalidate the cache in this case; see comments
+ // for lastRGBOrder
+ AccelGlyphCache_Invalidate(pGlyphCache);
+ lastRGBOrder = rgbOrder;
+ }
+ tileFormat = rgbOrder ? TILEFMT_3BYTE_RGB : TILEFMT_3BYTE_BGR;
+
+ return S_OK;
+}
+
+/**
+ * This method is invoked in the (relatively rare) case where one or
+ * more glyphs is about to be kicked out of the glyph cache texture.
+ * Here we simply flush the vertex queue of the current context in case
+ * any pending vertices are dependent upon the current glyph cache layout.
+ */
+static void
+D3DGlyphCache_FlushGlyphVertexCache()
+{
+ D3DContext *d3dc = D3DRQ_GetCurrentContext();
+ if (d3dc != NULL) {
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGlyphCache_FlushGlyphVertexCache");
+ d3dc->FlushVertexQueue();
+ }
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DGlyphCache.h b/jdk/src/windows/native/sun/java2d/d3d/D3DGlyphCache.h
new file mode 100644
index 00000000000..8edb499fb37
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DGlyphCache.h
@@ -0,0 +1,89 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DGLYPHCACHE_H
+#define D3DGLYPHCACHE_H
+
+#include "AccelGlyphCache.h"
+#include "D3DContext.h"
+#include "D3DResourceManager.h"
+
+typedef enum {
+ CACHE_GRAY,
+ CACHE_LCD
+} GlyphCacheType;
+
+class D3DContext;
+class D3DResource;
+
+class D3DGlyphCache {
+public:
+ // creates accel. glyph cache if it wasn't created, and the glyph
+ // cache texure
+ HRESULT Init(D3DContext *pCtx);
+ // releases the glyph cache texture, invalidates the accel. glyph cache
+ void ReleaseDefPoolResources();
+ // releases texture and deletes the accel. glyph cache
+ ~D3DGlyphCache();
+
+ // adds the glyph to the accel. glyph cache and uploads it into the glyph
+ // cache texture
+ HRESULT AddGlyph(GlyphInfo *glyph);
+
+ GlyphCacheInfo* GetGlyphCache() { return pGlyphCache; }
+ D3DResource* GetGlyphCacheTexture() { return pGlyphCacheRes; }
+
+ // Note: only applicable to CACHE_LCD type of the cache
+ // if the new rgb order doesn't match the current one, invalidates
+ // the accel. glyph cache, also resets the current tileFormat
+ HRESULT CheckGlyphCacheByteOrder(jboolean rgbOrder);
+
+static
+ HRESULT CreateInstance(D3DContext *pCtx,
+ GlyphCacheType gcType,
+ D3DGlyphCache **ppGlyphCache);
+
+private:
+ D3DGlyphCache(GlyphCacheType gcType);
+
+ D3DContext *pCtx;
+ GlyphCacheType gcType;
+ D3DResource *pGlyphCacheRes;
+ GlyphCacheInfo *pGlyphCache;
+ TileFormat tileFormat;
+ /**
+ * Relevant only for the CACHE_LCD cache type.
+ *
+ * This value tracks the previous LCD rgbOrder setting, so if the rgbOrder
+ * value has changed since the last time, it indicates that we need to
+ * invalidate the cache, which may already store glyph images in the
+ * reverse order. Note that in most real world applications this value
+ * will not change over the course of the application, but tests like
+ * Font2DTest allow for changing the ordering at runtime, so we need to
+ * handle that case.
+ */
+ jboolean lastRGBOrder;
+};
+#endif // D3DGLYPHCACHE_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DGraphicsDevice.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DGraphicsDevice.cpp
new file mode 100644
index 00000000000..03b016ab736
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DGraphicsDevice.cpp
@@ -0,0 +1,507 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "sun_java2d_d3d_D3DGraphicsDevice.h"
+#include "D3DGraphicsDevice.h"
+#include "D3DPipelineManager.h"
+#include "D3DRenderQueue.h"
+#include "Trace.h"
+#include "awt_Toolkit.h"
+#include "awt_Window.h"
+
+extern jobject CreateDisplayMode(JNIEnv* env, jint width, jint height,
+ jint bitDepth, jint refreshRate);
+extern void addDisplayMode(JNIEnv* env, jobject arrayList, jint width,
+ jint height, jint bitDepth, jint refreshRate);
+
+void InitD3D(void *pReturn)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "InitD3D");
+
+ jboolean *pRet = (jboolean *)pReturn;
+
+ D3DPipelineManager *pMgr = D3DPipelineManager::CreateInstance();
+ if (pMgr == NULL) {
+ J2dTraceLn(J2D_TRACE_ERROR, "InitD3D: could not create or init d3d");
+ *pRet = JNI_FALSE;
+ } else {
+ J2dTraceLn(J2D_TRACE_INFO, "InitD3D: successfully initialized d3d");
+ *pRet = JNI_TRUE;
+ }
+}
+
+extern "C" {
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: initD3D
+ * Signature: ()Z
+ */
+JNIEXPORT jboolean JNICALL Java_sun_java2d_d3d_D3DGraphicsDevice_initD3D
+ (JNIEnv *env, jclass)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_initD3D");
+
+ jboolean result = JNI_FALSE;
+ AwtToolkit::GetInstance().InvokeFunction(InitD3D, &result);
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DGD_initD3D: result=%x", result);
+ return result;
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: getDeviceIdNative
+ * Signature: (I)Ljava/lang/String;
+ */
+JNIEXPORT jstring JNICALL Java_sun_java2d_d3d_D3DGraphicsDevice_getDeviceIdNative
+ (JNIEnv *env, jclass d3dsdc, jint gdiScreen)
+{
+ D3DPipelineManager *pMgr;
+ UINT adapter;
+ D3DADAPTER_IDENTIFIER9 aid;
+ IDirect3D9 *pd3d9;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_getDeviceIdNative");
+
+ pMgr = D3DPipelineManager::GetInstance();
+ RETURN_STATUS_IF_NULL(pMgr, NULL);
+ pd3d9 = pMgr->GetD3DObject();
+ RETURN_STATUS_IF_NULL(pd3d9, NULL);
+
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+ if (FAILED(pd3d9->GetAdapterIdentifier(adapter, 0, &aid))) {
+ return NULL;
+ }
+
+ // ('%d.' will take no more than 6+1 chars since we are printing a WORD)
+ // AAAA&BBBB MAX_DEVICE_IDENTIFIER_STRING (%d.%d.%d.%d)0
+ size_t len = (4+1+4 +1+MAX_DEVICE_IDENTIFIER_STRING+1 +1+(6+1)*4+1 +1);
+ WCHAR *pAdapterId = new WCHAR[len];
+ RETURN_STATUS_IF_NULL(pAdapterId, NULL);
+
+ _snwprintf(pAdapterId, len, L"%x&%x %S (%d.%d.%d.%d)",
+ 0xffff & aid.VendorId, 0xffff & aid.DeviceId, aid.Description,
+ HIWORD(aid.DriverVersion.HighPart),
+ LOWORD(aid.DriverVersion.HighPart),
+ HIWORD(aid.DriverVersion.LowPart),
+ LOWORD(aid.DriverVersion.LowPart));
+ // _snwprintf doesn't add 0 at the end if the formatted string didn't fit
+ // in the buffer so we have to make sure it is null terminated
+ pAdapterId[len-1] = (WCHAR)0;
+
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " id=%S", pAdapterId);
+
+ jstring ret = JNU_NewStringPlatform(env, pAdapterId);
+
+ delete pAdapterId;
+
+ return ret;
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: getDeviceCapsNative
+ * Signature: (I)I
+ */
+JNIEXPORT jint JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_getDeviceCapsNative
+ (JNIEnv *env, jclass d3dsdc, jint gdiScreen)
+{
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+ UINT adapter;
+
+ J2dRlsTraceLn(J2D_TRACE_INFO, "D3DGD_getDeviceCapsNative");
+
+ pMgr = D3DPipelineManager::GetInstance();
+ RETURN_STATUS_IF_NULL(pMgr, CAPS_EMPTY);
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (FAILED(pMgr->GetD3DContext(adapter, &pCtx))) {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DGD_getDeviceCapsNative: device %d disabled", adapter);
+ return CAPS_EMPTY;
+ }
+ return pCtx->GetContextCaps();
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: enterFullScreenExclusiveNative
+ * Signature: (IJ)V
+ */
+JNIEXPORT jboolean JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_enterFullScreenExclusiveNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen, jlong window)
+{
+ HRESULT res;
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+ HWND hWnd;
+ AwtWindow *w;
+ D3DPRESENT_PARAMETERS newParams, *pCurParams;
+ D3DDISPLAYMODE dm;
+ UINT adapter;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_enterFullScreenExclusiveNative");
+
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (FAILED(res = pMgr->GetD3DContext(adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+ return JNI_FALSE;
+ }
+
+ w = (AwtWindow *)AwtComponent::GetComponent((HWND)window);
+ if (w == NULL || !::IsWindow(hWnd = w->GetTopLevelHWnd())) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DGD_enterFullScreenExclusiveNative: disposed window");
+ return JNI_FALSE;
+ }
+
+ // REMIND: should we also move the non-topleve window from
+ // being on top here (it's moved to front in GraphicsDevice.setFSW())?
+
+ pCtx->Get3DObject()->GetAdapterDisplayMode(adapter, &dm);
+ pCurParams = pCtx->GetPresentationParams();
+
+ // let the mananger know that we're entering the fs mode, it will
+ // set the proper current focus window for us, which ConfigureContext will
+ // use when creating the device
+ pMgr->SetFSFocusWindow(adapter, hWnd);
+
+ newParams = *pCurParams;
+ newParams.hDeviceWindow = hWnd;
+ newParams.Windowed = FALSE;
+ newParams.BackBufferCount = 1;
+ newParams.BackBufferFormat = dm.Format;
+ newParams.FullScreen_RefreshRateInHz = dm.RefreshRate;
+ newParams.BackBufferWidth = dm.Width;
+ newParams.BackBufferHeight = dm.Height;
+ newParams.PresentationInterval = D3DPRESENT_INTERVAL_DEFAULT;
+ newParams.SwapEffect = D3DSWAPEFFECT_DISCARD;
+
+ res = pCtx->ConfigureContext(&newParams);
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+ return SUCCEEDED(res);
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: exitFullScreenExclusiveNative
+ * Signature: (I)V
+ */
+JNIEXPORT jboolean JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_exitFullScreenExclusiveNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen)
+{
+ HRESULT res;
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+ D3DPRESENT_PARAMETERS newParams, *pCurParams;
+ UINT adapter;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_exitFullScreenExclusiveNative");
+
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (FAILED(res = pMgr->GetD3DContext(adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+ return JNI_FALSE;
+ }
+
+ pCurParams = pCtx->GetPresentationParams();
+
+ newParams = *pCurParams;
+ // we're exiting fs, the device window can be 0
+ newParams.hDeviceWindow = 0;
+ newParams.Windowed = TRUE;
+ newParams.BackBufferFormat = D3DFMT_UNKNOWN;
+ newParams.BackBufferCount = 1;
+ newParams.FullScreen_RefreshRateInHz = 0;
+ newParams.BackBufferWidth = 0;
+ newParams.BackBufferHeight = 0;
+ newParams.PresentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE;
+ newParams.SwapEffect = D3DSWAPEFFECT_COPY;
+
+ res = pCtx->ConfigureContext(&newParams);
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+
+ // exited fs, update current focus window
+ // note that we call this after this adapter exited fs mode so that
+ // the rest of the adapters can be reset
+ pMgr->SetFSFocusWindow(adapter, 0);
+
+ return SUCCEEDED(res);
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: configDisplayModeNative
+ * Signature: (IJIIII)V
+ */
+JNIEXPORT void JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_configDisplayModeNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen, jlong window,
+ jint width, jint height, jint bitDepth, jint refreshRate)
+{
+ HRESULT res;
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+ D3DPRESENT_PARAMETERS newParams, *pCurParams;
+ UINT adapter;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_configDisplayModeNative");
+
+ RETURN_IF_NULL(pMgr = D3DPipelineManager::GetInstance());
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (FAILED(res = pMgr->GetD3DContext(adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+ return;
+ }
+
+ pCurParams = pCtx->GetPresentationParams();
+
+ newParams = *pCurParams;
+ newParams.BackBufferWidth = width;
+ newParams.BackBufferHeight = height;
+ newParams.FullScreen_RefreshRateInHz = refreshRate;
+ newParams.PresentationInterval = D3DPRESENT_INTERVAL_DEFAULT;
+ // we leave the swap effect so that it's more likely
+ // to be the one user selected initially
+// newParams.SwapEffect = D3DSWAPEFFECT_DISCARD;
+
+ if (bitDepth == 32) {
+ newParams.BackBufferFormat = D3DFMT_X8R8G8B8;
+ } else if (bitDepth == 16) {
+ UINT modeNum;
+ D3DDISPLAYMODE mode;
+ IDirect3D9 *pd3d9;
+ UINT modesCount;
+
+ RETURN_IF_NULL(pd3d9 = pMgr->GetD3DObject());
+
+ modesCount = pd3d9->GetAdapterModeCount(adapter, D3DFMT_R5G6B5);
+ if (modesCount == 0) {
+ modesCount = pd3d9->GetAdapterModeCount(adapter, D3DFMT_X1R5G5B5);
+ }
+
+ newParams.BackBufferFormat = D3DFMT_UNKNOWN;
+ for (modeNum = 0; modeNum < modesCount; modeNum++) {
+ if (SUCCEEDED(pd3d9->EnumAdapterModes(adapter, D3DFMT_R5G6B5,
+ modeNum, &mode)))
+ {
+ if (mode.Width == width && mode.Height == height &&
+ mode.RefreshRate == refreshRate)
+ {
+ // prefer 565 over 555
+ if (mode.Format == D3DFMT_R5G6B5) {
+ newParams.BackBufferFormat = D3DFMT_R5G6B5;
+ break;
+ } else if (mode.Format == D3DFMT_X1R5G5B5) {
+ newParams.BackBufferFormat = D3DFMT_X1R5G5B5;
+ }
+ }
+ }
+ }
+ if (newParams.BackBufferFormat == D3DFMT_UNKNOWN) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DGD_configDisplayModeNative: no 16-bit formats");
+ return;
+ }
+ } else {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DGD_configDisplayModeNative: unsupported depth: %d",
+ bitDepth);
+ return;
+ }
+
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " changing to dm: %dx%dx%d@%d",
+ newParams.BackBufferWidth, newParams.BackBufferHeight,
+ bitDepth, refreshRate);
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " selected backbuffer format: %d",
+ newParams.BackBufferFormat);
+
+ res = pCtx->ConfigureContext(&newParams);
+ if (SUCCEEDED(res)) {
+ // the full screen window doesn't receive WM_SIZE event when
+ // the display mode changes (it does get resized though) so we need to
+ // generate the event ourselves
+ ::SendMessage(newParams.hDeviceWindow, WM_SIZE, width, height);
+ }
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+}
+
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: getCurrentDisplayModeNative
+ * Signature: (I)Ljava/awt/DisplayMode;
+ */
+JNIEXPORT jobject JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_getCurrentDisplayModeNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen)
+{
+ D3DPipelineManager *pMgr;
+ IDirect3D9 *pd3d9;
+ jobject ret = NULL;
+ D3DDISPLAYMODE mode;
+ UINT adapter;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_getCurrentDisplayModeNative");
+
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), NULL);
+ RETURN_STATUS_IF_NULL(pd3d9 = pMgr->GetD3DObject(), NULL);
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (SUCCEEDED(pd3d9->GetAdapterDisplayMode(adapter, &mode))) {
+ int bitDepth = -1;
+ // these are the only three valid screen formats
+ switch (mode.Format) {
+ case D3DFMT_X8R8G8B8: bitDepth = 32; break;
+ case D3DFMT_R5G6B5:
+ case D3DFMT_X1R5G5B5: bitDepth = 16; break;
+ }
+ J2dTraceLn4(J2D_TRACE_VERBOSE,
+ " current dm: %dx%dx%d@%d",
+ mode.Width, mode.Height, bitDepth, mode.RefreshRate);
+ ret = CreateDisplayMode(env, mode.Width, mode.Height, bitDepth,
+ mode.RefreshRate);
+ }
+ return ret;
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: enumDisplayModesNative
+ * Signature: (ILjava/util/ArrayList;)V
+ */
+JNIEXPORT void JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_enumDisplayModesNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen, jobject arrayList)
+{
+ D3DPipelineManager *pMgr;
+ IDirect3D9 *pd3d9;
+ jobject ret = NULL;
+ D3DDISPLAYMODE mode;
+ UINT formatNum, modeNum, modesCount;
+ UINT adapter;
+ // EnumAdapterModes treats 555 and 565 formats as equivalents
+ static D3DFORMAT formats[] =
+ { D3DFMT_X8R8G8B8, D3DFMT_R5G6B5 };
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_enumDisplayModesNative");
+
+ RETURN_IF_NULL(pMgr = D3DPipelineManager::GetInstance());
+ RETURN_IF_NULL(pd3d9 = pMgr->GetD3DObject());
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ for (formatNum = 0; formatNum < 3; formatNum++) {
+ modesCount = pd3d9->GetAdapterModeCount(adapter, formats[formatNum]);
+ for (modeNum = 0; modeNum < modesCount; modeNum++) {
+ if (SUCCEEDED(pd3d9->EnumAdapterModes(adapter, formats[formatNum],
+ modeNum, &mode)))
+ {
+ int bitDepth = -1;
+ // these are the only three valid screen formats,
+ // 30-bit is returned as X8R8G8B8
+ switch (mode.Format) {
+ case D3DFMT_X8R8G8B8: bitDepth = 32; break;
+ case D3DFMT_R5G6B5:
+ case D3DFMT_X1R5G5B5: bitDepth = 16; break;
+ }
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " found dm: %dx%dx%d@%d",
+ mode.Width, mode.Height, bitDepth,mode.RefreshRate);
+ addDisplayMode(env, arrayList, mode.Width, mode.Height,
+ bitDepth, mode.RefreshRate);
+ }
+ }
+ }
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: getAvailableAcceleratedMemoryNative
+ * Signature: (I)J
+ */
+JNIEXPORT jlong JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_getAvailableAcceleratedMemoryNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen)
+{
+ // REMIND: looks like Direct3D provides information about texture memory
+ // only via IDirect3DDevice9::GetAvailableTextureMem, however, it
+ // seems to report the same amount as direct draw used to.
+ HRESULT res;
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+ IDirect3DDevice9 *pd3dDevice;
+ UINT adapter;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_getAvailableAcceleratedMemoryNative");
+
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), 0L);
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (FAILED(res = pMgr->GetD3DContext(adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+ return 0L;
+ }
+ RETURN_STATUS_IF_NULL(pd3dDevice = pCtx->Get3DDevice(), 0L);
+
+ UINT mem = pd3dDevice->GetAvailableTextureMem();
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " available memory=%d", mem);
+ return mem;
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DGraphicsDevice
+ * Method: isD3DAvailableOnDeviceNative
+ * Signature: (I)Z
+ */
+JNIEXPORT jboolean JNICALL
+Java_sun_java2d_d3d_D3DGraphicsDevice_isD3DAvailableOnDeviceNative
+ (JNIEnv *env, jclass gdc, jint gdiScreen)
+{
+ HRESULT res;
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+ UINT adapter;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DGD_isD3DAvailableOnDeviceNative");
+
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+ adapter = pMgr->GetAdapterOrdinalForScreen(gdiScreen);
+
+ if (FAILED(res = pMgr->GetD3DContext(adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, D3DRQ_GetCurrentDestination());
+ return JNI_FALSE;
+ }
+
+ return JNI_TRUE;
+}
+
+} // extern "C"
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DGraphicsDevice.h b/jdk/src/windows/native/sun/java2d/d3d/D3DGraphicsDevice.h
new file mode 100644
index 00000000000..812d85f5e1c
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DGraphicsDevice.h
@@ -0,0 +1,29 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef _D3DGRAPHICSDEVICE_H_
+#define _D3DGRAPHICSDEVICE_H_
+
+#endif /* _D3DGRAPHICSDEVICE_H_ */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DMaskBlit.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskBlit.cpp
new file mode 100644
index 00000000000..0a92e85da64
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskBlit.cpp
@@ -0,0 +1,107 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include
+#include
+
+#include "D3DMaskBlit.h"
+#include "D3DRenderQueue.h"
+#include "D3DSurfaceData.h"
+
+/**
+ * REMIND: This method assumes that the dimensions of the incoming pixel
+ * array are less than or equal to the cached blit texture tile;
+ * these are rather fragile assumptions, and should be cleaned up...
+ */
+HRESULT
+D3DMaskBlit_MaskBlit(JNIEnv *env, D3DContext *d3dc,
+ jint dstx, jint dsty,
+ jint width, jint height,
+ void *pPixels)
+{
+ HRESULT res = S_OK;
+ jfloat dx1, dy1, dx2, dy2;
+ jfloat tx1, ty1, tx2, ty2;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DMaskBlit_MaskBlit");
+
+ if (width <= 0 || height <= 0) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DMaskBlit_MaskBlit: invalid dimensions");
+ return res;
+ }
+
+ RETURN_STATUS_IF_NULL(pPixels, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ if (FAILED(res = d3dc->BeginScene(STATE_TEXTUREOP))) {
+ return res;
+ }
+
+ D3DResource *pBlitTexRes;
+ if (FAILED(res =
+ d3dc->GetResourceManager()->GetBlitTexture(&pBlitTexRes)))
+ {
+ return res;
+ }
+ IDirect3DTexture9 *pBlitTex = pBlitTexRes->GetTexture();
+
+ if (FAILED(res = d3dc->SetTexture(pBlitTex, 0))) {
+ return res;
+ }
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ D3DTEXTUREFILTERTYPE fhint =
+ d3dc->IsTextureFilteringSupported(D3DTEXF_NONE) ?
+ D3DTEXF_NONE : D3DTEXF_POINT;
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, fhint);
+
+ // copy system memory IntArgbPre surface into cached texture
+ if (FAILED(res = d3dc->UploadTileToTexture(pBlitTexRes, pPixels,
+ 0, 0, 0, 0,
+ width, height,
+ width*4,
+ TILEFMT_4BYTE_ARGB_PRE)))
+ {
+ return res;
+ }
+
+ dx1 = (jfloat)dstx;
+ dy1 = (jfloat)dsty;
+ dx2 = dx1 + width;
+ dy2 = dy1 + height;
+
+ tx1 = 0.0f;
+ ty1 = 0.0f;
+ tx2 = ((jfloat)width) / D3DC_BLIT_TILE_SIZE;
+ ty2 = ((jfloat)height) / D3DC_BLIT_TILE_SIZE;
+
+ // render cached texture to the destination surface
+ res = d3dc->pVCacher->DrawTexture(dx1, dy1, dx2, dy2,
+ tx1, ty1, tx2, ty2);
+ res = d3dc->pVCacher->Render();
+
+ return res;
+}
diff --git a/jdk/src/windows/native/sun/java2d/windows/dxInit.h b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskBlit.h
similarity index 53%
rename from jdk/src/windows/native/sun/java2d/windows/dxInit.h
rename to jdk/src/windows/native/sun/java2d/d3d/D3DMaskBlit.h
index 1d8e5f6aa4c..51cecd8b8bc 100644
--- a/jdk/src/windows/native/sun/java2d/windows/dxInit.h
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskBlit.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -23,36 +23,14 @@
* have any questions.
*/
-#ifndef DXINIT_H
-#define DXINIT_H
+#ifndef D3DMaskBlit_h_Included
+#define D3DMaskBlit_h_Included
-#include "Win32SurfaceData.h"
-#include "Trace.h"
-#include "awt_MMStub.h"
-#include "dxCapabilities.h"
+#include "D3DContext.h"
-// Registry definitions: these values are used to determine whether
-// acceleration components are untested, working, or broken, depending
-// on the results of testing
-#define J2D_ACCEL_KEY_ROOT L"Software\\JavaSoft\\Java2D\\"
-#define J2D_ACCEL_DRIVER_SUBKEY L"Drivers\\"
-#define J2D_ACCEL_DX_NAME L"DXAcceleration"
+HRESULT D3DMaskBlit_MaskBlit(JNIEnv *env, D3DContext *d3dc,
+ jint dstx, jint dsty,
+ jint width, jint height,
+ void *pPixels);
-void InitDirectX();
-
-void GetDeviceKeyName(_DISPLAY_DEVICE *displayDevice, WCHAR *devName);
-
-void CheckFlags();
-
-void CheckRegistry();
-
-BOOL DDSetupDevice(DDrawObjectStruct *tmpDdInstance, DxCapabilities *dxCaps);
-
-DDrawObjectStruct *CreateDevice(GUID *lpGUID, HMONITOR hMonitor);
-
-BOOL CALLBACK EnumDeviceCallback(GUID FAR* lpGUID, LPSTR szName, LPSTR szDevice,
- LPVOID lParam, HMONITOR hMonitor);
-
-BOOL DDCreateObject();
-
-#endif DXINIT_H
+#endif /* D3DMaskBlit_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DMaskCache.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskCache.cpp
new file mode 100644
index 00000000000..1e56609d80d
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskCache.cpp
@@ -0,0 +1,157 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "D3DMaskCache.h"
+
+HRESULT
+D3DMaskCache::CreateInstance(D3DContext *pCtx, D3DMaskCache **ppMaskCache)
+{
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DMaskCache::CreateInstance");
+
+ *ppMaskCache = new D3DMaskCache();
+ if (FAILED(res = (*ppMaskCache)->Init(pCtx))) {
+ delete *ppMaskCache;
+ *ppMaskCache = NULL;
+ }
+ return res;
+}
+
+D3DMaskCache::D3DMaskCache()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DMaskCache::D3DMaskCache");
+ this->pCtx = NULL;
+ maskCacheIndex = 0;
+}
+
+D3DMaskCache::~D3DMaskCache()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DMaskCache::~D3DMaskCache");
+ pCtx = NULL;
+ maskCacheIndex = 0;
+}
+
+HRESULT
+D3DMaskCache::Init(D3DContext *pCtx)
+{
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DMaskCache::Init pCtx=%x", pCtx);
+ this->pCtx = pCtx;
+ this->maskCacheIndex = 0;
+ return S_OK;
+}
+
+HRESULT D3DMaskCache::Enable()
+{
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DMaskCache::Enable");
+
+ D3DResource *pMaskTexRes;
+ res = pCtx->GetResourceManager()->GetMaskTexture(&pMaskTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+
+ res = pCtx->SetTexture(pMaskTexRes->GetTexture(), 0);
+
+ IDirect3DDevice9 *pd3dDevice = pCtx->Get3DDevice();
+ D3DTEXTUREFILTERTYPE fhint =
+ pCtx->IsTextureFilteringSupported(D3DTEXF_NONE) ?
+ D3DTEXF_NONE : D3DTEXF_POINT;
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, fhint);
+
+ return res;
+}
+
+HRESULT D3DMaskCache::Disable()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DMaskCache::Disable");
+
+ maskCacheIndex = 0;
+
+ return pCtx->SetTexture(NULL, 0);
+}
+
+HRESULT D3DMaskCache::AddMaskQuad(int srcx, int srcy,
+ int dstx, int dsty,
+ int width, int height,
+ int maskscan, void *mask)
+{
+ HRESULT res;
+ float tx1, ty1, tx2, ty2;
+ float dx1, dy1, dx2, dy2;
+
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DVertexCacher::AddMaskQuad: %d",
+ maskCacheIndex);
+
+ if (maskCacheIndex >= D3D_MASK_CACHE_MAX_INDEX ||
+ pCtx->pVCacher->GetFreeVertices() < 6)
+ {
+ res = pCtx->pVCacher->Render();
+ RETURN_STATUS_IF_FAILED(res);
+ maskCacheIndex = 0;
+ }
+
+ if (mask != NULL) {
+ int texx = D3D_MASK_CACHE_TILE_WIDTH *
+ (maskCacheIndex % D3D_MASK_CACHE_WIDTH_IN_TILES);
+ int texy = D3D_MASK_CACHE_TILE_HEIGHT *
+ (maskCacheIndex / D3D_MASK_CACHE_WIDTH_IN_TILES);
+ D3DResource *pMaskTexRes;
+
+ res = pCtx->GetResourceManager()->GetMaskTexture(&pMaskTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // copy alpha mask into texture tile
+ pCtx->UploadTileToTexture(pMaskTexRes, mask,
+ texx, texy,
+ srcx, srcy,
+ width, height,
+ maskscan,
+ TILEFMT_1BYTE_ALPHA);
+
+ tx1 = ((float)texx) / D3D_MASK_CACHE_WIDTH_IN_TEXELS;
+ ty1 = ((float)texy) / D3D_MASK_CACHE_HEIGHT_IN_TEXELS;
+
+ maskCacheIndex++;
+ } else {
+ // use special fully opaque tile
+ tx1 = ((float)D3D_MASK_CACHE_SPECIAL_TILE_X) /
+ D3D_MASK_CACHE_WIDTH_IN_TEXELS;
+ ty1 = ((float)D3D_MASK_CACHE_SPECIAL_TILE_Y) /
+ D3D_MASK_CACHE_HEIGHT_IN_TEXELS;
+ }
+
+ tx2 = tx1 + (((float)width) / D3D_MASK_CACHE_WIDTH_IN_TEXELS);
+ ty2 = ty1 + (((float)height) / D3D_MASK_CACHE_HEIGHT_IN_TEXELS);
+
+ dx1 = (float)dstx;
+ dy1 = (float)dsty;
+ dx2 = dx1 + width;
+ dy2 = dy1 + height;
+
+ return pCtx->pVCacher->DrawTexture(dx1, dy1, dx2, dy2,
+ tx1, ty1, tx2, ty2);
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DMaskCache.h b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskCache.h
new file mode 100644
index 00000000000..ad2ac2f70e8
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskCache.h
@@ -0,0 +1,83 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DMASKCACHE_H
+#define D3DMASKCACHE_H
+
+#include "jni.h"
+#include "D3DContext.h"
+
+/**
+ * Constants that control the size of the texture tile cache used for
+ * mask operations.
+ */
+#define D3D_MASK_CACHE_TILE_WIDTH 32
+#define D3D_MASK_CACHE_TILE_HEIGHT 32
+#define D3D_MASK_CACHE_TILE_SIZE \
+ (D3D_MASK_CACHE_TILE_WIDTH * D3D_MASK_CACHE_TILE_HEIGHT)
+
+#define D3D_MASK_CACHE_WIDTH_IN_TILES 8
+#define D3D_MASK_CACHE_HEIGHT_IN_TILES 4
+
+#define D3D_MASK_CACHE_WIDTH_IN_TEXELS \
+ (D3D_MASK_CACHE_TILE_WIDTH * D3D_MASK_CACHE_WIDTH_IN_TILES)
+#define D3D_MASK_CACHE_HEIGHT_IN_TEXELS \
+ (D3D_MASK_CACHE_TILE_HEIGHT * D3D_MASK_CACHE_HEIGHT_IN_TILES)
+
+/*
+ * We reserve one (fully opaque) tile in the upper-right corner for
+ * operations where the mask is null.
+ */
+#define D3D_MASK_CACHE_MAX_INDEX \
+ ((D3D_MASK_CACHE_WIDTH_IN_TILES * D3D_MASK_CACHE_HEIGHT_IN_TILES) - 1)
+#define D3D_MASK_CACHE_SPECIAL_TILE_X \
+ (D3D_MASK_CACHE_WIDTH_IN_TEXELS - D3D_MASK_CACHE_TILE_WIDTH)
+#define D3D_MASK_CACHE_SPECIAL_TILE_Y \
+ (D3D_MASK_CACHE_HEIGHT_IN_TEXELS - D3D_MASK_CACHE_TILE_HEIGHT)
+
+class D3DContext;
+
+class D3DMaskCache {
+public:
+ HRESULT Init(D3DContext *pCtx);
+ void ReleaseDefPoolResources() {};
+ ~D3DMaskCache();
+ HRESULT Enable();
+ HRESULT Disable();
+ HRESULT AddMaskQuad(int srcx, int srcy,
+ int dstx, int dsty,
+ int width, int height,
+ int maskscan, void *mask);
+
+static
+ HRESULT CreateInstance(D3DContext *pCtx, D3DMaskCache **ppMaskCache);
+
+private:
+ D3DMaskCache();
+ UINT maskCacheIndex;
+ D3DContext *pCtx;
+};
+
+#endif // D3DMASKCACHE_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.cpp
index e1748607183..67bbfccbe47 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.cpp
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -23,187 +23,102 @@
* have any questions.
*/
-#include
-#include
-#include "ddrawUtils.h"
-#include "GraphicsPrimitiveMgr.h"
-#include "j2d_md.h"
-#include "jlong.h"
-
#include "sun_java2d_d3d_D3DMaskFill.h"
-#include "Win32SurfaceData.h"
+#include "D3DMaskFill.h"
+#include "D3DRenderQueue.h"
-#include "D3DContext.h"
-#include "D3DUtils.h"
-
-
-extern "C" {
-
-inline static HRESULT doMaskFill
- (JNIEnv *env, jobject self,
- Win32SDOps *wsdo, D3DContext *d3dc,
- jint x, jint y, jint w, jint h,
- jbyteArray maskArray,
- jint maskoff, jint maskscan);
-
-
-JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DMaskFill_MaskFill
- (JNIEnv *env, jobject self,
- jlong pData, jlong pCtx,
- jint x, jint y, jint w, jint h,
- jbyteArray maskArray,
- jint maskoff, jint maskscan)
+/**
+ * This implementation first copies the alpha tile into a texture and then
+ * maps that texture to the destination surface. This approach appears to
+ * offer the best performance despite being a two-step process.
+ *
+ * Here are some descriptions of the many variables used in this method:
+ * x,y - upper left corner of the tile destination
+ * w,h - width/height of the mask tile
+ * x0 - placekeeper for the original destination x location
+ * tw,th - width/height of the actual texture tile in pixels
+ * sx1,sy1 - upper left corner of the mask tile source region
+ * sx2,sy2 - lower left corner of the mask tile source region
+ * sx,sy - "current" upper left corner of the mask tile region of interest
+ */
+HRESULT
+D3DMaskFill_MaskFill(D3DContext *d3dc,
+ jint x, jint y, jint w, jint h,
+ jint maskoff, jint maskscan, jint masklen,
+ unsigned char *pMask)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
+ HRESULT res = S_OK;
J2dTraceLn(J2D_TRACE_INFO, "D3DMaskFill_MaskFill");
- J2dTraceLn4(J2D_TRACE_VERBOSE, " x=%-4d y=%-4d w=%-4d h=%-4d",
- x, y, w, h);
- J2dTraceLn2(J2D_TRACE_VERBOSE, " maskoff=%-4d maskscan=%-4d",
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " x=%d y=%d w=%d h=%d", x, y, w, h);
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " maskoff=%d maskscan=%d",
maskoff, maskscan);
- if (d3dc == NULL || wsdo == NULL) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "D3DMaskFill_MaskFill: context is null");
- return;
- }
-
- HRESULT res;
- D3D_EXEC_PRIM_LOOP(env, res, wsdo,
- doMaskFill(env, self, wsdo, d3dc,
- x, y, w, h,
- maskArray, maskoff, maskscan));
-}
-
-inline static HRESULT doMaskFill
- (JNIEnv *env, jobject self,
- Win32SDOps *wsdo, D3DContext *d3dc,
- jint x, jint y, jint w, jint h,
- jbyteArray maskArray,
- jint maskoff, jint maskscan)
-{
- DXSurface *maskTexture;
- static J2DLVERTEX quadVerts[4] = {
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f }
- };
-
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return DDERR_GENERIC;
- }
-
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
-
- HRESULT res = D3D_OK;
- if (maskArray) {
- jubyte *pMask =
- (jubyte*)env->GetPrimitiveArrayCritical(maskArray, 0);
- float tx1, ty1, tx2, ty2;
+ {
+ D3DMaskCache *maskCache = d3dc->GetMaskCache();
jint tw, th, x0;
jint sx1, sy1, sx2, sy2;
jint sx, sy, sw, sh;
- if (pMask == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return DDERR_GENERIC;
- }
-
- maskTexture = d3dc->GetMaskTexture();
- if (maskTexture == NULL ||
- FAILED(res = d3dc->BeginScene(STATE_MASKOP)))
- {
- env->ReleasePrimitiveArrayCritical(maskArray, pMask, JNI_ABORT);
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return DDERR_GENERIC;
- }
-
- if (FAILED(res = d3dc->SetTexture(maskTexture))) {
- d3dc->EndScene(res);
- env->ReleasePrimitiveArrayCritical(maskArray, pMask, JNI_ABORT);
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return res;
- }
+ res = d3dc->BeginScene(STATE_MASKOP);
+ RETURN_STATUS_IF_FAILED(res);
x0 = x;
- tx1 = 0.0f;
- ty1 = 0.0f;
- tw = D3DSD_MASK_TILE_SIZE;
- th = D3DSD_MASK_TILE_SIZE;
+ tw = D3D_MASK_CACHE_TILE_WIDTH;
+ th = D3D_MASK_CACHE_TILE_HEIGHT;
sx1 = maskoff % maskscan;
sy1 = maskoff / maskscan;
sx2 = sx1 + w;
sy2 = sy1 + h;
- D3DU_INIT_VERTEX_QUAD_COLOR(quadVerts, d3dc->colorPixel);
- for (sy = sy1; (sy < sy2) && SUCCEEDED(res); sy += th, y += th) {
+ for (sy = sy1; sy < sy2; sy += th, y += th) {
x = x0;
sh = ((sy + th) > sy2) ? (sy2 - sy) : th;
- for (sx = sx1; (sx < sx2) && SUCCEEDED(res); sx += tw, x += tw) {
+ for (sx = sx1; sx < sx2; sx += tw, x += tw) {
sw = ((sx + tw) > sx2) ? (sx2 - sx) : tw;
- if (FAILED(d3dc->UploadImageToTexture(maskTexture,
- pMask,
- 0, 0, sx, sy, sw, sh,
- maskscan)))
- {
- continue;
- }
-
- // update the lower right texture coordinates
- tx2 = ((float)sw) / tw;
- ty2 = ((float)sh) / th;
-
- D3DU_INIT_VERTEX_QUAD_XYUV(quadVerts,
- (float)x, (float)y,
- (float)(x+sw), (float)(y+sh),
- tx1, ty1, tx2, ty2);
- if (SUCCEEDED(res = ddTargetSurface->IsLost())) {
- // render texture tile to the destination surface
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN,
- D3DFVF_J2DLVERTEX,
- quadVerts, 4, 0);
- }
-
+ res = maskCache->AddMaskQuad(sx, sy, x, y, sw, sh,
+ maskscan, pMask);
}
}
-
- d3dc->EndScene(res);
-
- env->ReleasePrimitiveArrayCritical(maskArray, pMask, JNI_ABORT);
- } else {
- float x1 = (float)x;
- float y1 = (float)y;
- float x2 = x1 + (float)w;
- float y2 = y1 + (float)h;
- D3DU_INIT_VERTEX_QUAD_COLOR(quadVerts, d3dc->colorPixel);
- D3DU_INIT_VERTEX_QUAD_XY(quadVerts, x1, y1, x2, y2);
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_RENDEROP))) {
- if (SUCCEEDED(res = ddTargetSurface->IsLost())) {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN,
- D3DFVF_J2DLVERTEX,
- quadVerts, 4, 0);
- }
- d3dc->EndScene(res);
- }
}
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
-
return res;
}
+JNIEXPORT void JNICALL
+Java_sun_java2d_d3d_D3DMaskFill_maskFill
+ (JNIEnv *env, jobject self,
+ jint x, jint y, jint w, jint h,
+ jint maskoff, jint maskscan, jint masklen,
+ jbyteArray maskArray)
+{
+ D3DContext *d3dc = D3DRQ_GetCurrentContext();
+ unsigned char *mask;
+
+ J2dTraceLn(J2D_TRACE_ERROR, "D3DMaskFill_maskFill");
+
+ if (maskArray != NULL) {
+ mask = (unsigned char *)
+ env->GetPrimitiveArrayCritical(maskArray, NULL);
+ } else {
+ mask = NULL;
+ }
+
+ D3DMaskFill_MaskFill(d3dc,
+ x, y, w, h,
+ maskoff, maskscan, masklen, mask);
+
+ // reset current state, and ensure rendering is flushed to dest
+ if (d3dc != NULL) {
+ d3dc->FlushVertexQueue();
+ }
+
+ if (mask != NULL) {
+ env->ReleasePrimitiveArrayCritical(maskArray, mask, JNI_ABORT);
+ }
}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.h b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.h
new file mode 100644
index 00000000000..dd5d1678408
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DMaskFill.h
@@ -0,0 +1,36 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DMaskFill_h_Included
+#define D3DMaskFill_h_Included
+
+#include "D3DContext.h"
+
+HRESULT D3DMaskFill_MaskFill(D3DContext *d3dc,
+ jint x, jint y, jint w, jint h,
+ jint maskoff, jint maskscan, jint masklen,
+ unsigned char *pMask);
+
+#endif /* D3DMaskFill_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DPaints.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DPaints.cpp
new file mode 100644
index 00000000000..f1104990445
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DPaints.cpp
@@ -0,0 +1,532 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include
+#include
+
+#include "sun_java2d_d3d_D3DPaints_MultiGradient.h"
+
+#include "D3DPaints.h"
+#include "D3DContext.h"
+#include "D3DRenderQueue.h"
+#include "D3DSurfaceData.h"
+
+HRESULT
+D3DPaints_ResetPaint(D3DContext *d3dc)
+{
+ jint pixel, paintState;
+ jubyte ea;
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPaints_ResetPaint");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ paintState = d3dc->GetPaintState();
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " state=%d", paintState);
+
+ res = d3dc->UpdateState(STATE_OTHEROP);
+
+ // disable current complex paint state, if necessary
+ if (paintState > PAINT_ALPHACOLOR) {
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ DWORD sampler = d3dc->useMask ? 1 : 0;
+
+ d3dc->SetTexture(NULL, sampler);
+ pd3dDevice->SetSamplerState(sampler,
+ D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(sampler,
+ D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetTextureStageState(sampler,
+ D3DTSS_TEXCOORDINDEX, sampler);
+ res = pd3dDevice->SetTextureStageState(sampler,
+ D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_DISABLE);
+
+ if (paintState == PAINT_GRADIENT ||
+ paintState == PAINT_LIN_GRADIENT ||
+ paintState == PAINT_RAD_GRADIENT)
+ {
+ res = pd3dDevice->SetPixelShader(NULL);
+ }
+ }
+
+ // set each component of the current color state to the extra alpha
+ // value, which will effectively apply the extra alpha to each fragment
+ // in paint/texturing operations
+ ea = (jubyte)(d3dc->extraAlpha * 0xff + 0.5f);
+ pixel = (ea << 24) | (ea << 16) | (ea << 8) | (ea << 0);
+ d3dc->pVCacher->SetColor(pixel);
+ d3dc->useMask = JNI_FALSE;
+ d3dc->SetPaintState(-1);
+ return res;
+}
+
+HRESULT
+D3DPaints_SetColor(D3DContext *d3dc, jint pixel)
+{
+ HRESULT res = S_OK;
+
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DPaints_SetColor: pixel=%08x", pixel);
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+
+ // no need to reset the current op state here unless the paint
+ // state really needs to be changed
+ if (d3dc->GetPaintState() > PAINT_ALPHACOLOR) {
+ res = D3DPaints_ResetPaint(d3dc);
+ }
+
+ d3dc->pVCacher->SetColor(pixel);
+ d3dc->useMask = JNI_FALSE;
+ d3dc->SetPaintState(PAINT_ALPHACOLOR);
+ return res;
+}
+
+/************************* GradientPaint support ****************************/
+
+HRESULT
+D3DPaints_SetGradientPaint(D3DContext *d3dc,
+ jboolean useMask, jboolean cyclic,
+ jdouble p0, jdouble p1, jdouble p3,
+ jint pixel1, jint pixel2)
+{
+ IDirect3DDevice9 *pd3dDevice;
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPaints_SetGradientPaint");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ D3DPaints_ResetPaint(d3dc);
+
+#if 0
+ /*
+ * REMIND: The following code represents the original fast gradient
+ * implementation. The problem is that it relies on LINEAR
+ * texture filtering, which does not provide sufficient
+ * precision on certain hardware (from ATI, notably), which
+ * will cause visible banding (e.g. 64 shades of gray between
+ * black and white, instead of the expected 256 shades. For
+ * correctness on such hardware, it is necessary to use a
+ * shader-based approach that does not suffer from these
+ * precision issues (see below). This original implementation
+ * is about 16x faster than software, whereas the shader-based
+ * implementation is only about 4x faster than software (still
+ * impressive). For simplicity, we will always use the
+ * shader-based version for now, but in the future we could
+ * consider using the fast path for certain hardware (that does
+ * not exhibit the problem) or provide a flag to allow developers
+ * to control which path we take (for those that are less
+ * concerned about quality). Therefore, I'll leave this code
+ * here (currently disabled) for future use.
+ */
+ D3DResource *pGradientTexRes;
+ IDirect3DTexture9 *pGradientTex;
+
+ // this will initialize the gradient texture, if necessary
+ res = d3dc->GetResourceManager()->GetGradientTexture(&pGradientTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pGradientTex = pGradientTexRes->GetTexture();
+
+ // update the texture containing the gradient colors
+ {
+ D3DLOCKED_RECT lockedRect;
+ res = pGradientTex->LockRect(0, &lockedRect, NULL, D3DLOCK_NOSYSLOCK);
+ RETURN_STATUS_IF_FAILED(res);
+ jint *pPix = (jint*)lockedRect.pBits;
+ pPix[0] = pixel1;
+ pPix[1] = pixel2;
+ pGradientTex->UnlockRect(0);
+ }
+
+ DWORD sampler = useMask ? 1 : 0;
+ DWORD wrapMode = cyclic ? D3DTADDRESS_WRAP : D3DTADDRESS_CLAMP;
+ d3dc->SetTexture(pGradientTex, sampler);
+ d3dc->UpdateTextureColorState(D3DTA_TEXTURE, sampler);
+
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_ADDRESSU, wrapMode);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_ADDRESSV, wrapMode);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_MINFILTER, D3DTEXF_LINEAR);
+
+ D3DMATRIX mt;
+ ZeroMemory(&mt, sizeof(mt));
+ mt._11 = (float)p0;
+ mt._21 = (float)p1;
+ mt._31 = (float)0.0;
+ mt._41 = (float)p3;
+ mt._12 = 0.0f;
+ mt._22 = 1.0f;
+ mt._32 = 0.0f;
+ mt._42 = 0.0f;
+ pd3dDevice->SetTransform(useMask ? D3DTS_TEXTURE1 : D3DTS_TEXTURE0, &mt);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXCOORDINDEX,
+ D3DTSS_TCI_CAMERASPACEPOSITION);
+ res = pd3dDevice->SetTextureStageState(sampler,
+ D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_COUNT2);
+#else
+ jfloat params[4];
+ jfloat color[4];
+ jint flags = 0;
+
+ if (cyclic) flags |= BASIC_GRAD_IS_CYCLIC;
+ if (useMask) flags |= BASIC_GRAD_USE_MASK;
+
+ // locate/enable the shader program for the given flags
+ res = d3dc->EnableBasicGradientProgram(flags);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the "uniform" values
+ params[0] = (jfloat)p0;
+ params[1] = (jfloat)p1;
+ params[2] = (jfloat)p3;
+ params[3] = 0.0f; // unused
+ pd3dDevice = d3dc->Get3DDevice();
+ res = pd3dDevice->SetPixelShaderConstantF(0, params, 1);
+
+ color[0] = ((pixel1 >> 16) & 0xff) / 255.0f; // r
+ color[1] = ((pixel1 >> 8) & 0xff) / 255.0f; // g
+ color[2] = ((pixel1 >> 0) & 0xff) / 255.0f; // b
+ color[3] = ((pixel1 >> 24) & 0xff) / 255.0f; // a
+ res = pd3dDevice->SetPixelShaderConstantF(1, color, 1);
+
+ color[0] = ((pixel2 >> 16) & 0xff) / 255.0f; // r
+ color[1] = ((pixel2 >> 8) & 0xff) / 255.0f; // g
+ color[2] = ((pixel2 >> 0) & 0xff) / 255.0f; // b
+ color[3] = ((pixel2 >> 24) & 0xff) / 255.0f; // a
+ res = pd3dDevice->SetPixelShaderConstantF(2, color, 1);
+
+ // set up texture coordinate transform with identity matrix, which
+ // will have the effect of passing the current window-space coordinates
+ // through to the TEXCOORD0/1 register used by the basic gradient
+ // pixel shader
+ DWORD sampler = useMask ? 1 : 0;
+ D3DMATRIX mt;
+ ZeroMemory(&mt, sizeof(mt));
+ mt._11 = 1.0f;
+ mt._21 = 0.0f;
+ mt._31 = 0.0f;
+ mt._41 = 0.0f;
+ mt._12 = 0.0f;
+ mt._22 = 1.0f;
+ mt._32 = 0.0f;
+ mt._42 = 0.0f;
+ pd3dDevice->SetTransform(useMask ? D3DTS_TEXTURE1 : D3DTS_TEXTURE0, &mt);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXCOORDINDEX,
+ D3DTSS_TCI_CAMERASPACEPOSITION);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_COUNT2);
+#endif
+
+ // pixel state has been set appropriately in D3DPaints_ResetPaint()
+ d3dc->useMask = useMask;
+ d3dc->SetPaintState(PAINT_GRADIENT);
+ return res;
+}
+
+/************************** TexturePaint support ****************************/
+
+HRESULT
+D3DPaints_SetTexturePaint(D3DContext *d3dc,
+ jboolean useMask,
+ jlong pSrcOps, jboolean filter,
+ jdouble xp0, jdouble xp1, jdouble xp3,
+ jdouble yp0, jdouble yp1, jdouble yp3)
+{
+ D3DSDOps *srcOps = (D3DSDOps *)jlong_to_ptr(pSrcOps);
+ IDirect3DDevice9 *pd3dDevice;
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPaints_SetTexturePaint");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(srcOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(srcOps->pResource, E_FAIL);
+ D3DPaints_ResetPaint(d3dc);
+
+ DWORD sampler = useMask ? 1 : 0;
+ DWORD dwFilter = filter ? D3DTEXF_LINEAR : D3DTEXF_POINT;
+ res = d3dc->SetTexture(srcOps->pResource->GetTexture(), sampler);
+ d3dc->UpdateTextureColorState(D3DTA_TEXTURE, sampler);
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_ADDRESSU, D3DTADDRESS_WRAP);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_ADDRESSV, D3DTADDRESS_WRAP);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_MAGFILTER, dwFilter);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_MINFILTER, dwFilter);
+
+ D3DMATRIX mt;
+ ZeroMemory(&mt, sizeof(mt));
+
+ // offset by a half texel to correctly map texels to pixels
+ // m02 = tx * m00 + ty * m01 + m02;
+ // m12 = tx * m10 + ty * m11 + m12;
+ jdouble tx = (1 / (2.0f * srcOps->pResource->GetDesc()->Width));
+ jdouble ty = (1 / (2.0f * srcOps->pResource->GetDesc()->Height));
+ xp3 = tx * xp0 + ty * xp1 + xp3;
+ yp3 = tx * yp0 + ty * yp1 + yp3;
+
+ mt._11 = (float)xp0;
+ mt._21 = (float)xp1;
+ mt._31 = (float)0.0;
+ mt._41 = (float)xp3;
+ mt._12 = (float)yp0;
+ mt._22 = (float)yp1;
+ mt._32 = (float)0.0;
+ mt._42 = (float)yp3;
+ pd3dDevice->SetTransform(useMask ? D3DTS_TEXTURE1 : D3DTS_TEXTURE0, &mt);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXCOORDINDEX,
+ D3DTSS_TCI_CAMERASPACEPOSITION);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_COUNT2);
+
+ // pixel state has been set appropriately in D3DPaints_ResetPaint()
+ d3dc->useMask = useMask;
+ d3dc->SetPaintState(PAINT_TEXTURE);
+ return res;
+}
+
+/****************** Shared MultipleGradientPaint support ********************/
+
+/** Composes the given parameters as flags into the given flags variable.*/
+#define COMPOSE_FLAGS(flags, cycleMethod, large, useMask, linear) \
+ do { \
+ flags |= ((cycleMethod) & MULTI_GRAD_CYCLE_METHOD); \
+ if (large) flags |= MULTI_GRAD_LARGE; \
+ if (useMask) flags |= MULTI_GRAD_USE_MASK; \
+ if (linear) flags |= MULTI_GRAD_LINEAR_RGB; \
+ } while (0)
+
+/**
+ * The maximum number of gradient "stops" supported by the fragment shader
+ * and related code. When the MULTI_GRAD_LARGE flag is set, we will use
+ * MAX_FRACTIONS_LARGE; otherwise, we use MAX_FRACTIONS_SMALL. By having
+ * two separate values, we can have one highly optimized shader (SMALL) that
+ * supports only a few fractions/colors, and then another, less optimal
+ * shader that supports more stops.
+ */
+#define MAX_FRACTIONS \
+ sun_java2d_d3d_D3DPaints_MultiGradient_MULTI_MAX_FRACTIONS_D3D
+#define MAX_FRACTIONS_LARGE MAX_FRACTIONS
+#define MAX_FRACTIONS_SMALL 4
+
+/**
+ * Called from the D3DPaints_SetLinear/RadialGradientPaint() methods
+ * in order to setup the fraction/color values that are common to both.
+ */
+static HRESULT
+D3DPaints_SetMultiGradientPaint(D3DContext *d3dc,
+ jboolean useMask, jint numStops,
+ void *pFractions, void *pPixels)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ IDirect3DTexture9 *pMultiGradientTex;
+ D3DResource *pMultiGradientTexRes;
+ jint maxFractions = (numStops > MAX_FRACTIONS_SMALL) ?
+ MAX_FRACTIONS_LARGE : MAX_FRACTIONS_SMALL;
+ jfloat stopVals[MAX_FRACTIONS * 4];
+ jfloat *fractions = (jfloat *)pFractions;
+ juint *pixels = (juint *)pPixels;
+ int i;
+ int fIndex = 0;
+
+ pd3dDevice = d3dc->Get3DDevice();
+
+ // update the "uniform" fractions and scale factors
+ for (i = 0; i < maxFractions; i++) {
+ stopVals[fIndex+0] = (i < numStops) ?
+ fractions[i] : 0.0f;
+ stopVals[fIndex+1] = (i < numStops-1) ?
+ 1.0f / (fractions[i+1] - fractions[i]) : 0.0f;
+ stopVals[fIndex+2] = 0.0f; // unused
+ stopVals[fIndex+3] = 0.0f; // unused
+ fIndex += 4;
+ }
+ pd3dDevice->SetPixelShaderConstantF(0, stopVals, maxFractions);
+
+ // this will initialize the multi-gradient texture, if necessary
+ res = d3dc->GetResourceManager()->
+ GetMultiGradientTexture(&pMultiGradientTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pMultiGradientTex = pMultiGradientTexRes->GetTexture();
+
+ // update the texture containing the gradient colors
+ D3DLOCKED_RECT lockedRect;
+ res = pMultiGradientTex->LockRect(0, &lockedRect, NULL, D3DLOCK_NOSYSLOCK);
+ RETURN_STATUS_IF_FAILED(res);
+
+ juint *pPix = (juint*)lockedRect.pBits;
+ memcpy(pPix, pixels, numStops*sizeof(juint));
+ if (numStops < MAX_MULTI_GRADIENT_COLORS) {
+ // when we don't have enough colors to fill the entire
+ // color gradient, we have to replicate the last color
+ // in the right-most texel for the NO_CYCLE case where the
+ // texcoord is sometimes forced to 1.0
+ pPix[MAX_MULTI_GRADIENT_COLORS-1] = pixels[numStops-1];
+ }
+ pMultiGradientTex->UnlockRect(0);
+
+ // set the gradient texture and update relevant state
+ DWORD sampler = useMask ? 1 : 0;
+ res = d3dc->SetTexture(pMultiGradientTex, sampler);
+ d3dc->UpdateTextureColorState(D3DTA_TEXTURE, sampler);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR);
+ pd3dDevice->SetSamplerState(sampler, D3DSAMP_MINFILTER, D3DTEXF_LINEAR);
+
+ // set up texture coordinate transform with identity matrix, which
+ // will have the effect of passing the current window-space coordinates
+ // through to the TEXCOORD0/1 register used by the multi-stop
+ // gradient pixel shader
+ D3DMATRIX mt;
+ ZeroMemory(&mt, sizeof(mt));
+ mt._11 = 1.0f;
+ mt._21 = 0.0f;
+ mt._31 = 0.0f;
+ mt._41 = 0.0f;
+ mt._12 = 0.0f;
+ mt._22 = 1.0f;
+ mt._32 = 0.0f;
+ mt._42 = 0.0f;
+ pd3dDevice->SetTransform(useMask ? D3DTS_TEXTURE1 : D3DTS_TEXTURE0, &mt);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXCOORDINDEX,
+ D3DTSS_TCI_CAMERASPACEPOSITION);
+ pd3dDevice->SetTextureStageState(sampler, D3DTSS_TEXTURETRANSFORMFLAGS,
+ D3DTTFF_COUNT2);
+ return res;
+}
+
+/********************** LinearGradientPaint support *************************/
+
+HRESULT
+D3DPaints_SetLinearGradientPaint(D3DContext *d3dc, D3DSDOps *dstOps,
+ jboolean useMask, jboolean linear,
+ jint cycleMethod, jint numStops,
+ jfloat p0, jfloat p1, jfloat p3,
+ void *fractions, void *pixels)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ jfloat params[4];
+ jboolean large = (numStops > MAX_FRACTIONS_SMALL);
+ jint flags = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPaints_SetLinearGradientPaint");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ D3DPaints_ResetPaint(d3dc);
+
+ COMPOSE_FLAGS(flags, cycleMethod, large, useMask, linear);
+
+ // locate/enable the shader program for the given flags
+ res = d3dc->EnableLinearGradientProgram(flags);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the common "uniform" values (fractions and colors)
+ D3DPaints_SetMultiGradientPaint(d3dc, useMask,
+ numStops, fractions, pixels);
+
+ // update the other "uniform" values
+ params[0] = p0;
+ params[1] = p1;
+ params[2] = p3;
+ params[3] = 0.0f; // unused
+ pd3dDevice = d3dc->Get3DDevice();
+ res = pd3dDevice->SetPixelShaderConstantF(16, params, 1);
+
+ // pixel state has been set appropriately in D3DPaints_ResetPaint()
+ d3dc->useMask = useMask;
+ d3dc->SetPaintState(PAINT_LIN_GRADIENT);
+ return res;
+}
+
+/********************** RadialGradientPaint support *************************/
+
+HRESULT
+D3DPaints_SetRadialGradientPaint(D3DContext *d3dc, D3DSDOps *dstOps,
+ jboolean useMask, jboolean linear,
+ jint cycleMethod, jint numStops,
+ jfloat m00, jfloat m01, jfloat m02,
+ jfloat m10, jfloat m11, jfloat m12,
+ jfloat focusX,
+ void *fractions, void *pixels)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ jfloat denom, inv_denom;
+ jfloat params[4];
+ jboolean large = (numStops > MAX_FRACTIONS_SMALL);
+ jint flags = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPaints_SetRadialGradientPaint");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ D3DPaints_ResetPaint(d3dc);
+
+ COMPOSE_FLAGS(flags, cycleMethod, large, useMask, linear);
+
+ // locate/enable the shader program for the given flags
+ res = d3dc->EnableRadialGradientProgram(flags);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the common "uniform" values (fractions and colors)
+ D3DPaints_SetMultiGradientPaint(d3dc, useMask,
+ numStops, fractions, pixels);
+
+ // update the other "uniform" values
+ params[0] = m00;
+ params[1] = m01;
+ params[2] = m02;
+ params[3] = 0.0f; // unused
+ pd3dDevice = d3dc->Get3DDevice();
+ pd3dDevice->SetPixelShaderConstantF(16, params, 1);
+
+ params[0] = m10;
+ params[1] = m11;
+ params[2] = m12;
+ params[3] = 0.0f; // unused
+ pd3dDevice->SetPixelShaderConstantF(17, params, 1);
+
+ // pack a few unrelated, precalculated values into a single float4
+ denom = 1.0f - (focusX * focusX);
+ inv_denom = 1.0f / denom;
+ params[0] = focusX;
+ params[1] = denom;
+ params[2] = inv_denom;
+ params[3] = 0.0f; // unused
+ res = pd3dDevice->SetPixelShaderConstantF(18, params, 1);
+
+ // pixel state has been set appropriately in D3DPaints_ResetPaint()
+ d3dc->useMask = useMask;
+ d3dc->SetPaintState(PAINT_RAD_GRADIENT);
+ return res;
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DPaints.h b/jdk/src/windows/native/sun/java2d/d3d/D3DPaints.h
new file mode 100644
index 00000000000..52ef19007b8
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DPaints.h
@@ -0,0 +1,132 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DPaints_h_Included
+#define D3DPaints_h_Included
+
+#include "sun_java2d_SunGraphics2D.h"
+
+#include "D3DContext.h"
+#include "D3DSurfaceData.h"
+
+HRESULT D3DPaints_ResetPaint(D3DContext *d3dc);
+HRESULT D3DPaints_SetColor(D3DContext *d3dc, jint pixel);
+
+/************************* GradientPaint support ****************************/
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define BASIC_GRAD_IS_CYCLIC (1 << 0)
+#define BASIC_GRAD_USE_MASK (1 << 1)
+
+HRESULT D3DPaints_SetGradientPaint(D3DContext *d3dc,
+ jboolean useMask, jboolean cyclic,
+ jdouble p0, jdouble p1, jdouble p3,
+ jint pixel1, jint pixel2);
+
+/************************** TexturePaint support ****************************/
+
+HRESULT D3DPaints_SetTexturePaint(D3DContext *d3dc,
+ jboolean useMask,
+ jlong pSrcOps, jboolean filter,
+ jdouble xp0, jdouble xp1, jdouble xp3,
+ jdouble yp0, jdouble yp1, jdouble yp3);
+
+/****************** Shared MultipleGradientPaint support ********************/
+
+/**
+ * These constants are identical to those defined in the
+ * MultipleGradientPaint.CycleMethod enum; they are copied here for
+ * convenience (ideally we would pull them directly from the Java level,
+ * but that entails more hassle than it is worth).
+ */
+#define CYCLE_NONE 0
+#define CYCLE_REFLECT 1
+#define CYCLE_REPEAT 2
+
+/**
+ * The following constants are flags that can be bitwise-or'ed together
+ * to control how the MultipleGradientPaint shader source code is generated:
+ *
+ * MULTI_GRAD_CYCLE_METHOD
+ * Placeholder for the CycleMethod enum constant.
+ *
+ * MULTI_GRAD_LARGE
+ * If set, use the (slower) shader that supports a larger number of
+ * gradient colors; otherwise, use the optimized codepath. See
+ * the MAX_FRACTIONS_SMALL/LARGE constants below for more details.
+ *
+ * MULTI_GRAD_USE_MASK
+ * If set, apply the alpha mask value from texture unit 1 to the
+ * final color result (only used in the MaskFill case).
+ *
+ * MULTI_GRAD_LINEAR_RGB
+ * If set, convert the linear RGB result back into the sRGB color space.
+ */
+#define MULTI_GRAD_CYCLE_METHOD (3 << 0)
+#define MULTI_GRAD_LARGE (1 << 2)
+#define MULTI_GRAD_USE_MASK (1 << 3)
+#define MULTI_GRAD_LINEAR_RGB (1 << 4)
+
+/**
+ * The maximum number of gradient colors supported by all of the gradient
+ * fragment shaders. Note that this value must be a power of two, as it
+ * determines the size of the 1D texture created below. It also must be
+ * greater than or equal to MAX_FRACTIONS (there is no strict requirement
+ * that the two values be equal).
+ */
+#define MAX_MULTI_GRADIENT_COLORS 16
+
+/********************** LinearGradientPaint support *************************/
+
+HRESULT D3DPaints_SetLinearGradientPaint(D3DContext *d3dc, D3DSDOps *dstOps,
+ jboolean useMask, jboolean linear,
+ jint cycleMethod, jint numStops,
+ jfloat p0, jfloat p1, jfloat p3,
+ void *fractions, void *pixels);
+
+/********************** RadialGradientPaint support *************************/
+
+HRESULT D3DPaints_SetRadialGradientPaint(D3DContext *d3dc, D3DSDOps *dstOps,
+ jboolean useMask, jboolean linear,
+ jint cycleMethod, jint numStops,
+ jfloat m00, jfloat m01, jfloat m02,
+ jfloat m10, jfloat m11, jfloat m12,
+ jfloat focusX,
+ void *fractions, void *pixels);
+
+/************************ SunGraphics2D constants ***************************/
+
+#define PAINT_CUSTOM sun_java2d_SunGraphics2D_PAINT_CUSTOM
+#define PAINT_TEXTURE sun_java2d_SunGraphics2D_PAINT_TEXTURE
+#define PAINT_RAD_GRADIENT sun_java2d_SunGraphics2D_PAINT_RAD_GRADIENT
+#define PAINT_LIN_GRADIENT sun_java2d_SunGraphics2D_PAINT_LIN_GRADIENT
+#define PAINT_GRADIENT sun_java2d_SunGraphics2D_PAINT_GRADIENT
+#define PAINT_ALPHACOLOR sun_java2d_SunGraphics2D_PAINT_ALPHACOLOR
+#define PAINT_OPAQUECOLOR sun_java2d_SunGraphics2D_PAINT_OPAQUECOLOR
+
+#endif /* D3DPaints_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DPipeline.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DPipeline.cpp
new file mode 100644
index 00000000000..e2a207b9990
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DPipeline.cpp
@@ -0,0 +1,40 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "D3DPipeline.h"
+
+BOOL APIENTRY DllMain( HANDLE hModule,
+ DWORD ul_reason_for_call,
+ LPVOID lpReserved)
+{
+ switch (ul_reason_for_call) {
+ case DLL_PROCESS_ATTACH:
+ case DLL_THREAD_ATTACH:
+ case DLL_THREAD_DETACH:
+ case DLL_PROCESS_DETACH:
+ break;
+ }
+ return TRUE;
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DPipeline.h b/jdk/src/windows/native/sun/java2d/d3d/D3DPipeline.h
new file mode 100644
index 00000000000..2ae8b7df295
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DPipeline.h
@@ -0,0 +1,125 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+#pragma once
+
+#ifdef DEBUG
+#define D3D_DEBUG_INFO
+#endif // DEBUG
+
+#ifdef D3D_PPL_DLL
+
+
+ #ifndef WIN32_LEAN_AND_MEAN
+ #define WIN32_LEAN_AND_MEAN
+ #endif
+
+ #ifdef D3DPIPELINE_EXPORTS
+ #define D3DPIPELINE_API __declspec(dllexport)
+ #else
+ #define D3DPIPELINE_API __declspec(dllimport)
+ #endif
+
+ #include
+ #include
+ #include
+ #include "..\Import\Trace.h"
+
+ #define DebugPrintD3DError(res, msg) \
+ DXTRACE_ERR(msg, res)
+
+#else
+
+ #define D3DPIPELINE_API __declspec(dllexport)
+
+ // this include ensures that with debug build we get
+ // awt's overridden debug "new" and "delete" operators
+ #include "awt.h"
+
+ #include
+ #include
+ #include "Trace.h"
+
+ #define DebugPrintD3DError(res, msg) \
+ J2dTraceLn1(J2D_TRACE_ERROR, "D3D Error: " ## msg ## " res=%d", res)
+
+#endif /*D3D_PPL_DLL*/
+
+// some helper macros
+#define SAFE_RELEASE(RES) \
+do { \
+ if ((RES)!= NULL) { \
+ (RES)->Release(); \
+ (RES) = NULL; \
+ } \
+} while (0);
+
+#define SAFE_DELETE(RES) \
+do { \
+ if ((RES)!= NULL) { \
+ delete (RES); \
+ (RES) = NULL; \
+ } \
+} while (0);
+
+#ifdef DEBUG
+#define SAFE_PRINTLN(RES) \
+do { \
+ if ((RES)!= NULL) { \
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " " ## #RES ## "=0x%x", (RES)); \
+ } else { \
+ J2dTraceLn(J2D_TRACE_VERBOSE, " " ## #RES ## "=NULL"); \
+ } \
+} while (0);
+#else // DEBUG
+#define SAFE_PRINTLN(RES)
+#endif // DEBUG
+
+/*
+ * The following macros allow the caller to return (or continue) if the
+ * provided value is NULL. (The strange else clause is included below to
+ * allow for a trailing ';' after RETURN/CONTINUE_IF_NULL() invocations.)
+ */
+#define ACT_IF_NULL(ACTION, value) \
+ if ((value) == NULL) { \
+ J2dTraceLn3(J2D_TRACE_ERROR, \
+ "%s is null in %s:%d", #value, __FILE__, __LINE__); \
+ ACTION; \
+ } else do { } while (0)
+#define RETURN_IF_NULL(value) ACT_IF_NULL(return, value)
+#define CONTINUE_IF_NULL(value) ACT_IF_NULL(continue, value)
+#define RETURN_STATUS_IF_NULL(value, status) \
+ ACT_IF_NULL(return (status), value)
+
+#define RETURN_STATUS_IF_EXP_FAILED(EXPR) \
+ if (FAILED(res = (EXPR))) { \
+ DebugPrintD3DError(res, " " ## #EXPR ## " failed in " ## __FILE__); \
+ return res; \
+ } else do { } while (0)
+
+#define RETURN_STATUS_IF_FAILED(status) \
+ if (FAILED((status))) { \
+ DebugPrintD3DError((status), " failed in " ## __FILE__ ## ", return;");\
+ return (status); \
+ } else do { } while (0)
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DPipelineManager.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DPipelineManager.cpp
new file mode 100644
index 00000000000..f0838dd5625
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DPipelineManager.cpp
@@ -0,0 +1,936 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "D3DBadHardware.h"
+#include "D3DPipelineManager.h"
+#include "D3DRenderQueue.h"
+#include "WindowsFlags.h"
+#include "awt_Win32GraphicsDevice.h"
+
+// state of the adapter prior to initialization
+#define CONTEXT_NOT_INITED 0
+// this state is set if adapter initialization had failed
+#define CONTEXT_INIT_FAILED (-1)
+// this state is set if adapter was successfully created
+#define CONTEXT_CREATED 1
+
+static BOOL bNoHwCheck = (getenv("J2D_D3D_NO_HWCHECK") != NULL);
+
+D3DPipelineManager *D3DPipelineManager::pMgr = NULL;
+
+D3DPipelineManager * D3DPipelineManager::CreateInstance(void)
+{
+ if (!IsD3DEnabled() ||
+ FAILED((D3DPipelineManager::CheckOSVersion())) ||
+ FAILED((D3DPipelineManager::GDICheckForBadHardware())))
+ {
+ return NULL;
+ }
+
+ if (pMgr == NULL) {
+ pMgr = new D3DPipelineManager();
+ if (FAILED(pMgr->InitD3D())) {
+ SAFE_DELETE(pMgr);
+ }
+ } else {
+ // this should never happen so to be on the safe side do not
+ // use this unexpected pointer, do not try to release it, just null
+ // it out and fail safely
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::CreateInstance: unexpected instance: 0x%x,"\
+ " abort.", pMgr);
+ pMgr = NULL;
+ }
+ return pMgr;
+}
+
+void D3DPipelineManager::DeleteInstance()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::DeleteInstance()");
+ SAFE_DELETE(pMgr);
+}
+
+D3DPipelineManager * D3DPipelineManager::GetInstance(void)
+{
+ return pMgr;
+}
+
+D3DPipelineManager::D3DPipelineManager(void)
+{
+ pd3d9 = NULL;
+ hLibD3D9 = NULL;
+ pAdapters = NULL;
+ adapterCount = 0;
+ currentFSFocusAdapter = -1;
+ defaultFocusWindow = 0;
+ devType = SelectDeviceType();
+}
+
+D3DPipelineManager::~D3DPipelineManager(void)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::~D3DPipelineManager()");
+ ReleaseD3D();
+}
+
+HRESULT D3DPipelineManager::ReleaseD3D(void)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::ReleaseD3D()");
+
+ ReleaseAdapters();
+
+ SAFE_RELEASE(pd3d9);
+
+ if (hLibD3D9 != NULL) {
+ ::FreeLibrary(hLibD3D9);
+ hLibD3D9 = NULL;
+ }
+
+ return S_OK;
+}
+
+// Creates a Direct3D9 object and initializes adapters.
+// If succeeded, returns S_OK, otherwise returns the error code.
+HRESULT D3DPipelineManager::InitD3D(void)
+{
+ typedef IDirect3D9 * WINAPI FnDirect3DCreate9(UINT SDKVersion);
+
+ hLibD3D9 = ::LoadLibrary(TEXT("d3d9.dll"));
+ if (hLibD3D9 == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR, "InitD3D: no d3d9.dll");
+ return E_FAIL;
+ }
+
+ FnDirect3DCreate9 *d3dcreate9 = NULL;
+ d3dcreate9 = (FnDirect3DCreate9*)
+ ::GetProcAddress(hLibD3D9, "Direct3DCreate9");
+ if (d3dcreate9 == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR, "InitD3D: no Direct3DCreate9");
+ ::FreeLibrary(hLibD3D9);
+ return E_FAIL;
+ }
+
+ pd3d9 = d3dcreate9(D3D_SDK_VERSION);
+ if (pd3d9 == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "InitD3D: unable to create IDirect3D9 object");
+ ::FreeLibrary(hLibD3D9);
+ return E_FAIL;
+ }
+
+ HRESULT res;
+ if (FAILED(res = InitAdapters())) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR, "InitD3D: failed to init adapters");
+ ReleaseD3D();
+ return res;
+ }
+
+ return S_OK;
+}
+
+HRESULT D3DPipelineManager::ReleaseAdapters()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::ReleaseAdapters()");
+
+ D3DRQ_ResetCurrentContextAndDestination();
+ if (pAdapters != NULL) {
+ for (UINT i = 0; i < adapterCount; i++) {
+ if (pAdapters[i].pd3dContext != NULL) {
+ delete pAdapters[i].pd3dContext;
+ }
+ }
+ delete[] pAdapters;
+ pAdapters = NULL;
+ }
+ if (defaultFocusWindow != 0) {
+ DestroyWindow(defaultFocusWindow);
+ UnregisterClass(L"D3DFocusWindow", GetModuleHandle(NULL));
+ defaultFocusWindow = 0;
+ }
+ currentFSFocusAdapter = -1;
+ return S_OK;
+}
+
+// static
+void D3DPipelineManager::NotifyAdapterEventListeners(UINT adapter,
+ jint eventType)
+{
+ HMONITOR hMon;
+ int gdiScreen;
+ D3DPipelineManager *pMgr;
+ JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
+
+ pMgr = D3DPipelineManager::GetInstance();
+ RETURN_IF_NULL(pMgr);
+ hMon = pMgr->pd3d9->GetAdapterMonitor(adapter);
+ gdiScreen = AwtWin32GraphicsDevice::GetScreenFromMHND((MHND)hMon);
+
+ JNU_CallStaticMethodByName(env, NULL,
+ "sun/java2d/pipe/hw/AccelDeviceEventNotifier",
+ "eventOccured", "(II)V",
+ gdiScreen, eventType);
+}
+
+UINT D3DPipelineManager::GetAdapterOrdinalForScreen(jint gdiScreen)
+{
+ MHND mHnd = AwtWin32GraphicsDevice::GetMonitor(gdiScreen);
+ if (mHnd == (MHND)0) {
+ return D3DADAPTER_DEFAULT;
+ }
+ return GetAdapterOrdinalByHmon((HMONITOR)mHnd);
+}
+
+// static
+HRESULT D3DPipelineManager::HandleAdaptersChange(HMONITOR *pMHNDs, UINT monNum)
+{
+ HRESULT res = S_OK;
+ BOOL bResetD3D = FALSE, bFound;
+
+ D3DPipelineManager *pMgr = D3DPipelineManager::GetInstance();
+ RETURN_STATUS_IF_NULL(pMHNDs, E_FAIL);
+ if (pMgr == NULL) {
+ // NULL pMgr is valid when the pipeline is not enabled or if it hasn't
+ // been created yet
+ return S_OK;
+ }
+ RETURN_STATUS_IF_NULL(pMgr->pAdapters, E_FAIL);
+ RETURN_STATUS_IF_NULL(pMgr->pd3d9, E_FAIL);
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::HandleAdaptersChange");
+
+ if (monNum != pMgr->adapterCount) {
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ " number of adapters changed (old=%d, new=%d)",
+ pMgr->adapterCount, monNum);
+ bResetD3D = TRUE;
+ } else {
+ for (UINT i = 0; i < pMgr->adapterCount; i++) {
+ HMONITOR hMon = pMgr->pd3d9->GetAdapterMonitor(i);
+ if (hMon == (HMONITOR)0x0) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " adapter %d: removed", i);
+ bResetD3D = TRUE;
+ break;
+ }
+ bFound = FALSE;
+ for (UINT mon = 0; mon < monNum; mon++) {
+ if (pMHNDs[mon] == hMon) {
+ J2dTraceLn3(J2D_TRACE_VERBOSE,
+ " adapter %d: found hmnd[%d]=0x%x", i, mon, hMon);
+ bFound = TRUE;
+ break;
+ }
+ }
+ if (!bFound) {
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ " adapter %d: could not find hmnd=0x%x "\
+ "in the list of new hmnds", i, hMon);
+ bResetD3D = TRUE;
+ break;
+ }
+ }
+ }
+
+ if (bResetD3D) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " adapters changed: resetting d3d");
+ pMgr->ReleaseD3D();
+ res = pMgr->InitD3D();
+ }
+ return res;
+}
+
+HRESULT D3DPipelineManager::HandleLostDevices()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::HandleLostDevices()");
+ BOOL bAllClear = TRUE;
+
+ HWND hwnd = GetCurrentFocusWindow();
+ if (hwnd != defaultFocusWindow) {
+ // we're in full-screen mode
+ WINDOWPLACEMENT wp;
+ ::ZeroMemory(&wp, sizeof(WINDOWPLACEMENT));
+ wp.length = sizeof(WINDOWPLACEMENT);
+ ::GetWindowPlacement(hwnd, &wp);
+
+ // Only attempt to restore the devices if we're in full-screen mode
+ // and the fs window is active; sleep otherwise.
+ // Restoring a window while minimized causes problems on Vista:
+ // sometimes we restore the window too quickly and it pops up back from
+ // minimized state when the device is restored.
+ //
+ // WARNING: this is a sleep on the Toolkit thread! We may reconsider
+ // this if we find any issues later.
+ if ((wp.showCmd & SW_SHOWMINNOACTIVE) && !(wp.showCmd & SW_SHOWNORMAL)){
+ static DWORD prevCallTime = 0;
+ J2dTraceLn(J2D_TRACE_VERBOSE, " fs focus window is minimized");
+ DWORD currentTime = ::GetTickCount();
+ if ((currentTime - prevCallTime) < 100) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " tight loop detected, sleep");
+ ::Sleep(100);
+ }
+ prevCallTime = currentTime;
+ return D3DERR_DEVICELOST;
+ }
+ }
+ if (pAdapters != NULL) {
+ for (UINT i = 0; i < adapterCount; i++) {
+ if (pAdapters[i].pd3dContext != NULL) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ " HandleLostDevices: checking adapter %d", i);
+ D3DContext *d3dc = pAdapters[i].pd3dContext;
+ if (FAILED(d3dc->CheckAndResetDevice())) {
+ bAllClear = FALSE;
+ }
+ }
+ }
+ }
+ return bAllClear ? S_OK : D3DERR_DEVICELOST;
+}
+
+HRESULT D3DPipelineManager::InitAdapters()
+{
+ HRESULT res = E_FAIL;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::InitAdapters()");
+ if (pAdapters != NULL) {
+ ReleaseAdapters();
+ }
+
+ adapterCount = pd3d9->GetAdapterCount();
+ pAdapters = new D3DAdapter[adapterCount];
+ if (pAdapters == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR, "InitAdapters: out of memory");
+ adapterCount = 0;
+ return E_FAIL;
+ }
+ ZeroMemory(pAdapters, adapterCount * sizeof(D3DAdapter));
+
+ res = CheckAdaptersInfo();
+ RETURN_STATUS_IF_FAILED(res);
+
+ currentFSFocusAdapter = -1;
+ if (CreateDefaultFocusWindow() == 0) {
+ return E_FAIL;
+ }
+
+ return S_OK;
+}
+
+// static
+HRESULT
+D3DPipelineManager::CheckOSVersion()
+{
+ // require Windows XP or newer client-class OS
+ if (IS_WINVER_ATLEAST(5, 1) &&
+ !D3DPPLM_OsVersionMatches(OS_WINSERV_2008|OS_WINSERV_2003))
+ {
+ J2dTraceLn(J2D_TRACE_INFO,
+ "D3DPPLM::CheckOSVersion: Windows XP or newer client-classs"\
+ " OS detected, passed");
+ return S_OK;
+ }
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DPPLM::CheckOSVersion: Windows 2000 or earlier (or a "\
+ "server) OS detected, failed");
+ if (bNoHwCheck) {
+ J2dRlsTraceLn(J2D_TRACE_WARNING,
+ " OS check overridden via J2D_D3D_NO_HWCHECK");
+ return S_OK;
+ }
+ return E_FAIL;
+}
+
+// static
+HRESULT
+D3DPipelineManager::GDICheckForBadHardware()
+{
+ _DISPLAY_DEVICE dd;
+ dd.dwSize = sizeof(DISPLAY_DEVICE);
+
+ int failedDevices = 0;
+ int attachedDevices = 0;
+ int i = 0;
+ WCHAR *id;
+ WCHAR vendorId[5];
+ WCHAR deviceId[5];
+ DWORD dwDId, dwVId;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::GDICheckForBadHardware");
+
+ // i<20 is to guard against buggy drivers
+ while (EnumDisplayDevices(NULL, i, &dd, 0) && i < 20) {
+ if (dd.dwFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP) {
+ attachedDevices++;
+ id = dd.deviceID;
+ if (wcslen(id) > 21) {
+ // get vendor ID
+ wcsncpy(vendorId, id+8, 4);
+ int args1 = swscanf(vendorId, L"%X", &dwVId);
+
+ // get device ID
+ wcsncpy(deviceId, id+17, 4);
+ int args2 = swscanf(deviceId, L"%X", &dwDId);
+
+ if (args1 == 1 && args2 == 1) {
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ " device: vendorID=0x%04x, deviceId=0x%04x",
+ dwVId, dwDId);
+ // since we don't have a driver version here we will
+ // just ask to ignore the version for now; bad hw
+ // entries with specific drivers information will be
+ // processed later when d3d is initialized and we can
+ // obtain a driver version
+ if (FAILED(CheckForBadHardware(dwVId, dwDId, MAX_VERSION))){
+ failedDevices++;
+ }
+ }
+ }
+ }
+
+ i++;
+ }
+
+ if (failedDevices == attachedDevices) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DPPLM::GDICheckForBadHardware: no suitable devices found");
+ return E_FAIL;
+ }
+
+ return S_OK;
+}
+
+BOOL D3DPPLM_OsVersionMatches(USHORT osInfo) {
+ static USHORT currentOS = OS_UNDEFINED;
+
+ if (currentOS == OS_UNDEFINED) {
+ BOOL bVersOk;
+ OSVERSIONINFOEX osvi;
+
+ ZeroMemory(&osvi, sizeof(OSVERSIONINFOEX));
+ osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
+
+ bVersOk = GetVersionEx((OSVERSIONINFO *) &osvi);
+
+ J2dRlsTrace(J2D_TRACE_INFO, "[I] OS Version = ");
+ if (bVersOk && osvi.dwPlatformId == VER_PLATFORM_WIN32_NT &&
+ osvi.dwMajorVersion > 4)
+ {
+ if (osvi.dwMajorVersion >= 6 && osvi.dwMinorVersion >= 0) {
+ if (osvi.wProductType == VER_NT_WORKSTATION) {
+ J2dRlsTrace(J2D_TRACE_INFO, "OS_VISTA or newer\n");
+ currentOS = OS_VISTA;
+ } else {
+ J2dRlsTrace(J2D_TRACE_INFO, "OS_WINSERV_2008 or newer\n");
+ currentOS = OS_WINSERV_2008;
+ }
+ } else if (osvi.dwMajorVersion == 5 && osvi.dwMinorVersion == 2) {
+ if (osvi.wProductType == VER_NT_WORKSTATION) {
+ J2dRlsTrace(J2D_TRACE_INFO, "OS_WINXP_64\n");
+ currentOS = OS_WINXP_64;
+ } else {
+ J2dRlsTrace(J2D_TRACE_INFO, "OS_WINSERV_2003\n");
+ currentOS = OS_WINSERV_2003;
+ }
+ } else if (osvi.dwMajorVersion == 5 && osvi.dwMinorVersion == 1) {
+ J2dRlsTrace(J2D_TRACE_INFO, "OS_WINXP ");
+ currentOS = OS_WINXP;
+ if (osvi.wSuiteMask & VER_SUITE_PERSONAL) {
+ J2dRlsTrace(J2D_TRACE_INFO, "Home\n");
+ } else {
+ J2dRlsTrace(J2D_TRACE_INFO, "Pro\n");
+ }
+ } else {
+ J2dRlsTrace2(J2D_TRACE_INFO,
+ "OS_UNKNOWN: dwMajorVersion=%d dwMinorVersion=%d\n",
+ osvi.dwMajorVersion, osvi.dwMinorVersion);
+ currentOS = OS_UNKNOWN;
+ }
+ } else {
+ if (bVersOk) {
+ J2dRlsTrace2(J2D_TRACE_INFO,
+ "OS_UNKNOWN: dwPlatformId=%d dwMajorVersion=%d\n",
+ osvi.dwPlatformId, osvi.dwMajorVersion);
+ } else {
+ J2dRlsTrace(J2D_TRACE_INFO,"OS_UNKNOWN: GetVersionEx failed\n");
+ }
+ currentOS = OS_UNKNOWN;
+ }
+ }
+ return (currentOS & osInfo);
+}
+
+// static
+HRESULT
+D3DPipelineManager::CheckForBadHardware(DWORD vId, DWORD dId, LONGLONG version)
+{
+ DWORD vendorId, deviceId;
+ UINT adapterInfo = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::CheckForBadHardware");
+
+ while ((vendorId = badHardware[adapterInfo].VendorId) != 0x0000 &&
+ (deviceId = badHardware[adapterInfo].DeviceId) != 0x0000)
+ {
+ if (vendorId == vId && (deviceId == dId || deviceId == ALL_DEVICEIDS)) {
+ LONGLONG goodVersion = badHardware[adapterInfo].DriverVersion;
+ USHORT osInfo = badHardware[adapterInfo].OsInfo;
+ // the hardware check fails if:
+ // - we have an entry for this OS and
+ // - hardware is bad for all driver versions (NO_VERSION), or
+ // we have a driver version which is older than the
+ // minimum required for this OS
+ if (D3DPPLM_OsVersionMatches(osInfo) &&
+ (goodVersion == NO_VERSION || version < goodVersion))
+ {
+ J2dRlsTraceLn2(J2D_TRACE_ERROR,
+ "D3DPPLM::CheckForBadHardware: found matching "\
+ "hardware: VendorId=0x%04x DeviceId=0x%04x",
+ vendorId, deviceId);
+ if (goodVersion != NO_VERSION) {
+ // this was a match by the driver version
+ LARGE_INTEGER li;
+ li.QuadPart = goodVersion;
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ " bad driver found, device disabled");
+ J2dRlsTraceLn4(J2D_TRACE_ERROR,
+ " update your driver to at "\
+ "least version %d.%d.%d.%d",
+ HIWORD(li.HighPart), LOWORD(li.HighPart),
+ HIWORD(li.LowPart), LOWORD(li.LowPart));
+ } else {
+ // this was a match by the device (no good driver for this
+ // device)
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DPPLM::CheckForBadHardware: bad hardware "\
+ "found, device disabled");
+ }
+ if (!bNoHwCheck) {
+ return D3DERR_INVALIDDEVICE;
+ }
+ J2dRlsTraceLn(J2D_TRACE_WARNING, " Warning: hw/driver match "\
+ "overridden (via J2D_D3D_NO_HWCHECK)");
+ }
+ }
+ adapterInfo++;
+ }
+
+ return S_OK;
+}
+
+HRESULT D3DPipelineManager::CheckAdaptersInfo()
+{
+ D3DADAPTER_IDENTIFIER9 aid;
+ UINT failedAdaptersCount = 0;
+
+ J2dRlsTraceLn(J2D_TRACE_INFO, "CheckAdaptersInfo");
+ J2dRlsTraceLn(J2D_TRACE_INFO, "------------------");
+ for (UINT Adapter = 0; Adapter < adapterCount; Adapter++) {
+
+ if (FAILED(pd3d9->GetAdapterIdentifier(Adapter, 0, &aid))) {
+ pAdapters[Adapter].state = CONTEXT_INIT_FAILED;
+ failedAdaptersCount++;
+ continue;
+ }
+
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "Adapter Ordinal : %d", Adapter);
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "Adapter Handle : 0x%x",
+ pd3d9->GetAdapterMonitor(Adapter));
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "Description : %s",
+ aid.Description);
+ J2dRlsTraceLn2(J2D_TRACE_INFO, "GDI Name, Driver : %s, %s",
+ aid.DeviceName, aid.Driver);
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "Vendor Id : 0x%04x",
+ aid.VendorId);
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "Device Id : 0x%04x",
+ aid.DeviceId);
+ J2dRlsTraceLn1(J2D_TRACE_INFO, "SubSys Id : 0x%x",
+ aid.SubSysId);
+ J2dRlsTraceLn4(J2D_TRACE_INFO, "Driver Version : %d.%d.%d.%d",
+ HIWORD(aid.DriverVersion.HighPart),
+ LOWORD(aid.DriverVersion.HighPart),
+ HIWORD(aid.DriverVersion.LowPart),
+ LOWORD(aid.DriverVersion.LowPart));
+ J2dRlsTrace3(J2D_TRACE_INFO,
+ "[I] GUID : {%08X-%04X-%04X-",
+ aid.DeviceIdentifier.Data1,
+ aid.DeviceIdentifier.Data2,
+ aid.DeviceIdentifier.Data3);
+ J2dRlsTrace4(J2D_TRACE_INFO, "%02X%02X-%02X%02X",
+ aid.DeviceIdentifier.Data4[0],
+ aid.DeviceIdentifier.Data4[1],
+ aid.DeviceIdentifier.Data4[2],
+ aid.DeviceIdentifier.Data4[3]);
+ J2dRlsTrace4(J2D_TRACE_INFO, "%02X%02X%02X%02X}\n",
+ aid.DeviceIdentifier.Data4[4],
+ aid.DeviceIdentifier.Data4[5],
+ aid.DeviceIdentifier.Data4[6],
+ aid.DeviceIdentifier.Data4[7]);
+
+ if (FAILED(CheckForBadHardware(aid.VendorId, aid.DeviceId,
+ aid.DriverVersion.QuadPart)) ||
+ FAILED(CheckDeviceCaps(Adapter)) ||
+ FAILED(D3DEnabledOnAdapter(Adapter)))
+ {
+ pAdapters[Adapter].state = CONTEXT_INIT_FAILED;
+ failedAdaptersCount++;
+ }
+ J2dRlsTraceLn(J2D_TRACE_INFO, "------------------");
+ }
+
+ if (failedAdaptersCount == adapterCount) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DPPLM::CheckAdaptersInfo: no suitable adapters found");
+ return E_FAIL;
+ }
+
+ return S_OK;
+}
+
+D3DDEVTYPE D3DPipelineManager::SelectDeviceType()
+{
+ char *pRas = getenv("J2D_D3D_RASTERIZER");
+ D3DDEVTYPE dtype = D3DDEVTYPE_HAL;
+ if (pRas != NULL) {
+ J2dRlsTrace(J2D_TRACE_WARNING, "[W] D3DPPLM::SelectDeviceType: ");
+ if (strncmp(pRas, "ref", 3) == 0 || strncmp(pRas, "rgb", 3) == 0) {
+ J2dRlsTrace(J2D_TRACE_WARNING, "ref rasterizer selected");
+ dtype = D3DDEVTYPE_REF;
+ } else if (strncmp(pRas, "hal",3) == 0 || strncmp(pRas, "tnl",3) == 0) {
+ J2dRlsTrace(J2D_TRACE_WARNING, "hal rasterizer selected");
+ dtype = D3DDEVTYPE_HAL;
+ } else if (strncmp(pRas, "nul", 3) == 0) {
+ J2dRlsTrace(J2D_TRACE_WARNING, "nullref rasterizer selected");
+ dtype = D3DDEVTYPE_NULLREF;
+ } else {
+ J2dRlsTrace1(J2D_TRACE_WARNING,
+ "unknown rasterizer: %s, only (ref|hal|nul) "\
+ "supported, hal selected instead", pRas);
+ }
+ J2dRlsTrace(J2D_TRACE_WARNING, "\n");
+ }
+ return dtype;
+}
+
+#define CHECK_CAP(FLAG, CAP) \
+ do { \
+ if (!((FLAG)&CAP)) { \
+ J2dRlsTraceLn2(J2D_TRACE_ERROR, \
+ "D3DPPLM::CheckDeviceCaps: adapter %d: Failed "\
+ "(cap %s not supported)", \
+ adapter, #CAP); \
+ return E_FAIL; \
+ } \
+ } while (0)
+
+HRESULT D3DPipelineManager::CheckDeviceCaps(UINT adapter)
+{
+ HRESULT res;
+ D3DCAPS9 d3dCaps;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::CheckDeviceCaps");
+
+ res = pd3d9->GetDeviceCaps(adapter, devType, &d3dCaps);
+ RETURN_STATUS_IF_FAILED(res);
+
+ CHECK_CAP(d3dCaps.DevCaps, D3DDEVCAPS_DRAWPRIMTLVERTEX);
+
+ // by requiring hardware tnl we are hoping for better drivers quality
+ if (!IsD3DForced()) {
+ // fail if not hw tnl unless d3d was forced
+ CHECK_CAP(d3dCaps.DevCaps, D3DDEVCAPS_HWTRANSFORMANDLIGHT);
+ }
+ if (d3dCaps.DeviceType == D3DDEVTYPE_HAL) {
+ CHECK_CAP(d3dCaps.DevCaps, D3DDEVCAPS_HWRASTERIZATION);
+ }
+
+ CHECK_CAP(d3dCaps.RasterCaps, D3DPRASTERCAPS_SCISSORTEST);
+
+ CHECK_CAP(d3dCaps.Caps3, D3DCAPS3_ALPHA_FULLSCREEN_FLIP_OR_DISCARD);
+
+ CHECK_CAP(d3dCaps.PrimitiveMiscCaps, D3DPMISCCAPS_CULLNONE);
+ CHECK_CAP(d3dCaps.PrimitiveMiscCaps, D3DPMISCCAPS_BLENDOP);
+ CHECK_CAP(d3dCaps.PrimitiveMiscCaps, D3DPMISCCAPS_MASKZ);
+
+ CHECK_CAP(d3dCaps.ZCmpCaps, D3DPCMPCAPS_ALWAYS);
+ CHECK_CAP(d3dCaps.ZCmpCaps, D3DPCMPCAPS_LESS);
+
+ CHECK_CAP(d3dCaps.SrcBlendCaps, D3DPBLENDCAPS_ZERO);
+ CHECK_CAP(d3dCaps.SrcBlendCaps, D3DPBLENDCAPS_ONE);
+ CHECK_CAP(d3dCaps.SrcBlendCaps, D3DPBLENDCAPS_SRCALPHA);
+ CHECK_CAP(d3dCaps.SrcBlendCaps, D3DPBLENDCAPS_DESTALPHA);
+ CHECK_CAP(d3dCaps.SrcBlendCaps, D3DPBLENDCAPS_INVSRCALPHA);
+ CHECK_CAP(d3dCaps.SrcBlendCaps, D3DPBLENDCAPS_INVDESTALPHA);
+
+ CHECK_CAP(d3dCaps.DestBlendCaps, D3DPBLENDCAPS_ZERO);
+ CHECK_CAP(d3dCaps.DestBlendCaps, D3DPBLENDCAPS_ONE);
+ CHECK_CAP(d3dCaps.DestBlendCaps, D3DPBLENDCAPS_SRCALPHA);
+ CHECK_CAP(d3dCaps.DestBlendCaps, D3DPBLENDCAPS_DESTALPHA);
+ CHECK_CAP(d3dCaps.DestBlendCaps, D3DPBLENDCAPS_INVSRCALPHA);
+ CHECK_CAP(d3dCaps.DestBlendCaps, D3DPBLENDCAPS_INVDESTALPHA);
+
+ CHECK_CAP(d3dCaps.TextureAddressCaps, D3DPTADDRESSCAPS_CLAMP);
+ CHECK_CAP(d3dCaps.TextureAddressCaps, D3DPTADDRESSCAPS_WRAP);
+
+ CHECK_CAP(d3dCaps.TextureOpCaps, D3DTEXOPCAPS_MODULATE);
+
+ if (d3dCaps.PixelShaderVersion < D3DPS_VERSION(2,0) && !IsD3DForced()) {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::CheckDeviceCaps: adapter %d: Failed "\
+ "(pixel shaders 2.0 required)", adapter);
+ return E_FAIL;
+ }
+
+ J2dRlsTraceLn1(J2D_TRACE_INFO,
+ "D3DPPLM::CheckDeviceCaps: adapter %d: Passed", adapter);
+ return S_OK;
+}
+
+
+HRESULT D3DPipelineManager::D3DEnabledOnAdapter(UINT adapter)
+{
+ HRESULT res;
+ D3DDISPLAYMODE dm;
+
+ res = pd3d9->GetAdapterDisplayMode(adapter, &dm);
+ RETURN_STATUS_IF_FAILED(res);
+
+ res = pd3d9->CheckDeviceType(adapter, devType, dm.Format, dm.Format, TRUE);
+ if (FAILED(res)) {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::D3DEnabledOnAdapter: no " \
+ "suitable d3d device on adapter %d", adapter);
+ }
+
+ return res;
+}
+
+UINT D3DPipelineManager::GetAdapterOrdinalByHmon(HMONITOR hMon)
+{
+ UINT ret = D3DADAPTER_DEFAULT;
+
+ if (pd3d9 != NULL) {
+ UINT adapterCount = pd3d9->GetAdapterCount();
+ for (UINT adapter = 0; adapter < adapterCount; adapter++) {
+ HMONITOR hm = pd3d9->GetAdapterMonitor(adapter);
+ if (hm == hMon) {
+ ret = adapter;
+ break;
+ }
+ }
+ }
+ return ret;
+}
+
+D3DFORMAT
+D3DPipelineManager::GetMatchingDepthStencilFormat(UINT adapterOrdinal,
+ D3DFORMAT adapterFormat,
+ D3DFORMAT renderTargetFormat)
+{
+ static D3DFORMAT formats[] =
+ { D3DFMT_D16, D3DFMT_D32, D3DFMT_D24S8, D3DFMT_D24X8 };
+ D3DFORMAT newFormat = D3DFMT_UNKNOWN;
+ HRESULT res;
+ for (int i = 0; i < 4; i++) {
+ res = pd3d9->CheckDeviceFormat(adapterOrdinal,
+ devType, adapterFormat, D3DUSAGE_DEPTHSTENCIL,
+ D3DRTYPE_SURFACE, formats[i]);
+ if (FAILED(res)) continue;
+
+ res = pd3d9->CheckDepthStencilMatch(adapterOrdinal,
+ devType, adapterFormat, renderTargetFormat, formats[i]);
+ if (FAILED(res)) continue;
+ newFormat = formats[i];
+ break;
+ }
+ return newFormat;
+}
+
+HWND D3DPipelineManager::CreateDefaultFocusWindow()
+{
+ UINT adapterOrdinal = D3DADAPTER_DEFAULT;
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DPPLM::CreateDefaultFocusWindow: adapter=%d",
+ adapterOrdinal);
+
+ if (defaultFocusWindow != 0) {
+ J2dRlsTraceLn(J2D_TRACE_WARNING,
+ "D3DPPLM::CreateDefaultFocusWindow: "\
+ "existing default focus window!");
+ return defaultFocusWindow;
+ }
+
+ WNDCLASS wc;
+ ZeroMemory(&wc, sizeof(WNDCLASS));
+ wc.hInstance = GetModuleHandle(NULL);
+ wc.lpfnWndProc = DefWindowProc;
+ wc.lpszClassName = L"D3DFocusWindow";
+ if (RegisterClass(&wc) == 0) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DPPLM::CreateDefaultFocusWindow: "\
+ "error registering window class");
+ return 0;
+ }
+
+ MONITORINFO mi;
+ ZeroMemory(&mi, sizeof(MONITORINFO));
+ mi.cbSize = sizeof(MONITORINFO);
+ HMONITOR hMon = pd3d9->GetAdapterMonitor(adapterOrdinal);
+ if (hMon == 0 || !GetMonitorInfo(hMon, (PMONITOR_INFO)&mi)) {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::CreateDefaultFocusWindow: "\
+ "error getting monitor info for adapter=%d", adapterOrdinal);
+ return 0;
+ }
+
+ HWND hWnd = CreateWindow(L"D3DFocusWindow", L"D3DFocusWindow", 0,
+ mi.rcMonitor.left, mi.rcMonitor.top, 1, 1,
+ NULL, NULL, GetModuleHandle(NULL), NULL);
+ if (hWnd == 0) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DPPLM::CreateDefaultFocusWindow: CreateWindow failed");
+ } else {
+ J2dTraceLn2(J2D_TRACE_INFO,
+ " Created default focus window %x for adapter %d",
+ hWnd, adapterOrdinal);
+ defaultFocusWindow = hWnd;
+ }
+ return hWnd;
+}
+
+HWND D3DPipelineManager::GetCurrentFocusWindow()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::GetCurrentFocusWindow");
+ if (currentFSFocusAdapter < 0) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ " no fs windows, using default focus window=0x%x",
+ defaultFocusWindow);
+ return defaultFocusWindow;
+ }
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " using fs window=0x%x",
+ pAdapters[currentFSFocusAdapter].fsFocusWindow);
+ return pAdapters[currentFSFocusAdapter].fsFocusWindow;
+}
+
+HWND D3DPipelineManager::SetFSFocusWindow(UINT adapterOrdinal, HWND hWnd)
+{
+ J2dTraceLn2(J2D_TRACE_INFO,"D3DPPLM::SetFSFocusWindow hwnd=0x%x adapter=%d",
+ hWnd, adapterOrdinal);
+
+ HWND prev = pAdapters[adapterOrdinal].fsFocusWindow;
+ pAdapters[adapterOrdinal].fsFocusWindow = hWnd;
+ if (currentFSFocusAdapter < 0) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " first full-screen window");
+ // first fs window
+ currentFSFocusAdapter = adapterOrdinal;
+ // REMIND: we might want to reset the rest of the context here as well
+ // like we do when the an adapter exits fs mode; currently they will
+ // be reset sometime later
+ } else {
+ // there's already a fs window
+ if (currentFSFocusAdapter == adapterOrdinal) {
+ // it's current fs window => we're exiting fs mode on this adapter;
+ // look for a new fs focus window
+ if (hWnd == 0) {
+ UINT i;
+ currentFSFocusAdapter = -1;
+ for (i = 0; i < adapterCount; i++) {
+ if (pAdapters[i].fsFocusWindow != 0) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ " adapter %d is still in fs mode", i);
+ currentFSFocusAdapter = i;
+ break;
+ }
+ }
+ // we have to reset all devices any time current focus device
+ // exits fs mode, and also to prevent some of them being left in
+ // a lost state when the last device exits fs - when non-last
+ // adapters exit fs mode they would not be able to create the
+ // device and will be put in a lost state forever
+ HRESULT res;
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " adapter exited full-screen, reset all adapters");
+ for (i = 0; i < adapterCount; i++) {
+ if (pAdapters[i].pd3dContext != NULL) {
+ res = pAdapters[i].pd3dContext->ResetContext();
+ D3DRQ_MarkLostIfNeeded(res,
+ D3DRQ_GetCurrentDestination());
+ }
+ }
+ } else {
+ J2dTraceLn1(J2D_TRACE_WARNING,
+ "D3DPM::SetFSFocusWindow: setting the fs "\
+ "window again for adapter %d", adapterOrdinal);
+ }
+ }
+ }
+ return prev;
+}
+
+HRESULT D3DPipelineManager::GetD3DContext(UINT adapterOrdinal,
+ D3DContext **ppd3dContext)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DPPLM::GetD3DContext");
+
+ HRESULT res = S_OK;
+ if (adapterOrdinal < 0 || adapterOrdinal >= adapterCount ||
+ pAdapters == NULL ||
+ pAdapters[adapterOrdinal].state == CONTEXT_INIT_FAILED)
+ {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::GetD3DContext: invalid parameters or "\
+ "failed init for adapter %d", adapterOrdinal);
+ *ppd3dContext = NULL;
+ return E_FAIL;
+ }
+
+ if (pAdapters[adapterOrdinal].state == CONTEXT_NOT_INITED) {
+ D3DContext *pCtx = NULL;
+
+ if (pAdapters[adapterOrdinal].pd3dContext != NULL) {
+ J2dTraceLn1(J2D_TRACE_ERROR, " non-null context in "\
+ "uninitialized adapter %d", adapterOrdinal);
+ res = E_FAIL;
+ } else {
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ " initializing context for adapter %d",adapterOrdinal);
+
+ if (SUCCEEDED(res = D3DEnabledOnAdapter(adapterOrdinal))) {
+ res = D3DContext::CreateInstance(pd3d9, adapterOrdinal, &pCtx);
+ if (FAILED(res)) {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::GetD3DContext: failed to create context "\
+ "for adapter=%d", adapterOrdinal);
+ }
+ } else {
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DPPLM::GetContext: no d3d on adapter %d",adapterOrdinal);
+ }
+ }
+ pAdapters[adapterOrdinal].state =
+ SUCCEEDED(res) ? CONTEXT_CREATED : CONTEXT_INIT_FAILED;
+ pAdapters[adapterOrdinal].pd3dContext = pCtx;
+ }
+ *ppd3dContext = pAdapters[adapterOrdinal].pd3dContext;
+ return res;
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DPipelineManager.h b/jdk/src/windows/native/sun/java2d/d3d/D3DPipelineManager.h
new file mode 100644
index 00000000000..56948e859e0
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DPipelineManager.h
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+#pragma once
+
+#include "D3DPipeline.h"
+#include "D3DContext.h"
+
+typedef class D3DPipelineManager *LPD3DPIPELINEMANAGER;
+
+typedef struct D3DAdapter
+{
+ D3DContext *pd3dContext;
+ DWORD state;
+ HWND fsFocusWindow;
+} D3DAdapter;
+
+class D3DPIPELINE_API D3DPipelineManager
+{
+public:
+ // creates and initializes instance of D3DPipelineManager, may return NULL
+ static D3DPipelineManager* CreateInstance(void);
+ // deletes the single instance of the manager
+ static void DeleteInstance();
+ // returns the single instance of the manager, may return NULL
+ static D3DPipelineManager* GetInstance(void);
+
+ HRESULT GetD3DContext(UINT adapterOrdinal, D3DContext **ppd3dContext);
+
+ HRESULT HandleLostDevices();
+ // Checks if adapters were added or removed, or if the order had changed
+ // (which may happen with primary display is changed). If that's the case
+ // releases current adapters and d3d9 instance, reinitializes the pipeline.
+ // @param *monHds list of monitor handles retrieved from GDI
+ // @param monNum number of gdi monitors
+ static
+ HRESULT HandleAdaptersChange(HMONITOR *monHds, UINT monNum);
+ // returns depth stencil buffer format matching adapterFormat and render target
+ // format for the device specified by adapterOrdinal/devType
+ D3DFORMAT GetMatchingDepthStencilFormat(UINT adapterOrdinal,
+ D3DFORMAT adapterFormat,
+ D3DFORMAT renderTargetFormat);
+
+ HWND GetCurrentFocusWindow();
+ // returns previous fs window
+ HWND SetFSFocusWindow(UINT, HWND);
+
+ LPDIRECT3D9 GetD3DObject() { return pd3d9; }
+ D3DDEVTYPE GetDeviceType() { return devType; }
+
+ // returns the d3d adapter ordinal given GDI screen number:
+ // these may differ depending on which display is primary
+ UINT GetAdapterOrdinalForScreen(jint gdiScreen);
+
+ // notifies adapter event listeners by calling
+ // AccelDeviceEventNotifier.eventOccured()
+ static
+ void NotifyAdapterEventListeners(UINT adapter, jint eventType);
+
+private:
+ D3DPipelineManager(void);
+ ~D3DPipelineManager(void);
+
+ // Creates a Direct3D9 object and initializes adapters.
+ HRESULT InitD3D(void);
+ // Releases adapters, Direct3D9 object and the d3d9 library.
+ HRESULT ReleaseD3D();
+
+ // selects the device type based on user input and available
+ // device types
+ D3DDEVTYPE SelectDeviceType();
+
+ // creates array of adapters (releases the old one first)
+ HRESULT InitAdapters();
+ // releases each adapter's context, and then releases the array
+ HRESULT ReleaseAdapters();
+
+ HWND CreateDefaultFocusWindow();
+ // returns S_OK if the adapter is capable of running the Direct3D
+ // pipeline
+ HRESULT D3DEnabledOnAdapter(UINT Adapter);
+ // returns adapterOrdinal given a HMONITOR handle
+ UINT GetAdapterOrdinalByHmon(HMONITOR hMon);
+ HRESULT CheckAdaptersInfo();
+ HRESULT CheckDeviceCaps(UINT Adapter);
+ // Check the OS, succeeds if the OS is XP or newer client-class OS
+static HRESULT CheckOSVersion();
+ // used to check attached adapters using GDI against known bad hw database
+ // prior to the instantiation of the pipeline manager
+static HRESULT GDICheckForBadHardware();
+ // given VendorId, DeviceId and driver version, checks against a database
+ // of known bad hardware/driver combinations.
+ // If the driver version is not known MAX_VERSION can be used
+ // which is guaranteed to satisfy the check
+static HRESULT CheckForBadHardware(DWORD vId, DWORD dId, LONGLONG version);
+
+private:
+
+ // current adapter count
+ UINT adapterCount;
+ // Pointer to Direct3D9 Object mainained by the pipeline manager
+ LPDIRECT3D9 pd3d9;
+ // d3d9.dll lib
+ HINSTANCE hLibD3D9;
+
+ int currentFSFocusAdapter;
+ HWND defaultFocusWindow;
+
+ D3DDEVTYPE devType;
+
+ D3DAdapter *pAdapters;
+ // instance of this object
+ static LPD3DPIPELINEMANAGER pMgr;
+};
+
+#define OS_UNDEFINED (0 << 0)
+#define OS_VISTA (1 << 0)
+#define OS_WINSERV_2008 (1 << 1)
+#define OS_WINXP (1 << 2)
+#define OS_WINXP_64 (1 << 3)
+#define OS_WINSERV_2003 (1 << 4)
+#define OS_ALL (OS_VISTA|OS_WINSERV_2008|OS_WINXP|OS_WINXP_64|OS_WINSERV_2003)
+#define OS_UNKNOWN (~OS_ALL)
+BOOL D3DPPLM_OsVersionMatches(USHORT osInfo);
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DRenderQueue.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderQueue.cpp
new file mode 100644
index 00000000000..e33f1c4e1db
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderQueue.cpp
@@ -0,0 +1,954 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include
+#include
+#include "sun_java2d_pipe_BufferedOpCodes.h"
+
+#include "jlong.h"
+#include "D3DBlitLoops.h"
+#include "D3DBufImgOps.h"
+#include "D3DPipelineManager.h"
+#include "D3DContext.h"
+#include "D3DMaskBlit.h"
+#include "D3DMaskFill.h"
+#include "D3DPaints.h"
+#include "D3DRenderQueue.h"
+#include "D3DRenderer.h"
+#include "D3DSurfaceData.h"
+#include "D3DTextRenderer.h"
+#include "Trace.h"
+#include "awt_Toolkit.h"
+
+BOOL DWMIsCompositionEnabled();
+
+/**
+ * References to the "current" context and destination surface.
+ */
+static D3DContext *d3dc = NULL;
+static D3DSDOps *dstOps = NULL;
+static BOOL bLostDevices = FALSE;
+
+typedef struct {
+ byte *buffer;
+ int limit;
+ jobject runnable;
+} FlushBufferStruct;
+
+HRESULT
+D3DRQ_SwapBuffers(D3DPipelineManager *pMgr, D3DSDOps *d3dsdo,
+ int x1, int y1, int x2, int y2)
+{
+ HRESULT res;
+ D3DContext *pCtx;
+ IDirect3DSwapChain9 *pSwapChain;
+ RECT srcRect, dstRect, *pSrcRect, *pDstRect;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRQ_SwapBuffers");
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " x1=%d y1=%d x2=%d y2=%d",
+ x1, y1, x2, y2);
+
+ RETURN_STATUS_IF_NULL(d3dsdo, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dsdo->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(pSwapChain=d3dsdo->pResource->GetSwapChain(), E_FAIL);
+
+ pCtx = D3DRQ_GetCurrentContext();
+ if (pCtx != NULL) {
+ // flush the current vertex queue here, just in case
+ res = d3dc->FlushVertexQueue();
+ D3DRQ_MarkLostIfNeeded(res, dstOps);
+ pCtx = NULL;
+ }
+ // end scene for this destination
+ res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pCtx->EndScene();
+
+ // This is a workaround for what apparently is a DWM bug.
+ // If the dimensions of the back-buffer don't match the dimensions of
+ // the window, Present() will flash the whole window with black.
+ // The workaround is to detect this situation and not do a present.
+ // It is ok to do so since a repaint event is coming due to the resize that
+ // just happened.
+ //
+ // REMIND: this will need to be updated if we switch to creating
+ // back-buffers of the size of the client area instead of the whole window
+ // (use GetClientRect() instead of GetWindowRect()).
+ if (DWMIsCompositionEnabled()) {
+ RECT r;
+ D3DPRESENT_PARAMETERS params;
+
+ pSwapChain->GetPresentParameters(¶ms);
+ GetWindowRect(params.hDeviceWindow, &r);
+ int ww = r.right - r.left;
+ int wh = r.bottom - r.top;
+ if (ww != params.BackBufferWidth || wh != params.BackBufferHeight) {
+ J2dTraceLn4(J2D_TRACE_WARNING,
+ "D3DRQ_SwapBuffers: surface/window dimensions mismatch: "\
+ "win: w=%d h=%d, bb: w=%d h=%d",
+ ww, wh, params.BackBufferWidth, params.BackBufferHeight);
+
+ return S_OK;
+ }
+ }
+
+ if (d3dsdo->swapEffect == D3DSWAPEFFECT_COPY) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " D3DSWAPEFFECT_COPY");
+ if (x1 < 0) x1 = 0;
+ if (y1 < 0) y1 = 0;
+ if (x2 > d3dsdo->width) x2 = d3dsdo->width;
+ if (y2 > d3dsdo->height) y2 = d3dsdo->height;
+ if (x2 <= x1 || y2 <= y1) {
+ // nothing to present
+ return S_OK;
+ }
+ srcRect.left = x1;
+ srcRect.top = y1;
+ srcRect.right = x2;
+ srcRect.bottom = y2;
+
+ dstRect = srcRect;
+
+ pSrcRect = &srcRect;
+ pDstRect = &dstRect;
+ // only offset in windowed mode
+ if (pCtx!= NULL && pCtx->GetPresentationParams()->Windowed) {
+ OffsetRect(pDstRect, d3dsdo->xoff, d3dsdo->yoff);
+ } else {
+ // some boards (Nvidia) have problems with copy strategy and
+ // non-null src/dest rectangles in fs mode; unfortunately this
+ // means that we'll paint over fs window decorations
+ pSrcRect = NULL;
+ pDstRect = NULL;
+ }
+ } else {
+ if (d3dsdo->swapEffect == D3DSWAPEFFECT_FLIP) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " D3DSWAPEFFECT_FLIP");
+ } else {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " D3DSWAPEFFECT_DISCARD");
+ }
+ // src and dest rectangles must be NULL for FLIP/DISCARD
+ pSrcRect = NULL;
+ pDstRect = NULL;
+ }
+
+ res = pSwapChain->Present(pSrcRect, pDstRect, 0, NULL, 0);
+ res = D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+
+ return res;
+}
+
+HRESULT
+D3DRQ_MarkLostIfNeeded(HRESULT res, D3DSDOps *d3dops)
+{
+ if (res == D3DERR_DEVICELOST || res == D3DERR_DEVICENOTRESET) {
+ D3DContext *pCtx;
+
+ J2dTraceLn(J2D_TRACE_WARNING, "D3DRQ_MarkLostIfNeeded: device lost");
+ bLostDevices = TRUE;
+
+ // only mark surfaces belonging to the lost device
+ if (d3dops != NULL &&
+ SUCCEEDED(res = D3DPipelineManager::GetInstance()->
+ GetD3DContext(d3dops->adapter, &pCtx)))
+ {
+ IDirect3DDevice9 *pd3dDevice = pCtx->Get3DDevice();
+ if (pd3dDevice) {
+ HRESULT res1 = pd3dDevice->TestCooperativeLevel();
+ if (res1 != D3DERR_DEVICELOST && res1 != D3DERR_DEVICENOTRESET){
+ // this surface's device is not lost, do not mark it
+ return res;
+ }
+ }
+ }
+ D3DSD_MarkLost(d3dops);
+ }
+ return res;
+}
+
+void D3DRQ_FlushBuffer(void *pParam)
+{
+ FlushBufferStruct *pFlush = (FlushBufferStruct*)pParam;
+ JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
+ unsigned char *b, *end;
+ int limit;
+ HRESULT res = S_OK;
+ BOOL bSync = FALSE;
+
+ b = pFlush->buffer;
+ limit = pFlush->limit;
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DRQ_flushBuffer: limit=%d", limit);
+
+ end = b + limit;
+
+ D3DPipelineManager *pMgr = D3DPipelineManager::GetInstance();
+ if (pMgr == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_WARNING, "D3DRQ_flushBuffer: null manager");
+ return;
+ }
+
+ if (bLostDevices) {
+ if (SUCCEEDED(res = pMgr->HandleLostDevices())) {
+ bLostDevices = FALSE;
+ }
+ }
+
+ while (b < end) {
+ jint opcode = NEXT_INT(b);
+
+ J2dTraceLn1(J2D_TRACE_VERBOSE, "D3DRQ_flushBuffer: opcode=%d", opcode);
+
+ switch (opcode) {
+
+ // draw ops
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_LINE:
+ {
+ jint x1 = NEXT_INT(b);
+ jint y1 = NEXT_INT(b);
+ jint x2 = NEXT_INT(b);
+ jint y2 = NEXT_INT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_DrawLine(d3dc, x1, y1, x2, y2);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_RECT:
+ {
+ jint x = NEXT_INT(b);
+ jint y = NEXT_INT(b);
+ jint w = NEXT_INT(b);
+ jint h = NEXT_INT(b);
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_DrawRect(d3dc, x, y, w, h);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_POLY:
+ {
+ jint nPoints = NEXT_INT(b);
+ jboolean isClosed = NEXT_BOOLEAN(b);
+ jint transX = NEXT_INT(b);
+ jint transY = NEXT_INT(b);
+ jint *xPoints = (jint *)b;
+ jint *yPoints = ((jint *)b) + nPoints;
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_DrawPoly(d3dc, nPoints, isClosed,
+ transX, transY,
+ xPoints, yPoints);
+ SKIP_BYTES(b, nPoints * BYTES_PER_POLY_POINT);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_PIXEL:
+ {
+ jint x = NEXT_INT(b);
+ jint y = NEXT_INT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_DrawLine(d3dc, x, y, x, y);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_SCANLINES:
+ {
+ jint count = NEXT_INT(b);
+ res = D3DRenderer_DrawScanlines(d3dc, count, (jint *)b);
+ SKIP_BYTES(b, count * BYTES_PER_SCANLINE);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_PARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+ jfloat lwr21 = NEXT_FLOAT(b);
+ jfloat lwr12 = NEXT_FLOAT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_DrawParallelogram(d3dc,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12,
+ lwr21, lwr12);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_AAPARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+ jfloat lwr21 = NEXT_FLOAT(b);
+ jfloat lwr12 = NEXT_FLOAT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_DrawAAParallelogram(d3dc,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12,
+ lwr21, lwr12);
+ }
+ break;
+
+ // fill ops
+ case sun_java2d_pipe_BufferedOpCodes_FILL_RECT:
+ {
+ jint x = NEXT_INT(b);
+ jint y = NEXT_INT(b);
+ jint w = NEXT_INT(b);
+ jint h = NEXT_INT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_FillRect(d3dc, x, y, w, h);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_FILL_PARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_FillParallelogram(d3dc,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_FILL_AAPARALLELOGRAM:
+ {
+ jfloat x11 = NEXT_FLOAT(b);
+ jfloat y11 = NEXT_FLOAT(b);
+ jfloat dx21 = NEXT_FLOAT(b);
+ jfloat dy21 = NEXT_FLOAT(b);
+ jfloat dx12 = NEXT_FLOAT(b);
+ jfloat dy12 = NEXT_FLOAT(b);
+
+ CONTINUE_IF_NULL(d3dc);
+ res = D3DRenderer_FillAAParallelogram(d3dc,
+ x11, y11,
+ dx21, dy21,
+ dx12, dy12);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_FILL_SPANS:
+ {
+ jint count = NEXT_INT(b);
+ res = D3DRenderer_FillSpans(d3dc, count, (jint *)b);
+ SKIP_BYTES(b, count * BYTES_PER_SPAN);
+ }
+ break;
+
+ // text-related ops
+ case sun_java2d_pipe_BufferedOpCodes_DRAW_GLYPH_LIST:
+ {
+ jint numGlyphs = NEXT_INT(b);
+ jint packedParams = NEXT_INT(b);
+ jfloat glyphListOrigX = NEXT_FLOAT(b);
+ jfloat glyphListOrigY = NEXT_FLOAT(b);
+ jboolean usePositions = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_POSITIONS);
+ jboolean subPixPos = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_SUBPIXPOS);
+ jboolean rgbOrder = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_RGBORDER);
+ jint lcdContrast = EXTRACT_BYTE(packedParams,
+ OFFSET_CONTRAST);
+ unsigned char *images = b;
+ unsigned char *positions;
+ jint bytesPerGlyph;
+ if (usePositions) {
+ positions = (b + numGlyphs * BYTES_PER_GLYPH_IMAGE);
+ bytesPerGlyph = BYTES_PER_POSITIONED_GLYPH;
+ } else {
+ positions = NULL;
+ bytesPerGlyph = BYTES_PER_GLYPH_IMAGE;
+ }
+ res = D3DTR_DrawGlyphList(d3dc, dstOps,
+ numGlyphs, usePositions,
+ subPixPos, rgbOrder, lcdContrast,
+ glyphListOrigX, glyphListOrigY,
+ images, positions);
+ SKIP_BYTES(b, numGlyphs * bytesPerGlyph);
+ }
+ break;
+
+ // copy-related ops
+ case sun_java2d_pipe_BufferedOpCodes_COPY_AREA:
+ {
+ jint x = NEXT_INT(b);
+ jint y = NEXT_INT(b);
+ jint w = NEXT_INT(b);
+ jint h = NEXT_INT(b);
+ jint dx = NEXT_INT(b);
+ jint dy = NEXT_INT(b);
+ res = D3DBlitLoops_CopyArea(env, d3dc, dstOps,
+ x, y, w, h, dx, dy);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_BLIT:
+ {
+ jint packedParams = NEXT_INT(b);
+ jint sx1 = NEXT_INT(b);
+ jint sy1 = NEXT_INT(b);
+ jint sx2 = NEXT_INT(b);
+ jint sy2 = NEXT_INT(b);
+ jdouble dx1 = NEXT_DOUBLE(b);
+ jdouble dy1 = NEXT_DOUBLE(b);
+ jdouble dx2 = NEXT_DOUBLE(b);
+ jdouble dy2 = NEXT_DOUBLE(b);
+ jlong pSrc = NEXT_LONG(b);
+ jlong pDst = NEXT_LONG(b);
+ jint hint = EXTRACT_BYTE(packedParams, OFFSET_HINT);
+ jboolean texture = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_TEXTURE);
+ jboolean rtt = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_RTT);
+ jboolean xform = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_XFORM);
+ jboolean isoblit = EXTRACT_BOOLEAN(packedParams,
+ OFFSET_ISOBLIT);
+ if (isoblit) {
+ res = D3DBlitLoops_IsoBlit(env, d3dc, pSrc, pDst,
+ xform, hint, texture, rtt,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2);
+ D3DRQ_MarkLostIfNeeded(res, (D3DSDOps*)pSrc);
+ } else {
+ jint srctype = EXTRACT_BYTE(packedParams, OFFSET_SRCTYPE);
+ res = D3DBlitLoops_Blit(env, d3dc, pSrc, pDst,
+ xform, hint, srctype, texture,
+ sx1, sy1, sx2, sy2,
+ dx1, dy1, dx2, dy2);
+ }
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SURFACE_TO_SW_BLIT:
+ {
+ jint sx = NEXT_INT(b);
+ jint sy = NEXT_INT(b);
+ jint dx = NEXT_INT(b);
+ jint dy = NEXT_INT(b);
+ jint w = NEXT_INT(b);
+ jint h = NEXT_INT(b);
+ jint dsttype = NEXT_INT(b);
+ jlong pSrc = NEXT_LONG(b);
+ jlong pDst = NEXT_LONG(b);
+ res = D3DBlitLoops_SurfaceToSwBlit(env, d3dc,
+ pSrc, pDst, dsttype,
+ sx, sy, dx, dy, w, h);
+ D3DRQ_MarkLostIfNeeded(res, (D3DSDOps*)pSrc);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_MASK_FILL:
+ {
+ jint x = NEXT_INT(b);
+ jint y = NEXT_INT(b);
+ jint w = NEXT_INT(b);
+ jint h = NEXT_INT(b);
+ jint maskoff = NEXT_INT(b);
+ jint maskscan = NEXT_INT(b);
+ jint masklen = NEXT_INT(b);
+ unsigned char *pMask = (masklen > 0) ? b : NULL;
+ res = D3DMaskFill_MaskFill(d3dc, x, y, w, h,
+ maskoff, maskscan, masklen, pMask);
+ SKIP_BYTES(b, masklen);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_MASK_BLIT:
+ {
+ jint dstx = NEXT_INT(b);
+ jint dsty = NEXT_INT(b);
+ jint width = NEXT_INT(b);
+ jint height = NEXT_INT(b);
+ jint masklen = width * height * sizeof(jint);
+ res = D3DMaskBlit_MaskBlit(env, d3dc,
+ dstx, dsty, width, height, b);
+ SKIP_BYTES(b, masklen);
+ }
+ break;
+
+ // state-related ops
+ case sun_java2d_pipe_BufferedOpCodes_SET_RECT_CLIP:
+ {
+ jint x1 = NEXT_INT(b);
+ jint y1 = NEXT_INT(b);
+ jint x2 = NEXT_INT(b);
+ jint y2 = NEXT_INT(b);
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->SetRectClip(x1, y1, x2, y2);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_BEGIN_SHAPE_CLIP:
+ {
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->BeginShapeClip();
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_SHAPE_CLIP_SPANS:
+ {
+ jint count = NEXT_INT(b);
+ res = D3DRenderer_FillSpans(d3dc, count, (jint *)b);
+ SKIP_BYTES(b, count * BYTES_PER_SPAN);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_END_SHAPE_CLIP:
+ {
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->EndShapeClip();
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_RESET_CLIP:
+ {
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->ResetClip();
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_ALPHA_COMPOSITE:
+ {
+ jint rule = NEXT_INT(b);
+ jfloat extraAlpha = NEXT_FLOAT(b);
+ jint flags = NEXT_INT(b);
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->SetAlphaComposite(rule, extraAlpha, flags);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_XOR_COMPOSITE:
+ {
+ jint xorPixel = NEXT_INT(b);
+// res = d3dc->SetXorComposite(d3dc, xorPixel);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_RESET_COMPOSITE:
+ {
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->ResetComposite();
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_TRANSFORM:
+ {
+ jdouble m00 = NEXT_DOUBLE(b);
+ jdouble m10 = NEXT_DOUBLE(b);
+ jdouble m01 = NEXT_DOUBLE(b);
+ jdouble m11 = NEXT_DOUBLE(b);
+ jdouble m02 = NEXT_DOUBLE(b);
+ jdouble m12 = NEXT_DOUBLE(b);
+ res = d3dc->SetTransform(m00, m10, m01, m11, m02, m12);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_RESET_TRANSFORM:
+ {
+ CONTINUE_IF_NULL(d3dc);
+ res = d3dc->ResetTransform();
+ }
+ break;
+
+ // context-related ops
+ case sun_java2d_pipe_BufferedOpCodes_SET_SURFACES:
+ {
+ jlong pSrc = NEXT_LONG(b);
+ jlong pDst = NEXT_LONG(b);
+ D3DContext *oldd3dc = NULL;
+ if (d3dc != NULL) {
+ oldd3dc = d3dc;
+ d3dc = NULL;
+ oldd3dc->UpdateState(STATE_CHANGE);
+ }
+ dstOps = (D3DSDOps *)jlong_to_ptr(pDst);
+ res = pMgr->GetD3DContext(dstOps->adapter, &d3dc);
+ if (FAILED(res)) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DRQ_FlushBuffer: failed to get context");
+ D3DRQ_ResetCurrentContextAndDestination();
+ break;
+ }
+ // REMIND: we may also want to do EndScene on each
+ // render target change so that the GPU can go work on
+ // whatever is already in the queue
+ if (oldd3dc != d3dc && oldd3dc != NULL) {
+ res = oldd3dc->EndScene();
+ }
+ CONTINUE_IF_NULL(dstOps->pResource);
+ res = d3dc->SetRenderTarget(dstOps->pResource->GetSurface());
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_SCRATCH_SURFACE:
+ {
+ jint screen = NEXT_INT(b);
+ jint adapter = pMgr->GetAdapterOrdinalForScreen(screen);
+ D3DContext *oldd3dc = NULL;
+
+ if (d3dc != NULL) {
+ oldd3dc = d3dc;
+ d3dc = NULL;
+ }
+ res = pMgr->GetD3DContext(adapter, &d3dc);
+ if (FAILED(res)) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DRQ_FlushBuffer: failed to get context");
+ D3DRQ_ResetCurrentContextAndDestination();
+ } else if (oldd3dc != d3dc && oldd3dc != NULL) {
+ res = oldd3dc->EndScene();
+ }
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_FLUSH_SURFACE:
+ {
+ jlong pData = NEXT_LONG(b);
+ D3DSDOps *d3dsdo = (D3DSDOps *)jlong_to_ptr(pData);
+ D3DSD_Flush(d3dsdo);
+ if (dstOps == d3dsdo) {
+ dstOps = NULL;
+ }
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DISPOSE_SURFACE:
+ {
+ jlong pData = NEXT_LONG(b);
+ D3DSDOps *d3dsdo = (D3DSDOps *)jlong_to_ptr(pData);
+ D3DSD_Flush(d3dsdo);
+ if (dstOps == d3dsdo) {
+ dstOps = NULL;
+ }
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DISPOSE_CONFIG:
+ {
+ jlong pConfigInfo = NEXT_LONG(b);
+ CONTINUE_IF_NULL(d3dc);
+ // REMIND: does this need to be implemented for D3D?
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_INVALIDATE_CONTEXT:
+ {
+ // flush just in case there are any pending operations in
+ // the hardware pipe
+ if (d3dc != NULL) {
+ res = d3dc->EndScene();
+ }
+
+ // invalidate the references to the current context and
+ // destination surface that are maintained at the native level
+ D3DRQ_ResetCurrentContextAndDestination();
+ }
+ break;
+
+ case sun_java2d_pipe_BufferedOpCodes_SYNC:
+ {
+ bSync = TRUE;
+ }
+ break;
+
+ case sun_java2d_pipe_BufferedOpCodes_RESTORE_DEVICES:
+ {
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRQ_FlushBuffer: RESTORE_DEVICES");
+ if (SUCCEEDED(res = pMgr->HandleLostDevices())) {
+ bLostDevices = FALSE;
+ } else {
+ bLostDevices = TRUE;
+ }
+ }
+ break;
+
+ case sun_java2d_pipe_BufferedOpCodes_SAVE_STATE:
+ {
+ CONTINUE_IF_NULL(d3dc);
+
+ res = d3dc->SaveState();
+ }
+ break;
+
+ case sun_java2d_pipe_BufferedOpCodes_RESTORE_STATE:
+ {
+ CONTINUE_IF_NULL(d3dc);
+
+ res = d3dc->RestoreState();
+ }
+ break;
+
+ // multibuffering ops
+ case sun_java2d_pipe_BufferedOpCodes_SWAP_BUFFERS:
+ {
+ jlong sdo = NEXT_LONG(b);
+ jint x1 = NEXT_INT(b);
+ jint y1 = NEXT_INT(b);
+ jint x2 = NEXT_INT(b);
+ jint y2 = NEXT_INT(b);
+
+ res = D3DRQ_SwapBuffers(pMgr, (D3DSDOps *)jlong_to_ptr(sdo),
+ x1, y1, x2, y2);
+ }
+ break;
+
+ // special no-op (mainly used for achieving 8-byte alignment)
+ case sun_java2d_pipe_BufferedOpCodes_NOOP:
+ break;
+
+ // paint-related ops
+ case sun_java2d_pipe_BufferedOpCodes_RESET_PAINT:
+ {
+ res = D3DPaints_ResetPaint(d3dc);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_COLOR:
+ {
+ jint pixel = NEXT_INT(b);
+ res = D3DPaints_SetColor(d3dc, pixel);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_GRADIENT_PAINT:
+ {
+ jboolean useMask= NEXT_BOOLEAN(b);
+ jboolean cyclic = NEXT_BOOLEAN(b);
+ jdouble p0 = NEXT_DOUBLE(b);
+ jdouble p1 = NEXT_DOUBLE(b);
+ jdouble p3 = NEXT_DOUBLE(b);
+ jint pixel1 = NEXT_INT(b);
+ jint pixel2 = NEXT_INT(b);
+ res = D3DPaints_SetGradientPaint(d3dc, useMask, cyclic,
+ p0, p1, p3,
+ pixel1, pixel2);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_LINEAR_GRADIENT_PAINT:
+ {
+ jboolean useMask = NEXT_BOOLEAN(b);
+ jboolean linear = NEXT_BOOLEAN(b);
+ jint cycleMethod = NEXT_INT(b);
+ jint numStops = NEXT_INT(b);
+ jfloat p0 = NEXT_FLOAT(b);
+ jfloat p1 = NEXT_FLOAT(b);
+ jfloat p3 = NEXT_FLOAT(b);
+ void *fractions, *pixels;
+ fractions = b; SKIP_BYTES(b, numStops * sizeof(jfloat));
+ pixels = b; SKIP_BYTES(b, numStops * sizeof(jint));
+ res = D3DPaints_SetLinearGradientPaint(d3dc, dstOps,
+ useMask, linear,
+ cycleMethod, numStops,
+ p0, p1, p3,
+ fractions, pixels);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_RADIAL_GRADIENT_PAINT:
+ {
+ jboolean useMask = NEXT_BOOLEAN(b);
+ jboolean linear = NEXT_BOOLEAN(b);
+ jint numStops = NEXT_INT(b);
+ jint cycleMethod = NEXT_INT(b);
+ jfloat m00 = NEXT_FLOAT(b);
+ jfloat m01 = NEXT_FLOAT(b);
+ jfloat m02 = NEXT_FLOAT(b);
+ jfloat m10 = NEXT_FLOAT(b);
+ jfloat m11 = NEXT_FLOAT(b);
+ jfloat m12 = NEXT_FLOAT(b);
+ jfloat focusX = NEXT_FLOAT(b);
+ void *fractions, *pixels;
+ fractions = b; SKIP_BYTES(b, numStops * sizeof(jfloat));
+ pixels = b; SKIP_BYTES(b, numStops * sizeof(jint));
+ res = D3DPaints_SetRadialGradientPaint(d3dc, dstOps,
+ useMask, linear,
+ cycleMethod, numStops,
+ m00, m01, m02,
+ m10, m11, m12,
+ focusX,
+ fractions, pixels);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_SET_TEXTURE_PAINT:
+ {
+ jboolean useMask= NEXT_BOOLEAN(b);
+ jboolean filter = NEXT_BOOLEAN(b);
+ jlong pSrc = NEXT_LONG(b);
+ jdouble xp0 = NEXT_DOUBLE(b);
+ jdouble xp1 = NEXT_DOUBLE(b);
+ jdouble xp3 = NEXT_DOUBLE(b);
+ jdouble yp0 = NEXT_DOUBLE(b);
+ jdouble yp1 = NEXT_DOUBLE(b);
+ jdouble yp3 = NEXT_DOUBLE(b);
+ res = D3DPaints_SetTexturePaint(d3dc, useMask, pSrc, filter,
+ xp0, xp1, xp3,
+ yp0, yp1, yp3);
+ }
+ break;
+
+ // BufferedImageOp-related ops
+ case sun_java2d_pipe_BufferedOpCodes_ENABLE_CONVOLVE_OP:
+ {
+ jlong pSrc = NEXT_LONG(b);
+ jboolean edgeZero = NEXT_BOOLEAN(b);
+ jint kernelWidth = NEXT_INT(b);
+ jint kernelHeight = NEXT_INT(b);
+ res = D3DBufImgOps_EnableConvolveOp(d3dc, pSrc, edgeZero,
+ kernelWidth, kernelHeight, b);
+ SKIP_BYTES(b, kernelWidth * kernelHeight * sizeof(jfloat));
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DISABLE_CONVOLVE_OP:
+ {
+ res = D3DBufImgOps_DisableConvolveOp(d3dc);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_ENABLE_RESCALE_OP:
+ {
+ jlong pSrc = NEXT_LONG(b); // unused
+ jboolean nonPremult = NEXT_BOOLEAN(b);
+ jint numFactors = 4;
+ unsigned char *scaleFactors = b;
+ unsigned char *offsets = (b + numFactors * sizeof(jfloat));
+ res = D3DBufImgOps_EnableRescaleOp(d3dc, nonPremult,
+ scaleFactors, offsets);
+ SKIP_BYTES(b, numFactors * sizeof(jfloat) * 2);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DISABLE_RESCALE_OP:
+ {
+ D3DBufImgOps_DisableRescaleOp(d3dc);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_ENABLE_LOOKUP_OP:
+ {
+ jlong pSrc = NEXT_LONG(b); // unused
+ jboolean nonPremult = NEXT_BOOLEAN(b);
+ jboolean shortData = NEXT_BOOLEAN(b);
+ jint numBands = NEXT_INT(b);
+ jint bandLength = NEXT_INT(b);
+ jint offset = NEXT_INT(b);
+ jint bytesPerElem = shortData ? sizeof(jshort):sizeof(jbyte);
+ void *tableValues = b;
+ res = D3DBufImgOps_EnableLookupOp(d3dc, nonPremult, shortData,
+ numBands, bandLength, offset,
+ tableValues);
+ SKIP_BYTES(b, numBands * bandLength * bytesPerElem);
+ }
+ break;
+ case sun_java2d_pipe_BufferedOpCodes_DISABLE_LOOKUP_OP:
+ {
+ res = D3DBufImgOps_DisableLookupOp(d3dc);
+ }
+ break;
+
+ default:
+ J2dRlsTraceLn1(J2D_TRACE_ERROR,
+ "D3DRQ_flushBuffer: invalid opcode=%d", opcode);
+ return;
+ }
+ // we may mark the surface lost repeatedly but that won't do much harm
+ res = D3DRQ_MarkLostIfNeeded(res, dstOps);
+ }
+
+ if (d3dc != NULL) {
+ res = d3dc->EndScene();
+ // REMIND: EndScene is not really enough to flush the
+ // whole d3d pipeline
+
+ // REMIND: there may be an issue with BeginScene/EndScene
+ // for each flushQueue, because of the blits, which flush
+ // the queue
+ if (bSync) {
+ res = d3dc->Sync();
+ }
+ }
+
+ // REMIND: we need to also handle hard errors here as well, and disable
+ // particular context if needed
+ D3DRQ_MarkLostIfNeeded(res, dstOps);
+
+ if (!JNU_IsNull(env, pFlush->runnable)) {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " executing runnable");
+ JNU_CallMethodByName(env, NULL, pFlush->runnable, "run", "()V");
+ }
+}
+
+/**
+ * Returns a pointer to the "current" context, as set by the last SET_SURFACES
+ * or SET_SCRATCH_SURFACE operation.
+ */
+D3DContext *
+D3DRQ_GetCurrentContext()
+{
+ return d3dc;
+}
+
+/**
+ * Returns a pointer to the "current" destination surface, as set by the last
+ * SET_SURFACES operation.
+ */
+D3DSDOps *
+D3DRQ_GetCurrentDestination()
+{
+ return dstOps;
+}
+
+/**
+ * Resets current context and destination surface.
+ */
+void
+D3DRQ_ResetCurrentContextAndDestination()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRQ_ResetCurrentContextAndDestination");
+
+ d3dc = NULL;
+ dstOps = NULL;
+}
+
+extern "C"
+{
+
+/*
+ * Class: sun_java2d_d3d_D3DRenderQueue
+ * Method: flushBuffer
+ * Signature: (JILjava/lang/Runnable;)V
+ */
+JNIEXPORT void JNICALL
+Java_sun_java2d_d3d_D3DRenderQueue_flushBuffer
+ (JNIEnv *env, jobject d3drq, jlong buf, jint limit, jobject runnable)
+{
+ FlushBufferStruct bufstr;
+ // just in case we forget to init any new fields
+ ZeroMemory(&bufstr, sizeof(FlushBufferStruct));
+
+ bufstr.buffer = (unsigned char *)jlong_to_ptr(buf);
+ if (bufstr.buffer == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DRenderQueue_flushBuffer: cannot get direct buffer address");
+ return;
+ }
+ bufstr.limit = limit;
+
+ bufstr.runnable = JNU_IsNull(env, runnable) ?
+ NULL : env->NewGlobalRef(runnable);
+ AwtToolkit::GetInstance().InvokeFunction(D3DRQ_FlushBuffer, &bufstr);
+ if (!JNU_IsNull(env, bufstr.runnable)) {
+ env->DeleteGlobalRef(bufstr.runnable);
+ }
+}
+
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DRenderQueue.h b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderQueue.h
new file mode 100644
index 00000000000..01bcc777bd5
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderQueue.h
@@ -0,0 +1,64 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DRenderQueue_h_Included
+#define D3DRenderQueue_h_Included
+
+#include "D3DContext.h"
+#include "D3DSurfaceData.h"
+
+/*
+ * The following macros are used to pick values (of the specified type) off
+ * the queue.
+ */
+#define NEXT_VAL(buf, type) (((type *)((buf) += sizeof(type)))[-1])
+#define NEXT_BYTE(buf) NEXT_VAL(buf, unsigned char)
+#define NEXT_INT(buf) NEXT_VAL(buf, jint)
+#define NEXT_FLOAT(buf) NEXT_VAL(buf, jfloat)
+#define NEXT_BOOLEAN(buf) (jboolean)NEXT_INT(buf)
+#define NEXT_LONG(buf) NEXT_VAL(buf, jlong)
+#define NEXT_DOUBLE(buf) NEXT_VAL(buf, jdouble)
+
+/*
+ * Increments a pointer (buf) by the given number of bytes.
+ */
+#define SKIP_BYTES(buf, numbytes) buf += (numbytes)
+
+/*
+ * Extracts a value at the given offset from the provided packed value.
+ */
+#define EXTRACT_VAL(packedval, offset, mask) \
+ (((packedval) >> (offset)) & (mask))
+#define EXTRACT_BYTE(packedval, offset) \
+ (unsigned char)EXTRACT_VAL(packedval, offset, 0xff)
+#define EXTRACT_BOOLEAN(packedval, offset) \
+ (jboolean)EXTRACT_VAL(packedval, offset, 0x1)
+
+D3DContext *D3DRQ_GetCurrentContext();
+D3DSDOps *D3DRQ_GetCurrentDestination();
+void D3DRQ_ResetCurrentContextAndDestination();
+HRESULT D3DRQ_MarkLostIfNeeded(HRESULT res, D3DSDOps *d3dops);
+
+#endif /* D3DRenderQueue_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.cpp
index 4b0b358e8ad..8da2398ebcf 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.cpp
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -24,450 +24,367 @@
*/
#include "sun_java2d_d3d_D3DRenderer.h"
-#include "sun_java2d_windows_DDRenderer.h"
-#include "Win32SurfaceData.h"
-#include
+#include "D3DContext.h"
+#include "D3DRenderer.h"
+#include "D3DRenderQueue.h"
-#include "D3DUtils.h"
-#include "ddrawUtils.h"
-
-#include "j2d_md.h"
-#include "jlong.h"
-
-/*
- * Class: sun_java2d_d3d_D3DRenderer
- * Method: doDrawLineD3D
- * Signature: (Lsun/java2d/SurfaceData;IIIII)Z
- */
-JNIEXPORT jboolean JNICALL
-Java_sun_java2d_d3d_D3DRenderer_doDrawLineD3D
- (JNIEnv *env, jobject d3dr,
- jlong pData, jlong pCtx,
- jint x1, jint y1, jint x2, jint y2)
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawLine(D3DContext *d3dc,
+ jint x1, jint y1, jint x2, jint y2)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
- static J2D_XY_C_VERTEX lineVerts[] = {
-#ifdef USE_SINGLE_VERTEX_FORMAT
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
-#else
- // x, y, z, color
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
-#endif // USE_SINGLE_VERTEX_FORMAT
- };
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_doDrawLineD3D");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " x1=%-4d y1=%-4d x2=%-4d y2=%-4d",
+ J2dTraceLn4(J2D_TRACE_INFO,
+ "D3DRenderer_doDrawLineD3D x1=%-4d y1=%-4d x2=%-4d y2=%-4d",
x1, y1, x2, y2);
- HRESULT res = DDERR_GENERIC;
- if (d3dc != NULL) {
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return FALSE;
- }
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return FALSE;
- }
-
- // +0.5f is needed to compensate for the -0.5f we
- // force when setting the transform
- lineVerts[0].x = (float)x1 + 0.5f;
- lineVerts[0].y = (float)y1 + 0.5f;
- lineVerts[0].color = d3dc->colorPixel;
- lineVerts[1].x = (float)x2 + 0.5f;
- lineVerts[1].y = (float)y2 + 0.5f;
- lineVerts[1].color = d3dc->colorPixel;
-
- D3DU_PRIM_LOOP_BEGIN(res, wsdo);
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_RENDEROP))) {
- res = d3dDevice->DrawPrimitive(D3DPT_LINESTRIP, D3DFVF_J2D_XY_C,
- lineVerts, 2, 0);
- // REMIND: need to be using the results of device testing
- res = d3dDevice->DrawPrimitive(D3DPT_POINTLIST, D3DFVF_J2D_XY_C,
- &(lineVerts[1]), 1, 0);
- d3dc->EndScene(res);
- }
- D3DU_PRIM_LOOP_END(env, res, wsdo, "DrawPrimitive(D3DPT_LINESTRIP)");
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- }
- return SUCCEEDED(res);
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->DrawLine(x1, y1, x2, y2);
}
-JNIEXPORT jboolean JNICALL
-Java_sun_java2d_d3d_D3DRenderer_doDrawRectD3D
- (JNIEnv *env, jobject d3dr,
- jlong pData, jlong pCtx,
- jint x, jint y, jint w, jint h)
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawRect(D3DContext *d3dc,
+ jint x, jint y, jint w, jint h)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
- float x1, y1, x2, y2;
- static J2D_XY_C_VERTEX lineVerts[] = {
-#ifdef USE_SINGLE_VERTEX_FORMAT
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
-#else
- // x, y, z, color
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
-#endif // USE_SINGLE_VERTEX_FORMAT
- };
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_doDrawRectD3D");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " x=%-4d y=%-4d w=%-4d h=%-4d",
+ J2dTraceLn4(J2D_TRACE_INFO,
+ "D3DRenderer_DrawRect x=%-4d y=%-4d w=%-4d h=%-4d",
x, y, w, h);
- HRESULT res = DDERR_GENERIC;
- if (d3dc != NULL) {
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return FALSE;
- }
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return FALSE;
- }
-
- // +0.5f is needed to compensate for the -0.5f we
- // force when setting the transform
- x1 = (float)x + 0.5f;
- y1 = (float)y + 0.5f;
- x2 = x1 + (float)w;
- y2 = y1 + (float)h;
- D3DU_INIT_VERTEX_PENT_XY(lineVerts, x1, y1, x2, y2);
- D3DU_INIT_VERTEX_PENT_COLOR(lineVerts, d3dc->colorPixel);
-
- D3DU_PRIM_LOOP_BEGIN(res, wsdo);
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_RENDEROP))) {
- res = d3dDevice->DrawPrimitive(D3DPT_LINESTRIP, D3DFVF_J2D_XY_C,
- lineVerts, 5, 0);
- d3dc->EndScene(res);
- }
- D3DU_PRIM_LOOP_END(env, res, wsdo, "DrawPrimitive(D3DPT_LINESTRIP)");
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- }
- return SUCCEEDED(res);
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->DrawRect(x, y, x + w, y + h);
}
-/*
- * Class: sun_java2d_d3d_D3DRenderer
- * Method: doFillRectD3D
- * Signature: (JIIII)Z
- */
-JNIEXPORT jboolean JNICALL Java_sun_java2d_d3d_D3DRenderer_doFillRectD3D
- (JNIEnv *env, jobject d3dr,
- jlong pData, jlong pCtx,
- jint x, jint y, jint w, jint h)
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillRect(D3DContext *d3dc,
+ jint x, jint y, jint w, jint h)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
- HRESULT res = DDERR_GENERIC;
- float x1, y1, x2, y2;
- static J2D_XY_C_VERTEX quadVerts[] = {
-#ifdef USE_SINGLE_VERTEX_FORMAT
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f }
-#else
- // x, y, z, color
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
- { 0, 0, 0, 0x0 },
-#endif // USE_SINGLE_VERTEX_FORMAT
- };
+ J2dTraceLn4(J2D_TRACE_INFO,
+ "D3DRenderer_FillRect x=%-4d y=%-4d w=%-4d h=%-4d",
+ x, y, w, h);
- J2dTraceLn(J2D_TRACE_INFO, "doFillRectD3D");
- J2dTraceLn4(J2D_TRACE_VERBOSE, " x=%-4d y=%-4d w=%-4d h=%-4d", x, y, w, h);
-
- if (d3dc != NULL) {
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return FALSE;
- }
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return FALSE;
- }
-
- x1 = (float)x;
- y1 = (float)y;
- x2 = x1 + (float)w;
- y2 = y1 + (float)h;
- D3DU_INIT_VERTEX_QUAD_COLOR(quadVerts, d3dc->colorPixel);
- D3DU_INIT_VERTEX_QUAD_XY(quadVerts, x1, y1, x2, y2);
-
- D3DU_PRIM_LOOP_BEGIN(res, wsdo);
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_RENDEROP))) {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN, D3DFVF_J2D_XY_C,
- quadVerts, 4, 0);
- d3dc->EndScene(res);
- }
- D3DU_PRIM_LOOP_END(env, res, wsdo, "DrawPrimitive(D3DPT_TRIANGLEFAN)");
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- }
- return SUCCEEDED(res);
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->FillRect(x, y, x + w, y + h);
}
-/*
- * Class: sun_java2d_d3d_D3DRenderer
- * Method: doDrawPoly
- * Signature: (JII[I[IIZ)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DRenderer_doDrawPoly
- (JNIEnv *env, jobject d3dr,
- jlong pData, jlong pCtx,
- jint transx, jint transy,
- jintArray xcoordsArray, jintArray ycoordsArray, jint npoints,
- jboolean needToClose)
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawPoly(D3DContext *d3dc,
+ jint nPoints, jboolean isClosed,
+ jint transX, jint transY,
+ jint *xPoints, jint *yPoints)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
- jint *xcoords, *ycoords;
- jint i;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_DrawPoly");
- J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_doDrawPoly");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " transx=%-4d transy=%-4d "\
- "npoints=%-4d needToClose=%-4d",
- transx, transy, npoints, needToClose);
+ if (d3dc == NULL || xPoints == NULL || yPoints == NULL) {
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DRenderer_DrawPoly: d3dc, xPoints or yPoints is NULL");
+ return E_FAIL;
+ }
+
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->DrawPoly(nPoints, isClosed, transX, transY,
+ xPoints, yPoints);
+}
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawScanlines(D3DContext *d3dc,
+ jint scanlineCount, jint *scanlines)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_DrawScanlines");
if (d3dc == NULL) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "D3DRenderer_doDrawPoly: null device context");
- return;
+ return E_FAIL;
+ }
+ if (scanlines == NULL || scanlineCount <= 0) {
+ return D3D_OK;
}
- if (JNU_IsNull(env, xcoordsArray) || JNU_IsNull(env, ycoordsArray)) {
- JNU_ThrowNullPointerException(env, "coordinate array");
- return;
- }
- if (env->GetArrayLength(ycoordsArray) < npoints ||
- env->GetArrayLength(xcoordsArray) < npoints)
- {
- JNU_ThrowArrayIndexOutOfBoundsException(env, "coordinate array");
- return;
- }
-
- xcoords = (jint *)
- env->GetPrimitiveArrayCritical(xcoordsArray, NULL);
- if (xcoords == NULL) {
- return;
- }
-
- ycoords = (jint *)
- env->GetPrimitiveArrayCritical(ycoordsArray, NULL);
- if (ycoords == NULL) {
- env->ReleasePrimitiveArrayCritical(xcoordsArray, xcoords, JNI_ABORT);
- return;
- }
-
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- env->ReleasePrimitiveArrayCritical(ycoordsArray, ycoords, JNI_ABORT);
- env->ReleasePrimitiveArrayCritical(xcoordsArray, xcoords, JNI_ABORT);
- return;
- }
-
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- env->ReleasePrimitiveArrayCritical(ycoordsArray, ycoords, JNI_ABORT);
- env->ReleasePrimitiveArrayCritical(xcoordsArray, xcoords, JNI_ABORT);
- return;
- }
-
- int totalPoints = npoints;
- if (needToClose) {
- if (xcoords[npoints - 1] != xcoords[0] ||
- ycoords[npoints - 1] != ycoords[0])
- {
- totalPoints++;
- } else {
- needToClose = FALSE;
- }
- }
- J2D_XY_C_VERTEX *lpVerts =
- (J2D_XY_C_VERTEX *)safe_Malloc(totalPoints*sizeof(J2D_XY_C_VERTEX));
- if (!lpVerts) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- env->ReleasePrimitiveArrayCritical(ycoordsArray, ycoords, JNI_ABORT);
- env->ReleasePrimitiveArrayCritical(xcoordsArray, xcoords, JNI_ABORT);
- return;
- }
-
- ZeroMemory(lpVerts, totalPoints*sizeof(J2D_XY_C_VERTEX));
- for (i = 0; i < npoints; i++) {
- // +0.5f is needed to compensate for the -0.5f we
- // force when setting the transform
- lpVerts[i].x = (float)(xcoords[i] + transx) + 0.5f;
- lpVerts[i].y = (float)(ycoords[i] + transy) + 0.5f;
- lpVerts[i].color = d3dc->colorPixel;
- }
- if (needToClose) {
- lpVerts[npoints].x = (float)(xcoords[0] + transx) + 0.5f;
- lpVerts[npoints].y = (float)(ycoords[0] + transy) + 0.5f;
- lpVerts[npoints].color = d3dc->colorPixel;
- }
- env->ReleasePrimitiveArrayCritical(ycoordsArray, ycoords, JNI_ABORT);
- env->ReleasePrimitiveArrayCritical(xcoordsArray, xcoords, JNI_ABORT);
-
- HRESULT res;
- D3DU_PRIM_LOOP_BEGIN(res, wsdo);
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_RENDEROP))) {
- res = d3dDevice->DrawPrimitive(D3DPT_LINESTRIP, D3DFVF_J2D_XY_C,
- lpVerts, totalPoints, 0);
- // REMIND: temp hack, need to be using the results of device testing
- if (!needToClose) {
- res = d3dDevice->DrawPrimitive(D3DPT_POINTLIST, D3DFVF_J2D_XY_C,
- &(lpVerts[totalPoints-1]), 1, 0);
- }
- d3dc->EndScene(res);
- }
- D3DU_PRIM_LOOP_END(env, res, wsdo, "DrawPrimitive(D3DPT_LINESTRIP)");
-
- free(lpVerts);
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->DrawScanlines(scanlineCount, scanlines);
}
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillSpans(D3DContext *d3dc, jint spanCount, jint *spans)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_FillSpans");
+ if (d3dc == NULL) {
+ return E_FAIL;
+ }
+
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->FillSpans(spanCount, spans);
+}
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12)
+{
+ J2dTraceLn6(J2D_TRACE_INFO,
+ "D3DRenderer_FillParallelogram "
+ "x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f ",
+ fx11, fy11,
+ dx21, dy21,
+ dx12, dy12);
+
+ d3dc->BeginScene(STATE_RENDEROP);
+ return d3dc->pVCacher->FillParallelogram(fx11, fy11,
+ dx21, dy21,
+ dx12, dy12);
+}
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lwr21, jfloat lwr12)
+{
+ HRESULT res;
+
+ J2dTraceLn8(J2D_TRACE_INFO,
+ "D3DRenderer_DrawParallelogram "
+ "x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f lwr1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f lwr2=%6.2f ",
+ fx11, fy11,
+ dx21, dy21, lwr21,
+ dx12, dy12, lwr12);
+
+ // dx,dy for line width in the "21" and "12" directions.
+ jfloat ldx21 = dx21 * lwr21;
+ jfloat ldy21 = dy21 * lwr21;
+ jfloat ldx12 = dx12 * lwr12;
+ jfloat ldy12 = dy12 * lwr12;
+
+ // calculate origin of the outer parallelogram
+ jfloat ox11 = fx11 - (ldx21 + ldx12) / 2.0f;
+ jfloat oy11 = fy11 - (ldy21 + ldy12) / 2.0f;
+
+ res = d3dc->BeginScene(STATE_RENDEROP);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // Only need to generate 4 quads if the interior still
+ // has a hole in it (i.e. if the line width ratio was
+ // less than 1.0)
+ if (lwr21 < 1.0f && lwr12 < 1.0f) {
+ // Note: "TOP", "BOTTOM", "LEFT" and "RIGHT" here are
+ // relative to whether the dxNN variables are positive
+ // and negative. The math works fine regardless of
+ // their signs, but for conceptual simplicity the
+ // comments will refer to the sides as if the dxNN
+ // were all positive. "TOP" and "BOTTOM" segments
+ // are defined by the dxy21 deltas. "LEFT" and "RIGHT"
+ // segments are defined by the dxy12 deltas.
+
+ // Each segment includes its starting corner and comes
+ // to just short of the following corner. Thus, each
+ // corner is included just once and the only lengths
+ // needed are the original parallelogram delta lengths
+ // and the "line width deltas". The sides will cover
+ // the following relative territories:
+ //
+ // T T T T T R
+ // L R
+ // L R
+ // L R
+ // L R
+ // L B B B B B
+
+ // TOP segment, to left side of RIGHT edge
+ // "width" of original pgram, "height" of hor. line size
+ fx11 = ox11;
+ fy11 = oy11;
+ res = d3dc->pVCacher->FillParallelogram(fx11, fy11,
+ dx21, dy21,
+ ldx12, ldy12);
+
+ // RIGHT segment, to top of BOTTOM edge
+ // "width" of vert. line size , "height" of original pgram
+ fx11 = ox11 + dx21;
+ fy11 = oy11 + dy21;
+ res = d3dc->pVCacher->FillParallelogram(fx11, fy11,
+ ldx21, ldy21,
+ dx12, dy12);
+
+ // BOTTOM segment, from right side of LEFT edge
+ // "width" of original pgram, "height" of hor. line size
+ fx11 = ox11 + dx12 + ldx21;
+ fy11 = oy11 + dy12 + ldy21;
+ res = d3dc->pVCacher->FillParallelogram(fx11, fy11,
+ dx21, dy21,
+ ldx12, ldy12);
+
+ // LEFT segment, from bottom of TOP edge
+ // "width" of vert. line size , "height" of inner pgram
+ fx11 = ox11 + ldx12;
+ fy11 = oy11 + ldy12;
+ res = d3dc->pVCacher->FillParallelogram(fx11, fy11,
+ ldx21, ldy21,
+ dx12, dy12);
+ } else {
+ // The line width ratios were large enough to consume
+ // the entire hole in the middle of the parallelogram
+ // so we can just issue one large quad for the outer
+ // parallelogram.
+ dx21 += ldx21;
+ dy21 += ldy21;
+ dx12 += ldx12;
+ dy12 += ldy12;
+
+ res = d3dc->pVCacher->FillParallelogram(ox11, oy11,
+ dx21, dy21,
+ dx12, dy12);
+ }
+
+ return res;
+}
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillAAParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12)
+{
+ IDirect3DDevice9 *pd3dDevice;
+ HRESULT res;
+
+ J2dTraceLn6(J2D_TRACE_INFO,
+ "D3DRenderer_FillAAParallelogram "
+ "x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f ",
+ fx11, fy11,
+ dx21, dy21,
+ dx12, dy12);
+
+ res = d3dc->BeginScene(STATE_AAPGRAMOP);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pd3dDevice = d3dc->Get3DDevice();
+ if (pd3dDevice == NULL) {
+ return E_FAIL;
+ }
+
+ res = d3dc->pVCacher->FillParallelogramAA(fx11, fy11,
+ dx21, dy21,
+ dx12, dy12);
+ return res;
+}
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawAAParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lwr21, jfloat lwr12)
+{
+ IDirect3DDevice9 *pd3dDevice;
+ // dx,dy for line width in the "21" and "12" directions.
+ jfloat ldx21, ldy21, ldx12, ldy12;
+ // parameters for "outer" parallelogram
+ jfloat ofx11, ofy11, odx21, ody21, odx12, ody12;
+ // parameters for "inner" parallelogram
+ jfloat ifx11, ify11, idx21, idy21, idx12, idy12;
+ HRESULT res;
+
+ J2dTraceLn8(J2D_TRACE_INFO,
+ "D3DRenderer_DrawAAParallelogram "
+ "x=%6.2f y=%6.2f "
+ "dx1=%6.2f dy1=%6.2f lwr1=%6.2f "
+ "dx2=%6.2f dy2=%6.2f lwr2=%6.2f ",
+ fx11, fy11,
+ dx21, dy21, lwr21,
+ dx12, dy12, lwr12);
+
+ res = d3dc->BeginScene(STATE_AAPGRAMOP);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pd3dDevice = d3dc->Get3DDevice();
+ if (pd3dDevice == NULL) {
+ return E_FAIL;
+ }
+
+ // calculate true dx,dy for line widths from the "line width ratios"
+ ldx21 = dx21 * lwr21;
+ ldy21 = dy21 * lwr21;
+ ldx12 = dx12 * lwr12;
+ ldy12 = dy12 * lwr12;
+
+ // calculate coordinates of the outer parallelogram
+ ofx11 = fx11 - (ldx21 + ldx12) / 2.0f;
+ ofy11 = fy11 - (ldy21 + ldy12) / 2.0f;
+ odx21 = dx21 + ldx21;
+ ody21 = dy21 + ldy21;
+ odx12 = dx12 + ldx12;
+ ody12 = dy12 + ldy12;
+
+ // Only process the inner parallelogram if the line width ratio
+ // did not consume the entire interior of the parallelogram
+ // (i.e. if the width ratio was less than 1.0)
+ if (lwr21 < 1.0f && lwr12 < 1.0f) {
+ // calculate coordinates of the inner parallelogram
+ ifx11 = fx11 + (ldx21 + ldx12) / 2.0f;
+ ify11 = fy11 + (ldy21 + ldy12) / 2.0f;
+ idx21 = dx21 - ldx21;
+ idy21 = dy21 - ldy21;
+ idx12 = dx12 - ldx12;
+ idy12 = dy12 - ldy12;
+
+ res = d3dc->pVCacher->DrawParallelogramAA(ofx11, ofy11,
+ odx21, ody21,
+ odx12, ody12,
+ ifx11, ify11,
+ idx21, idy21,
+ idx12, idy12);
+ } else {
+ // Just invoke a regular fill on the outer parallelogram
+ res = d3dc->pVCacher->FillParallelogramAA(ofx11, ofy11,
+ odx21, ody21,
+ odx12, ody12);
+ }
+
+ return res;
+}
+
+#ifndef D3D_PPL_DLL
+
+extern "C"
+{
+
JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DRenderer_devFillSpans
+Java_sun_java2d_d3d_D3DRenderer_drawPoly
(JNIEnv *env, jobject d3dr,
- jlong pData, jlong pCtx,
- jobject si, jlong pIterator, jint transx, jint transy)
+ jintArray xpointsArray, jintArray ypointsArray,
+ jint nPoints, jboolean isClosed,
+ jint transX, jint transY)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
- SpanIteratorFuncs *pFuncs = (SpanIteratorFuncs *)jlong_to_ptr(pIterator);
- void *srData;
- float x1, y1, x2, y2;
- jint spanbox[4];
- J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_devFillSpans");
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- " transx=%-4d transy=%-4d", transx, transy);
+ jint *xPoints, *yPoints;
- if (JNU_IsNull(env, si)) {
- JNU_ThrowNullPointerException(env, "span iterator");
- return;
- }
- if (pFuncs == NULL) {
- JNU_ThrowNullPointerException(env, "native iterator not supplied");
- return;
- }
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRenderer_drawPoly");
- if (d3dc == NULL) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "D3DRenderer_devFillSpans: context is null");
- return;
- }
+ xPoints = (jint *)env->GetPrimitiveArrayCritical(xpointsArray, NULL);
+ if (xPoints != NULL) {
+ yPoints = (jint *)env->GetPrimitiveArrayCritical(ypointsArray, NULL);
+ if (yPoints != NULL) {
+ D3DContext *d3dc = D3DRQ_GetCurrentContext();
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return;
- }
+ D3DRenderer_DrawPoly(d3dc,
+ nPoints, isClosed,
+ transX, transY,
+ xPoints, yPoints);
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return;
- }
-
- HRESULT res = D3D_OK;
-
- // buffer for the span vertexes (6 vertexes per span)
- static J2DXYC_HEXA spanVx[MAX_CACHED_SPAN_VX_NUM];
- J2DXYC_HEXA *pHexa = (J2DXYC_HEXA*)spanVx;
- jint numOfCachedSpans = 0;
-
- if (SUCCEEDED(res = d3dc->BeginScene(STATE_RENDEROP))) {
- srData = (*pFuncs->open)(env, si);
-
- // REMIND: this is wrong, if something has failed, we need to
- // do a EndScene/BeginScene()
- D3DU_PRIM_LOOP_BEGIN(res, wsdo);
- while ((*pFuncs->nextSpan)(srData, spanbox)) {
- x1 = (float)(spanbox[0] + transx);
- y1 = (float)(spanbox[1] + transy);
- x2 = (float)(spanbox[2] + transx);
- y2 = (float)(spanbox[3] + transy);
-
- D3DU_INIT_VERTEX_COLOR_6(*pHexa, d3dc->colorPixel);
- D3DU_INIT_VERTEX_XY_6(*pHexa, x1, y1, x2, y2);
- numOfCachedSpans++;
- pHexa = (J2DXYC_HEXA*)PtrAddBytes(pHexa, sizeof(J2DXYC_HEXA));
- if (numOfCachedSpans >= MAX_CACHED_SPAN_VX_NUM) {
- if (FAILED(res = ddTargetSurface->IsLost())) {
- numOfCachedSpans = 0;
- break;
- }
-
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLELIST,
- D3DFVF_J2D_XY_C,
- (void*)spanVx,
- 6*numOfCachedSpans, 0);
- numOfCachedSpans = 0;
- pHexa = (J2DXYC_HEXA*)spanVx;
- if (FAILED(res)) {
- break;
- }
+ if (d3dc != NULL) {
+ HRESULT res = d3dc->EndScene();
+ D3DRQ_MarkLostIfNeeded(res,
+ D3DRQ_GetCurrentDestination());
}
+ env->ReleasePrimitiveArrayCritical(ypointsArray, yPoints, JNI_ABORT);
}
- if (numOfCachedSpans > 0) {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLELIST, D3DFVF_J2D_XY_C,
- (void*)spanVx,
- 6*numOfCachedSpans, 0);
- }
- D3DU_PRIM_LOOP_END(env, res, wsdo, "DrawPrimitive(D3DPT_TRIANGLEFAN)");
-
- (*pFuncs->close)(env, srData);
-
- d3dc->EndScene(res);
+ env->ReleasePrimitiveArrayCritical(xpointsArray, xPoints, JNI_ABORT);
}
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
}
+
+}
+
+#endif // D3D_PPL_DLL
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.h b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.h
new file mode 100644
index 00000000000..594b6ab322d
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DRenderer.h
@@ -0,0 +1,85 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "sun_java2d_pipe_BufferedRenderPipe.h"
+#include "D3DContext.h"
+
+#define BYTES_PER_POLY_POINT \
+ sun_java2d_pipe_BufferedRenderPipe_BYTES_PER_POLY_POINT
+#define BYTES_PER_SCANLINE \
+ sun_java2d_pipe_BufferedRenderPipe_BYTES_PER_SCANLINE
+#define BYTES_PER_SPAN \
+ sun_java2d_pipe_BufferedRenderPipe_BYTES_PER_SPAN
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawLine(D3DContext *d3dc,
+ jint x1, jint y1, jint x2, jint y2);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawRect(D3DContext *d3dc,
+ jint x, jint y, jint w, jint h);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillRect(D3DContext *d3dc,
+ jint x, jint y, jint w, jint h);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawPoly(D3DContext *d3dc,
+ jint nPoints, jboolean isClosed,
+ jint transX, jint transY,
+ jint *xPoints, jint *yPoints);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawScanlines(D3DContext *d3dc,
+ jint scanlineCount, jint *scanlines);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillSpans(D3DContext *d3dc, jint spanCount, jint *spans);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lw21, jfloat lw12);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_FillAAParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12);
+
+HRESULT D3DPIPELINE_API
+D3DRenderer_DrawAAParallelogram(D3DContext *d3dc,
+ jfloat fx11, jfloat fy11,
+ jfloat dx21, jfloat dy21,
+ jfloat dx12, jfloat dy12,
+ jfloat lw21, jfloat lw12);
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DResourceManager.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DResourceManager.cpp
new file mode 100644
index 00000000000..724d1a0161d
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DResourceManager.cpp
@@ -0,0 +1,765 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "D3DResourceManager.h"
+#include "awt.h"
+#include "D3DPaints.h"
+#include "D3DTextRenderer.h"
+
+void
+D3DResource::Init(IDirect3DResource9 *pRes, IDirect3DSwapChain9 *pSC)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DResource::Init");
+
+ pResource = NULL;
+ pSwapChain = pSC;
+ pSurface = NULL;
+ pTexture = NULL;
+ pOps = NULL;
+ ZeroMemory(&desc, sizeof(desc));
+ desc.Format = D3DFMT_UNKNOWN;
+
+ if (pRes != NULL) {
+ pResource = pRes;
+
+ D3DRESOURCETYPE type = pResource->GetType();
+ switch (type) {
+ case D3DRTYPE_TEXTURE:
+ // addRef is needed because both pResource and pTexture will be
+ // Release()d, and they point to the same object
+ pResource->AddRef();
+ pTexture = (IDirect3DTexture9*)pResource;
+ pTexture->GetSurfaceLevel(0, &pSurface);
+ break;
+ case D3DRTYPE_SURFACE:
+ pResource->AddRef();
+ pSurface = (IDirect3DSurface9*)pResource;
+ break;
+ case D3DRTYPE_CUBETEXTURE:
+ ((IDirect3DCubeTexture9*)pResource)->GetLevelDesc(0, &desc);
+ break;
+ default:
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " resource type=%d", type);
+ }
+ } else if (pSwapChain != NULL) {
+ pSwapChain->GetBackBuffer(0, D3DBACKBUFFER_TYPE_MONO, &pSurface);
+ } else {
+ J2dTraceLn(J2D_TRACE_VERBOSE, " pResource == pSwapChain == NULL");
+ }
+
+ if (pSurface != NULL) {
+ pSurface->GetDesc(&desc);
+ }
+
+ SAFE_PRINTLN(pResource);
+ SAFE_PRINTLN(pSurface);
+ SAFE_PRINTLN(pTexture);
+ SAFE_PRINTLN(pSwapChain);
+}
+
+D3DResource::~D3DResource()
+{
+ Release();
+}
+
+void
+D3DResource::SetSDOps(D3DSDOps *pOps)
+{
+ if (pOps != NULL && this->pOps != NULL) {
+ // something's wrong, we're overwriting
+ // a non-null field (setting it to null is allowed)
+ J2dTraceLn2(J2D_TRACE_WARNING,
+ "D3DResource::SetSDOps: overwriting "\
+ "this->pOps=0x%x with pOps=0x%x", this->pOps, pOps);
+ }
+ this->pOps = pOps;
+}
+
+BOOL
+D3DResource::IsDefaultPool()
+{
+ if (desc.Format != D3DFMT_UNKNOWN) {
+ return (desc.Pool == D3DPOOL_DEFAULT);
+ }
+ return TRUE;
+}
+
+void
+D3DResource::Release()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DResource::Release");
+
+ SAFE_PRINTLN(pResource);
+ SAFE_PRINTLN(pSurface);
+ SAFE_PRINTLN(pTexture);
+ SAFE_PRINTLN(pSwapChain);
+
+ SAFE_RELEASE(pSurface);
+ SAFE_RELEASE(pTexture);
+ SAFE_RELEASE(pResource);
+ SAFE_RELEASE(pSwapChain);
+
+ if (pOps != NULL) {
+ // if sdOps is not NULL it means that the release was initiated
+ // from the native level, and is caused by a surface loss
+ D3DSD_MarkLost(pOps);
+ pOps->pResource = NULL;
+ pOps = NULL;
+ }
+}
+
+HRESULT
+D3DResourceManager::CreateInstance(D3DContext *pCtx,
+ D3DResourceManager** ppResourceMgr)
+{
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::CreateInstance");
+
+ *ppResourceMgr = new D3DResourceManager();
+ if (FAILED(res = (*ppResourceMgr)->Init(pCtx))) {
+ delete *ppResourceMgr;
+ *ppResourceMgr = NULL;
+ }
+ return res;
+}
+
+D3DResourceManager::D3DResourceManager()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::D3DRM");
+
+ this->pCtx = NULL;
+ this->pHead = NULL;
+}
+
+HRESULT
+D3DResourceManager::Init(D3DContext *pCtx)
+{
+ J2dTraceLn1(J2D_TRACE_INFO, "D3DRM::Init pCtx=%x", pCtx);
+ if (this->pCtx != pCtx ||
+ (this->pCtx != NULL &&
+ this->pCtx->Get3DDevice() != pCtx->Get3DDevice()))
+ {
+ ReleaseAll();
+ }
+ this->pCtx = pCtx;
+ return S_OK;
+}
+
+D3DResourceManager::~D3DResourceManager()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::~D3DRM");
+ ReleaseAll();
+ pCtx = NULL;
+ pHead = NULL;
+}
+
+void
+D3DResourceManager::ReleaseAll()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::ReleaseAll");
+ IManagedResource* pCurrent;
+ while (pHead != NULL) {
+ pCurrent = pHead;
+ pHead = pHead->pNext;
+ delete pCurrent;
+ }
+ pCachedDestTexture = NULL;
+ pBlitTexture = NULL;
+ pBlitRTTexture = NULL;
+ pBlitOSPSurface = NULL;
+ pGradientTexture = NULL;
+ pLookupOpLutTexture = NULL;
+ pMaskTexture = NULL;
+ pMultiGradientTexture = NULL;
+ pLockableRTSurface = NULL;
+}
+
+void
+D3DResourceManager::ReleaseDefPoolResources()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::ReleaseDefPoolResources");
+ // REMIND: for now, release all resources
+ ReleaseAll();
+}
+
+HRESULT
+D3DResourceManager::ReleaseResource(IManagedResource* pResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::ReleaseResource");
+
+ if (pResource != NULL) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " releasing pResource=%x", pResource);
+ if (pResource->pPrev != NULL) {
+ pResource->pPrev->pNext = pResource->pNext;
+ } else {
+ // it's the head
+ pHead = pResource->pNext;
+ if (pHead != NULL) {
+ pHead->pPrev = NULL;
+ }
+ }
+ if (pResource->pNext != NULL) {
+ pResource->pNext->pPrev = pResource->pPrev;
+ }
+ delete pResource;
+ }
+ return S_OK;
+}
+
+HRESULT
+D3DResourceManager::AddResource(IManagedResource* pResource)
+{
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::AddResource");
+
+ if (pResource != NULL) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " pResource=%x", pResource);
+ pResource->pPrev = NULL;
+ pResource->pNext = pHead;
+ if (pHead != NULL) {
+ pHead->pPrev = pResource;
+ }
+ pHead = pResource;
+ }
+
+ return S_OK;
+}
+
+HRESULT
+D3DResourceManager::CreateTexture(UINT width, UINT height,
+ BOOL isRTT, BOOL isOpaque,
+ D3DFORMAT *pFormat, DWORD dwUsage,
+ D3DResource **ppTextureResource)
+{
+ D3DPOOL pool;
+ D3DFORMAT format;
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::CreateTexture");
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " w=%d h=%d isRTT=%d isOpaque=%d",
+ width, height, isRTT, isOpaque);
+
+ if (ppTextureResource == NULL || pCtx == NULL ||
+ (pd3dDevice = pCtx->Get3DDevice()) == NULL)
+ {
+ return E_FAIL;
+ }
+ if (FAILED(res = pd3dDevice->TestCooperativeLevel())) {
+ return res;
+ }
+
+ if (pFormat != NULL && *pFormat != D3DFMT_UNKNOWN) {
+ format = *pFormat;
+ } else {
+ if (isOpaque) {
+ format = D3DFMT_X8R8G8B8;
+ } else {
+ format = D3DFMT_A8R8G8B8;
+ }
+ }
+
+ if (isRTT) {
+ dwUsage = D3DUSAGE_RENDERTARGET;
+ pool = D3DPOOL_DEFAULT;
+ } else {
+ if (dwUsage == D3DUSAGE_DYNAMIC && !pCtx->IsDynamicTextureSupported()) {
+ dwUsage = 0;
+ }
+ if (dwUsage == D3DUSAGE_DYNAMIC) {
+ pool = D3DPOOL_DEFAULT;
+ } else {
+ pool = pCtx->IsHWRasterizer() ?
+ D3DPOOL_MANAGED : D3DPOOL_SYSTEMMEM;
+ }
+ }
+
+ if (pCtx->IsPow2TexturesOnly()) {
+ UINT w, h;
+ for (w = 1; width > w; w <<= 1);
+ for (h = 1; height > h; h <<= 1);
+ width = w;
+ height = h;
+ }
+ if (pCtx->IsSquareTexturesOnly()) {
+ if (width > height) {
+ height = width;
+ } else {
+ width = height;
+ }
+ }
+
+ IDirect3DTexture9 *pTexture = NULL;
+ res = pd3dDevice->CreateTexture(width, height, 1/*levels*/, dwUsage,
+ format, pool, &pTexture, 0);
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " created texture: 0x%x", pTexture);
+ *ppTextureResource = new D3DResource((IDirect3DResource9*)pTexture);
+ res = AddResource(*ppTextureResource);
+ } else {
+ DebugPrintD3DError(res, "D3DRM::CreateTexture failed");
+ *ppTextureResource = NULL;
+ format = D3DFMT_UNKNOWN;
+ }
+
+ if (pFormat != NULL) {
+ *pFormat = format;
+ }
+
+ return res;
+}
+
+HRESULT D3DResourceManager::CreateRTSurface(UINT width, UINT height,
+ BOOL isOpaque, BOOL isLockable,
+ D3DFORMAT *pFormat/*out*/,
+ D3DResource** ppSurfaceResource/*out*/)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::CreateRTSurface");
+ J2dTraceLn3(J2D_TRACE_VERBOSE, " w=%d h=%d isOpaque=%d",
+ width, height, isOpaque);
+
+ if (pCtx == NULL || ppSurfaceResource == NULL ||
+ (pd3dDevice = pCtx->Get3DDevice()) == NULL)
+ {
+ return E_FAIL;
+ }
+ if (FAILED(res = pd3dDevice->TestCooperativeLevel())) {
+ return res;
+ }
+
+ D3DPRESENT_PARAMETERS *curParams = pCtx->GetPresentationParams();
+ D3DFORMAT format = isOpaque ? curParams->BackBufferFormat : D3DFMT_A8R8G8B8;
+ IDirect3DSurface9 *pSurface = NULL;
+
+ res = pd3dDevice->CreateRenderTarget(width, height, format,
+ D3DMULTISAMPLE_NONE, 0,
+ isLockable,
+ &pSurface, NULL);
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " created RT Surface: 0x%x ", pSurface);
+ if (pFormat != NULL) {
+ *pFormat = format;
+ }
+ *ppSurfaceResource = new D3DResource((IDirect3DResource9*)pSurface);
+ res = AddResource(*ppSurfaceResource);
+ } else {
+ DebugPrintD3DError(res, "D3DRM::CreateRTSurface failed");
+ ppSurfaceResource = NULL;
+ }
+ return res;
+}
+
+// REMIND: this method is currently unused; consider removing it later...
+HRESULT D3DResourceManager::CreateOSPSurface(UINT width, UINT height,
+ D3DFORMAT fmt,
+ D3DResource** ppSurfaceResource/*out*/)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::CreateOSPSurface");
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " w=%d h=%d", width, height);
+
+ if (pCtx == NULL || ppSurfaceResource == NULL ||
+ (pd3dDevice = pCtx->Get3DDevice()) == NULL)
+ {
+ return E_FAIL;
+ }
+ if (FAILED(res = pd3dDevice->TestCooperativeLevel())) {
+ return res;
+ }
+
+ // since the off-screen plain surface is intended to be used with
+ // the UpdateSurface() method, it is essential that it be created
+ // in the same format as the destination and allocated in the
+ // SYSTEMMEM pool (otherwise UpdateSurface() will fail)
+ D3DFORMAT format;
+ if (fmt == D3DFMT_UNKNOWN) {
+ format = pCtx->GetPresentationParams()->BackBufferFormat;
+ } else {
+ format = fmt;
+ }
+ D3DPOOL pool = D3DPOOL_SYSTEMMEM;
+ IDirect3DSurface9 *pSurface = NULL;
+
+ res = pd3dDevice->CreateOffscreenPlainSurface(width, height,
+ format, pool,
+ &pSurface, NULL);
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " created OSP Surface: 0x%x ",pSurface);
+ *ppSurfaceResource = new D3DResource((IDirect3DResource9*)pSurface);
+ res = AddResource(*ppSurfaceResource);
+ } else {
+ DebugPrintD3DError(res, "D3DRM::CreateOSPSurface failed");
+ ppSurfaceResource = NULL;
+ }
+ return res;
+}
+
+HRESULT
+D3DResourceManager::CreateSwapChain(HWND hWnd, UINT numBuffers,
+ UINT width, UINT height,
+ D3DSWAPEFFECT swapEffect,
+ UINT presentationInterval,
+ D3DResource ** ppSwapChainResource)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice;
+ IDirect3DSwapChain9 *pSwapChain = NULL;
+ D3DPRESENT_PARAMETERS newParams, *curParams;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::CreateSwapChain");
+ J2dTraceLn4(J2D_TRACE_VERBOSE, " w=%d h=%d hwnd=%x numBuffers=%d",
+ width, height, hWnd, numBuffers);
+
+ if (pCtx == NULL || ppSwapChainResource == NULL ||
+ (pd3dDevice = pCtx->Get3DDevice()) == NULL)
+ {
+ return E_FAIL;
+ }
+ RETURN_STATUS_IF_FAILED(res = pd3dDevice->TestCooperativeLevel());
+
+ curParams = pCtx->GetPresentationParams();
+
+ if (curParams->Windowed == FALSE) {
+ // there's a single swap chain in full-screen mode, use it if
+ // it fits our parameters, reset the device otherwise
+ if (curParams->BackBufferCount != numBuffers ||
+ curParams->SwapEffect != swapEffect ||
+ curParams->PresentationInterval != presentationInterval)
+ {
+ newParams = *curParams;
+ newParams.BackBufferCount = numBuffers;
+ newParams.SwapEffect = swapEffect;
+ newParams.PresentationInterval = presentationInterval;
+
+ res = pCtx->ConfigureContext(&newParams);
+ RETURN_STATUS_IF_FAILED(res);
+ // this reset will not have released the device, so our pd3dDevice
+ // is still valid, but to be on a safe side, reset it
+ pd3dDevice = pCtx->Get3DDevice();
+ }
+ res = pd3dDevice->GetSwapChain(0, &pSwapChain);
+ } else {
+ ZeroMemory(&newParams, sizeof(D3DPRESENT_PARAMETERS));
+ newParams.BackBufferWidth = width;
+ newParams.BackBufferHeight = height;
+ newParams.hDeviceWindow = hWnd;
+ newParams.Windowed = TRUE;
+ newParams.BackBufferCount = numBuffers;
+ newParams.SwapEffect = swapEffect;
+ newParams.PresentationInterval = presentationInterval;
+
+ res = pd3dDevice->CreateAdditionalSwapChain(&newParams, &pSwapChain);
+ }
+
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE," created swap chain: 0x%x ",pSwapChain);
+ *ppSwapChainResource = new D3DResource(pSwapChain);
+ res = AddResource(*ppSwapChainResource);
+ } else {
+ DebugPrintD3DError(res, "D3DRM::CreateSwapChain failed");
+ *ppSwapChainResource = NULL;
+ }
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetMaskTexture(D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetMaskTexture");
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ D3DFORMAT format = pCtx->IsTextureFormatSupported(D3DFMT_A8) ?
+ D3DFMT_A8 : D3DFMT_A8R8G8B8;
+
+ jboolean needsInit = (pMaskTexture == NULL);
+ HRESULT res;
+ if (FAILED(res =
+ GetStockTextureResource(D3D_MASK_CACHE_WIDTH_IN_TEXELS,
+ D3D_MASK_CACHE_HEIGHT_IN_TEXELS,
+ FALSE/*isRTT*/, FALSE/*isOpaque*/, &format, 0,
+ &pMaskTexture)))
+ {
+ return res;
+ }
+
+ if (needsInit) {
+ // init special fully opaque tile in the upper-right corner of
+ // the mask cache texture
+ jubyte allOnes[D3D_MASK_CACHE_TILE_SIZE];
+ memset(allOnes, 0xff, D3D_MASK_CACHE_TILE_SIZE);
+ if (FAILED(res = pCtx->UploadTileToTexture(
+ pMaskTexture,
+ allOnes,
+ D3D_MASK_CACHE_SPECIAL_TILE_X,
+ D3D_MASK_CACHE_SPECIAL_TILE_Y,
+ 0, 0,
+ D3D_MASK_CACHE_TILE_WIDTH,
+ D3D_MASK_CACHE_TILE_HEIGHT,
+ D3D_MASK_CACHE_TILE_WIDTH,
+ TILEFMT_1BYTE_ALPHA)))
+ {
+ return res;
+ }
+ }
+
+ *ppTextureResource = pMaskTexture;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetBlitTexture(D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetBlitTexture");
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ HRESULT res =
+ GetStockTextureResource(D3DC_BLIT_TILE_SIZE, D3DC_BLIT_TILE_SIZE,
+ FALSE/*isRTT*/, FALSE/*isOpaque*/, NULL,
+ D3DUSAGE_DYNAMIC,
+ &pBlitTexture);
+ *ppTextureResource = pBlitTexture;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetGradientTexture(D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetGradientTexture");
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ HRESULT res =
+ GetStockTextureResource(2, 1,
+ FALSE/*isRTT*/, FALSE/*isOpaque*/, NULL, 0,
+ &pGradientTexture);
+ *ppTextureResource = pGradientTexture;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetMultiGradientTexture(D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetMultiGradientTexture");
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ HRESULT res =
+ GetStockTextureResource(MAX_MULTI_GRADIENT_COLORS, 1,
+ FALSE/*isRTT*/, FALSE/*isOpaque*/, NULL, 0,
+ &pMultiGradientTexture);
+ *ppTextureResource = pMultiGradientTexture;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetLookupOpLutTexture(D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetLookupOpTexture");
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ D3DFORMAT format = D3DFMT_L16;
+ HRESULT res =
+ GetStockTextureResource(256, 4,
+ FALSE/*isRTT*/, FALSE/*isOpaque*/, &format, 0,
+ &pLookupOpLutTexture);
+ *ppTextureResource = pLookupOpLutTexture;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetBlitRTTexture(UINT width, UINT height, D3DFORMAT format,
+ D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetBlitRTTexture");
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ HRESULT res = GetStockTextureResource(width, height,
+ TRUE/*isRTT*/, FALSE/*isOpaque*/,
+ &format, 0,
+ &pBlitRTTexture);
+ if (SUCCEEDED(res)) {
+ D3DSURFACE_DESC *pDesc = pBlitRTTexture->GetDesc();
+ D3DCAPS9 *pDevCaps = pCtx->GetDeviceCaps();
+ if ((width <= pDesc->Width && height <= pDesc->Height) &&
+ (format == pDesc->Format ||
+ SUCCEEDED(pCtx->Get3DObject()->CheckDeviceFormatConversion(
+ pDevCaps->AdapterOrdinal,
+ pDevCaps->DeviceType, format, pDesc->Format))))
+ {
+ *ppTextureResource = pBlitRTTexture;
+ return res;
+ }
+ // current texture doesn't fit, release and allocate a new one
+ ReleaseResource(pBlitRTTexture);
+ pBlitRTTexture = NULL;
+ }
+ if (width < D3DC_BLIT_TILE_SIZE) width = D3DC_BLIT_TILE_SIZE;
+ if (height < D3DC_BLIT_TILE_SIZE) height = D3DC_BLIT_TILE_SIZE;
+
+ res = CreateTexture(width, height, TRUE, FALSE, &format, 0,&pBlitRTTexture);
+ *ppTextureResource = pBlitRTTexture;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetBlitOSPSurface(UINT width, UINT height, D3DFORMAT fmt,
+ D3DResource **ppSurfaceResource)
+{
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetBlitOSPSurface");
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppSurfaceResource, E_FAIL);
+
+ if (pBlitOSPSurface != NULL) {
+ D3DSURFACE_DESC *pDesc = pBlitOSPSurface->GetDesc();
+ if (width == pDesc->Width && height == pDesc->Height &&
+ (fmt == pDesc->Format || fmt == D3DFMT_UNKNOWN))
+ {
+ *ppSurfaceResource = pBlitOSPSurface;
+ return res;
+ }
+ // current surface doesn't fit, release and allocate a new one
+ ReleaseResource(pBlitOSPSurface);
+ pBlitOSPSurface = NULL;
+ }
+
+ res = CreateOSPSurface(width, height, fmt, &pBlitOSPSurface);
+ *ppSurfaceResource = pBlitOSPSurface;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetLockableRTSurface(UINT width, UINT height,
+ D3DFORMAT format,
+ D3DResource **ppSurfaceResource)
+{
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetLockableRTSurface");
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppSurfaceResource, E_FAIL);
+
+ if (pLockableRTSurface != NULL) {
+ D3DSURFACE_DESC *pDesc = pLockableRTSurface->GetDesc();
+ if (width <= pDesc->Width && height <= pDesc->Height &&
+ format == pDesc->Format)
+ {
+ *ppSurfaceResource = pLockableRTSurface;
+ return res;
+ }
+ // current surface doesn't fit, release and allocate a new one
+ ReleaseResource(pLockableRTSurface);
+ pLockableRTSurface = NULL;
+ }
+ if (width < D3DC_BLIT_TILE_SIZE) width = D3DC_BLIT_TILE_SIZE;
+ if (height < D3DC_BLIT_TILE_SIZE) height = D3DC_BLIT_TILE_SIZE;
+
+ res = CreateRTSurface(width,height,
+ (format != D3DFMT_A8R8G8B8), TRUE /*lockable*/,
+ &format, &pLockableRTSurface);
+ *ppSurfaceResource = pLockableRTSurface;
+
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetCachedDestTexture(D3DFORMAT format,
+ D3DResource **ppTextureResource)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DRM::GetCachedDestTexture");
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+ RETURN_STATUS_IF_NULL(ppTextureResource, E_FAIL);
+
+ HRESULT res =
+ GetStockTextureResource(D3DTR_CACHED_DEST_WIDTH,
+ D3DTR_CACHED_DEST_HEIGHT,
+ TRUE/*isRTT*/, FALSE/*isOpaque*/,
+ &format, 0, &pCachedDestTexture);
+ if (SUCCEEDED(res)) {
+ D3DSURFACE_DESC *pDesc = pCachedDestTexture->GetDesc();
+ D3DCAPS9 *pDevCaps = pCtx->GetDeviceCaps();
+ if ((format == pDesc->Format ||
+ SUCCEEDED(pCtx->Get3DObject()->CheckDeviceFormatConversion(
+ pDevCaps->AdapterOrdinal,
+ pDevCaps->DeviceType, format, pDesc->Format))))
+ {
+ *ppTextureResource = pCachedDestTexture;
+ return res;
+ }
+ // current texture doesn't fit, release and allocate a new one
+ ReleaseResource(pCachedDestTexture);
+ pCachedDestTexture = NULL;
+ }
+ res = CreateTexture(D3DTR_CACHED_DEST_WIDTH, D3DTR_CACHED_DEST_HEIGHT,
+ TRUE, FALSE, &format, 0,
+ &pCachedDestTexture);
+ *ppTextureResource = pCachedDestTexture;
+ return res;
+}
+
+HRESULT
+D3DResourceManager::GetStockTextureResource(UINT width, UINT height,
+ BOOL isRTT, BOOL isOpaque,
+ D3DFORMAT *pFormat/*in/out*/,
+ DWORD dwUsage,
+ D3DResource **ppTextureResource)
+{
+ D3DResource *pResource = *ppTextureResource;
+ if (pResource != NULL) {
+ if (pResource->GetTexture() != NULL) {
+ return S_OK;
+ }
+ ReleaseResource(pResource);
+ *ppTextureResource = NULL;
+ }
+
+ return CreateTexture(width, height, isRTT, isOpaque, pFormat, dwUsage,
+ ppTextureResource);
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DResourceManager.h b/jdk/src/windows/native/sun/java2d/d3d/D3DResourceManager.h
new file mode 100644
index 00000000000..87b8d8ccc6b
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DResourceManager.h
@@ -0,0 +1,214 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+
+#ifndef _D3DRESOURCEMANAGER_H_
+#define _D3DRESOURCEMANAGER_H_
+
+#include "D3DContext.h"
+#include "D3DSurfaceData.h"
+
+class D3DResourceManager;
+class D3DContext;
+
+/**
+ * This interface represents a Direct3D resource which is managed by the
+ * D3DResourceManager.
+ *
+ * Subclasses will need to override Release() and the destructor to release
+ * the resources held by the object.
+ *
+ * The subclasses then can be used like this:
+ * class D3DShaderResource : public IManagedResource {
+ * D3DShaderResource(IDirect3DPixelShader9 *pShader) { // ... }
+ * virtual ~D3DShaderResource() { Release(); }
+ * virtual Release() { SAFE_RELEASE(pShader); }
+ * virtual IsDefaultPool() { return FALSE; }
+ * private:
+ * IDirect3DPixelShader9 *pShader;
+ * }
+ *
+ * pD3DDevice->CreatePixelShader(..., &pShader);
+ * IManagedResource *pShaderRes = new D3DShaderResource(pShader);
+ * pCtx->GetResourceManager()->AddResource(pShaderRes);
+ *
+ * D3DResourceManager::ReleaseResource() must be used to dispose of the
+ * resource:
+ * pCtx->GetResourceManager()->ReleaseResource(pShaderRes);
+ * // pShaderRes is now invalid, it was deleted
+ * shaderRes = NULL;
+ *
+ * In certain cases the D3DResourceManager may need to release all its
+ * resources (like when resetting the device), so the subclasses must be
+ * ready to be released at any time, and be able to notify their users.
+ * For an example of how this can be achieved see how D3DSDO's
+ * pResource field and D3DResource subclass. d3dsdo->pResource is reset when
+ * the D3DResource it was pointing to is disposed.
+ */
+class IManagedResource {
+friend class D3DResourceManager;
+public:
+ // determines whether the resource should be released by the manager
+ // when defaul pool resources are to be released
+ virtual BOOL IsDefaultPool() = 0;
+protected:
+ IManagedResource() { pPrev = pNext = NULL; };
+ virtual ~IManagedResource() { pPrev = pNext = NULL; };
+ virtual void Release() = 0;
+private:
+ // prevents accidental bad things like copying the object
+ IManagedResource& operator=(const IManagedResource&);
+
+ IManagedResource* pPrev;
+ IManagedResource* pNext;
+};
+
+/**
+ * This class handles either IDirect3DResource9 or IDirect3DSwapChain9
+ * type of resources and provides access to Texture, Surface or SwapChain,
+ * as well as the surface description.
+ */
+class D3DResource : public IManagedResource {
+public:
+ D3DResource(IDirect3DResource9 *pRes)
+ { Init(pRes, NULL); }
+ D3DResource(IDirect3DSwapChain9 *pSC)
+ { Init(NULL, pSC); }
+ IDirect3DResource9* GetResource() { return pResource; }
+ IDirect3DTexture9* GetTexture() { return pTexture; }
+ IDirect3DSurface9* GetSurface() { return pSurface; }
+ IDirect3DSwapChain9* GetSwapChain() { return pSwapChain; }
+ D3DSDOps* GetSDOps() { return pOps; }
+ void SetSDOps(D3DSDOps *pOps);
+ D3DSURFACE_DESC* GetDesc() { return &desc; }
+ virtual BOOL IsDefaultPool();
+
+protected:
+ // these are protected because we want D3DResource to be only released via
+ // ResourceManager
+virtual ~D3DResource();
+virtual void Release();
+ void Init(IDirect3DResource9*, IDirect3DSwapChain9*);
+
+private:
+ // prevents accidental bad things like copying the object
+ D3DResource() {}
+ D3DResource& operator=(const D3DResource&);
+
+ IDirect3DResource9* pResource;
+ IDirect3DSwapChain9* pSwapChain;
+ IDirect3DSurface9* pSurface;
+ IDirect3DTexture9* pTexture;
+ D3DSDOps* pOps;
+ D3DSURFACE_DESC desc;
+};
+
+/**
+ * This class maintains a list of d3d resources created by the pipeline or
+ * other clients. It is needed because in some cases all resources have to be
+ * released in order to reset the device so we must keep track of them.
+ *
+ * There is one instance of this class per D3DContext. Clients can either
+ * use factory methods for creating resources or create their own encapsulated
+ * in an IManagedResource interface subclass and add them to the list
+ * using the AddResource() method. Resources added to the list must be released
+ * via the ReleaseResource() method so that they can be stopped being managed.
+ */
+class D3DResourceManager {
+
+public:
+ ~D3DResourceManager();
+ HRESULT Init(D3DContext *pCtx);
+ // Releases and deletes all resources managed by this manager.
+ void ReleaseAll();
+ // Releases (and deletes) all resources belonging to the default pool.
+ // Note: this method may release other resources as well.
+ void ReleaseDefPoolResources();
+
+ // Adds the resource to the list managed by this class.
+ HRESULT AddResource(IManagedResource* pResource);
+ // Removes the resource from the list of managed resources, and deletes
+ // it. The argument pointer is invalid after this method returns.
+ HRESULT ReleaseResource(IManagedResource* pResource);
+
+ HRESULT CreateTexture(UINT width, UINT height,
+ BOOL isRTT, BOOL isOpaque,
+ D3DFORMAT *pFormat/*in/out*/,
+ DWORD dwUsage,
+ D3DResource **ppTextureResource/*out*/);
+
+ HRESULT CreateRTSurface(UINT width, UINT height,
+ BOOL isOpaque, BOOL isLockable,
+ D3DFORMAT *pFormat/*in/out*/,
+ D3DResource ** ppSurfaceResource/*out*/);
+
+ HRESULT CreateSwapChain(HWND hWnd, UINT numBuffers, UINT width, UINT height,
+ D3DSWAPEFFECT swapEffect, UINT presentationInterval,
+ D3DResource ** ppSwapChainResource/*out*/);
+
+ HRESULT GetCachedDestTexture(D3DFORMAT format,
+ D3DResource **ppTextureResource);
+ HRESULT GetBlitTexture(D3DResource **ppTextureResource);
+ HRESULT GetBlitRTTexture(UINT width, UINT height, D3DFORMAT format,
+ D3DResource **ppTextureResource);
+ HRESULT GetBlitOSPSurface(UINT width, UINT height, D3DFORMAT fmt,
+ D3DResource **ppSurfaceResource);
+ HRESULT GetMaskTexture(D3DResource **ppTextureResource);
+ HRESULT GetGradientTexture(D3DResource **ppTextureResource);
+ HRESULT GetMultiGradientTexture(D3DResource **ppTextureResource);
+ HRESULT GetLookupOpLutTexture(D3DResource **ppTextureResource);
+ HRESULT GetLockableRTSurface(UINT width, UINT height, D3DFORMAT format,
+ D3DResource **ppSurfaceResource);
+
+static
+ HRESULT CreateInstance(D3DContext *pCtx, D3DResourceManager **ppResMgr);
+
+private:
+ D3DResourceManager();
+ HRESULT GetStockTextureResource(UINT width, UINT height,
+ BOOL isRTT, BOOL isOpaque,
+ D3DFORMAT *pFormat/*in/out*/,
+ DWORD dwUsage,
+ D3DResource **ppTextureResource/*out*/);
+
+ HRESULT CreateOSPSurface(UINT width, UINT height,
+ D3DFORMAT fmt,
+ D3DResource ** ppSurfaceResource/*out*/);
+
+ D3DResource* pCachedDestTexture;
+ D3DResource* pBlitTexture;
+ D3DResource* pBlitRTTexture;
+ D3DResource* pBlitOSPSurface;
+ D3DResource* pGradientTexture;
+ D3DResource* pLookupOpLutTexture;
+ D3DResource* pMaskTexture;
+ D3DResource* pMultiGradientTexture;
+ D3DResource* pLockableRTSurface;
+
+ D3DContext* pCtx;
+
+ IManagedResource* pHead;
+};
+#endif _D3DRESOURCEMANAGER_H_
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DRuntimeTest.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DRuntimeTest.cpp
deleted file mode 100644
index 46aebdb80a5..00000000000
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DRuntimeTest.cpp
+++ /dev/null
@@ -1,537 +0,0 @@
-/*
- * Copyright 2005-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include "dxInit.h"
-#include "ddrawUtils.h"
-#include "RegistryKey.h"
-#include "D3DTestRaster.h"
-#include "WindowsFlags.h"
-#include "D3DRuntimeTest.h"
-#include "D3DSurfaceData.h"
-#include "D3DUtils.h"
-
-#ifdef DEBUG
-void TestRasterOutput(byte *rasPtr, int x, int y, int w, int h,
- int scanStride, int pixelStride,
- TIntTestRaster goldenArray = NULL);
-#endif // DEBUG
-void PrintD3DCaps(int caps);
-
-/**
- * Test whether we should enable d3d rendering on this device.
- * This includes checking whether there were problems creating
- * the necessary offscreen surface, problems during any of the
- * rendering calls (Blts and d3d lines) and any rendering artifacts
- * caused by d3d lines. The rendering artifact tests are
- * performed by checking a pre-rendered test pattern (produced
- * by our software renderer) against that same pattern rendered
- * on this device. If there are any pixels which differ between
- * the two patterns we disable d3d line rendering on the device.
- * Differences in the test pattern rendering can be caused
- * by different rendering algorithms used by our software
- * renderer and the driver or hardware on this device. For example,
- * some Intel cards (e.g., i815) are known to use polygon renderers
- * for their lines, which sometimes result in wide lines.
- * The test pattern is stored in d3dTestRaster.h, which is generated
- * by a Java test program
- * (src/share/test/java2d/VolatileImage/D3DTestPattern/D3DTestPattern.java).
- */
-
-int TestForBadHardware(DxCapabilities *dxCaps)
-{
- // Check this device against a list of bad d3d devices and
- // disable as necessary
- static WCHAR *badDeviceStrings[] = {
- L"Trident Video Accelerator",
- L"RAGE PRO",
- L"RAGE XL",
- L"Rage Fury",
- };
- static int numBadDevices = 4;
- WCHAR *dxDeviceName = dxCaps->GetDeviceName();
- for (int i = 0; i < numBadDevices; ++i) {
- if (wcsstr(dxDeviceName, badDeviceStrings[i]) != NULL) {
- // REMIND: For now, we disable d3d for all operations because
- // of one bad d3d device in the system. This is because we
- // should avoid registering the d3d rendering loops at the
- // Java level since we cannot use d3d at the native level.
- // A real fix would instead understand the difference between
- // a surface that could handle d3d native rendering and one
- // that could not and would use the appropriate rendering loop
- // so that disabling d3d on simply one device would be
- // sufficient.
- // Note that this disable-all approach is okay for now because
- // the single bad device (Trident) that triggers this error
- // is generally found on laptops, where multiple graphics
- // devices are not even possible, so disabling d3d for all
- // devices is equivalent to disabling d3d for this single
- // device.
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "TestForBadHardware: Found match: %S. Test FAILED",
- badDeviceStrings[i]);
- return J2D_D3D_FAILURE;
- }
- }
- return J2D_D3D_HW_OK;
-}
-
-int TestTextureFormats(D3DContext *d3dContext)
-{
- int testRes = J2D_D3D_FAILURE;
-
- D3DTextureTable &table = d3dContext->GetTextureTable();
- int pfExists;
- // Check that there's at least one valid pixel format
- // for each transparency type (opaque, bitmask, translucent)
- for (int t = TR_OPAQUE_IDX; t < TR_MAX_IDX; t++) {
- pfExists = FALSE;
- for (int d = DEPTH16_IDX; d < DEPTH_MAX_IDX; d++) {
- if (table[t][d].pfType != PF_INVALID) {
- pfExists = TRUE;
- break;
- }
- }
- if (pfExists == FALSE) {
- // couldn't find a pixel formap for this transparency type
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "D3DTest::TestTextureFormats no texture formats"\
- " for %d transparency", t);
- break;
- }
- }
-
- // we must have ARGB texture format (may be used for text rendering)
- if (pfExists == TRUE &&
- table[TR_TRANSLUCENT_IDX][DEPTH32_IDX].pfType == PF_INT_ARGB)
- {
- testRes |= J2D_D3D_PIXEL_FORMATS_OK;
- } else {
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "D3DTest::TestTextureFormats: FAILED pfType=%d",
- table[TR_TRANSLUCENT_IDX][DEPTH32_IDX].pfType);
- }
- return testRes;
-}
-
-int TestSetClip(JNIEnv *env, D3DContext *d3dContext,
- DDrawSurface *lpPlainSurface)
-{
- int testRes = J2D_D3D_FAILURE;
-
- if (SUCCEEDED(d3dContext->SetRenderTarget(lpPlainSurface))) {
- jobject clip =
- JNU_CallStaticMethodByName(env, NULL,
- "sun/java2d/pipe/Region",
- "getInstanceXYWH",
- "(IIII)Lsun/java2d/pipe/Region;",
- 0, 0, D3D_TEST_RASTER_W, D3D_TEST_RASTER_H).l;
- if (!JNU_IsNull(env, clip)) {
- if (SUCCEEDED(d3dContext->SetClip(env, clip, JNI_TRUE,
- 0, 0,
- D3D_TEST_RASTER_W,
- D3D_TEST_RASTER_H)))
- {
- testRes |= J2D_D3D_DEPTH_SURFACE_OK;
- }
- env->DeleteLocalRef(clip);
- }
- }
- return testRes;
-}
-
-int TestRenderingResults(DDrawSurface *lpPlainSurface,
- TIntTestRaster goldenArray)
-{
- // Now, check the results of the test raster against our d3d drawing
- SurfaceDataRasInfo rasInfo;
- if (FAILED(lpPlainSurface->Lock(NULL, &rasInfo, DDLOCK_WAIT, NULL))) {
- return J2D_D3D_FAILURE;
- }
-
- byte *rasPtr = (byte*)rasInfo.rasBase;
- int pixelStride = rasInfo.pixelStride;
- int scanStride = rasInfo.scanStride;
- for (int row = 0; row < D3D_TEST_RASTER_H; ++row) {
- byte *tmpRasPtr = rasPtr + row * scanStride;
- for (int col = 0; col < D3D_TEST_RASTER_W; ++col) {
- DWORD pixelVal;
- switch (pixelStride) {
- case 1:
- pixelVal = *tmpRasPtr;
- break;
- case 2:
- pixelVal = *((unsigned short*)tmpRasPtr);
- break;
- default:
- pixelVal = *((unsigned int*)tmpRasPtr);
- break;
- }
- tmpRasPtr += pixelStride;
- // The test is simple: if the test raster pixel has value 0, then
- // we expect 0 in the d3d surface. If the test raster has a nonzero
- // value, then we expect the d3d surface to also have non-zero value.
- // All other results represent failure.
- int goldenValue = (goldenArray[row][col] & 0x00ffffff);
- if ((goldenValue == 0 && pixelVal != 0) ||
- (goldenValue != 0 && pixelVal == 0))
- {
- J2dRlsTraceLn3(J2D_TRACE_WARNING,
- "TestRenderingResults: Quality test failed due "\
- "to value %x at (%d, %d)", pixelVal, col, row);
-#ifdef DEBUG
- // This section is not necessary, but it might be
- // nice to know why we are failing D3DTest on some
- // systems. If tracing is enabled, this section will
- // produce an ascii representation of the test pattern,
- // the result on this device, and the pixels that were
- // in error.
- J2dTraceLn(J2D_TRACE_VERBOSE, "TestRaster:");
- TestRasterOutput((byte*)goldenArray, 0, 0, D3D_TEST_RASTER_W,
- D3D_TEST_RASTER_H, D3D_TEST_RASTER_W*4, 4);
- J2dTraceLn(J2D_TRACE_VERBOSE, "D3D Raster:");
- TestRasterOutput(rasPtr, 0, 0, D3D_TEST_RASTER_W,
- D3D_TEST_RASTER_H, scanStride, pixelStride);
- J2dTraceLn(J2D_TRACE_VERBOSE, "Deltas (x indicates problem pixel):");
- TestRasterOutput(rasPtr, 0, 0, D3D_TEST_RASTER_W,
- D3D_TEST_RASTER_H, scanStride, pixelStride,
- goldenArray);
-#endif // DEBUG
- lpPlainSurface->Unlock(NULL);
- return J2D_D3D_FAILURE;
- }
- }
- }
-
- lpPlainSurface->Unlock(NULL);
- return (J2D_D3D_LINES_OK | J2D_D3D_LINE_CLIPPING_OK);
-}
-
-int TestLineRenderingQuality(JNIEnv *env, D3DContext *d3dContext,
- DDrawSurface *lpPlainSurface)
-{
- static J2D_XY_C_VERTEX lineVerts[] = {
-#ifdef USE_SINGLE_VERTEX_FORMAT
- { 0, 0, 0, 0xffffffff, 0.0f, 0.0f },
- { 0, 0, 0, 0xffffffff, 0.0f, 0.0f },
- { 0, 0, 0, 0xffffffff, 0.0f, 0.0f },
- { 0, 0, 0, 0xffffffff, 0.0f, 0.0f },
- { 0, 0, 0, 0xffffffff, 0.0f, 0.0f },
-#else
- { 0, 0, 0, 0xffffffff }, // x, y, z, color
- { 0, 0, 0, 0xffffffff },
- { 0, 0, 0, 0xffffffff },
- { 0, 0, 0, 0xffffffff },
- { 0, 0, 0, 0xffffffff },
-#endif // USE_SINGLE_VERTEX_FORMAT
- };
- IDirect3DDevice7 *d3dDevice = d3dContext->Get3DDevice();
- HRESULT res;
-
- d3dDevice->Clear(0, NULL, D3DCLEAR_TARGET, 0x0, 0.0, 0);
-
- if (FAILED(d3dContext->BeginScene(STATE_RENDEROP))) {
- return J2D_D3D_FAILURE;
- }
-
- int i;
-
- for (i = 0; i < d3dNumTestLines * 4; i += 4) {
- lineVerts[0].x = d3dTestLines[i + 0];
- lineVerts[0].y = d3dTestLines[i + 1];
- lineVerts[1].x = d3dTestLines[i + 2];
- lineVerts[1].y = d3dTestLines[i + 3];
- if (FAILED(res = d3dDevice->DrawPrimitive(D3DPT_LINESTRIP,
- D3DFVF_J2D_XY_C,
- lineVerts, 2, 0)))
- {
- d3dContext->ForceEndScene();
- return J2D_D3D_FAILURE;
- }
- // REMIND: needed for the test to pass on ATI some boards
- d3dDevice->DrawPrimitive(D3DPT_POINTLIST, D3DFVF_J2D_XY_C,
- &(lineVerts[1]), 1, 0);
- }
-
- for (i = 0; i < d3dNumTestRects * 4; i += 4) {
- float x1 = d3dTestRects[i + 0];
- float y1 = d3dTestRects[i + 1];
- float x2 = d3dTestRects[i + 2];
- float y2 = d3dTestRects[i + 3];
- D3DU_INIT_VERTEX_PENT_XY(lineVerts, x1, y1, x2, y2);
- if (FAILED(res = d3dDevice->DrawPrimitive(D3DPT_LINESTRIP,
- D3DFVF_J2D_XY_C,
- lineVerts, 5, 0)))
- {
- d3dContext->ForceEndScene();
- return J2D_D3D_FAILURE;
- }
- }
- d3dContext->ForceEndScene();
-
- // REMIND: add rendering of clipped lines
-
- return TestRenderingResults(lpPlainSurface, d3dTestRaster);
-}
-
-int TestTextureMappingQuality(JNIEnv *env, DDraw *ddObject,
- D3DContext *d3dContext,
- DDrawSurface *lpPlainSurface)
-{
- static J2DLVERTEX quadVerts[4] = {
- { 0.0f, 0.0f, 0.0f, 0xffffffff, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0xffffffff, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0xffffffff, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0xffffffff, 0.0f, 0.0f }
- };
-
- int testRes = TestTextureFormats(d3dContext);
-
- if (testRes & J2D_D3D_PIXEL_FORMATS_OK) {
-
- DDrawSurface *lpTexture =
- D3DUtils_CreateTexture(env, ddObject, d3dContext, TR_TRANSLUCENT,
- D3D_TEXTURE_RASTER_W, D3D_TEXTURE_RASTER_H);
- if (lpTexture) {
- D3DUtils_UploadIntImageToXRGBTexture(lpTexture,
- (int *)srcImageArray,
- D3D_TEXTURE_RASTER_W,
- D3D_TEXTURE_RASTER_H);
-
- float u2 = ((float)D3D_TEXTURE_RASTER_W) /
- (float)lpTexture->GetDXSurface()->GetWidth();
- float v2 = ((float)D3D_TEXTURE_RASTER_H) /
- (float)lpTexture->GetDXSurface()->GetHeight();
- D3DU_INIT_VERTEX_QUAD_UV(quadVerts, 0.0f, 0.0f, u2, v2);
-
- IDirect3DDevice7 *d3dDevice = d3dContext->Get3DDevice();
- d3dDevice->Clear(0, NULL, D3DCLEAR_TARGET, 0x00000000, 0.0, 0);
-
- d3dContext->SetAlphaComposite(3/*SrcOver*/,
- 1.0f, D3DC_NO_CONTEXT_FLAGS);
- d3dDevice->SetTextureStageState(0, D3DTSS_MAGFILTER, D3DTFG_POINT);
- d3dDevice->SetTextureStageState(0, D3DTSS_MINFILTER, D3DTFG_POINT);
-
- HRESULT res;
- if (SUCCEEDED(res = d3dContext->BeginScene(STATE_BLITOP))) {
- DXSurface *dxSurface = lpTexture->GetDXSurface();
- if (SUCCEEDED(d3dContext->SetTexture(dxSurface))) {
- for (int i = 0; i < d3dNumTextureRects * 4; i += 4) {
- float x1 = d3dTextureRects[i + 0];
- float y1 = d3dTextureRects[i + 1];
- float x2 = d3dTextureRects[i + 2];
- float y2 = d3dTextureRects[i + 3];
- D3DU_INIT_VERTEX_QUAD_XY(quadVerts, x1, y1, x2, y2);
- d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN,
- D3DFVF_J2DLVERTEX,
- quadVerts, 4, 0);
- }
- }
- res = d3dContext->ForceEndScene();
- d3dContext->SetTexture(NULL);
- }
- // REMIND: at this point we ignore the results of
- // the test.
- TestRenderingResults(lpPlainSurface, linInterpArray);
- if (SUCCEEDED(res)) {
- testRes |= (J2D_D3D_TR_TEXTURE_SURFACE_OK |
- J2D_D3D_TEXTURE_BLIT_OK |
- J2D_D3D_TEXTURE_TRANSFORM_OK);
-
- // REMIND: add tests for opaque and bitmask textures
- testRes |= (J2D_D3D_OP_TEXTURE_SURFACE_OK |
- J2D_D3D_BM_TEXTURE_SURFACE_OK);
- }
- delete lpTexture;
- } else {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "TestTextureMappingQuality: "\
- "CreateTexture(TRANSLUCENT) FAILED");
- }
- }
- return testRes;
-}
-
-int TestD3DDevice(DDraw *ddObject,
- D3DContext *d3dContext,
- DxCapabilities *dxCaps)
-{
- int testRes = TestForBadHardware(dxCaps);
- if (!(testRes & J2D_D3D_HW_OK) || !d3dContext) {
- return testRes;
- }
-
- D3DDEVICEDESC7 d3dDevDesc;
- IDirect3DDevice7 *d3dDevice = d3dContext->Get3DDevice();
- if (d3dDevice == NULL ||
- FAILED(d3dDevice->GetCaps(&d3dDevDesc)) ||
- FAILED(D3DUtils_CheckDeviceCaps(&d3dDevDesc)))
- {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "TestD3DDevice: device caps testing FAILED");
- return testRes;
- }
- testRes |= J2D_D3D_DEVICE_OK;
-
- JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
- DDrawSurface *lpPlainSurface =
- D3DUtils_CreatePlainSurface(env, ddObject, d3dContext,
- D3D_TEST_RASTER_W, D3D_TEST_RASTER_H);
- if (!lpPlainSurface) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "TestD3DDevice: CreatePlainSurface FAILED");
- return testRes;
- }
- testRes |= J2D_D3D_PLAIN_SURFACE_OK;
-
- // Set identity transform
- if (FAILED(d3dContext->SetTransform(NULL, 0, 0, 0, 0, 0, 0))) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "TestD3DDevice: SetTransform FAILED");
- delete lpPlainSurface;
- return testRes;
- }
- testRes |= J2D_D3D_SET_TRANSFORM_OK;
-
- // Test setting the target surface, create depth buffer, and
- // clip
- testRes |= TestSetClip(env, d3dContext, lpPlainSurface);
- if (!(testRes & J2D_D3D_DEPTH_SURFACE_OK)) {
- J2dRlsTraceLn(J2D_TRACE_ERROR, "TestD3DDevice: SetClip FAILED");
- delete lpPlainSurface;
- return testRes;
- }
-
- // Test drawLines
- testRes |= TestLineRenderingQuality(env, d3dContext, lpPlainSurface);
-
- // Test texture mapping
- testRes |= TestTextureMappingQuality(env, ddObject, d3dContext,
- lpPlainSurface);
-
- d3dContext->SetRenderTarget(NULL);
-
- delete lpPlainSurface;
- return testRes;
-}
-
-#ifdef DEBUG
-/**
- * Output test raster (produced in D3DTest function). Utility
- * used in debugging only. Enable by setting J2D_TRACE_LEVEL=J2D_VERBOSE
- * prior to running application with debug java. The output from this will
- * be seen only if D3DTest fails.
- */
-void TestRasterOutput(byte *rasPtr, int x, int y, int w, int h,
- int scanStride, int pixelStride,
- TIntTestRaster goldenArray)
-{
- int goldenValue;
- for (int traceRow = y; traceRow < h; ++traceRow) {
- byte *tmpRasPtr = rasPtr + traceRow * scanStride;
- for (int traceCol = x; traceCol < w; ++traceCol) {
- DWORD pixelVal;
- switch (pixelStride) {
- case 1:
- pixelVal = *tmpRasPtr;
- break;
- case 2:
- pixelVal = *((unsigned short*)tmpRasPtr);
- break;
- default:
- pixelVal = *((unsigned int*)tmpRasPtr) & 0x00ffffff;
- break;
- }
- tmpRasPtr += pixelStride;
- if (goldenArray == NULL) {
- if (pixelVal) {
- J2dTrace(J2D_TRACE_VERBOSE, "1");
- } else {
- J2dTrace(J2D_TRACE_VERBOSE, "0");
- }
- } else {
- goldenValue = (goldenArray[traceRow][traceCol] & 0x00ffffff);
- if ((goldenValue == 0 && pixelVal != 0) ||
- (goldenValue != 0 && pixelVal == 0))
- {
- J2dTrace(J2D_TRACE_VERBOSE, "x");
- } else {
- J2dTrace(J2D_TRACE_VERBOSE, "-");
- }
- }
-
- }
- J2dTrace(J2D_TRACE_VERBOSE, "\n");
- }
-}
-#endif // DEBUG
-
-void PrintD3DCaps(int caps)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "{")
- if (caps == J2D_D3D_FAILURE) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_FAILURE");
- } else {
- if (caps & J2D_D3D_DEPTH_SURFACE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_DEPTH_SURFACE_OK,");
- }
- if (caps & J2D_D3D_PLAIN_SURFACE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_PLAIN_SURFACE_OK,");
- }
- if (caps & J2D_D3D_OP_TEXTURE_SURFACE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_OP_TEXTURE_SURFACE_OK,");
- }
- if (caps & J2D_D3D_BM_TEXTURE_SURFACE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_BM_TEXTURE_SURFACE_OK,");
- }
- if (caps & J2D_D3D_TR_TEXTURE_SURFACE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_TR_TEXTURE_SURFACE_OK,");
- }
- if (caps & J2D_D3D_OP_RTT_SURFACE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_OP_RTT_SURFACE_OK,");
- }
- if (caps & J2D_D3D_LINE_CLIPPING_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_LINE_CLIPPING_OK,");
- }
- if (caps & J2D_D3D_LINES_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_LINES_OK,");
- }
- if (caps & J2D_D3D_TEXTURE_BLIT_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_TEXTURE_BLIT_OK,");
- }
- if (caps & J2D_D3D_TEXTURE_TRANSFORM_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_TEXTURE_TRANSFORM_OK,");
- }
- if (caps & J2D_D3D_DEVICE_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_DEVICE_OK,");
- }
- if (caps & J2D_D3D_PIXEL_FORMATS_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_SET_TRANSFORM_OK,");
- }
- if (caps & J2D_D3D_HW_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " J2D_D3D_HW_OK,");
- }
- }
- J2dTraceLn(J2D_TRACE_VERBOSE, "}");
-}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DRuntimeTest.h b/jdk/src/windows/native/sun/java2d/d3d/D3DRuntimeTest.h
deleted file mode 100644
index c33496b45c7..00000000000
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DRuntimeTest.h
+++ /dev/null
@@ -1,82 +0,0 @@
-/*
- * Copyright 2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#ifndef D3DRUNTIMETEST_H
-#define D3DRUNTIMETEST_H
-
-#include "ddrawObject.h"
-#include "dxCapabilities.h"
-#include "D3DContext.h"
-
-/*
- * This is a minimum set of capabilities required for
- * enabling D3D pipeline. If any of these is
- * missing, d3d will be disabled completely.
- *
- * This set is used if the use of d3d pipeline is
- * forced via flag or env. variable.
- */
-#define J2D_D3D_REQUIRED_RESULTS ( \
- J2D_D3D_HW_OK | \
- J2D_D3D_DEVICE_OK | \
- J2D_D3D_DEPTH_SURFACE_OK | \
- J2D_D3D_PLAIN_SURFACE_OK | \
- J2D_D3D_PIXEL_FORMATS_OK | \
- J2D_D3D_OP_TEXTURE_SURFACE_OK | \
- J2D_D3D_TR_TEXTURE_SURFACE_OK | \
- J2D_D3D_SET_TRANSFORM_OK)
-
-/*
- * This is a set of capabilities desired for
- * enabling D3D pipeline. It includes the set
- * of required caps, plus a number of rendering
- * quality related caps.
- *
- * This is the set of caps checked by default when
- * deciding on whether to enable the d3d pipeline.
- */
-#define J2D_D3D_DESIRED_RESULTS ( \
- J2D_D3D_REQUIRED_RESULTS | \
- J2D_D3D_BM_TEXTURE_SURFACE_OK | \
- J2D_D3D_TEXTURE_BLIT_OK | \
- J2D_D3D_TEXTURE_TRANSFORM_OK | \
- J2D_D3D_LINES_OK | \
- J2D_D3D_LINE_CLIPPING_OK)
-
-
-/*
- * This function tests the direct3d device associated
- * with the passed ddraw object.
- *
- * The function returns the capabilities of the tested device, and the
- * results of the quality testing.
- * Enabling the d3d pipeline for this particular device is based on the
- * result of this function.
- */
-int TestD3DDevice(DDraw *ddObject, D3DContext *d3dContext, DxCapabilities *dxCaps);
-
-void PrintD3DCaps(int caps);
-
-#endif //D3DRUNTIMETEST_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DShaderGen.c b/jdk/src/windows/native/sun/java2d/d3d/D3DShaderGen.c
new file mode 100644
index 00000000000..69cb2989380
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DShaderGen.c
@@ -0,0 +1,985 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/**
+ * This file contains a standalone program that is used to generate the
+ * D3DShaders.h file. The program invokes the fxc (D3D Shader Compiler)
+ * utility, which is part of the DirectX 9/10 SDK. Since most JDK
+ * developers (other than some Java 2D engineers) do not have the full DXSDK
+ * installed, and since we do not want to make the JDK build process
+ * dependent on the full DXSDK installation, we have chosen not to make
+ * this shader compilation step part of the build process. Instead, it is
+ * only necessary to compile and run this program when changes need to be
+ * made to the shader code contained within. Typically, this only happens
+ * on an as-needed basis by someone familiar with the D3D pipeline. Running
+ * this program is fairly straightforward:
+ *
+ * % rm D3DShaders.h
+ * % cl D3DShaderGen.c
+ * % D3DShaderGen.exe
+ *
+ * (And don't forget to putback the updated D3DShaders.h file!)
+ */
+
+#include
+#include
+#include
+
+static FILE *fpHeader = NULL;
+static char *strHeaderFile = "D3DShaders.h";
+
+/** Evaluates to true if the given bit is set on the local flags variable. */
+#define IS_SET(flagbit) \
+ (((flags) & (flagbit)) != 0)
+
+// REMIND
+//#define J2dTraceLn(a, b) fprintf(stderr, "%s\n", b);
+//#define J2dTraceLn1(a, b, c) fprintf(stderr, b, c);
+#define J2dTraceLn(a, b)
+#define J2dTraceLn1(a, b, c)
+
+/************************* General shader support ***************************/
+
+static void
+D3DShaderGen_WriteShader(char *source, char *target, char *name, int flags)
+{
+ FILE *fpTmp;
+ char varname[50];
+ char *args[8];
+ int val;
+
+ // write source to tmp.hlsl
+ fpTmp = fopen("tmp.hlsl", "w");
+ fprintf(fpTmp, "%s\n", source);
+ fclose(fpTmp);
+
+ {
+ PROCESS_INFORMATION pi;
+ STARTUPINFO si;
+ char pargs[300];
+ sprintf(pargs,
+ "c:\\progra~1\\mi5889~1\\utilit~1\\bin\\x86\\fxc.exe "
+ "/T %s /Vn %s%d /Fh tmp.h tmp.hlsl",
+ // uncomment the following line to generate debug
+ // info in the shader header file (may be useful
+ // for testing/debuggging purposes, but it nearly
+ // doubles the size of the header file and compiled
+ // shader programs - off for production builds)
+ //"/Zi /T %s /Vn %s%d /Fh tmp.h tmp.hlsl",
+ target, name, flags);
+ fprintf(stderr, "%s\n", pargs);
+ memset(&si, 0, sizeof(si));
+ si.cb = sizeof(si);
+ si.dwFlags = STARTF_USESTDHANDLES;
+ //si.hStdOutput = GetStdHandle(STD_OUTPUT_HANDLE);
+ //fprintf(stderr, "%s\n", pargs);
+ val = CreateProcess(0, pargs, 0, 0, TRUE,
+ CREATE_NO_WINDOW, NULL, NULL, &si, &pi);
+
+ {
+ DWORD code;
+ do {
+ GetExitCodeProcess(pi.hProcess, &code);
+ //fprintf(stderr, "waiting...");
+ Sleep(100);
+ } while (code == STILL_ACTIVE);
+
+ if (code != 0) {
+ fprintf(stderr, "fxc failed for %s%d\n", name, flags);
+ }
+ }
+
+ CloseHandle(pi.hThread);
+ CloseHandle(pi.hProcess);
+ }
+
+ // append tmp.h to D3DShaders.h
+ {
+ int ch;
+ fpTmp = fopen("tmp.h", "r");
+ while ((ch = fgetc(fpTmp)) != EOF) {
+ fputc(ch, fpHeader);
+ }
+ fclose(fpTmp);
+ }
+}
+
+static void
+D3DShaderGen_WritePixelShader(char *source, char *name, int flags)
+{
+ D3DShaderGen_WriteShader(source, "ps_2_0", name, flags);
+}
+
+#define MULTI_GRAD_CYCLE_METHOD (3 << 0)
+/** Extracts the CycleMethod enum value from the given flags variable. */
+#define EXTRACT_CYCLE_METHOD(flags) \
+ ((flags) & MULTI_GRAD_CYCLE_METHOD)
+
+static void
+D3DShaderGen_WriteShaderArray(char *name, int num)
+{
+ char array[5000];
+ char elem[30];
+ int i;
+
+ sprintf(array, "const DWORD *%sShaders[] =\n{\n", name);
+ for (i = 0; i < num; i++) {
+ if (num == 32 && EXTRACT_CYCLE_METHOD(i) == 3) {
+ // REMIND: what a hack!
+ sprintf(elem, " NULL,\n");
+ } else {
+ sprintf(elem, " %s%d,\n", name, i);
+ }
+ strcat(array, elem);
+ }
+ strcat(array, "};\n");
+
+ // append to D3DShaders.h
+ fprintf(fpHeader, "%s\n", array);
+}
+
+/**************************** ConvolveOp support ****************************/
+
+static const char *convolveShaderSource =
+ // image to be convolved
+ "sampler2D baseImage : register(s0);"
+ // image edge limits:
+ // imgEdge.xy = imgMin.xy (anything < will be treated as edge case)
+ // imgEdge.zw = imgMax.xy (anything > will be treated as edge case)
+ "float4 imgEdge : register(c0);"
+ // value for each location in the convolution kernel:
+ // kernelVals[i].x = offsetX[i]
+ // kernelVals[i].y = offsetY[i]
+ // kernelVals[i].z = kernel[i]
+ "float3 kernelVals[%d] : register(c1);"
+ ""
+ "void main(in float2 tc : TEXCOORD0,"
+ " inout float4 color : COLOR0)"
+ "{"
+ " float4 sum = imgEdge - tc.xyxy;"
+ ""
+ " if (sum.x > 0 || sum.y > 0 || sum.z < 0 || sum.w < 0) {"
+ // (placeholder for edge condition code)
+ " color = %s;"
+ " } else {"
+ " int i;"
+ " sum = float4(0, 0, 0, 0);"
+ " for (i = 0; i < %d; i++) {"
+ " sum +="
+ " kernelVals[i].z *"
+ " tex2D(baseImage, tc + kernelVals[i].xy);"
+ " }"
+ // modulate with current color in order to apply extra alpha
+ " color *= sum;"
+ " }"
+ ""
+ "}";
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define CONVOLVE_EDGE_ZERO_FILL (1 << 0)
+#define CONVOLVE_5X5 (1 << 1)
+#define MAX_CONVOLVE (1 << 2)
+
+static void
+D3DShaderGen_GenerateConvolveShader(int flags)
+{
+ int kernelMax = IS_SET(CONVOLVE_5X5) ? 25 : 9;
+ char *edge;
+ char finalSource[2000];
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DShaderGen_GenerateConvolveShader: flags=%d",
+ flags);
+
+ if (IS_SET(CONVOLVE_EDGE_ZERO_FILL)) {
+ // EDGE_ZERO_FILL: fill in zero at the edges
+ edge = "float4(0, 0, 0, 0)";
+ } else {
+ // EDGE_NO_OP: use the source pixel color at the edges
+ edge = "tex2D(baseImage, tc)";
+ }
+
+ // compose the final source code string from the various pieces
+ sprintf(finalSource, convolveShaderSource,
+ kernelMax, edge, kernelMax);
+
+ D3DShaderGen_WritePixelShader(finalSource, "convolve", flags);
+}
+
+/**************************** RescaleOp support *****************************/
+
+static const char *rescaleShaderSource =
+ // image to be rescaled
+ "sampler2D baseImage : register(s0);"
+ // vector containing scale factors
+ "float4 scaleFactors : register(c0);"
+ // vector containing offsets
+ "float4 offsets : register(c1);"
+ ""
+ "void main(in float2 tc : TEXCOORD0,"
+ " inout float4 color : COLOR0)"
+ "{"
+ " float4 srcColor = tex2D(baseImage, tc);"
+ ""
+ // (placeholder for un-premult code)
+ " %s"
+ ""
+ // rescale source value
+ " float4 result = (srcColor * scaleFactors) + offsets;"
+ ""
+ // (placeholder for re-premult code)
+ " %s"
+ ""
+ // modulate with current color in order to apply extra alpha
+ " color *= result;"
+ "}";
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define RESCALE_NON_PREMULT (1 << 0)
+#define MAX_RESCALE (1 << 1)
+
+static void
+D3DShaderGen_GenerateRescaleShader(int flags)
+{
+ char *preRescale = "";
+ char *postRescale = "";
+ char finalSource[2000];
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DShaderGen_GenerateRescaleShader: flags=%d",
+ flags);
+
+ if (IS_SET(RESCALE_NON_PREMULT)) {
+ preRescale = "srcColor.rgb /= srcColor.a;";
+ postRescale = "result.rgb *= result.a;";
+ }
+
+ // compose the final source code string from the various pieces
+ sprintf(finalSource, rescaleShaderSource,
+ preRescale, postRescale);
+
+ D3DShaderGen_WritePixelShader(finalSource, "rescale", flags);
+}
+
+/**************************** LookupOp support ******************************/
+
+static const char *lookupShaderSource =
+ // source image (bound to texture unit 0)
+ "sampler2D baseImage : register(s0);"
+ // lookup table (bound to texture unit 1)
+ "sampler2D lookupTable : register(s1);"
+ // offset subtracted from source index prior to lookup step
+ "float4 offset : register(c0);"
+ ""
+ "void main(in float2 tc : TEXCOORD0,"
+ " inout float4 color : COLOR0)"
+ "{"
+ " float4 srcColor = tex2D(baseImage, tc);"
+ // (placeholder for un-premult code)
+ " %s"
+ // subtract offset from original index
+ " float4 srcIndex = srcColor - offset;"
+ // use source value as input to lookup table (note that
+ // "v" texcoords are hardcoded to hit texel centers of
+ // each row/band in texture)
+ " float4 result;"
+ " result.r = tex2D(lookupTable, float2(srcIndex.r, 0.125)).r;"
+ " result.g = tex2D(lookupTable, float2(srcIndex.g, 0.375)).r;"
+ " result.b = tex2D(lookupTable, float2(srcIndex.b, 0.625)).r;"
+ // (placeholder for alpha store code)
+ " %s"
+ // (placeholder for re-premult code)
+ " %s"
+ // modulate with current color in order to apply extra alpha
+ " color *= result;"
+ "}";
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define LOOKUP_USE_SRC_ALPHA (1 << 0)
+#define LOOKUP_NON_PREMULT (1 << 1)
+#define MAX_LOOKUP (1 << 2)
+
+static void
+D3DShaderGen_GenerateLookupShader(int flags)
+{
+ char *alpha;
+ char *preLookup = "";
+ char *postLookup = "";
+ char finalSource[2000];
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DShaderGen_GenerateLookupShader: flags=%d",
+ flags);
+
+ if (IS_SET(LOOKUP_USE_SRC_ALPHA)) {
+ // when numComps is 1 or 3, the alpha is not looked up in the table;
+ // just keep the alpha from the source fragment
+ alpha = "result.a = srcColor.a;";
+ } else {
+ // when numComps is 4, the alpha is looked up in the table, just
+ // like the other color components from the source fragment
+ alpha = "result.a = tex2D(lookupTable, float2(srcIndex.a, 0.875)).r;";
+ }
+ if (IS_SET(LOOKUP_NON_PREMULT)) {
+ preLookup = "srcColor.rgb /= srcColor.a;";
+ postLookup = "result.rgb *= result.a;";
+ }
+
+ // compose the final source code string from the various pieces
+ sprintf(finalSource, lookupShaderSource,
+ preLookup, alpha, postLookup);
+
+ D3DShaderGen_WritePixelShader(finalSource, "lookup", flags);
+}
+
+/************************* GradientPaint support ****************************/
+
+/*
+ * To simplify the code and to make it easier to upload a number of
+ * uniform values at once, we pack a bunch of scalar (float) values
+ * into a single float3 below. Here's how the values are related:
+ *
+ * params.x = p0
+ * params.y = p1
+ * params.z = p3
+ */
+static const char *basicGradientShaderSource =
+ "float3 params : register (c0);"
+ "float4 color1 : register (c1);"
+ "float4 color2 : register (c2);"
+ // (placeholder for mask variable)
+ "%s"
+ ""
+ // (placeholder for mask texcoord input)
+ "void main(%s"
+ " in float4 winCoord : TEXCOORD%d,"
+ " inout float4 color : COLOR0)"
+ "{"
+ " float3 fragCoord = float3(winCoord.x, winCoord.y, 1.0);"
+ " float dist = dot(params.xyz, fragCoord);"
+ ""
+ // the setup code for p0/p1/p3 translates/scales to hit texel
+ // centers (at 0.25 and 0.75) because it is needed for the
+ // original/fast texture-based implementation, but it is not
+ // desirable for this shader-based implementation, so we
+ // re-transform the value here...
+ " dist = (dist - 0.25) * 2.0;"
+ ""
+ " float fraction;"
+ // (placeholder for cycle code)
+ " %s"
+ ""
+ " float4 result = lerp(color1, color2, fraction);"
+ ""
+ // (placeholder for mask modulation code)
+ " %s"
+ ""
+ // modulate with current color in order to apply extra alpha
+ " color *= result;"
+ "}";
+
+/**
+ * Flags that can be bitwise-or'ed together to control how the shader
+ * source code is generated.
+ */
+#define BASIC_GRAD_IS_CYCLIC (1 << 0)
+#define BASIC_GRAD_USE_MASK (1 << 1)
+#define MAX_BASIC_GRAD (1 << 2)
+
+static void
+D3DShaderGen_GenerateBasicGradShader(int flags)
+{
+ int colorSampler = IS_SET(BASIC_GRAD_USE_MASK) ? 1 : 0;
+ char *cycleCode;
+ char *maskVars = "";
+ char *maskInput = "";
+ char *maskCode = "";
+ char finalSource[3000];
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DShaderGen_GenerateBasicGradShader",
+ flags);
+
+ if (IS_SET(BASIC_GRAD_IS_CYCLIC)) {
+ cycleCode =
+ "fraction = 1.0 - (abs(frac(dist * 0.5) - 0.5) * 2.0);";
+ } else {
+ cycleCode =
+ "fraction = clamp(dist, 0.0, 1.0);";
+ }
+
+ if (IS_SET(BASIC_GRAD_USE_MASK)) {
+ /*
+ * This code modulates the calculated result color with the
+ * corresponding alpha value from the alpha mask texture active
+ * on texture unit 0. Only needed when useMask is true (i.e., only
+ * for MaskFill operations).
+ */
+ maskVars = "sampler2D mask : register(s0);";
+ maskInput = "in float4 maskCoord : TEXCOORD0,";
+ maskCode = "result *= tex2D(mask, maskCoord.xy).a;";
+ }
+
+ // compose the final source code string from the various pieces
+ sprintf(finalSource, basicGradientShaderSource,
+ maskVars, maskInput, colorSampler, cycleCode, maskCode);
+
+ D3DShaderGen_WritePixelShader(finalSource, "grad", flags);
+}
+
+/****************** Shared MultipleGradientPaint support ********************/
+
+/**
+ * These constants are identical to those defined in the
+ * MultipleGradientPaint.CycleMethod enum; they are copied here for
+ * convenience (ideally we would pull them directly from the Java level,
+ * but that entails more hassle than it is worth).
+ */
+#define CYCLE_NONE 0
+#define CYCLE_REFLECT 1
+#define CYCLE_REPEAT 2
+
+/**
+ * The following constants are flags that can be bitwise-or'ed together
+ * to control how the MultipleGradientPaint shader source code is generated:
+ *
+ * MULTI_GRAD_CYCLE_METHOD
+ * Placeholder for the CycleMethod enum constant.
+ *
+ * MULTI_GRAD_LARGE
+ * If set, use the (slower) shader that supports a larger number of
+ * gradient colors; otherwise, use the optimized codepath. See
+ * the MAX_FRACTIONS_SMALL/LARGE constants below for more details.
+ *
+ * MULTI_GRAD_USE_MASK
+ * If set, apply the alpha mask value from texture unit 1 to the
+ * final color result (only used in the MaskFill case).
+ *
+ * MULTI_GRAD_LINEAR_RGB
+ * If set, convert the linear RGB result back into the sRGB color space.
+ */
+//#define MULTI_GRAD_CYCLE_METHOD (3 << 0)
+#define MULTI_GRAD_LARGE (1 << 2)
+#define MULTI_GRAD_USE_MASK (1 << 3)
+#define MULTI_GRAD_LINEAR_RGB (1 << 4)
+
+// REMIND
+#define MAX_MULTI_GRAD (1 << 5)
+
+/** Extracts the CycleMethod enum value from the given flags variable. */
+//#define EXTRACT_CYCLE_METHOD(flags) \
+// ((flags) & MULTI_GRAD_CYCLE_METHOD)
+
+/**
+ * The maximum number of gradient "stops" supported by the fragment shader
+ * and related code. When the MULTI_GRAD_LARGE flag is set, we will use
+ * MAX_FRACTIONS_LARGE; otherwise, we use MAX_FRACTIONS_SMALL. By having
+ * two separate values, we can have one highly optimized shader (SMALL) that
+ * supports only a few fractions/colors, and then another, less optimal
+ * shader that supports more stops.
+ */
+#define MAX_FRACTIONS 8
+#define MAX_FRACTIONS_LARGE MAX_FRACTIONS
+#define MAX_FRACTIONS_SMALL 4
+
+/**
+ * The maximum number of gradient colors supported by all of the gradient
+ * fragment shaders. Note that this value must be a power of two, as it
+ * determines the size of the 1D texture created below. It also must be
+ * greater than or equal to MAX_FRACTIONS (there is no strict requirement
+ * that the two values be equal).
+ */
+#define MAX_COLORS 16
+
+static const char *multiGradientShaderSource =
+ // gradient texture size (in texels)
+ "#define TEXTURE_SIZE %d\n"
+ // maximum number of fractions/colors supported by this shader
+ "#define MAX_FRACTIONS %d\n"
+ // size of a single texel
+ "#define FULL_TEXEL (1.0 / float(TEXTURE_SIZE))\n"
+ // size of half of a single texel
+ "#define HALF_TEXEL (FULL_TEXEL / 2.0)\n"
+ // texture containing the gradient colors
+ "sampler2D colors : register (s%d);"
+ // array of gradient stops/fractions and corresponding scale factors
+ // fractions[i].x = gradientStop[i]
+ // fractions[i].y = scaleFactor[i]
+ "float2 fractions[MAX_FRACTIONS] : register (c0);"
+ // (placeholder for mask variable)
+ "%s"
+ // (placeholder for Linear/RadialGP-specific variables)
+ "%s"
+ ""
+ // (placeholder for mask texcoord input)
+ "void main(%s"
+ " in float4 winCoord : TEXCOORD%d,"
+ " inout float4 color : COLOR0)"
+ "{"
+ " float dist;"
+ // (placeholder for Linear/RadialGradientPaint-specific code)
+ " %s"
+ ""
+ " float4 result;"
+ // (placeholder for CycleMethod-specific code)
+ " %s"
+ ""
+ // (placeholder for ColorSpace conversion code)
+ " %s"
+ ""
+ // (placeholder for mask modulation code)
+ " %s"
+ ""
+ // modulate with current color in order to apply extra alpha
+ " color *= result;"
+ "}";
+
+/*
+ * Note: An earlier version of this code would simply calculate a single
+ * texcoord:
+ * "tc = HALF_TEXEL + (FULL_TEXEL * relFraction);"
+ * and then use that value to do a single texture lookup, taking advantage
+ * of the LINEAR texture filtering mode which in theory will do the
+ * appropriate linear interpolation between adjacent texels, like this:
+ * "float4 result = tex2D(colors, float2(tc, 0.5));"
+ *
+ * The problem with that approach is that on certain hardware (from ATI,
+ * notably) the LINEAR texture fetch unit has low precision, and would
+ * for instance only produce 64 distinct grayscales between white and black,
+ * instead of the expected 256. The visual banding caused by this issue
+ * is severe enough to likely cause complaints from developers, so we have
+ * devised a new approach below that instead manually fetches the two
+ * relevant neighboring texels and then performs the linear interpolation
+ * using the lerp() instruction (which does not suffer from the precision
+ * issues of the fixed-function texture filtering unit). This new approach
+ * requires a few more instructions and is therefore slightly slower than
+ * the old approach (not more than 10% or so).
+ */
+static const char *texCoordCalcCode =
+ "int i;"
+ "float relFraction = 0.0;"
+ "for (i = 0; i < MAX_FRACTIONS-1; i++) {"
+ " relFraction +="
+ " clamp((dist - fractions[i].x) * fractions[i].y, 0.0, 1.0);"
+ "}"
+ // we offset by half a texel so that we find the linearly interpolated
+ // color between the two texel centers of interest
+ "float intPart = floor(relFraction);"
+ "float tc1 = HALF_TEXEL + (FULL_TEXEL * intPart);"
+ "float tc2 = HALF_TEXEL + (FULL_TEXEL * (intPart + 1.0));"
+ "float4 clr1 = tex2D(colors, float2(tc1, 0.5));"
+ "float4 clr2 = tex2D(colors, float2(tc2, 0.5));"
+ "result = lerp(clr1, clr2, frac(relFraction));";
+
+/** Code for NO_CYCLE that gets plugged into the CycleMethod placeholder. */
+static const char *noCycleCode =
+ "if (dist <= 0.0) {"
+ " result = tex2D(colors, float2(0.0, 0.5));"
+ "} else if (dist >= 1.0) {"
+ " result = tex2D(colors, float2(1.0, 0.5));"
+ "} else {"
+ // (placeholder for texcoord calculation)
+ " %s"
+ "}";
+
+/** Code for REFLECT that gets plugged into the CycleMethod placeholder. */
+static const char *reflectCode =
+ "dist = 1.0 - (abs(frac(dist * 0.5) - 0.5) * 2.0);"
+ // (placeholder for texcoord calculation)
+ "%s";
+
+/** Code for REPEAT that gets plugged into the CycleMethod placeholder. */
+static const char *repeatCode =
+ "dist = frac(dist);"
+ // (placeholder for texcoord calculation)
+ "%s";
+
+static void
+D3DShaderGen_GenerateMultiGradShader(int flags, char *name,
+ char *paintVars, char *distCode)
+{
+ char *maskVars = "";
+ char *maskInput = "";
+ char *maskCode = "";
+ char *colorSpaceCode = "";
+ char cycleCode[1500];
+ char finalSource[3000];
+ int colorSampler = IS_SET(MULTI_GRAD_USE_MASK) ? 1 : 0;
+ int cycleMethod = EXTRACT_CYCLE_METHOD(flags);
+ int maxFractions = IS_SET(MULTI_GRAD_LARGE) ?
+ MAX_FRACTIONS_LARGE : MAX_FRACTIONS_SMALL;
+
+ J2dTraceLn(J2D_TRACE_INFO, "OGLPaints_CreateMultiGradProgram");
+
+ if (IS_SET(MULTI_GRAD_USE_MASK)) {
+ /*
+ * This code modulates the calculated result color with the
+ * corresponding alpha value from the alpha mask texture active
+ * on texture unit 0. Only needed when useMask is true (i.e., only
+ * for MaskFill operations).
+ */
+ maskVars = "sampler2D mask : register(s0);";
+ maskInput = "in float4 maskCoord : TEXCOORD0,";
+ maskCode = "result *= tex2D(mask, maskCoord.xy).a;";
+ }
+
+ if (IS_SET(MULTI_GRAD_LINEAR_RGB)) {
+ /*
+ * This code converts a single pixel in linear RGB space back
+ * into sRGB (note: this code was adapted from the
+ * MultipleGradientPaintContext.convertLinearRGBtoSRGB() method).
+ */
+ colorSpaceCode =
+ "result.rgb = 1.055 * pow(result.rgb, 0.416667) - 0.055;";
+ }
+
+ if (cycleMethod == CYCLE_NONE) {
+ sprintf(cycleCode, noCycleCode, texCoordCalcCode);
+ } else if (cycleMethod == CYCLE_REFLECT) {
+ sprintf(cycleCode, reflectCode, texCoordCalcCode);
+ } else { // (cycleMethod == CYCLE_REPEAT)
+ sprintf(cycleCode, repeatCode, texCoordCalcCode);
+ }
+
+ // compose the final source code string from the various pieces
+ sprintf(finalSource, multiGradientShaderSource,
+ MAX_COLORS, maxFractions, colorSampler,
+ maskVars, paintVars, maskInput, colorSampler,
+ distCode, cycleCode, colorSpaceCode, maskCode);
+
+ D3DShaderGen_WritePixelShader(finalSource, name, flags);
+}
+
+/********************** LinearGradientPaint support *************************/
+
+static void
+D3DShaderGen_GenerateLinearGradShader(int flags)
+{
+ char *paintVars;
+ char *distCode;
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DShaderGen_GenerateLinearGradShader",
+ flags);
+
+ /*
+ * To simplify the code and to make it easier to upload a number of
+ * uniform values at once, we pack a bunch of scalar (float) values
+ * into a single float3 below. Here's how the values are related:
+ *
+ * params.x = p0
+ * params.y = p1
+ * params.z = p3
+ */
+ paintVars =
+ "float3 params : register(c16);";
+ distCode =
+ "float3 fragCoord = float3(winCoord.x, winCoord.y, 1.0);"
+ "dist = dot(params.xyz, fragCoord);";
+
+ D3DShaderGen_GenerateMultiGradShader(flags, "linear",
+ paintVars, distCode);
+}
+
+/********************** RadialGradientPaint support *************************/
+
+static void
+D3DShaderGen_GenerateRadialGradShader(int flags)
+{
+ char *paintVars;
+ char *distCode;
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DShaderGen_GenerateRadialGradShader",
+ flags);
+
+ /*
+ * To simplify the code and to make it easier to upload a number of
+ * uniform values at once, we pack a bunch of scalar (float) values
+ * into float3 values below. Here's how the values are related:
+ *
+ * m0.x = m00
+ * m0.y = m01
+ * m0.z = m02
+ *
+ * m1.x = m10
+ * m1.y = m11
+ * m1.z = m12
+ *
+ * precalc.x = focusX
+ * precalc.y = 1.0 - (focusX * focusX)
+ * precalc.z = 1.0 / precalc.z
+ */
+ paintVars =
+ "float3 m0 : register(c16);"
+ "float3 m1 : register(c17);"
+ "float3 precalc : register(c18);";
+
+ /*
+ * The following code is derived from Daniel Rice's whitepaper on
+ * radial gradient performance (attached to the bug report for 6521533).
+ * Refer to that document as well as the setup code in the Java-level
+ * BufferedPaints.setRadialGradientPaint() method for more details.
+ */
+ distCode =
+ "float3 fragCoord = float3(winCoord.x, winCoord.y, 1.0);"
+ "float x = dot(fragCoord, m0);"
+ "float y = dot(fragCoord, m1);"
+ "float xfx = x - precalc.x;"
+ "dist = (precalc.x*xfx + sqrt(xfx*xfx + y*y*precalc.y))*precalc.z;";
+
+ D3DShaderGen_GenerateMultiGradShader(flags, "radial",
+ paintVars, distCode);
+}
+
+/*************************** LCD text support *******************************/
+
+// REMIND: Shader uses texture addressing operations in a dependency chain
+// that is too complex for the target shader model (ps_2_0) to handle
+// (ugh, I guess we can either require ps_3_0 or just use
+// the slower pow intrinsic)
+#define POW_LUT 0
+
+static const char *lcdTextShaderSource =
+ "float3 srcAdj : register(c0);"
+ "sampler2D glyphTex : register(s0);"
+ "sampler2D dstTex : register(s1);"
+#if POW_LUT
+ "sampler3D invgammaTex : register(s2);"
+ "sampler3D gammaTex : register(s3);"
+#else
+ "float3 invgamma : register(c1);"
+ "float3 gamma : register(c2);"
+#endif
+ ""
+ "void main(in float2 tc0 : TEXCOORD0,"
+ " in float2 tc1 : TEXCOORD1,"
+ " inout float4 color : COLOR0)"
+ "{"
+ // load the RGB value from the glyph image at the current texcoord
+ " float3 glyphClr = tex2D(glyphTex, tc0).rgb;"
+ " if (!any(glyphClr)) {"
+ // zero coverage, so skip this fragment
+ " discard;"
+ " }"
+ // load the RGB value from the corresponding destination pixel
+ " float3 dstClr = tex2D(dstTex, tc1).rgb;"
+ // gamma adjust the dest color using the invgamma LUT
+#if POW_LUT
+ " float3 dstAdj = tex3D(invgammaTex, dstClr).rgb;"
+#else
+ " float3 dstAdj = pow(dstClr, invgamma);"
+#endif
+ // linearly interpolate the three color values
+ " float3 result = lerp(dstAdj, srcAdj, glyphClr);"
+ // gamma re-adjust the resulting color (alpha is always set to 1.0)
+#if POW_LUT
+ " color = float4(tex3D(gammaTex, result).rgb, 1.0);"
+#else
+ " color = float4(pow(result, gamma), 1.0);"
+#endif
+ "}";
+
+static void
+D3DShaderGen_GenerateLCDTextShader()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DShaderGen_GenerateLCDTextShader");
+
+ D3DShaderGen_WritePixelShader((char *)lcdTextShaderSource, "lcdtext", 0);
+}
+
+/*************************** AA support *******************************/
+
+/*
+ * This shader fills the space between an outer and inner parallelogram.
+ * It can be used to draw an outline by specifying both inner and outer
+ * values. It fills pixels by estimating what portion falls inside the
+ * outer shape, and subtracting an estimate of what portion falls inside
+ * the inner shape. Specifying both inner and outer values produces a
+ * standard "wide outline". Specifying an inner shape that falls far
+ * outside the outer shape allows the same shader to fill the outer
+ * shape entirely since pixels that fall within the outer shape are never
+ * inside the inner shape and so they are filled based solely on their
+ * coverage of the outer shape.
+ *
+ * The setup code renders this shader over the bounds of the outer
+ * shape (or the only shape in the case of a fill operation) and
+ * sets the texture 0 coordinates so that 0,0=>0,1=>1,1=>1,0 in those
+ * texture coordinates map to the four corners of the parallelogram.
+ * Similarly the texture 1 coordinates map the inner shape to the
+ * unit square as well, but in a different coordinate system.
+ *
+ * When viewed in the texture coordinate systems the parallelograms
+ * we are filling are unit squares, but the pixels have then become
+ * tiny parallelograms themselves. Both of the texture coordinate
+ * systems are affine transforms so the rate of change in X and Y
+ * of the texture coordinates are essentially constants and happen
+ * to correspond to the size and direction of the slanted sides of
+ * the distorted pixels relative to the "square mapped" boundary
+ * of the parallelograms.
+ *
+ * The shader uses the ddx() and ddy() functions to measure the "rate
+ * of change" of these texture coordinates and thus gets an accurate
+ * measure of the size and shape of a pixel relative to the two
+ * parallelograms. It then uses the bounds of the size and shape
+ * of a pixel to intersect with the unit square to estimate the
+ * coverage of the pixel. Unfortunately, without a lot more work
+ * to calculate the exact area of intersection between a unit
+ * square (the original parallelogram) and a parallelogram (the
+ * distorted pixel), this shader only approximates the pixel
+ * coverage, but emperically the estimate is very useful and
+ * produces visually pleasing results, if not theoretically accurate.
+ */
+static const char *aaShaderSource =
+ "void main(in float2 tco : TEXCOORD0,"
+ " in float2 tci : TEXCOORD1,"
+ " inout float4 color : COLOR0)"
+ "{"
+ // Calculate the vectors for the "legs" of the pixel parallelogram
+ // for the outer parallelogram.
+ " float2 oleg1 = ddx(tco);"
+ " float2 oleg2 = ddy(tco);"
+ // Calculate the bounds of the distorted pixel parallelogram.
+ " float2 omin = min(tco, tco+oleg1);"
+ " omin = min(omin, tco+oleg2);"
+ " omin = min(omin, tco+oleg1+oleg2);"
+ " float2 omax = max(tco, tco+oleg1);"
+ " omax = max(omax, tco+oleg2);"
+ " omax = max(omax, tco+oleg1+oleg2);"
+ // Calculate the vectors for the "legs" of the pixel parallelogram
+ // for the inner parallelogram.
+ " float2 ileg1 = ddx(tci);"
+ " float2 ileg2 = ddy(tci);"
+ // Calculate the bounds of the distorted pixel parallelogram.
+ " float2 imin = min(tci, tci+ileg1);"
+ " imin = min(imin, tci+ileg2);"
+ " imin = min(imin, tci+ileg1+ileg2);"
+ " float2 imax = max(tci, tci+ileg1);"
+ " imax = max(imax, tci+ileg2);"
+ " imax = max(imax, tci+ileg1+ileg2);"
+ // Clamp the bounds of the parallelograms to the unit square to
+ // estimate the intersection of the pixel parallelogram with
+ // the unit square. The ratio of the 2 rectangle areas is a
+ // reasonable estimate of the proportion of coverage.
+ " float2 o1 = clamp(omin, 0.0, 1.0);"
+ " float2 o2 = clamp(omax, 0.0, 1.0);"
+ " float oint = (o2.y-o1.y)*(o2.x-o1.x);"
+ " float oarea = (omax.y-omin.y)*(omax.x-omin.x);"
+ " float2 i1 = clamp(imin, 0.0, 1.0);"
+ " float2 i2 = clamp(imax, 0.0, 1.0);"
+ " float iint = (i2.y-i1.y)*(i2.x-i1.x);"
+ " float iarea = (imax.y-imin.y)*(imax.x-imin.x);"
+ // Proportion of pixel in outer shape minus the proportion
+ // of pixel in the inner shape == the coverage of the pixel
+ // in the area between the two.
+ " float coverage = oint/oarea - iint / iarea;"
+ " color *= coverage;"
+ "}";
+
+static void
+D3DShaderGen_GenerateAAParallelogramShader()
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DShaderGen_GenerateAAParallelogramShader");
+
+ D3DShaderGen_WriteShader((char *)aaShaderSource, "ps_2_a", "aapgram", 0);
+}
+
+/**************************** Main entrypoint *******************************/
+
+static void
+D3DShaderGen_GenerateAllShaders()
+{
+ int i;
+
+#if 1
+ // Generate BufferedImageOp shaders
+ for (i = 0; i < MAX_RESCALE; i++) {
+ D3DShaderGen_GenerateRescaleShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("rescale", MAX_RESCALE);
+ for (i = 0; i < MAX_CONVOLVE; i++) {
+ D3DShaderGen_GenerateConvolveShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("convolve", MAX_CONVOLVE);
+ for (i = 0; i < MAX_LOOKUP; i++) {
+ D3DShaderGen_GenerateLookupShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("lookup", MAX_LOOKUP);
+
+ // Generate Paint shaders
+ for (i = 0; i < MAX_BASIC_GRAD; i++) {
+ D3DShaderGen_GenerateBasicGradShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("grad", MAX_BASIC_GRAD);
+ for (i = 0; i < MAX_MULTI_GRAD; i++) {
+ if (EXTRACT_CYCLE_METHOD(i) == 3) continue; // REMIND
+ D3DShaderGen_GenerateLinearGradShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("linear", MAX_MULTI_GRAD);
+ for (i = 0; i < MAX_MULTI_GRAD; i++) {
+ if (EXTRACT_CYCLE_METHOD(i) == 3) continue; // REMIND
+ D3DShaderGen_GenerateRadialGradShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("radial", MAX_MULTI_GRAD);
+
+ // Generate LCD text shader
+ D3DShaderGen_GenerateLCDTextShader();
+
+ // Genereate Shader to fill Antialiased parallelograms
+ D3DShaderGen_GenerateAAParallelogramShader();
+#else
+ /*
+ for (i = 0; i < MAX_RESCALE; i++) {
+ D3DShaderGen_GenerateRescaleShader(i);
+ }
+ D3DShaderGen_WriteShaderArray("rescale", MAX_RESCALE);
+ */
+ //D3DShaderGen_GenerateConvolveShader(2);
+ //D3DShaderGen_GenerateLCDTextShader();
+ //D3DShaderGen_GenerateLinearGradShader(16);
+ D3DShaderGen_GenerateBasicGradShader(0);
+#endif
+}
+
+int
+main(int argc, char **argv)
+{
+ fpHeader = fopen(strHeaderFile, "a");
+
+ D3DShaderGen_GenerateAllShaders();
+
+ fclose(fpHeader);
+
+ return 0;
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DShaders.h b/jdk/src/windows/native/sun/java2d/d3d/D3DShaders.h
new file mode 100644
index 00000000000..39683be0df9
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DShaders.h
@@ -0,0 +1,7734 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn rescale0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// float4 offsets;
+// float4 scaleFactors;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// scaleFactors c0 1
+// offsets c1 1
+// baseImage s0 1
+//
+
+ ps_2_0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ texld r0, t0, s0
+ mov r1, c0
+ mad r0, r0, r1, c1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 5 instruction slots used (1 texture, 4 arithmetic)
+#endif
+
+const DWORD rescale0[] =
+{
+ 0xffff0200, 0x0036fffe, 0x42415443, 0x0000001c, 0x000000a0, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000099, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00010002, 0x00060001,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00000002, 0x00020001, 0x0000007c,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x7366666f, 0x00737465, 0x00030001, 0x00040001,
+ 0x00000001, 0x00000000, 0x6c616373, 0x63614665, 0x73726f74, 0x5f737000,
+ 0x00305f32, 0x7263694d, 0x666f736f, 0x52282074, 0x4c482029, 0x53204c53,
+ 0x65646168, 0x6f432072, 0x6c69706d, 0x39207265, 0x2e39312e, 0x2e393439,
+ 0x31313132, 0xababab00, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f,
+ 0x80000000, 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x03000042,
+ 0x800f0000, 0xb0e40000, 0xa0e40800, 0x02000001, 0x800f0001, 0xa0e40000,
+ 0x04000004, 0x800f0000, 0x80e40000, 0x80e40001, 0xa0e40001, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn rescale1 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// float4 offsets;
+// float4 scaleFactors;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// scaleFactors c0 1
+// offsets c1 1
+// baseImage s0 1
+//
+
+ ps_2_0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ texld r0, t0, s0
+ rcp r1.w, r0.w
+ mul r0.xyz, r0, r1.w
+ mov r1, c0
+ mad r0, r0, r1, c1
+ mul r0.xyz, r0.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 8 instruction slots used (1 texture, 7 arithmetic)
+#endif
+
+const DWORD rescale1[] =
+{
+ 0xffff0200, 0x0036fffe, 0x42415443, 0x0000001c, 0x000000a0, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000099, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00010002, 0x00060001,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00000002, 0x00020001, 0x0000007c,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x7366666f, 0x00737465, 0x00030001, 0x00040001,
+ 0x00000001, 0x00000000, 0x6c616373, 0x63614665, 0x73726f74, 0x5f737000,
+ 0x00305f32, 0x7263694d, 0x666f736f, 0x52282074, 0x4c482029, 0x53204c53,
+ 0x65646168, 0x6f432072, 0x6c69706d, 0x39207265, 0x2e39312e, 0x2e393439,
+ 0x31313132, 0xababab00, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f,
+ 0x80000000, 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x03000042,
+ 0x800f0000, 0xb0e40000, 0xa0e40800, 0x02000006, 0x80080001, 0x80ff0000,
+ 0x03000005, 0x80070000, 0x80e40000, 0x80ff0001, 0x02000001, 0x800f0001,
+ 0xa0e40000, 0x04000004, 0x800f0000, 0x80e40000, 0x80e40001, 0xa0e40001,
+ 0x03000005, 0x80070000, 0x80ff0000, 0x80e40000, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+const DWORD *rescaleShaders[] =
+{
+ rescale0,
+ rescale1,
+};
+
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn convolve0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// float4 imgEdge;
+// float3 kernelVals[9];
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// imgEdge c0 1
+// kernelVals c1 9
+// baseImage s0 1
+//
+
+ ps_2_0
+ def c10, 0, 1, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ add r0.xy, t0, c2
+ add r1.xy, t0, c1
+ add r2.xy, t0, c3
+ add r3.xy, t0, c4
+ add r4.xy, t0, c5
+ add r5.xy, t0, c6
+ add r6.xy, t0, c7
+ add r7.xy, t0, c8
+ add r8.xy, t0, c9
+ texld r0, r0, s0
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r9, t0, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, r8, s0
+ mul r0, r0, c2.z
+ mad r0, c1.z, r1, r0
+ mad r0, c3.z, r2, r0
+ mad r0, c4.z, r3, r0
+ mad r0, c5.z, r4, r0
+ mad r0, c6.z, r5, r0
+ mad r0, c7.z, r6, r0
+ mad r0, c8.z, r7, r0
+ mad r0, c9.z, r8, r0
+ mul r0, r0, v0
+ add r1.yw, -t0.y, c0
+ add r1.xz, -t0.x, c0
+ cmp r1.x, -r1.x, c10.x, c10.y
+ cmp r1.y, -r1.y, c10.x, c10.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c10.x, c10.y
+ cmp r1.y, r1.z, c10.x, c10.y
+ cmp r1.z, r1.w, c10.x, c10.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c10.x, c10.y
+ add r1.x, r1.z, r1.x
+ cmp r0, -r1.x, r0, r9
+ mov oC0, r0
+
+// approximately 42 instruction slots used (10 texture, 32 arithmetic)
+#endif
+
+const DWORD convolve0[] =
+{
+ 0xffff0200, 0x0039fffe, 0x42415443, 0x0000001c, 0x000000af, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a8, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00000002, 0x00020001,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00010002, 0x00060009, 0x00000098,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x45676d69, 0x00656764, 0x00030001, 0x00040001,
+ 0x00000001, 0x00000000, 0x6e72656b, 0x61566c65, 0xab00736c, 0x00030001,
+ 0x00030001, 0x00000009, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369,
+ 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320,
+ 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051,
+ 0xa00f000a, 0x00000000, 0x3f800000, 0x00000000, 0x00000000, 0x0200001f,
+ 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f,
+ 0x90000000, 0xa00f0800, 0x03000002, 0x80030000, 0xb0e40000, 0xa0e40002,
+ 0x03000002, 0x80030001, 0xb0e40000, 0xa0e40001, 0x03000002, 0x80030002,
+ 0xb0e40000, 0xa0e40003, 0x03000002, 0x80030003, 0xb0e40000, 0xa0e40004,
+ 0x03000002, 0x80030004, 0xb0e40000, 0xa0e40005, 0x03000002, 0x80030005,
+ 0xb0e40000, 0xa0e40006, 0x03000002, 0x80030006, 0xb0e40000, 0xa0e40007,
+ 0x03000002, 0x80030007, 0xb0e40000, 0xa0e40008, 0x03000002, 0x80030008,
+ 0xb0e40000, 0xa0e40009, 0x03000042, 0x800f0000, 0x80e40000, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002,
+ 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0009, 0xb0e40000, 0xa0e40800,
+ 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004,
+ 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40005, 0xa0e40800,
+ 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800, 0x03000042, 0x800f0007,
+ 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008, 0x80e40008, 0xa0e40800,
+ 0x03000005, 0x800f0000, 0x80e40000, 0xa0aa0002, 0x04000004, 0x800f0000,
+ 0xa0aa0001, 0x80e40001, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0003,
+ 0x80e40002, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0004, 0x80e40003,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0005, 0x80e40004, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa0006, 0x80e40005, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0007, 0x80e40006, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0008, 0x80e40007, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0009,
+ 0x80e40008, 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x03000002, 0x800a0001, 0xb1550000, 0xa0e40000, 0x03000002, 0x80050001,
+ 0xb1000000, 0xa0e40000, 0x04000058, 0x80010001, 0x81000001, 0xa000000a,
+ 0xa055000a, 0x04000058, 0x80020001, 0x81550001, 0xa000000a, 0xa055000a,
+ 0x03000002, 0x80010001, 0x80000001, 0x80550001, 0x04000058, 0x80010001,
+ 0x81000001, 0xa000000a, 0xa055000a, 0x04000058, 0x80020001, 0x80aa0001,
+ 0xa000000a, 0xa055000a, 0x04000058, 0x80040001, 0x80ff0001, 0xa000000a,
+ 0xa055000a, 0x03000002, 0x80010001, 0x80000001, 0x80550001, 0x04000058,
+ 0x80010001, 0x81000001, 0xa000000a, 0xa055000a, 0x03000002, 0x80010001,
+ 0x80aa0001, 0x80000001, 0x04000058, 0x800f0000, 0x81000001, 0x80e40000,
+ 0x80e40009, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn convolve1 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// float4 imgEdge;
+// float3 kernelVals[9];
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// imgEdge c0 1
+// kernelVals c1 9
+// baseImage s0 1
+//
+
+ ps_2_0
+ def c10, 0, 1, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ add r0.xy, t0, c2
+ add r1.xy, t0, c1
+ add r2.xy, t0, c3
+ add r3.xy, t0, c4
+ add r4.xy, t0, c5
+ add r5.xy, t0, c6
+ add r6.xy, t0, c7
+ add r7.xy, t0, c8
+ add r8.xy, t0, c9
+ texld r0, r0, s0
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, r8, s0
+ mul r0, r0, c2.z
+ mad r0, c1.z, r1, r0
+ mad r0, c3.z, r2, r0
+ mad r0, c4.z, r3, r0
+ mad r0, c5.z, r4, r0
+ mad r0, c6.z, r5, r0
+ mad r0, c7.z, r6, r0
+ mad r0, c8.z, r7, r0
+ mad r0, c9.z, r8, r0
+ mul r0, r0, v0
+ add r1.yw, -t0.y, c0
+ add r1.xz, -t0.x, c0
+ cmp r1.x, -r1.x, c10.x, c10.y
+ cmp r1.y, -r1.y, c10.x, c10.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c10.x, c10.y
+ cmp r1.y, r1.z, c10.x, c10.y
+ cmp r1.z, r1.w, c10.x, c10.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c10.x, c10.y
+ add r1.x, r1.z, r1.x
+ cmp r0, -r1.x, r0, c10.x
+ mov oC0, r0
+
+// approximately 41 instruction slots used (9 texture, 32 arithmetic)
+#endif
+
+const DWORD convolve1[] =
+{
+ 0xffff0200, 0x0039fffe, 0x42415443, 0x0000001c, 0x000000af, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a8, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00000002, 0x00020001,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00010002, 0x00060009, 0x00000098,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x45676d69, 0x00656764, 0x00030001, 0x00040001,
+ 0x00000001, 0x00000000, 0x6e72656b, 0x61566c65, 0xab00736c, 0x00030001,
+ 0x00030001, 0x00000009, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369,
+ 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320,
+ 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051,
+ 0xa00f000a, 0x00000000, 0x3f800000, 0x00000000, 0x00000000, 0x0200001f,
+ 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f,
+ 0x90000000, 0xa00f0800, 0x03000002, 0x80030000, 0xb0e40000, 0xa0e40002,
+ 0x03000002, 0x80030001, 0xb0e40000, 0xa0e40001, 0x03000002, 0x80030002,
+ 0xb0e40000, 0xa0e40003, 0x03000002, 0x80030003, 0xb0e40000, 0xa0e40004,
+ 0x03000002, 0x80030004, 0xb0e40000, 0xa0e40005, 0x03000002, 0x80030005,
+ 0xb0e40000, 0xa0e40006, 0x03000002, 0x80030006, 0xb0e40000, 0xa0e40007,
+ 0x03000002, 0x80030007, 0xb0e40000, 0xa0e40008, 0x03000002, 0x80030008,
+ 0xb0e40000, 0xa0e40009, 0x03000042, 0x800f0000, 0x80e40000, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002,
+ 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800,
+ 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005,
+ 0x80e40005, 0xa0e40800, 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800,
+ 0x03000042, 0x800f0007, 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008,
+ 0x80e40008, 0xa0e40800, 0x03000005, 0x800f0000, 0x80e40000, 0xa0aa0002,
+ 0x04000004, 0x800f0000, 0xa0aa0001, 0x80e40001, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0003, 0x80e40002, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0004, 0x80e40003, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0005,
+ 0x80e40004, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0006, 0x80e40005,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0007, 0x80e40006, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa0008, 0x80e40007, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0009, 0x80e40008, 0x80e40000, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x03000002, 0x800a0001, 0xb1550000, 0xa0e40000,
+ 0x03000002, 0x80050001, 0xb1000000, 0xa0e40000, 0x04000058, 0x80010001,
+ 0x81000001, 0xa000000a, 0xa055000a, 0x04000058, 0x80020001, 0x81550001,
+ 0xa000000a, 0xa055000a, 0x03000002, 0x80010001, 0x80000001, 0x80550001,
+ 0x04000058, 0x80010001, 0x81000001, 0xa000000a, 0xa055000a, 0x04000058,
+ 0x80020001, 0x80aa0001, 0xa000000a, 0xa055000a, 0x04000058, 0x80040001,
+ 0x80ff0001, 0xa000000a, 0xa055000a, 0x03000002, 0x80010001, 0x80000001,
+ 0x80550001, 0x04000058, 0x80010001, 0x81000001, 0xa000000a, 0xa055000a,
+ 0x03000002, 0x80010001, 0x80aa0001, 0x80000001, 0x04000058, 0x800f0000,
+ 0x81000001, 0x80e40000, 0xa000000a, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn convolve2 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// float4 imgEdge;
+// float3 kernelVals[25];
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// imgEdge c0 1
+// kernelVals c1 25
+// baseImage s0 1
+//
+
+ ps_2_0
+ def c26, 0, 1, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ add r0.xy, t0, c2
+ add r1.xy, t0, c1
+ add r2.xy, t0, c3
+ add r3.xy, t0, c4
+ add r4.xy, t0, c5
+ add r5.xy, t0, c6
+ add r6.xy, t0, c7
+ add r7.xy, t0, c8
+ add r8.xy, t0, c9
+ texld r0, r0, s0
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, r8, s0
+ mul r0, r0, c2.z
+ mad r0, c1.z, r1, r0
+ mad r0, c3.z, r2, r0
+ mad r0, c4.z, r3, r0
+ mad r0, c5.z, r4, r0
+ mad r0, c6.z, r5, r0
+ mad r0, c7.z, r6, r0
+ mad r0, c8.z, r7, r0
+ mad r0, c9.z, r8, r0
+ add r1.xy, t0, c10
+ add r2.xy, t0, c11
+ add r3.xy, t0, c12
+ add r4.xy, t0, c13
+ add r5.xy, t0, c14
+ add r6.xy, t0, c15
+ add r7.xy, t0, c16
+ add r8.xy, t0, c17
+ add r9.xy, t0, c18
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, r8, s0
+ texld r9, r9, s0
+ mad r0, c10.z, r1, r0
+ mad r0, c11.z, r2, r0
+ mad r0, c12.z, r3, r0
+ mad r0, c13.z, r4, r0
+ mad r0, c14.z, r5, r0
+ mad r0, c15.z, r6, r0
+ mad r0, c16.z, r7, r0
+ mad r0, c17.z, r8, r0
+ mad r0, c18.z, r9, r0
+ add r1.xy, t0, c19
+ add r2.xy, t0, c20
+ add r3.xy, t0, c21
+ add r4.xy, t0, c22
+ add r5.xy, t0, c23
+ add r6.xy, t0, c24
+ add r7.xy, t0, c25
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, t0, s0
+ mad r0, c19.z, r1, r0
+ mad r0, c20.z, r2, r0
+ mad r0, c21.z, r3, r0
+ mad r0, c22.z, r4, r0
+ mad r0, c23.z, r5, r0
+ mad r0, c24.z, r6, r0
+ mad r0, c25.z, r7, r0
+ mul r0, r0, v0
+ add r1.yw, -t0.y, c0
+ add r1.xz, -t0.x, c0
+ cmp r1.x, -r1.x, c26.x, c26.y
+ cmp r1.y, -r1.y, c26.x, c26.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c26.x, c26.y
+ cmp r1.y, r1.z, c26.x, c26.y
+ cmp r1.z, r1.w, c26.x, c26.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c26.x, c26.y
+ add r1.x, r1.z, r1.x
+ cmp r0, -r1.x, r0, r8
+ mov oC0, r0
+
+// approximately 90 instruction slots used (26 texture, 64 arithmetic)
+#endif
+
+const DWORD convolve2[] =
+{
+ 0xffff0200, 0x0039fffe, 0x42415443, 0x0000001c, 0x000000af, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a8, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00000002, 0x00020001,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00010002, 0x00060019, 0x00000098,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x45676d69, 0x00656764, 0x00030001, 0x00040001,
+ 0x00000001, 0x00000000, 0x6e72656b, 0x61566c65, 0xab00736c, 0x00030001,
+ 0x00030001, 0x00000019, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369,
+ 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320,
+ 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051,
+ 0xa00f001a, 0x00000000, 0x3f800000, 0x00000000, 0x00000000, 0x0200001f,
+ 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f,
+ 0x90000000, 0xa00f0800, 0x03000002, 0x80030000, 0xb0e40000, 0xa0e40002,
+ 0x03000002, 0x80030001, 0xb0e40000, 0xa0e40001, 0x03000002, 0x80030002,
+ 0xb0e40000, 0xa0e40003, 0x03000002, 0x80030003, 0xb0e40000, 0xa0e40004,
+ 0x03000002, 0x80030004, 0xb0e40000, 0xa0e40005, 0x03000002, 0x80030005,
+ 0xb0e40000, 0xa0e40006, 0x03000002, 0x80030006, 0xb0e40000, 0xa0e40007,
+ 0x03000002, 0x80030007, 0xb0e40000, 0xa0e40008, 0x03000002, 0x80030008,
+ 0xb0e40000, 0xa0e40009, 0x03000042, 0x800f0000, 0x80e40000, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002,
+ 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800,
+ 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005,
+ 0x80e40005, 0xa0e40800, 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800,
+ 0x03000042, 0x800f0007, 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008,
+ 0x80e40008, 0xa0e40800, 0x03000005, 0x800f0000, 0x80e40000, 0xa0aa0002,
+ 0x04000004, 0x800f0000, 0xa0aa0001, 0x80e40001, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0003, 0x80e40002, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0004, 0x80e40003, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0005,
+ 0x80e40004, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0006, 0x80e40005,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0007, 0x80e40006, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa0008, 0x80e40007, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0009, 0x80e40008, 0x80e40000, 0x03000002, 0x80030001,
+ 0xb0e40000, 0xa0e4000a, 0x03000002, 0x80030002, 0xb0e40000, 0xa0e4000b,
+ 0x03000002, 0x80030003, 0xb0e40000, 0xa0e4000c, 0x03000002, 0x80030004,
+ 0xb0e40000, 0xa0e4000d, 0x03000002, 0x80030005, 0xb0e40000, 0xa0e4000e,
+ 0x03000002, 0x80030006, 0xb0e40000, 0xa0e4000f, 0x03000002, 0x80030007,
+ 0xb0e40000, 0xa0e40010, 0x03000002, 0x80030008, 0xb0e40000, 0xa0e40011,
+ 0x03000002, 0x80030009, 0xb0e40000, 0xa0e40012, 0x03000042, 0x800f0001,
+ 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800,
+ 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004,
+ 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40005, 0xa0e40800,
+ 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800, 0x03000042, 0x800f0007,
+ 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008, 0x80e40008, 0xa0e40800,
+ 0x03000042, 0x800f0009, 0x80e40009, 0xa0e40800, 0x04000004, 0x800f0000,
+ 0xa0aa000a, 0x80e40001, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa000b,
+ 0x80e40002, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa000c, 0x80e40003,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa000d, 0x80e40004, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa000e, 0x80e40005, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa000f, 0x80e40006, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0010, 0x80e40007, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0011,
+ 0x80e40008, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0012, 0x80e40009,
+ 0x80e40000, 0x03000002, 0x80030001, 0xb0e40000, 0xa0e40013, 0x03000002,
+ 0x80030002, 0xb0e40000, 0xa0e40014, 0x03000002, 0x80030003, 0xb0e40000,
+ 0xa0e40015, 0x03000002, 0x80030004, 0xb0e40000, 0xa0e40016, 0x03000002,
+ 0x80030005, 0xb0e40000, 0xa0e40017, 0x03000002, 0x80030006, 0xb0e40000,
+ 0xa0e40018, 0x03000002, 0x80030007, 0xb0e40000, 0xa0e40019, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x03000042,
+ 0x800f0004, 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40005,
+ 0xa0e40800, 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800, 0x03000042,
+ 0x800f0007, 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008, 0xb0e40000,
+ 0xa0e40800, 0x04000004, 0x800f0000, 0xa0aa0013, 0x80e40001, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa0014, 0x80e40002, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0015, 0x80e40003, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0016, 0x80e40004, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0017,
+ 0x80e40005, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0018, 0x80e40006,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0019, 0x80e40007, 0x80e40000,
+ 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x03000002, 0x800a0001,
+ 0xb1550000, 0xa0e40000, 0x03000002, 0x80050001, 0xb1000000, 0xa0e40000,
+ 0x04000058, 0x80010001, 0x81000001, 0xa000001a, 0xa055001a, 0x04000058,
+ 0x80020001, 0x81550001, 0xa000001a, 0xa055001a, 0x03000002, 0x80010001,
+ 0x80000001, 0x80550001, 0x04000058, 0x80010001, 0x81000001, 0xa000001a,
+ 0xa055001a, 0x04000058, 0x80020001, 0x80aa0001, 0xa000001a, 0xa055001a,
+ 0x04000058, 0x80040001, 0x80ff0001, 0xa000001a, 0xa055001a, 0x03000002,
+ 0x80010001, 0x80000001, 0x80550001, 0x04000058, 0x80010001, 0x81000001,
+ 0xa000001a, 0xa055001a, 0x03000002, 0x80010001, 0x80aa0001, 0x80000001,
+ 0x04000058, 0x800f0000, 0x81000001, 0x80e40000, 0x80e40008, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn convolve3 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// float4 imgEdge;
+// float3 kernelVals[25];
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// imgEdge c0 1
+// kernelVals c1 25
+// baseImage s0 1
+//
+
+ ps_2_0
+ def c26, 0, 1, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ add r0.xy, t0, c2
+ add r1.xy, t0, c1
+ add r2.xy, t0, c3
+ add r3.xy, t0, c4
+ add r4.xy, t0, c5
+ add r5.xy, t0, c6
+ add r6.xy, t0, c7
+ add r7.xy, t0, c8
+ add r8.xy, t0, c9
+ texld r0, r0, s0
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, r8, s0
+ mul r0, r0, c2.z
+ mad r0, c1.z, r1, r0
+ mad r0, c3.z, r2, r0
+ mad r0, c4.z, r3, r0
+ mad r0, c5.z, r4, r0
+ mad r0, c6.z, r5, r0
+ mad r0, c7.z, r6, r0
+ mad r0, c8.z, r7, r0
+ mad r0, c9.z, r8, r0
+ add r1.xy, t0, c10
+ add r2.xy, t0, c11
+ add r3.xy, t0, c12
+ add r4.xy, t0, c13
+ add r5.xy, t0, c14
+ add r6.xy, t0, c15
+ add r7.xy, t0, c16
+ add r8.xy, t0, c17
+ add r9.xy, t0, c18
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ texld r8, r8, s0
+ texld r9, r9, s0
+ mad r0, c10.z, r1, r0
+ mad r0, c11.z, r2, r0
+ mad r0, c12.z, r3, r0
+ mad r0, c13.z, r4, r0
+ mad r0, c14.z, r5, r0
+ mad r0, c15.z, r6, r0
+ mad r0, c16.z, r7, r0
+ mad r0, c17.z, r8, r0
+ mad r0, c18.z, r9, r0
+ add r1.xy, t0, c19
+ add r2.xy, t0, c20
+ add r3.xy, t0, c21
+ add r4.xy, t0, c22
+ add r5.xy, t0, c23
+ add r6.xy, t0, c24
+ add r7.xy, t0, c25
+ texld r1, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ texld r5, r5, s0
+ texld r6, r6, s0
+ texld r7, r7, s0
+ mad r0, c19.z, r1, r0
+ mad r0, c20.z, r2, r0
+ mad r0, c21.z, r3, r0
+ mad r0, c22.z, r4, r0
+ mad r0, c23.z, r5, r0
+ mad r0, c24.z, r6, r0
+ mad r0, c25.z, r7, r0
+ mul r0, r0, v0
+ add r1.yw, -t0.y, c0
+ add r1.xz, -t0.x, c0
+ cmp r1.x, -r1.x, c26.x, c26.y
+ cmp r1.y, -r1.y, c26.x, c26.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c26.x, c26.y
+ cmp r1.y, r1.z, c26.x, c26.y
+ cmp r1.z, r1.w, c26.x, c26.y
+ add r1.x, r1.x, r1.y
+ cmp r1.x, -r1.x, c26.x, c26.y
+ add r1.x, r1.z, r1.x
+ cmp r0, -r1.x, r0, c26.x
+ mov oC0, r0
+
+// approximately 89 instruction slots used (25 texture, 64 arithmetic)
+#endif
+
+const DWORD convolve3[] =
+{
+ 0xffff0200, 0x0039fffe, 0x42415443, 0x0000001c, 0x000000af, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a8, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00000002, 0x00020001,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00010002, 0x00060019, 0x00000098,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x45676d69, 0x00656764, 0x00030001, 0x00040001,
+ 0x00000001, 0x00000000, 0x6e72656b, 0x61566c65, 0xab00736c, 0x00030001,
+ 0x00030001, 0x00000019, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369,
+ 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320,
+ 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051,
+ 0xa00f001a, 0x00000000, 0x3f800000, 0x00000000, 0x00000000, 0x0200001f,
+ 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f,
+ 0x90000000, 0xa00f0800, 0x03000002, 0x80030000, 0xb0e40000, 0xa0e40002,
+ 0x03000002, 0x80030001, 0xb0e40000, 0xa0e40001, 0x03000002, 0x80030002,
+ 0xb0e40000, 0xa0e40003, 0x03000002, 0x80030003, 0xb0e40000, 0xa0e40004,
+ 0x03000002, 0x80030004, 0xb0e40000, 0xa0e40005, 0x03000002, 0x80030005,
+ 0xb0e40000, 0xa0e40006, 0x03000002, 0x80030006, 0xb0e40000, 0xa0e40007,
+ 0x03000002, 0x80030007, 0xb0e40000, 0xa0e40008, 0x03000002, 0x80030008,
+ 0xb0e40000, 0xa0e40009, 0x03000042, 0x800f0000, 0x80e40000, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002,
+ 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800,
+ 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005,
+ 0x80e40005, 0xa0e40800, 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800,
+ 0x03000042, 0x800f0007, 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008,
+ 0x80e40008, 0xa0e40800, 0x03000005, 0x800f0000, 0x80e40000, 0xa0aa0002,
+ 0x04000004, 0x800f0000, 0xa0aa0001, 0x80e40001, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0003, 0x80e40002, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0004, 0x80e40003, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0005,
+ 0x80e40004, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0006, 0x80e40005,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0007, 0x80e40006, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa0008, 0x80e40007, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0009, 0x80e40008, 0x80e40000, 0x03000002, 0x80030001,
+ 0xb0e40000, 0xa0e4000a, 0x03000002, 0x80030002, 0xb0e40000, 0xa0e4000b,
+ 0x03000002, 0x80030003, 0xb0e40000, 0xa0e4000c, 0x03000002, 0x80030004,
+ 0xb0e40000, 0xa0e4000d, 0x03000002, 0x80030005, 0xb0e40000, 0xa0e4000e,
+ 0x03000002, 0x80030006, 0xb0e40000, 0xa0e4000f, 0x03000002, 0x80030007,
+ 0xb0e40000, 0xa0e40010, 0x03000002, 0x80030008, 0xb0e40000, 0xa0e40011,
+ 0x03000002, 0x80030009, 0xb0e40000, 0xa0e40012, 0x03000042, 0x800f0001,
+ 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800,
+ 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004,
+ 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40005, 0xa0e40800,
+ 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800, 0x03000042, 0x800f0007,
+ 0x80e40007, 0xa0e40800, 0x03000042, 0x800f0008, 0x80e40008, 0xa0e40800,
+ 0x03000042, 0x800f0009, 0x80e40009, 0xa0e40800, 0x04000004, 0x800f0000,
+ 0xa0aa000a, 0x80e40001, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa000b,
+ 0x80e40002, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa000c, 0x80e40003,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa000d, 0x80e40004, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa000e, 0x80e40005, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa000f, 0x80e40006, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0010, 0x80e40007, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0011,
+ 0x80e40008, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0012, 0x80e40009,
+ 0x80e40000, 0x03000002, 0x80030001, 0xb0e40000, 0xa0e40013, 0x03000002,
+ 0x80030002, 0xb0e40000, 0xa0e40014, 0x03000002, 0x80030003, 0xb0e40000,
+ 0xa0e40015, 0x03000002, 0x80030004, 0xb0e40000, 0xa0e40016, 0x03000002,
+ 0x80030005, 0xb0e40000, 0xa0e40017, 0x03000002, 0x80030006, 0xb0e40000,
+ 0xa0e40018, 0x03000002, 0x80030007, 0xb0e40000, 0xa0e40019, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x03000042,
+ 0x800f0004, 0x80e40004, 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40005,
+ 0xa0e40800, 0x03000042, 0x800f0006, 0x80e40006, 0xa0e40800, 0x03000042,
+ 0x800f0007, 0x80e40007, 0xa0e40800, 0x04000004, 0x800f0000, 0xa0aa0013,
+ 0x80e40001, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0014, 0x80e40002,
+ 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0015, 0x80e40003, 0x80e40000,
+ 0x04000004, 0x800f0000, 0xa0aa0016, 0x80e40004, 0x80e40000, 0x04000004,
+ 0x800f0000, 0xa0aa0017, 0x80e40005, 0x80e40000, 0x04000004, 0x800f0000,
+ 0xa0aa0018, 0x80e40006, 0x80e40000, 0x04000004, 0x800f0000, 0xa0aa0019,
+ 0x80e40007, 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x03000002, 0x800a0001, 0xb1550000, 0xa0e40000, 0x03000002, 0x80050001,
+ 0xb1000000, 0xa0e40000, 0x04000058, 0x80010001, 0x81000001, 0xa000001a,
+ 0xa055001a, 0x04000058, 0x80020001, 0x81550001, 0xa000001a, 0xa055001a,
+ 0x03000002, 0x80010001, 0x80000001, 0x80550001, 0x04000058, 0x80010001,
+ 0x81000001, 0xa000001a, 0xa055001a, 0x04000058, 0x80020001, 0x80aa0001,
+ 0xa000001a, 0xa055001a, 0x04000058, 0x80040001, 0x80ff0001, 0xa000001a,
+ 0xa055001a, 0x03000002, 0x80010001, 0x80000001, 0x80550001, 0x04000058,
+ 0x80010001, 0x81000001, 0xa000001a, 0xa055001a, 0x03000002, 0x80010001,
+ 0x80aa0001, 0x80000001, 0x04000058, 0x800f0000, 0x81000001, 0x80e40000,
+ 0xa000001a, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+const DWORD *convolveShaders[] =
+{
+ convolve0,
+ convolve1,
+ convolve2,
+ convolve3,
+};
+
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn lookup0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// sampler2D lookupTable;
+// float4 offset;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// offset c0 1
+// baseImage s0 1
+// lookupTable s1 1
+//
+
+ ps_2_0
+ def c1, 0.125, 0.375, 0.625, 0.875
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ texld r0, t0, s0
+ mov r1.y, c1.x
+ add r0, r0, -c0
+ mov r1.x, r0.x
+ mov r0.x, r0.y
+ mov r0.y, c1.y
+ mov r2.x, r0.z
+ mov r3.x, r0.w
+ mov r2.y, c1.z
+ mov r3.y, c1.w
+ texld r1, r1, s1
+ texld r0, r0, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ mov r1.y, r0.x
+ mov r1.z, r2.x
+ mov r1.w, r3.x
+ mul r0, r1, v0
+ mov oC0, r0
+
+// approximately 19 instruction slots used (5 texture, 14 arithmetic)
+#endif
+
+const DWORD lookup0[] =
+{
+ 0xffff0200, 0x0035fffe, 0x42415443, 0x0000001c, 0x0000009f, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000098, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00010003, 0x00060001,
+ 0x00000064, 0x00000000, 0x00000080, 0x00000002, 0x00020001, 0x00000088,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x6b6f6f6c, 0x61547075, 0x00656c62, 0x7366666f,
+ 0xab007465, 0x00030001, 0x00040001, 0x00000001, 0x00000000, 0x325f7370,
+ 0x4d00305f, 0x6f726369, 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c,
+ 0x72656461, 0x6d6f4320, 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934,
+ 0x00313131, 0x05000051, 0xa00f0001, 0x3e000000, 0x3ec00000, 0x3f200000,
+ 0x3f600000, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x03000042, 0x800f0000, 0xb0e40000, 0xa0e40800, 0x02000001,
+ 0x80020001, 0xa0000001, 0x03000002, 0x800f0000, 0x80e40000, 0xa1e40000,
+ 0x02000001, 0x80010001, 0x80000000, 0x02000001, 0x80010000, 0x80550000,
+ 0x02000001, 0x80020000, 0xa0550001, 0x02000001, 0x80010002, 0x80aa0000,
+ 0x02000001, 0x80010003, 0x80ff0000, 0x02000001, 0x80020002, 0xa0aa0001,
+ 0x02000001, 0x80020003, 0xa0ff0001, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0000, 0x80e40000, 0xa0e40801, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0003, 0x80e40003,
+ 0xa0e40801, 0x02000001, 0x80020001, 0x80000000, 0x02000001, 0x80040001,
+ 0x80000002, 0x02000001, 0x80080001, 0x80000003, 0x03000005, 0x800f0000,
+ 0x80e40001, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn lookup1 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// sampler2D lookupTable;
+// float4 offset;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// offset c0 1
+// baseImage s0 1
+// lookupTable s1 1
+//
+
+ ps_2_0
+ def c1, 0.125, 0.375, 0.625, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ texld r0, t0, s0
+ mov r1.y, c1.x
+ add r2.xyz, r0, -c0
+ mov r1.x, r2.x
+ mov r2.x, r2.y
+ mov r3.x, r2.z
+ mov r2.y, c1.y
+ mov r3.y, c1.z
+ texld r1, r1, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ mov r0.x, r1.x
+ mov r0.y, r2.x
+ mov r0.z, r3.x
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 16 instruction slots used (4 texture, 12 arithmetic)
+#endif
+
+const DWORD lookup1[] =
+{
+ 0xffff0200, 0x0035fffe, 0x42415443, 0x0000001c, 0x0000009f, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000098, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00010003, 0x00060001,
+ 0x00000064, 0x00000000, 0x00000080, 0x00000002, 0x00020001, 0x00000088,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x6b6f6f6c, 0x61547075, 0x00656c62, 0x7366666f,
+ 0xab007465, 0x00030001, 0x00040001, 0x00000001, 0x00000000, 0x325f7370,
+ 0x4d00305f, 0x6f726369, 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c,
+ 0x72656461, 0x6d6f4320, 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934,
+ 0x00313131, 0x05000051, 0xa00f0001, 0x3e000000, 0x3ec00000, 0x3f200000,
+ 0x00000000, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x03000042, 0x800f0000, 0xb0e40000, 0xa0e40800, 0x02000001,
+ 0x80020001, 0xa0000001, 0x03000002, 0x80070002, 0x80e40000, 0xa1e40000,
+ 0x02000001, 0x80010001, 0x80000002, 0x02000001, 0x80010002, 0x80550002,
+ 0x02000001, 0x80010003, 0x80aa0002, 0x02000001, 0x80020002, 0xa0550001,
+ 0x02000001, 0x80020003, 0xa0aa0001, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0003, 0x80e40003, 0xa0e40801, 0x02000001, 0x80010000, 0x80000001,
+ 0x02000001, 0x80020000, 0x80000002, 0x02000001, 0x80040000, 0x80000003,
+ 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn lookup2 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// sampler2D lookupTable;
+// float4 offset;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// offset c0 1
+// baseImage s0 1
+// lookupTable s1 1
+//
+
+ ps_2_0
+ def c1, 0.125, 0.375, 0.625, 0.875
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ texld r0, t0, s0
+ mov r1.y, c1.x
+ rcp r1.z, r0.w
+ mul r0.xyz, r0, r1.z
+ add r0, r0, -c0
+ mov r1.x, r0.x
+ mov r0.x, r0.y
+ mov r0.y, c1.y
+ mov r2.x, r0.z
+ mov r3.x, r0.w
+ mov r2.y, c1.z
+ mov r3.y, c1.w
+ texld r1, r1, s1
+ texld r0, r0, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ mov r1.y, r0.x
+ mov r1.z, r2.x
+ mul r0.xyz, r3.x, r1
+ mov r0.w, r3.x
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 22 instruction slots used (5 texture, 17 arithmetic)
+#endif
+
+const DWORD lookup2[] =
+{
+ 0xffff0200, 0x0035fffe, 0x42415443, 0x0000001c, 0x0000009f, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000098, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00010003, 0x00060001,
+ 0x00000064, 0x00000000, 0x00000080, 0x00000002, 0x00020001, 0x00000088,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x6b6f6f6c, 0x61547075, 0x00656c62, 0x7366666f,
+ 0xab007465, 0x00030001, 0x00040001, 0x00000001, 0x00000000, 0x325f7370,
+ 0x4d00305f, 0x6f726369, 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c,
+ 0x72656461, 0x6d6f4320, 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934,
+ 0x00313131, 0x05000051, 0xa00f0001, 0x3e000000, 0x3ec00000, 0x3f200000,
+ 0x3f600000, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x03000042, 0x800f0000, 0xb0e40000, 0xa0e40800, 0x02000001,
+ 0x80020001, 0xa0000001, 0x02000006, 0x80040001, 0x80ff0000, 0x03000005,
+ 0x80070000, 0x80e40000, 0x80aa0001, 0x03000002, 0x800f0000, 0x80e40000,
+ 0xa1e40000, 0x02000001, 0x80010001, 0x80000000, 0x02000001, 0x80010000,
+ 0x80550000, 0x02000001, 0x80020000, 0xa0550001, 0x02000001, 0x80010002,
+ 0x80aa0000, 0x02000001, 0x80010003, 0x80ff0000, 0x02000001, 0x80020002,
+ 0xa0aa0001, 0x02000001, 0x80020003, 0xa0ff0001, 0x03000042, 0x800f0001,
+ 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0000, 0x80e40000, 0xa0e40801,
+ 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40801, 0x02000001, 0x80020001, 0x80000000, 0x02000001,
+ 0x80040001, 0x80000002, 0x03000005, 0x80070000, 0x80000003, 0x80e40001,
+ 0x02000001, 0x80080000, 0x80000003, 0x03000005, 0x800f0000, 0x80e40000,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn lookup3 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D baseImage;
+// sampler2D lookupTable;
+// float4 offset;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// offset c0 1
+// baseImage s0 1
+// lookupTable s1 1
+//
+
+ ps_2_0
+ def c1, 0.125, 0.375, 0.625, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ texld r0, t0, s0
+ mov r1.y, c1.x
+ rcp r1.z, r0.w
+ mad r2.xyz, r0, r1.z, -c0
+ mov r1.x, r2.x
+ mov r3.y, c1.y
+ mov r3.x, r2.y
+ mov r2.x, r2.z
+ mov r2.y, c1.z
+ texld r1, r1, s1
+ texld r3, r3, s1
+ texld r2, r2, s1
+ mov r1.y, r3.x
+ mov r1.z, r2.x
+ mul r0.xyz, r0.w, r1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 17 instruction slots used (4 texture, 13 arithmetic)
+#endif
+
+const DWORD lookup3[] =
+{
+ 0xffff0200, 0x0035fffe, 0x42415443, 0x0000001c, 0x0000009f, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000098, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000064, 0x00000000, 0x00000074, 0x00010003, 0x00060001,
+ 0x00000064, 0x00000000, 0x00000080, 0x00000002, 0x00020001, 0x00000088,
+ 0x00000000, 0x65736162, 0x67616d49, 0xabab0065, 0x000c0004, 0x00010001,
+ 0x00000001, 0x00000000, 0x6b6f6f6c, 0x61547075, 0x00656c62, 0x7366666f,
+ 0xab007465, 0x00030001, 0x00040001, 0x00000001, 0x00000000, 0x325f7370,
+ 0x4d00305f, 0x6f726369, 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c,
+ 0x72656461, 0x6d6f4320, 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934,
+ 0x00313131, 0x05000051, 0xa00f0001, 0x3e000000, 0x3ec00000, 0x3f200000,
+ 0x00000000, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x03000042, 0x800f0000, 0xb0e40000, 0xa0e40800, 0x02000001,
+ 0x80020001, 0xa0000001, 0x02000006, 0x80040001, 0x80ff0000, 0x04000004,
+ 0x80070002, 0x80e40000, 0x80aa0001, 0xa1e40000, 0x02000001, 0x80010001,
+ 0x80000002, 0x02000001, 0x80020003, 0xa0550001, 0x02000001, 0x80010003,
+ 0x80550002, 0x02000001, 0x80010002, 0x80aa0002, 0x02000001, 0x80020002,
+ 0xa0aa0001, 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042,
+ 0x800f0003, 0x80e40003, 0xa0e40801, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40801, 0x02000001, 0x80020001, 0x80000003, 0x02000001, 0x80040001,
+ 0x80000002, 0x03000005, 0x80070000, 0x80ff0000, 0x80e40001, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+const DWORD *lookupShaders[] =
+{
+ lookup0,
+ lookup1,
+ lookup2,
+ lookup3,
+};
+
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn grad0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// float4 color1;
+// float4 color2;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// params c0 1
+// color1 c1 1
+// color2 c2 1
+//
+
+ ps_2_0
+ def c3, 1, -0.25, 0, 0
+ dcl t0.xy
+ dcl v0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.x, c0, r0
+ add r0.x, r0.x, c3.y
+ add_sat r0.x, r0.x, r0.x
+ mov r1, c1
+ add r1, -r1, c2
+ mad r0, r0.x, r1, c1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 10 instruction slots used
+#endif
+
+const DWORD grad0[] =
+{
+ 0xffff0200, 0x0033fffe, 0x42415443, 0x0000001c, 0x00000097, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000090, 0x00000058, 0x00010002,
+ 0x00060001, 0x00000060, 0x00000000, 0x00000070, 0x00020002, 0x000a0001,
+ 0x00000060, 0x00000000, 0x00000077, 0x00000002, 0x00020001, 0x00000080,
+ 0x00000000, 0x6f6c6f63, 0xab003172, 0x00030001, 0x00040001, 0x00000001,
+ 0x00000000, 0x6f6c6f63, 0x70003272, 0x6d617261, 0xabab0073, 0x00030001,
+ 0x00030001, 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369,
+ 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320,
+ 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051,
+ 0xa00f0003, 0x3f800000, 0xbe800000, 0x00000000, 0x00000000, 0x0200001f,
+ 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x02000001,
+ 0x80030000, 0xb0e40000, 0x02000001, 0x80040000, 0xa0000003, 0x03000008,
+ 0x80010000, 0xa0e40000, 0x80e40000, 0x03000002, 0x80010000, 0x80000000,
+ 0xa0550003, 0x03000002, 0x80110000, 0x80000000, 0x80000000, 0x02000001,
+ 0x800f0001, 0xa0e40001, 0x03000002, 0x800f0001, 0x81e40001, 0xa0e40002,
+ 0x04000004, 0x800f0000, 0x80000000, 0x80e40001, 0xa0e40001, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn grad1 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// float4 color1;
+// float4 color2;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// params c0 1
+// color1 c1 1
+// color2 c2 1
+//
+
+ ps_2_0
+ def c3, 1, -0.25, -0.5, 2
+ dcl t0.xy
+ dcl v0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.x, c0, r0
+ add r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ mov r1, c1
+ add r1, -r1, c2
+ mad r0, r0.x, r1, c1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 13 instruction slots used
+#endif
+
+const DWORD grad1[] =
+{
+ 0xffff0200, 0x0033fffe, 0x42415443, 0x0000001c, 0x00000097, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x00000090, 0x00000058, 0x00010002,
+ 0x00060001, 0x00000060, 0x00000000, 0x00000070, 0x00020002, 0x000a0001,
+ 0x00000060, 0x00000000, 0x00000077, 0x00000002, 0x00020001, 0x00000080,
+ 0x00000000, 0x6f6c6f63, 0xab003172, 0x00030001, 0x00040001, 0x00000001,
+ 0x00000000, 0x6f6c6f63, 0x70003272, 0x6d617261, 0xabab0073, 0x00030001,
+ 0x00030001, 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369,
+ 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320,
+ 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051,
+ 0xa00f0003, 0x3f800000, 0xbe800000, 0xbf000000, 0x40000000, 0x0200001f,
+ 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x02000001,
+ 0x80030000, 0xb0e40000, 0x02000001, 0x80040000, 0xa0000003, 0x03000008,
+ 0x80010000, 0xa0e40000, 0x80e40000, 0x03000002, 0x80010000, 0x80000000,
+ 0xa0550003, 0x02000013, 0x80010000, 0x80000000, 0x03000002, 0x80010000,
+ 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000, 0x80000000, 0x04000004,
+ 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003, 0x02000001, 0x800f0001,
+ 0xa0e40001, 0x03000002, 0x800f0001, 0x81e40001, 0xa0e40002, 0x04000004,
+ 0x800f0000, 0x80000000, 0x80e40001, 0xa0e40001, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn grad2 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// float4 color1;
+// float4 color2;
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// params c0 1
+// color1 c1 1
+// color2 c2 1
+// mask s0 1
+//
+
+ ps_2_0
+ def c3, 1, -0.25, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ texld r0, t0, s0
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.x, c0, r0
+ add r0.x, r0.x, c3.y
+ add_sat r0.x, r0.x, r0.x
+ mov r1, c1
+ add r1, -r1, c2
+ mad r1, r0.x, r1, c1
+ mul r0, r0.w, r1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 12 instruction slots used (1 texture, 11 arithmetic)
+#endif
+
+const DWORD grad2[] =
+{
+ 0xffff0200, 0x003dfffe, 0x42415443, 0x0000001c, 0x000000bf, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000b8, 0x0000006c, 0x00010002,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00020002, 0x000a0001,
+ 0x00000074, 0x00000000, 0x0000008b, 0x00000003, 0x00020001, 0x00000090,
+ 0x00000000, 0x000000a0, 0x00000002, 0x00020001, 0x000000a8, 0x00000000,
+ 0x6f6c6f63, 0xab003172, 0x00030001, 0x00040001, 0x00000001, 0x00000000,
+ 0x6f6c6f63, 0x6d003272, 0x006b7361, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001, 0x00000001,
+ 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73, 0x29522820,
+ 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970, 0x2e392072,
+ 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003, 0x3f800000,
+ 0xbe800000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000, 0xb0030000,
+ 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000, 0x900f0000,
+ 0x0200001f, 0x90000000, 0xa00f0800, 0x03000042, 0x800f0000, 0xb0e40000,
+ 0xa0e40800, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40000, 0x80e40000, 0x03000002,
+ 0x80010000, 0x80000000, 0xa0550003, 0x03000002, 0x80110000, 0x80000000,
+ 0x80000000, 0x02000001, 0x800f0001, 0xa0e40001, 0x03000002, 0x800f0001,
+ 0x81e40001, 0xa0e40002, 0x04000004, 0x800f0001, 0x80000000, 0x80e40001,
+ 0xa0e40001, 0x03000005, 0x800f0000, 0x80ff0000, 0x80e40001, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn grad3 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// float4 color1;
+// float4 color2;
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// params c0 1
+// color1 c1 1
+// color2 c2 1
+// mask s0 1
+//
+
+ ps_2_0
+ def c3, 1, -0.25, -0.5, 2
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ texld r0, t0, s0
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.x, c0, r0
+ add r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ mov r1, c1
+ add r1, -r1, c2
+ mad r1, r0.x, r1, c1
+ mul r0, r0.w, r1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 15 instruction slots used (1 texture, 14 arithmetic)
+#endif
+
+const DWORD grad3[] =
+{
+ 0xffff0200, 0x003dfffe, 0x42415443, 0x0000001c, 0x000000bf, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000b8, 0x0000006c, 0x00010002,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00020002, 0x000a0001,
+ 0x00000074, 0x00000000, 0x0000008b, 0x00000003, 0x00020001, 0x00000090,
+ 0x00000000, 0x000000a0, 0x00000002, 0x00020001, 0x000000a8, 0x00000000,
+ 0x6f6c6f63, 0xab003172, 0x00030001, 0x00040001, 0x00000001, 0x00000000,
+ 0x6f6c6f63, 0x6d003272, 0x006b7361, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001, 0x00000001,
+ 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73, 0x29522820,
+ 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970, 0x2e392072,
+ 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003, 0x3f800000,
+ 0xbe800000, 0xbf000000, 0x40000000, 0x0200001f, 0x80000000, 0xb0030000,
+ 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000, 0x900f0000,
+ 0x0200001f, 0x90000000, 0xa00f0800, 0x03000042, 0x800f0000, 0xb0e40000,
+ 0xa0e40800, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40000, 0x80e40000, 0x03000002,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x02000001, 0x800f0001, 0xa0e40001, 0x03000002, 0x800f0001, 0x81e40001,
+ 0xa0e40002, 0x04000004, 0x800f0001, 0x80000000, 0x80e40001, 0xa0e40001,
+ 0x03000005, 0x800f0000, 0x80ff0000, 0x80e40001, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+const DWORD *gradShaders[] =
+{
+ grad0,
+ grad1,
+ grad2,
+ grad3,
+};
+
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c3.x
+ mov r0.xy, t0
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c3.x
+ mad r1.x, r0.w, c4.z, c4.w
+ add r0.w, r0.x, c3.w
+ mad r2.x, r0.y, c4.z, c4.w
+ mov r2.y, c3.z
+ mov r1.y, c3.z
+ mov r3.xy, c4
+ mov r4.xy, c3.yzxw
+ texld r2, r2, s0
+ texld r1, r1, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ lrp r5, r0.z, r1, r2
+ cmp r1, r0.w, r3, r5
+ cmp r0, -r0.x, r4, r1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 30 instruction slots used (4 texture, 26 arithmetic)
+#endif
+
+const DWORD linear0[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020003,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000004, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000003,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0004, 0xa0ff0004, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0003, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020002, 0xa0aa0003, 0x02000001,
+ 0x80020001, 0xa0aa0003, 0x02000001, 0x80030003, 0xa0e40004, 0x02000001,
+ 0x80030004, 0xa0c90003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800,
+ 0x04000012, 0x800f0005, 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058,
+ 0x800f0001, 0x80ff0000, 0x80e40003, 0x80e40005, 0x04000058, 0x800f0000,
+ 0x81000000, 0x80e40004, 0x80e40001, 0x03000005, 0x800f0000, 0x80e40000,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear1 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 28 instruction slots used (2 texture, 26 arithmetic)
+#endif
+
+const DWORD linear1[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020003,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000004, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear2 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 24 instruction slots used (2 texture, 22 arithmetic)
+#endif
+
+const DWORD linear2[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020003,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000004, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear4 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c7.x
+ mov r0.xy, t0
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c6.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c7.x
+ mad r1.x, r0.w, c8.z, c8.w
+ add r0.w, r0.x, c7.w
+ mad r2.x, r0.y, c8.z, c8.w
+ mov r2.y, c7.z
+ mov r1.y, c7.z
+ mov r3.xy, c8
+ mov r4.xy, c7.yzxw
+ texld r2, r2, s0
+ texld r1, r1, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ lrp r5, r0.z, r1, r2
+ cmp r1, r0.w, r3, r5
+ cmp r0, -r0.x, r4, r1
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 42 instruction slots used (4 texture, 38 arithmetic)
+#endif
+
+const DWORD linear4[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020007,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000008, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000006, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0008, 0xa0ff0008, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0007, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0008, 0xa0ff0008, 0x02000001, 0x80020002, 0xa0aa0007, 0x02000001,
+ 0x80020001, 0xa0aa0007, 0x02000001, 0x80030003, 0xa0e40008, 0x02000001,
+ 0x80030004, 0xa0c90007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800,
+ 0x04000012, 0x800f0005, 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058,
+ 0x800f0001, 0x80ff0000, 0x80e40003, 0x80e40005, 0x04000058, 0x800f0000,
+ 0x81000000, 0x80e40004, 0x80e40001, 0x03000005, 0x800f0000, 0x80e40000,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear5 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 40 instruction slots used (2 texture, 38 arithmetic)
+#endif
+
+const DWORD linear5[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020007,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000008, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear6 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 36 instruction slots used (2 texture, 34 arithmetic)
+#endif
+
+const DWORD linear6[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020007,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000008, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear8 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c3.x
+ mov r0.xy, t1
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c3.x
+ mad r1.x, r0.w, c4.z, c4.w
+ add r0.w, r0.x, c3.w
+ mad r2.x, r0.y, c4.z, c4.w
+ mov r2.y, c3.z
+ mov r1.y, c3.z
+ mov r3.xy, c4
+ mov r4.xy, c3.yzxw
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, r3, s1
+ texld r4, r4, s1
+ texld r5, t0, s0
+ lrp r6, r0.z, r1, r2
+ cmp r1, r0.w, r3, r6
+ cmp r0, -r0.x, r4, r1
+ mul r0, r5.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 32 instruction slots used (5 texture, 27 arithmetic)
+#endif
+
+const DWORD linear8[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020003,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000003,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0004, 0xa0ff0004, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0003, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020002, 0xa0aa0003, 0x02000001,
+ 0x80020001, 0xa0aa0003, 0x02000001, 0x80030003, 0xa0e40004, 0x02000001,
+ 0x80030004, 0xa0c90003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40801, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40801,
+ 0x03000042, 0x800f0005, 0xb0e40000, 0xa0e40800, 0x04000012, 0x800f0006,
+ 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058, 0x800f0001, 0x80ff0000,
+ 0x80e40003, 0x80e40006, 0x04000058, 0x800f0000, 0x81000000, 0x80e40004,
+ 0x80e40001, 0x03000005, 0x800f0000, 0x80ff0005, 0x80e40000, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear9 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 30 instruction slots used (3 texture, 27 arithmetic)
+#endif
+
+const DWORD linear9[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020003,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000,
+ 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear10 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 26 instruction slots used (3 texture, 23 arithmetic)
+#endif
+
+const DWORD linear10[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020003,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear12 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c7.x
+ mov r0.xy, t1
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c6.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c7.x
+ mad r1.x, r0.w, c8.z, c8.w
+ add r0.w, r0.x, c7.w
+ mad r2.x, r0.y, c8.z, c8.w
+ mov r2.y, c7.z
+ mov r1.y, c7.z
+ mov r3.xy, c8
+ mov r4.xy, c7.yzxw
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, r3, s1
+ texld r4, r4, s1
+ texld r5, t0, s0
+ lrp r6, r0.z, r1, r2
+ cmp r1, r0.w, r3, r6
+ cmp r0, -r0.x, r4, r1
+ mul r0, r5.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 44 instruction slots used (5 texture, 39 arithmetic)
+#endif
+
+const DWORD linear12[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020007,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000006, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0008, 0xa0ff0008, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0007, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0008, 0xa0ff0008, 0x02000001, 0x80020002, 0xa0aa0007, 0x02000001,
+ 0x80020001, 0xa0aa0007, 0x02000001, 0x80030003, 0xa0e40008, 0x02000001,
+ 0x80030004, 0xa0c90007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40801, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40801,
+ 0x03000042, 0x800f0005, 0xb0e40000, 0xa0e40800, 0x04000012, 0x800f0006,
+ 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058, 0x800f0001, 0x80ff0000,
+ 0x80e40003, 0x80e40006, 0x04000058, 0x800f0000, 0x81000000, 0x80e40004,
+ 0x80e40001, 0x03000005, 0x800f0000, 0x80ff0005, 0x80e40000, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear13 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 42 instruction slots used (3 texture, 39 arithmetic)
+#endif
+
+const DWORD linear13[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020007,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000,
+ 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear14 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 38 instruction slots used (3 texture, 35 arithmetic)
+#endif
+
+const DWORD linear14[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020007,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear16 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ def c5, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c3.x
+ mov r0.xy, t0
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c3.x
+ mad r1.x, r0.w, c4.z, c4.w
+ add r0.w, r0.x, c3.w
+ mad r2.x, r0.y, c4.z, c4.w
+ mov r2.y, c3.z
+ mov r1.y, c3.z
+ mov r3.xy, c4
+ mov r4.xy, c3.yzxw
+ texld r2, r2, s0
+ texld r1, r1, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ lrp r5, r0.z, r1, r2
+ cmp r1, r0.w, r3, r5
+ cmp r0, -r0.x, r4, r1
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c5.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c5.y, c5.z
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 38 instruction slots used (4 texture, 34 arithmetic)
+#endif
+
+const DWORD linear16[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020003,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000004, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0005,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000003,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0004, 0xa0ff0004, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0003, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020002, 0xa0aa0003, 0x02000001,
+ 0x80020001, 0xa0aa0003, 0x02000001, 0x80030003, 0xa0e40004, 0x02000001,
+ 0x80030004, 0xa0c90003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800,
+ 0x04000012, 0x800f0005, 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058,
+ 0x800f0001, 0x80ff0000, 0x80e40003, 0x80e40005, 0x04000058, 0x800f0000,
+ 0x81000000, 0x80e40004, 0x80e40001, 0x0200000f, 0x80010001, 0x80000000,
+ 0x0200000f, 0x80020001, 0x80550000, 0x0200000f, 0x80040001, 0x80aa0000,
+ 0x03000005, 0x80070001, 0x80e40001, 0xa0000005, 0x0200000e, 0x80010002,
+ 0x80000001, 0x0200000e, 0x80020002, 0x80550001, 0x0200000e, 0x80040002,
+ 0x80aa0001, 0x04000004, 0x80070000, 0x80e40002, 0xa0550005, 0xa0aa0005,
+ 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear17 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0.416667014, 0
+ def c5, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c4.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c5.x, c5.y
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 36 instruction slots used (2 texture, 34 arithmetic)
+#endif
+
+const DWORD linear17[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020003,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000004, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0005,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000003, 0x0200000f, 0x80020000, 0x80550003,
+ 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0aa0004, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070003,
+ 0x80e40001, 0xa0000005, 0xa0550005, 0x03000005, 0x800f0000, 0x80e40003,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear18 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ def c4, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c4.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c4.y, c4.z
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 32 instruction slots used (2 texture, 30 arithmetic)
+#endif
+
+const DWORD linear18[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020003,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000004, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0004,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000, 0x80000003, 0x0200000f,
+ 0x80020000, 0x80550003, 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005,
+ 0x80070000, 0x80e40000, 0xa0000004, 0x0200000e, 0x80010001, 0x80000000,
+ 0x0200000e, 0x80020001, 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000,
+ 0x04000004, 0x80070003, 0x80e40001, 0xa0550004, 0xa0aa0004, 0x03000005,
+ 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear20 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ def c9, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c7.x
+ mov r0.xy, t0
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c6.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c7.x
+ mad r1.x, r0.w, c8.z, c8.w
+ add r0.w, r0.x, c7.w
+ mad r2.x, r0.y, c8.z, c8.w
+ mov r2.y, c7.z
+ mov r1.y, c7.z
+ mov r3.xy, c8
+ mov r4.xy, c7.yzxw
+ texld r2, r2, s0
+ texld r1, r1, s0
+ texld r3, r3, s0
+ texld r4, r4, s0
+ lrp r5, r0.z, r1, r2
+ cmp r1, r0.w, r3, r5
+ cmp r0, -r0.x, r4, r1
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c9.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c9.y, c9.z
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 50 instruction slots used (4 texture, 46 arithmetic)
+#endif
+
+const DWORD linear20[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020007,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000008, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0009,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000006, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0008, 0xa0ff0008, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0007, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0008, 0xa0ff0008, 0x02000001, 0x80020002, 0xa0aa0007, 0x02000001,
+ 0x80020001, 0xa0aa0007, 0x02000001, 0x80030003, 0xa0e40008, 0x02000001,
+ 0x80030004, 0xa0c90007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40800, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40800,
+ 0x04000012, 0x800f0005, 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058,
+ 0x800f0001, 0x80ff0000, 0x80e40003, 0x80e40005, 0x04000058, 0x800f0000,
+ 0x81000000, 0x80e40004, 0x80e40001, 0x0200000f, 0x80010001, 0x80000000,
+ 0x0200000f, 0x80020001, 0x80550000, 0x0200000f, 0x80040001, 0x80aa0000,
+ 0x03000005, 0x80070001, 0x80e40001, 0xa0000009, 0x0200000e, 0x80010002,
+ 0x80000001, 0x0200000e, 0x80020002, 0x80550001, 0x0200000e, 0x80040002,
+ 0x80aa0001, 0x04000004, 0x80070000, 0x80e40002, 0xa0550009, 0xa0aa0009,
+ 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear21 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0.416667014, 0
+ def c9, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c8.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c9.x, c9.y
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 48 instruction slots used (2 texture, 46 arithmetic)
+#endif
+
+const DWORD linear21[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020007,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000008, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0009,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000003, 0x0200000f, 0x80020000, 0x80550003,
+ 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0aa0008, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070003,
+ 0x80e40001, 0xa0000009, 0xa0550009, 0x03000005, 0x800f0000, 0x80e40003,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear22 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ def c8, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c8.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c8.y, c8.z
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 44 instruction slots used (2 texture, 42 arithmetic)
+#endif
+
+const DWORD linear22[] =
+{
+ 0xffff0200, 0x0038fffe, 0x42415443, 0x0000001c, 0x000000ab, 0xffff0200,
+ 0x00000003, 0x0000001c, 0x20000100, 0x000000a4, 0x00000058, 0x00000003,
+ 0x00020001, 0x00000060, 0x00000000, 0x00000070, 0x00000002, 0x00020007,
+ 0x0000007c, 0x00000000, 0x0000008c, 0x00100002, 0x00420001, 0x00000094,
+ 0x00000000, 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001,
+ 0x00000000, 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001,
+ 0x00000008, 0x00000000, 0x61726170, 0xab00736d, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0008,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000, 0x80000003, 0x0200000f,
+ 0x80020000, 0x80550003, 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005,
+ 0x80070000, 0x80e40000, 0xa0000008, 0x0200000e, 0x80010001, 0x80000000,
+ 0x0200000e, 0x80020001, 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000,
+ 0x04000004, 0x80070003, 0x80e40001, 0xa0550008, 0xa0aa0008, 0x03000005,
+ 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear24 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ def c5, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c3.x
+ mov r0.xy, t1
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c3.x
+ mad r1.x, r0.w, c4.z, c4.w
+ add r0.w, r0.x, c3.w
+ mad r2.x, r0.y, c4.z, c4.w
+ mov r2.y, c3.z
+ mov r1.y, c3.z
+ mov r3.xy, c4
+ mov r4.xy, c3.yzxw
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, r3, s1
+ texld r4, r4, s1
+ texld r5, t0, s0
+ lrp r6, r0.z, r1, r2
+ cmp r1, r0.w, r3, r6
+ cmp r0, -r0.x, r4, r1
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c5.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c5.y, c5.z
+ mul r0, r5.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 40 instruction slots used (5 texture, 35 arithmetic)
+#endif
+
+const DWORD linear24[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020003,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0005,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000003,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0004, 0xa0ff0004, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0003, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020002, 0xa0aa0003, 0x02000001,
+ 0x80020001, 0xa0aa0003, 0x02000001, 0x80030003, 0xa0e40004, 0x02000001,
+ 0x80030004, 0xa0c90003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40801, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40801,
+ 0x03000042, 0x800f0005, 0xb0e40000, 0xa0e40800, 0x04000012, 0x800f0006,
+ 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058, 0x800f0001, 0x80ff0000,
+ 0x80e40003, 0x80e40006, 0x04000058, 0x800f0000, 0x81000000, 0x80e40004,
+ 0x80e40001, 0x0200000f, 0x80010001, 0x80000000, 0x0200000f, 0x80020001,
+ 0x80550000, 0x0200000f, 0x80040001, 0x80aa0000, 0x03000005, 0x80070001,
+ 0x80e40001, 0xa0000005, 0x0200000e, 0x80010002, 0x80000001, 0x0200000e,
+ 0x80020002, 0x80550001, 0x0200000e, 0x80040002, 0x80aa0001, 0x04000004,
+ 0x80070000, 0x80e40002, 0xa0550005, 0xa0aa0005, 0x03000005, 0x800f0000,
+ 0x80ff0005, 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear25 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0.416667014, 0
+ def c5, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c4.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c5.x, c5.y
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 38 instruction slots used (3 texture, 35 arithmetic)
+#endif
+
+const DWORD linear25[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020003,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0005,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000,
+ 0x80000004, 0x0200000f, 0x80020000, 0x80550004, 0x0200000f, 0x80040000,
+ 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000, 0xa0aa0004, 0x0200000e,
+ 0x80010001, 0x80000000, 0x0200000e, 0x80020001, 0x80550000, 0x0200000e,
+ 0x80040001, 0x80aa0000, 0x04000004, 0x80070004, 0x80e40001, 0xa0000005,
+ 0xa0550005, 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear26 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ def c4, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c4.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c4.y, c4.z
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 34 instruction slots used (3 texture, 31 arithmetic)
+#endif
+
+const DWORD linear26[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020003,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0004,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000004, 0x0200000f, 0x80020000, 0x80550004,
+ 0x0200000f, 0x80040000, 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0000004, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070004,
+ 0x80e40001, 0xa0550004, 0xa0aa0004, 0x03000005, 0x800f0000, 0x80ff0003,
+ 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear28 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ def c9, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c7.x
+ mov r0.xy, t1
+ dp3 r0.x, c16, r0
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c6.y
+ add r0.y, r0.y, r0.z
+ frc r0.z, r0.y
+ add r0.y, r0.y, -r0.z
+ add r0.w, r0.y, c7.x
+ mad r1.x, r0.w, c8.z, c8.w
+ add r0.w, r0.x, c7.w
+ mad r2.x, r0.y, c8.z, c8.w
+ mov r2.y, c7.z
+ mov r1.y, c7.z
+ mov r3.xy, c8
+ mov r4.xy, c7.yzxw
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, r3, s1
+ texld r4, r4, s1
+ texld r5, t0, s0
+ lrp r6, r0.z, r1, r2
+ cmp r1, r0.w, r3, r6
+ cmp r0, -r0.x, r4, r1
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c9.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c9.y, c9.z
+ mul r0, r5.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 52 instruction slots used (5 texture, 47 arithmetic)
+#endif
+
+const DWORD linear28[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020007,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0009,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000002,
+ 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000006, 0x03000005,
+ 0x80140000, 0x80aa0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80aa0000, 0x02000013, 0x80040000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81aa0000, 0x03000002, 0x80080000, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0000, 0xa0aa0008, 0xa0ff0008, 0x03000002,
+ 0x80080000, 0x80000000, 0xa0ff0007, 0x04000004, 0x80010002, 0x80550000,
+ 0xa0aa0008, 0xa0ff0008, 0x02000001, 0x80020002, 0xa0aa0007, 0x02000001,
+ 0x80020001, 0xa0aa0007, 0x02000001, 0x80030003, 0xa0e40008, 0x02000001,
+ 0x80030004, 0xa0c90007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801,
+ 0x03000042, 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003,
+ 0x80e40003, 0xa0e40801, 0x03000042, 0x800f0004, 0x80e40004, 0xa0e40801,
+ 0x03000042, 0x800f0005, 0xb0e40000, 0xa0e40800, 0x04000012, 0x800f0006,
+ 0x80aa0000, 0x80e40001, 0x80e40002, 0x04000058, 0x800f0001, 0x80ff0000,
+ 0x80e40003, 0x80e40006, 0x04000058, 0x800f0000, 0x81000000, 0x80e40004,
+ 0x80e40001, 0x0200000f, 0x80010001, 0x80000000, 0x0200000f, 0x80020001,
+ 0x80550000, 0x0200000f, 0x80040001, 0x80aa0000, 0x03000005, 0x80070001,
+ 0x80e40001, 0xa0000009, 0x0200000e, 0x80010002, 0x80000001, 0x0200000e,
+ 0x80020002, 0x80550001, 0x0200000e, 0x80040002, 0x80aa0001, 0x04000004,
+ 0x80070000, 0x80e40002, 0xa0550009, 0xa0aa0009, 0x03000005, 0x800f0000,
+ 0x80ff0005, 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear29 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0.416667014, 0
+ def c9, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c8.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c9.x, c9.y
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 50 instruction slots used (3 texture, 47 arithmetic)
+#endif
+
+const DWORD linear29[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020007,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0009,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000,
+ 0x80000004, 0x0200000f, 0x80020000, 0x80550004, 0x0200000f, 0x80040000,
+ 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000, 0xa0aa0008, 0x0200000e,
+ 0x80010001, 0x80000000, 0x0200000e, 0x80020001, 0x80550000, 0x0200000e,
+ 0x80040001, 0x80aa0000, 0x04000004, 0x80070004, 0x80e40001, 0xa0000009,
+ 0xa0550009, 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn linear30 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// sampler2D mask;
+// float3 params;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// params c16 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ def c8, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.x, c16, r0
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c8.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c8.y, c8.z
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 46 instruction slots used (3 texture, 43 arithmetic)
+#endif
+
+const DWORD linear30[] =
+{
+ 0xffff0200, 0x003efffe, 0x42415443, 0x0000001c, 0x000000c3, 0xffff0200,
+ 0x00000004, 0x0000001c, 0x20000100, 0x000000bc, 0x0000006c, 0x00010003,
+ 0x00060001, 0x00000074, 0x00000000, 0x00000084, 0x00000002, 0x00020007,
+ 0x00000090, 0x00000000, 0x000000a0, 0x00000003, 0x00020001, 0x00000074,
+ 0x00000000, 0x000000a5, 0x00100002, 0x00420001, 0x000000ac, 0x00000000,
+ 0x6f6c6f63, 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000,
+ 0x63617266, 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008,
+ 0x00000000, 0x6b73616d, 0x72617000, 0x00736d61, 0x00030001, 0x00030001,
+ 0x00000001, 0x00000000, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0008,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80010000, 0xa0e40010, 0x80e40000, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000004, 0x0200000f, 0x80020000, 0x80550004,
+ 0x0200000f, 0x80040000, 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0000008, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070004,
+ 0x80e40001, 0xa0550008, 0xa0aa0008, 0x03000005, 0x800f0000, 0x80ff0003,
+ 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+const DWORD *linearShaders[] =
+{
+ linear0,
+ linear1,
+ linear2,
+ NULL,
+ linear4,
+ linear5,
+ linear6,
+ NULL,
+ linear8,
+ linear9,
+ linear10,
+ NULL,
+ linear12,
+ linear13,
+ linear14,
+ NULL,
+ linear16,
+ linear17,
+ linear18,
+ NULL,
+ linear20,
+ linear21,
+ linear22,
+ NULL,
+ linear24,
+ linear25,
+ linear26,
+ NULL,
+ linear28,
+ linear29,
+ linear30,
+ NULL,
+};
+
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c3.x
+ mov r0.xy, t0
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c3.x
+ mad r1.x, r1.w, c4.z, c4.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c3.w
+ mad r0.x, r0.y, c4.z, c4.w
+ mov r0.y, c3.z
+ mov r1.y, c3.z
+ mov r2.xy, c4
+ mov r3.xy, c3.yzxw
+ texld r4, r0, s0
+ texld r5, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ lrp r6, r0.w, r5, r4
+ cmp r0, r1.w, r2, r6
+ cmp r0, -r1.z, r3, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 40 instruction slots used (4 texture, 36 arithmetic)
+#endif
+
+const DWORD radial0[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020003,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x02000013, 0x80080000, 0x80550000,
+ 0x03000002, 0x80020000, 0x80550000, 0x81ff0000, 0x03000002, 0x80080001,
+ 0x80550000, 0xa0000003, 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0004,
+ 0xa0ff0004, 0x03000005, 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004,
+ 0x80080001, 0x80000000, 0x80aa0000, 0xa0ff0003, 0x04000004, 0x80010000,
+ 0x80550000, 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020000, 0xa0aa0003,
+ 0x02000001, 0x80020001, 0xa0aa0003, 0x02000001, 0x80030002, 0xa0e40004,
+ 0x02000001, 0x80030003, 0xa0c90003, 0x03000042, 0x800f0004, 0x80e40000,
+ 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40001, 0xa0e40800, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003,
+ 0xa0e40800, 0x04000012, 0x800f0006, 0x80ff0000, 0x80e40005, 0x80e40004,
+ 0x04000058, 0x800f0000, 0x80ff0001, 0x80e40002, 0x80e40006, 0x04000058,
+ 0x800f0000, 0x81aa0001, 0x80e40003, 0x80e40000, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial1 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 37 instruction slots used (2 texture, 35 arithmetic)
+#endif
+
+const DWORD radial1[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020003,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial2 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 33 instruction slots used (2 texture, 31 arithmetic)
+#endif
+
+const DWORD radial2[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020003,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial4 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c7.x
+ mov r0.xy, t0
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c3.x
+ mul_sat r0.w, r0.w, c3.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c4.x
+ mul_sat r0.w, r0.w, c4.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c5.x
+ mul_sat r0.w, r0.w, c5.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c6.x
+ mul_sat r0.w, r0.w, c6.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c7.x
+ mad r1.x, r1.w, c8.z, c8.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c7.w
+ mad r0.x, r0.y, c8.z, c8.w
+ mov r0.y, c7.z
+ mov r1.y, c7.z
+ mov r2.xy, c8
+ mov r3.xy, c7.yzxw
+ texld r4, r0, s0
+ texld r5, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ lrp r6, r0.w, r5, r4
+ cmp r0, r1.w, r2, r6
+ cmp r0, -r1.z, r3, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 52 instruction slots used (4 texture, 48 arithmetic)
+#endif
+
+const DWORD radial4[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020007,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000,
+ 0x80aa0000, 0xa1000003, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000,
+ 0x80000000, 0x80aa0000, 0xa1000004, 0x03000005, 0x80180000, 0x80ff0000,
+ 0xa0550004, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004,
+ 0x80080000, 0x80000000, 0x80aa0000, 0xa1000005, 0x03000005, 0x80180000,
+ 0x80ff0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000,
+ 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000006, 0x03000005,
+ 0x80180000, 0x80ff0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80ff0000, 0x02000013, 0x80080000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81ff0000, 0x03000002, 0x80080001, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0008, 0xa0ff0008, 0x03000005,
+ 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004, 0x80080001, 0x80000000,
+ 0x80aa0000, 0xa0ff0007, 0x04000004, 0x80010000, 0x80550000, 0xa0aa0008,
+ 0xa0ff0008, 0x02000001, 0x80020000, 0xa0aa0007, 0x02000001, 0x80020001,
+ 0xa0aa0007, 0x02000001, 0x80030002, 0xa0e40008, 0x02000001, 0x80030003,
+ 0xa0c90007, 0x03000042, 0x800f0004, 0x80e40000, 0xa0e40800, 0x03000042,
+ 0x800f0005, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x04000012,
+ 0x800f0006, 0x80ff0000, 0x80e40005, 0x80e40004, 0x04000058, 0x800f0000,
+ 0x80ff0001, 0x80e40002, 0x80e40006, 0x04000058, 0x800f0000, 0x81aa0001,
+ 0x80e40003, 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial5 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 49 instruction slots used (2 texture, 47 arithmetic)
+#endif
+
+const DWORD radial5[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020007,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial6 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 45 instruction slots used (2 texture, 43 arithmetic)
+#endif
+
+const DWORD radial6[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020007,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000, 0x80e40003, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial8 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c3.x
+ mov r0.xy, t1
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c3.x
+ mad r1.x, r1.w, c4.z, c4.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c3.w
+ mad r0.x, r0.y, c4.z, c4.w
+ mov r0.y, c3.z
+ mov r1.y, c3.z
+ mov r2.xy, c4
+ mov r3.xy, c3.yzxw
+ texld r4, r0, s1
+ texld r5, r1, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ texld r6, t0, s0
+ lrp r7, r0.w, r5, r4
+ cmp r0, r1.w, r2, r7
+ cmp r0, -r1.z, r3, r0
+ mul r0, r6.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 42 instruction slots used (5 texture, 37 arithmetic)
+#endif
+
+const DWORD radial8[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020003,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x02000013, 0x80080000, 0x80550000,
+ 0x03000002, 0x80020000, 0x80550000, 0x81ff0000, 0x03000002, 0x80080001,
+ 0x80550000, 0xa0000003, 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0004,
+ 0xa0ff0004, 0x03000005, 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004,
+ 0x80080001, 0x80000000, 0x80aa0000, 0xa0ff0003, 0x04000004, 0x80010000,
+ 0x80550000, 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020000, 0xa0aa0003,
+ 0x02000001, 0x80020001, 0xa0aa0003, 0x02000001, 0x80030002, 0xa0e40004,
+ 0x02000001, 0x80030003, 0xa0c90003, 0x03000042, 0x800f0004, 0x80e40000,
+ 0xa0e40801, 0x03000042, 0x800f0005, 0x80e40001, 0xa0e40801, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0003, 0x80e40003,
+ 0xa0e40801, 0x03000042, 0x800f0006, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0007, 0x80ff0000, 0x80e40005, 0x80e40004, 0x04000058, 0x800f0000,
+ 0x80ff0001, 0x80e40002, 0x80e40007, 0x04000058, 0x800f0000, 0x81aa0001,
+ 0x80e40003, 0x80e40000, 0x03000005, 0x800f0000, 0x80ff0006, 0x80e40000,
+ 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800,
+ 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial9 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 39 instruction slots used (3 texture, 36 arithmetic)
+#endif
+
+const DWORD radial9[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020003,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000,
+ 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial10 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 35 instruction slots used (3 texture, 32 arithmetic)
+#endif
+
+const DWORD radial10[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020003,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial12 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c7.x
+ mov r0.xy, t1
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c3.x
+ mul_sat r0.w, r0.w, c3.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c4.x
+ mul_sat r0.w, r0.w, c4.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c5.x
+ mul_sat r0.w, r0.w, c5.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c6.x
+ mul_sat r0.w, r0.w, c6.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c7.x
+ mad r1.x, r1.w, c8.z, c8.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c7.w
+ mad r0.x, r0.y, c8.z, c8.w
+ mov r0.y, c7.z
+ mov r1.y, c7.z
+ mov r2.xy, c8
+ mov r3.xy, c7.yzxw
+ texld r4, r0, s1
+ texld r5, r1, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ texld r6, t0, s0
+ lrp r7, r0.w, r5, r4
+ cmp r0, r1.w, r2, r7
+ cmp r0, -r1.z, r3, r0
+ mul r0, r6.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 54 instruction slots used (5 texture, 49 arithmetic)
+#endif
+
+const DWORD radial12[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020007,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000,
+ 0x80aa0000, 0xa1000003, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000,
+ 0x80000000, 0x80aa0000, 0xa1000004, 0x03000005, 0x80180000, 0x80ff0000,
+ 0xa0550004, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004,
+ 0x80080000, 0x80000000, 0x80aa0000, 0xa1000005, 0x03000005, 0x80180000,
+ 0x80ff0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000,
+ 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000006, 0x03000005,
+ 0x80180000, 0x80ff0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80ff0000, 0x02000013, 0x80080000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81ff0000, 0x03000002, 0x80080001, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0008, 0xa0ff0008, 0x03000005,
+ 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004, 0x80080001, 0x80000000,
+ 0x80aa0000, 0xa0ff0007, 0x04000004, 0x80010000, 0x80550000, 0xa0aa0008,
+ 0xa0ff0008, 0x02000001, 0x80020000, 0xa0aa0007, 0x02000001, 0x80020001,
+ 0xa0aa0007, 0x02000001, 0x80030002, 0xa0e40008, 0x02000001, 0x80030003,
+ 0xa0c90007, 0x03000042, 0x800f0004, 0x80e40000, 0xa0e40801, 0x03000042,
+ 0x800f0005, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40801, 0x03000042,
+ 0x800f0006, 0xb0e40000, 0xa0e40800, 0x04000012, 0x800f0007, 0x80ff0000,
+ 0x80e40005, 0x80e40004, 0x04000058, 0x800f0000, 0x80ff0001, 0x80e40002,
+ 0x80e40007, 0x04000058, 0x800f0000, 0x81aa0001, 0x80e40003, 0x80e40000,
+ 0x03000005, 0x800f0000, 0x80ff0006, 0x80e40000, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial13 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 51 instruction slots used (3 texture, 48 arithmetic)
+#endif
+
+const DWORD radial13[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020007,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x03000005, 0x800f0000,
+ 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000,
+ 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial14 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 47 instruction slots used (3 texture, 44 arithmetic)
+#endif
+
+const DWORD radial14[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020007,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005, 0x800f0000,
+ 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial16 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ def c5, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c3.x
+ mov r0.xy, t0
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c3.x
+ mad r1.x, r1.w, c4.z, c4.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c3.w
+ mad r0.x, r0.y, c4.z, c4.w
+ mov r0.y, c3.z
+ mov r1.y, c3.z
+ mov r2.xy, c4
+ mov r3.xy, c3.yzxw
+ texld r4, r0, s0
+ texld r5, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ lrp r6, r0.w, r5, r4
+ cmp r0, r1.w, r2, r6
+ cmp r0, -r1.z, r3, r0
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c5.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c5.y, c5.z
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 48 instruction slots used (4 texture, 44 arithmetic)
+#endif
+
+const DWORD radial16[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020003,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0005,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x02000013, 0x80080000, 0x80550000,
+ 0x03000002, 0x80020000, 0x80550000, 0x81ff0000, 0x03000002, 0x80080001,
+ 0x80550000, 0xa0000003, 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0004,
+ 0xa0ff0004, 0x03000005, 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004,
+ 0x80080001, 0x80000000, 0x80aa0000, 0xa0ff0003, 0x04000004, 0x80010000,
+ 0x80550000, 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020000, 0xa0aa0003,
+ 0x02000001, 0x80020001, 0xa0aa0003, 0x02000001, 0x80030002, 0xa0e40004,
+ 0x02000001, 0x80030003, 0xa0c90003, 0x03000042, 0x800f0004, 0x80e40000,
+ 0xa0e40800, 0x03000042, 0x800f0005, 0x80e40001, 0xa0e40800, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003,
+ 0xa0e40800, 0x04000012, 0x800f0006, 0x80ff0000, 0x80e40005, 0x80e40004,
+ 0x04000058, 0x800f0000, 0x80ff0001, 0x80e40002, 0x80e40006, 0x04000058,
+ 0x800f0000, 0x81aa0001, 0x80e40003, 0x80e40000, 0x0200000f, 0x80010001,
+ 0x80000000, 0x0200000f, 0x80020001, 0x80550000, 0x0200000f, 0x80040001,
+ 0x80aa0000, 0x03000005, 0x80070001, 0x80e40001, 0xa0000005, 0x0200000e,
+ 0x80010002, 0x80000001, 0x0200000e, 0x80020002, 0x80550001, 0x0200000e,
+ 0x80040002, 0x80aa0001, 0x04000004, 0x80070000, 0x80e40002, 0xa0550005,
+ 0xa0aa0005, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial17 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0.416667014, 0
+ def c5, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c4.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c5.x, c5.y
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 45 instruction slots used (2 texture, 43 arithmetic)
+#endif
+
+const DWORD radial17[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020003,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0005,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000003, 0x0200000f, 0x80020000, 0x80550003,
+ 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0aa0004, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070003,
+ 0x80e40001, 0xa0000005, 0xa0550005, 0x03000005, 0x800f0000, 0x80e40003,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial18 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ def c4, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c4.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c4.y, c4.z
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 41 instruction slots used (2 texture, 39 arithmetic)
+#endif
+
+const DWORD radial18[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020003,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0004,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000, 0x80000003, 0x0200000f,
+ 0x80020000, 0x80550003, 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005,
+ 0x80070000, 0x80e40000, 0xa0000004, 0x0200000e, 0x80010001, 0x80000000,
+ 0x0200000e, 0x80020001, 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000,
+ 0x04000004, 0x80070003, 0x80e40001, 0xa0550004, 0xa0aa0004, 0x03000005,
+ 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial20 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ def c9, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.z, c7.x
+ mov r0.xy, t0
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c3.x
+ mul_sat r0.w, r0.w, c3.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c4.x
+ mul_sat r0.w, r0.w, c4.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c5.x
+ mul_sat r0.w, r0.w, c5.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c6.x
+ mul_sat r0.w, r0.w, c6.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c7.x
+ mad r1.x, r1.w, c8.z, c8.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c7.w
+ mad r0.x, r0.y, c8.z, c8.w
+ mov r0.y, c7.z
+ mov r1.y, c7.z
+ mov r2.xy, c8
+ mov r3.xy, c7.yzxw
+ texld r4, r0, s0
+ texld r5, r1, s0
+ texld r2, r2, s0
+ texld r3, r3, s0
+ lrp r6, r0.w, r5, r4
+ cmp r0, r1.w, r2, r6
+ cmp r0, -r1.z, r3, r0
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c9.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c9.y, c9.z
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 60 instruction slots used (4 texture, 56 arithmetic)
+#endif
+
+const DWORD radial20[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020007,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0009,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40000, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000,
+ 0x80aa0000, 0xa1000003, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000,
+ 0x80000000, 0x80aa0000, 0xa1000004, 0x03000005, 0x80180000, 0x80ff0000,
+ 0xa0550004, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004,
+ 0x80080000, 0x80000000, 0x80aa0000, 0xa1000005, 0x03000005, 0x80180000,
+ 0x80ff0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000,
+ 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000006, 0x03000005,
+ 0x80180000, 0x80ff0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80ff0000, 0x02000013, 0x80080000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81ff0000, 0x03000002, 0x80080001, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0008, 0xa0ff0008, 0x03000005,
+ 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004, 0x80080001, 0x80000000,
+ 0x80aa0000, 0xa0ff0007, 0x04000004, 0x80010000, 0x80550000, 0xa0aa0008,
+ 0xa0ff0008, 0x02000001, 0x80020000, 0xa0aa0007, 0x02000001, 0x80020001,
+ 0xa0aa0007, 0x02000001, 0x80030002, 0xa0e40008, 0x02000001, 0x80030003,
+ 0xa0c90007, 0x03000042, 0x800f0004, 0x80e40000, 0xa0e40800, 0x03000042,
+ 0x800f0005, 0x80e40001, 0xa0e40800, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40800, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40800, 0x04000012,
+ 0x800f0006, 0x80ff0000, 0x80e40005, 0x80e40004, 0x04000058, 0x800f0000,
+ 0x80ff0001, 0x80e40002, 0x80e40006, 0x04000058, 0x800f0000, 0x81aa0001,
+ 0x80e40003, 0x80e40000, 0x0200000f, 0x80010001, 0x80000000, 0x0200000f,
+ 0x80020001, 0x80550000, 0x0200000f, 0x80040001, 0x80aa0000, 0x03000005,
+ 0x80070001, 0x80e40001, 0xa0000009, 0x0200000e, 0x80010002, 0x80000001,
+ 0x0200000e, 0x80020002, 0x80550001, 0x0200000e, 0x80040002, 0x80aa0001,
+ 0x04000004, 0x80070000, 0x80e40002, 0xa0550009, 0xa0aa0009, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial21 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0.416667014, 0
+ def c9, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c8.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c9.x, c9.y
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 57 instruction slots used (2 texture, 55 arithmetic)
+#endif
+
+const DWORD radial21[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020007,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0009,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000003, 0x0200000f, 0x80020000, 0x80550003,
+ 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0aa0008, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070003,
+ 0x80e40001, 0xa0000009, 0xa0550009, 0x03000005, 0x800f0000, 0x80e40003,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial22 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// colors s0 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ def c8, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl v0
+ dcl_2d s0
+ mov r0.xy, t0
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s0
+ texld r1, r1, s0
+ lrp r3, r0.y, r2, r1
+ log r0.x, r3.x
+ log r0.y, r3.y
+ log r0.z, r3.z
+ mul r0.xyz, r0, c8.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r3.xyz, r1, c8.y, c8.z
+ mul r0, r3, v0
+ mov oC0, r0
+
+// approximately 53 instruction slots used (2 texture, 51 arithmetic)
+#endif
+
+const DWORD radial22[] =
+{
+ 0xffff0200, 0x0044fffe, 0x42415443, 0x0000001c, 0x000000da, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000d3, 0x00000080, 0x00000003,
+ 0x00020001, 0x00000088, 0x00000000, 0x00000098, 0x00000002, 0x00020007,
+ 0x000000a4, 0x00000000, 0x000000b4, 0x00100002, 0x00420001, 0x000000b8,
+ 0x00000000, 0x000000c8, 0x00110002, 0x00460001, 0x000000b8, 0x00000000,
+ 0x000000cb, 0x00120002, 0x004a0001, 0x000000b8, 0x00000000, 0x6f6c6f63,
+ 0xab007372, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266,
+ 0x6e6f6974, 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000,
+ 0xab00306d, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x7000316d,
+ 0x61636572, 0x7000636c, 0x5f325f73, 0x694d0030, 0x736f7263, 0x2074666f,
+ 0x20295228, 0x4c534c48, 0x61685320, 0x20726564, 0x706d6f43, 0x72656c69,
+ 0x312e3920, 0x34392e39, 0x31322e39, 0xab003131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0008,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0x900f0000, 0x0200001f, 0x90000000,
+ 0xa00f0800, 0x02000001, 0x80030000, 0xb0e40000, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40800, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40800, 0x04000012, 0x800f0003, 0x80550000,
+ 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000, 0x80000003, 0x0200000f,
+ 0x80020000, 0x80550003, 0x0200000f, 0x80040000, 0x80aa0003, 0x03000005,
+ 0x80070000, 0x80e40000, 0xa0000008, 0x0200000e, 0x80010001, 0x80000000,
+ 0x0200000e, 0x80020001, 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000,
+ 0x04000004, 0x80070003, 0x80e40001, 0xa0550008, 0xa0aa0008, 0x03000005,
+ 0x800f0000, 0x80e40003, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial24 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0, 0.5, -1
+ def c4, 1, 0.5, 0.0625, 0.03125
+ def c5, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c3.x
+ mov r0.xy, t1
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c3.x
+ mad r1.x, r1.w, c4.z, c4.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c3.w
+ mad r0.x, r0.y, c4.z, c4.w
+ mov r0.y, c3.z
+ mov r1.y, c3.z
+ mov r2.xy, c4
+ mov r3.xy, c3.yzxw
+ texld r4, r0, s1
+ texld r5, r1, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ texld r6, t0, s0
+ lrp r7, r0.w, r5, r4
+ cmp r0, r1.w, r2, r7
+ cmp r0, -r1.z, r3, r0
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c5.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c5.y, c5.z
+ mul r0, r6.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 50 instruction slots used (5 texture, 45 arithmetic)
+#endif
+
+const DWORD radial24[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020003,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0004,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0005,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000003, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x02000013, 0x80080000, 0x80550000,
+ 0x03000002, 0x80020000, 0x80550000, 0x81ff0000, 0x03000002, 0x80080001,
+ 0x80550000, 0xa0000003, 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0004,
+ 0xa0ff0004, 0x03000005, 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004,
+ 0x80080001, 0x80000000, 0x80aa0000, 0xa0ff0003, 0x04000004, 0x80010000,
+ 0x80550000, 0xa0aa0004, 0xa0ff0004, 0x02000001, 0x80020000, 0xa0aa0003,
+ 0x02000001, 0x80020001, 0xa0aa0003, 0x02000001, 0x80030002, 0xa0e40004,
+ 0x02000001, 0x80030003, 0xa0c90003, 0x03000042, 0x800f0004, 0x80e40000,
+ 0xa0e40801, 0x03000042, 0x800f0005, 0x80e40001, 0xa0e40801, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0003, 0x80e40003,
+ 0xa0e40801, 0x03000042, 0x800f0006, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0007, 0x80ff0000, 0x80e40005, 0x80e40004, 0x04000058, 0x800f0000,
+ 0x80ff0001, 0x80e40002, 0x80e40007, 0x04000058, 0x800f0000, 0x81aa0001,
+ 0x80e40003, 0x80e40000, 0x0200000f, 0x80010001, 0x80000000, 0x0200000f,
+ 0x80020001, 0x80550000, 0x0200000f, 0x80040001, 0x80aa0000, 0x03000005,
+ 0x80070001, 0x80e40001, 0xa0000005, 0x0200000e, 0x80010002, 0x80000001,
+ 0x0200000e, 0x80020002, 0x80550001, 0x0200000e, 0x80040002, 0x80aa0001,
+ 0x04000004, 0x80070000, 0x80e40002, 0xa0550005, 0xa0aa0005, 0x03000005,
+ 0x800f0000, 0x80ff0006, 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000,
+ 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial25 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.5, -0.5, 2
+ def c4, 0.0625, 0.03125, 0.416667014, 0
+ def c5, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c3.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c3.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c3.w, c3.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c4.x, c4.y
+ mad r2.x, r0.z, c4.x, c4.y
+ mov r2.y, c3.y
+ mov r1.y, c3.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c4.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c5.x, c5.y
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 47 instruction slots used (3 texture, 44 arithmetic)
+#endif
+
+const DWORD radial25[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020003,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0004,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0005,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550003, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0003, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0003, 0xa0000003,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000002, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550002, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000003, 0x04000004, 0x80010001, 0x80000000, 0xa0000004, 0xa0550004,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000004, 0xa0550004, 0x02000001,
+ 0x80020002, 0xa0550003, 0x02000001, 0x80020001, 0xa0550003, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000,
+ 0x80000004, 0x0200000f, 0x80020000, 0x80550004, 0x0200000f, 0x80040000,
+ 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000, 0xa0aa0004, 0x0200000e,
+ 0x80010001, 0x80000000, 0x0200000e, 0x80020001, 0x80550000, 0x0200000e,
+ 0x80040001, 0x80aa0000, 0x04000004, 0x80070004, 0x80e40001, 0xa0000005,
+ 0xa0550005, 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial26 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[4];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 3
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c3, 1, 0.0625, 0.03125, 0.5
+ def c4, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c3.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ add r0.x, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c2.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c3.x
+ mad r1.x, r0.x, c3.y, c3.z
+ mad r2.x, r0.z, c3.y, c3.z
+ mov r2.y, c3.w
+ mov r1.y, c3.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c4.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c4.y, c4.z
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 43 instruction slots used (3 texture, 40 arithmetic)
+#endif
+
+const DWORD radial26[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020003,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000004, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0003,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0004,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000003, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000002, 0x80010000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550002,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000003, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550003, 0xa0aa0003, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550003,
+ 0xa0aa0003, 0x02000001, 0x80020002, 0xa0ff0003, 0x02000001, 0x80020001,
+ 0xa0ff0003, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000004, 0x0200000f, 0x80020000, 0x80550004,
+ 0x0200000f, 0x80040000, 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0000004, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070004,
+ 0x80e40001, 0xa0550004, 0xa0aa0004, 0x03000005, 0x800f0000, 0x80ff0003,
+ 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial28 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0, 0.5, -1
+ def c8, 1, 0.5, 0.0625, 0.03125
+ def c9, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.z, c7.x
+ mov r0.xy, t1
+ dp3 r0.w, r0, c17
+ mul r0.w, r0.w, r0.w
+ mul r0.w, r0.w, c18.y
+ dp3 r0.x, r0, c16
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.w
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mov r0.z, c18.z
+ mad r0.y, r0.x, r0.z, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ mad r0.w, r0.x, r0.z, -c1.x
+ mul_sat r0.w, r0.w, c1.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c2.x
+ mul_sat r0.w, r0.w, c2.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c3.x
+ mul_sat r0.w, r0.w, c3.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c4.x
+ mul_sat r0.w, r0.w, c4.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c5.x
+ mul_sat r0.w, r0.w, c5.y
+ add r0.y, r0.y, r0.w
+ mad r0.w, r0.x, r0.z, -c6.x
+ mul_sat r0.w, r0.w, c6.y
+ add r0.y, r0.y, r0.w
+ frc r0.w, r0.y
+ add r0.y, r0.y, -r0.w
+ add r1.w, r0.y, c7.x
+ mad r1.x, r1.w, c8.z, c8.w
+ mul r1.z, r0.x, c18.z
+ mad r1.w, r0.x, r0.z, c7.w
+ mad r0.x, r0.y, c8.z, c8.w
+ mov r0.y, c7.z
+ mov r1.y, c7.z
+ mov r2.xy, c8
+ mov r3.xy, c7.yzxw
+ texld r4, r0, s1
+ texld r5, r1, s1
+ texld r2, r2, s1
+ texld r3, r3, s1
+ texld r6, t0, s0
+ lrp r7, r0.w, r5, r4
+ cmp r0, r1.w, r2, r7
+ cmp r0, -r1.z, r3, r0
+ log r1.x, r0.x
+ log r1.y, r0.y
+ log r1.z, r0.z
+ mul r1.xyz, r1, c9.x
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ mad r0.xyz, r2, c9.y, c9.z
+ mul r0, r6.w, r0
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 62 instruction slots used (5 texture, 57 arithmetic)
+#endif
+
+const DWORD radial28[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020007,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x00000000, 0x3f000000, 0xbf800000, 0x05000051, 0xa00f0008,
+ 0x3f800000, 0x3f000000, 0x3d800000, 0x3d000000, 0x05000051, 0xa00f0009,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80040000, 0xa0000007, 0x02000001, 0x80030000,
+ 0xb0e40001, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000005,
+ 0x80080000, 0x80ff0000, 0x80ff0000, 0x03000005, 0x80080000, 0x80ff0000,
+ 0xa0550012, 0x03000008, 0x80010000, 0x80e40000, 0xa0e40010, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80ff0000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x02000001, 0x80040000, 0xa0aa0012, 0x04000004, 0x80020000,
+ 0x80000000, 0x80aa0000, 0xa1000000, 0x03000005, 0x80120000, 0x80550000,
+ 0xa0550000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000001,
+ 0x03000005, 0x80180000, 0x80ff0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000, 0x80aa0000,
+ 0xa1000002, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550002, 0x03000002,
+ 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000, 0x80000000,
+ 0x80aa0000, 0xa1000003, 0x03000005, 0x80180000, 0x80ff0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004, 0x80080000,
+ 0x80000000, 0x80aa0000, 0xa1000004, 0x03000005, 0x80180000, 0x80ff0000,
+ 0xa0550004, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000, 0x04000004,
+ 0x80080000, 0x80000000, 0x80aa0000, 0xa1000005, 0x03000005, 0x80180000,
+ 0x80ff0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80ff0000,
+ 0x04000004, 0x80080000, 0x80000000, 0x80aa0000, 0xa1000006, 0x03000005,
+ 0x80180000, 0x80ff0000, 0xa0550006, 0x03000002, 0x80020000, 0x80550000,
+ 0x80ff0000, 0x02000013, 0x80080000, 0x80550000, 0x03000002, 0x80020000,
+ 0x80550000, 0x81ff0000, 0x03000002, 0x80080001, 0x80550000, 0xa0000007,
+ 0x04000004, 0x80010001, 0x80ff0001, 0xa0aa0008, 0xa0ff0008, 0x03000005,
+ 0x80040001, 0x80000000, 0xa0aa0012, 0x04000004, 0x80080001, 0x80000000,
+ 0x80aa0000, 0xa0ff0007, 0x04000004, 0x80010000, 0x80550000, 0xa0aa0008,
+ 0xa0ff0008, 0x02000001, 0x80020000, 0xa0aa0007, 0x02000001, 0x80020001,
+ 0xa0aa0007, 0x02000001, 0x80030002, 0xa0e40008, 0x02000001, 0x80030003,
+ 0xa0c90007, 0x03000042, 0x800f0004, 0x80e40000, 0xa0e40801, 0x03000042,
+ 0x800f0005, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0002, 0x80e40002,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0x80e40003, 0xa0e40801, 0x03000042,
+ 0x800f0006, 0xb0e40000, 0xa0e40800, 0x04000012, 0x800f0007, 0x80ff0000,
+ 0x80e40005, 0x80e40004, 0x04000058, 0x800f0000, 0x80ff0001, 0x80e40002,
+ 0x80e40007, 0x04000058, 0x800f0000, 0x81aa0001, 0x80e40003, 0x80e40000,
+ 0x0200000f, 0x80010001, 0x80000000, 0x0200000f, 0x80020001, 0x80550000,
+ 0x0200000f, 0x80040001, 0x80aa0000, 0x03000005, 0x80070001, 0x80e40001,
+ 0xa0000009, 0x0200000e, 0x80010002, 0x80000001, 0x0200000e, 0x80020002,
+ 0x80550001, 0x0200000e, 0x80040002, 0x80aa0001, 0x04000004, 0x80070000,
+ 0x80e40002, 0xa0550009, 0xa0aa0009, 0x03000005, 0x800f0000, 0x80ff0006,
+ 0x80e40000, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial29 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.5, -0.5, 2
+ def c8, 0.0625, 0.03125, 0.416667014, 0
+ def c9, 1.05499995, -0.0549999997, 0, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ mul r0.x, r0.x, c7.y
+ frc r0.x, r0.x
+ add r0.x, r0.x, c7.z
+ abs r0.x, r0.x
+ mad r0.x, r0.x, -c7.w, c7.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c8.x, c8.y
+ mad r2.x, r0.z, c8.x, c8.y
+ mov r2.y, c7.y
+ mov r1.y, c7.y
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c8.z
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c9.x, c9.y
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 59 instruction slots used (3 texture, 56 arithmetic)
+#endif
+
+const DWORD radial29[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020007,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3f000000, 0xbf000000, 0x40000000, 0x05000051, 0xa00f0008,
+ 0x3d800000, 0x3d000000, 0x3ed55561, 0x00000000, 0x05000051, 0xa00f0009,
+ 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x03000005,
+ 0x80010000, 0x80000000, 0xa0550007, 0x02000013, 0x80010000, 0x80000000,
+ 0x03000002, 0x80010000, 0x80000000, 0xa0aa0007, 0x02000023, 0x80010000,
+ 0x80000000, 0x04000004, 0x80010000, 0x80000000, 0xa1ff0007, 0xa0000007,
+ 0x03000002, 0x80020000, 0x80000000, 0xa1000000, 0x03000005, 0x80120000,
+ 0x80550000, 0xa0550000, 0x03000002, 0x80040000, 0x80000000, 0xa1000001,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000002,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000003,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000004,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000002, 0x80040000, 0x80000000, 0xa1000005,
+ 0x03000002, 0x80010000, 0x80000000, 0xa1000006, 0x03000005, 0x80140000,
+ 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000, 0x80550000, 0x80aa0000,
+ 0x03000005, 0x80110000, 0x80000000, 0xa0550006, 0x03000002, 0x80010000,
+ 0x80550000, 0x80000000, 0x02000013, 0x80020000, 0x80000000, 0x03000002,
+ 0x80010000, 0x80000000, 0x81550000, 0x03000002, 0x80040000, 0x80000000,
+ 0xa0000007, 0x04000004, 0x80010001, 0x80000000, 0xa0000008, 0xa0550008,
+ 0x04000004, 0x80010002, 0x80aa0000, 0xa0000008, 0xa0550008, 0x02000001,
+ 0x80020002, 0xa0550007, 0x02000001, 0x80020001, 0xa0550007, 0x03000042,
+ 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042, 0x800f0001, 0x80e40001,
+ 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000, 0xa0e40800, 0x04000012,
+ 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001, 0x0200000f, 0x80010000,
+ 0x80000004, 0x0200000f, 0x80020000, 0x80550004, 0x0200000f, 0x80040000,
+ 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000, 0xa0aa0008, 0x0200000e,
+ 0x80010001, 0x80000000, 0x0200000e, 0x80020001, 0x80550000, 0x0200000e,
+ 0x80040001, 0x80aa0000, 0x04000004, 0x80070004, 0x80e40001, 0xa0000009,
+ 0xa0550009, 0x03000005, 0x800f0000, 0x80ff0003, 0x80e40004, 0x03000005,
+ 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000,
+ 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn radial30 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D colors;
+// float2 fractions[8];
+// float3 m0;
+// float3 m1;
+// sampler2D mask;
+// float3 precalc;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// fractions c0 7
+// m0 c16 1
+// m1 c17 1
+// precalc c18 1
+// mask s0 1
+// colors s1 1
+//
+
+ ps_2_0
+ def c7, 1, 0.0625, 0.03125, 0.5
+ def c8, 0.416667014, 1.05499995, -0.0549999997, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dcl_2d s0
+ dcl_2d s1
+ mov r0.xy, t1
+ mov r0.z, c7.x
+ dp3 r0.w, r0, c17
+ dp3 r0.x, r0, c16
+ mul r0.y, r0.w, r0.w
+ mul r0.y, r0.y, c18.y
+ add r0.x, r0.x, -c18.x
+ mad r0.y, r0.x, r0.x, r0.y
+ rsq r0.y, r0.y
+ rcp r0.y, r0.y
+ mad r0.x, c18.x, r0.x, r0.y
+ mul r0.x, r0.x, c18.z
+ frc r0.x, r0.x
+ add r0.y, r0.x, -c0.x
+ mul_sat r0.y, r0.y, c0.y
+ add r0.z, r0.x, -c1.x
+ mul_sat r0.z, r0.z, c1.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c2.x
+ mul_sat r0.z, r0.z, c2.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c3.x
+ mul_sat r0.z, r0.z, c3.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c4.x
+ mul_sat r0.z, r0.z, c4.y
+ add r0.y, r0.y, r0.z
+ add r0.z, r0.x, -c5.x
+ add r0.x, r0.x, -c6.x
+ mul_sat r0.z, r0.z, c5.y
+ add r0.y, r0.y, r0.z
+ mul_sat r0.x, r0.x, c6.y
+ add r0.x, r0.y, r0.x
+ frc r0.y, r0.x
+ add r0.x, r0.x, -r0.y
+ add r0.z, r0.x, c7.x
+ mad r1.x, r0.x, c7.y, c7.z
+ mad r2.x, r0.z, c7.y, c7.z
+ mov r2.y, c7.w
+ mov r1.y, c7.w
+ texld r2, r2, s1
+ texld r1, r1, s1
+ texld r3, t0, s0
+ lrp r4, r0.y, r2, r1
+ log r0.x, r4.x
+ log r0.y, r4.y
+ log r0.z, r4.z
+ mul r0.xyz, r0, c8.x
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mad r4.xyz, r1, c8.y, c8.z
+ mul r0, r3.w, r4
+ mul r0, r0, v0
+ mov oC0, r0
+
+// approximately 55 instruction slots used (3 texture, 52 arithmetic)
+#endif
+
+const DWORD radial30[] =
+{
+ 0xffff0200, 0x004afffe, 0x42415443, 0x0000001c, 0x000000f3, 0xffff0200,
+ 0x00000006, 0x0000001c, 0x20000100, 0x000000ec, 0x00000094, 0x00010003,
+ 0x00060001, 0x0000009c, 0x00000000, 0x000000ac, 0x00000002, 0x00020007,
+ 0x000000b8, 0x00000000, 0x000000c8, 0x00100002, 0x00420001, 0x000000cc,
+ 0x00000000, 0x000000dc, 0x00110002, 0x00460001, 0x000000cc, 0x00000000,
+ 0x000000df, 0x00000003, 0x00020001, 0x0000009c, 0x00000000, 0x000000e4,
+ 0x00120002, 0x004a0001, 0x000000cc, 0x00000000, 0x6f6c6f63, 0xab007372,
+ 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x63617266, 0x6e6f6974,
+ 0xabab0073, 0x00030001, 0x00020001, 0x00000008, 0x00000000, 0xab00306d,
+ 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x6d00316d, 0x006b7361,
+ 0x63657270, 0x00636c61, 0x325f7370, 0x4d00305f, 0x6f726369, 0x74666f73,
+ 0x29522820, 0x534c4820, 0x6853204c, 0x72656461, 0x6d6f4320, 0x656c6970,
+ 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131, 0x05000051, 0xa00f0007,
+ 0x3f800000, 0x3d800000, 0x3d000000, 0x3f000000, 0x05000051, 0xa00f0008,
+ 0x3ed55561, 0x3f870a3d, 0xbd6147ae, 0x00000000, 0x0200001f, 0x80000000,
+ 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001, 0x0200001f, 0x80000000,
+ 0x900f0000, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f, 0x90000000,
+ 0xa00f0801, 0x02000001, 0x80030000, 0xb0e40001, 0x02000001, 0x80040000,
+ 0xa0000007, 0x03000008, 0x80080000, 0x80e40000, 0xa0e40011, 0x03000008,
+ 0x80010000, 0x80e40000, 0xa0e40010, 0x03000005, 0x80020000, 0x80ff0000,
+ 0x80ff0000, 0x03000005, 0x80020000, 0x80550000, 0xa0550012, 0x03000002,
+ 0x80010000, 0x80000000, 0xa1000012, 0x04000004, 0x80020000, 0x80000000,
+ 0x80000000, 0x80550000, 0x02000007, 0x80020000, 0x80550000, 0x02000006,
+ 0x80020000, 0x80550000, 0x04000004, 0x80010000, 0xa0000012, 0x80000000,
+ 0x80550000, 0x03000005, 0x80010000, 0x80000000, 0xa0aa0012, 0x02000013,
+ 0x80010000, 0x80000000, 0x03000002, 0x80020000, 0x80000000, 0xa1000000,
+ 0x03000005, 0x80120000, 0x80550000, 0xa0550000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000001, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550001,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000002, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550002,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000003, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550003,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000004, 0x03000005, 0x80140000, 0x80aa0000, 0xa0550004,
+ 0x03000002, 0x80020000, 0x80550000, 0x80aa0000, 0x03000002, 0x80040000,
+ 0x80000000, 0xa1000005, 0x03000002, 0x80010000, 0x80000000, 0xa1000006,
+ 0x03000005, 0x80140000, 0x80aa0000, 0xa0550005, 0x03000002, 0x80020000,
+ 0x80550000, 0x80aa0000, 0x03000005, 0x80110000, 0x80000000, 0xa0550006,
+ 0x03000002, 0x80010000, 0x80550000, 0x80000000, 0x02000013, 0x80020000,
+ 0x80000000, 0x03000002, 0x80010000, 0x80000000, 0x81550000, 0x03000002,
+ 0x80040000, 0x80000000, 0xa0000007, 0x04000004, 0x80010001, 0x80000000,
+ 0xa0550007, 0xa0aa0007, 0x04000004, 0x80010002, 0x80aa0000, 0xa0550007,
+ 0xa0aa0007, 0x02000001, 0x80020002, 0xa0ff0007, 0x02000001, 0x80020001,
+ 0xa0ff0007, 0x03000042, 0x800f0002, 0x80e40002, 0xa0e40801, 0x03000042,
+ 0x800f0001, 0x80e40001, 0xa0e40801, 0x03000042, 0x800f0003, 0xb0e40000,
+ 0xa0e40800, 0x04000012, 0x800f0004, 0x80550000, 0x80e40002, 0x80e40001,
+ 0x0200000f, 0x80010000, 0x80000004, 0x0200000f, 0x80020000, 0x80550004,
+ 0x0200000f, 0x80040000, 0x80aa0004, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0000008, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x04000004, 0x80070004,
+ 0x80e40001, 0xa0550008, 0xa0aa0008, 0x03000005, 0x800f0000, 0x80ff0003,
+ 0x80e40004, 0x03000005, 0x800f0000, 0x80e40000, 0x90e40000, 0x02000001,
+ 0x800f0800, 0x80e40000, 0x0000ffff
+};
+const DWORD *radialShaders[] =
+{
+ radial0,
+ radial1,
+ radial2,
+ NULL,
+ radial4,
+ radial5,
+ radial6,
+ NULL,
+ radial8,
+ radial9,
+ radial10,
+ NULL,
+ radial12,
+ radial13,
+ radial14,
+ NULL,
+ radial16,
+ radial17,
+ radial18,
+ NULL,
+ radial20,
+ radial21,
+ radial22,
+ NULL,
+ radial24,
+ radial25,
+ radial26,
+ NULL,
+ radial28,
+ radial29,
+ radial30,
+ NULL,
+};
+
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_0 /Vn lcdtext0 /Fh tmp.h tmp.hlsl
+//
+//
+// Parameters:
+//
+// sampler2D dstTex;
+// float3 gamma;
+// sampler2D glyphTex;
+// float3 invgamma;
+// float3 srcAdj;
+//
+//
+// Registers:
+//
+// Name Reg Size
+// ------------ ----- ----
+// srcAdj c0 1
+// invgamma c1 1
+// gamma c2 1
+// glyphTex s0 1
+// dstTex s1 1
+//
+
+ ps_2_0
+ def c3, -1, 0, 1, 0
+ dcl t0.xy
+ dcl t1.xy
+ dcl_2d s0
+ dcl_2d s1
+ texld r0, t0, s0
+ dp3 r0.w, r0, r0
+ cmp r1, -r0.w, c3.x, c3.y
+ texkill r1
+ texld r1, t1, s1
+ log r2.x, r1.x
+ log r2.y, r1.y
+ log r2.z, r1.z
+ mul r1.xyz, r2, c1
+ exp r2.x, r1.x
+ exp r2.y, r1.y
+ exp r2.z, r1.z
+ lrp r1.xyz, r0, c0, r2
+ log r0.x, r1.x
+ log r0.y, r1.y
+ log r0.z, r1.z
+ mul r0.xyz, r0, c2
+ exp r1.x, r0.x
+ exp r1.y, r0.y
+ exp r1.z, r0.z
+ mov r1.w, c3.z
+ mov oC0, r1
+
+// approximately 22 instruction slots used (2 texture, 20 arithmetic)
+#endif
+
+const DWORD lcdtext0[] =
+{
+ 0xffff0200, 0x0042fffe, 0x42415443, 0x0000001c, 0x000000d0, 0xffff0200,
+ 0x00000005, 0x0000001c, 0x20000100, 0x000000c9, 0x00000080, 0x00010003,
+ 0x00060001, 0x00000088, 0x00000000, 0x00000098, 0x00020002, 0x000a0001,
+ 0x000000a0, 0x00000000, 0x000000b0, 0x00000003, 0x00020001, 0x00000088,
+ 0x00000000, 0x000000b9, 0x00010002, 0x00060001, 0x000000a0, 0x00000000,
+ 0x000000c2, 0x00000002, 0x00020001, 0x000000a0, 0x00000000, 0x54747364,
+ 0xab007865, 0x000c0004, 0x00010001, 0x00000001, 0x00000000, 0x6d6d6167,
+ 0xabab0061, 0x00030001, 0x00030001, 0x00000001, 0x00000000, 0x70796c67,
+ 0x78655468, 0x766e6900, 0x6d6d6167, 0x72730061, 0x6a644163, 0x5f737000,
+ 0x00305f32, 0x7263694d, 0x666f736f, 0x52282074, 0x4c482029, 0x53204c53,
+ 0x65646168, 0x6f432072, 0x6c69706d, 0x39207265, 0x2e39312e, 0x2e393439,
+ 0x31313132, 0xababab00, 0x05000051, 0xa00f0003, 0xbf800000, 0x80000000,
+ 0x3f800000, 0x00000000, 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f,
+ 0x80000000, 0xb0030001, 0x0200001f, 0x90000000, 0xa00f0800, 0x0200001f,
+ 0x90000000, 0xa00f0801, 0x03000042, 0x800f0000, 0xb0e40000, 0xa0e40800,
+ 0x03000008, 0x80080000, 0x80e40000, 0x80e40000, 0x04000058, 0x800f0001,
+ 0x81ff0000, 0xa0000003, 0xa0550003, 0x01000041, 0x800f0001, 0x03000042,
+ 0x800f0001, 0xb0e40001, 0xa0e40801, 0x0200000f, 0x80010002, 0x80000001,
+ 0x0200000f, 0x80020002, 0x80550001, 0x0200000f, 0x80040002, 0x80aa0001,
+ 0x03000005, 0x80070001, 0x80e40002, 0xa0e40001, 0x0200000e, 0x80010002,
+ 0x80000001, 0x0200000e, 0x80020002, 0x80550001, 0x0200000e, 0x80040002,
+ 0x80aa0001, 0x04000012, 0x80070001, 0x80e40000, 0xa0e40000, 0x80e40002,
+ 0x0200000f, 0x80010000, 0x80000001, 0x0200000f, 0x80020000, 0x80550001,
+ 0x0200000f, 0x80040000, 0x80aa0001, 0x03000005, 0x80070000, 0x80e40000,
+ 0xa0e40002, 0x0200000e, 0x80010001, 0x80000000, 0x0200000e, 0x80020001,
+ 0x80550000, 0x0200000e, 0x80040001, 0x80aa0000, 0x02000001, 0x80080001,
+ 0xa0aa0003, 0x02000001, 0x800f0800, 0x80e40001, 0x0000ffff
+};
+#if 0
+//
+// Generated by Microsoft (R) HLSL Shader Compiler 9.19.949.2111
+//
+// fxc /T ps_2_a /Vn aapgram0 /Fh tmp.h tmp.hlsl
+//
+ ps_2_x
+ dcl t0.xy
+ dcl t1.xy
+ dcl v0
+ dsx r0.xy, t1.yxzw
+ add r0.xy, r0, t1.yxzw
+ min r1.xy, r0, t1.yxzw
+ dsy r0.zw, t1.xyyx
+ add r1.zw, r0, t1.xyyx
+ add r0.zw, r0.xyxy, r0
+ max r2.xy, t1.yxzw, r0
+ min r0.xy, r1.zwzw, r1
+ max r3.xy, r2, r1.zwzw
+ min r1.xy, r0.zwzw, r0
+ max r1.zw, r3.xyxy, r0
+ mov_sat r0.xy, r1
+ add r0.zw, -r1.xyxy, r1
+ mov_sat r1.zw, r1
+ add r0.xy, -r0, r1.zwzw
+ mul r0.xy, r0.ywzw, r0.xzzw
+ rcp r0.y, r0.y
+ mul r0.x, r0.x, r0.y
+ dsx r0.yz, t0.xyxw
+ add r0.yz, r0, t0.xyxw
+ min r1.xy, r0.yzzw, t0.yxzw
+ dsy r1.zw, t0.xyyx
+ add r2.xy, r1.zwzw, t0.yxzw
+ add r1.zw, r0.xyyz, r1
+ max r2.zw, t0.xyyx, r0.xyyz
+ min r0.yz, r2.xxyw, r1.xxyw
+ max r1.xy, r2.zwzw, r2
+ min r2.xy, r1.zwzw, r0.yzzw
+ max r0.yz, r1.xxyw, r1.xzww
+ mov_sat r1.xy, r2
+ add r1.zw, -r2.xyxy, r0.xyyz
+ mov_sat r0.yz, r0
+ add r0.yz, -r1.xxyw, r0
+ mul r0.y, r0.z, r0.y
+ mul r0.z, r1.w, r1.z
+ rcp r0.z, r0.z
+ mad r0.x, r0.y, r0.z, -r0.x
+ mul r0, r0.x, v0
+ mov oC0, r0
+
+// approximately 43 instruction slots used
+#endif
+
+const DWORD aapgram0[] =
+{
+ 0xffff0201, 0x0016fffe, 0x42415443, 0x0000001c, 0x00000023, 0xffff0201,
+ 0x00000000, 0x00000000, 0x20000100, 0x0000001c, 0x325f7370, 0x4d00615f,
+ 0x6f726369, 0x74666f73, 0x29522820, 0x534c4820, 0x6853204c, 0x72656461,
+ 0x6d6f4320, 0x656c6970, 0x2e392072, 0x392e3931, 0x322e3934, 0x00313131,
+ 0x0200001f, 0x80000000, 0xb0030000, 0x0200001f, 0x80000000, 0xb0030001,
+ 0x0200001f, 0x80000000, 0x900f0000, 0x0200005b, 0x80030000, 0xb0e10001,
+ 0x03000002, 0x80030000, 0x80e40000, 0xb0e10001, 0x0300000a, 0x80030001,
+ 0x80e40000, 0xb0e10001, 0x0200005c, 0x800c0000, 0xb0140001, 0x03000002,
+ 0x800c0001, 0x80e40000, 0xb0140001, 0x03000002, 0x800c0000, 0x80440000,
+ 0x80e40000, 0x0300000b, 0x80030002, 0xb0e10001, 0x80e40000, 0x0300000a,
+ 0x80030000, 0x80ee0001, 0x80e40001, 0x0300000b, 0x80030003, 0x80e40002,
+ 0x80ee0001, 0x0300000a, 0x80030001, 0x80ee0000, 0x80e40000, 0x0300000b,
+ 0x800c0001, 0x80440003, 0x80e40000, 0x02000001, 0x80130000, 0x80e40001,
+ 0x03000002, 0x800c0000, 0x81440001, 0x80e40001, 0x02000001, 0x801c0001,
+ 0x80e40001, 0x03000002, 0x80030000, 0x81e40000, 0x80ee0001, 0x03000005,
+ 0x80030000, 0x80ed0000, 0x80e80000, 0x02000006, 0x80020000, 0x80550000,
+ 0x03000005, 0x80010000, 0x80000000, 0x80550000, 0x0200005b, 0x80060000,
+ 0xb0c40000, 0x03000002, 0x80060000, 0x80e40000, 0xb0c40000, 0x0300000a,
+ 0x80030001, 0x80e90000, 0xb0e10000, 0x0200005c, 0x800c0001, 0xb0140000,
+ 0x03000002, 0x80030002, 0x80ee0001, 0xb0e10000, 0x03000002, 0x800c0001,
+ 0x80940000, 0x80e40001, 0x0300000b, 0x800c0002, 0xb0140000, 0x80940000,
+ 0x0300000a, 0x80060000, 0x80d00002, 0x80d00001, 0x0300000b, 0x80030001,
+ 0x80ee0002, 0x80e40002, 0x0300000a, 0x80030002, 0x80ee0001, 0x80e90000,
+ 0x0300000b, 0x80060000, 0x80d00001, 0x80f80001, 0x02000001, 0x80130001,
+ 0x80e40002, 0x03000002, 0x800c0001, 0x81440002, 0x80940000, 0x02000001,
+ 0x80160000, 0x80e40000, 0x03000002, 0x80060000, 0x81d00001, 0x80e40000,
+ 0x03000005, 0x80020000, 0x80aa0000, 0x80550000, 0x03000005, 0x80040000,
+ 0x80ff0001, 0x80aa0001, 0x02000006, 0x80040000, 0x80aa0000, 0x04000004,
+ 0x80010000, 0x80550000, 0x80aa0000, 0x81000000, 0x03000005, 0x800f0000,
+ 0x80000000, 0x90e40000, 0x02000001, 0x800f0800, 0x80e40000, 0x0000ffff
+};
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.cpp
index 5920f8f66f1..0cadb3bd677 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.cpp
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -23,291 +23,611 @@
* have any questions.
*/
+#include
+#include
+#include
#include "D3DSurfaceData.h"
-#include "D3DContext.h"
-#include "jlong.h"
-#include "jni_util.h"
+#include "D3DPipelineManager.h"
#include "Trace.h"
-#include "ddrawUtils.h"
-#include "Devices.h"
+#include "awt_Toolkit.h"
+#include "awt_Window.h"
+#include "awt_BitmapUtil.h"
+#include "D3DRenderQueue.h"
-#include "Win32SurfaceData.h"
-#include "sun_java2d_d3d_D3DBackBufferSurfaceData.h"
+// REMIND: move to awt_Component.h
+extern "C" HWND AwtComponent_GetHWnd(JNIEnv *env, jlong pData);
-extern LockFunc Win32OSSD_Lock;
-extern GetRasInfoFunc Win32OSSD_GetRasInfo;
-extern UnlockFunc Win32OSSD_Unlock;
-extern DisposeFunc Win32OSSD_Dispose;
-extern GetDCFunc Win32OSSD_GetDC;
-extern ReleaseDCFunc Win32OSSD_ReleaseDC;
-extern InvalidateSDFunc Win32OSSD_InvalidateSD;
-extern RestoreSurfaceFunc Win32OSSD_RestoreSurface;
-extern DisposeFunc Win32BBSD_Dispose;
-
-extern "C" {
-
-RestoreSurfaceFunc D3DSD_RestoreSurface;
-
-/*
- * D3D-surface specific restore function.
- * We need to make sure the D3DContext is notified if the
- * surface is lost (only if this surface is the current target,
- * otherwise it's possible that it'll get restored (along with its
- * depth buffer), and the context will still think that the clipping
- * that's set for this surface is valid.
- * Consider this scenario:
- * do {
- * vi.validate(gc); // validated, vi's surface is restored, clipping is lost
- * // render stuff using d3d, clipping is reset
- * // -> surface loss event happens
- * // do a DD blit of the VI to the screen
- * // at this point the VI surface will be marked lost
- * // and will be restored in validate() next time around,
- * // losing the clipping w/o notifying the D3D context
- * } while (vi.surfaceLost());
+/**
+ * Initializes nativeWidth/Height fields of the SurfaceData object with
+ * dimensions on the native surface.
*/
-void D3DSD_RestoreSurface(JNIEnv *env, Win32SDOps *wsdo) {
- J2dTraceLn(J2D_TRACE_INFO, "D3DSD_RestoreSurface");
- D3DSDOps *d3dsdo = (D3DSDOps *)wsdo;
- // This is needed only for non-textures, since textures can't
- // lose their surfaces, as they're managed.
- if (!(d3dsdo->d3dType & D3D_TEXTURE_SURFACE) && wsdo->lpSurface != NULL)
- {
- if (wsdo->ddInstance != NULL && wsdo->ddInstance->ddObject != NULL) {
- D3DContext *d3dContext =
- wsdo->ddInstance->ddObject->GetD3dContext();
- if (d3dContext != NULL) {
- d3dContext->InvalidateIfTarget(env, wsdo->lpSurface);
+void D3DSD_SetNativeDimensions(JNIEnv *env, D3DSDOps *d3dsdo) {
+ jobject sdObject;
+ jint width, height;
+
+ RETURN_IF_NULL(sdObject = env->NewLocalRef(d3dsdo->sdOps.sdObject));
+
+ if (d3dsdo->pResource != NULL) {
+ width = d3dsdo->pResource->GetDesc()->Width;
+ height = d3dsdo->pResource->GetDesc()->Height;
+ } else {
+ width = d3dsdo->width;
+ height = d3dsdo->height;
+ }
+
+ JNU_SetFieldByName(env, NULL, sdObject, "nativeWidth", "I", width);
+ JNU_SetFieldByName(env, NULL, sdObject, "nativeHeight", "I", height);
+
+ env->DeleteLocalRef(sdObject);
+}
+
+void D3DSD_Flush(void *pData)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSD_Flush");
+ RETURN_IF_NULL(pData);
+
+ D3DSDOps *d3dsdo = (D3DSDOps*)pData;
+ if (d3dsdo->pResource != NULL) {
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+
+ d3dsdo->pResource->SetSDOps(NULL);
+
+ if ((pMgr = D3DPipelineManager::GetInstance()) != NULL &&
+ SUCCEEDED(pMgr->GetD3DContext(d3dsdo->adapter, &pCtx)))
+ {
+ if (pCtx->GetResourceManager()) {
+ pCtx->GetResourceManager()->ReleaseResource(d3dsdo->pResource);
}
}
+ d3dsdo->pResource = NULL;
}
- Win32OSSD_RestoreSurface(env, wsdo);
}
+void
+D3DSD_MarkLost(void *pData)
+{
+ D3DSDOps *d3dsdo;
+ jobject sdObject;
+ JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSD_MarkLost");
+
+ RETURN_IF_NULL(pData);
+
+ d3dsdo = (D3DSDOps*)pData;
+ RETURN_IF_NULL(sdObject = env->NewLocalRef(d3dsdo->sdOps.sdObject));
+
+ JNU_CallMethodByName(env, NULL, sdObject,
+ "setSurfaceLost", "(Z)V", JNI_TRUE);
+
+ env->DeleteLocalRef(sdObject);
+}
+
+// ------------ generic SurfaceData.h functions ----------------
+
+void
+D3DSD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
+{
+ D3DSDOps *d3dsdo = (D3DSDOps *)ops;
+ RETURN_IF_NULL(d3dsdo);
+
+ JNU_CallStaticMethodByName(env, NULL, "sun/java2d/d3d/D3DSurfaceData",
+ "dispose", "(J)V",
+ ptr_to_jlong(ops));
+}
+
+/**
+ * This is the implementation of the general surface LockFunc defined in
+ * SurfaceData.h.
+ */
+jint
+D3DSD_Lock(JNIEnv *env,
+ SurfaceDataOps *ops,
+ SurfaceDataRasInfo *pRasInfo,
+ jint lockflags)
+{
+ JNU_ThrowInternalError(env, "D3DSD_Lock not implemented!");
+ return SD_FAILURE;
+}
+
+/**
+ * This is the implementation of the general GetRasInfoFunc defined in
+ * SurfaceData.h.
+ */
+void
+D3DSD_GetRasInfo(JNIEnv *env,
+ SurfaceDataOps *ops,
+ SurfaceDataRasInfo *pRasInfo)
+{
+ JNU_ThrowInternalError(env, "D3DSD_GetRasInfo not implemented!");
+}
+
+/**
+ * This is the implementation of the general surface UnlockFunc defined in
+ * SurfaceData.h.
+ */
+void
+D3DSD_Unlock(JNIEnv *env,
+ SurfaceDataOps *ops,
+ SurfaceDataRasInfo *pRasInfo)
+{
+ JNU_ThrowInternalError(env, "D3DSD_Unlock not implemented!");
+}
+
+// ------------ D3DSurfaceData's JNI methods ----------------
+
+
+extern "C" {
+
/*
* Class: sun_java2d_d3d_D3DSurfaceData
* Method: initOps
- * Signature: (Ljava/lang/Object;)V
+ * Signature: (III)V
*/
-JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DSurfaceData_initOps(JNIEnv *env,
- jobject wsd,
- jint depth,
- jint transparency)
+JNIEXPORT void
+JNICALL Java_sun_java2d_d3d_D3DSurfaceData_initOps
+ (JNIEnv *env, jobject d3dsd, jint gdiScreen, jint width, jint height)
{
+ D3DPipelineManager *pMgr;
+ D3DSDOps *d3dsdo = (D3DSDOps *)SurfaceData_InitOps(env, d3dsd,
+ sizeof(D3DSDOps));
+
J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_initOps");
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_InitOps(env, wsd,
- sizeof(D3DSDOps));
- wsdo->sdOps.Lock = Win32OSSD_Lock;
- wsdo->sdOps.GetRasInfo = Win32OSSD_GetRasInfo;
- wsdo->sdOps.Unlock = Win32OSSD_Unlock;
- wsdo->sdOps.Dispose = Win32OSSD_Dispose;
- wsdo->RestoreSurface = D3DSD_RestoreSurface;
- wsdo->GetDC = Win32OSSD_GetDC;
- wsdo->ReleaseDC = Win32OSSD_ReleaseDC;
- wsdo->InvalidateSD = Win32OSSD_InvalidateSD;
- wsdo->invalid = JNI_FALSE;
- wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
- wsdo->window = NULL;
- wsdo->backBufferCount = 0;
- wsdo->depth = depth;
- switch (depth) {
- case 8:
- wsdo->pixelStride = 1;
- break;
- case 15: //555
- wsdo->pixelStride = 2;
- wsdo->pixelMasks[0] = 0x1f << 10;
- wsdo->pixelMasks[1] = 0x1f << 5;
- wsdo->pixelMasks[2] = 0x1f;
- break;
- case 16: //565
- wsdo->pixelStride = 2;
- wsdo->pixelMasks[0] = 0x1f << 11;
- wsdo->pixelMasks[1] = 0x3f << 5;
- wsdo->pixelMasks[2] = 0x1f;
- break;
- case 24:
- wsdo->pixelStride = 3;
- break;
- case 32: //x888
- wsdo->pixelStride = 4;
- wsdo->pixelMasks[0] = 0xff0000;
- wsdo->pixelMasks[1] = 0x00ff00;
- wsdo->pixelMasks[2] = 0x0000ff;
- break;
+
+ if (d3dsdo == NULL) {
+ JNU_ThrowOutOfMemoryError(env, "creating native d3d ops");
+ return;
}
- wsdo->surfaceLock = new CriticalSection();
- wsdo->surfaceLost = FALSE;
- wsdo->transparency = transparency;
- wsdo->surfacePuntData.usingDDSystem = FALSE;
- wsdo->surfacePuntData.lpSurfaceSystem = NULL;
- wsdo->surfacePuntData.lpSurfaceVram = NULL;
- wsdo->surfacePuntData.numBltsSinceRead = 0;
- wsdo->surfacePuntData.pixelsReadSinceBlt = 0;
- wsdo->surfacePuntData.numBltsThreshold = 2;
- wsdo->gdiOpPending = FALSE;
+
+ d3dsdo->sdOps.Lock = D3DSD_Lock;
+ d3dsdo->sdOps.GetRasInfo = D3DSD_GetRasInfo;
+ d3dsdo->sdOps.Unlock = D3DSD_Unlock;
+ d3dsdo->sdOps.Dispose = D3DSD_Dispose;
+
+ d3dsdo->xoff = 0;
+ d3dsdo->yoff = 0;
+ d3dsdo->width = width;
+ d3dsdo->height = height;
+
+ d3dsdo->pResource = NULL;
+
+ d3dsdo->adapter =
+ (pMgr = D3DPipelineManager::GetInstance()) == NULL ?
+ D3DADAPTER_DEFAULT :
+ pMgr->GetAdapterOrdinalForScreen(gdiScreen);
}
-jboolean init_D3DSDO(JNIEnv* env, Win32SDOps* wsdo, jint width, jint height,
- jint d3dSurfaceType, jint screen)
-{
- // default in case of an error
- wsdo->lpSurface = NULL;
- wsdo->ddInstance = NULL;
- {
- Devices::InstanceAccess devices;
- wsdo->device = devices->GetDeviceReference(screen, FALSE);
- }
- if (wsdo->device == NULL) {
- J2dTraceLn1(J2D_TRACE_WARNING,
- "init_D3DSDO: Incorrect "\
- "screen number (screen=%d)", screen);
- wsdo->invalid = TRUE;
+/*
+ * Class: sun_java2d_d3d_D3DSurfaceData
+ * Method: initTexture
+ * Signature: (JZZ)Z
+ */
+JNIEXPORT jboolean
+JNICALL Java_sun_java2d_d3d_D3DSurfaceData_initTexture
+ (JNIEnv *env, jobject d3dsd,
+ jlong pData, jboolean isRTT, jboolean isOpaque)
+{
+ HRESULT res;
+ D3DSDOps *d3dsdo;
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+ D3DFORMAT format;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_initTexture");
+
+ RETURN_STATUS_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pData), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
return JNI_FALSE;
}
- wsdo->w = width;
- wsdo->h = height;
- wsdo->surfacePuntData.disablePunts = TRUE;
- return JNI_TRUE;
+ RETURN_STATUS_IF_NULL(pCtx->GetResourceManager(), JNI_FALSE);
+
+ pCtx->GetResourceManager()->ReleaseResource(d3dsdo->pResource);
+ d3dsdo->pResource = NULL;
+
+ if (isRTT && isOpaque) {
+ format = pCtx->GetPresentationParams()->BackBufferFormat;
+ } else {
+ format = D3DFMT_UNKNOWN;
+ }
+
+ res = pCtx->GetResourceManager()->
+ CreateTexture(d3dsdo->width, d3dsdo->height,
+ isRTT, isOpaque,
+ &format, 0/*usage*/, &d3dsdo->pResource);
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE,
+ " created texture pResource=%x", d3dsdo->pResource);
+ d3dsdo->pResource->SetSDOps(d3dsdo);
+ } else {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ }
+ D3DSD_SetNativeDimensions(env, d3dsdo);
+
+ return SUCCEEDED(res);
}
/*
* Class: sun_java2d_d3d_D3DSurfaceData
- * Method: initOffScreenSurface
- * Signature: (JJJIIII)I
+ * Method: initPlain
+ * Signature: (JZ)Z
*/
-JNIEXPORT jint JNICALL
-Java_sun_java2d_d3d_D3DSurfaceData_initOffScreenSurface
- (JNIEnv *env, jobject sData,
- jlong pCtx,
- jlong pData, jlong parentPdata,
- jint width, jint height,
- jint d3dSurfaceType, jint screen)
+JNIEXPORT jboolean JNICALL
+Java_sun_java2d_d3d_D3DSurfaceData_initRTSurface
+ (JNIEnv *env, jobject d3dsd, jlong pData, jboolean isOpaque)
{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *pd3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_initOffScreenSurface");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " width=%-4d height=%-4d type=%-3d scr=%-3d",
- width, height, d3dSurfaceType, screen);
-
- // REMIND: ideally this should be done in initOps
- if (d3dSurfaceType == D3D_ATTACHED_SURFACE) {
- wsdo->sdOps.Dispose = Win32BBSD_Dispose;
- }
-
- if (init_D3DSDO(env, wsdo, width, height,
- d3dSurfaceType, screen) == JNI_FALSE)
- {
- SurfaceData_ThrowInvalidPipeException(env,
- "Can't create offscreen surface");
- return PF_INVALID;
- }
-
- HMONITOR hMon = (HMONITOR)wsdo->device->GetMonitor();
- DDrawObjectStruct *ddInstance = GetDDInstanceForDevice(hMon);
- if (!ddInstance || !ddInstance->valid || !pd3dc) {
- return PF_INVALID;
- }
-
- if (d3dSurfaceType == D3D_ATTACHED_SURFACE) {
- // REMIND: still using the old path. ideally the creation of attached
- // surface shoudld be done in the same way as other types of surfaces,
- // that is, in D3DContext::CreateSurface, but we really don't use
- // anything from D3DContext to get an attached surface, so this
- // was left here.
-
- Win32SDOps *wsdo_parent = (Win32SDOps *)jlong_to_ptr(parentPdata);
- // we're being explicit here: requesting backbuffer, and render target
- DDrawSurface* pNew = wsdo_parent->lpSurface == NULL ?
- NULL :
- wsdo_parent->lpSurface->
- GetDDAttachedSurface(DDSCAPS_BACKBUFFER|DDSCAPS_3DDEVICE);
- if (pNew == NULL ||
- FAILED(pd3dc->AttachDepthBuffer(pNew->GetDXSurface())))
- {
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "D3DSD_initSurface: GetAttachedSurface for parent"\
- " wsdo_parent->lpSurface=0x%x failed",
- wsdo_parent->lpSurface);
- if (pNew != NULL) {
- delete pNew;
- }
- SurfaceData_ThrowInvalidPipeException(env,
- "Can't create attached offscreen surface");
- return PF_INVALID;
- }
-
- wsdo->lpSurface = pNew;
- wsdo->ddInstance = ddInstance;
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- "D3DSD_initSurface: created attached surface: "\
- "wsdo->lpSurface=0x%x for parent "\
- "wsdo_parent->lpSurface=0x%x",
- wsdo->lpSurface, wsdo_parent->lpSurface);
- // we don't care about pixel format for non-texture surfaces
- return PF_INVALID;
- }
-
- DXSurface *dxSurface = NULL;
- jint pf = PF_INVALID;
HRESULT res;
- if (SUCCEEDED(res = pd3dc->CreateSurface(env, wsdo->w, wsdo->h,
- wsdo->depth, wsdo->transparency,
- d3dSurfaceType,
- &dxSurface, &pf)))
- {
- // REMIND: put all the error-handling stuff here from
- // DDCreateOffScreenSurface
- wsdo->lpSurface = new DDrawSurface(ddInstance->ddObject, dxSurface);
- wsdo->surfacePuntData.lpSurfaceVram = wsdo->lpSurface;
- wsdo->ddInstance = ddInstance;
- // the dimensions of the surface may be adjusted in case of
- // textures
- wsdo->w = dxSurface->GetWidth();
- wsdo->h = dxSurface->GetHeight();
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "D3DSurfaceData_initSurface: created surface: "\
- "wsdo->lpSurface=0x%x", wsdo->lpSurface);
- } else {
- DebugPrintDirectDrawError(res,
- "D3DSurfaceData_initSurface: "\
- "CreateSurface failed");
- // REMIND: should use some other way to signal that
- // surface creation was unsuccessful
- SurfaceData_ThrowInvalidPipeException(env,
- "Can't create offscreen surf");
+ D3DSDOps *d3dsdo;
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+ D3DFORMAT format = D3DFMT_UNKNOWN;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_initRTSurface");
+
+ RETURN_STATUS_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pData), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return JNI_FALSE;
}
- return pf;
+ RETURN_STATUS_IF_NULL(pCtx->GetResourceManager(), JNI_FALSE);
+
+ pCtx->GetResourceManager()->ReleaseResource(d3dsdo->pResource);
+ d3dsdo->pResource = NULL;
+
+ res = pCtx->GetResourceManager()->
+ CreateRTSurface(d3dsdo->width, d3dsdo->height,
+ isOpaque, FALSE /*lockable*/,
+ &format, &d3dsdo->pResource);
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " created RT surface pResource=0x%x",
+ d3dsdo->pResource);
+ d3dsdo->pResource->SetSDOps(d3dsdo);
+ } else {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ }
+ D3DSD_SetNativeDimensions(env, d3dsdo);
+
+ return SUCCEEDED(res);
}
/*
- * Class: sun_java2d_d3d_D3DBackBufferSurfaceData
- * Method: restoreDepthBuffer
- * Signature: ()V
+ * Class: sun_java2d_d3d_D3DSurfaceData
+ * Method: initFlipBackbuffer
+ * Signature: (JJIZ)Z
*/
-JNIEXPORT void JNICALL
-Java_sun_java2d_d3d_D3DBackBufferSurfaceData_restoreDepthBuffer(JNIEnv *env,
- jobject sData)
+JNIEXPORT jboolean
+JNICALL Java_sun_java2d_d3d_D3DSurfaceData_initFlipBackbuffer
+ (JNIEnv *env, jobject d3dsd, jlong pData, jlong pPeerData,
+ jint numBuffers, jint swapEffect,
+ jint vSyncType)
{
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, sData);
- J2dTraceLn1(J2D_TRACE_INFO,
- "D3DBBSD_restoreDepthBuffer: wsdo=0x%x", wsdo);
+ HRESULT res;
+ D3DSDOps *d3dsdo;
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+ HWND hWnd;
+ UINT presentationInterval;
+ AwtComponent *pPeer;
+ RECT r = { 0, 0, 0, 0 };
- if (wsdo != NULL) {
- if (!DDRestoreSurface(wsdo)) {
- // Failure - throw exception
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3DBBSD_restoreDepthBuffer: failed to "\
- "restore depth buffer");
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_initFlipBackbuffer");
- SurfaceData_ThrowInvalidPipeException(env,
- "RestoreDepthBuffer failure");
+ RETURN_STATUS_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pData), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(pPeer = (AwtComponent *)jlong_to_ptr(pPeerData),
+ JNI_FALSE);
+
+ hWnd = pPeer->GetHWnd();
+ if (!IsWindow(hWnd)) {
+ J2dTraceLn(J2D_TRACE_WARNING,
+ "D3DSurfaceData_initFlipBackbuffer: disposed component");
+ return JNI_FALSE;
+ }
+
+ pPeer->GetInsets(&r);
+ d3dsdo->xoff = -r.left;
+ d3dsdo->yoff = -r.top;
+
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return JNI_FALSE;
+ }
+ RETURN_STATUS_IF_NULL(pCtx->GetResourceManager(), JNI_FALSE);
+
+ pCtx->GetResourceManager()->ReleaseResource(d3dsdo->pResource);
+ d3dsdo->pResource = NULL;
+
+ d3dsdo->swapEffect = (D3DSWAPEFFECT)swapEffect;
+
+ // in full-screen mode we should v-sync
+ if (pCtx->GetPresentationParams()->Windowed) {
+ if (vSyncType == VSYNC_ON) {
+ presentationInterval = D3DPRESENT_INTERVAL_ONE;
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " windowed, forced interval: ONE");
+ } else {
+ presentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE;
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " windowed, default interval: IMMEDIATE");
+ }
+
+ // REMIND: this is a workaround for the current issue
+ // we have with non-copy flip chains: since we can not specify
+ // the dest rectangle for Present for these modes, the result of
+ // Present(NULL, NULL) is scaled to the client area.
+ if (d3dsdo->xoff != 0 || d3dsdo->yoff != 0) {
+ d3dsdo->swapEffect = D3DSWAPEFFECT_COPY;
+ }
+ } else {
+ if (vSyncType == VSYNC_OFF) {
+ presentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE;
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " full-screen, forced interval: IMMEDIATE");
+ } else {
+ presentationInterval = D3DPRESENT_INTERVAL_ONE;
+ J2dTraceLn(J2D_TRACE_VERBOSE,
+ " full-screen, default interval: ONE");
}
}
+
+ res = pCtx->GetResourceManager()->
+ CreateSwapChain(hWnd, numBuffers,
+ d3dsdo->width, d3dsdo->height,
+ d3dsdo->swapEffect, presentationInterval,
+ &d3dsdo->pResource);
+ if (SUCCEEDED(res)) {
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " created swap chain pResource=0x%x",
+ d3dsdo->pResource);
+ d3dsdo->pResource->SetSDOps(d3dsdo);
+ } else {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ }
+ D3DSD_SetNativeDimensions(env, d3dsdo);
+
+ return SUCCEEDED(res);
}
+/*
+ * Class: sun_java2d_d3d_D3DSurfaceData
+ * Method: dbGetPixelNative
+ * Signature: (JII)I
+ */
+JNIEXPORT jint JNICALL Java_sun_java2d_d3d_D3DSurfaceData_dbGetPixelNative
+ (JNIEnv *env, jclass clazz, jlong pData, jint x, jint y)
+{
+ HRESULT res;
+ D3DSDOps *d3dsdo;
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+ D3DResource *pLockableRes;
+ jint pixel = 0;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_dbGetPixelNative");
+
+ RETURN_STATUS_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pData), pixel);
+ RETURN_STATUS_IF_NULL(d3dsdo->pResource, pixel);
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), pixel);
+
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return pixel;
+ }
+ RETURN_STATUS_IF_NULL(pCtx->GetResourceManager(), 0);
+
+ IDirect3DDevice9 *pd3dDevice = pCtx->Get3DDevice();
+ IDirect3DSurface9 *pSrc = d3dsdo->pResource->GetSurface();
+ D3DFORMAT srcFmt = d3dsdo->pResource->GetDesc()->Format;
+
+ pCtx->UpdateState(STATE_OTHEROP);
+
+ res = pCtx->GetResourceManager()->
+ GetLockableRTSurface(1, 1, srcFmt, &pLockableRes);
+ if (SUCCEEDED(res)) {
+ IDirect3DSurface9 *pTmpSurface;
+ RECT srcRect = { x, y, x+1, y+1};
+ RECT dstRect = { 0l, 0l, 1, 1 };
+
+ pTmpSurface = pLockableRes->GetSurface();
+ res = pd3dDevice->StretchRect(pSrc, &srcRect, pTmpSurface, &dstRect,
+ D3DTEXF_NONE);
+ if (SUCCEEDED(res)) {
+ D3DLOCKED_RECT lRect;
+
+ res = pTmpSurface->LockRect(&lRect, &dstRect, D3DLOCK_NOSYSLOCK);
+ if (SUCCEEDED(res)) {
+ if (srcFmt == D3DFMT_X8R8G8B8) {
+ pixel = *(jint*)lRect.pBits;
+ } else {
+ pixel = *(unsigned short*)lRect.pBits;
+ }
+ pTmpSurface->UnlockRect();
+ }
+ }
+ }
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+
+ return pixel;
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DSurfaceData
+ * Method: dbSetPixelNative
+ * Signature: (JIII)V
+ */
+JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DSurfaceData_dbSetPixelNative
+ (JNIEnv *env, jclass clazz, jlong pData, jint x, jint y, jint pixel)
+{
+ HRESULT res;
+ D3DSDOps *d3dsdo;
+ D3DResource *pLockableRes;
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_dbSetPixelNative");
+
+ RETURN_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pData));
+ RETURN_IF_NULL(d3dsdo->pResource);
+ RETURN_IF_NULL(pMgr = D3DPipelineManager::GetInstance());
+
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return;
+ }
+ RETURN_IF_NULL(pCtx->GetResourceManager());
+
+ IDirect3DDevice9 *pd3dDevice = pCtx->Get3DDevice();
+ IDirect3DSurface9 *pSrc = d3dsdo->pResource->GetSurface();
+ D3DFORMAT srcFmt = d3dsdo->pResource->GetDesc()->Format;
+
+ pCtx->UpdateState(STATE_OTHEROP);
+
+ res = pCtx->GetResourceManager()->
+ GetLockableRTSurface(1, 1, srcFmt, &pLockableRes);
+ if (SUCCEEDED(res)) {
+ IDirect3DSurface9 *pTmpSurface;
+ D3DLOCKED_RECT lRect;
+ RECT srcRect = { 0l, 0l, 1, 1 };
+ RECT dstRect = { x, y, x+1, y+1};
+
+ pTmpSurface = pLockableRes->GetSurface();
+ res = pTmpSurface->LockRect(&lRect, &srcRect, D3DLOCK_NOSYSLOCK);
+ if (SUCCEEDED(res)) {
+ if (srcFmt == D3DFMT_X8R8G8B8) {
+ *(jint*)lRect.pBits = pixel;
+ } else {
+ *(unsigned short*)lRect.pBits = (unsigned short)pixel;
+ }
+ pTmpSurface->UnlockRect();
+
+ res = pd3dDevice->StretchRect(pTmpSurface, &srcRect, pSrc, &dstRect,
+ D3DTEXF_NONE);
+ }
+ }
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DSurfaceData
+ * Method: getNativeResourceNative
+ * Signature: (JI)J
+ */
+JNIEXPORT jlong JNICALL
+ Java_sun_java2d_d3d_D3DSurfaceData_getNativeResourceNative
+ (JNIEnv *env, jclass d3sdc, jlong pData, jint resType)
+{
+ D3DSDOps *d3dsdo;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DSurfaceData_getNativeResourceNative")
+
+ RETURN_STATUS_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pData), 0L);
+
+ if (resType == D3D_DEVICE_RESOURCE) {
+ HRESULT res;
+ D3DPipelineManager *pMgr;
+ D3DContext *pCtx;
+
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), 0L);
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return 0L;
+ }
+ return ptr_to_jlong(pCtx->Get3DDevice());
+ }
+
+ RETURN_STATUS_IF_NULL(d3dsdo->pResource, 0L);
+
+ if (resType == RT_PLAIN || resType == RT_TEXTURE) {
+ return ptr_to_jlong(d3dsdo->pResource->GetSurface());
+ }
+ if (resType == TEXTURE) {
+ return ptr_to_jlong(d3dsdo->pResource->GetTexture());
+ }
+ if (resType == FLIP_BACKBUFFER) {
+ return ptr_to_jlong(d3dsdo->pResource->GetSwapChain());
+ }
+
+ return 0L;
+}
+
+/*
+ * Class: sun_java2d_d3d_D3DSurfaceData
+ * Method: updateWindowAccelImpl
+ * Signature: (JJII)Z
+ */
+JNIEXPORT jboolean
+JNICALL Java_sun_java2d_d3d_D3DSurfaceData_updateWindowAccelImpl
+ (JNIEnv *env, jclass clazz, jlong pd3dsd, jlong pData, jint w, jint h)
+{
+ HRESULT res;
+ AwtWindow *window;
+ HBITMAP hBitmap = NULL;
+ D3DSDOps *d3dsdo;
+ D3DResource *pSrcRes;
+ D3DContext *pCtx;
+ D3DPipelineManager *pMgr;
+ D3DResource *pLockableRes = NULL;
+ IDirect3DSurface9 *pTmpSurface = NULL;
+ IDirect3DDevice9 *pd3dDevice = NULL;
+ D3DLOCKED_RECT lockedRect;
+
+ J2dTraceLn(J2D_TRACE_ERROR, "D3DSurfaceData_updateWindowAccelImpl");
+
+ if (w <= 0 || h <= 0) {
+ return JNI_TRUE;
+ }
+
+ RETURN_STATUS_IF_NULL(window = (AwtWindow *)jlong_to_ptr(pData), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(d3dsdo = (D3DSDOps *)jlong_to_ptr(pd3dsd), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(pMgr = D3DPipelineManager::GetInstance(), JNI_FALSE);
+ RETURN_STATUS_IF_NULL(pSrcRes = d3dsdo->pResource, JNI_FALSE);
+
+ if (FAILED(res = pMgr->GetD3DContext(d3dsdo->adapter, &pCtx))) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return JNI_FALSE;
+ }
+
+ RETURN_STATUS_IF_NULL(pd3dDevice = pCtx->Get3DDevice(), JNI_FALSE);
+ pCtx->UpdateState(STATE_OTHEROP);
+
+ res = pCtx->GetResourceManager()->
+ GetBlitOSPSurface(pSrcRes->GetDesc()->Width,
+ pSrcRes->GetDesc()->Height,
+ pSrcRes->GetDesc()->Format,
+ &pLockableRes);
+ if (FAILED(res)) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return JNI_FALSE;
+ }
+ pTmpSurface = pLockableRes->GetSurface();
+
+ res = pd3dDevice->GetRenderTargetData(pSrcRes->GetSurface(), pTmpSurface);
+ if (FAILED(res)) {
+ D3DRQ_MarkLostIfNeeded(res, d3dsdo);
+ return JNI_FALSE;
+ }
+
+ res = pTmpSurface->LockRect(&lockedRect, NULL, D3DLOCK_NOSYSLOCK);
+ if (SUCCEEDED(res)) {
+ // REMIND: commented until translucent window support is integrated
+// hBitmap =
+// BitmapUtil::CreateBitmapFromARGBPre(w, h,
+// lockedRect.Pitch,
+// (int*)lockedRect.pBits);
+ pTmpSurface->UnlockRect();
+ }
+ RETURN_STATUS_IF_NULL(hBitmap, JNI_FALSE);
+
+ // REMIND: commented until translucent window support is integrated
+// window->UpdateWindow(env, NULL, w, h, hBitmap);
+
+ // hBitmap is released in UpdateWindow
+
+ return JNI_TRUE;
+}
}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.h b/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.h
index 29a6548efcf..22a89bc156d 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.h
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DSurfaceData.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -23,45 +23,71 @@
* have any questions.
*/
-#ifndef D3DSurfaceData_h_Included
-#define D3DSurfaceData_h_Included
+#ifndef _D3DSURFACEDATA_H_
+#define _D3DSURFACEDATA_H_
#include "java_awt_image_AffineTransformOp.h"
#include "sun_java2d_d3d_D3DSurfaceData.h"
-#include "Win32SurfaceData.h"
-
-// Shortcut macros
-
-#define D3D_PLAIN_SURFACE sun_java2d_d3d_D3DSurfaceData_D3D_PLAIN_SURFACE
-#define D3D_TEXTURE_SURFACE sun_java2d_d3d_D3DSurfaceData_D3D_TEXTURE_SURFACE
-#define D3D_BACKBUFFER_SURFACE sun_java2d_d3d_D3DSurfaceData_D3D_BACKBUFFER_SURFACE
-#define D3D_RTT_SURFACE sun_java2d_d3d_D3DSurfaceData_D3D_RTT_SURFACE
-
-#define D3D_RENDER_TARGET sun_java2d_d3d_D3DSurfaceData_D3D_RENDER_TARGET
-#define D3D_ATTACHED_SURFACE sun_java2d_d3d_D3DSurfaceData_D3D_ATTACHED_SURFACE
-
-#define PF_INVALID sun_java2d_d3d_D3DSurfaceData_PF_INVALID
-#define PF_INT_ARGB sun_java2d_d3d_D3DSurfaceData_PF_INT_ARGB
-#define PF_INT_RGB sun_java2d_d3d_D3DSurfaceData_PF_INT_RGB
-#define PF_INT_RGBX sun_java2d_d3d_D3DSurfaceData_PF_INT_RGBX
-#define PF_INT_BGR sun_java2d_d3d_D3DSurfaceData_PF_INT_BGR
-#define PF_USHORT_565_RGB sun_java2d_d3d_D3DSurfaceData_PF_USHORT_565_RGB
-#define PF_USHORT_555_RGB sun_java2d_d3d_D3DSurfaceData_PF_USHORT_555_RGB
-#define PF_USHORT_555_RGBX sun_java2d_d3d_D3DSurfaceData_PF_USHORT_555_RGBX
-#define PF_INT_ARGB_PRE sun_java2d_d3d_D3DSurfaceData_PF_INT_ARGB_PRE
-#define PF_USHORT_4444_ARGB sun_java2d_d3d_D3DSurfaceData_PF_USHORT_4444_ARGB
+#include "sun_java2d_pipe_hw_AccelSurface.h"
+#include "SurfaceData.h"
+#include
typedef struct _D3DSDOps D3DSDOps;
+class D3DResource;
+
struct _D3DSDOps {
- Win32SDOps dxOps;
- jint d3dType; // surface type (plain/texture/bb/rtt) - see D3DSurfaceData.java
+ SurfaceDataOps sdOps;
+
+ // the ordinal of the d3d adapter this surface belongs to
+ // (may be different from GDI display number)
+ jint adapter;
+ jint width, height;
+
+ // backbuffer-related data
+ jint xoff, yoff;
+ D3DSWAPEFFECT swapEffect;
+
+ D3DResource *pResource;
};
+#define UNDEFINED sun_java2d_pipe_hw_AccelSurface_UNDEFINED
+#define RT_PLAIN sun_java2d_pipe_hw_AccelSurface_RT_PLAIN
+#define TEXTURE sun_java2d_pipe_hw_AccelSurface_TEXTURE
+#define RT_TEXTURE sun_java2d_pipe_hw_AccelSurface_RT_TEXTURE
+#define FLIP_BACKBUFFER sun_java2d_pipe_hw_AccelSurface_FLIP_BACKBUFFER
+#define D3D_DEVICE_RESOURCE \
+ sun_java2d_d3d_D3DSurfaceData_D3D_DEVICE_RESOURCE
+
+#define ST_INT_ARGB sun_java2d_d3d_D3DSurfaceData_ST_INT_ARGB
+#define ST_INT_ARGB_PRE sun_java2d_d3d_D3DSurfaceData_ST_INT_ARGB_PRE
+#define ST_INT_ARGB_BM sun_java2d_d3d_D3DSurfaceData_ST_INT_ARGB_BM
+#define ST_INT_RGB sun_java2d_d3d_D3DSurfaceData_ST_INT_RGB
+#define ST_INT_BGR sun_java2d_d3d_D3DSurfaceData_ST_INT_BGR
+#define ST_USHORT_565_RGB sun_java2d_d3d_D3DSurfaceData_ST_USHORT_565_RGB
+#define ST_USHORT_555_RGB sun_java2d_d3d_D3DSurfaceData_ST_USHORT_555_RGB
+#define ST_BYTE_INDEXED sun_java2d_d3d_D3DSurfaceData_ST_BYTE_INDEXED
+#define ST_BYTE_INDEXED_BM sun_java2d_d3d_D3DSurfaceData_ST_BYTE_INDEXED_BM
+
+/**
+ * These are defined to be the same as ExtendedBufferCapabilities.VSyncType
+ * enum.
+ */
+#define VSYNC_DEFAULT 0
+#define VSYNC_ON 1
+#define VSYNC_OFF 2
+
+/**
+ * These are shorthand names for the filtering method constants used by
+ * image transform methods.
+ */
+#define D3DSD_XFORM_DEFAULT 0
#define D3DSD_XFORM_NEAREST_NEIGHBOR \
java_awt_image_AffineTransformOp_TYPE_NEAREST_NEIGHBOR
#define D3DSD_XFORM_BILINEAR \
java_awt_image_AffineTransformOp_TYPE_BILINEAR
+void D3DSD_Flush(void *pData);
+void D3DSD_MarkLost(void *pData);
-#endif /* D3DSurfaceData_h_Included */
+#endif /* _D3DSURFACEDATA_H_ */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DTestRaster.h b/jdk/src/windows/native/sun/java2d/d3d/D3DTestRaster.h
deleted file mode 100644
index de0aa47eec0..00000000000
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DTestRaster.h
+++ /dev/null
@@ -1,131 +0,0 @@
-/*
- * Copyright 2002-2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-/*
- * DO NOT EDIT THIS FILE, IT IS AUTOMATICALLY GENERATED BY D3DTestPattern.java
- */
-
-#ifndef D3DTESTRASTER_H
-#define D3DTESTRASTER_H
-
-#define D3D_TEST_RASTER_W (60)
-#define D3D_TEST_RASTER_H (21)
-
-#define D3D_TEXTURE_RASTER_W (12)
-#define D3D_TEXTURE_RASTER_H (8)
-
-typedef int TIntTestRaster[D3D_TEST_RASTER_H][D3D_TEST_RASTER_W];
-
-typedef int TIntTextureRaster[D3D_TEXTURE_RASTER_H][D3D_TEXTURE_RASTER_W];
-
-typedef int TIntRaster[1][1];
-
-int d3dNumTestLines = 11;
-float d3dTestLines[] = {
- 20.5, 20.5, 0.5, 20.5,
- 20.5, 20.5, 3.5, 12.5,
- 20.5, 20.5, 6.5, 6.5,
- 20.5, 20.5, 12.5, 3.5,
- 20.5, 20.5, 20.5, 0.5,
- 20.5, 20.5, 28.5, 3.5,
- 20.5, 20.5, 34.5, 6.5,
- 20.5, 20.5, 37.5, 12.5,
- 20.5, 20.5, 40.5, 20.5,
- 41.5, 20.5, 60.5, 19.5,
- 41.5, 20.5, 42.5, 1.5,
-};
-int d3dNumTestRects = 1;
-float d3dTestRects[] = {
- 44.5, 0.5, 59.5, 15.5,
-};
-TIntTestRaster d3dTestRaster = {
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 0, 1, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 0, 1, 1, 1, 0, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 1, 1, 1, 1, 1, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0},
-};
-
-int d3dNumTextureRects = 6;
-float d3dTextureRects[] = {
- 0, 0, 4, 21,
- 5, 0, 20, 10,
- 21, 0, 33, 8,
- 34, 0, 40, 4,
- 34, 6, 60, 10,
- 5, 12, 60, 22,
-};
-TIntTestRaster linInterpArray = {
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {0, 1, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0},
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0},
- {0, 1, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0},
- {1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0},
- {1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
- {0, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
-};
-
-TIntTextureRaster srcImageArray = {
- {0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000},
- {0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000},
- {0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000},
- {0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000},
- {0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000},
- {0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000},
- {0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000, 0xff0000ff, 0xffff0000},
- {0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000, 0xff0000ff, 0xff000000},
-};
-
-#endif //D3DTESTRASTER_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.cpp
index 07bc8142886..ea9bb9a2270 100644
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.cpp
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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
@@ -24,109 +24,888 @@
*/
#include
+#include
+#include
+
#include "sun_java2d_d3d_D3DTextRenderer.h"
+#include "sun_java2d_pipe_BufferedTextPipe.h"
+
#include "SurfaceData.h"
-#include "Region.h"
-#include "glyphblitting.h"
-
-extern "C" {
-
-#ifndef D3D_GCACHE_WIDTH
- #define D3D_GCACHE_WIDTH 512
- #define D3D_GCACHE_HEIGHT 512
- #define D3D_GCACHE_CELL_WIDTH 16
- #define D3D_GCACHE_CELL_HEIGHT 16
-#endif
+#include "D3DContext.h"
+#include "D3DSurfaceData.h"
+#include "D3DRenderQueue.h"
+#include "D3DTextRenderer.h"
+#include "D3DGlyphCache.h"
+#include "AccelGlyphCache.h"
+#include "fontscalerdefs.h"
/**
- * This method is almost exactly the same as the RefineBounds() method
- * defined in DrawGlyphList.c. The goal is to determine whether the given
- * GlyphBlitVector intersects with the given bounding box. If any part of
- * the GBV intersects with the bounding box, this method returns true;
- * otherwise false is returned. The only step that differs in this method
- * from RefineBounds() is that we check to see whether all the glyphs in
- * the GBV will fit in the glyph cache. If any glyph is too big for a
- * glyph cache cell, we return FALSE in the useCache out parameter; otherwise
- * useCache is TRUE, indicating that the caller can be assured that all
- * the glyphs can be stored in the accelerated glyph cache.
+ * The current "glyph mode" state. This variable is used to track the
+ * codepath used to render a particular glyph. This variable is reset to
+ * MODE_NOT_INITED at the beginning of every call to D3DTR_DrawGlyphList().
+ * As each glyph is rendered, the glyphMode variable is updated to reflect
+ * the current mode, so if the current mode is the same as the mode used
+ * to render the previous glyph, we can avoid doing costly setup operations
+ * each time.
*/
-jboolean
-D3DRefineBounds(GlyphBlitVector *gbv, SurfaceDataBounds *bounds,
- jboolean *useCache)
+typedef enum {
+ MODE_NOT_INITED,
+ MODE_USE_CACHE_GRAY,
+ MODE_USE_CACHE_LCD,
+ MODE_NO_CACHE_GRAY,
+ MODE_NO_CACHE_LCD
+} GlyphMode;
+static GlyphMode glyphMode = MODE_NOT_INITED;
+
+/**
+ * The current bounds of the "cached destination" texture, in destination
+ * coordinate space. The width/height of these bounds will not exceed the
+ * D3DTR_CACHED_DEST_WIDTH/HEIGHT values defined above. These bounds are
+ * only considered valid when the isCachedDestValid flag is JNI_TRUE.
+ */
+static SurfaceDataBounds cachedDestBounds;
+
+/**
+ * This flag indicates whether the "cached destination" texture contains
+ * valid data. This flag is reset to JNI_FALSE at the beginning of every
+ * call to D3DTR_DrawGlyphList(). Once we copy valid destination data
+ * into the cached texture, this flag is set to JNI_TRUE. This way, we
+ * can limit the number of times we need to copy destination data, which
+ * is a very costly operation.
+ */
+static jboolean isCachedDestValid = JNI_FALSE;
+
+/**
+ * The bounds of the previously rendered LCD glyph, in destination
+ * coordinate space. We use these bounds to determine whether the glyph
+ * currently being rendered overlaps the previously rendered glyph (i.e.
+ * its bounding box intersects that of the previously rendered glyph).
+ * If so, we need to re-read the destination area associated with that
+ * previous glyph so that we can correctly blend with the actual
+ * destination data.
+ */
+static SurfaceDataBounds previousGlyphBounds;
+
+/**
+ * Updates the gamma and inverse gamma values for the LCD text shader.
+ */
+static HRESULT
+D3DTR_UpdateLCDTextContrast(D3DContext *d3dc, jint contrast)
+{
+ HRESULT res;
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+
+ jfloat fcon = ((jfloat)contrast) / 100.0f;
+ jfloat invgamma = fcon;
+ jfloat gamma = 1.0f / invgamma;
+ jfloat vals[4];
+
+ // update the "invgamma" parameter of the shader program
+ vals[0] = invgamma;
+ vals[1] = invgamma;
+ vals[2] = invgamma;
+ vals[3] = 0.0f; // unused
+ pd3dDevice->SetPixelShaderConstantF(1, vals, 1);
+
+ // update the "gamma" parameter of the shader program
+ vals[0] = gamma;
+ vals[1] = gamma;
+ vals[2] = gamma;
+ vals[3] = 0.0f; // unused
+ res = pd3dDevice->SetPixelShaderConstantF(2, vals, 1);
+
+ return res;
+}
+
+/**
+ * Updates the current gamma-adjusted source color ("src_adj") of the LCD
+ * text shader program. Note that we could calculate this value in the
+ * shader (e.g. just as we do for "dst_adj"), but would be unnecessary work
+ * (and a measurable performance hit, maybe around 5%) since this value is
+ * constant over the entire glyph list. So instead we just calculate the
+ * gamma-adjusted value once and update the uniform parameter of the LCD
+ * shader as needed.
+ */
+static HRESULT
+D3DTR_UpdateLCDTextColor(D3DContext *d3dc, jint contrast)
+{
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ jfloat gamma = ((jfloat)contrast) / 100.0f;
+ jfloat clr[4];
+
+ J2dTraceLn1(J2D_TRACE_INFO,
+ "D3DTR_UpdateLCDTextColor: contrast=%d", contrast);
+
+ /*
+ * Note: Ideally we would update the "srcAdj" uniform parameter only
+ * when there is a change in the source color. Fortunately, the cost
+ * of querying the current D3D color state and updating the uniform
+ * value is quite small, and in the common case we only need to do this
+ * once per GlyphList, so we gain little from trying to optimize too
+ * eagerly here.
+ */
+
+ // get the current D3D primary color state
+ jint color = d3dc->pVCacher->GetColor();
+ clr[0] = (jfloat)((color >> 16) & 0xff) / 255.0f;
+ clr[1] = (jfloat)((color >> 8) & 0xff) / 255.0f;
+ clr[2] = (jfloat)((color >> 0) & 0xff) / 255.0f;
+ clr[3] = 0.0f; // unused
+
+ // gamma adjust the primary color
+ clr[0] = (jfloat)pow(clr[0], gamma);
+ clr[1] = (jfloat)pow(clr[1], gamma);
+ clr[2] = (jfloat)pow(clr[2], gamma);
+
+ // update the "srcAdj" parameter of the shader program with this value
+ return pd3dDevice->SetPixelShaderConstantF(0, clr, 1);
+}
+
+/**
+ * Enables the LCD text shader and updates any related state, such as the
+ * gamma values.
+ */
+static HRESULT
+D3DTR_EnableLCDGlyphModeState(D3DContext *d3dc, D3DSDOps *dstOps,
+ jboolean useCache, jint contrast)
+{
+ D3DResource *pGlyphTexRes, *pCachedDestTexRes;
+ IDirect3DTexture9 *pGlyphTex, *pCachedDestTex;
+
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+
+ HRESULT res = S_OK;
+ if (useCache) {
+ // glyph cache had been already initialized
+ pGlyphTexRes = d3dc->GetLCDGlyphCache()->GetGlyphCacheTexture();
+ } else {
+ res = d3dc->GetResourceManager()->GetBlitTexture(&pGlyphTexRes);
+ }
+ RETURN_STATUS_IF_FAILED(res);
+
+ pGlyphTex = pGlyphTexRes->GetTexture();
+
+ res = d3dc->GetResourceManager()->
+ GetCachedDestTexture(dstOps->pResource->GetDesc()->Format,
+ &pCachedDestTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+ pCachedDestTex = pCachedDestTexRes->GetTexture();
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ D3DTEXTUREFILTERTYPE fhint =
+ d3dc->IsTextureFilteringSupported(D3DTEXF_NONE) ?
+ D3DTEXF_NONE : D3DTEXF_POINT;
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, fhint);
+ pd3dDevice->SetSamplerState(1, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(1, D3DSAMP_MINFILTER, fhint);
+ d3dc->UpdateTextureColorState(D3DTA_TEXTURE, 1);
+
+ // bind the texture containing glyph data to texture unit 0
+ d3dc->SetTexture(pGlyphTex, 0);
+
+ // bind the texture tile containing destination data to texture unit 1
+ d3dc->SetTexture(pCachedDestTex, 1);
+
+ // create/enable the LCD text shader
+ res = d3dc->EnableLCDTextProgram();
+ RETURN_STATUS_IF_FAILED(res);
+
+ // update the current contrast settings (note: these change very rarely,
+ // but it seems that D3D pixel shader registers aren't maintained as
+ // part of the pixel shader instance, so we need to update these
+ // everytime around in case another shader blew away the contents
+ // of those registers)
+ D3DTR_UpdateLCDTextContrast(d3dc, contrast);
+
+ // update the current color settings
+ return D3DTR_UpdateLCDTextColor(d3dc, contrast);
+}
+
+HRESULT
+D3DTR_EnableGlyphVertexCache(D3DContext *d3dc)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DTR_EnableGlyphVertexCache");
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ D3DTEXTUREFILTERTYPE fhint =
+ d3dc->IsTextureFilteringSupported(D3DTEXF_NONE) ?
+ D3DTEXF_NONE : D3DTEXF_POINT;
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, fhint);
+ pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, fhint);
+
+ // glyph cache had been successfully initialized if we got here
+ D3DResource *pGlyphCacheTexRes =
+ d3dc->GetGrayscaleGlyphCache()->GetGlyphCacheTexture();
+ return d3dc->SetTexture(pGlyphCacheTexRes->GetTexture(), 0);
+}
+
+HRESULT
+D3DTR_DisableGlyphVertexCache(D3DContext *d3dc)
+{
+ J2dTraceLn(J2D_TRACE_INFO, "D3DTR_DisableGlyphVertexCache");
+
+ return d3dc->SetTexture(NULL, 0);
+}
+
+/**
+ * Disables any pending state associated with the current "glyph mode".
+ */
+static HRESULT
+D3DTR_DisableGlyphModeState(D3DContext *d3dc)
+{
+ HRESULT res = S_OK;
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+
+ switch (glyphMode) {
+ case MODE_NO_CACHE_LCD:
+ case MODE_USE_CACHE_LCD:
+ d3dc->FlushVertexQueue();
+ pd3dDevice->SetPixelShader(NULL);
+ res = d3dc->SetTexture(NULL, 1);
+ break;
+
+ case MODE_NO_CACHE_GRAY:
+ case MODE_USE_CACHE_GRAY:
+ case MODE_NOT_INITED:
+ default:
+ break;
+ }
+ return res;
+}
+
+static HRESULT
+D3DTR_DrawGrayscaleGlyphViaCache(D3DContext *d3dc,
+ GlyphInfo *ginfo, jint x, jint y)
+{
+ HRESULT res = S_OK;
+ D3DGlyphCache *pGrayscaleGCache;
+ CacheCellInfo *cell;
+ GlyphCacheInfo *gcache;
+ jfloat x1, y1, x2, y2;
+
+ J2dTraceLn(J2D_TRACE_VERBOSE, "D3DTR_DrawGrayscaleGlyphViaCache");
+
+ if (glyphMode != MODE_USE_CACHE_GRAY) {
+ D3DTR_DisableGlyphModeState(d3dc);
+
+ res = d3dc->BeginScene(STATE_GLYPHOP);
+ RETURN_STATUS_IF_FAILED(res);
+
+ glyphMode = MODE_USE_CACHE_GRAY;
+ }
+
+ pGrayscaleGCache = d3dc->GetGrayscaleGlyphCache();
+ gcache = pGrayscaleGCache->GetGlyphCache();
+ cell = AccelGlyphCache_GetCellInfoForCache(ginfo, gcache);
+ if (cell == NULL) {
+ // attempt to add glyph to accelerated glyph cache
+ res = pGrayscaleGCache->AddGlyph(ginfo);
+ RETURN_STATUS_IF_FAILED(res);
+
+ cell = AccelGlyphCache_GetCellInfoForCache(ginfo, gcache);
+ RETURN_STATUS_IF_NULL(cell, E_FAIL);
+ }
+
+ cell->timesRendered++;
+
+ x1 = (jfloat)x;
+ y1 = (jfloat)y;
+ x2 = x1 + ginfo->width;
+ y2 = y1 + ginfo->height;
+
+ return d3dc->pVCacher->DrawTexture(x1, y1, x2, y2,
+ cell->tx1, cell->ty1,
+ cell->tx2, cell->ty2);
+}
+
+/**
+ * Evaluates to true if the rectangle defined by gx1/gy1/gx2/gy2 is
+ * inside outerBounds.
+ */
+#define INSIDE(gx1, gy1, gx2, gy2, outerBounds) \
+ (((gx1) >= outerBounds.x1) && ((gy1) >= outerBounds.y1) && \
+ ((gx2) <= outerBounds.x2) && ((gy2) <= outerBounds.y2))
+
+/**
+ * Evaluates to true if the rectangle defined by gx1/gy1/gx2/gy2 intersects
+ * the rectangle defined by bounds.
+ */
+#define INTERSECTS(gx1, gy1, gx2, gy2, bounds) \
+ ((bounds.x2 > (gx1)) && (bounds.y2 > (gy1)) && \
+ (bounds.x1 < (gx2)) && (bounds.y1 < (gy2)))
+
+/**
+ * This method checks to see if the given LCD glyph bounds fall within the
+ * cached destination texture bounds. If so, this method can return
+ * immediately. If not, this method will copy a chunk of framebuffer data
+ * into the cached destination texture and then update the current cached
+ * destination bounds before returning.
+ *
+ * The agx1, agx2 are "adjusted" glyph bounds, which are only used when checking
+ * against the previous glyph bounds.
+ */
+static HRESULT
+D3DTR_UpdateCachedDestination(D3DContext *d3dc, D3DSDOps *dstOps,
+ GlyphInfo *ginfo,
+ jint gx1, jint gy1, jint gx2, jint gy2,
+ jint agx1, jint agx2,
+ jint glyphIndex, jint totalGlyphs)
{
- int index;
jint dx1, dy1, dx2, dy2;
- ImageRef glyphImage;
- int num = gbv->numGlyphs;
- SurfaceDataBounds glyphs;
- jboolean tryCache = JNI_TRUE;
+ D3DResource *pCachedDestTexRes;
+ IDirect3DSurface9 *pCachedDestSurface, *pDst;
+ HRESULT res;
- glyphs.x1 = glyphs.y1 = 0x7fffffff;
- glyphs.x2 = glyphs.y2 = 0x80000000;
- for (index = 0; index < num; index++) {
- glyphImage = gbv->glyphs[index];
- dx1 = (jint) glyphImage.x;
- dy1 = (jint) glyphImage.y;
- dx2 = dx1 + glyphImage.width;
- dy2 = dy1 + glyphImage.height;
- if (glyphs.x1 > dx1) glyphs.x1 = dx1;
- if (glyphs.y1 > dy1) glyphs.y1 = dy1;
- if (glyphs.x2 < dx2) glyphs.x2 = dx2;
- if (glyphs.y2 < dy2) glyphs.y2 = dy2;
+ if (isCachedDestValid && INSIDE(gx1, gy1, gx2, gy2, cachedDestBounds)) {
+ // glyph is already within the cached destination bounds; no need
+ // to read back the entire destination region again, but we do
+ // need to see if the current glyph overlaps the previous glyph...
- if (tryCache &&
- ((glyphImage.width > D3D_GCACHE_CELL_WIDTH) ||
- (glyphImage.height > D3D_GCACHE_CELL_HEIGHT)))
- {
- tryCache = JNI_FALSE;
+ // only use the "adjusted" glyph bounds when checking against
+ // previous glyph's bounds
+ gx1 = agx1;
+ gx2 = agx2;
+
+ if (INTERSECTS(gx1, gy1, gx2, gy2, previousGlyphBounds)) {
+ // the current glyph overlaps the destination region touched
+ // by the previous glyph, so now we need to read back the part
+ // of the destination corresponding to the previous glyph
+ dx1 = previousGlyphBounds.x1;
+ dy1 = previousGlyphBounds.y1;
+ dx2 = previousGlyphBounds.x2;
+ dy2 = previousGlyphBounds.y2;
+
+ // REMIND: make sure we flush any pending primitives that are
+ // dependent on the current contents of the cached dest
+ d3dc->FlushVertexQueue();
+
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(pDst = dstOps->pResource->GetSurface(),
+ E_FAIL);
+ res = d3dc->GetResourceManager()->
+ GetCachedDestTexture(dstOps->pResource->GetDesc()->Format,
+ &pCachedDestTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+ pCachedDestSurface = pCachedDestTexRes->GetSurface();
+
+ // now dxy12 represent the "desired" destination bounds, but the
+ // StretchRect() call may fail if these fall outside the actual
+ // surface bounds; therefore, we use cxy12 to represent the
+ // clamped bounds, and dxy12 are saved for later
+ jint cx1 = (dx1 < 0) ? 0 : dx1;
+ jint cy1 = (dy1 < 0) ? 0 : dy1;
+ jint cx2 = (dx2 > dstOps->width) ? dstOps->width : dx2;
+ jint cy2 = (dy2 > dstOps->height) ? dstOps->height : dy2;
+
+ if (cx2 > cx1 && cy2 > cy1) {
+ // copy destination into subregion of cached texture tile
+ // cx1-cachedDestBounds.x1 == +xoffset from left of texture
+ // cy1-cachedDestBounds.y1 == +yoffset from top of texture
+ // cx2-cachedDestBounds.x1 == +xoffset from left of texture
+ // cy2-cachedDestBounds.y1 == +yoffset from top of texture
+ jint cdx1 = cx1-cachedDestBounds.x1;
+ jint cdy1 = cy1-cachedDestBounds.y1;
+ jint cdx2 = cx2-cachedDestBounds.x1;
+ jint cdy2 = cy2-cachedDestBounds.y1;
+ RECT srcRect = { cx1, cy1, cx2, cy2 };
+ RECT dstRect = { cdx1, cdy1, cdx2, cdy2 };
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ res = pd3dDevice->StretchRect(pDst, &srcRect,
+ pCachedDestSurface, &dstRect,
+ D3DTEXF_NONE);
+ }
+ }
+ } else {
+ // destination region is not valid, so we need to read back a
+ // chunk of the destination into our cached texture
+
+ // position the upper-left corner of the destination region on the
+ // "top" line of glyph list
+ // REMIND: this isn't ideal; it would be better if we had some idea
+ // of the bounding box of the whole glyph list (this is
+ // do-able, but would require iterating through the whole
+ // list up front, which may present its own problems)
+ dx1 = gx1;
+ dy1 = gy1;
+
+ jint remainingWidth;
+ if (ginfo->advanceX > 0) {
+ // estimate the width based on our current position in the glyph
+ // list and using the x advance of the current glyph (this is just
+ // a quick and dirty heuristic; if this is a "thin" glyph image,
+ // then we're likely to underestimate, and if it's "thick" then we
+ // may end up reading back more than we need to)
+ remainingWidth =
+ (jint)(ginfo->advanceX * (totalGlyphs - glyphIndex));
+ if (remainingWidth > D3DTR_CACHED_DEST_WIDTH) {
+ remainingWidth = D3DTR_CACHED_DEST_WIDTH;
+ } else if (remainingWidth < ginfo->width) {
+ // in some cases, the x-advance may be slightly smaller
+ // than the actual width of the glyph; if so, adjust our
+ // estimate so that we can accomodate the entire glyph
+ remainingWidth = ginfo->width;
+ }
+ } else {
+ // a negative advance is possible when rendering rotated text,
+ // in which case it is difficult to estimate an appropriate
+ // region for readback, so we will pick a region that
+ // encompasses just the current glyph
+ remainingWidth = ginfo->width;
+ }
+ dx2 = dx1 + remainingWidth;
+
+ // estimate the height (this is another sloppy heuristic; we'll
+ // make the cached destination region tall enough to encompass most
+ // glyphs that are small enough to fit in the glyph cache, and then
+ // we add a little something extra to account for descenders
+ dy2 = dy1 + D3DTR_CACHE_CELL_HEIGHT + 2;
+
+ // REMIND: make sure we flush any pending primitives that are
+ // dependent on the current contents of the cached dest
+ d3dc->FlushVertexQueue();
+
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(pDst = dstOps->pResource->GetSurface(), E_FAIL);
+ res = d3dc->GetResourceManager()->
+ GetCachedDestTexture(dstOps->pResource->GetDesc()->Format,
+ &pCachedDestTexRes);
+ RETURN_STATUS_IF_FAILED(res);
+ pCachedDestSurface = pCachedDestTexRes->GetSurface();
+
+ // now dxy12 represent the "desired" destination bounds, but the
+ // StretchRect() call may fail if these fall outside the actual
+ // surface bounds; therefore, we use cxy12 to represent the
+ // clamped bounds, and dxy12 are saved for later
+ jint cx1 = (dx1 < 0) ? 0 : dx1;
+ jint cy1 = (dy1 < 0) ? 0 : dy1;
+ jint cx2 = (dx2 > dstOps->width) ? dstOps->width : dx2;
+ jint cy2 = (dy2 > dstOps->height) ? dstOps->height : dy2;
+
+ if (cx2 > cx1 && cy2 > cy1) {
+ // copy destination into cached texture tile (the upper-left
+ // corner of the destination region will be positioned at the
+ // upper-left corner (0,0) of the texture)
+ RECT srcRect = { cx1, cy1, cx2, cy2 };
+ RECT dstRect = { cx1-dx1, cy1-dy1, cx2-dx1, cy2-dy1 };
+
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ res = pd3dDevice->StretchRect(pDst, &srcRect,
+ pCachedDestSurface, &dstRect,
+ D3DTEXF_NONE);
+ }
+
+ // update the cached bounds and mark it valid
+ cachedDestBounds.x1 = dx1;
+ cachedDestBounds.y1 = dy1;
+ cachedDestBounds.x2 = dx2;
+ cachedDestBounds.y2 = dy2;
+ isCachedDestValid = JNI_TRUE;
+ }
+
+ // always update the previous glyph bounds
+ previousGlyphBounds.x1 = gx1;
+ previousGlyphBounds.y1 = gy1;
+ previousGlyphBounds.x2 = gx2;
+ previousGlyphBounds.y2 = gy2;
+
+ return res;
+}
+
+static HRESULT
+D3DTR_DrawLCDGlyphViaCache(D3DContext *d3dc, D3DSDOps *dstOps,
+ GlyphInfo *ginfo, jint x, jint y,
+ jint glyphIndex, jint totalGlyphs,
+ jboolean rgbOrder, jint contrast)
+{
+ HRESULT res;
+ D3DGlyphCache *pLCDGCache;
+ CacheCellInfo *cell;
+ GlyphCacheInfo *gcache;
+ jint dx1, dy1, dx2, dy2;
+ jfloat dtx1, dty1, dtx2, dty2;
+
+ J2dTraceLn(J2D_TRACE_VERBOSE, "D3DTR_DrawLCDGlyphViaCache");
+
+ // the glyph cache is initialized before this method is called
+ pLCDGCache = d3dc->GetLCDGlyphCache();
+
+ if (glyphMode != MODE_USE_CACHE_LCD) {
+ D3DTR_DisableGlyphModeState(d3dc);
+
+ res = d3dc->BeginScene(STATE_TEXTUREOP);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pLCDGCache->CheckGlyphCacheByteOrder(rgbOrder);
+
+ res = D3DTR_EnableLCDGlyphModeState(d3dc, dstOps, JNI_TRUE, contrast);
+ RETURN_STATUS_IF_FAILED(res);
+
+ glyphMode = MODE_USE_CACHE_LCD;
+ }
+
+ gcache = pLCDGCache->GetGlyphCache();
+ cell = AccelGlyphCache_GetCellInfoForCache(ginfo, gcache);
+ if (cell == NULL) {
+ // attempt to add glyph to accelerated glyph cache
+ res = pLCDGCache->AddGlyph(ginfo);
+ RETURN_STATUS_IF_FAILED(res);
+
+ // we'll just no-op in the rare case that the cell is NULL
+ cell = AccelGlyphCache_GetCellInfoForCache(ginfo, gcache);
+ RETURN_STATUS_IF_NULL(cell, E_FAIL);
+ }
+
+ cell->timesRendered++;
+
+ // location of the glyph in the destination's coordinate space
+ dx1 = x;
+ dy1 = y;
+ dx2 = dx1 + ginfo->width;
+ dy2 = dy1 + ginfo->height;
+
+ // copy destination into second cached texture, if necessary
+ D3DTR_UpdateCachedDestination(d3dc,
+ dstOps, ginfo,
+ dx1, dy1,
+ dx2, dy2,
+ dx1 + cell->leftOff, // adjusted dx1
+ dx2 + cell->rightOff, // adjusted dx2
+ glyphIndex, totalGlyphs);
+
+ // texture coordinates of the destination tile
+ dtx1 = ((jfloat)(dx1 - cachedDestBounds.x1)) / D3DTR_CACHED_DEST_WIDTH;
+ dty1 = ((jfloat)(dy1 - cachedDestBounds.y1)) / D3DTR_CACHED_DEST_HEIGHT;
+ dtx2 = ((jfloat)(dx2 - cachedDestBounds.x1)) / D3DTR_CACHED_DEST_WIDTH;
+ dty2 = ((jfloat)(dy2 - cachedDestBounds.y1)) / D3DTR_CACHED_DEST_HEIGHT;
+
+ // render composed texture to the destination surface
+ return d3dc->pVCacher->DrawTexture((jfloat)dx1, (jfloat)dy1,
+ (jfloat)dx2, (jfloat)dy2,
+ cell->tx1, cell->ty1,
+ cell->tx2, cell->ty2,
+ dtx1, dty1, dtx2, dty2);
+}
+
+static HRESULT
+D3DTR_DrawGrayscaleGlyphNoCache(D3DContext *d3dc,
+ GlyphInfo *ginfo, jint x, jint y)
+{
+ jint tw, th;
+ jint sx, sy, sw, sh;
+ jint x0;
+ jint w = ginfo->width;
+ jint h = ginfo->height;
+ HRESULT res = S_OK;
+
+ J2dTraceLn(J2D_TRACE_VERBOSE, "D3DTR_DrawGrayscaleGlyphNoCache");
+
+ if (glyphMode != MODE_NO_CACHE_GRAY) {
+ D3DTR_DisableGlyphModeState(d3dc);
+
+ res = d3dc->BeginScene(STATE_MASKOP);
+ RETURN_STATUS_IF_FAILED(res);
+
+ glyphMode = MODE_NO_CACHE_GRAY;
+ }
+
+ x0 = x;
+ tw = D3D_MASK_CACHE_TILE_WIDTH;
+ th = D3D_MASK_CACHE_TILE_HEIGHT;
+
+ for (sy = 0; sy < h; sy += th, y += th) {
+ x = x0;
+ sh = ((sy + th) > h) ? (h - sy) : th;
+
+ for (sx = 0; sx < w; sx += tw, x += tw) {
+ sw = ((sx + tw) > w) ? (w - sx) : tw;
+
+ res = d3dc->GetMaskCache()->AddMaskQuad(sx, sy, x, y, sw, sh,
+ w, ginfo->image);
}
}
- *useCache = tryCache;
-
- SurfaceData_IntersectBounds(bounds, &glyphs);
- return (bounds->x1 < bounds->x2 && bounds->y1 < bounds->y2);
+ return res;
}
-extern JNIEXPORT void JNICALL
- D3DDrawGlyphList(JNIEnv *env, jobject d3dtr,
- jlong pData, jlong pCtx,
- ImageRef *glyphs, jint totalGlyphs,
- jboolean useCache);
-
-/*
- * Class: sun_java2d_d3d_D3DTextRenderer
- * Method: doDrawGlyphList
- * Signature: (JLsun/java2d/pipe/Region;Lsun/font/GlyphList;)V
- */
-JNIEXPORT void JNICALL Java_sun_java2d_d3d_D3DTextRenderer_doDrawGlyphList
- (JNIEnv *env, jobject d3dtr,
- jlong pData,
- jlong pCtx, jobject clip, jobject glyphlist)
+static HRESULT
+D3DTR_DrawLCDGlyphNoCache(D3DContext *d3dc, D3DSDOps *dstOps,
+ GlyphInfo *ginfo, jint x, jint y,
+ jint rowBytesOffset,
+ jboolean rgbOrder, jint contrast)
{
- GlyphBlitVector* gbv;
- SurfaceDataBounds bounds;
- jboolean useCache;
+ jfloat tx1, ty1, tx2, ty2;
+ jfloat dx1, dy1, dx2, dy2;
+ jfloat dtx1, dty1, dtx2, dty2;
+ jint tw, th;
+ jint sx, sy, sw, sh;
+ jint cx1, cy1, cx2, cy2;
+ jint x0;
+ jint w = ginfo->width;
+ jint h = ginfo->height;
+ TileFormat tileFormat = rgbOrder ? TILEFMT_3BYTE_RGB : TILEFMT_3BYTE_BGR;
- if ((pData == 0) || (pCtx == 0)) {
- return;
+ IDirect3DDevice9 *pd3dDevice = d3dc->Get3DDevice();
+ D3DResource *pBlitTextureRes, *pCachedDestTextureRes;
+ IDirect3DTexture9 *pBlitTexture;
+ IDirect3DSurface9 *pCachedDestSurface, *pDst;
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_VERBOSE, "D3DTR_DrawLCDGlyphNoCache");
+
+ RETURN_STATUS_IF_NULL(dstOps->pResource, E_FAIL);
+ RETURN_STATUS_IF_NULL(pDst = dstOps->pResource->GetSurface(), E_FAIL);
+
+ res = d3dc->GetResourceManager()->GetBlitTexture(&pBlitTextureRes);
+ RETURN_STATUS_IF_FAILED(res);
+
+ res = d3dc->GetResourceManager()->
+ GetCachedDestTexture(dstOps->pResource->GetDesc()->Format,
+ &pCachedDestTextureRes);
+ RETURN_STATUS_IF_FAILED(res);
+
+ pBlitTexture = pBlitTextureRes->GetTexture();
+ pCachedDestSurface = pCachedDestTextureRes->GetSurface();
+
+ if (glyphMode != MODE_NO_CACHE_LCD) {
+ D3DTR_DisableGlyphModeState(d3dc);
+
+ res = d3dc->BeginScene(STATE_TEXTUREOP);
+ RETURN_STATUS_IF_FAILED(res);
+ res = D3DTR_EnableLCDGlyphModeState(d3dc,dstOps, JNI_FALSE, contrast);
+ RETURN_STATUS_IF_FAILED(res);
+
+ glyphMode = MODE_NO_CACHE_LCD;
}
- Region_GetBounds(env, clip, &bounds);
+ x0 = x;
+ tx1 = 0.0f;
+ ty1 = 0.0f;
+ dtx1 = 0.0f;
+ dty1 = 0.0f;
+ tw = D3DTR_NOCACHE_TILE_SIZE;
+ th = D3DTR_NOCACHE_TILE_SIZE;
- if ((gbv = setupBlitVector(env, glyphlist)) == NULL) {
- return;
+ for (sy = 0; sy < h; sy += th, y += th) {
+ x = x0;
+ sh = ((sy + th) > h) ? (h - sy) : th;
+
+ for (sx = 0; sx < w; sx += tw, x += tw) {
+ sw = ((sx + tw) > w) ? (w - sx) : tw;
+
+ // calculate the bounds of the tile to be copied from the
+ // destination into the cached tile
+ cx1 = x;
+ cy1 = y;
+ cx2 = cx1 + sw;
+ cy2 = cy1 + sh;
+
+ // need to clamp to the destination bounds, otherwise the
+ // StretchRect() call may fail
+ if (cx1 < 0) cx1 = 0;
+ if (cy1 < 0) cy1 = 0;
+ if (cx2 > dstOps->width) cx2 = dstOps->width;
+ if (cy2 > dstOps->height) cy2 = dstOps->height;
+
+ if (cx2 > cx1 && cy2 > cy1) {
+ // copy LCD mask into glyph texture tile
+ d3dc->UploadTileToTexture(pBlitTextureRes,
+ ginfo->image+rowBytesOffset,
+ 0, 0, sx, sy, sw, sh,
+ ginfo->rowBytes, tileFormat);
+
+ // update the lower-right glyph texture coordinates
+ tx2 = ((jfloat)sw) / D3DC_BLIT_TILE_SIZE;
+ ty2 = ((jfloat)sh) / D3DC_BLIT_TILE_SIZE;
+
+ // calculate the actual destination vertices
+ dx1 = (jfloat)x;
+ dy1 = (jfloat)y;
+ dx2 = dx1 + sw;
+ dy2 = dy1 + sh;
+
+ // copy destination into cached texture tile (the upper-left
+ // corner of the destination region will be positioned at the
+ // upper-left corner (0,0) of the texture)
+ RECT srcRect = { cx1, cy1, cx2, cy2 };
+ RECT dstRect = { cx1-x, cy1-y, cx2-x, cy2-y };
+ pd3dDevice->StretchRect(pDst, &srcRect,
+ pCachedDestSurface,
+ &dstRect,
+ D3DTEXF_NONE);
+
+ // update the remaining destination texture coordinates
+ dtx2 = ((jfloat)sw) / D3DTR_CACHED_DEST_WIDTH;
+ dty2 = ((jfloat)sh) / D3DTR_CACHED_DEST_HEIGHT;
+
+ // render composed texture to the destination surface
+ res = d3dc->pVCacher->DrawTexture( dx1, dy1, dx2, dy2,
+ tx1, ty1, tx2, ty2,
+ dtx1, dty1, dtx2, dty2);
+
+ // unfortunately we need to flush after each tile
+ d3dc->FlushVertexQueue();
+ }
+ }
}
- if (!D3DRefineBounds(gbv, &bounds, &useCache)) {
- free(gbv);
- return;
- }
-
- D3DDrawGlyphList(env, d3dtr,
- pData, pCtx,
- gbv->glyphs, gbv->numGlyphs, useCache);
- free(gbv);
+ return res;
}
+// see DrawGlyphList.c for more on this macro...
+#define FLOOR_ASSIGN(l, r) \
+ if ((r)<0) (l) = ((int)floor(r)); else (l) = ((int)(r))
+
+HRESULT
+D3DTR_DrawGlyphList(D3DContext *d3dc, D3DSDOps *dstOps,
+ jint totalGlyphs, jboolean usePositions,
+ jboolean subPixPos, jboolean rgbOrder, jint lcdContrast,
+ jfloat glyphListOrigX, jfloat glyphListOrigY,
+ unsigned char *images, unsigned char *positions)
+{
+ int glyphCounter;
+ HRESULT res = S_OK;
+ J2dTraceLn(J2D_TRACE_INFO, "D3DTR_DrawGlyphList");
+
+ RETURN_STATUS_IF_NULL(d3dc, E_FAIL);
+ RETURN_STATUS_IF_NULL(d3dc->Get3DDevice(), E_FAIL);
+ RETURN_STATUS_IF_NULL(dstOps, E_FAIL);
+ RETURN_STATUS_IF_NULL(images, E_FAIL);
+ if (usePositions) {
+ RETURN_STATUS_IF_NULL(positions, E_FAIL);
+ }
+
+ glyphMode = MODE_NOT_INITED;
+ isCachedDestValid = JNI_FALSE;
+
+ for (glyphCounter = 0; glyphCounter < totalGlyphs; glyphCounter++) {
+ jint x, y;
+ jfloat glyphx, glyphy;
+ jboolean grayscale;
+ GlyphInfo *ginfo = (GlyphInfo *)jlong_to_ptr(NEXT_LONG(images));
+
+ if (ginfo == NULL) {
+ // this shouldn't happen, but if it does we'll just break out...
+ J2dRlsTraceLn(J2D_TRACE_ERROR,
+ "D3DTR_DrawGlyphList: glyph info is null");
+ break;
+ }
+
+ grayscale = (ginfo->rowBytes == ginfo->width);
+
+ if (usePositions) {
+ jfloat posx = NEXT_FLOAT(positions);
+ jfloat posy = NEXT_FLOAT(positions);
+ glyphx = glyphListOrigX + posx + ginfo->topLeftX;
+ glyphy = glyphListOrigY + posy + ginfo->topLeftY;
+ FLOOR_ASSIGN(x, glyphx);
+ FLOOR_ASSIGN(y, glyphy);
+ } else {
+ glyphx = glyphListOrigX + ginfo->topLeftX;
+ glyphy = glyphListOrigY + ginfo->topLeftY;
+ FLOOR_ASSIGN(x, glyphx);
+ FLOOR_ASSIGN(y, glyphy);
+ glyphListOrigX += ginfo->advanceX;
+ glyphListOrigY += ginfo->advanceY;
+ }
+
+ if (ginfo->image == NULL) {
+ continue;
+ }
+
+ if (grayscale) {
+ // grayscale or monochrome glyph data
+ if (ginfo->width <= D3DTR_CACHE_CELL_WIDTH &&
+ ginfo->height <= D3DTR_CACHE_CELL_HEIGHT &&
+ SUCCEEDED(d3dc->InitGrayscaleGlyphCache()))
+ {
+ res = D3DTR_DrawGrayscaleGlyphViaCache(d3dc, ginfo, x, y);
+ } else {
+ res = D3DTR_DrawGrayscaleGlyphNoCache(d3dc, ginfo, x, y);
+ }
+ } else {
+ // LCD-optimized glyph data
+ jint rowBytesOffset = 0;
+
+ if (subPixPos) {
+ jint frac = (jint)((glyphx - x) * 3);
+ if (frac != 0) {
+ rowBytesOffset = 3 - frac;
+ x += 1;
+ }
+ }
+
+ if (rowBytesOffset == 0 &&
+ ginfo->width <= D3DTR_CACHE_CELL_WIDTH &&
+ ginfo->height <= D3DTR_CACHE_CELL_HEIGHT &&
+ SUCCEEDED(d3dc->InitLCDGlyphCache()))
+ {
+ res = D3DTR_DrawLCDGlyphViaCache(d3dc, dstOps,
+ ginfo, x, y,
+ glyphCounter, totalGlyphs,
+ rgbOrder, lcdContrast);
+ } else {
+ res = D3DTR_DrawLCDGlyphNoCache(d3dc, dstOps,
+ ginfo, x, y,
+ rowBytesOffset,
+ rgbOrder, lcdContrast);
+ }
+ }
+
+ if (FAILED(res)) {
+ break;
+ }
+ }
+
+ D3DTR_DisableGlyphModeState(d3dc);
+ return res;
+}
+
+JNIEXPORT void JNICALL
+Java_sun_java2d_d3d_D3DTextRenderer_drawGlyphList
+ (JNIEnv *env, jobject self,
+ jint numGlyphs, jboolean usePositions,
+ jboolean subPixPos, jboolean rgbOrder, jint lcdContrast,
+ jfloat glyphListOrigX, jfloat glyphListOrigY,
+ jlongArray imgArray, jfloatArray posArray)
+{
+ unsigned char *images;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DTextRenderer_drawGlyphList");
+
+ images = (unsigned char *)
+ env->GetPrimitiveArrayCritical(imgArray, NULL);
+ if (images != NULL) {
+ D3DContext *d3dc = D3DRQ_GetCurrentContext();
+ D3DSDOps *dstOps = D3DRQ_GetCurrentDestination();
+
+ if (usePositions) {
+ unsigned char *positions = (unsigned char *)
+ env->GetPrimitiveArrayCritical(posArray, NULL);
+ if (positions != NULL) {
+ D3DTR_DrawGlyphList(d3dc, dstOps,
+ numGlyphs, usePositions,
+ subPixPos, rgbOrder, lcdContrast,
+ glyphListOrigX, glyphListOrigY,
+ images, positions);
+ env->ReleasePrimitiveArrayCritical(posArray,
+ positions, JNI_ABORT);
+ }
+ } else {
+ D3DTR_DrawGlyphList(d3dc, dstOps,
+ numGlyphs, usePositions,
+ subPixPos, rgbOrder, lcdContrast,
+ glyphListOrigX, glyphListOrigY,
+ images, NULL);
+ }
+
+ // reset current state, and ensure rendering is flushed to dest
+ if (d3dc != NULL) {
+ d3dc->FlushVertexQueue();
+ }
+
+ env->ReleasePrimitiveArrayCritical(imgArray,
+ images, JNI_ABORT);
+ }
}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.h b/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.h
new file mode 100644
index 00000000000..a251d993bc3
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer.h
@@ -0,0 +1,96 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DTextRenderer_h_Included
+#define D3DTextRenderer_h_Included
+
+#include
+#include
+#include "sun_java2d_pipe_BufferedTextPipe.h"
+#include "AccelGlyphCache.h"
+#include "D3DContext.h"
+#include "D3DSurfaceData.h"
+
+/**
+ * The following constants define the inner and outer bounds of the
+ * accelerated glyph cache.
+ */
+#define D3DTR_CACHE_WIDTH 512
+#define D3DTR_CACHE_HEIGHT 512
+#define D3DTR_CACHE_CELL_WIDTH 16
+#define D3DTR_CACHE_CELL_HEIGHT 16
+
+/**
+ * This constant defines the size of the tile to use in the
+ * D3DTR_DrawLCDGlyphNoCache() method. See below for more on why we
+ * restrict this value to a particular size.
+ */
+#define D3DTR_NOCACHE_TILE_SIZE 32
+
+/**
+ * These constants define the size of the "cached destination" texture.
+ * This texture is only used when rendering LCD-optimized text, as that
+ * codepath needs direct access to the destination. There is no way to
+ * access the framebuffer directly from a Direct3D shader, so we need to first
+ * copy the destination region corresponding to a particular glyph into
+ * this cached texture, and then that texture will be accessed inside the
+ * shader. Copying the destination into this cached texture can be a very
+ * expensive operation (accounting for about half the rendering time for
+ * LCD text), so to mitigate this cost we try to bulk read a horizontal
+ * region of the destination at a time. (These values are empirically
+ * derived for the common case where text runs horizontally.)
+ *
+ * Note: It is assumed in various calculations below that:
+ * (D3DTR_CACHED_DEST_WIDTH >= D3DTR_CACHE_CELL_WIDTH) &&
+ * (D3DTR_CACHED_DEST_WIDTH >= D3DTR_NOCACHE_TILE_SIZE) &&
+ * (D3DTR_CACHED_DEST_HEIGHT >= D3DTR_CACHE_CELL_HEIGHT) &&
+ * (D3DTR_CACHED_DEST_HEIGHT >= D3DTR_NOCACHE_TILE_SIZE)
+ */
+#define D3DTR_CACHED_DEST_WIDTH 512
+#define D3DTR_CACHED_DEST_HEIGHT 32
+
+#define BYTES_PER_GLYPH_IMAGE \
+ sun_java2d_pipe_BufferedTextPipe_BYTES_PER_GLYPH_IMAGE
+#define BYTES_PER_GLYPH_POSITION \
+ sun_java2d_pipe_BufferedTextPipe_BYTES_PER_GLYPH_POSITION
+#define BYTES_PER_POSITIONED_GLYPH \
+ (BYTES_PER_GLYPH_IMAGE + BYTES_PER_GLYPH_POSITION)
+
+#define OFFSET_CONTRAST sun_java2d_pipe_BufferedTextPipe_OFFSET_CONTRAST
+#define OFFSET_RGBORDER sun_java2d_pipe_BufferedTextPipe_OFFSET_RGBORDER
+#define OFFSET_SUBPIXPOS sun_java2d_pipe_BufferedTextPipe_OFFSET_SUBPIXPOS
+#define OFFSET_POSITIONS sun_java2d_pipe_BufferedTextPipe_OFFSET_POSITIONS
+
+HRESULT D3DTR_EnableGlyphVertexCache(D3DContext *d3dc);
+HRESULT D3DTR_DisableGlyphVertexCache(D3DContext *d3dc);
+
+HRESULT D3DTR_DrawGlyphList(D3DContext *d3dc, D3DSDOps *dstOps,
+ jint totalGlyphs, jboolean usePositions,
+ jboolean subPixPos, jboolean rgbOrder,
+ jint lcdContrast,
+ jfloat glyphListOrigX, jfloat glyphListOrigY,
+ unsigned char *images, unsigned char *positions);
+
+#endif /* D3DTextRenderer_h_Included */
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer_md.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer_md.cpp
deleted file mode 100644
index 980f62b8164..00000000000
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DTextRenderer_md.cpp
+++ /dev/null
@@ -1,341 +0,0 @@
-/*
- * Copyright 2005-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include
-#include "sun_java2d_d3d_D3DTextRenderer.h"
-#include "SurfaceData.h"
-#include "Region.h"
-#include "glyphblitting.h"
-#include "fontscalerdefs.h"
-#include "AccelGlyphCache.h"
-#include "j2d_md.h"
-#include "jlong.h"
-
-#include "ddrawUtils.h"
-#include "D3DContext.h"
-#include "D3DUtils.h"
-#include "Win32SurfaceData.h"
-
-extern "C" {
-
-#define MAX_STATIC_QUADS_NUM 40
-static int indicesInited = 0;
-static short vertexIndices[MAX_STATIC_QUADS_NUM*6];
-static J2DLV_QUAD vertexQuads[MAX_STATIC_QUADS_NUM];
-
-/**
- * Initializes the array of index vertices used for rendering
- * glyphs using cached texture.
- */
-static void
-InitIndexArray()
-{
- int ii, vi;
- memset(vertexQuads, 0, sizeof(vertexQuads));
- for (ii = 0, vi = 0; ii < MAX_STATIC_QUADS_NUM*6; ii += 6, vi += 4) {
- vertexIndices[ii + 0] = vi + 0;
- vertexIndices[ii + 1] = vi + 1;
- vertexIndices[ii + 2] = vi + 2;
- vertexIndices[ii + 3] = vi + 0;
- vertexIndices[ii + 4] = vi + 2;
- vertexIndices[ii + 5] = vi + 3;
- }
-}
-
-/**
- * Renders each glyph directly from the glyph texture cache.
- */
-static HRESULT
-D3DDrawGlyphList_UseCache(JNIEnv *env, Win32SDOps *wsdo,
- D3DContext *d3dc,
- ImageRef *glyphs, jint totalGlyphs)
-{
- int glyphCounter;
- HRESULT res = DDERR_GENERIC;
- DXSurface *glyphCacheTexture;
- J2dTraceLn(J2D_TRACE_INFO, "D3DDrawGlyphList_UseCache");
- int color = d3dc->colorPixel;
- int quadCounter = 0;
-
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return DDERR_GENERIC;
- }
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- glyphCacheTexture = d3dc->GetGlyphCacheTexture();
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL ||
- FAILED(res = d3dc->BeginScene(STATE_MASKOP)))
- {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return res;
- }
-
- if (FAILED(res = d3dc->SetTexture(glyphCacheTexture))) {
- d3dc->EndScene(res);
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return res;
- }
-
- if (!indicesInited) {
- InitIndexArray();
- indicesInited = 1;
- }
-
- for (glyphCounter = 0; (glyphCounter < totalGlyphs) && SUCCEEDED(res);
- glyphCounter++)
- {
- // render glyph cached in texture object
- const jubyte *pixels = (const jubyte *)glyphs[glyphCounter].pixels;
- GlyphInfo *ginfo = (GlyphInfo *)glyphs[glyphCounter].glyphInfo;
- CacheCellInfo *cell;
- float x1, y1, x2, y2;
- float tx1, ty1, tx2, ty2;
- J2DLV_QUAD *quad;
-
- // it the glyph is an empty space, skip it
- if (!pixels) {
- continue;
- }
-
- if (ginfo->cellInfo == NULL ||
- // REMIND: this is a temp fix to allow a glyph be cached
- // in caches for different devices.
- // REMIND: check if this is even a problem: we're using
- // managed textures, they may be automatically accelerated
- // on a different device.
- // If the glyph is cached on a different device, cache
- // it on this context's device.
- // This may result in thrashing if the same glyphs
- // get rendered on different devices.
- // Note: this is not thread-safe: we may change the coordinates
- // while another thread is using this cell.
- // A proper fix would allow a glyph to be cached in multiple
- // caches at the same time.
- d3dc->GetGlyphCache() != ginfo->cellInfo->cacheInfo)
- {
- // attempt to add glyph to accelerated glyph cache
- if (FAILED(d3dc->GlyphCacheAdd(env, ginfo)) ||
- ginfo->cellInfo == NULL)
- {
- continue;
- }
- }
-
- cell = ginfo->cellInfo;
- cell->timesRendered++;
-
- x1 = (float)glyphs[glyphCounter].x;
- y1 = (float)glyphs[glyphCounter].y;
- x2 = x1 + (float)glyphs[glyphCounter].width;
- y2 = y1 + (float)glyphs[glyphCounter].height;
- tx1 = cell->tx1;
- ty1 = cell->ty1;
- tx2 = cell->tx2;
- ty2 = cell->ty2;
- quad = &vertexQuads[quadCounter++];
-
- D3DU_INIT_VERTEX_QUAD(*quad, x1, y1, x2, y2, color ,
- tx1, ty1, tx2, ty2);
-
- if (quadCounter == MAX_STATIC_QUADS_NUM &&
- SUCCEEDED(res = ddTargetSurface->IsLost()))
- {
- res = d3dDevice->DrawIndexedPrimitive(D3DPT_TRIANGLELIST,
- D3DFVF_J2DLVERTEX,
- vertexQuads,
- 4*quadCounter,
- (LPWORD)vertexIndices,
- 6*quadCounter, 0);
- quadCounter = 0;
- }
- }
-
- if (quadCounter > 0 && SUCCEEDED(res)) {
- res = d3dDevice->DrawIndexedPrimitive(D3DPT_TRIANGLELIST,
- D3DFVF_J2DLVERTEX,
- vertexQuads,
- 4*quadCounter,
- (LPWORD)vertexIndices,
- 6*quadCounter, 0);
- }
-
- d3dc->EndScene(res);
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
-
- return res;
-}
-
-static HRESULT
-D3DDrawGlyphList_NoCache(JNIEnv *env, Win32SDOps *wsdo,
- D3DContext *d3dc,
- ImageRef *glyphs, jint totalGlyphs)
-{
- int glyphCounter;
- float tx1, ty1, tx2, ty2;
- jint tw, th;
- DXSurface *maskTexture;
- static J2DLVERTEX quadVerts[4] = {
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f },
- { 0.0f, 0.0f, 0.0f, 0x0, 0.0f, 0.0f }
- };
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DDrawGlyphList_NoCache");
-
- DDrawSurface *ddTargetSurface = d3dc->GetTargetSurface();
- if (ddTargetSurface == NULL) {
- return DDERR_GENERIC;
- }
- ddTargetSurface->GetExclusiveAccess();
- d3dc->GetExclusiveAccess();
-
- HRESULT res = DDERR_GENERIC;
-
- IDirect3DDevice7 *d3dDevice = d3dc->Get3DDevice();
- if (d3dDevice == NULL) {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return res;
- }
-
- maskTexture = d3dc->GetMaskTexture();
- if (maskTexture == NULL ||
- FAILED(res = d3dc->BeginScene(STATE_MASKOP)))
- {
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return DDERR_GENERIC;
- }
-
- if (FAILED(res = d3dc->SetTexture(maskTexture))) {
- d3dc->EndScene(res);
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
- return res;
- }
-
- tx1 = 0.0f;
- ty1 = 0.0f;
- tw = D3DSD_MASK_TILE_SIZE;
- th = D3DSD_MASK_TILE_SIZE;
-
- D3DU_INIT_VERTEX_QUAD_COLOR(quadVerts, d3dc->colorPixel);
- for (glyphCounter = 0; (glyphCounter < totalGlyphs) && SUCCEEDED(res);
- glyphCounter++)
- {
- // render system memory glyph image
- jint sx, sy, sw, sh;
- jint x, y, w, h, x0;
- const jubyte *pixels = (const jubyte *)glyphs[glyphCounter].pixels;
-
- if (!pixels) {
- continue;
- }
-
- x = glyphs[glyphCounter].x;
- y = glyphs[glyphCounter].y;
- w = glyphs[glyphCounter].width;
- h = glyphs[glyphCounter].height;
- x0 = x;
-
- for (sy = 0; sy < h; sy += th, y += th) {
- x = x0;
- sh = ((sy + th) > h) ? (h - sy) : th;
-
- for (sx = 0; sx < w; sx += tw, x += tw) {
- sw = ((sx + tw) > w) ? (w - sx) : tw;
-
- if (FAILED(d3dc->UploadImageToTexture(maskTexture,
- (jubyte*)pixels,
- 0, 0, sx, sy,
- sw, sh, w)))
- {
- continue;
- }
-
- // update the lower right texture coordinates
- tx2 = ((float)sw) / tw;
- ty2 = ((float)sh) / th;
-
- D3DU_INIT_VERTEX_QUAD_XYUV(quadVerts,
- (float)x, (float)y,
- (float)(x+sw), (float)(y+sh),
- tx1, ty1, tx2, ty2);
- if (SUCCEEDED(res = ddTargetSurface->IsLost())) {
- res = d3dDevice->DrawPrimitive(D3DPT_TRIANGLEFAN,
- D3DFVF_J2DLVERTEX,
- quadVerts, 4, 0);
- }
- }
- }
- }
-
- d3dc->EndScene(res);
-
- d3dc->ReleaseExclusiveAccess();
- ddTargetSurface->ReleaseExclusiveAccess();
-
- return res;
-}
-
-
-JNIEXPORT void JNICALL
-D3DDrawGlyphList(JNIEnv *env, jobject d3dtr,
- jlong pData, jlong pCtx,
- ImageRef *glyphs, jint totalGlyphs,
- jboolean useCache)
-{
- Win32SDOps *wsdo = (Win32SDOps *)jlong_to_ptr(pData);
- D3DContext *d3dc = (D3DContext *)jlong_to_ptr(pCtx);
-
- HRESULT res;
- // Note: uncomment to control glyph caching via env. variable.
- // useCache = useCache && !getenv("J2D_D3D_NOGLYPHCACHING");
-
- if (d3dc == NULL) {
- return;
- }
-
- if (useCache && SUCCEEDED(res = d3dc->InitGlyphCache())) {
- D3D_EXEC_PRIM_LOOP(env, res, wsdo,
- D3DDrawGlyphList_UseCache(env, wsdo, d3dc, glyphs,
- totalGlyphs));
- return;
- }
-
- D3D_EXEC_PRIM_LOOP(env, res, wsdo,
- D3DDrawGlyphList_NoCache(env, wsdo, d3dc, glyphs,
- totalGlyphs));
-}
-
-}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DUtils.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DUtils.cpp
deleted file mode 100644
index af5c6a7bd30..00000000000
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DUtils.cpp
+++ /dev/null
@@ -1,606 +0,0 @@
-/*
- * Copyright 2005-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-
-#include "ddrawUtils.h"
-#include "D3DUtils.h"
-#include "D3DSurfaceData.h"
-
-#ifdef DEBUG
-// These strings must be in the same order as pixel
-// formats in D3DSurfaceData.java
-char * TR_NAMES[] = {
- "TR_OPAQUE",
- "TR_BITMASK",
- "TR_TRANSLUCENT"
-};
-
-char * PF_NAMES[] = {
- "PF_INVALID" ,
- "PF_INT_ARGB" ,
- "PF_INT_RGB" ,
- "PF_INT_RGBX",
- "PF_INT_BGR" ,
- "PF_USHORT_565_RGB" ,
- "PF_USHORT_555_RGB" ,
- "PF_USHORT_555_RGBX" ,
- "PF_INT_ARGB_PRE" ,
- "PF_USHORT_4444_ARGB"
-};
-#endif // DEBUG
-
-/**
- * This structure could be used when searching for a pixel
- * format with preferred bit depth.
- */
-typedef struct {
- // Pointer to a DDPIXELFORMAT structure where the found pixel
- // format will be copied to
- DDPIXELFORMAT *pddpf;
- // If TRUE, the search was successful, FALSE otherwise
- BOOL bFoundFormat;
- // Preferred bit depth
- int preferredDepth;
-} PixelFormatSearchStruct;
-
-jint D3DUtils_GetPixelFormatType(DDPIXELFORMAT*lpddpf);
-
-HRESULT WINAPI EnumAlphaTextureFormatsCallback(DDPIXELFORMAT* pddpf,
- VOID* pContext )
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "EnumAlphaTextureFormatsCallback");
- DDPIXELFORMAT* pddpfOut = (DDPIXELFORMAT*)pContext;
-
- // Looking for a 8-bit luminance texture (and probably not alpha-luminance)
- if((pddpf->dwFlags & DDPF_ALPHA) && (pddpf->dwAlphaBitDepth == 8))
- {
- memcpy(pddpfOut, pddpf, sizeof(DDPIXELFORMAT));
- return D3DENUMRET_CANCEL;
- }
-
- return D3DENUMRET_OK;
-}
-
-HRESULT CALLBACK
-D3DUtils_TextureSearchCallback(DDPIXELFORMAT *lpddpf,
- void *param)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "D3DUtils_TextureSearchCallback");
- jint pfType = D3DUtils_GetPixelFormatType(lpddpf);
- if (pfType == PF_INVALID) {
- return DDENUMRET_OK;
- }
-
- DWORD dwAlphaBitCount = 0;
- if (lpddpf->dwFlags & DDPF_ALPHAPIXELS) {
- DWORD dwMask = lpddpf->dwRGBAlphaBitMask;
- while( dwMask ) {
- dwMask = dwMask & ( dwMask - 1 );
- dwAlphaBitCount++;
- }
- }
-
- DWORD dwRGBBitCount = lpddpf->dwRGBBitCount;
- WORD wDepthIndex = D3D_DEPTH_IDX(dwRGBBitCount);
- WORD wTransparencyIndex =
- dwAlphaBitCount > 0 ? TR_TRANSLUCENT_IDX : TR_OPAQUE_IDX;
-
- D3DTextureTable *table = (D3DTextureTable*)param;
- D3DTextureTableCell *cell = &(*table)[wTransparencyIndex][wDepthIndex];
- if (cell->pfType == PF_INVALID || pfType < cell->pfType) {
- // set only if it wasn't set or if current pfType is better than
- // the one found previously: it's better to use 565 than 555
- memcpy(&cell->pddpf, lpddpf, sizeof(DDPIXELFORMAT));
- cell->pfType = pfType;
- }
- // continue for all pixel formats
- return DDENUMRET_OK;
-}
-
-HRESULT
-WINAPI EnumZBufferFormatsCallback(DDPIXELFORMAT* pddpf,
- VOID* pContext )
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "EnumZBufferFormatsCallback");
- PixelFormatSearchStruct *ppfss = (PixelFormatSearchStruct*)pContext;
- DDPIXELFORMAT* pddpfOut = ppfss->pddpf;
-
- // if found a format with the exact depth, return it
- if (pddpf->dwZBufferBitDepth == (DWORD)ppfss->preferredDepth) {
- ppfss->bFoundFormat = TRUE;
- memcpy(pddpfOut, pddpf, sizeof(DDPIXELFORMAT));
- return D3DENUMRET_CANCEL;
- }
- // If a format with exact depth can't be found, look for the best
- // available, preferring those with the lowest bit depth to save
- // video memory. Also, prefer formats with no stencil bits.
- if (!ppfss->bFoundFormat ||
- (pddpfOut->dwZBufferBitDepth > pddpf->dwZBufferBitDepth &&
- !(pddpf->dwFlags & DDPF_STENCILBUFFER)))
- {
- ppfss->bFoundFormat = TRUE;
- memcpy(pddpfOut, pddpf, sizeof(DDPIXELFORMAT));
- }
-
- return D3DENUMRET_OK;
-}
-
-HRESULT
-WINAPI DeviceEnumCallback(LPSTR strDesc, LPSTR strName,
- LPD3DDEVICEDESC7 pDesc,
- LPVOID pParentInfo)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "DeviceEnumCallback");
- DEVICES_INFO *devinfo = (DEVICES_INFO*)pParentInfo;
-
- if (pDesc->deviceGUID == IID_IDirect3DHALDevice) {
- devinfo->pGUIDs[HAL_IDX] = &IID_IDirect3DHALDevice;
- } else if (pDesc->deviceGUID == IID_IDirect3DTnLHalDevice) {
- devinfo->pGUIDs[TNL_IDX] = &IID_IDirect3DTnLHalDevice;
- } else if (pDesc->deviceGUID == IID_IDirect3DRGBDevice) {
- devinfo->pGUIDs[RGB_IDX] = &IID_IDirect3DRGBDevice;
- } else if (pDesc->deviceGUID == IID_IDirect3DRefDevice) {
- devinfo->pGUIDs[REF_IDX] = &IID_IDirect3DRefDevice;
- }
- return D3DENUMRET_OK;
-}
-
-HRESULT
-D3DUtils_FindMaskTileTextureFormat(IDirect3DDevice7 *d3dDevice,
- DDPIXELFORMAT* pddpf)
-{
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_FindMaskTileTextureFormat");
- d3dDevice->EnumTextureFormats(EnumAlphaTextureFormatsCallback,
- (void*)pddpf);
- if (pddpf->dwAlphaBitDepth == 8) {
- return D3D_OK;
- }
- return DDERR_GENERIC;
-}
-
-HRESULT
-D3DUtils_FindDepthBufferFormat(IDirect3D7 *d3dObject,
- int preferredDepth,
- DDPIXELFORMAT* pddpf,
- const GUID *pDeviceGUID)
-{
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_FindDepthBufferFormat");
- PixelFormatSearchStruct pfss;
- pfss.pddpf = pddpf;
- pfss.bFoundFormat = FALSE;
- pfss.preferredDepth = preferredDepth;
-
- d3dObject->EnumZBufferFormats(*pDeviceGUID,
- EnumZBufferFormatsCallback,
- (void*)&pfss);
-
- return pfss.bFoundFormat ? D3D_OK : DDERR_GENERIC;
-}
-
-jint D3DUtils_GetPixelFormatType(DDPIXELFORMAT*lpddpf)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "D3DUtils_GetPixelFormatType");
-
- if (lpddpf == NULL) return PF_INVALID;
-
- DWORD dwFlags = lpddpf->dwFlags;
- // skip weird formats
- if (lpddpf->dwRGBBitCount < 16 ||
- dwFlags & DDPF_ALPHA || dwFlags & DDPF_ZBUFFER ||
- dwFlags & DDPF_ZPIXELS || dwFlags & DDPF_LUMINANCE ||
- dwFlags & DDPF_FOURCC || dwFlags & DDPF_STENCILBUFFER ||
- dwFlags & DDPF_BUMPLUMINANCE || dwFlags & DDPF_BUMPDUDV)
- {
- return PF_INVALID;
- }
-
- jint pfType = PF_INVALID;
- DWORD aMask = lpddpf->dwRGBAlphaBitMask;
- DWORD rMask = lpddpf->dwRBitMask;
- DWORD gMask = lpddpf->dwGBitMask;
- DWORD bMask = lpddpf->dwBBitMask;
-
- if (rMask == 0x0000f800 &&
- gMask == 0x000007e0 &&
- bMask == 0x0000001f &&
- aMask == 0x00000000)
- {
- pfType = PF_USHORT_565_RGB;
- } else if (rMask == 0x00007C00 &&
- gMask == 0x000003E0 &&
- bMask == 0x0000001f &&
- aMask == 0x00000000)
- {
- pfType = PF_USHORT_555_RGB;
- } else if (rMask == 0x00000f00 &&
- gMask == 0x000000f0 &&
- bMask == 0x0000000f &&
- aMask == 0x0000f000)
- {
- // REMIND: we currently don't support this
- // pixel format, since we don't have the loops for a
- // premultiplied version of it. So we'll just use INT_ARGB
- // for now
- pfType = PF_INVALID;
- // pfType = PF_USHORT_4444_ARGB;
- } else if (rMask == 0x00ff0000 &&
- gMask == 0x0000ff00 &&
- bMask == 0x000000ff)
- {
- if (lpddpf->dwRGBBitCount == 32) {
- pfType = (dwFlags & DDPF_ALPHAPIXELS) ?
- PF_INT_ARGB : PF_INT_RGB;
- } else {
- // We currently don't support this format.
- // pfType = PF_3BYTE_BGR;
- pfType = PF_INVALID;
- }
- }
-
- return pfType;
-}
-
-void
-D3DUtils_SetupTextureFormats(IDirect3DDevice7 *d3dDevice,
- D3DTextureTable &table)
-{
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_SetupTextureFormats");
- if (d3dDevice == NULL || table == NULL) {
- return;
- }
-
- ZeroMemory(table, sizeof(D3DTextureTable));
- int t;
- for (t = TR_OPAQUE_IDX; t < TR_MAX_IDX; t++) {
- for (int d = DEPTH16_IDX; d < DEPTH_MAX_IDX; d++) {
- table[t][d].pfType = PF_INVALID;
- }
- }
- d3dDevice->EnumTextureFormats(D3DUtils_TextureSearchCallback, table);
-
- // We've retrieved the pixel formats for this device. The matrix may
- // look something like this, depending on the formats the device supports:
- // Transparency/Depth Depth 16 Depth 24 Depth 32
- // ------------------------------------------------------------------------
- // TR_OPAQUE PF_USHORT_565_RGB PF_INVALID PF_INT_RGB
- // TR_BITMASK PF_INVALID PF_INVALID PF_INVALID
- // TR_TRANSLUCENT PF_INVALID PF_INVALID PF_INT_ARGB
-
-
- // we'll be using translucent pixel formats for bitmask images
- // for now, this may change later
- memcpy(&table[TR_BITMASK_IDX], &table[TR_TRANSLUCENT_IDX],
- sizeof(D3DTextureTableCell[DEPTH_MAX_IDX]));
- // Transparency/Depth Depth 16 Depth 24 Depth 32
- // ------------------------------------------------------------------------
- // TR_OPAQUE PF_USHORT_565_RGB PF_INVALID PF_INT_RGB
- // TR_BITMASK PF_INVALID PF_INVALID PF_INT_ARGB
- // TR_TRANSLUCENT PF_INVALID PF_INVALID PF_INT_ARGB
-
- // REMIND: crude force
- // Find substitutes for pixel formats which we didn't find.
- // For example, if we didn't find a 24-bit format, 32-bit will be
- // a first choice for substitution. But if it wasn't found either,
- // then use 16-bit format
- D3DTextureTableCell *cell16, *cell24, *cell32;
- for (t = TR_OPAQUE_IDX; t < TR_MAX_IDX; t++) {
- cell16 = &table[t][DEPTH16_IDX];
- cell24 = &table[t][DEPTH24_IDX];
- cell32 = &table[t][DEPTH32_IDX];
- if (cell32->pfType == PF_INVALID) {
- if (cell24->pfType != PF_INVALID) {
- memcpy(cell32, cell24, sizeof(D3DTextureTableCell));
- } else if (cell16->pfType != PF_INVALID) {
- memcpy(cell32, cell16, sizeof(D3DTextureTableCell));
- } else {
- // no valid pixel formats for this transparency
- // type were found
- continue;
- }
- }
- // now we know that 32-bit is valid
- if (cell24->pfType == PF_INVALID) {
- // use 32-bit format as a substitution for 24-bit
- memcpy(cell24, cell32, sizeof(D3DTextureTableCell));
- }
- // now we know that 32- and 24-bit are valid
- if (cell16->pfType == PF_INVALID) {
- // use 24-bit format as a substitution for 16-bit
- memcpy(cell16, cell24, sizeof(D3DTextureTableCell));
- }
- }
- // After this loop the matrix may look something like this:
- // Transparency/Depth Depth 16 Depth 24 Depth 32
- // ------------------------------------------------------------------------
- // TR_OPAQUE PF_USHORT_565_RGB PF_INT_RGB PF_INT_RGB
- // TR_BITMASK PF_INT_ARGB PF_INT_ARGB PF_INT_ARGB
- // TR_TRANSLUCENT PF_INT_ARGB PF_INT_ARGB PF_INT_ARGB
-
-#ifdef DEBUG
- // Print out the matrix (should look something like the comment above)
- J2dTraceLn1(J2D_TRACE_INFO,
- "Texutre formats table for device %x", d3dDevice);
- J2dTraceLn(J2D_TRACE_INFO, "Transparency/Depth Depth 16 "\
- "Depth 24 Depth 32");
- J2dTraceLn(J2D_TRACE_INFO, "-------------------------------------------"\
- "----------------------------");
- for (t = TR_OPAQUE_IDX; t < TR_MAX_IDX; t++) {
- J2dTrace1(J2D_TRACE_INFO, "%15s", TR_NAMES[t]);
- for (int d = DEPTH16_IDX; d < DEPTH_MAX_IDX; d++) {
- J2dTrace1(J2D_TRACE_INFO, "%20s",
- PF_NAMES[table[t][d].pfType]);
- }
- J2dTrace(J2D_TRACE_INFO, "\n");
- }
-#endif // DEBUG
-}
-
-const GUID *
-D3DUtils_SelectDeviceGUID(IDirect3D7 *d3dObject)
-{
- static char * RASTERIZER_NAMES[] = {
- "TNL", "HAL", "REFERENCE", "RGB"
- };
- // try to use TnL rasterizer by default
- int defIndex = TNL_IDX;
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_SelectDeviceGUID");
- // unless a different one was requested
- char *pRasterizer = getenv("J2D_D3D_RASTERIZER");
- if (pRasterizer != NULL) {
- if (strncmp(pRasterizer, "ref", 3) == 0) {
- defIndex = REF_IDX;
- } else if (strncmp(pRasterizer, "rgb", 3) == 0) {
- defIndex = RGB_IDX;
- } else if (strncmp(pRasterizer, "hal", 3) == 0) {
- defIndex = HAL_IDX;
- } else if (strncmp(pRasterizer, "tnl", 3) == 0) {
- defIndex = TNL_IDX;
- }
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- " rasterizer requested: %s",
- RASTERIZER_NAMES[defIndex]);
- }
-
- DEVICES_INFO devInfo;
- memset(&devInfo, 0, sizeof(devInfo));
- HRESULT res;
- if (FAILED(res = d3dObject->EnumDevices(DeviceEnumCallback,
- (VOID*)&devInfo)))
- {
- DebugPrintDirectDrawError(res, "D3DUtils_SelectDeviceGUID: "\
- "EnumDevices failed");
- return NULL;
- }
-
- // return requested rasterizer's guid if it's present
- if (devInfo.pGUIDs[defIndex] != NULL) {
- J2dRlsTraceLn1(J2D_TRACE_VERBOSE,
- "D3DUtils_SelectDeviceGUID: using %s rasterizer",
- RASTERIZER_NAMES[defIndex]);
- return devInfo.pGUIDs[defIndex];
- }
- // if not, try to find one, starting with the best available
- defIndex = TNL_IDX;
- do {
- if (devInfo.pGUIDs[defIndex] != NULL) {
- J2dRlsTraceLn1(J2D_TRACE_VERBOSE,
- "D3DUtils_SelectDeviceGUID: using %s rasterizer",
- RASTERIZER_NAMES[defIndex]);
- return devInfo.pGUIDs[defIndex];
- }
- // While we could use the rgb and ref rasterizers if tnl and
- // hal aren't present, it's not practical for performance purposes.
- // so we just leave an opportunity to force them.
- } while (++defIndex < REF_IDX /*DEV_IDX_MAX*/);
-
-
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "D3DUtils_SelectDeviceGUID: "\
- "No Accelerated Rasterizers Found");
- return NULL;
-}
-
-
-/*
- * This function sets passed matrix to be a custom left-hand off-center
- * orthogonal matrix. The output is identical to D3DX's function call
- * D3DXMatrixOrthoOffCenterLH((D3DXMATRIX*)&tx,
- * 0.0, width, height, 0.0, -1.0, 1.0);
- */
-void
-D3DUtils_SetOrthoMatrixOffCenterLH(D3DMATRIX *m,
- float width, float height)
-{
- DASSERT((m != NULL) && (width > 0.0f) && (height > 0.0f));
- memset(m, 0, sizeof(D3DMATRIX));
- m->_11 = 2.0f/width;
- m->_22 = -2.0f/height;
- m->_33 = 0.5f;
- m->_44 = 1.0f;
-
- m->_41 = -1.0f;
- m->_42 = 1.0f;
- m->_43 = 0.5f;
-}
-
-void
-D3DUtils_SetIdentityMatrix(D3DMATRIX *m, BOOL adjust)
-{
- DASSERT(m != NULL);
- m->_12 = m->_13 = m->_14 = m->_21 = m->_23 = m->_24 = 0.0f;
- m->_31 = m->_32 = m->_34 = m->_43 = 0.0f;
- m->_11 = m->_22 = m->_33 = m->_44 = 1.0f;
- if (adjust) {
- // This is required for proper texel alignment
- m->_41 = m->_42 = -0.5f;
- } else {
- m->_41 = m->_42 = 0.0f;
- }
-}
-
-DDrawSurface *
-D3DUtils_CreatePlainSurface(JNIEnv *env,
- DDraw *ddObject,
- D3DContext *d3dContext,
- int w, int h)
-{
- DXSurface *dxSurface;
- jint pType;
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_CreatePlainSurface");
- if (FAILED(d3dContext->CreateSurface(env, w, h, 32,
- TR_OPAQUE, D3D_PLAIN_SURFACE,
- &dxSurface, &pType)))
- {
- return NULL;
- }
- return new DDrawSurface(ddObject, dxSurface);
-}
-
-DDrawSurface *
-D3DUtils_CreateTexture(JNIEnv *env,
- DDraw *ddObject,
- D3DContext *d3dContext,
- int transparency,
- int w, int h)
-{
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_CreateTexture");
- DXSurface *dxSurface;
- jint pType;
- if (FAILED(d3dContext->CreateSurface(env, w, h, 32,
- transparency, D3D_TEXTURE_SURFACE,
- &dxSurface, &pType)))
- {
- return NULL;
- }
- return new DDrawSurface(ddObject, dxSurface);
-}
-
-HRESULT
-D3DUtils_UploadIntImageToXRGBTexture(DDrawSurface *lpTexture,
- int *pSrc, int width, int height)
-{
- HRESULT res;
- int texW = lpTexture->GetDXSurface()->GetWidth();
- int texH = lpTexture->GetDXSurface()->GetHeight();
- int srcStride = width * 4;
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_UploadIntImageToXRGBTexture");
- if (width > texW) {
- width = texW;
- }
- if (height > texH) {
- height = texH;
- }
-
- SurfaceDataRasInfo rasInfo;
- if (SUCCEEDED(res = lpTexture->Lock(NULL, &rasInfo,
- DDLOCK_WAIT|DDLOCK_NOSYSLOCK, NULL)))
- {
- void *pDstPixels = rasInfo.rasBase;
- void *pSrcPixels = (void*)pSrc;
-
- // REMIND: clear the dest first
- memset(pDstPixels, 0, texH * rasInfo.scanStride);
- do {
- memcpy(pDstPixels, pSrcPixels, width * 4);
- pSrcPixels = PtrAddBytes(pSrcPixels, srcStride);
- pDstPixels = PtrAddBytes(pDstPixels, rasInfo.scanStride);
- } while (--height > 0);
- res = lpTexture->Unlock(NULL);
- }
- return res;
-}
-
-HRESULT
-D3DUtils_CheckD3DCaps(LPD3DDEVICEDESC7 lpDesc7)
-{
- // The device must support fast rasterization
- static DWORD dwDevCaps =
- (D3DDEVCAPS_DRAWPRIMTLVERTEX | D3DDEVCAPS_HWRASTERIZATION);
- BOOL vt = lpDesc7->dwDevCaps & D3DDEVCAPS_DRAWPRIMTLVERTEX;
- BOOL rz = lpDesc7->dwDevCaps & D3DDEVCAPS_HWRASTERIZATION;
-
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_CheckD3DCaps");
- return (lpDesc7->dwDevCaps & dwDevCaps) ?
- D3D_OK :
- DDERR_GENERIC;
-}
-
-HRESULT
-D3DUtils_CheckTextureCaps(LPD3DDEVICEDESC7 lpDesc7)
-{
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_CheckTextureCaps");
- // REMIND: we should really check both Tri and Lin caps,
- // but hopefully we won't be using line strips soon
- LPD3DPRIMCAPS lpDpcTriCaps = &lpDesc7->dpcTriCaps;
- // Filtering requirements
- static DWORD dwFilterCaps =
- (D3DPTFILTERCAPS_LINEAR | D3DPTFILTERCAPS_NEAREST);
- // Check for caps used for alpha compositing (implementation of
- // Porter-Duff rules)
- static DWORD dwBlendCaps =
- (D3DPBLENDCAPS_ZERO | D3DPBLENDCAPS_ONE |
- D3DPBLENDCAPS_SRCALPHA | D3DPBLENDCAPS_INVSRCALPHA |
- D3DPBLENDCAPS_DESTALPHA | D3DPBLENDCAPS_INVDESTALPHA);
-
- if ((lpDesc7->dwTextureOpCaps & D3DTEXOPCAPS_MODULATE) &&
- (lpDpcTriCaps->dwTextureFilterCaps & dwFilterCaps) &&
- (lpDpcTriCaps->dwSrcBlendCaps & dwBlendCaps) &&
- (lpDpcTriCaps->dwDestBlendCaps & dwBlendCaps))
- {
- return D3D_OK;
- }
- return DDERR_GENERIC;
-}
-
-HRESULT
-D3DUtils_CheckDeviceCaps(LPD3DDEVICEDESC7 lpDesc7) {
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_CheckDeviceCaps");
- if (SUCCEEDED(D3DUtils_CheckD3DCaps(lpDesc7)) &&
- SUCCEEDED(D3DUtils_CheckTextureCaps(lpDesc7)) &&
- SUCCEEDED(D3DUtils_CheckDepthCaps(lpDesc7)))
- {
- return D3D_OK;
- }
- return DDERR_GENERIC;
-}
-
-HRESULT
-D3DUtils_CheckDepthCaps(LPD3DDEVICEDESC7 lpDesc7)
-{
- J2dTraceLn(J2D_TRACE_INFO, "D3DUtils_CheckDepthCaps");
- // Check for required depth-buffer operations
- // (see D3DContext::SetClip() for more info).
- static DWORD dwZCmpCaps = (D3DPCMPCAPS_ALWAYS | D3DPCMPCAPS_LESS);
- // D3DPMISCCAPS_MASKZ capability allows enabling/disabling
- // depth buffer updates.
- if ((lpDesc7->dpcTriCaps.dwMiscCaps & D3DPMISCCAPS_MASKZ) &&
- (lpDesc7->dpcTriCaps.dwZCmpCaps & dwZCmpCaps))
- {
- return D3D_OK;
- }
- return DDERR_GENERIC;
-}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DUtils.h b/jdk/src/windows/native/sun/java2d/d3d/D3DUtils.h
deleted file mode 100644
index e246f8ba905..00000000000
--- a/jdk/src/windows/native/sun/java2d/d3d/D3DUtils.h
+++ /dev/null
@@ -1,287 +0,0 @@
-/*
- * Copyright 2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#ifndef D3DUTILS_H
-#define D3DUTILS_H
-
-#include "D3DContext.h"
-
-// - Types and macros used in SelectDeviceGUID -----------------------
-// Indexes for the rasterizers:
-// TNL, HAL, REF, RGB
-#define TNL_IDX (0)
-#define HAL_IDX (1)
-#define REF_IDX (2)
-#define RGB_IDX (3)
-#define DEV_IDX_MAX (RGB_IDX+1)
-
-typedef struct {
- const GUID *pGUIDs[4];
-} DEVICES_INFO;
-
-// - Utility funcions for dealing with pixel formats ----------------
-const GUID *
-D3DUtils_SelectDeviceGUID(IDirect3D7 *d3dObject);
-
-HRESULT
-D3DUtils_FindDepthBufferFormat(IDirect3D7 *d3dObject,
- int preferredDepth,
- DDPIXELFORMAT* pddpf,
- const GUID *pDeviceGUID);
-HRESULT
-D3DUtils_FindMaskTileTextureFormat(IDirect3DDevice7 *d3dDevice,
- DDPIXELFORMAT* pddpf);
-void
-D3DUtils_SetupTextureFormats(IDirect3DDevice7 *d3dDevice,
- D3DTextureTable &table);
-
-// - Utility funcions for working with matricies ---------------------
-void
-D3DUtils_SetIdentityMatrix(D3DMATRIX *m, BOOL adjust = TRUE);
-
-void
-D3DUtils_SetOrthoMatrixOffCenterLH(D3DMATRIX *m,
- float width, float height);
-DDrawSurface *
-D3DUtils_CreatePlainSurface(JNIEnv *env, DDraw *ddObject,
- D3DContext *d3dContext,
- int w, int h);
-
-DDrawSurface *
-D3DUtils_CreateTexture(JNIEnv *env, DDraw *ddObject,
- D3DContext *d3dContext,
- int transparency,
- int w, int h);
-
-HRESULT
-D3DUtils_UploadIntImageToXRGBTexture(DDrawSurface *lpTexture,
- int *pSrc, int width, int height);
-
-// - Utility functions for checking various capabilities of the device
-
-HRESULT
-D3DUtils_CheckD3DCaps(LPD3DDEVICEDESC7 lpDesc7);
-
-HRESULT
-D3DUtils_CheckDepthCaps(LPD3DDEVICEDESC7 lpDesc7);
-
-HRESULT
-D3DUtils_CheckTextureCaps(LPD3DDEVICEDESC7 lpDesc7);
-
-HRESULT
-D3DUtils_CheckDeviceCaps(LPD3DDEVICEDESC7 lpDesc7);
-
-// - Utility macros error handling of d3d operations -----------------
-
-/* #define NO_D3D_CHECKING */
-
-#ifdef NO_D3D_CHECKING
-
-#define D3DU_PRIM_LOOP_BEGIN(RES, DST_WSDO)
-#define D3DU_PRIM2_LOOP_BEGIN(RES, SRC_WSDO, DST_WSDO)
-#define D3DU_PRIM_LOOP_END(ENV, RES, DST_WSDO, PRIM)
-#define D3DU_PRIM2_LOOP_END(ENV, RES, SRC_WSDO, DST_WSDO, PRIM)
-
-#else /* NO_D3D_CHECKING */
-
-#ifndef MAX_BUSY_ATTEMPTS
- #define MAX_BUSY_ATTEMPTS 50 // Arbitrary number of times to attempt
-#endif
-
-
-#define D3DU_PRIM_LOOP_BEGIN(RES, DST_WSDO) \
-do { \
- int attempts = 0; \
- while (attempts++ < MAX_BUSY_ATTEMPTS) { \
- if (FAILED((DST_WSDO)->lpSurface->IsLost())) { \
- RES = DDERR_SURFACELOST; \
- } else {
-
-#define D3DU_PRIM2_LOOP_BEGIN(RES, SRC_WSDO, DST_WSDO) \
-do { \
- int attempts = 0; \
- while (attempts++ < MAX_BUSY_ATTEMPTS) { \
- if (FAILED((DST_WSDO)->lpSurface->IsLost()) || \
- FAILED((SRC_WSDO)->lpSurface->IsLost())) \
- { \
- RES = DDERR_SURFACELOST; \
- } else {
-
-#define D3DU_PRIM_LOOP_END(ENV, RES, DST_WSDO, PRIM) \
- } \
- if (SUCCEEDED(RES)) { \
- break; \
- } else if (RES == DDERR_SURFACEBUSY || RES == DDERR_WASSTILLDRAWING) { \
- J2dTraceLn(J2D_TRACE_VERBOSE, #PRIM ## ": surface is busy."); \
- continue; \
- } else if (RES == DDERR_SURFACELOST) { \
- J2dTraceLn(J2D_TRACE_INFO, #PRIM ## ": dest surface lost."); \
- DST_WSDO->RestoreSurface(ENV, DST_WSDO); \
- break; \
- } else { \
- DebugPrintDirectDrawError(RES, #PRIM); \
- } \
- } \
-} while (0)
-
-#define D3DU_PRIM2_LOOP_END(ENV, RES, SRC_WSDO, DST_WSDO, PRIM) \
- } \
- if (SUCCEEDED(RES)) { \
- break; \
- } else if (RES == DDERR_SURFACEBUSY || RES == DDERR_WASSTILLDRAWING) { \
- J2dTraceLn(J2D_TRACE_VERBOSE, #PRIM ## ": surface is busy."); \
- continue; \
- } else if (RES == DDERR_SURFACELOST) { \
- if (FAILED((DST_WSDO)->lpSurface->IsLost())) { \
- J2dTraceLn(J2D_TRACE_INFO, #PRIM ## ": dst surface lost."); \
- (DST_WSDO)->RestoreSurface(ENV, (DST_WSDO)); \
- } \
- if (FAILED((SRC_WSDO)->lpSurface->IsLost())) { \
- J2dTraceLn(J2D_TRACE_INFO, #PRIM ## ": src surface lost."); \
- (SRC_WSDO)->RestoreSurface(ENV, (SRC_WSDO)); \
- } \
- break; \
- } else { \
- DebugPrintDirectDrawError(RES, #PRIM); \
- } \
- } \
-} while (0)
-
-#endif /* NO_D3D_CHECKING */
-
-// - Utility macros for initializing vertex structures ---------------
-
-#define D3D_EXEC_PRIM_LOOP(ENV, RES, DST_WSDO, PRIM) \
- D3DU_PRIM_LOOP_BEGIN(RES, DST_WSDO); \
- RES = (PRIM); \
- D3DU_PRIM_LOOP_END(ENV, RES, DST_WSDO, PRIM);
-
-#define D3DU_INIT_VERTEX_PENT_XY(VQUAD, X1, Y1, X2, Y2) \
-do { \
- D3DU_INIT_VERTEX_QUAD_XY(VQUAD, X1, Y1, X2, Y2); \
- (VQUAD)[4].x = (X1); (VQUAD)[4].y = (Y1); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_PENT_COLOR(VQUAD, VCOLOR) \
-do { \
- D3DU_INIT_VERTEX_QUAD_COLOR(VQUAD, VCOLOR); \
- (VQUAD)[4].color = (VCOLOR); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_QUAD_XY(VQUAD, X1, Y1, X2, Y2) \
-do { \
- (VQUAD)[0].x = (X1); (VQUAD)[0].y = (Y1); \
- (VQUAD)[1].x = (X2); (VQUAD)[1].y = (Y1); \
- (VQUAD)[2].x = (X2); (VQUAD)[2].y = (Y2); \
- (VQUAD)[3].x = (X1); (VQUAD)[3].y = (Y2); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_QUAD_XYZ(VQUAD, X1, Y1, X2, Y2, Z) \
-do { \
- D3DU_INIT_VERTEX_QUAD_XY(VQUAD, X1, Y1, X2, Y2); \
- (VQUAD)[0].z = (Z); \
- (VQUAD)[1].z = (Z); \
- (VQUAD)[2].z = (Z); \
- (VQUAD)[3].z = (Z); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_QUAD_COLOR(VQUAD, VCOLOR) \
-do { \
- (VQUAD)[0].color = (VCOLOR); \
- (VQUAD)[1].color = (VCOLOR); \
- (VQUAD)[2].color = (VCOLOR); \
- (VQUAD)[3].color = (VCOLOR); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_QUAD_UV(VQUAD, TU1, TV1, TU2, TV2) \
-do { \
- (VQUAD)[0].tu = (TU1); (VQUAD)[0].tv = (TV1); \
- (VQUAD)[1].tu = (TU2); (VQUAD)[1].tv = (TV1); \
- (VQUAD)[2].tu = (TU2); (VQUAD)[2].tv = (TV2); \
- (VQUAD)[3].tu = (TU1); (VQUAD)[3].tv = (TV2); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_QUAD_XYUV(VQUAD, X1, Y1, X2, Y2, TU1, TV1, TU2, TV2) \
-do { \
- D3DU_INIT_VERTEX_QUAD_XY(VQUAD, X1, Y1, X2, Y2); \
- D3DU_INIT_VERTEX_QUAD_UV(VQUAD, TU1, TV1, TU2, TV2); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_QUAD(VQUAD, X1, Y1, X2, Y2, VCOLOR, TU1, TV1, TU2, TV2) \
-do { \
- D3DU_INIT_VERTEX_QUAD_XYUV(VQUAD, X1, Y1, X2, Y2, TU1, TV1, TU2, TV2); \
- D3DU_INIT_VERTEX_QUAD_COLOR(VQUAD, VCOLOR); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_6(VQUAD, X1, Y1, X2, Y2, VCOLOR, TU1, TV1, TU2, TV2) \
-do { \
- D3DU_INIT_VERTEX_XY_6(VHEX, X1, Y1, X2, Y2); \
- D3DU_INIT_VERTEX_UV_6(VHEX, TU1, TV1, TU2, TV2); \
- D3DU_INIT_VERTEX_COLOR_6(VHEX, VCOLOR); \
-} while (0)
-
-#define D3DU_INIT_VERTEX_UV_6(VHEX, TU1, TV1, TU2, TV2) \
-do { \
- (VHEX)[0].tu = TU1; (VHEX)[0].tv = TV1; \
- (VHEX)[1].tu = TU2; (VHEX)[1].tv = TV1; \
- (VHEX)[2].tu = TU1; (VHEX)[2].tv = TV2; \
- (VHEX)[3].tu = TU1; (VHEX)[3].tv = TV2; \
- (VHEX)[4].tu = TU2; (VHEX)[4].tv = TV1; \
- (VHEX)[5].tu = TU2; (VHEX)[5].tv = TV2; \
-} while (0)
-
-#define D3DU_INIT_VERTEX_COLOR_6(VHEX, VCOLOR) \
-do { \
- (VHEX)[0].color = VCOLOR; \
- (VHEX)[1].color = VCOLOR; \
- (VHEX)[2].color = VCOLOR; \
- (VHEX)[3].color = VCOLOR; \
- (VHEX)[4].color = VCOLOR; \
- (VHEX)[5].color = VCOLOR; \
-} while (0)
-
-#define D3DU_INIT_VERTEX_XY_6(VHEX, X1, Y1, X2, Y2) \
-do { \
- (VHEX)[0].x = X1; (VHEX)[0].y = Y1; \
- (VHEX)[1].x = X2; (VHEX)[1].y = Y1; \
- (VHEX)[2].x = X1; (VHEX)[2].y = Y2; \
- (VHEX)[3].x = X1; (VHEX)[3].y = Y2; \
- (VHEX)[4].x = X2; (VHEX)[4].y = Y1; \
- (VHEX)[5].x = X2; (VHEX)[5].y = Y2; \
-} while (0)
-
-#define D3DU_INIT_VERTEX_XYZ_6(VHEX, X1, Y1, X2, Y2, Z) \
-do { \
- D3DU_INIT_VERTEX_XY_6(VHEX, X1, Y1, X2, Y2); \
- (VHEX)[0].z = (Z); \
- (VHEX)[1].z = (Z); \
- (VHEX)[2].z = (Z); \
- (VHEX)[3].z = (Z); \
- (VHEX)[4].z = (Z); \
- (VHEX)[5].z = (Z); \
-} while (0)
-
-
-#endif // D3DUTILS_H
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DVertexCacher.cpp b/jdk/src/windows/native/sun/java2d/d3d/D3DVertexCacher.cpp
new file mode 100644
index 00000000000..6cf2a90b8ad
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DVertexCacher.cpp
@@ -0,0 +1,823 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#include "D3DPipeline.h"
+#include "D3DVertexCacher.h"
+#include "D3DPaints.h"
+
+#include "math.h"
+
+// non-texturized macros
+
+#define ADD_VERTEX_XYC(X, Y, VCOLOR) \
+do { \
+ vertices[firstUnusedVertex].x = (X); \
+ vertices[firstUnusedVertex].y = (Y); \
+ vertices[firstUnusedVertex].color = (DWORD)(VCOLOR); \
+ firstUnusedVertex++; \
+} while (0)
+
+#define ADD_LINE_XYC(X1, Y1, X2, Y2, VCOLOR) \
+do { \
+ ADD_VERTEX_XYC(X1, Y1, VCOLOR); \
+ ADD_VERTEX_XYC(X2, Y2, VCOLOR); \
+ batches[currentBatch].pNum++; \
+} while (0)
+
+#define ADD_LINE_SEG_XYC(X, Y, VCOLOR) \
+do { \
+ ADD_VERTEX_XYC(X, Y, VCOLOR); \
+ batches[currentBatch].pNum++; \
+} while (0)
+
+#define ADD_TRIANGLE_XYC(X1, Y1, X2, Y2, X3, Y3, VCOLOR) \
+do { \
+ ADD_VERTEX_XYC(X1, Y1, VCOLOR); \
+ ADD_VERTEX_XYC(X2, Y2, VCOLOR); \
+ ADD_VERTEX_XYC(X3, Y3, VCOLOR); \
+ batches[currentBatch].pNum++; \
+} while (0)
+
+// texturized macros
+
+#define ADD_VERTEX_XYUVC(X, Y, U1, V1, VCOLOR) \
+do { \
+ vertices[firstUnusedVertex].x = (X); \
+ vertices[firstUnusedVertex].y = (Y); \
+ vertices[firstUnusedVertex].tu1 = (U1); \
+ vertices[firstUnusedVertex].tv1 = (V1); \
+ vertices[firstUnusedVertex].color = (DWORD)(VCOLOR); \
+ firstUnusedVertex++; \
+} while (0)
+
+#define ADD_VERTEX_XYUVUVC(X, Y, U1, V1, U2, V2, VCOLOR) \
+do { \
+ vertices[firstUnusedVertex].tu2 = (U2); \
+ vertices[firstUnusedVertex].tv2 = (V2); \
+ ADD_VERTEX_XYUVC(X, Y, U1, V1, VCOLOR); \
+} while (0)
+
+#define ADD_TRIANGLE_XYUVC(X1, Y1, X2, Y2, X3, Y3, \
+ U1, V1, U2, V2, U3, V3, VCOLOR) \
+do { \
+ ADD_VERTEX_XYUVC(X1, Y1, U1, V1, VCOLOR); \
+ ADD_VERTEX_XYUVC(X2, Y2, U2, V2, VCOLOR); \
+ ADD_VERTEX_XYUVC(X3, Y3, U3, V3, VCOLOR); \
+ batches[currentBatch].pNum++; \
+} while (0)
+
+#define ADD_TRIANGLE_XYUVUVC(X1, Y1, X2, Y2, X3, Y3, \
+ U11, V11, U12, V12, U13, V13, \
+ U21, V21, U22, V22, U23, V23, \
+ VCOLOR) \
+do { \
+ ADD_VERTEX_XYUVUVC(X1, Y1, U11, V11, U21, V21, VCOLOR); \
+ ADD_VERTEX_XYUVUVC(X2, Y2, U12, V12, U22, V22, VCOLOR); \
+ ADD_VERTEX_XYUVUVC(X3, Y3, U13, V13, U23, V23, VCOLOR); \
+ batches[currentBatch].pNum++; \
+} while (0)
+
+// These are fudge factors for rendering lines found by experimenting.
+// They are used to tweak the geometry such that the rendering (mostly) matches
+// our software rendering on most hardware. The main goal was to pick the
+// numbers such that the beginning and ending pixels of lines match.
+#define LINE_FUDGE
+// fudge factors
+#ifdef LINE_FUDGE
+
+// Horiz/vertical
+#define HV_FF1 ( 0.0f)
+#define HV_FF2 ( 0.51f)
+// For the record: value below (or larger) is required for Intel 855, but
+// breaks Nvidia, ATI and Intel 965, and since the pipeline is disabled on
+// 855 anyway we'll use 0.51f.
+//#define HV_FF2 ( 0.5315f)
+#define HV_FF3 (-0.2f)
+// single pixel
+#define SP_FF4 ( 0.3f)
+
+// diagonal, down
+#define DD_FX1 (-0.1f)
+#define DD_FY1 (-0.25f)
+#define DD_FX2 ( 0.2f)
+#define DD_FY2 ( 0.304f)
+// For the record: with this value diagonal-down lines with Texture paint
+// are a bit off on all chipsets but Intel 965. So instead we'll use
+// .304f which makes it better for the rest, but at a price of a bit
+// of pixel/texel shifting on 965G
+//#define DD_FY2 ( 0.4f)
+// diagonal, up
+#define DU_FX1 (-0.1f)
+#define DU_FY1 ( 0.4f)
+#define DU_FX2 ( 0.3f)
+#define DU_FY2 (-0.3f)
+
+#else
+
+#define HV_FF1 (0.0f)
+#define HV_FF2 (0.0f)
+#define HV_FF3 (0.0f)
+#define SP_FF4 (0.0f)
+
+#define DD_FX1 (0.0f)
+#define DD_FY1 (0.0f)
+#define DD_FX2 (0.0f)
+#define DD_FY2 (0.0f)
+#define DU_FX1 (0.0f)
+#define DU_FY1 (0.0f)
+#define DU_FX2 (0.0f)
+#define DU_FY2 (0.0f)
+
+#endif
+
+HRESULT
+D3DVertexCacher::CreateInstance(D3DContext *pCtx, D3DVertexCacher **ppVC)
+{
+ HRESULT res;
+
+ J2dTraceLn(J2D_TRACE_INFO, "D3DVertexCacher::CreateInstance");
+
+ *ppVC = new D3DVertexCacher();
+ if (FAILED(res = (*ppVC)->Init(pCtx))) {
+ delete *ppVC;
+ *ppVC = NULL;
+ }
+ return res;
+}
+
+D3DVertexCacher::D3DVertexCacher()
+{
+ lpD3DDevice = NULL;
+ lpD3DVertexBuffer = NULL;
+}
+
+HRESULT
+D3DVertexCacher::Init(D3DContext *pCtx)
+{
+ D3DCAPS9 caps;
+
+ RETURN_STATUS_IF_NULL(pCtx, E_FAIL);
+
+ ReleaseDefPoolResources();
+
+ this->pCtx = pCtx;
+
+ firstPendingBatch = 0;
+ firstPendingVertex = 0;
+ firstUnusedVertex = 0;
+ currentBatch = 0;
+ ZeroMemory(vertices, sizeof(vertices));
+ ZeroMemory(batches, sizeof(batches));
+
+ lpD3DDevice = pCtx->Get3DDevice();
+ RETURN_STATUS_IF_NULL(lpD3DDevice, E_FAIL);
+
+ ZeroMemory(&caps, sizeof(caps));
+ lpD3DDevice->GetDeviceCaps(&caps);
+
+ D3DPOOL pool = (caps.DeviceType == D3DDEVTYPE_HAL) ?
+ D3DPOOL_DEFAULT : D3DPOOL_SYSTEMMEM;
+ // usage depends on whether we use hw or sw vertex processing
+ HRESULT res =
+ lpD3DDevice->CreateVertexBuffer(MAX_BATCH_SIZE*sizeof(J2DLVERTEX),
+ D3DUSAGE_DYNAMIC|D3DUSAGE_WRITEONLY, D3DFVF_J2DLVERTEX,
+ pool, &lpD3DVertexBuffer, NULL);
+ RETURN_STATUS_IF_FAILED(res);
+
+ res = lpD3DDevice->SetStreamSource(0, lpD3DVertexBuffer, 0,
+ sizeof(J2DLVERTEX));
+ RETURN_STATUS_IF_FAILED(res);
+
+ lpD3DDevice->SetFVF(D3DFVF_J2DLVERTEX);
+ return res;
+}
+
+void
+D3DVertexCacher::ReleaseDefPoolResources()
+{
+ SAFE_RELEASE(lpD3DVertexBuffer);
+ pCtx = NULL;
+}
+
+HRESULT D3DVertexCacher::DrawLine(int x1, int y1, int x2, int y2)
+{
+ HRESULT res;
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_LINELIST, 1*2))) {
+ float fx1, fy1, fx2, fy2;
+ if (y1 == y2) {
+ // horizontal
+ fy1 = (float)y1+HV_FF1;
+ fy2 = fy1;
+
+ if (x1 > x2) {
+ fx1 = (float)x2+HV_FF3;
+ fx2 = (float)x1+HV_FF2;
+ } else if (x1 < x2) {
+ fx1 = (float)x1+HV_FF3;
+ fx2 = (float)x2+HV_FF2;
+ } else {
+ // single point, offset a little so that a single
+ // pixel is rendered
+ fx1 = (float)x1-SP_FF4;
+ fy1 = (float)y1-SP_FF4;
+ fx2 = (float)x2+SP_FF4;
+ fy2 = (float)y2+SP_FF4;
+ }
+ } else if (x1 == x2) {
+ // vertical
+ fx1 = (float)x1+HV_FF1;
+ fx2 = fx1;
+ if (y1 > y2) {
+ fy1 = (float)y2+HV_FF3;
+ fy2 = (float)y1+HV_FF2;
+ } else {
+ fy1 = (float)y1+HV_FF3;
+ fy2 = (float)y2+HV_FF2;
+ }
+ } else {
+ // diagonal
+ if (x1 > x2 && y1 > y2) {
+ // ^
+ // \ case -> inverse
+ fx1 = (float)x2;
+ fy1 = (float)y2;
+ fx2 = (float)x1;
+ fy2 = (float)y1;
+ } else if (x1 > x2 && y2 > y1) {
+ // /
+ // v case - inverse
+ fx1 = (float)x2;
+ fy1 = (float)y2;
+ fx2 = (float)x1;
+ fy2 = (float)y1;
+ } else {
+ // \ ^
+ // v or / - leave as is
+ fx1 = (float)x1;
+ fy1 = (float)y1;
+ fx2 = (float)x2;
+ fy2 = (float)y2;
+ }
+
+ if (fx2 > fx1 && fy2 > fy1) {
+ // \
+ // v
+ fx1 += DD_FX1;
+ fy1 += DD_FY1;
+ fx2 += DD_FX2;
+ fy2 += DD_FY2;
+ } else {
+ // ^
+ // /
+ fx1 += DU_FX1;
+ fy1 += DU_FY1;
+ fx2 += DU_FX2;
+ fy2 += DU_FY2;
+ }
+ }
+ ADD_LINE_XYC(fx1, fy1, fx2, fy2, color);
+ }
+ return res;
+}
+
+HRESULT
+D3DVertexCacher::DrawPoly(jint nPoints, jboolean isClosed,
+ jint transX, jint transY,
+ jint *xPoints, jint *yPoints)
+{
+ HRESULT res;
+ jfloat mx = (jfloat)xPoints[0];
+ jfloat my = (jfloat)yPoints[0];
+ jboolean isEmpty = TRUE;
+
+ if (nPoints == 0) {
+ return S_OK;
+ }
+
+ if (isClosed &&
+ xPoints[nPoints - 1] == xPoints[0] &&
+ yPoints[nPoints - 1] == yPoints[0])
+ {
+ isClosed = FALSE;
+ }
+
+ // npoints is exactly the number of vertices we need,
+ // possibly plus one (if the path is closed)
+ UINT reqVerts = nPoints * 1;
+ int i = 0;
+ do {
+ // leave room for one possible additional closing point
+ UINT vertsInBatch = min(MAX_BATCH_SIZE-1, max(2, reqVerts));
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_LINESTRIP, vertsInBatch+1))) {
+ reqVerts -= vertsInBatch;
+ do {
+ jfloat x = (jfloat)xPoints[i];
+ jfloat y = (jfloat)yPoints[i];
+
+ isEmpty = isEmpty && (x == mx && y == my);
+
+ ADD_LINE_SEG_XYC(x + transX, y + transY, color);
+ i++;
+ vertsInBatch--;
+ } while (vertsInBatch > 0);
+ // include the last point from the current batch into the next
+ if (reqVerts > 0) {
+ i--;
+ reqVerts++;
+ // loop continues
+ } else if (isClosed && !isEmpty) {
+ // if this was the last batch, see if the closing point is needed;
+ // note that we have left the room for it
+ ADD_LINE_SEG_XYC(mx + transX, my + transY, color);
+ // for clarity, the loop is ended anyway
+ break;
+ } else if (isEmpty || !isClosed) {
+ // - either we went nowhere, then change the last point
+ // so that a single pixel is rendered
+ // - or it's not empty and not closed - add another
+ // point because on some boards the last point is not rendered
+ mx = xPoints[nPoints-1] + transX +SP_FF4;
+ my = yPoints[nPoints-1] + transY +SP_FF4;
+ ADD_LINE_SEG_XYC(mx, my, color);
+ // for clarity
+ break;
+ }
+ }
+ } while (reqVerts > 0 && SUCCEEDED(res));
+
+ return res;
+}
+
+HRESULT
+D3DVertexCacher::DrawScanlines(jint scanlineCount, jint *scanlines)
+{
+ HRESULT res;
+ float x1, x2, y;
+ UINT reqVerts = scanlineCount*2/*vertices per line*/;
+
+ if (scanlineCount == 0) {
+ return S_OK;
+ }
+
+ do {
+ UINT vertsInBatch = min(2*(MAX_BATCH_SIZE/2), reqVerts);
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_LINELIST, vertsInBatch))) {
+ reqVerts -= vertsInBatch;
+ do {
+ x1 = ((float)*(scanlines++)) +HV_FF3;
+ x2 = ((float)*(scanlines++)) +HV_FF2;
+ y = ((float)*(scanlines++)) +HV_FF1;
+ ADD_LINE_XYC(x1, y, x2, y, color);
+ vertsInBatch -= 2;
+ } while (vertsInBatch > 0);
+ }
+ } while (reqVerts > 0 && SUCCEEDED(res));
+ return res;
+}
+
+HRESULT
+D3DVertexCacher::FillSpans(jint spanCount, jint *spans)
+{
+ HRESULT res;
+ float x1, y1, x2, y2;
+ UINT reqVerts = spanCount*2*3/*vertices per span: two triangles*/;
+
+ if (spanCount == 0) {
+ return S_OK;
+ }
+
+ do {
+ UINT vertsInBatch = min(6*(MAX_BATCH_SIZE/6), reqVerts);
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, vertsInBatch))) {
+ reqVerts -= vertsInBatch;
+ do {
+ x1 = ((float)*(spans++));
+ y1 = ((float)*(spans++));
+ x2 = ((float)*(spans++));
+ y2 = ((float)*(spans++));
+
+ ADD_TRIANGLE_XYC(x1, y1, x2, y1, x1, y2, color);
+ ADD_TRIANGLE_XYC(x1, y2, x2, y1, x2, y2, color);
+ vertsInBatch -= 6;
+ } while (vertsInBatch > 0);
+ }
+ } while (reqVerts > 0 && SUCCEEDED(res));
+
+ return res;
+}
+
+HRESULT D3DVertexCacher::DrawRect(int x1, int y1, int x2, int y2)
+{
+ HRESULT res;
+
+ if ((x2 - x1) < 2 || (y2 - y1) < 2) {
+ return FillRect(x1, y1, x2+1, y2+1);
+ }
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_LINELIST, 4*2))) {
+
+ float fx1 = (float)x1;
+ float fy1 = (float)y1;
+ float fx2 = (float)x2;
+ float fy2 = (float)y2;
+
+ // horiz: top left - top right
+ ADD_LINE_XYC(fx1+HV_FF3, fy1+HV_FF1, fx2-1.0f+HV_FF2, fy1+HV_FF1,color);
+ // horiz: bottom left - bottom right
+ ADD_LINE_XYC(fx1+1.0f+HV_FF3, fy2+HV_FF1, fx2+HV_FF2, fy2+HV_FF1,color);
+ // vert : top right - bottom right
+ ADD_LINE_XYC(fx2+HV_FF1, fy1+HV_FF3, fx2+HV_FF1, fy2-1.0f+HV_FF2,color);
+ // vert : top left - bottom left
+ ADD_LINE_XYC(fx1+HV_FF1, fy1+1.0f+HV_FF3, fx1+HV_FF1, fy2+HV_FF2,color);
+ }
+ return res;
+}
+
+HRESULT D3DVertexCacher::FillRect(int x1, int y1, int x2, int y2)
+{
+ HRESULT res;
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, 2*3))) {
+ float fx1 = (float)x1;
+ float fy1 = (float)y1;
+ float fx2 = (float)x2;
+ float fy2 = (float)y2;
+ ADD_TRIANGLE_XYUVC(fx1, fy1, fx2, fy1, fx1, fy2,
+ 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f,
+ color);
+ ADD_TRIANGLE_XYUVC(fx1, fy2, fx2, fy1, fx2, fy2,
+ 0.0f, 1.0f, 1.0f, 0.0f, 1.0f, 1.0f,
+ color);
+ }
+ return res;
+}
+
+HRESULT D3DVertexCacher::FillParallelogram(float fx11, float fy11,
+ float dx21, float dy21,
+ float dx12, float dy12)
+{
+ HRESULT res;
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, 2*3))) {
+ // correct texel to pixel mapping; see D3DContext::SetTransform()
+ // for non-id tx case
+ if (pCtx->IsIdentityTx()) {
+ fx11 -= 0.5f;
+ fy11 -= 0.5f;
+ }
+ dx21 += fx11;
+ dy21 += fy11;
+ float fx22 = dx21 + dx12;
+ float fy22 = dy21 + dy12;
+ dx12 += fx11;
+ dy12 += fy11;
+
+ ADD_TRIANGLE_XYUVC(fx11, fy11, dx21, dy21, dx12, dy12,
+ 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f,
+ color);
+ ADD_TRIANGLE_XYUVC(dx12, dy12, dx21, dy21, fx22, fy22,
+ 0.0f, 1.0f, 1.0f, 0.0f, 1.0f, 1.0f,
+ color);
+ }
+ return res;
+}
+
+#define ADJUST_PGRAM(V, DV, DIM) \
+ do { \
+ if ((DV) >= 0) { \
+ (DIM) += (DV); \
+ } else { \
+ (DIM) -= (DV); \
+ (V) += (DV); \
+ } \
+ } while (0)
+
+// Invert the following transform:
+// DeltaT(0, 0) == (0, 0)
+// DeltaT(1, 0) == (DX1, DY1)
+// DeltaT(0, 1) == (DX2, DY2)
+// DeltaT(1, 1) == (DX1+DX2, DY1+DY2)
+// TM00 = DX1, TM01 = DX2, (TM02 = X11)
+// TM10 = DY1, TM11 = DY2, (TM12 = Y11)
+// Determinant = TM00*TM11 - TM01*TM10
+// = DX1*DY2 - DX2*DY1
+// Inverse is:
+// IM00 = TM11/det, IM01 = -TM01/det
+// IM10 = -TM10/det, IM11 = TM00/det
+// IM02 = (TM01 * TM12 - TM11 * TM02) / det,
+// IM12 = (TM10 * TM02 - TM00 * TM12) / det,
+
+#define DECLARE_MATRIX(MAT) \
+ float MAT ## 00, MAT ## 01, MAT ## 02, MAT ## 10, MAT ## 11, MAT ## 12
+
+#define GET_INVERTED_MATRIX(MAT, X11, Y11, DX1, DY1, DX2, DY2, RET_CODE) \
+ do { \
+ float det = DX1*DY2 - DX2*DY1; \
+ if (det == 0) { \
+ RET_CODE; \
+ } \
+ MAT ## 00 = DY2/det; \
+ MAT ## 01 = -DX2/det; \
+ MAT ## 10 = -DY1/det; \
+ MAT ## 11 = DX1/det; \
+ MAT ## 02 = (DX2 * Y11 - DY2 * X11) / det; \
+ MAT ## 12 = (DY1 * X11 - DX1 * Y11) / det; \
+ } while (0)
+
+#define TRANSFORM(MAT, TX, TY, X, Y) \
+ do { \
+ TX = (X) * MAT ## 00 + (Y) * MAT ## 01 + MAT ## 02; \
+ TY = (X) * MAT ## 10 + (Y) * MAT ## 11 + MAT ## 12; \
+ } while (0)
+
+HRESULT D3DVertexCacher::FillParallelogramAA(float fx11, float fy11,
+ float dx21, float dy21,
+ float dx12, float dy12)
+{
+ HRESULT res;
+ DECLARE_MATRIX(om);
+
+ GET_INVERTED_MATRIX(om, fx11, fy11, dx21, dy21, dx12, dy12,
+ return D3D_OK);
+
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, 2*3))) {
+ float px = fx11, py = fy11;
+ float pw = 0.0f, ph = 0.0f;
+ ADJUST_PGRAM(px, dx21, pw);
+ ADJUST_PGRAM(py, dy21, ph);
+ ADJUST_PGRAM(px, dx12, pw);
+ ADJUST_PGRAM(py, dy12, ph);
+ float px1 = floor(px);
+ float py1 = floor(py);
+ float px2 = ceil(px + pw);
+ float py2 = ceil(py + ph);
+ float u11, v11, u12, v12, u21, v21, u22, v22;
+ TRANSFORM(om, u11, v11, px1, py1);
+ TRANSFORM(om, u21, v21, px2, py1);
+ TRANSFORM(om, u12, v12, px1, py2);
+ TRANSFORM(om, u22, v22, px2, py2);
+ ADD_TRIANGLE_XYUVUVC(px1, py1, px2, py1, px1, py2,
+ u11, v11, u21, v21, u12, v12,
+ 5.0, 5.0, 6.0, 5.0, 5.0, 6.0,
+ color);
+ ADD_TRIANGLE_XYUVUVC(px1, py2, px2, py1, px2, py2,
+ u12, v12, u21, v21, u22, v22,
+ 5.0, 6.0, 6.0, 5.0, 6.0, 6.0,
+ color);
+ }
+ return res;
+}
+
+HRESULT D3DVertexCacher::DrawParallelogramAA(float ox11, float oy11,
+ float ox21, float oy21,
+ float ox12, float oy12,
+ float ix11, float iy11,
+ float ix21, float iy21,
+ float ix12, float iy12)
+{
+ HRESULT res;
+ DECLARE_MATRIX(om);
+ DECLARE_MATRIX(im);
+
+ GET_INVERTED_MATRIX(im, ix11, iy11, ix21, iy21, ix12, iy12,
+ // inner parallelogram is degenerate
+ // therefore it encloses no area
+ // fill outer
+ return FillParallelogramAA(ox11, oy11,
+ ox21, oy21,
+ ox12, oy12));
+ GET_INVERTED_MATRIX(om, ox11, oy11, ox21, oy21, ox12, oy12,
+ return D3D_OK);
+
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, 2*3))) {
+ float ox = ox11, oy = oy11;
+ float ow = 0.0f, oh = 0.0f;
+ ADJUST_PGRAM(ox, ox21, ow);
+ ADJUST_PGRAM(oy, oy21, oh);
+ ADJUST_PGRAM(ox, ox12, ow);
+ ADJUST_PGRAM(oy, oy12, oh);
+ float ox11 = floor(ox);
+ float oy11 = floor(oy);
+ float ox22 = ceil(ox + ow);
+ float oy22 = ceil(oy + oh);
+ float ou11, ov11, ou12, ov12, ou21, ov21, ou22, ov22;
+ TRANSFORM(om, ou11, ov11, ox11, oy11);
+ TRANSFORM(om, ou21, ov21, ox22, oy11);
+ TRANSFORM(om, ou12, ov12, ox11, oy22);
+ TRANSFORM(om, ou22, ov22, ox22, oy22);
+ float iu11, iv11, iu12, iv12, iu21, iv21, iu22, iv22;
+ TRANSFORM(im, iu11, iv11, ox11, oy11);
+ TRANSFORM(im, iu21, iv21, ox22, oy11);
+ TRANSFORM(im, iu12, iv12, ox11, oy22);
+ TRANSFORM(im, iu22, iv22, ox22, oy22);
+ ADD_TRIANGLE_XYUVUVC(ox11, oy11, ox22, oy11, ox11, oy22,
+ ou11, ov11, ou21, ov21, ou12, ov12,
+ iu11, iv11, iu21, iv21, iu12, iv12,
+ color);
+ ADD_TRIANGLE_XYUVUVC(ox11, oy22, ox22, oy11, ox22, oy22,
+ ou12, ov12, ou21, ov21, ou22, ov22,
+ iu12, iv12, iu21, iv21, iu22, iv22,
+ color);
+ }
+ return res;
+}
+
+HRESULT
+D3DVertexCacher::DrawTexture(float x1, float y1, float x2, float y2,
+ float u1, float v1, float u2, float v2)
+{
+ HRESULT res;
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, 2*3))) {
+ // correct texel to pixel mapping; see D3DContext::SetTransform()
+ // for non-id tx case
+ if (pCtx->IsIdentityTx()) {
+ x1 -= 0.5f;
+ y1 -= 0.5f;
+ x2 -= 0.5f;
+ y2 -= 0.5f;
+ }
+
+ ADD_TRIANGLE_XYUVC(x1, y1, x2, y1, x1, y2,
+ u1, v1, u2, v1, u1, v2,
+ color);
+ ADD_TRIANGLE_XYUVC(x1, y2, x2, y1, x2, y2,
+ u1, v2, u2, v1, u2, v2,
+ color);
+ }
+ return res;
+}
+
+HRESULT
+D3DVertexCacher::DrawTexture(float x1, float y1, float x2, float y2,
+ float u11, float v11, float u12, float v12,
+ float u21, float v21, float u22, float v22)
+{
+ HRESULT res;
+ if (SUCCEEDED(res = EnsureCapacity(D3DPT_TRIANGLELIST, 2*3))) {
+ // correct texel to pixel mapping; see D3DContext::SetTransform()
+ // for non-id tx case
+ if (pCtx->IsIdentityTx()) {
+ x1 -= 0.5f;
+ y1 -= 0.5f;
+ x2 -= 0.5f;
+ y2 -= 0.5f;
+ }
+
+ ADD_TRIANGLE_XYUVUVC(x1, y1, x2, y1, x1, y2,
+ u11, v11, u12, v11, u11, v12,
+ u21, v21, u22, v21, u21, v22,
+ color);
+ ADD_TRIANGLE_XYUVUVC(x1, y2, x2, y1, x2, y2,
+ u11, v12, u12, v11, u12, v12,
+ u21, v22, u22, v21, u22, v22,
+ color);
+ }
+ return res;
+}
+
+HRESULT D3DVertexCacher::Render(int actionType)
+{
+ J2DLVERTEX *lpVert;
+ HRESULT res;
+ DWORD dwLockFlags;
+ UINT pendingVertices = firstUnusedVertex - firstPendingVertex;
+
+ // nothing to render
+ if (pendingVertices == 0) {
+ if (actionType == RESET_ACTION) {
+ firstPendingBatch = 0;
+ firstPendingVertex = 0;
+ firstUnusedVertex = 0;
+ currentBatch = 0;
+ }
+ return D3D_OK;
+ }
+
+ if (firstPendingVertex == 0) {
+ // no data in the buffer yet, we don't care about
+ // vertex buffer's contents
+ dwLockFlags = D3DLOCK_DISCARD;
+ } else {
+ // append to the existing data in the vertex buffer
+ dwLockFlags = D3DLOCK_NOOVERWRITE;
+ }
+
+ if (SUCCEEDED(res =
+ lpD3DVertexBuffer->Lock((UINT)firstPendingVertex*sizeof(J2DLVERTEX),
+ (UINT)pendingVertices*sizeof(J2DLVERTEX),
+ (void**)&lpVert, dwLockFlags)))
+ {
+ // copy only new vertices
+ memcpy((void *)lpVert,
+ (void *)(vertices + firstPendingVertex),
+ pendingVertices * sizeof(J2DLVERTEX));
+ res = lpD3DVertexBuffer->Unlock();
+ UINT currentVertex = firstPendingVertex;
+ UINT batchSize;
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ "D3DVC::Render Starting flushing of %d vertices "\
+ "in %d batches",
+ pendingVertices,
+ (currentBatch - firstPendingBatch + 1));
+
+
+ for (UINT b = firstPendingBatch; b <= currentBatch; b++) {
+ D3DPRIMITIVETYPE pType = batches[b].pType;
+ UINT primCount = batches[b].pNum;
+ switch (pType) {
+ // the macro for adding a line segment adds one too many prims
+ case D3DPT_LINESTRIP: batchSize = primCount; primCount--; break;
+ case D3DPT_LINELIST: batchSize = primCount*2; break;
+ default: batchSize = primCount*3; break;
+ }
+ res = lpD3DDevice->DrawPrimitive(pType, currentVertex, primCount);
+ currentVertex += batchSize;
+ // init to something it can never be
+ batches[b].pType = (D3DPRIMITIVETYPE)0;
+ batches[b].pNum = 0;
+ }
+ } else {
+ DebugPrintD3DError(res, "Can't lock vertex buffer");
+ }
+
+ // REMIND: may need to rethink what to do in case of an error,
+ // should we try to render them later?
+ if (actionType == RESET_ACTION) {
+ firstPendingBatch = 0;
+ firstPendingVertex = 0;
+ firstUnusedVertex = 0;
+ currentBatch = 0;
+ } else {
+ firstPendingBatch = currentBatch;
+ firstPendingVertex = firstUnusedVertex;
+ }
+
+ return res;
+}
+
+HRESULT D3DVertexCacher::EnsureCapacity(D3DPRIMITIVETYPE newPType, UINT vNum)
+{
+ HRESULT res = D3D_OK;
+ if (vNum > MAX_BATCH_SIZE) {
+ // REMIND: need to define our own errors
+ return D3DERR_NOTAVAILABLE;
+ }
+ if ((firstUnusedVertex + vNum) > MAX_BATCH_SIZE) {
+ // if we can't fit new vertices in the vertex buffer,
+ // render whatever we have in the buffer and start
+ // from the beginning of the vertex buffer
+ J2dTraceLn2(J2D_TRACE_VERBOSE,
+ "D3DVC::EnsureCapacity exceeded capacity. "\
+ "current v: %d, requested vertices: %d\n",
+ firstUnusedVertex, vNum);
+ if (FAILED(res = Render(RESET_ACTION))) {
+ return res;
+ }
+ }
+
+ J2dTraceLn5(J2D_TRACE_VERBOSE,
+ "D3DVC::EnsureCapacity current batch: %d "\
+ " batch.type=%d newType=%d vNum=%d firstUnusedV=%d",
+ currentBatch, batches[currentBatch].pType, newPType, vNum,
+ firstUnusedVertex);
+ // there should not be multiple linestrips in a batch,
+ // or they will be counted as a single line strip
+ if (batches[currentBatch].pType != newPType ||
+ batches[currentBatch].pType == D3DPT_LINESTRIP)
+ {
+ // if this is a first unused batch, use it
+ if (firstUnusedVertex == firstPendingVertex) {
+ // record the first batch and vertex scheduled for rendering
+ firstPendingBatch = currentBatch;
+ firstPendingVertex = firstUnusedVertex;
+ } else {
+ // otherwise go to the next batch
+ currentBatch++;
+ }
+ batches[currentBatch].pType = newPType;
+ batches[currentBatch].pNum = 0;
+ }
+ // firstUnusedVertex is updated when new vertices are added
+ // to the vertices array
+
+ return res;
+}
diff --git a/jdk/src/windows/native/sun/java2d/d3d/D3DVertexCacher.h b/jdk/src/windows/native/sun/java2d/d3d/D3DVertexCacher.h
new file mode 100644
index 00000000000..a3872e36145
--- /dev/null
+++ b/jdk/src/windows/native/sun/java2d/d3d/D3DVertexCacher.h
@@ -0,0 +1,108 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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. Sun designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+#ifndef D3DVERTEXCACHER_H
+#define D3DVERTEXCACHER_H
+
+#include "jni.h"
+#include "D3DContext.h"
+
+#define MAX_BATCH_SIZE 1024
+#define APPEND_ACTION 0x0
+#define RESET_ACTION 0x1
+#define D3DFVF_J2DLVERTEX \
+ (D3DFVF_XYZ | D3DFVF_DIFFUSE | D3DFVF_TEX2 | \
+ D3DFVF_TEXCOORDSIZE2(0) | D3DFVF_TEXCOORDSIZE2(1) )
+typedef struct _J2DLVERTEX {
+ float x, y, z;
+ DWORD color;
+ float tu1, tv1;
+ float tu2, tv2;
+} J2DLVERTEX;
+
+typedef struct {
+ D3DPRIMITIVETYPE pType; // type of primitives in this batch
+ UINT pNum; // number of primitives of pType in this batch
+} VertexBatch;
+
+class D3DContext;
+
+class D3DPIPELINE_API D3DVertexCacher {
+public:
+ HRESULT Init(D3DContext *pCtx);
+ ~D3DVertexCacher() { ReleaseDefPoolResources(); }
+ void ReleaseDefPoolResources();
+
+ jint GetColor() { return color; }
+ void SetColor(jint newColor) { color = newColor; }
+ HRESULT DrawLine(int x1, int y1, int x2, int y2);
+ HRESULT DrawPoly(jint nPoints, jboolean isClosed,
+ jint transX, jint transY,
+ jint *xPoints, jint *yPoints);
+ HRESULT DrawScanlines(jint scanlineCount, jint *scanlines);
+ HRESULT DrawRect(int x1, int y1, int x2, int y2);
+ HRESULT FillRect(int x1, int y1, int x2, int y2);
+ HRESULT FillParallelogramAA(float fx11, float fy11,
+ float dx21, float dy21,
+ float dx12, float dy12);
+ HRESULT DrawParallelogramAA(float ox11, float oy11,
+ float ox21, float oy21,
+ float ox12, float oy12,
+ float ix11, float iy11,
+ float ix21, float iy21,
+ float ix12, float iy12);
+ HRESULT FillParallelogram(float fx11, float fy11,
+ float dx21, float dy21,
+ float dx12, float dy12);
+ HRESULT FillSpans(jint spansCount, jint *spans);
+ HRESULT DrawTexture(float dx1, float dy1, float dx2, float dy2,
+ float tx1, float ty1, float tx2, float ty2);
+ HRESULT DrawTexture(float dx1, float dy1, float dx2, float dy2,
+ float t1x1, float t1y1, float t1x2, float t1y2,
+ float t2x1, float t2y1, float t2x2, float t2y2);
+ HRESULT Render(int actionType = APPEND_ACTION);
+ UINT GetFreeVertices() { return (MAX_BATCH_SIZE - firstUnusedVertex); }
+
+static
+ HRESULT CreateInstance(D3DContext *pCtx, D3DVertexCacher **ppVC);
+
+private:
+ D3DVertexCacher();
+ HRESULT EnsureCapacity(D3DPRIMITIVETYPE newPType, UINT vNum);
+
+private:
+ UINT firstPendingBatch;
+ UINT firstPendingVertex;
+ UINT firstUnusedVertex;
+ UINT currentBatch;
+ J2DLVERTEX vertices[MAX_BATCH_SIZE];
+ VertexBatch batches[MAX_BATCH_SIZE];
+ IDirect3DVertexBuffer9 *lpD3DVertexBuffer;
+ IDirect3DDevice9 *lpD3DDevice;
+ D3DContext *pCtx;
+ jint color;
+};
+
+#endif // D3DVERTEXCACHER_H
diff --git a/jdk/src/windows/native/sun/java2d/opengl/WGLGraphicsConfig.c b/jdk/src/windows/native/sun/java2d/opengl/WGLGraphicsConfig.c
index ee914db32de..3e05a65eab1 100644
--- a/jdk/src/windows/native/sun/java2d/opengl/WGLGraphicsConfig.c
+++ b/jdk/src/windows/native/sun/java2d/opengl/WGLGraphicsConfig.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -501,7 +501,7 @@ Java_sun_java2d_opengl_WGLGraphicsConfig_getWGLConfigInfo(JNIEnv *env,
WGLGraphicsConfigInfo *wglinfo;
const unsigned char *versionstr;
const char *extstr;
- jint caps = sun_java2d_opengl_OGLContext_CAPS_EMPTY;
+ jint caps = CAPS_EMPTY;
int attrKeys[] = { WGL_DOUBLE_BUFFER_ARB, WGL_ALPHA_BITS_ARB };
int attrVals[2];
@@ -626,10 +626,10 @@ Java_sun_java2d_opengl_WGLGraphicsConfig_getWGLConfigInfo(JNIEnv *env,
// get config-specific capabilities
j2d_wglGetPixelFormatAttribivARB(hdc, pixfmt, 0, 2, attrKeys, attrVals);
if (attrVals[0]) {
- caps |= sun_java2d_opengl_OGLContext_CAPS_DOUBLEBUFFERED;
+ caps |= CAPS_DOUBLEBUFFERED;
}
if (attrVals[1] > 0) {
- caps |= sun_java2d_opengl_OGLContext_CAPS_STORED_ALPHA;
+ caps |= CAPS_STORED_ALPHA;
}
// create the scratch pbuffer
@@ -712,7 +712,7 @@ Java_sun_java2d_opengl_WGLGraphicsConfig_getOGLCapabilities(JNIEnv *env,
J2dTraceLn(J2D_TRACE_INFO, "WGLGraphicsConfig_getOGLCapabilities");
if (wglinfo == NULL || wglinfo->context == NULL) {
- return sun_java2d_opengl_OGLContext_CAPS_EMPTY;
+ return CAPS_EMPTY;
}
return wglinfo->context->caps;
diff --git a/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.c b/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.c
index b34872aaf7c..71d5e876406 100644
--- a/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.c
+++ b/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.c
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -44,11 +44,21 @@ extern GetRasInfoFunc OGLSD_GetRasInfo;
extern UnlockFunc OGLSD_Unlock;
extern DisposeFunc OGLSD_Dispose;
+extern OGLPixelFormat PixelFormats[];
+extern void AwtWindow_UpdateWindow(JNIEnv *env, jobject peer,
+ jint w, jint h, HBITMAP hBitmap);
+extern HBITMAP BitmapUtil_CreateBitmapFromARGBPre(int width, int height,
+ int srcStride,
+ int* imageData);
+extern void AwtComponent_GetInsets(JNIEnv *env, jobject peer, RECT *insets);
+
+extern void
+ OGLSD_SetNativeDimensions(JNIEnv *env, OGLSDOps *oglsdo, jint w, jint h);
+
JNIEXPORT void JNICALL
Java_sun_java2d_opengl_WGLSurfaceData_initOps(JNIEnv *env, jobject wglsd,
jlong pConfigInfo,
- jlong pPeerData,
- jint xoff, jint yoff)
+ jobject peer, jlong hwnd)
{
OGLSDOps *oglsdo = (OGLSDOps *)SurfaceData_InitOps(env, wglsd,
sizeof(OGLSDOps));
@@ -71,10 +81,17 @@ Java_sun_java2d_opengl_WGLSurfaceData_initOps(JNIEnv *env, jobject wglsd,
oglsdo->drawableType = OGLSD_UNDEFINED;
oglsdo->activeBuffer = GL_FRONT;
oglsdo->needsInit = JNI_TRUE;
- oglsdo->xOffset = xoff;
- oglsdo->yOffset = yoff;
+ if (peer != NULL) {
+ RECT insets;
+ AwtComponent_GetInsets(env, peer, &insets);
+ oglsdo->xOffset = -insets.left;
+ oglsdo->yOffset = -insets.bottom;
+ } else {
+ oglsdo->xOffset = 0;
+ oglsdo->yOffset = 0;
+ }
- wglsdo->peerData = pPeerData;
+ wglsdo->window = (HWND)jlong_to_ptr(hwnd);
wglsdo->configInfo = (WGLGraphicsConfigInfo *)jlong_to_ptr(pConfigInfo);
if (wglsdo->configInfo == NULL) {
free(wglsdo);
@@ -96,14 +113,14 @@ OGLSD_DestroyOGLSurface(JNIEnv *env, OGLSDOps *oglsdo)
J2dTraceLn(J2D_TRACE_INFO, "OGLSD_DestroyOGLSurface");
if (oglsdo->drawableType == OGLSD_PBUFFER) {
- if (wglsdo->drawable.pbuffer != 0) {
+ if (wglsdo->pbuffer != 0) {
if (wglsdo->pbufferDC != 0) {
- j2d_wglReleasePbufferDCARB(wglsdo->drawable.pbuffer,
+ j2d_wglReleasePbufferDCARB(wglsdo->pbuffer,
wglsdo->pbufferDC);
wglsdo->pbufferDC = 0;
}
- j2d_wglDestroyPbufferARB(wglsdo->drawable.pbuffer);
- wglsdo->drawable.pbuffer = 0;
+ j2d_wglDestroyPbufferARB(wglsdo->pbuffer);
+ wglsdo->pbuffer = 0;
}
}
}
@@ -256,7 +273,7 @@ OGLSD_MakeOGLContextCurrent(JNIEnv *env, OGLSDOps *srcOps, OGLSDOps *dstOps)
if (dstOps->drawableType == OGLSD_PBUFFER) {
dstHDC = dstWGLOps->pbufferDC;
} else {
- dstHDC = GetDC(dstWGLOps->drawable.window);
+ dstHDC = GetDC(dstWGLOps->window);
}
// get the hdc for the source surface
@@ -284,7 +301,7 @@ OGLSD_MakeOGLContextCurrent(JNIEnv *env, OGLSDOps *srcOps, OGLSDOps *dstOps)
J2dRlsTraceLn(J2D_TRACE_ERROR,
"OGLSD_MakeOGLContextCurrent: could not make current");
if (dstOps->drawableType != OGLSD_PBUFFER) {
- ReleaseDC(dstWGLOps->drawable.window, dstHDC);
+ ReleaseDC(dstWGLOps->window, dstHDC);
}
return NULL;
}
@@ -297,7 +314,7 @@ OGLSD_MakeOGLContextCurrent(JNIEnv *env, OGLSDOps *srcOps, OGLSDOps *dstOps)
}
if (dstOps->drawableType != OGLSD_PBUFFER) {
- ReleaseDC(dstWGLOps->drawable.window, dstHDC);
+ ReleaseDC(dstWGLOps->window, dstHDC);
}
return oglc;
@@ -340,7 +357,7 @@ OGLSD_InitOGLWindow(JNIEnv *env, OGLSDOps *oglsdo)
return JNI_FALSE;
}
- window = AwtComponent_GetHWnd(env, wglsdo->peerData);
+ window = wglsdo->window;
if (!IsWindow(window)) {
J2dRlsTraceLn(J2D_TRACE_ERROR,
"OGLSD_InitOGLWindow: disposed component");
@@ -369,7 +386,6 @@ OGLSD_InitOGLWindow(JNIEnv *env, OGLSDOps *oglsdo)
oglsdo->isOpaque = JNI_TRUE;
oglsdo->width = wbounds.right - wbounds.left;
oglsdo->height = wbounds.bottom - wbounds.top;
- wglsdo->drawable.window = window;
wglsdo->pbufferDC = 0;
J2dTraceLn2(J2D_TRACE_VERBOSE, " created window: w=%d h=%d",
@@ -505,9 +521,11 @@ Java_sun_java2d_opengl_WGLSurfaceData_initPbuffer
oglsdo->isOpaque = isOpaque;
oglsdo->width = width;
oglsdo->height = height;
- wglsdo->drawable.pbuffer = pbuffer;
+ wglsdo->pbuffer = pbuffer;
wglsdo->pbufferDC = pbufferDC;
+ OGLSD_SetNativeDimensions(env, oglsdo, width, height);
+
return JNI_TRUE;
}
@@ -543,3 +561,83 @@ OGLSD_SwapBuffers(JNIEnv *env, jlong pPeerData)
"OGLSD_SwapBuffers: error while releasing dc");
}
}
+
+/*
+ * Class: sun_java2d_opengl_WGLSurfaceData
+ * Method: updateWindowAccelImpl
+ * Signature: (JJII)Z
+ */
+JNIEXPORT jboolean JNICALL
+ Java_sun_java2d_opengl_WGLSurfaceData_updateWindowAccelImpl
+ (JNIEnv *env, jclass clazz, jlong pData, jobject peer, jint w, jint h)
+{
+ OGLSDOps *oglsdo = (OGLSDOps *)jlong_to_ptr(pData);
+ OGLPixelFormat pf = PixelFormats[0/*PF_INT_ARGB_PRE*/];
+ HBITMAP hBitmap = NULL;
+ void *pDst;
+ jint srcx, srcy, dstx, dsty, width, height;
+ jint pixelStride = 4;
+ jint scanStride = pixelStride * w;
+
+ J2dTraceLn(J2D_TRACE_INFO, "WGLSurfaceData_updateWindowAccelImpl");
+
+ if (w <= 0 || h <= 0) {
+ return JNI_TRUE;
+ }
+ if (oglsdo == NULL) {
+ return JNI_FALSE;
+ }
+ RESET_PREVIOUS_OP();
+
+ width = w;
+ height = h;
+ srcx = srcy = dstx = dsty = 0;
+
+ pDst = malloc(height * scanStride);
+ if (pDst == NULL) {
+ return JNI_FALSE;
+ }
+ ZeroMemory(pDst, height * scanStride);
+
+ // the code below is mostly copied from OGLBlitLoops_SurfaceToSwBlit
+
+ j2d_glPixelStorei(GL_PACK_SKIP_PIXELS, dstx);
+ j2d_glPixelStorei(GL_PACK_ROW_LENGTH, scanStride / pixelStride);
+ j2d_glPixelStorei(GL_PACK_ALIGNMENT, pf.alignment);
+
+ // this accounts for lower-left origin of the source region
+ srcx = oglsdo->xOffset + srcx;
+ srcy = oglsdo->yOffset + oglsdo->height - (srcy + 1);
+ // we must read one scanline at a time because there is no way
+ // to read starting at the top-left corner of the source region
+ while (height > 0) {
+ j2d_glPixelStorei(GL_PACK_SKIP_ROWS, dsty);
+ j2d_glReadPixels(srcx, srcy, width, 1,
+ pf.format, pf.type, pDst);
+ srcy--;
+ dsty++;
+ height--;
+ }
+
+ j2d_glPixelStorei(GL_PACK_SKIP_PIXELS, 0);
+ j2d_glPixelStorei(GL_PACK_SKIP_ROWS, 0);
+ j2d_glPixelStorei(GL_PACK_ROW_LENGTH, 0);
+ j2d_glPixelStorei(GL_PACK_ALIGNMENT, 4);
+
+ // the pixels read from the surface are already premultiplied
+ // REMIND: commented until translucent window support is integrated
+// hBitmap = BitmapUtil_CreateBitmapFromARGBPre(w, h, scanStride,
+// (int*)pDst);
+ free(pDst);
+
+ if (hBitmap == NULL) {
+ return JNI_FALSE;
+ }
+
+ // REMIND: commented until translucent window support is integrated
+ // AwtWindow_UpdateWindow(env, peer, w, h, hBitmap);
+
+ // hBitmap is released in UpdateWindow
+
+ return JNI_TRUE;
+}
diff --git a/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.h b/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.h
index ef877b97e9c..acc7f46bb1b 100644
--- a/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.h
+++ b/jdk/src/windows/native/sun/java2d/opengl/WGLSurfaceData.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2004-2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2004-2008 Sun Microsystems, Inc. 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
@@ -35,12 +35,9 @@
* OGLSurfaceData. It is referenced by the native OGLSDOps structure.
*/
typedef struct _WGLSDOps {
- jlong peerData;
WGLGraphicsConfigInfo *configInfo;
- union {
- HWND window;
- HPBUFFERARB pbuffer;
- } drawable;
+ HWND window;
+ HPBUFFERARB pbuffer;
HDC pbufferDC;
} WGLSDOps;
diff --git a/jdk/src/windows/native/sun/java2d/windows/DDBlitLoops.cpp b/jdk/src/windows/native/sun/java2d/windows/DDBlitLoops.cpp
deleted file mode 100644
index 2caa03d3a4c..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/DDBlitLoops.cpp
+++ /dev/null
@@ -1,413 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include
-#include
-#include
-#include
-#include "ddrawUtils.h"
-#include "GraphicsPrimitiveMgr.h"
-#include "Region.h"
-#include "Trace.h"
-
-extern int currNumDevices;
-extern CriticalSection windowMoveLock;
-
-extern "C" {
-
-/**
- * Return TRUE if rCheck is contained within rContainer
- */
-INLINE BOOL RectInRect(RECT *rCheck, RECT *rContainer)
-{
- // Assumption: left <= right, top <= bottom
- if (rCheck->left >= rContainer->left &&
- rCheck->right <= rContainer->right &&
- rCheck->top >= rContainer->top &&
- rCheck->bottom <= rContainer->bottom)
- {
- return TRUE;
- } else {
- return FALSE;
- }
-}
-
-/**
- * Returns whether the given rectangle (in screen-relative
- * coords) is within the rectangle of the given device.
- * NOTE: A side-effect of this function is offsetting the
- * rectangle by the left/top of the monitor rectangle.
- */
-INLINE BOOL RectInDevice(RECT *rect, AwtWin32GraphicsDevice *device)
-{
- MONITOR_INFO *mi = device->GetMonitorInfo();
- ::OffsetRect(rect, mi->rMonitor.left, mi->rMonitor.top);
- if (!RectInRect(rect, &mi->rMonitor)) {
- return TRUE;
- }
- return FALSE;
-}
-
-/**
- * Need to handle Blt to other devices iff:
- * - there are >1 devices on the system
- * - at least one of src/dest is an onscreen window
- * - the onscreen window overlaps with
- * a monitor which is not the monitor associated with the window
- */
-void MultimonBlt(JNIEnv *env, Win32SDOps *wsdoSrc, Win32SDOps *wsdoDst,
- jobject clip,
- jint srcx, jint srcy,
- jint dstx, jint dsty,
- RECT *rSrc, RECT *rDst)
-{
- J2dTraceLn(J2D_TRACE_INFO, "MultimonBlt");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " srcx=%-4d srcy=%-4d dstx=%-4d dsty=%-4d",
- srcx, srcy, dstx, dsty);
- if (rSrc == NULL) {
- J2dTraceLn(J2D_TRACE_ERROR, "MultimonBlt: null rSrc");
- return;
- }
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " rSrc: l=%-4d t=%-4d r=%-4d b=%-4d",
- rSrc->left, rSrc->top, rSrc->right, rSrc->bottom);
- if (rDst == NULL) {
- J2dTraceLn(J2D_TRACE_ERROR, "MultimonBlt: null rDst");
- return;
- }
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " rDst: l=%-4d t=%-4d r=%-4d b=%-4d",
- rDst->left, rDst->top, rDst->right, rDst->bottom);
- int currentDevice = -1;
- RECT rectToIntersect;
-
- if (!(wsdoSrc->window || wsdoDst->window))
- {
- // Neither surface is onscreen: nothing to do
- return;
- }
- BOOL doGdiBlt = FALSE;
- if (wsdoSrc->window) {
- doGdiBlt = RectInDevice(rSrc, wsdoSrc->device);
- if (doGdiBlt) {
- currentDevice = wsdoSrc->device->GetDeviceIndex();
- rectToIntersect = *rSrc;
- }
- } else if (wsdoDst->window) {
- doGdiBlt = RectInDevice(rDst, wsdoDst->device);
- if (doGdiBlt) {
- currentDevice = wsdoDst->device->GetDeviceIndex();
- rectToIntersect = *rDst;
- }
- }
- if (doGdiBlt) {
- // Need to invoke Setup functions to setup HDCs because we used
- // the NoSetup versions of GetOps for performance reasons
- SurfaceData_InvokeSetup(env, (SurfaceDataOps*)wsdoSrc);
- SurfaceData_InvokeSetup(env, (SurfaceDataOps*)wsdoDst);
- HDC hdcSrc = wsdoSrc->GetDC(env, wsdoSrc, 0, NULL, NULL, NULL, 0);
- if (!hdcSrc) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "MultimonBlt: Null src HDC in MultimonBlt");
- return;
- }
- HDC hdcDst = wsdoDst->GetDC(env, wsdoDst, 0, NULL, clip, NULL, 0);
- if (!hdcDst) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "MultimonBlt: Null dst HDC in MultimonBlt");
- wsdoSrc->ReleaseDC(env, wsdoSrc, hdcSrc);
- return;
- }
- for (int i = 0; i < currNumDevices; ++i) {
- // Assumption: can't end up here for copies between two
- // different windows; it must be a copy between offscreen
- // surfaces or a window and an offscreen surface. We've
- // already handled the Blt to window on the window's native
- // GraphicsDevice, so skip that device now.
- if (i == currentDevice) {
- continue;
- }
- MONITOR_INFO *mi = AwtWin32GraphicsDevice::GetMonitorInfo(i);
- RECT rIntersect;
- ::IntersectRect(&rIntersect, &rectToIntersect, &mi->rMonitor);
- if (!::IsRectEmpty(&rIntersect)) {
- int newSrcX = srcx + (rIntersect.left - rectToIntersect.left);
- int newSrcY = srcy + (rIntersect.top - rectToIntersect.top);
- int newDstX = dstx + (rIntersect.left - rectToIntersect.left);
- int newDstY = dsty + (rIntersect.top - rectToIntersect.top);
- int newW = rIntersect.right - rIntersect.left;
- int newH = rIntersect.bottom - rIntersect.top;
- ::BitBlt(hdcDst, newDstX, newDstY, newW, newH, hdcSrc,
- newSrcX, newSrcY, SRCCOPY);
- }
- }
- wsdoSrc->ReleaseDC(env, wsdoSrc, hdcSrc);
- wsdoDst->ReleaseDC(env, wsdoDst, hdcDst);
- }
-}
-
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_DDBlitLoops_Blit
- (JNIEnv *env, jobject joSelf,
- jobject srcData, jobject dstData,
- jobject composite, jobject clip,
- jint srcx, jint srcy,
- jint dstx, jint dsty,
- jint width, jint height)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDBlitLoops_Blit");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " srcx=%-4d srcy=%-4d dstx=%-4d dsty=%-4d",
- srcx, srcy, dstx, dsty);
- J2dTraceLn2(J2D_TRACE_VERBOSE, " width=%-4d height=%-4d", width, height);
- POINT ptDst = {0, 0};
- POINT ptSrc = {0, 0};
- Win32SDOps *wsdoSrc = Win32SurfaceData_GetOpsNoSetup(env, srcData);
- Win32SDOps *wsdoDst = Win32SurfaceData_GetOpsNoSetup(env, dstData);
- RegionData clipInfo;
-
- if (!wsdoSrc->ddInstance || !wsdoDst->ddInstance) {
- // Some situations can cause us to fail on primary
- // creation, resulting in null lpSurface and null ddInstance
- // for a Win32Surface object.. Just noop this call in that case.
- return;
- }
-
- if (wsdoSrc->invalid || wsdoDst->invalid) {
- SurfaceData_ThrowInvalidPipeException(env,
- "DDBlitLoops_Blit: invalid surface data");
- return;
- }
-
- RECT rSrc = {srcx, srcy, srcx + width, srcy + height};
- RECT rDst = {dstx, dsty, dstx + width, dsty + height};
- if (Region_GetInfo(env, clip, &clipInfo)) {
- return;
- }
-
- /* If dst and/or src are offscreen surfaces, need to make sure
- that Blt is within the boundaries of those surfaces. If not,
- clip the surface in question and also clip the other
- surface by the same amount.
- */
- if (!wsdoDst->window) {
- CLIP2RECTS(rDst, 0, 0, wsdoDst->w, wsdoDst->h, rSrc);
- }
- CLIP2RECTS(rDst,
- clipInfo.bounds.x1, clipInfo.bounds.y1,
- clipInfo.bounds.x2, clipInfo.bounds.y2,
- rSrc);
- if (!wsdoSrc->window) {
- CLIP2RECTS(rSrc, 0, 0, wsdoSrc->w, wsdoSrc->h, rDst);
- }
- Region_IntersectBoundsXYXY(&clipInfo,
- rDst.left, rDst.top,
- rDst.right, rDst.bottom);
- if (Region_IsEmpty(&clipInfo)) {
- return;
- }
- if (wsdoDst->window || wsdoSrc->window) {
- if ((wsdoDst->window && !::IsWindowVisible(wsdoDst->window)) ||
- (wsdoSrc->window && !::IsWindowVisible(wsdoSrc->window)))
- {
- return;
- }
- // The windowMoveLock CriticalSection ensures that a window cannot
- // move while we are in the middle of copying pixels into it. See
- // the WM_WINDOWPOSCHANGING code in awt_Component.cpp for more
- // information.
- windowMoveLock.Enter();
- if (wsdoDst->window) {
- ::ClientToScreen(wsdoDst->window, &ptDst);
- MONITOR_INFO *mi = wsdoDst->device->GetMonitorInfo();
- ptDst.x -= wsdoDst->insets.left;
- ptDst.y -= wsdoDst->insets.top;
- ptDst.x -= mi->rMonitor.left;
- ptDst.y -= mi->rMonitor.top;
- ::OffsetRect(&rDst, ptDst.x, ptDst.y);
- }
- if (wsdoSrc->window) {
- MONITOR_INFO *mi = wsdoDst->device->GetMonitorInfo();
- ::ClientToScreen(wsdoSrc->window, &ptSrc);
- ptSrc.x -= wsdoSrc->insets.left;
- ptSrc.y -= wsdoSrc->insets.top;
- ptSrc.x -= mi->rMonitor.left;
- ptSrc.y -= mi->rMonitor.top;
- ::OffsetRect(&rSrc, ptSrc.x, ptSrc.y);
- }
- }
- if (Region_IsRectangular(&clipInfo)) {
- DDBlt(env, wsdoSrc, wsdoDst, &rDst, &rSrc);
- } else {
- SurfaceDataBounds span;
- RECT rSrcSpan, rDstSpan;
- ptSrc.x += srcx - dstx;
- ptSrc.y += srcy - dsty;
- Region_StartIteration(env, &clipInfo);
- while (Region_NextIteration(&clipInfo, &span)) {
- ::SetRect(&rDstSpan, span.x1, span.y1, span.x2, span.y2);
- ::CopyRect(&rSrcSpan, &rDstSpan);
- ::OffsetRect(&rDstSpan, ptDst.x, ptDst.y);
- ::OffsetRect(&rSrcSpan, ptSrc.x, ptSrc.y);
- DDBlt(env, wsdoSrc, wsdoDst, &rDstSpan, &rSrcSpan);
- }
- Region_EndIteration(env, &clipInfo);
- }
- if (wsdoDst->window || wsdoSrc->window) {
- windowMoveLock.Leave();
- }
-
- if (currNumDevices > 1) {
- // Also need to handle Blit in multimon case, where part of the
- // source or dest lies on a different device
- MultimonBlt(env, wsdoSrc, wsdoDst, clip, srcx, srcy, dstx, dsty,
- &rSrc, &rDst);
- }
-}
-
-
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_DDScaleLoops_Scale
- (JNIEnv *env, jobject joSelf,
- jobject srcData, jobject dstData,
- jobject composite,
- jint srcx, jint srcy,
- jint dstx, jint dsty,
- jint srcWidth, jint srcHeight,
- jint dstWidth, jint dstHeight)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDScaleLoops_Scale");
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " srcx=%-4d srcy=%-4d dstx=%-4d dsty=%-4d",
- srcx, srcy, dstx, dsty);
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- " srcWidth=%-4d srcHeight=%-4d dstWidth=%-4d dstHeight=%-4d",
- srcWidth, srcHeight, dstWidth, dstHeight);
- POINT ptDst = {0, 0};
- POINT ptSrc = {0, 0};
- Win32SDOps *wsdoSrc = Win32SurfaceData_GetOpsNoSetup(env, srcData);
- Win32SDOps *wsdoDst = Win32SurfaceData_GetOpsNoSetup(env, dstData);
-
- if (!wsdoSrc->ddInstance || !wsdoDst->ddInstance) {
- // Some situations can cause us to fail on primary
- // creation, resulting in null lpSurface and null ddInstance
- // for a Win32Surface object.. Just noop this call in that case.
- return;
- }
-
- if (wsdoSrc->invalid || wsdoDst->invalid) {
- SurfaceData_ThrowInvalidPipeException(env,
- "DDBlitLoops_Scale: invalid surface data");
- return;
- }
-
- RECT rSrc = {srcx, srcy, srcx + srcWidth, srcy + srcHeight};
- RECT rDst = {dstx, dsty, dstx + dstWidth, dsty + dstHeight};
-
- /* If dst and/or src are offscreen surfaces, need to make sure
- that Blt is within the boundaries of those surfaces. If not,
- clip the surface in question and also rescale the other
- surface according to the new scaling rectangle.
- */
- if (!wsdoDst->window &&
- (dstx < 0 || dsty < 0 ||
- rDst.right > wsdoDst->w || rDst.bottom > wsdoDst->h))
- {
- RECT newRDst;
- newRDst.left = max(0, rDst.left);
- newRDst.top = max(0, rDst.top);
- newRDst.right = min(wsdoDst->w, rDst.right);
- newRDst.bottom = min(wsdoDst->h, rDst.bottom);
- double srcDstScaleW = (double)srcWidth/(double)dstWidth;
- double srcDstScaleH = (double)srcHeight/(double)dstHeight;
- rSrc.left += (int)(srcDstScaleW * (newRDst.left - rDst.left));
- rSrc.top += (int)(srcDstScaleH * (newRDst.top - rDst.top));
- rSrc.right += (int)(srcDstScaleW * (newRDst.right - rDst.right));
- rSrc.bottom += (int)(srcDstScaleH * (newRDst.bottom - rDst.bottom));
- rDst = newRDst;
- }
- if (!wsdoSrc->window &&
- (srcx < 0 || srcy < 0 ||
- rSrc.right > wsdoSrc->w || rSrc.bottom > wsdoSrc->h))
- {
- RECT newRSrc;
- newRSrc.left = max(0, rSrc.left);
- newRSrc.top = max(0, rSrc.top);
- newRSrc.right = min(wsdoSrc->w, rSrc.right);
- newRSrc.bottom = min(wsdoSrc->h, rSrc.bottom);
- double dstSrcScaleW = (double)dstWidth/(double)srcWidth;
- double dstSrcScaleH = (double)dstHeight/(double)srcHeight;
- rDst.left += (int)(dstSrcScaleW * (newRSrc.left - rSrc.left));
- rDst.top += (int)(dstSrcScaleH * (newRSrc.top - rSrc.top));
- rDst.right += (int)(dstSrcScaleW * (newRSrc.right - rSrc.right));
- rDst.bottom += (int)(dstSrcScaleH * (newRSrc.bottom - rSrc.bottom));
- rSrc = newRSrc;
- }
- if (wsdoDst->window || wsdoSrc->window) {
- if ((wsdoDst->window && !::IsWindowVisible(wsdoDst->window)) ||
- (wsdoSrc->window && !::IsWindowVisible(wsdoSrc->window)))
- {
- return;
- }
- // The windowMoveLock CriticalSection ensures that a window cannot
- // move while we are in the middle of copying pixels into it. See
- // the WM_WINDOWPOSCHANGING code in awt_Component.cpp for more
- // information.
- windowMoveLock.Enter();
- if (wsdoDst->window) {
- ::ClientToScreen(wsdoDst->window, &ptDst);
- MONITOR_INFO *mi = wsdoDst->device->GetMonitorInfo();
- ptDst.x -= wsdoDst->insets.left;
- ptDst.y -= wsdoDst->insets.top;
- ptDst.x -= mi->rMonitor.left;
- ptDst.y -= mi->rMonitor.top;
- ::OffsetRect(&rDst, ptDst.x, ptDst.y);
- }
- if (wsdoSrc->window) {
- MONITOR_INFO *mi = wsdoDst->device->GetMonitorInfo();
- ::ClientToScreen(wsdoSrc->window, &ptSrc);
- ptSrc.x -= wsdoSrc->insets.left;
- ptSrc.y -= wsdoSrc->insets.top;
- ptSrc.x -= mi->rMonitor.left;
- ptSrc.y -= mi->rMonitor.top;
- ::OffsetRect(&rSrc, ptSrc.x, ptSrc.y);
- }
- }
- DDBlt(env, wsdoSrc, wsdoDst, &rDst, &rSrc);
- if (wsdoDst->window || wsdoSrc->window) {
- windowMoveLock.Leave();
- }
-
- if (currNumDevices > 1) {
- // Also need to handle Blit in multimon case, where part of the
- // source or dest lies on a different device
- MultimonBlt(env, wsdoSrc, wsdoDst, NULL, srcx, srcy, dstx, dsty,
- &rSrc, &rDst);
- }
-}
-
-
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/DDRenderer.cpp b/jdk/src/windows/native/sun/java2d/windows/DDRenderer.cpp
deleted file mode 100644
index 258a7beedeb..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/DDRenderer.cpp
+++ /dev/null
@@ -1,172 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include "sun_java2d_windows_DDRenderer.h"
-#include "Win32SurfaceData.h"
-
-#include
-#include "ddrawUtils.h"
-#include "Trace.h"
-
-/*
- * Class: sun_java2d_windows_DDRenderer
- * Method: doDrawLineDD
- * Signature: (Lsun/java2d/SurfaceData;IIIII)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_DDRenderer_doDrawLineDD
- (JNIEnv *env, jobject wr,
- jobject sData,
- jint color,
- jint x1, jint y1, jint x2, jint y2)
-{
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, sData);
- RECT fillRect;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDRenderer_doDrawLineDD");
-
- // Assume x1 <= x2 and y1 <= y2 (that's the way the
- // Java code is written)
- fillRect.left = x1;
- fillRect.top = y1;
- fillRect.right = x2+1;
- fillRect.bottom = y2+1;
- DDColorFill(env, sData, wsdo, &fillRect, color);
-}
-
-
-/*
- * Class: sun_java2d_windows_DDRenderer
- * Method: doFillRect
- * Signature: (Lsun/java2d/SurfaceData;IIIII)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_DDRenderer_doFillRectDD
- (JNIEnv *env, jobject wr,
- jobject sData,
- jint color,
- jint left, jint top, jint right, jint bottom)
-{
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, sData);
- RECT fillRect;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDRenderer_doFillRectDD");
-
- fillRect.left = left;
- fillRect.top = top;
- fillRect.right = right;
- fillRect.bottom = bottom;
- DDColorFill(env, sData, wsdo, &fillRect, color);
-}
-
-
-/*
- * Class: sun_java2d_windows_DDRenderer
- * Method: doDrawRectDD
- * Signature: (Lsun/java2d/SurfaceData;IIIII)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_DDRenderer_doDrawRectDD
- (JNIEnv *env, jobject wr,
- jobject sData,
- jint color,
- jint x, jint y, jint w, jint h)
-{
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, sData);
- RECT fillRect;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDRenderer_doDrawRectDD");
-
- if (w == 0 || h == 0) {
- fillRect.left = x;
- fillRect.top = y;
- fillRect.right = w + 1;
- fillRect.bottom = h + 1;
- DDColorFill(env, sData, wsdo, &fillRect, color);
- }
- else {
- fillRect.left = x;
- fillRect.top = y;
- fillRect.right = x + w + 1;
- fillRect.bottom = y + 1;
- if (!DDColorFill(env, sData, wsdo, &fillRect, color))
- return;
- fillRect.top = y + 1;
- fillRect.right = x + 1;
- fillRect.bottom = y + h + 1;
- if (!DDColorFill(env, sData, wsdo, &fillRect, color))
- return;
- fillRect.left = x + 1;
- fillRect.top = y + h;
- fillRect.right = x + w + 1;
- fillRect.bottom = y + h + 1;
- if (!DDColorFill(env, sData, wsdo, &fillRect, color))
- return;
- fillRect.left = x + w;
- fillRect.top = y + 1;
- fillRect.bottom = y + h;
- if (!DDColorFill(env, sData, wsdo, &fillRect, color))
- return;
- }
-}
-
-/*
- * Class: sun_java2d_windows_DDRenderer
- * Method: devCopyArea
- * Signature: (Lsun/awt/windows/SurfaceData;IIIIII)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_DDRenderer_devCopyArea
- (JNIEnv *env, jobject wr,
- jobject sData,
- jint srcx, jint srcy,
- jint dx, jint dy,
- jint width, jint height)
-{
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, sData);
- J2dTraceLn(J2D_TRACE_INFO, "DDRenderer_devCopyArea");
- J2dTrace4(J2D_TRACE_VERBOSE, " sx=%-4d sy=%-4d dx=%-4d dy=%-4d",
- srcx, srcy, dx, dy);
- J2dTraceLn2(J2D_TRACE_VERBOSE, " w=%-4d h=%-4d", width, height);
- if (wsdo == NULL) {
- return;
- }
-
- RECT rSrc = {srcx, srcy, srcx + width, srcy + height};
- if (DDCanBlt(wsdo)) {
- RECT rDst = rSrc;
- ::OffsetRect(&rDst, dx, dy);
-
- DDBlt(env,wsdo, wsdo, &rDst, &rSrc);
- return;
- }
- HDC hDC = wsdo->GetDC(env, wsdo, 0, NULL, NULL, NULL, 0);
- if (hDC == NULL) {
- return;
- }
-
- VERIFY(::ScrollDC(hDC, dx, dy, &rSrc, NULL, NULL, NULL));
- wsdo->ReleaseDC(env, wsdo, hDC);
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/GDIBlitLoops.cpp b/jdk/src/windows/native/sun/java2d/windows/GDIBlitLoops.cpp
index 6ca907d867d..06d3c5f3fb5 100644
--- a/jdk/src/windows/native/sun/java2d/windows/GDIBlitLoops.cpp
+++ b/jdk/src/windows/native/sun/java2d/windows/GDIBlitLoops.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2002-2008 Sun Microsystems, Inc. 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
@@ -25,8 +25,8 @@
#include
#include "gdefs.h"
-#include "ddrawUtils.h"
#include "Trace.h"
+#include "GDIWindowSurfaceData.h"
static RGBQUAD *byteGrayPalette = NULL;
@@ -60,7 +60,7 @@ Java_sun_java2d_windows_GDIBlitLoops_nativeBlit
SurfaceDataRasInfo srcInfo;
SurfaceDataOps *srcOps = SurfaceData_GetOps(env, srcData);
- Win32SDOps *dstOps = Win32SurfaceData_GetOps(env, dstData);
+ GDIWinSDOps *dstOps = GDIWindowSurfaceData_GetOps(env, dstData);
jint lockFlags;
srcInfo.bounds.x1 = srcx;
diff --git a/jdk/src/windows/native/sun/java2d/windows/GDIRenderer.cpp b/jdk/src/windows/native/sun/java2d/windows/GDIRenderer.cpp
index 025dd0d7aef..5a06df00bcc 100644
--- a/jdk/src/windows/native/sun/java2d/windows/GDIRenderer.cpp
+++ b/jdk/src/windows/native/sun/java2d/windows/GDIRenderer.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -26,11 +26,10 @@
#include "sun_java2d_windows_GDIRenderer.h"
#include "java_awt_geom_PathIterator.h"
-#include "Win32SurfaceData.h"
+#include "GDIWindowSurfaceData.h"
#include "awt_Component.h"
#include "awt_Pen.h"
#include "awt_Brush.h"
-#include "ddrawUtils.h"
#include "jni.h"
@@ -133,7 +132,7 @@ Java_sun_java2d_windows_GDIRenderer_doDrawLine
J2dTraceLn5(J2D_TRACE_VERBOSE,
" color=0x%x x1=%-4d y1=%-4d x2=%-4d y2=%-4d",
color, x1, y1, x2, y2);
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -184,7 +183,7 @@ Java_sun_java2d_windows_GDIRenderer_doDrawRect
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -241,7 +240,7 @@ Java_sun_java2d_windows_GDIRenderer_doDrawRoundRect
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -279,7 +278,7 @@ Java_sun_java2d_windows_GDIRenderer_doDrawOval
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -315,7 +314,7 @@ Java_sun_java2d_windows_GDIRenderer_doDrawArc
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -385,7 +384,7 @@ Java_sun_java2d_windows_GDIRenderer_doDrawPoly
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -432,7 +431,7 @@ Java_sun_java2d_windows_GDIRenderer_doFillRect
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -476,7 +475,7 @@ Java_sun_java2d_windows_GDIRenderer_doFillRoundRect
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -543,7 +542,7 @@ Java_sun_java2d_windows_GDIRenderer_doFillOval
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -589,7 +588,7 @@ Java_sun_java2d_windows_GDIRenderer_doFillArc
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -652,7 +651,7 @@ Java_sun_java2d_windows_GDIRenderer_doFillPoly
return;
}
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -698,7 +697,7 @@ Java_sun_java2d_windows_GDIRenderer_doShape
J2dTraceLn4(J2D_TRACE_VERBOSE,
" color=0x%x transx=%-4d transy=%-4d isfill=%-4d",
color, transX, transY, isfill);
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, sData);
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, sData);
if (wsdo == NULL) {
return;
}
@@ -849,8 +848,6 @@ Java_sun_java2d_windows_GDIRenderer_doShape
} /* extern "C" */
-extern int currNumDevices;
-
INLINE BOOL RectInMonitorRect(RECT *rCheck, RECT *rContainer)
{
// Assumption: left <= right, top <= bottom
@@ -878,8 +875,8 @@ Java_sun_java2d_windows_GDIRenderer_devCopyArea
jint dx, jint dy,
jint width, jint height)
{
- Win32SDOps *wsdo = Win32SurfaceData_GetOps(env, wsd);
- J2dTraceLn(J2D_TRACE_INFO, "Win32SurfaceData_devCopyArea");
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOps(env, wsd);
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWindowSurfaceData_devCopyArea");
J2dTraceLn4(J2D_TRACE_VERBOSE, " srcx=%-4d srcy=%-4d dx=%-4d dy=%-4d",
srcx, srcy, dx, dy);
J2dTraceLn2(J2D_TRACE_VERBOSE, " w=%-4d h=%-4d", width, height);
@@ -892,61 +889,6 @@ Java_sun_java2d_windows_GDIRenderer_devCopyArea
return;
}
- if (DDCanBlt(wsdo)) {
- // We try to use ddraw for this copy because it tends to be faster
- // than GDI. We punt if:
- // - the source rect is clipped (GDI does a better job of
- // validating the clipped area)
- // - there is only one monitor
- // OR
- // - the src and dest rectangles are both wholly within the device
- // associated with this surfaceData
- // REMIND: There may be a bug looming out here where the user
- // moves an overlapping window between the time that we check the
- // clip and the time that the Blt is executed, then we may not be
- // invalidating the src region the way we should.
-
- // Here, rSrc, rDst are the src/dst rectangles in screen coordinates.
- // r[Src|Dst]Absolute are the src/dst rectangles in virtual
- // screen space (where all monitors occupy the same coords).
- RECT rSrc = {srcx, srcy, srcx + width, srcy + height};
- RECT rDst, rSrcAbsolute, rDstAbsolute;
- POINT clientOrigin = {0, 0};
- MONITOR_INFO *mi = wsdo->device->GetMonitorInfo();
- ::ScreenToClient(wsdo->window, &clientOrigin);
- ::OffsetRect(&rSrc,
- (-clientOrigin.x - wsdo->insets.left),
- (-clientOrigin.y - wsdo->insets.top));
- rSrcAbsolute = rSrc;
- ::OffsetRect(&rSrc, (-mi->rMonitor.left), (-mi->rMonitor.top));
- rDst = rSrc;
- ::OffsetRect(&rDst, dx, dy);
- rDstAbsolute = rSrcAbsolute;
- ::OffsetRect(&rDstAbsolute, dx, dy);
-
- if (DDClipCheck(wsdo, &rSrc) &&
- (currNumDevices <= 1 ||
- (RectInMonitorRect(&rSrcAbsolute, &mi->rMonitor) &&
- RectInMonitorRect(&rDstAbsolute, &mi->rMonitor))))
- {
- AwtComponent *comp = NULL;
-
- // Bug 4362500: Win2k pointer garbage on screen->screen DD blts
- if (IS_WIN2000) {
- comp = Win32SurfaceData_GetComp(env, wsdo);
- if (comp != NULL) {
- comp->SendMessage(WM_AWT_HIDECURSOR, NULL);
- }
- }
-
- DDBlt(env,wsdo, wsdo, &rDst, &rSrc);
-
- if (comp != NULL) {
- comp->SendMessage(WM_AWT_SHOWCURSOR, NULL);
- }
- return;
- }
- }
HDC hDC = wsdo->GetDC(env, wsdo, 0, NULL, NULL, NULL, 0);
if (hDC == NULL) {
return;
diff --git a/jdk/src/windows/native/sun/java2d/windows/Win32SurfaceData.cpp b/jdk/src/windows/native/sun/java2d/windows/GDIWindowSurfaceData.cpp
similarity index 74%
rename from jdk/src/windows/native/sun/java2d/windows/Win32SurfaceData.cpp
rename to jdk/src/windows/native/sun/java2d/windows/GDIWindowSurfaceData.cpp
index eee57e0df97..b7032df7b3c 100644
--- a/jdk/src/windows/native/sun/java2d/windows/Win32SurfaceData.cpp
+++ b/jdk/src/windows/native/sun/java2d/windows/GDIWindowSurfaceData.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2007 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -23,32 +23,30 @@
* have any questions.
*/
-#include "sun_java2d_windows_Win32SurfaceData.h"
+#include "sun_java2d_windows_GDIWindowSurfaceData.h"
-#include "Win32SurfaceData.h"
+#include "GDIWindowSurfaceData.h"
#include "GraphicsPrimitiveMgr.h"
#include "Region.h"
#include "Disposer.h"
#include "WindowsFlags.h"
#include "awt_Component.h"
-#include "ddrawUtils.h"
#include "awt_Palette.h"
#include "awt_Win32GraphicsDevice.h"
#include "gdefs.h"
-#include "D3DContext.h"
#include "Trace.h"
#include "Devices.h"
#include "jni_util.h"
-static LockFunc Win32SD_Lock;
-static GetRasInfoFunc Win32SD_GetRasInfo;
-static UnlockFunc Win32SD_Unlock;
-static DisposeFunc Win32SD_Dispose;
-static SetupFunc Win32SD_Setup;
-static GetDCFunc Win32SD_GetDC;
-static ReleaseDCFunc Win32SD_ReleaseDC;
-static InvalidateSDFunc Win32SD_InvalidateSD;
+static LockFunc GDIWinSD_Lock;
+static GetRasInfoFunc GDIWinSD_GetRasInfo;
+static UnlockFunc GDIWinSD_Unlock;
+static DisposeFunc GDIWinSD_Dispose;
+static SetupFunc GDIWinSD_Setup;
+static GetDCFunc GDIWinSD_GetDC;
+static ReleaseDCFunc GDIWinSD_ReleaseDC;
+static InvalidateSDFunc GDIWinSD_InvalidateSD;
static HBRUSH nullbrush;
static HPEN nullpen;
@@ -56,11 +54,6 @@ static HPEN nullpen;
static jclass xorCompClass;
static jboolean beingShutdown = JNI_FALSE;
-extern BOOL isFullScreen; // TODO/REMIND : Multi-montior fullscreen support
-extern HWND hwndFullScreen;
-extern int gBackBufferCount;
-extern BOOL ddrawParamsChanged;
-extern int currNumDevices;
extern CriticalSection windowMoveLock;
extern "C"
@@ -72,7 +65,7 @@ int threadInfoIndex = TLS_OUT_OF_INDEXES;
static jclass threadClass = NULL;
static jmethodID currentThreadMethodID = NULL;
-void SetupThreadGraphicsInfo(JNIEnv *env, Win32SDOps *wsdo) {
+void SetupThreadGraphicsInfo(JNIEnv *env, GDIWinSDOps *wsdo) {
J2dTraceLn(J2D_TRACE_INFO, "SetupThreadGraphicsInfo");
// REMIND: handle error when creation fails
@@ -116,7 +109,7 @@ void SetupThreadGraphicsInfo(JNIEnv *env, Win32SDOps *wsdo) {
HDC hDC;
// This is a window surface
// First, init the HDC object
- AwtComponent *comp = Win32SurfaceData_GetComp(env, wsdo);
+ AwtComponent *comp = GDIWindowSurfaceData_GetComp(env, wsdo);
if (comp == NULL) {
return;
}
@@ -213,15 +206,15 @@ JNI_GetCurrentThread(JNIEnv *env) {
* NOTE: This function assumes that the SetupThreadGraphicsInfo()
* function has already been called for this situation (thread,
* window, etc.), so we can assume that the thread info contains
- * a valid hDC. This should usually be the case since Win32SD_Setup
+ * a valid hDC. This should usually be the case since GDIWinSD_Setup
* is called as part of the GetOps() process.
*/
ThreadGraphicsInfo *GetThreadGraphicsInfo(JNIEnv *env,
- Win32SDOps *wsdo) {
+ GDIWinSDOps *wsdo) {
return (ThreadGraphicsInfo*)TlsGetValue(threadInfoIndex);
}
-__inline HDC GetThreadDC(JNIEnv *env, Win32SDOps *wsdo) {
+__inline HDC GetThreadDC(JNIEnv *env, GDIWinSDOps *wsdo) {
ThreadGraphicsInfo *info =
(ThreadGraphicsInfo *)GetThreadGraphicsInfo(env, wsdo);
if (!info) {
@@ -238,14 +231,14 @@ __inline HDC GetThreadDC(JNIEnv *env, Win32SDOps *wsdo) {
* code.
*/
-static BOOL Win32SD_CheckMonitorArea(Win32SDOps *wsdo,
+static BOOL GDIWinSD_CheckMonitorArea(GDIWinSDOps *wsdo,
SurfaceDataBounds *bounds,
HDC hDC)
{
HWND hW = wsdo->window;
BOOL retCode = TRUE;
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_CheckMonitorArea");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_CheckMonitorArea");
int numScreens;
{
Devices::InstanceAccess devices;
@@ -286,17 +279,60 @@ initThreadInfoIndex()
}
}
+
+/**
+ * Utility function to make sure that native and java-level
+ * surface depths are matched. They can be mismatched when display-depths
+ * change, either between the creation of the Java surfaceData structure
+ * and the native ddraw surface, or later when a surface is automatically
+ * adjusted to be the new display depth (even if it was created in a different
+ * depth to begin with)
+ */
+BOOL SurfaceDepthsCompatible(int javaDepth, int nativeDepth)
+{
+ if (nativeDepth != javaDepth) {
+ switch (nativeDepth) {
+ case 0: // Error condition: something is wrong with the surface
+ case 8:
+ case 24:
+ // Java and native surface depths should match exactly for
+ // these cases
+ return FALSE;
+ break;
+ case 16:
+ // Java surfaceData should be 15 or 16 bits
+ if (javaDepth < 15 || javaDepth > 16) {
+ return FALSE;
+ }
+ break;
+ case 32:
+ // Could have this native depth for either 24- or 32-bit
+ // Java surfaceData
+ if (javaDepth != 24 && javaDepth != 32) {
+ return FALSE;
+ }
+ break;
+ default:
+ // should not get here, but if we do something is odd, so
+ // just register a failure
+ return FALSE;
+ }
+ }
+ return TRUE;
+}
+
+
/*
- * Class: sun_java2d_windows_Win32SurfaceData
+ * Class: sun_java2d_windows_GDIWindowSurfaceData
* Method: initIDs
* Signature: ()V
*/
JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32SurfaceData_initIDs(JNIEnv *env, jclass wsd,
+Java_sun_java2d_windows_GDIWindowSurfaceData_initIDs(JNIEnv *env, jclass wsd,
jclass XORComp)
{
jclass tc;
- J2dTraceLn(J2D_TRACE_INFO, "Win32SurfaceData_initIDs");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWindowSurfaceData_initIDs");
nullbrush = (HBRUSH) ::GetStockObject(NULL_BRUSH);
nullpen = (HPEN) ::GetStockObject(NULL_PEN);
@@ -314,43 +350,27 @@ Java_sun_java2d_windows_Win32SurfaceData_initIDs(JNIEnv *env, jclass wsd,
DASSERT(currentThreadMethodID != NULL);
}
-void Win32SD_RestoreSurface(JNIEnv *env, Win32SDOps *wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_RestoreSurface: restoring primary");
-
- if (!DDRestoreSurface(wsdo)) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "Win32SD_RestoreSurface: problems restoring primary");
- } else {
- // Force repaint of the window when we restore the surface
- ::InvalidateRect(wsdo->window, NULL, FALSE);
- }
-}
-
-
/*
- * Class: sun_java2d_windows_Win32SurfaceData
+ * Class: sun_java2d_windows_GDIWindowSurfaceData
* Method: initOps
* Signature: (Ljava/lang/Object;IIIIII)V
*/
JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32SurfaceData_initOps(JNIEnv *env, jobject wsd,
+Java_sun_java2d_windows_GDIWindowSurfaceData_initOps(JNIEnv *env, jobject wsd,
jobject peer, jint depth,
jint redMask, jint greenMask,
- jint blueMask, jint backBufferCount,
- jint screen)
+ jint blueMask, jint screen)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SurfaceData_initOps");
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_InitOps(env, wsd, sizeof(Win32SDOps));
- wsdo->sdOps.Lock = Win32SD_Lock;
- wsdo->sdOps.GetRasInfo = Win32SD_GetRasInfo;
- wsdo->sdOps.Unlock = Win32SD_Unlock;
- wsdo->sdOps.Dispose = Win32SD_Dispose;
- wsdo->sdOps.Setup = Win32SD_Setup;
- wsdo->RestoreSurface = Win32SD_RestoreSurface;
- wsdo->GetDC = Win32SD_GetDC;
- wsdo->ReleaseDC = Win32SD_ReleaseDC;
- wsdo->InvalidateSD = Win32SD_InvalidateSD;
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWindowSurfaceData_initOps");
+ GDIWinSDOps *wsdo = (GDIWinSDOps *)SurfaceData_InitOps(env, wsd, sizeof(GDIWinSDOps));
+ wsdo->sdOps.Lock = GDIWinSD_Lock;
+ wsdo->sdOps.GetRasInfo = GDIWinSD_GetRasInfo;
+ wsdo->sdOps.Unlock = GDIWinSD_Unlock;
+ wsdo->sdOps.Dispose = GDIWinSD_Dispose;
+ wsdo->sdOps.Setup = GDIWinSD_Setup;
+ wsdo->GetDC = GDIWinSD_GetDC;
+ wsdo->ReleaseDC = GDIWinSD_ReleaseDC;
+ wsdo->InvalidateSD = GDIWinSD_InvalidateSD;
wsdo->invalid = JNI_FALSE;
wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
wsdo->peer = env->NewWeakGlobalRef(peer);
@@ -379,97 +399,69 @@ Java_sun_java2d_windows_Win32SurfaceData_initOps(JNIEnv *env, jobject wsd,
wsdo->pixelStride = 4;
break;
}
- // Win32SurfaceData_GetWindow will throw NullPointerException
+ // GDIWindowSurfaceData_GetWindow will throw NullPointerException
// if wsdo->window is NULL
- wsdo->window = Win32SurfaceData_GetWindow(env, wsdo);
+ wsdo->window = GDIWindowSurfaceData_GetWindow(env, wsdo);
J2dTraceLn2(J2D_TRACE_VERBOSE, " wsdo=0x%x wsdo->window=0x%x",
wsdo, wsdo->window);
- wsdo->backBufferCount = backBufferCount;
{
Devices::InstanceAccess devices;
wsdo->device = devices->GetDeviceReference(screen, FALSE);
}
if (wsdo->device == NULL ||
- !DDSurfaceDepthsCompatible(depth, wsdo->device->GetBitDepth()))
+ !SurfaceDepthsCompatible(depth, wsdo->device->GetBitDepth()))
{
if (wsdo->device != NULL) {
J2dTraceLn2(J2D_TRACE_WARNING,
- "Win32SurfaceData_initOps: Surface depth mismatch: "\
+ "GDIWindowSurfaceData_initOps: Surface depth mismatch: "\
"wsdo->depth=%d device depth=%d. Surface invalidated.",
wsdo->depth, wsdo->device->GetBitDepth());
} else {
J2dTraceLn1(J2D_TRACE_WARNING,
- "Win32SurfaceData_initOps: Incorrect "\
+ "GDIWindowSurfaceData_initOps: Incorrect "\
"screen number (screen=%d). Surface invalidated.",
screen);
}
wsdo->invalid = JNI_TRUE;
- wsdo->lpSurface = NULL;
- wsdo->ddInstance = NULL;
-
- } else if (wsdo->window != NULL &&
- DDCanCreatePrimary((HMONITOR)wsdo->device->GetMonitor()))
- {
- // Create the surface on the windows event thread to avoid
- // problems with fullscreen window creation and manipulation
- if (!((BOOL)::SendMessage(wsdo->window, WM_AWT_DD_CREATE_SURFACE,
- (WPARAM)wsdo, NULL)))
- {
- // We only get here if the device can handle ddraw but
- // cannot create a primary. This can happen if, for example,
- // another application has exclusive access to the device.
- // In this situation, we null-out the ddraw-related pointers
- // which ends up noop'ing ddraw operations on this surface
- // (our window is effectively not visible in this situation,
- // so noops are fine).
- wsdo->lpSurface = NULL;
- wsdo->ddInstance = NULL;
- }
- }
- else {
- wsdo->lpSurface = NULL;
- wsdo->ddInstance = NULL;
}
wsdo->surfaceLock = new CriticalSection();
- wsdo->transparency = TR_OPAQUE;
wsdo->bitmap = NULL;
wsdo->bmdc = NULL;
wsdo->bmCopyToScreen = FALSE;
- wsdo->gdiOpPending = FALSE;
}
-JNIEXPORT Win32SDOps * JNICALL
-Win32SurfaceData_GetOps(JNIEnv *env, jobject sData)
+JNIEXPORT GDIWinSDOps * JNICALL
+GDIWindowSurfaceData_GetOps(JNIEnv *env, jobject sData)
{
SurfaceDataOps *ops = SurfaceData_GetOps(env, sData);
// REMIND: There was originally a condition check here to make sure
- // that we were really dealing with a Win32SurfaceData object, but
+ // that we were really dealing with a GDIWindowSurfaceData object, but
// it did not allow for the existence of other win32-accelerated
// surface data objects (e.g., Win32OffScreenSurfaceData). I've
// removed the check for now, but we should replace it with another,
// more general check against Win32-related surfaces.
- return (Win32SDOps *) ops;
+ return (GDIWinSDOps *) ops;
}
-JNIEXPORT Win32SDOps * JNICALL
-Win32SurfaceData_GetOpsNoSetup(JNIEnv *env, jobject sData)
+JNIEXPORT GDIWinSDOps * JNICALL
+GDIWindowSurfaceData_GetOpsNoSetup(JNIEnv *env, jobject sData)
{
// use the 'no setup' version of GetOps
SurfaceDataOps *ops = SurfaceData_GetOpsNoSetup(env, sData);
- return (Win32SDOps *) ops;
+ return (GDIWinSDOps *) ops;
}
JNIEXPORT AwtComponent * JNICALL
-Win32SurfaceData_GetComp(JNIEnv *env, Win32SDOps *wsdo)
+GDIWindowSurfaceData_GetComp(JNIEnv *env, GDIWinSDOps *wsdo)
{
PDATA pData;
jobject localObj = env->NewLocalRef(wsdo->peer);
if (localObj == NULL || (pData = JNI_GET_PDATA(localObj)) == NULL) {
J2dTraceLn1(J2D_TRACE_WARNING,
- "Win32SurfaceData_GetComp: Null pData? pData=0x%x",
+ "GDIWindowSurfaceData_GetComp: Null pData? pData=0x%x",
pData);
if (beingShutdown == JNI_TRUE) {
wsdo->invalid = JNI_TRUE;
@@ -484,7 +476,7 @@ Win32SurfaceData_GetComp(JNIEnv *env, Win32SDOps *wsdo)
}
if (wsdo->invalid == JNI_TRUE) {
SurfaceData_ThrowInvalidPipeException(env,
- "Win32SurfaceData: bounds changed");
+ "GDIWindowSurfaceData: bounds changed");
} else {
JNU_ThrowNullPointerException(env, "component argument pData");
}
@@ -494,22 +486,22 @@ Win32SurfaceData_GetComp(JNIEnv *env, Win32SDOps *wsdo)
}
JNIEXPORT HWND JNICALL
-Win32SurfaceData_GetWindow(JNIEnv *env, Win32SDOps *wsdo)
+GDIWindowSurfaceData_GetWindow(JNIEnv *env, GDIWinSDOps *wsdo)
{
HWND window = wsdo->window;
if (window == (HWND) NULL) {
- AwtComponent *comp = Win32SurfaceData_GetComp(env, wsdo);
+ AwtComponent *comp = GDIWindowSurfaceData_GetComp(env, wsdo);
if (comp == NULL) {
J2dTraceLn(J2D_TRACE_WARNING,
- "Win32SurfaceData_GetWindow: null component");
+ "GDIWindowSurfaceData_GetWindow: null component");
return (HWND) NULL;
}
comp->GetInsets(&wsdo->insets);
window = comp->GetHWnd();
if (::IsWindow(window) == FALSE) {
J2dRlsTraceLn(J2D_TRACE_ERROR,
- "Win32SurfaceData_GetWindow: disposed component");
+ "GDIWindowSurfaceData_GetWindow: disposed component");
JNU_ThrowNullPointerException(env, "disposed component");
return (HWND) NULL;
}
@@ -521,13 +513,13 @@ Win32SurfaceData_GetWindow(JNIEnv *env, Win32SDOps *wsdo)
} /* extern "C" */
-static jboolean Win32SD_SimpleClip(JNIEnv *env, Win32SDOps *wsdo,
+static jboolean GDIWinSD_SimpleClip(JNIEnv *env, GDIWinSDOps *wsdo,
SurfaceDataBounds *bounds,
HDC hDC)
{
RECT rClip;
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_SimpleClip");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_SimpleClip");
if (hDC == NULL) {
return JNI_FALSE;
}
@@ -557,7 +549,7 @@ static jboolean Win32SD_SimpleClip(JNIEnv *env, Win32SDOps *wsdo,
int clipStatus = ::GetClipRgn(hDC, rgnSave);
if (-1 == clipStatus) {
J2dTraceLn(J2D_TRACE_WARNING,
- "Win32SD_SimpleClip: failed due to clip status");
+ "GDIWinSD_SimpleClip: failed due to clip status");
::DeleteObject(rgnSave);
return JNI_FALSE;
}
@@ -577,7 +569,7 @@ static jboolean Win32SD_SimpleClip(JNIEnv *env, Win32SDOps *wsdo,
// the clipping article
if (SIMPLEREGION != nComplexity) {
J2dTraceLn(J2D_TRACE_WARNING,
- "Win32SD_SimpleClip: failed due to complexity");
+ "GDIWinSD_SimpleClip: failed due to complexity");
return JNI_FALSE;
}
}
@@ -610,7 +602,7 @@ static jboolean Win32SD_SimpleClip(JNIEnv *env, Win32SDOps *wsdo,
// Perhaps we need a new lock code that will indicate that we
// shouldn't bother drawing?
J2dTraceLn(J2D_TRACE_WARNING,
- "Win32SD_SimpleClip: failed due to empty bounds");
+ "GDIWinSD_SimpleClip: failed due to empty bounds");
return JNI_FALSE;
}
break;
@@ -618,7 +610,7 @@ static jboolean Win32SD_SimpleClip(JNIEnv *env, Win32SDOps *wsdo,
case ERROR:
default:
J2dTraceLn1(J2D_TRACE_ERROR,
- "Win32SD_SimpleClip: failed due to incorrect complexity=%d",
+ "GDIWinSD_SimpleClip: failed due to incorrect complexity=%d",
nComplexity);
return JNI_FALSE;
}
@@ -626,60 +618,15 @@ static jboolean Win32SD_SimpleClip(JNIEnv *env, Win32SDOps *wsdo,
return JNI_TRUE;
}
-static void Win32SD_TryLockByDD(JNIEnv *env, Win32SDOps *wsdo,
- SurfaceDataRasInfo *pRasInfo, HDC hDC)
-{
- RECT winrect;
-
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_LockByDD");
-
- ::GetWindowRect(wsdo->window, &winrect);
- ::OffsetRect(&winrect, wsdo->insets.left, wsdo->insets.top);
-
- /* On NT-based OS's, winddow can move even while we have the primary
- * surface locked. Must prevent this or else we may render to the
- * wrong place on the screen. */
- windowMoveLock.Enter();
-
- if (!DDLock(env, wsdo, NULL, pRasInfo)) {
- windowMoveLock.Leave();
- J2dTraceLn(J2D_TRACE_WARNING,
- "Win32SD_TryLockByDD: DDLock failed");
- return;
- }
-
- // If DD lock successful, update window location in wsdo
- ::GetWindowRect(wsdo->window, &winrect);
- wsdo->x = winrect.left;
- wsdo->y = winrect.top;
- if (currNumDevices > 1) {
- MONITOR_INFO *mi = wsdo->device->GetMonitorInfo();
- wsdo->x -= mi->rMonitor.left;
- wsdo->y -= mi->rMonitor.top;
- }
- if (!Win32SD_SimpleClip(env, wsdo, &pRasInfo->bounds, hDC) ||
- !Win32SD_CheckMonitorArea(wsdo, &pRasInfo->bounds, hDC))
- {
- DDUnlock(env, wsdo);
- windowMoveLock.Leave();
- J2dTraceLn(J2D_TRACE_WARNING, "Win32SD_TryLockByDD: failed because "\
- "of clip, cursor or monitor area");
- return;
- }
-
- wsdo->lockType = WIN32SD_LOCK_BY_DDRAW;
- J2dTraceLn(J2D_TRACE_VERBOSE, "Win32SD_TryLockByDD: succeeded");
-}
-
-static jint Win32SD_Lock(JNIEnv *env,
+static jint GDIWinSD_Lock(JNIEnv *env,
SurfaceDataOps *ops,
SurfaceDataRasInfo *pRasInfo,
jint lockflags)
{
- Win32SDOps *wsdo = (Win32SDOps *) ops;
+ GDIWinSDOps *wsdo = (GDIWinSDOps *) ops;
int ret = SD_SUCCESS;
HDC hDC;
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_Lock");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_Lock");
/* This surfaceLock replaces an earlier implementation which used a
monitor associated with the peer. That implementation was prone
@@ -698,11 +645,11 @@ static jint Win32SD_Lock(JNIEnv *env,
code at the mercy of driver bugs.*/
wsdo->surfaceLock->Enter();
if (wsdo->invalid == JNI_TRUE) {
- J2dTraceLn(J2D_TRACE_WARNING, "Win32SD_Lock: surface is invalid");
+ J2dTraceLn(J2D_TRACE_WARNING, "GDIWinSD_Lock: surface is invalid");
wsdo->surfaceLock->Leave();
if (beingShutdown != JNI_TRUE) {
SurfaceData_ThrowInvalidPipeException(env,
- "Win32SurfaceData: bounds changed");
+ "GDIWindowSurfaceData: bounds changed");
}
return SD_FAILURE;
}
@@ -749,33 +696,22 @@ static jint Win32SD_Lock(JNIEnv *env,
bounds->y2 = crect.bottom;
}
- if (useDDLock && DDUseDDraw(wsdo)) {
- Win32SD_TryLockByDD(env, wsdo, pRasInfo, hDC);
- }
- if (wsdo->lockType == WIN32SD_LOCK_UNLOCKED) {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "Win32SD_Lock: dd lock failed, try to lock by DIB");
- if (bounds->x2 > bounds->x1 && bounds->y2 > bounds->y1) {
- wsdo->lockType = WIN32SD_LOCK_BY_DIB;
- if (lockflags & SD_LOCK_FASTEST) {
- ret = SD_SLOWLOCK;
- }
- J2dTraceLn(J2D_TRACE_VERBOSE, " locked by DIB");
- } else {
- wsdo->ReleaseDC(env, wsdo, hDC);
- wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
- wsdo->surfaceLock->Leave();
- ret = SD_FAILURE;
- J2dTraceLn(J2D_TRACE_ERROR,
- "Win32SD_Lock: error locking by DIB");
+ if (bounds->x2 > bounds->x1 && bounds->y2 > bounds->y1) {
+ wsdo->lockType = WIN32SD_LOCK_BY_DIB;
+ if (lockflags & SD_LOCK_FASTEST) {
+ ret = SD_SLOWLOCK;
}
+ J2dTraceLn(J2D_TRACE_VERBOSE, " locked by DIB");
} else {
- J2dTraceLn(J2D_TRACE_VERBOSE, "Win32SD_Lock: locked by DDraw");
- // Surface is already locked - release DC
wsdo->ReleaseDC(env, wsdo, hDC);
+ wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
+ wsdo->surfaceLock->Leave();
+ ret = SD_FAILURE;
+ J2dTraceLn(J2D_TRACE_ERROR,
+ "GDIWinSD_Lock: error locking by DIB");
}
} else {
- J2dTraceLn(J2D_TRACE_VERBOSE, "Win32SD_Lock: surface wasn't locked");
+ J2dTraceLn(J2D_TRACE_VERBOSE, "GDIWinSD_Lock: surface wasn't locked");
/* They didn't lock for anything - we won't give them anything */
wsdo->ReleaseDC(env, wsdo, hDC);
wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
@@ -787,13 +723,13 @@ static jint Win32SD_Lock(JNIEnv *env,
return ret;
}
-static void Win32SD_GetRasInfo(JNIEnv *env,
+static void GDIWinSD_GetRasInfo(JNIEnv *env,
SurfaceDataOps *ops,
SurfaceDataRasInfo *pRasInfo)
{
- Win32SDOps *wsdo = (Win32SDOps *) ops;
+ GDIWinSDOps *wsdo = (GDIWinSDOps *) ops;
jint lockflags = wsdo->lockFlags;
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_GetRasInfo");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_GetRasInfo");
HDC hDC = GetThreadDC(env, wsdo);
if (wsdo->lockType == WIN32SD_LOCK_UNLOCKED) {
@@ -801,22 +737,7 @@ static void Win32SD_GetRasInfo(JNIEnv *env,
return;
}
- if (useDDLock && DDUseDDraw(wsdo) &&
- wsdo->lockType == WIN32SD_LOCK_BY_DIB &&
- (lockflags & SD_LOCK_FASTEST))
- {
- Win32SD_TryLockByDD(env, wsdo, pRasInfo, hDC);
- if (wsdo->lockType == WIN32SD_LOCK_BY_DDRAW) {
- wsdo->ReleaseDC(env, wsdo, hDC);
- }
- }
-
- if (wsdo->lockType == WIN32SD_LOCK_BY_DDRAW) {
- // Adjust the rasBase to point to the upper left pixel of our drawing area
- pRasInfo->rasBase = (void *) ((intptr_t)pRasInfo->rasBase +
- (wsdo->x * pRasInfo->pixelStride) +
- (wsdo->y * pRasInfo->scanStride));
- } else if (wsdo->lockType == WIN32SD_LOCK_BY_DIB) {
+ if (wsdo->lockType == WIN32SD_LOCK_BY_DIB) {
int x, y, w, h;
int pixelStride = wsdo->pixelStride;
// do not subtract insets from x,y as we take care of it in SD_GetDC
@@ -846,7 +767,7 @@ static void Win32SD_GetRasInfo(JNIEnv *env,
if (wsdo->bitmap) {
// delete old objects
J2dTraceLn(J2D_TRACE_VERBOSE,
- "Win32SD_GetRasInfo: recreating GDI bitmap");
+ "GDIWinSD_GetRasInfo: recreating GDI bitmap");
if (wsdo->bmdc) { // should not be null
::SelectObject(wsdo->bmdc, wsdo->oldmap);
::DeleteDC(wsdo->bmdc);
@@ -963,7 +884,7 @@ static void Win32SD_GetRasInfo(JNIEnv *env,
}
}
-static void Win32SD_Setup(JNIEnv *env,
+static void GDIWinSD_Setup(JNIEnv *env,
SurfaceDataOps *ops)
{
// Call SetupTGI to ensure that this thread already has a DC that is
@@ -973,16 +894,16 @@ static void Win32SD_Setup(JNIEnv *env,
// Note that calling SetupTGI here means that anybody needing a DC
// later in this rendering process need only call GetTGI, which
// assumes that the TGI structure is valid for this thread/window.
- SetupThreadGraphicsInfo(env, (Win32SDOps*)ops);
+ SetupThreadGraphicsInfo(env, (GDIWinSDOps*)ops);
}
-static void Win32SD_Unlock(JNIEnv *env,
+static void GDIWinSD_Unlock(JNIEnv *env,
SurfaceDataOps *ops,
SurfaceDataRasInfo *pRasInfo)
{
- Win32SDOps *wsdo = (Win32SDOps *) ops;
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_Unlock");
+ GDIWinSDOps *wsdo = (GDIWinSDOps *) ops;
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_Unlock");
HDC hDC = GetThreadDC(env, wsdo);
if (wsdo->lockType == WIN32SD_LOCK_UNLOCKED) {
@@ -993,14 +914,10 @@ static void Win32SD_Unlock(JNIEnv *env,
return;
}
- if (wsdo->lockType == WIN32SD_LOCK_BY_DDRAW) {
- DDUnlock(env, wsdo);
- windowMoveLock.Leave();
- wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
- } else if (wsdo->lockType == WIN32SD_LOCK_BY_DIB) {
+ if (wsdo->lockType == WIN32SD_LOCK_BY_DIB) {
if (wsdo->lockFlags & SD_LOCK_WRITE) {
J2dTraceLn(J2D_TRACE_VERBOSE,
- "Win32SD_Unlock: do Blt of the bitmap");
+ "GDIWinSD_Unlock: do Blt of the bitmap");
if (wsdo->bmCopyToScreen && ::IsWindowVisible(wsdo->window)) {
// Don't bother copying to screen if our window has gone away
// or if the bitmap was not actually written to during this
@@ -1035,7 +952,7 @@ static void Win32SD_Unlock(JNIEnv *env,
#define COLORFOR(c) (PALETTERGB(((c)>>16)&0xff,((c)>>8)&0xff,((c)&0xff)))
-COLORREF CheckGrayColor(Win32SDOps *wsdo, int c) {
+COLORREF CheckGrayColor(GDIWinSDOps *wsdo, int c) {
if (wsdo->device->GetGrayness() != GS_NOTGRAY) {
int g = (77 *(c & 0xFF) +
150*((c >> 8) & 0xFF) +
@@ -1045,11 +962,11 @@ COLORREF CheckGrayColor(Win32SDOps *wsdo, int c) {
return COLORFOR(c);
}
-static HDC Win32SD_GetDC(JNIEnv *env, Win32SDOps *wsdo,
+static HDC GDIWinSD_GetDC(JNIEnv *env, GDIWinSDOps *wsdo,
jint type, jint *patrop,
jobject clip, jobject comp, jint color)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_GetDC");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_GetDC");
if (wsdo->invalid == JNI_TRUE) {
if (beingShutdown != JNI_TRUE) {
@@ -1059,16 +976,16 @@ static HDC Win32SD_GetDC(JNIEnv *env, Win32SDOps *wsdo,
}
ThreadGraphicsInfo *info = GetThreadGraphicsInfo(env, wsdo);
- Win32SD_InitDC(env, wsdo, info, type, patrop, clip, comp, color);
+ GDIWinSD_InitDC(env, wsdo, info, type, patrop, clip, comp, color);
return info->hDC;
}
JNIEXPORT void JNICALL
-Win32SD_InitDC(JNIEnv *env, Win32SDOps *wsdo, ThreadGraphicsInfo *info,
+GDIWinSD_InitDC(JNIEnv *env, GDIWinSDOps *wsdo, ThreadGraphicsInfo *info,
jint type, jint *patrop,
jobject clip, jobject comp, jint color)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_InitDC");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_InitDC");
// init clip
if (clip == NULL) {
@@ -1222,23 +1139,18 @@ Win32SD_InitDC(JNIEnv *env, Win32SDOps *wsdo, ThreadGraphicsInfo *info,
}
}
-static void Win32SD_ReleaseDC(JNIEnv *env, Win32SDOps *wsdo, HDC hDC)
+static void GDIWinSD_ReleaseDC(JNIEnv *env, GDIWinSDOps *wsdo, HDC hDC)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_ReleaseDC");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_ReleaseDC");
// Don't actually do anything here: every thread holds its own
// wsdo-specific DC until the thread goes away or the wsdo
// is disposed.
-
- // we don't track gdi operations for on-screen surfaces
- // by setting wsdo->gdiOpPending to TRUE here
- // because we do not need an extra sync before DD-locking
- // the on-screen surface (sync is a dd-lock)
}
-static void Win32SD_InvalidateSD(JNIEnv *env, Win32SDOps *wsdo)
+static void GDIWinSD_InvalidateSD(JNIEnv *env, GDIWinSDOps *wsdo)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_InvalidateSD");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_InvalidateSD");
J2dTraceLn2(J2D_TRACE_VERBOSE, " wsdo=0x%x wsdo->window=0x%x",
wsdo, wsdo->window);
@@ -1248,14 +1160,14 @@ static void Win32SD_InvalidateSD(JNIEnv *env, Win32SDOps *wsdo)
/*
- * Method: Win32SD_Dispose
+ * Method: GDIWinSD_Dispose
*/
static void
-Win32SD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
+GDIWinSD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SD_Dispose");
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWinSD_Dispose");
// ops is assumed non-null as it is checked in SurfaceData_DisposeOps
- Win32SDOps *wsdo = (Win32SDOps*)ops;
+ GDIWinSDOps *wsdo = (GDIWinSDOps*)ops;
if (wsdo->bitmap) {
// delete old objects
J2dTraceLn(J2D_TRACE_VERBOSE, " disposing the GDI bitmap");
@@ -1267,7 +1179,6 @@ Win32SD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
::DeleteObject(wsdo->bitmap);
wsdo->bitmap = 0;
}
- DDDestroySurface(wsdo);
env->DeleteWeakGlobalRef(wsdo->peer);
if (wsdo->device != NULL) {
wsdo->device->Release();
@@ -1278,67 +1189,16 @@ Win32SD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
/*
- * Class: sun_java2d_windows_Win32SurfaceData
+ * Class: sun_java2d_windows_GDIWindowSurfaceData
* Method: invalidateSD
* Signature: ()V
*/
JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32SurfaceData_invalidateSD(JNIEnv *env, jobject wsd)
+Java_sun_java2d_windows_GDIWindowSurfaceData_invalidateSD(JNIEnv *env, jobject wsd)
{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SurfaceData_invalidateSD");
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, wsd);
+ J2dTraceLn(J2D_TRACE_INFO, "GDIWindowSurfaceData_invalidateSD");
+ GDIWinSDOps *wsdo = GDIWindowSurfaceData_GetOpsNoSetup(env, wsd);
if (wsdo != NULL) {
wsdo->InvalidateSD(env, wsdo);
}
}
-
-/*
- * Class: sun_java2d_windows_Win32SurfaceData
- * Method: restoreSurface
- * Signature: ()V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32SurfaceData_restoreSurface(JNIEnv *env,
- jobject sData)
-{
- J2dTraceLn(J2D_TRACE_INFO,
- "Win32SurfaceData_restoreSurface: restoring primary");
- Win32SDOps *wsdo = Win32SurfaceData_GetOpsNoSetup(env, sData);
-
- // Attempt to restore and lock the surface (to make sure the restore worked)
- if (DDRestoreSurface(wsdo) && DDLock(env, wsdo, NULL, NULL)) {
- DDUnlock(env, wsdo);
- } else {
- // Failure - throw exception
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "Win32SurfaceData_restoreSurface: "\
- "problems restoring primary");
- SurfaceData_ThrowInvalidPipeException(env, "RestoreSurface failure");
- }
-}
-
-/*
- * Class: sun_java2d_windows_Win32SurfaceData
- * Method: flip
- * Signature: (Lsun/awt/windows/Win32SurfaceData;)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32SurfaceData_flip(JNIEnv *env,
- jobject sData, jobject dData)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32SurfaceData_flip");
- if (dData == NULL) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "Win32SurfaceData_flip: dData is null");
- SurfaceData_ThrowInvalidPipeException(env,
- "Could not flip offscreen Surface");
- return;
- }
- Win32SDOps *wsdo_s = Win32SurfaceData_GetOpsNoSetup(env, sData);
- Win32SDOps *wsdo_d = Win32SurfaceData_GetOpsNoSetup(env, dData);
- if (!DDFlip(env, wsdo_s, wsdo_d)) {
- J2dTraceLn(J2D_TRACE_WARNING, "Win32SurfaceData_flip: flipping error");
- SurfaceData_ThrowInvalidPipeException(env,
- "Could not flip offscreen Surface");
- }
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/Win32SurfaceData.h b/jdk/src/windows/native/sun/java2d/windows/GDIWindowSurfaceData.h
similarity index 72%
rename from jdk/src/windows/native/sun/java2d/windows/Win32SurfaceData.h
rename to jdk/src/windows/native/sun/java2d/windows/GDIWindowSurfaceData.h
index 425a5b48f5a..6c37c8e2aea 100644
--- a/jdk/src/windows/native/sun/java2d/windows/Win32SurfaceData.h
+++ b/jdk/src/windows/native/sun/java2d/windows/GDIWindowSurfaceData.h
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -35,7 +35,6 @@
#include "awt_Win32GraphicsDevice.h"
#include "stdhdrs.h"
-#include
#define TEST_SURFACE_BITS(a,f) (((a)&(f)) == (f))
@@ -45,39 +44,7 @@
* SurfaceData interface to talk to a Win32 drawable from native code.
*/
-typedef struct _Win32SDOps Win32SDOps;
-
-/* These defined in ddrawObject.h */
-class DDraw;
-class DDrawSurface;
-class DDrawClipper;
-class DDrawDisplayMode;
-
-/*
- * Typedef for the structure which contains the ddObject reference, as
- * well as a refCount so that we know when we can delete the ddObject
- * and this structure. We have to keep around the old ddObject even
- * after creating a new one because there may be multiple other surfaces
- * that still reference the old ddObject. When all of those surfaces
- * have been recreated with the new ddObject, the old one can be
- * released.
- */
-typedef struct {
- DDraw *ddObject;
- int refCount;
- DDrawSurface *primary;
- DDrawSurface *syncSurface;
- DDrawClipper *clipper;
- CriticalSection *primaryLock;
- BOOL valid;
- HMONITOR hMonitor;
- BOOL capsSet;
- BOOL canBlt;
- HWND hwndFullScreen;
- int backBufferCount;
- int context;
- BOOL accelerated;
-} DDrawObjectStruct;
+typedef struct _GDIWinSDOps GDIWinSDOps;
#define CONTEXT_NORMAL 0
#define CONTEXT_DISPLAY_CHANGE 1
@@ -101,7 +68,7 @@ typedef struct {
/*
* This function retrieves an HDC for rendering to the destination
- * managed by the indicated Win32SDOps structure.
+ * managed by the indicated GDIWinSDOps structure.
*
* The env parameter should be the JNIEnv of the surrounding JNI context.
*
@@ -134,7 +101,7 @@ typedef struct {
* GetStringCritical locks which have not been released.
*/
typedef HDC GetDCFunc(JNIEnv *env,
- Win32SDOps *wsdo,
+ GDIWinSDOps *wsdo,
jint flags,
jint *patrop,
jobject clip,
@@ -143,7 +110,7 @@ typedef HDC GetDCFunc(JNIEnv *env,
/*
* This function releases an HDC that was retrieved from the GetDC
- * function of the indicated Win32SDOps structure.
+ * function of the indicated GDIWinSDOps structure.
*
* The env parameter should be the JNIEnv of the surrounding JNI context.
*
@@ -159,27 +126,13 @@ typedef HDC GetDCFunc(JNIEnv *env,
* GetStringCritical locks which have not been released.
*/
typedef void ReleaseDCFunc(JNIEnv *env,
- Win32SDOps *wsdo,
+ GDIWinSDOps *wsdo,
HDC hdc);
typedef void InvalidateSDFunc(JNIEnv *env,
- Win32SDOps *wsdo);
+ GDIWinSDOps *wsdo);
-typedef void RestoreSurfaceFunc(JNIEnv *env, Win32SDOps *wsdo);
-
-#define READS_PUNT_THRESHOLD 2
-#define BLTS_UNPUNT_THRESHOLD 4
-typedef struct {
- jboolean disablePunts;
- jint pixelsReadThreshold;
- jint numBltsThreshold;
- jboolean usingDDSystem;
- DDrawSurface *lpSurfaceSystem;
- DDrawSurface *lpSurfaceVram;
- jint numBltsSinceRead;
- jint pixelsReadSinceBlt;
-} SurfacePuntData;
/*
* A structure that holds all state global to the native surfaceData
* object.
@@ -193,7 +146,7 @@ typedef struct {
* window field remains because once it is set for a given wsdo
* structure it stays the same until that structure is destroyed.
*/
-struct _Win32SDOps {
+struct _GDIWinSDOps {
SurfaceDataOps sdOps;
jboolean invalid;
GetDCFunc *GetDC;
@@ -223,23 +176,13 @@ struct _Win32SDOps {
jint penclr;
int x, y, w, h; // REMIND: store in TLS
- RestoreSurfaceFunc *RestoreSurface;
- DDrawSurface *lpSurface;
- int backBufferCount;
- DDrawObjectStruct *ddInstance;
CriticalSection *surfaceLock; // REMIND: try to remove
- jboolean surfaceLost;
- jboolean gdiOpPending; // whether a GDI operation is pending for this
- // surface (Get/ReleaseDC were called)
- jint transparency;
AwtWin32GraphicsDevice *device;
- SurfacePuntData surfacePuntData;
};
#define WIN32SD_LOCK_UNLOCKED 0 /* surface is not locked */
#define WIN32SD_LOCK_BY_NULL 1 /* surface locked for NOP */
-#define WIN32SD_LOCK_BY_DDRAW 2 /* surface locked by DirectDraw */
-#define WIN32SD_LOCK_BY_DIB 3 /* surface locked by BitBlt */
+#define WIN32SD_LOCK_BY_DIB 2 /* surface locked by BitBlt */
extern "C" {
@@ -248,7 +191,7 @@ extern "C" {
*/
typedef struct {
HDC hDC;
- Win32SDOps *wsdo;
+ GDIWinSDOps *wsdo;
RECT bounds;
jobject clip;
jobject comp;
@@ -263,10 +206,10 @@ typedef struct {
/*
- * This function returns a pointer to a native Win32SDOps structure
+ * This function returns a pointer to a native GDIWinSDOps structure
* for accessing the indicated Win32 SurfaceData Java object. It
* verifies that the indicated SurfaceData object is an instance
- * of Win32SurfaceData before returning and will return NULL if the
+ * of GDIWindowSurfaceData before returning and will return NULL if the
* wrong SurfaceData object is being accessed. This function will
* throw the appropriate Java exception if it returns NULL so that
* the caller can simply return.
@@ -280,22 +223,22 @@ typedef struct {
* is called since this function will not leave any outstanding
* JNI Critical locks unreleased.
*/
-JNIEXPORT Win32SDOps * JNICALL
-Win32SurfaceData_GetOps(JNIEnv *env, jobject sData);
+JNIEXPORT GDIWinSDOps * JNICALL
+GDIWindowSurfaceData_GetOps(JNIEnv *env, jobject sData);
-JNIEXPORT Win32SDOps * JNICALL
-Win32SurfaceData_GetOpsNoSetup(JNIEnv *env, jobject sData);
+JNIEXPORT GDIWinSDOps * JNICALL
+GDIWindowSurfaceData_GetOpsNoSetup(JNIEnv *env, jobject sData);
JNIEXPORT HWND JNICALL
-Win32SurfaceData_GetWindow(JNIEnv *env, Win32SDOps *wsdo);
+GDIWindowSurfaceData_GetWindow(JNIEnv *env, GDIWinSDOps *wsdo);
JNIEXPORT void JNICALL
-Win32SD_InitDC(JNIEnv *env, Win32SDOps *wsdo, ThreadGraphicsInfo *info,
+GDIWinSD_InitDC(JNIEnv *env, GDIWinSDOps *wsdo, ThreadGraphicsInfo *info,
jint type, jint *patrop,
jobject clip, jobject comp, jint color);
JNIEXPORT AwtComponent * JNICALL
-Win32SurfaceData_GetComp(JNIEnv *env, Win32SDOps *wsdo);
+GDIWindowSurfaceData_GetComp(JNIEnv *env, GDIWinSDOps *wsdo);
} /* extern "C" */
diff --git a/jdk/src/windows/native/sun/java2d/windows/RegistryKey.cpp b/jdk/src/windows/native/sun/java2d/windows/RegistryKey.cpp
deleted file mode 100644
index 148fd2f411b..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/RegistryKey.cpp
+++ /dev/null
@@ -1,270 +0,0 @@
-/*
- * Copyright 2003 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-/**
- * This class encapsulates simple interaction with the Windows Registry.
- * Use of the class should generally follow one of two forms:
- * (1) Need to get set just one value:
- * int val = RegistryKey::GetIntValue(keyName, valueName);
- * This function creates a temporary registry key object, reads the value
- * from it, and closes the key.
- * (2) Need to get/set several values:
- * {
- * RegistryKey key(keyName, permissions);
- * int val = key.GetIntValue(valueName);
- * // other key operations
- * }
- * Upon going out of scope, the RegistryKey object is automatically disposed,
- * which closes the key. This is important: if you instead create an instance
- * like this:
- * RegistryKey *key = new RegistryKey(keyName, permissions);
- * then you need to remember to delete that object, else you will leave a
- * registry key open, which could cause various problems such as leaks
- * and synchronization.
- *
- * One important item implemented in this class is the ability to force
- * a flush during a registry set operation. This was implemented because
- * the primary usage for the registry at this time is in storing results
- * of testing; if we happen to crash (the application or system) during the
- * tests, we want to ensure that that information was recorded. If we
- * rely on the default lazy behavior of the registry, then we have no way of
- * knowing whether our last settings into the registry were recorded before
- * the process died.
- */
-
-#include
-#include
-#include "Trace.h"
-#include "WindowsFlags.h"
-#include "RegistryKey.h"
-
-/**
- * Constructs a registry key object. permissions can be any of the
- * allowable values for keys, but are generally KEY_WRITE or
- * KEY_QUERY_VALUE. If the key does not yet exist in the registry, it
- * will be created here.
- * Note that we use HKEY_CURRENT_USER as the registry hierarchy; this is
- * because we want any user (restricted or administrator) to be able to
- * read and write these test results; storing the results in a more central
- * location (e.g., HKEY_LOCAL_MACHINE) would prevent usage by users without
- * permission to read and write in that registry hierarchy.
- */
-RegistryKey::RegistryKey(WCHAR *keyName, REGSAM permissions)
-{
- hKey = NULL; // default value
- if (disableRegistry) {
- return;
- }
- DWORD disposition;
- DWORD ret = RegCreateKeyEx(HKEY_CURRENT_USER, keyName, 0, 0,
- REG_OPTION_NON_VOLATILE, permissions,
- NULL, &hKey, &disposition);
- if (ret != ERROR_SUCCESS) {
- PrintRegistryError(ret, "RegCreateKeyEx");
- }
-}
-
-/**
- * Destruction of the registry key object; this closes the key if
- * if was opened.
- */
-RegistryKey::~RegistryKey()
-{
- if (hKey) {
- RegCloseKey(hKey);
- }
-}
-
-DWORD RegistryKey::EnumerateSubKeys(DWORD index, WCHAR *subKeyName,
- DWORD *buffSize)
-{
- if (disableRegistry) {
- // truncate the enumeration
- return ERROR_NO_MORE_ITEMS;
- }
- FILETIME lastWriteTime;
- return RegEnumKeyEx(hKey, index, subKeyName, buffSize, NULL, NULL, NULL,
- &lastWriteTime);
-}
-
-/**
- * Retrieves the value of the given parameter from the registry.
- * If no such value exists in the registry, it returns the default
- * value of J2D_ACCEL_UNVERIFIED.
- */
-int RegistryKey::GetIntValue(WCHAR *valueName)
-{
- DWORD valueLength = 4;
- int regValue = J2D_ACCEL_UNVERIFIED;
- if (!disableRegistry) {
- RegQueryValueEx(hKey, valueName, NULL, NULL, (LPBYTE) & regValue,
- & valueLength);
- }
- // QueryValue could fail if value does not exist, but in this
- // case regValue still equals the UNVERIFIED state, so no need to
- // catch failure.
- return regValue;
-}
-
-/**
- * Static method which opens a registry key with the given keyName and
- * calls GetIntValue(valueName) on that key.
- */
-int RegistryKey::GetIntValue(WCHAR *keyName, WCHAR *valueName)
-{
- RegistryKey key(keyName, KEY_QUERY_VALUE);
- return key.GetIntValue(valueName);
-}
-
-/**
- * Sets the specified value in the given key, returning TRUE for
- * success and FALSE for failure (errors are not expected in
- * this function and indicate some unknown problem with registry
- * interaction). The flush parameter indicates that we should force
- * the registry to record this value after setting it (as opposed
- * to allowing the registry to write the value lazily).
- */
-BOOL RegistryKey::SetIntValue(WCHAR *valueName, int regValue, BOOL flush)
-{
- if (disableRegistry) {
- return TRUE;
- }
- if (!hKey) {
- PrintRegistryError(0, "Null hKey in SetIntValue");
- return FALSE;
- }
- DWORD valueLength = 4;
- DWORD ret = RegSetValueEx(hKey, valueName, 0, REG_DWORD, (LPBYTE)®Value,
- valueLength);
- if (ret != ERROR_SUCCESS) {
- PrintRegistryError(ret, "RegSetValueEx");
- return FALSE;
- }
- if (flush) {
- ret = RegFlushKey(hKey);
- if (ret != ERROR_SUCCESS) {
- PrintRegistryError(ret, "RegFlushKey");
- return FALSE;
- }
- }
- return TRUE;
-}
-
-/**
- * Static method which opens a registry key with the given keyName and
- * calls SetIntValue(valueName, regValue, flush) on that key.
- */
-BOOL RegistryKey::SetIntValue(WCHAR *keyName, WCHAR *valueName,
- int regValue, BOOL flush)
-{
- RegistryKey key(keyName, KEY_WRITE);
- return key.SetIntValue(valueName, regValue, flush);
-}
-
-/**
- * Deletes the key with the given key name. This is useful when using
- * the -Dsun.java2d.accelReset flag, which resets the registry values
- * to force the startup tests to be rerun and re-recorded.
- *
- */
-void RegistryKey::DeleteKey(WCHAR *keyName)
-{
- if (disableRegistry) {
- return;
- }
- // We should be able to do this with the ShDeleteKey() function, but
- // that is apparently not available on the ia64 sdk, so we revert back
- // to recursively deleting all subkeys until we can delete the key in
- // question
- DWORD buffSize = 1024;
- WCHAR subKeyName[1024];
- int subKeyIndex = 0;
- FILETIME lastWriteTime;
- HKEY hKey;
- DWORD ret = RegOpenKeyEx(HKEY_CURRENT_USER, keyName, 0, KEY_ALL_ACCESS,
- &hKey);
- if (ret != ERROR_SUCCESS) {
- PrintRegistryError(ret, "DeleteKey, during RegOpenKeyEx");
- }
- while ((ret = RegEnumKeyEx(hKey, subKeyIndex, subKeyName, &buffSize,
- NULL, NULL, NULL, &lastWriteTime)) ==
- ERROR_SUCCESS)
- {
- WCHAR subKeyBuffer[1024];
- swprintf(subKeyBuffer, L"%s\\%s", keyName, subKeyName);
- DeleteKey(subKeyBuffer);
- ++subKeyIndex;
- buffSize = 1024;
- }
- ret = RegCloseKey(hKey);
- if (ret != ERROR_SUCCESS) {
- PrintRegistryError(ret, "DeleteKey, during RegCloseKey");
- }
- ret = RegDeleteKey(HKEY_CURRENT_USER, keyName);
- if (ret != ERROR_SUCCESS) {
- PrintRegistryError(ret, "DeleteKey, during RegDeleteKey");
- }
-}
-
-void RegistryKey::PrintValue(WCHAR *keyName, WCHAR *valueName,
- WCHAR *msg)
-{
- int value = GetIntValue(keyName, valueName);
- switch (value) {
- case J2D_ACCEL_UNVERIFIED:
- printf("%S: %s\n", msg, "UNVERIFIED");
- break;
- case J2D_ACCEL_TESTING:
- printf("%S: %s\n", msg, "TESTING (may indicate crash during test)");
- break;
- case J2D_ACCEL_FAILURE:
- printf("%S: %s\n", msg, "FAILURE");
- break;
- case J2D_ACCEL_SUCCESS:
- printf("%S: %s\n", msg, "SUCCESS");
- break;
- default:
- printf("No registry value for key, value %S, %S\n",
- keyName, valueName);
- break;
- }
-}
-
-/**
- * Debugging utility: prints information about errors received
- * during interaction with the registry.
- */
-void RegistryKey::PrintRegistryError(LONG errNum, char *message)
-{
- WCHAR errString[255];
- int numChars = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, errNum, 0,
- errString, 255, NULL);
- if (numChars == 0) {
- J2dTraceLn1(J2D_TRACE_ERROR, "problem with formatmessage, err = %d\n",
- GetLastError());
- }
- J2dTraceLn3(J2D_TRACE_ERROR, "problem with %s, errNum, string = %d, %S\n",
- message, errNum, errString);
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/RegistryKey.h b/jdk/src/windows/native/sun/java2d/windows/RegistryKey.h
deleted file mode 100644
index d1221bd8ce2..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/RegistryKey.h
+++ /dev/null
@@ -1,72 +0,0 @@
-/*
- * Copyright 2003 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-
-#ifndef REGISTRYKEY_H
-#define REGISTRYKEY_H
-
-
-/**
- * Meaning of the following variables:
- * - UNVERIFIED: this value has not yet been tested (and needs to be)
- * - TESTING: this value is currently being tested. If we get this value
- * from the registry, this indicates that we probably crashed while we were
- * testing it last time, so we should disable the appropriate component.
- * - FAILURE: this component failed testing, so it should be disabled
- * - SUCCESS: this component succeeded, so we can enable it
- */
-#define J2D_ACCEL_UNVERIFIED -1
-#define J2D_ACCEL_TESTING 0
-#define J2D_ACCEL_FAILURE 1
-#define J2D_ACCEL_SUCCESS 2
-
-class RegistryKey {
-public:
- RegistryKey(WCHAR *keyName, REGSAM permissions);
- ~RegistryKey();
-
- DWORD EnumerateSubKeys(DWORD index, WCHAR *subKeyName,
- DWORD *buffSize);
-
- int GetIntValue(WCHAR *valueName);
-
- static int GetIntValue(WCHAR *keyName, WCHAR *valueName);
-
- BOOL SetIntValue(WCHAR *valueName, int regValue, BOOL flush);
-
- static BOOL SetIntValue(WCHAR *keyName, WCHAR *valueName, int regValue,
- BOOL flush);
-
- static void DeleteKey(WCHAR *keyName);
-
- static void PrintValue(WCHAR *keyName, WCHAR *valueName,
- WCHAR *msg);
-
-private:
- HKEY hKey;
- static void PrintRegistryError(LONG errNum, char *message);
-};
-
-#endif REGISTRYKEY_H
diff --git a/jdk/src/windows/native/sun/java2d/windows/Win32OffScreenSurfaceData.cpp b/jdk/src/windows/native/sun/java2d/windows/Win32OffScreenSurfaceData.cpp
deleted file mode 100644
index b6dcb5bf560..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/Win32OffScreenSurfaceData.cpp
+++ /dev/null
@@ -1,727 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include "sun_java2d_windows_Win32OffScreenSurfaceData.h"
-
-#include "Win32SurfaceData.h"
-
-#include "Trace.h"
-#include "Region.h"
-#include "awt_Component.h"
-#include "debug_trace.h"
-#include "ddrawUtils.h"
-#include "awt_Win32GraphicsDevice.h"
-#include "D3DContext.h"
-
-#include "jni_util.h"
-
-/**
- * This source file contains support code for loops using the
- * SurfaceData interface to talk to a Win32 drawable from native
- * code.
- */
-
-JNIEXPORT void JNICALL
-Win32OSSD_InitDC(JNIEnv *env, Win32SDOps *wsdo, HDC hdc,
- jint type, jint *patrop,
- jobject clip, jobject comp, jint color);
-jfieldID ddSurfacePuntedID;
-jmethodID markSurfaceLostMID;
-static HBRUSH nullbrush;
-static HPEN nullpen;
-
-extern BOOL ddVramForced;
-
-LockFunc Win32OSSD_Lock;
-GetRasInfoFunc Win32OSSD_GetRasInfo;
-UnlockFunc Win32OSSD_Unlock;
-DisposeFunc Win32OSSD_Dispose;
-GetDCFunc Win32OSSD_GetDC;
-ReleaseDCFunc Win32OSSD_ReleaseDC;
-InvalidateSDFunc Win32OSSD_InvalidateSD;
-RestoreSurfaceFunc Win32OSSD_RestoreSurface;
-
-extern "C" {
-
-/*
- * Class: sun_java2d_windows_Win32OffScreenSurfaceData
- * Method: initIDs
- * Signature: ()V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_initIDs(JNIEnv *env,
- jclass wsd)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OffScreenSurfaceData_initIDs");
- ddSurfacePuntedID = env->GetFieldID(wsd, "ddSurfacePunted", "Z");
- markSurfaceLostMID = env->GetMethodID(wsd, "markSurfaceLost", "()V");
- nullbrush = (HBRUSH) ::GetStockObject(NULL_BRUSH);
- nullpen = (HPEN) ::GetStockObject(NULL_PEN);
-}
-
-void Win32OSSD_DisableDD(JNIEnv *env, Win32SDOps *wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_DisableDD");
-
- wsdo->RestoreSurface(env, wsdo);
- jobject sdObject = env->NewLocalRef(wsdo->sdOps.sdObject);
- if (sdObject != NULL) {
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "Win32OSSD_DisableDD: disabling DirectDraw"\
- " for surface 0x%x", wsdo);
- JNU_CallMethodByName(env, NULL, sdObject, "disableDD", "()V");
- env->DeleteLocalRef(sdObject);
- }
-}
-
-void disposeOSSD_WSDO(JNIEnv* env, Win32SDOps* wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "disposeOSSD_WSDO");
- if (wsdo->device != NULL) {
- wsdo->device->Release();
- wsdo->device = NULL;
- }
- delete wsdo->surfaceLock;
-}
-
-jboolean initOSSD_WSDO(JNIEnv* env, Win32SDOps* wsdo, jint width, jint height,
- jint screen, jint transparency)
-{
- J2dTraceLn1(J2D_TRACE_INFO, "initOSSD_WSDO screen=%d", screen);
-
- {
- Devices::InstanceAccess devices;
- wsdo->device = devices->GetDeviceReference(screen, FALSE);
- }
- if (wsdo->device == NULL) {
- J2dTraceLn1(J2D_TRACE_WARNING,
- "initOSSD_WSDO: Incorrect "\
- "screen number (screen=%d)", screen);
- wsdo->invalid = TRUE;
- return JNI_FALSE;
- }
-
- wsdo->transparency = transparency;
- wsdo->w = width;
- wsdo->h = height;
- wsdo->surfacePuntData.disablePunts = TRUE;
- return JNI_TRUE;
-}
-
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_initSurface
- (JNIEnv *env,
- jobject sData,
- jint depth,
- jint width, jint height,
- jint screen,
- jboolean isVolatile,
- jint transparency)
-{
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_GetOps(env, sData);
-
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_initSurface");
- jboolean status =
- initOSSD_WSDO(env, wsdo, width, height, screen, transparency);
-
- if (status == JNI_FALSE || !DDCreateSurface(wsdo)) {
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "Win32OffScreenSurfaceData_initSurface: Error creating "\
- "offscreen surface (transparency=%d), throwing IPE",
- transparency);
- SurfaceData_ThrowInvalidPipeException(env,
- "Can't create offscreen surf");
- return;
- } else {
- wsdo->surfacePuntData.lpSurfaceVram = wsdo->lpSurface;
- }
- // 8 is somewhat arbitrary; we want the threshhold to represent a
- // significant portion of the surface area in order to avoid
- // punting for occasional, small reads
- wsdo->surfacePuntData.pixelsReadThreshold = width * height / 8;
- /**
- * Only enable our punt-to-sysmem-surface scheme for surfaces that are:
- * - non-transparent (we really only intended this workaround for
- * back buffers, which are usually opaque)
- * - volatile (non-volatile images should not even get into the punt
- * situation since they should not be a rendering destination, but
- * we check this just to make sure)
- * And only do so if the user did not specify that punting be disabled
- */
- wsdo->surfacePuntData.disablePunts = (transparency != TR_OPAQUE) ||
- !isVolatile ||
- ddVramForced;
-}
-
-/*
- * Class: sun_java2d_windows_Win32OffScreenSurfaceData
- * Method: restoreSurface
- * Signature: ()V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_restoreSurface(JNIEnv *env,
- jobject sData)
-{
- J2dTraceLn(J2D_TRACE_INFO,
- "Win32OSSD_restoreSurface: restoring offscreen");
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_GetOps(env, sData);
-
- // Might have gotten here by some default action. Make sure that the
- // surface is marked as lost before bothering to try to restore it.
- if (!wsdo->surfaceLost) {
- return;
- }
-
- // Attempt to restore and lock the surface (to make sure the restore worked)
- if (DDRestoreSurface(wsdo) && DDLock(env, wsdo, NULL, NULL)) {
- DDUnlock(env, wsdo);
- wsdo->surfaceLost = FALSE;
- } else {
- // Failure - throw exception
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "Win32OSSD_restoreSurface: problems"\
- " restoring, throwing IPE");
- SurfaceData_ThrowInvalidPipeException(env, "RestoreSurface failure");
- }
-}
-
-
-/*
- * Class: sun_java2d_windows_Win32OffScreenSurfaceData
- * Method: initOps
- * Signature: (Ljava/lang/Object;)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_initOps(JNIEnv *env,
- jobject wsd,
- jint depth,
- jint transparency)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OffScreenSurfaceData_initOps");
- Win32SDOps *wsdo =
- (Win32SDOps *)SurfaceData_InitOps(env, wsd, sizeof(Win32SDOps));
- wsdo->sdOps.Lock = Win32OSSD_Lock;
- wsdo->sdOps.GetRasInfo = Win32OSSD_GetRasInfo;
- wsdo->sdOps.Unlock = Win32OSSD_Unlock;
- wsdo->sdOps.Dispose = Win32OSSD_Dispose;
- wsdo->RestoreSurface = Win32OSSD_RestoreSurface;
- wsdo->GetDC = Win32OSSD_GetDC;
- wsdo->ReleaseDC = Win32OSSD_ReleaseDC;
- wsdo->InvalidateSD = Win32OSSD_InvalidateSD;
- wsdo->invalid = JNI_FALSE;
- wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
- wsdo->window = NULL;
- wsdo->backBufferCount = 0;
- wsdo->depth = depth;
- switch (depth) {
- case 8:
- wsdo->pixelStride = 1;
- break;
- case 15: //555
- wsdo->pixelStride = 2;
- wsdo->pixelMasks[0] = 0x1f << 10;
- wsdo->pixelMasks[1] = 0x1f << 5;
- wsdo->pixelMasks[2] = 0x1f;
- break;
- case 16: //565
- wsdo->pixelStride = 2;
- wsdo->pixelMasks[0] = 0x1f << 11;
- wsdo->pixelMasks[1] = 0x3f << 5;
- wsdo->pixelMasks[2] = 0x1f;
- break;
- case 24:
- wsdo->pixelStride = 3;
- break;
- case 32: //888
- wsdo->pixelStride = 4;
- wsdo->pixelMasks[0] = 0xff0000;
- wsdo->pixelMasks[1] = 0x00ff00;
- wsdo->pixelMasks[2] = 0x0000ff;
- break;
- }
- wsdo->surfaceLock = new CriticalSection();
- wsdo->surfaceLost = FALSE;
- wsdo->transparency = transparency;
- wsdo->surfacePuntData.usingDDSystem = FALSE;
- wsdo->surfacePuntData.lpSurfaceSystem = NULL;
- wsdo->surfacePuntData.lpSurfaceVram = NULL;
- wsdo->surfacePuntData.numBltsSinceRead = 0;
- wsdo->surfacePuntData.pixelsReadSinceBlt = 0;
- wsdo->surfacePuntData.numBltsThreshold = 2;
- wsdo->gdiOpPending = FALSE;
-}
-
-JNIEXPORT Win32SDOps * JNICALL
-Win32OffScreenSurfaceData_GetOps(JNIEnv *env, jobject sData)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "Win32OffScreenSurfaceData_GetOps");
- SurfaceDataOps *ops = SurfaceData_GetOps(env, sData);
- if (ops == NULL) {
- JNU_ThrowNullPointerException(env, "SurfaceData native ops");
- } else if (ops->Lock != Win32OSSD_Lock) {
- SurfaceData_ThrowInvalidPipeException(env, "not a Win32 SurfaceData");
- ops = NULL;
- }
- return (Win32SDOps *) ops;
-}
-
-} /* extern "C" */
-
-void Win32OSSD_RestoreSurface(JNIEnv *env, Win32SDOps *wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_RestoreSurface");
- wsdo->surfaceLost = TRUE;
- jobject sdObject = env->NewLocalRef(wsdo->sdOps.sdObject);
- if (sdObject != NULL) {
- // markSurfaceLost will end up throwing an InvalidPipeException
- // if this surface belongs to a managed image.
- env->CallVoidMethod(sdObject, markSurfaceLostMID);
- env->DeleteLocalRef(sdObject);
- }
-}
-
-void Win32OSSD_LockByDD(JNIEnv *env, Win32SDOps *wsdo, jint lockflags,
- SurfaceDataRasInfo *pRasInfo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_LockByDD");
-
- if ((lockflags & SD_LOCK_READ) &&
- !wsdo->surfacePuntData.disablePunts)
- {
- wsdo->surfacePuntData.numBltsSinceRead = 0;
- if (!wsdo->surfacePuntData.usingDDSystem) {
- int w = pRasInfo->bounds.x2 - pRasInfo->bounds.x1;
- int h = pRasInfo->bounds.y2 - pRasInfo->bounds.y1;
- wsdo->surfacePuntData.pixelsReadSinceBlt += w * h;
- // Note that basing this decision on the bounds is somewhat
- // incorrect because locks of type FASTEST will simply send
- // in bounds that equal the area of the entire surface.
- // To do this correctly, we would need to return
- // SLOWLOCK and recalculate the punt data in GetRasInfo()
- if (wsdo->surfacePuntData.pixelsReadSinceBlt >
- wsdo->surfacePuntData.pixelsReadThreshold)
- {
- // Create the system surface if it doesn't exist
- if (!wsdo->surfacePuntData.lpSurfaceSystem) {
- wsdo->surfacePuntData.lpSurfaceSystem =
- wsdo->ddInstance->ddObject->CreateDDOffScreenSurface(
- wsdo->w, wsdo->h, wsdo->depth,
- wsdo->transparency, DDSCAPS_SYSTEMMEMORY);
- if (wsdo->surfacePuntData.lpSurfaceSystem) {
- // 4941350: Double-check that the surface we created
- // matches the depth expected.
- int sysmemDepth =
- wsdo->surfacePuntData.lpSurfaceSystem->GetSurfaceDepth();
- if (!DDSurfaceDepthsCompatible(wsdo->depth, sysmemDepth)) {
- // There is clearly a problem here; release
- // the punting surface
- J2dTraceLn2(J2D_TRACE_WARNING,
- "Win32OSSD_LockByDD: Punting error: "\
- "wsdo->depth=%d memory surface depth=%d",
- wsdo->depth, sysmemDepth);
- DDReleaseSurfaceMemory(wsdo->surfacePuntData.
- lpSurfaceSystem);
- wsdo->surfacePuntData.lpSurfaceSystem = NULL;
- } else {
- DDCOLORKEY ddck;
- HRESULT ddResult =
- wsdo->surfacePuntData.lpSurfaceVram->GetColorKey(
- DDCKEY_SRCBLT, &ddck);
- if (ddResult == DD_OK) {
- // Vram surface has colorkey - use same colorkey on sys
- ddResult =
- wsdo->surfacePuntData.lpSurfaceSystem->SetColorKey(
- DDCKEY_SRCBLT, &ddck);
- }
- }
- }
- }
- // Assuming no errors in system creation, copy contents
- if (wsdo->surfacePuntData.lpSurfaceSystem) {
- if (wsdo->surfacePuntData.lpSurfaceSystem->Blt(NULL,
- wsdo->surfacePuntData.lpSurfaceVram, NULL,
- DDBLT_WAIT, NULL) == DD_OK)
- {
- J2dTraceLn2(J2D_TRACE_INFO,
- "Win32OSSD_LockByDD: punting VRAM to sys: "\
- "0x%x -> 0x%x",
- wsdo->surfacePuntData.lpSurfaceVram,
- wsdo->surfacePuntData.lpSurfaceSystem);
- wsdo->lpSurface = wsdo->surfacePuntData.lpSurfaceSystem;
- wsdo->surfacePuntData.usingDDSystem = TRUE;
- // Notify the Java level that this surface has
- // been punted to avoid performance penalties from
- // copying from VRAM cached versions of other images
- // when we should use system memory versions instead.
- jobject sdObject =
- env->NewLocalRef(wsdo->sdOps.sdObject);
- if (sdObject) {
- // Only bother with this optimization if the
- // reference is still valid
- env->SetBooleanField(sdObject, ddSurfacePuntedID,
- JNI_TRUE);
- env->DeleteLocalRef(sdObject);
- }
- }
- }
- }
- }
- }
-
- if (!DDLock(env, wsdo, NULL, pRasInfo))
- return;
-
- wsdo->lockType = WIN32SD_LOCK_BY_DDRAW;
-}
-
-
-jint Win32OSSD_Lock(JNIEnv *env,
- SurfaceDataOps *ops,
- SurfaceDataRasInfo *pRasInfo,
- jint lockflags)
-{
- Win32SDOps *wsdo = (Win32SDOps *) ops;
- J2dTraceLn1(J2D_TRACE_INFO, "Win32OSSD_Lock: lockflags=0x%x",
- lockflags);
- wsdo->surfaceLock->Enter();
- if (wsdo->invalid) {
- wsdo->surfaceLock->Leave();
- SurfaceData_ThrowInvalidPipeException(env, "invalid sd");
- return SD_FAILURE;
- }
-
- if (wsdo->lockType != WIN32SD_LOCK_UNLOCKED) {
- wsdo->surfaceLock->Leave();
- JNU_ThrowInternalError(env, "Win32OSSD_Lock cannot nest locks");
- return SD_FAILURE;
- }
-
- if (lockflags & SD_LOCK_RD_WR) {
- if (pRasInfo->bounds.x1 < 0) pRasInfo->bounds.x1 = 0;
- if (pRasInfo->bounds.y1 < 0) pRasInfo->bounds.y1 = 0;
- if (pRasInfo->bounds.x2 > wsdo->w) pRasInfo->bounds.x2 = wsdo->w;
- if (pRasInfo->bounds.y2 > wsdo->h) pRasInfo->bounds.y2 = wsdo->h;
- if (DDUseDDraw(wsdo)) {
- Win32OSSD_LockByDD(env, wsdo, lockflags, pRasInfo);
- }
- if (wsdo->lockType == WIN32SD_LOCK_UNLOCKED) {
- wsdo->lockFlags = lockflags;
- wsdo->surfaceLock->Leave();
- return SD_FAILURE;
- }
- } else {
- // They didn't ask for a lock, so they don't get one
- wsdo->lockType = WIN32SD_LOCK_BY_NULL;
- }
- wsdo->lockFlags = lockflags;
- J2dTraceLn2(J2D_TRACE_VERBOSE, "Win32OSSD_Lock: flags=0x%x type=%d",
- wsdo->lockFlags, wsdo->lockType);
- return 0;
-}
-
-void Win32OSSD_GetRasInfo(JNIEnv *env,
- SurfaceDataOps *ops,
- SurfaceDataRasInfo *pRasInfo)
-{
- Win32SDOps *wsdo = (Win32SDOps *) ops;
- jint lockflags = wsdo->lockFlags;
-
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_GetRasInfo");
-
- if (wsdo->lockType == WIN32SD_LOCK_UNLOCKED) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "Win32OSSD_GetRasInfo: lockType=UNLOCKED");
- memset(pRasInfo, 0, sizeof(*pRasInfo));
- return;
- }
-
- if (wsdo->lockType != WIN32SD_LOCK_BY_DDRAW) {
- /* They didn't lock for anything - we won't give them anything */
- pRasInfo->rasBase = NULL;
- pRasInfo->pixelStride = 0;
- pRasInfo->pixelBitOffset = 0;
- pRasInfo->scanStride = 0;
- }
- if (wsdo->lockFlags & SD_LOCK_LUT) {
- pRasInfo->lutBase =
- (long *) wsdo->device->GetSystemPaletteEntries();
- pRasInfo->lutSize = 256;
- } else {
- pRasInfo->lutBase = NULL;
- pRasInfo->lutSize = 0;
- }
- if (wsdo->lockFlags & SD_LOCK_INVCOLOR) {
- pRasInfo->invColorTable = wsdo->device->GetSystemInverseLUT();
- ColorData *cData = wsdo->device->GetColorData();
- pRasInfo->redErrTable = cData->img_oda_red;
- pRasInfo->grnErrTable = cData->img_oda_green;
- pRasInfo->bluErrTable = cData->img_oda_blue;
- } else {
- pRasInfo->invColorTable = NULL;
- pRasInfo->redErrTable = NULL;
- pRasInfo->grnErrTable = NULL;
- pRasInfo->bluErrTable = NULL;
- }
- if (wsdo->lockFlags & SD_LOCK_INVGRAY) {
- pRasInfo->invGrayTable =
- wsdo->device->GetColorData()->pGrayInverseLutData;
- } else {
- pRasInfo->invGrayTable = NULL;
- }
-}
-
-void Win32OSSD_Unlock(JNIEnv *env,
- SurfaceDataOps *ops,
- SurfaceDataRasInfo *pRasInfo)
-{
- Win32SDOps *wsdo = (Win32SDOps *) ops;
-
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_Unlock");
-
- if (wsdo->lockType == WIN32SD_LOCK_UNLOCKED) {
- JNU_ThrowInternalError(env, "Unmatched unlock on Win32OS SurfaceData");
- return;
- }
-
- if (wsdo->lockType == WIN32SD_LOCK_BY_DDRAW) {
- DDUnlock(env, wsdo);
- }
- wsdo->lockType = WIN32SD_LOCK_UNLOCKED;
- wsdo->surfaceLock->Leave();
-}
-
-static void
-GetClipFromRegion(JNIEnv *env, jobject clip, RECT &r)
-{
- SurfaceDataBounds bounds;
- Region_GetBounds(env, clip, &bounds);
- r.left = bounds.x1;
- r.top = bounds.y1;
- r.right = bounds.x2;
- r.bottom = bounds.y2;
-}
-
-/*
- * REMIND: This mechanism is just a prototype of a way to manage a
- * small cache of DC objects. It is incomplete in the following ways:
- *
- * - It is not thread-safe! It needs appropriate locking and release calls
- * (perhaps the AutoDC mechanisms from Kestrel)
- * - It does hardly any error checking (What if GetDCEx returns NULL?)
- * - It cannot handle printer DCs, their resolution, or Embedded DCs
- * - It always selects a clip region, even if the clip is the window bounds
- * - There is very little error checking (null DC returned from GetDCEx, etc)
- * - It should probably "live" in the native SurfaceData object to allow
- * alternate implementations for printing and embedding
- * - It doesn't handle XOR
- * - It caches the client bounds to determine if clipping is really needed
- * (no way to invalidate the cached bounds and there is probably a better
- * way to manage clip validation in any case)
- */
-
-extern COLORREF CheckGrayColor(Win32SDOps *wsdo, int c);
-HDC Win32OSSD_GetDC(JNIEnv *env, Win32SDOps *wsdo,
- jint type, jint *patrop,
- jobject clip, jobject comp, jint color)
-{
- // REMIND: Should lock around all accesses to "last"
- J2dTraceLn1(J2D_TRACE_INFO, "Win32OSSD_GetDC: color=0x%x", color);
-
- if (wsdo->invalid) {
- SurfaceData_ThrowInvalidPipeException(env, "invalid sd");
- return (HDC) NULL;
- }
-
- HDC hdc;
- HRESULT res = wsdo->lpSurface->GetDC(&hdc);
- if (res != DD_OK) {
- if (res == DDERR_CANTCREATEDC) {
- // this may be a manifestations of an unrecoverable error caused by
- // address space exaustion when heap size is too large
- Win32OSSD_DisableDD(env, wsdo);
- }
- // Note: DDrawSurface::GetDC() releases its surfaceLock
- // when it returns an error here, so do not call ReleaseDC()
- // to force the release of surfaceLock
- SurfaceData_ThrowInvalidPipeException(env, "invalid sd");
- return (HDC) NULL;
- }
-
- // Initialize the DC
- Win32OSSD_InitDC(env, wsdo, hdc, type, patrop, clip, comp, color);
- return hdc;
-}
-
-JNIEXPORT void JNICALL
-Win32OSSD_InitDC(JNIEnv *env, Win32SDOps *wsdo, HDC hdc,
- jint type, jint *patrop,
- jobject clip, jobject comp, jint color)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_InitDC");
- // Initialize DC. Assume nothing about the DC since ddraw DC's are
- // created from scratch every time
-
- // Since we can't get here in XOR mode (ISCOPY only), we will ignore the
- // comp and force the patrop to PATCOPY if necessary.
- if (patrop != NULL) {
- *patrop = PATCOPY;
- }
-
- if (clip == NULL) {
- ::SelectClipRgn(hdc, (HRGN) NULL);
- } else {
- RECT r;
- GetClipFromRegion(env, clip, r);
- // Only bother setting clip if it's smaller than our window
- if ((r.left > 0) || (r.top > 0) ||
- (r.right < wsdo->w) || (r.bottom < wsdo->h)) {
- J2dTraceLn4(J2D_TRACE_VERBOSE,
- "Win32OSSD_InitDC: clipRect "\
- "l=%-4d t=%-4d r=%-4d b=%-4d",
- r.left, r.top, r.right, r.bottom);
- //Make the window-relative rect a client-relative one for Windows
- ::OffsetRect(&r, -wsdo->insets.left, -wsdo->insets.top);
- if (r.left > r.right) r.left = r.right;
- if (r.top > r.bottom) r.top = r.bottom;
- HRGN hrgn = ::CreateRectRgnIndirect(&r);
- ::SelectClipRgn(hdc, hrgn);
- ::DeleteObject(hrgn);
- }
- }
- if (type & BRUSH) {
- if (wsdo->brushclr != color || (wsdo->brush == NULL)) {
- if (wsdo->brush != NULL) {
- wsdo->brush->Release();
- }
- wsdo->brush = AwtBrush::Get(CheckGrayColor(wsdo, color));
- wsdo->brushclr = color;
- }
- // always select a new brush - the DC is new every time
- ::SelectObject(hdc, wsdo->brush->GetHandle());
- } else if (type & NOBRUSH) {
- ::SelectObject(hdc, nullbrush);
- }
- if (type & PEN) {
- if (wsdo->penclr != color || (wsdo->pen == NULL)) {
- if (wsdo->pen != NULL) {
- wsdo->pen->Release();
- }
- wsdo->pen = AwtPen::Get(CheckGrayColor(wsdo, color));
- wsdo->penclr = color;
- }
- // always select a new pen - the DC is new every time
- ::SelectObject(hdc, wsdo->pen->GetHandle());
- } else if (type & NOPEN) {
- ::SelectObject(hdc, nullpen);
- }
-}
-
-void Win32OSSD_ReleaseDC(JNIEnv *env, Win32SDOps *wsdo, HDC hdc)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_ReleaseDC");
- wsdo->lpSurface->ReleaseDC(hdc);
- wsdo->gdiOpPending = TRUE;
-}
-
-void Win32OSSD_InvalidateSD(JNIEnv *env, Win32SDOps *wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OSSD_InvalidateSD");
- wsdo->invalid = JNI_TRUE;
-}
-
-/*
- * Class: sun_java2d_windows_Win32OffScreenSurfaceData
- * Method: invalidateSD
- * Signature: ()V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_nativeInvalidate(JNIEnv *env,
- jobject wsd)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OffScreenSurfaceData_nativeInvalidate");
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_GetOps(env, wsd);
- if (wsdo != NULL) {
- wsdo->InvalidateSD(env, wsdo);
- }
-}
-
-
-/*
- * Method: Win32OSSD_Dispose
- */
-void
-Win32OSSD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
-{
- Win32SDOps *wsdo = (Win32SDOps*)ops;
- J2dTraceLn2(J2D_TRACE_VERBOSE, "Win32OSSD_Dispose vram=0%x sysm=0%x",
- wsdo->surfacePuntData.lpSurfaceVram,
- wsdo->surfacePuntData.lpSurfaceSystem);
- // REMIND: Need to delete a lot of other things here as well, starting
- // with the offscreen surface
-
- // ops is assumed non-null as it is checked in SurfaceData_DisposeOps
- if (wsdo->surfacePuntData.lpSurfaceVram) {
- delete wsdo->surfacePuntData.lpSurfaceVram;
- }
- if (wsdo->surfacePuntData.lpSurfaceSystem) {
- delete wsdo->surfacePuntData.lpSurfaceSystem;
- }
- if (wsdo->brush != NULL) {
- wsdo->brush->Release();
- }
- if (wsdo->pen != NULL) {
- wsdo->pen->Release();
- }
- wsdo->lpSurface = NULL;
- disposeOSSD_WSDO(env, wsdo);
-}
-
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_setTransparentPixel(JNIEnv *env,
- jobject wsd,
- jint pixel)
-{
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_GetOps(env, wsd);
- DDSetColorKey(env, wsdo, pixel);
-}
-
-/*
- * Class: sun_java2d_windows_Win32OffScreenSurfaceData
- * Method: flush
- * Signature: ()V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_Win32OffScreenSurfaceData_flush(JNIEnv *env,
- jobject wsd)
-{
- J2dTraceLn(J2D_TRACE_INFO, "Win32OffScreenSurfaceData_flush");
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_GetOps(env, wsd);
- if (wsdo != NULL) {
- // Note that wsdo may be null if there was some error during
- // construction, such as a surface depth we could not handle
- DDReleaseSurfaceMemory(wsdo->surfacePuntData.lpSurfaceSystem);
- DDReleaseSurfaceMemory(wsdo->surfacePuntData.lpSurfaceVram);
- }
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/WinBackBufferSurfaceData.cpp b/jdk/src/windows/native/sun/java2d/windows/WinBackBufferSurfaceData.cpp
deleted file mode 100644
index ac954b4c3bf..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/WinBackBufferSurfaceData.cpp
+++ /dev/null
@@ -1,101 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include "sun_java2d_windows_WinBackBufferSurfaceData.h"
-
-#include "Win32SurfaceData.h"
-
-#include "awt_Component.h"
-#include "Trace.h"
-#include "ddrawUtils.h"
-
-#include "jni_util.h"
-
-extern "C" jboolean initOSSD_WSDO(JNIEnv* env, Win32SDOps* wsdo, jint width,
- jint height, jint screen, jint transparency);
-
-extern "C" void disposeOSSD_WSDO(JNIEnv* env, Win32SDOps* wsdo);
-
-DisposeFunc Win32BBSD_Dispose;
-
-/*
- * Class: sun_java2d_windows_WinBackBufferSurfaceData
- * Method: initSurface
- * Signature: (IIILsun/awt/windows/WinBackBufferSurfaceData;)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_WinBackBufferSurfaceData_initSurface(JNIEnv *env,
- jobject sData, jint depth, jint width, jint height, jint screen,
- jobject parentData)
-{
- Win32SDOps *wsdo = (Win32SDOps *)SurfaceData_GetOps(env, sData);
-
- J2dTraceLn(J2D_TRACE_INFO, "Win32BBSD_initSurface");
- /* Set the correct dispose method */
- wsdo->sdOps.Dispose = Win32BBSD_Dispose;
- jboolean status =
- initOSSD_WSDO(env, wsdo, width, height, screen, JNI_FALSE);
- if (status == JNI_FALSE || parentData == NULL) {
- SurfaceData_ThrowInvalidPipeException(env,
- "Error initalizing back-buffer surface");
- return;
- }
- Win32SDOps *wsdo_parent = (Win32SDOps*)SurfaceData_GetOps(env, parentData);
- if (!DDGetAttachedSurface(env, wsdo_parent, wsdo)) {
- SurfaceData_ThrowInvalidPipeException(env,
- "Can't create attached surface");
- }
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "Win32BackBufferSurfaceData_initSurface: "\
- "completed wsdo->lpSurface=0x%x", wsdo->lpSurface);
-}
-
-/*
- * Class: sun_java2d_windows_WinBackBufferSurfaceData
- * Method: restoreSurface
- * Signature: (Lsun/awt/windows/WinBackBufferSurfaceData;)V
- */
-JNIEXPORT void JNICALL
-Java_sun_java2d_windows_WinBackBufferSurfaceData_restoreSurface(JNIEnv *env,
- jobject sData, jobject parentData)
-{
- // Noop: back buffer restoration implicit in primary restore
-}
-
-/*
- * Method: Win32BBSD_Dispose
- */
-void
-Win32BBSD_Dispose(JNIEnv *env, SurfaceDataOps *ops)
-{
- // ops is assumed non-null as it is checked in SurfaceData_DisposeOps
- Win32SDOps *wsdo = (Win32SDOps*)ops;
- J2dTraceLn(J2D_TRACE_INFO, "Win32BBSD_Dispose");
- if (wsdo->lpSurface != NULL && !wsdo->surfaceLost) {
- delete wsdo->lpSurface;
- wsdo->lpSurface = NULL;
- }
- disposeOSSD_WSDO(env, wsdo);
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.cpp b/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.cpp
index 8da8df92aa7..b7040b8d803 100644
--- a/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.cpp
+++ b/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2003-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -28,15 +28,11 @@
#include
#include "Trace.h"
#include "WindowsFlags.h"
-#include "dxInit.h"
-BOOL ddVramForced; // disable punting of ddraw buffers
BOOL accelReset; // reset registry 2d acceleration settings
-BOOL useDD; // ddraw enabled flag
BOOL useD3D; // d3d enabled flag
BOOL forceD3DUsage; // force d3d on or off
jboolean g_offscreenSharing; // JAWT accelerated surface sharing
-BOOL useDDLock; // Disabled for win2k/XP
BOOL checkRegistry; // Diagnostic tool: outputs 2d registry settings
BOOL disableRegistry; // Diagnostic tool: disables registry interaction
BOOL setHighDPIAware; // Whether to set the high-DPI awareness flag
@@ -44,19 +40,15 @@ BOOL setHighDPIAware; // Whether to set the high-DPI awareness flag
extern WCHAR *j2dAccelKey; // Name of java2d root key
extern WCHAR *j2dAccelDriverKey; // Name of j2d per-device key
-static jfieldID ddEnabledID;
static jfieldID d3dEnabledID;
static jfieldID d3dSetID;
-static jfieldID ddSetID;
static jclass wFlagsClassID;
void SetIDs(JNIEnv *env, jclass wFlagsClass)
{
wFlagsClassID = (jclass)env->NewGlobalRef(wFlagsClass);
- ddEnabledID = env->GetStaticFieldID(wFlagsClass, "ddEnabled", "Z");
d3dEnabledID = env->GetStaticFieldID(wFlagsClass, "d3dEnabled", "Z");
d3dSetID = env->GetStaticFieldID(wFlagsClass, "d3dSet", "Z");
- ddSetID = env->GetStaticFieldID(wFlagsClass, "ddSet", "Z");
}
BOOL GetStaticBoolean(JNIEnv *env, jclass wfClass, const char *fieldName)
@@ -74,8 +66,6 @@ jobject GetStaticObject(JNIEnv *env, jclass wfClass, const char *fieldName,
void GetFlagValues(JNIEnv *env, jclass wFlagsClass)
{
- useDD = env->GetStaticBooleanField(wFlagsClass, ddEnabledID);
- jboolean ddSet = env->GetStaticBooleanField(wFlagsClass, ddSetID);
jboolean d3dEnabled = env->GetStaticBooleanField(wFlagsClass, d3dEnabledID);
jboolean d3dSet = env->GetStaticBooleanField(wFlagsClass, d3dSetID);
if (!d3dSet) {
@@ -99,90 +89,23 @@ void GetFlagValues(JNIEnv *env, jclass wFlagsClass)
}
useD3D = d3dEnabled;
forceD3DUsage = d3dSet;
- ddVramForced = GetStaticBoolean(env, wFlagsClass, "ddVramForced");
g_offscreenSharing = GetStaticBoolean(env, wFlagsClass,
"offscreenSharingEnabled");
- useDDLock = GetStaticBoolean(env, wFlagsClass, "ddLockEnabled");
- jboolean ddLockSet = GetStaticBoolean(env, wFlagsClass, "ddLockSet");
accelReset = GetStaticBoolean(env, wFlagsClass, "accelReset");
checkRegistry = GetStaticBoolean(env, wFlagsClass, "checkRegistry");
disableRegistry = GetStaticBoolean(env, wFlagsClass, "disableRegistry");
jstring javaVersionString = (jstring)GetStaticObject(env, wFlagsClass,
"javaVersion",
"Ljava/lang/String;");
- jboolean isCopy;
- const jchar *javaVersion = env->GetStringChars(javaVersionString,
- &isCopy);
- jsize versionLength = env->GetStringLength(javaVersionString);
- size_t j2dRootKeyLength = wcslen(J2D_ACCEL_KEY_ROOT);
- j2dAccelKey = (WCHAR *)safe_Calloc((j2dRootKeyLength + versionLength + 2),
- sizeof(WCHAR));
- wcscpy(j2dAccelKey, J2D_ACCEL_KEY_ROOT);
- wcscat(j2dAccelKey, javaVersion);
- wcscat(j2dAccelKey, L"\\");
- j2dAccelDriverKey =
- (WCHAR *)safe_Calloc((wcslen(j2dAccelKey) +
- wcslen(J2D_ACCEL_DRIVER_SUBKEY) + 1),
- sizeof(WCHAR));
- wcscpy(j2dAccelDriverKey, j2dAccelKey);
- wcscat(j2dAccelDriverKey, J2D_ACCEL_DRIVER_SUBKEY);
- env->ReleaseStringChars(javaVersionString, javaVersion);
setHighDPIAware =
(IS_WINVISTA && GetStaticBoolean(env, wFlagsClass, "setHighDPIAware"));
- // Change default value of some flags based on OS-specific requirements
- if (IS_WINVISTA && !(ddSet && useDD)) {
- // Disable ddraw on vista due to issues with mixing GDI and ddraw
- // unless ddraw is forced
- SetDDEnabledFlag(env, FALSE);
- J2dRlsTraceLn(J2D_TRACE_WARNING,
- "GetFlagValues: DDraw/D3D is disabled on Windows Vista");
- }
-
- if (IS_NT && !(IS_WIN2000)) {
- // Do not enable d3d on NT4; d3d is only supported through
- // software on that platform
- SetD3DEnabledFlag(env, FALSE, FALSE);
- J2dRlsTraceLn(J2D_TRACE_WARNING,
- "GetFlagValues: D3D is disabled on Win NT");
- }
- if (IS_WIN64 && !d3dSet) {
- // Only enable d3d on Itanium if user forces it on.
- // D3d was not functioning on initial XP Itanium releases
- // so we do not want it suddenly enabled in the field without
- // having tested that codepath first.
- SetD3DEnabledFlag(env, FALSE, FALSE);
- J2dRlsTraceLn(J2D_TRACE_WARNING,
- "GetFlagValues: D3D is disabled on 64-bit OSs");
- }
- if (IS_WIN2000 && !ddLockSet) { // valid for win2k, XP, and future OSs
- // Fix for cursor flicker on win2k and XP (bug 4409306). The
- // fix is to avoid using DDraw for locking the
- // screen. Ideally, we will handle most operations to the
- // screen through new GDI Blt loops (GDIBlitLoops.cpp),
- // but failing there we will punt to GDI instead of DDraw for
- // locking the screen.
- useDDLock = FALSE;
- J2dRlsTraceLn(J2D_TRACE_WARNING,
- "GetFlagValues: DDraw screen locking is "\
- "disabled (W2K, XP+)");
- }
J2dTraceLn(J2D_TRACE_INFO, "WindowsFlags (native):");
- J2dTraceLn1(J2D_TRACE_INFO, " ddEnabled = %s",
- (useDD ? "true" : "false"));
- J2dTraceLn1(J2D_TRACE_INFO, " ddSet = %s",
- (ddSet ? "true" : "false"));
- J2dTraceLn1(J2D_TRACE_INFO, " ddVramForced = %s",
- (ddVramForced ? "true" : "false"));
J2dTraceLn1(J2D_TRACE_INFO, " d3dEnabled = %s",
(useD3D ? "true" : "false"));
J2dTraceLn1(J2D_TRACE_INFO, " d3dSet = %s",
(forceD3DUsage ? "true" : "false"));
- J2dTraceLn1(J2D_TRACE_INFO, " ddLockEnabled = %s",
- (useDDLock ? "true" : "false"));
- J2dTraceLn1(J2D_TRACE_INFO, " ddLockSet = %s",
- (ddLockSet ? "true" : "false"));
J2dTraceLn1(J2D_TRACE_INFO, " offscreenSharing = %s",
(g_offscreenSharing ? "true" : "false"));
J2dTraceLn1(J2D_TRACE_INFO, " accelReset = %s",
@@ -208,13 +131,12 @@ void SetD3DEnabledFlag(JNIEnv *env, BOOL d3dEnabled, BOOL d3dSet)
}
}
-void SetDDEnabledFlag(JNIEnv *env, BOOL ddEnabled)
-{
- useDD = ddEnabled;
- if (env == NULL) {
- env = (JNIEnv * ) JNU_GetEnv(jvm, JNI_VERSION_1_2);
- }
- env->SetStaticBooleanField(wFlagsClassID, ddEnabledID, ddEnabled);
+BOOL IsD3DEnabled() {
+ return useD3D;
+}
+
+BOOL IsD3DForced() {
+ return forceD3DUsage;
}
extern "C" {
diff --git a/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.h b/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.h
index 1dc6d257b24..37d3ea8bd08 100644
--- a/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.h
+++ b/jdk/src/windows/native/sun/java2d/windows/WindowsFlags.h
@@ -1,6 +1,6 @@
/*
- * Copyright 2003-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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
@@ -27,19 +27,17 @@
#ifndef WINDOWSFLAGS_H
#define WINDOWSFLAGS_H
-extern BOOL ddVramForced; // disable punting of ddraw buffers
extern BOOL accelReset; // reset registry 2d acceleration settings
-extern BOOL useDD; // ddraw enabled flag
extern BOOL useD3D; // d3d enabled flag
extern BOOL forceD3DUsage; // force d3d on or off
extern jboolean g_offscreenSharing; // JAWT accelerated surface sharing
-extern BOOL useDDLock; // Disabled for win2k/XP
extern BOOL checkRegistry; // Diag tool: outputs 2d registry settings
extern BOOL disableRegistry; // Diag tool: disables registry interaction
extern BOOL setHighDPIAware; // whether to set High DPI Aware flag on Vista
void SetD3DEnabledFlag(JNIEnv *env, BOOL d3dEnabled, BOOL d3dSet);
-void SetDDEnabledFlag(JNIEnv *env, BOOL ddEnabled);
+BOOL IsD3DEnabled();
+BOOL IsD3DForced();
#endif WINDOWSFLAGS_H
diff --git a/jdk/src/windows/native/sun/java2d/windows/ddrawObject.cpp b/jdk/src/windows/native/sun/java2d/windows/ddrawObject.cpp
deleted file mode 100644
index 54df82c50f5..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/ddrawObject.cpp
+++ /dev/null
@@ -1,1452 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
- /**
- * ddrawObject.cpp
- *
- * This file holds classes used to access DirectDraw functionality.
- * There are two main classes here used by the outside world:
- * DDraw and DDrawSurface. DDraw holds the actual DirectDraw
- * device object, responsible for creating surfaces and doing other
- * device-wide operations. DDraw also holds a pointer to a D3DContext,
- * which has the d3dObject and shared d3d drawing device for the
- * display device (see d3dObject.cpp). DDrawSurface holds an individual
- * surface, such as the primary or an offscreen surface.
- * DDrawSurface also holds a pointer to the device-wide d3dContext
- * because some operations on the surface may actually be 3D methods
- * that need to be forwarded to the 3d drawing device.
- * The DirectDraw object and surfaces are wrapped by DXObject
- * and DXSurface classes in order to be able to generically handle
- * DDraw method calls without the caller having to worry about which
- * version of DirectX we are currently running with.
- * A picture might help to explain the hierarchy of objects herein:
- *
- * DDraw (one per display device)
- * field: DXObject *dxObject
- * field: DXSurface *lpPrimary
- * field: D3DContext *d3dContext
- *
- *
- * DXObject (one per display device)
- * field: IDirectDraw7 (Actual DirectX objects)
- *
- *
- * DDrawSurface (one per offscreen or onscreen surface)
- * field: DXSurface (for ddraw operations)
- *
- * DXSurface (wrapper for DirectDraw operations)
- * field: IDirectDrawSurface7 (DirectX object)
- *
- * The wrapper classes work by using the same method calls as the
- * actual DirectX calls and simply forwarding those calls into the
- * the appropriate DirectX object that they contain. The reason for
- * the indirection is that the subclasses can thus call into the
- * appropriate interface without the caller having to do that
- * explicitly. So instead of something like:
- * if (usingDX7) {
- * dx7Surface->Lock();
- * } else if (usingDXN) {
- * dxNSurface->Lock();
- * }
- * the caller can simply call:
- * dxSurface->Lock();
- * and let the magic of subclassing handle the details of which interface
- * to call, depending on which interface was loaded (and thus which
- * subclass was instantiated).
- * The main difference between actual DirectX method calls and the
- * method calls of these wrapper classes is that we avoid using any
- * structures or parameters that are different between the versions
- * of DirectX that we currently support (DX7). For example,
- * Lock takes DDSURFACEDESC2 structure for DX7.
- * For these methods, we pick an appropriate higher-level data
- * structure that can be cast and queried as appropriate at the
- * subclass level (in the Lock example, we pass a
- * SurfaceDataRasInfo structure, holds the data from the
- * call that we need.
- *
- * Note that the current implementation of the d3d and ddraw pipelines
- * relies heavily on DX7, so some of the abstraction concepts aren't
- * applicable. They may become more relevant once we get back to
- * version-independent implementation.
- */
-
-#include "ddrawUtils.h"
-#include "ddrawObject.h"
-#include "WindowsFlags.h"
-#include "java_awt_DisplayMode.h"
-
-#include "D3DContext.h"
-
-extern HINSTANCE hLibDDraw; // DDraw Library handle
-
-
-#ifdef DEBUG
-void StackTrace() {
- JNIEnv* env;
- jvm->AttachCurrentThread((void**)&env, NULL);
- jclass threadClass = env->FindClass("java/lang/Thread");
- jmethodID dumpStackMID = env->GetStaticMethodID(threadClass, "dumpStack", "()V");
- env->CallStaticVoidMethod(threadClass, dumpStackMID);
-}
-#endif
-
-/**
- * Class DDrawDisplayMode
- */
-DDrawDisplayMode::DDrawDisplayMode() :
- width(0), height(0), bitDepth(0), refreshRate(0) {}
-DDrawDisplayMode::DDrawDisplayMode(DDrawDisplayMode& rhs) :
- width(rhs.width), height(rhs.height), bitDepth(rhs.bitDepth),
- refreshRate(rhs.refreshRate) {}
-DDrawDisplayMode::DDrawDisplayMode(jint w, jint h, jint b, jint r) :
- width(w), height(h), bitDepth(b), refreshRate(r) {}
-
-DDrawDisplayMode::~DDrawDisplayMode() {}
-
-
-/**
- * Class DDraw::EnumDisplayModesParam
- */
-DXObject::EnumDisplayModesParam::EnumDisplayModesParam(
- DDrawDisplayMode::Callback cb, void* ct) : callback(cb), context(ct) {}
-
-DXObject::EnumDisplayModesParam::~EnumDisplayModesParam() {}
-
-/**
- * DXObject
- * These classes handle operations specific to the DX7 interfaces
- */
-
-DXObject::~DXObject()
-{
- J2dTraceLn1(J2D_TRACE_INFO, "~DXObject: ddObject = 0x%x", ddObject);
- ddObject->Release();
- ddObject = NULL;
-}
-
-HRESULT DXObject::GetAvailableVidMem(DWORD caps, DWORD *total,
- DWORD *free)
-{
- DDSCAPS2 ddsCaps;
- memset(&ddsCaps, 0, sizeof(ddsCaps));
- ddsCaps.dwCaps = caps;
- return ddObject->GetAvailableVidMem(&ddsCaps, total, free);
-}
-
-HRESULT DXObject::CreateSurface(DWORD dwFlags,
- DWORD ddsCaps,
- DWORD ddsCaps2,
- LPDDPIXELFORMAT lpPf,
- int width, int height,
- DXSurface **lpDDSurface,
- int numBackBuffers)
-{
- IDirectDrawSurface7 *lpSurface;
- HRESULT ddResult;
- DDSURFACEDESC2 ddsd;
- memset(&ddsd, 0, sizeof(ddsd));
- ddsd.dwSize = sizeof(ddsd);
- ddsd.dwFlags = dwFlags;
- ddsd.ddsCaps.dwCaps = ddsCaps;
- ddsd.ddsCaps.dwCaps2 = ddsCaps2;
- ddsd.dwWidth = width;
- ddsd.dwHeight = height;
- ddsd.dwBackBufferCount = numBackBuffers;
- if (lpPf) {
- memcpy(&ddsd.ddpfPixelFormat, lpPf, sizeof(DDPIXELFORMAT));
- }
- ddResult = ddObject->CreateSurface(&ddsd, &lpSurface, NULL);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DXObject::CreateSurface");
- return ddResult;
- }
- *lpDDSurface = new DXSurface(lpSurface);
- J2dTraceLn3(J2D_TRACE_INFO,
- "DXObject::CreateSurface: w=%-4d h=%-4d dxSurface=0x%x",
- width, height, *lpDDSurface);
- return DD_OK;
-}
-
-HRESULT DXObject::GetDisplayMode(DDrawDisplayMode &dm)
-{
- HRESULT ddResult;
- DDSURFACEDESC2 ddsd;
- memset(&ddsd, 0, sizeof(ddsd));
- ddsd.dwSize = sizeof(ddsd);
- ddResult = ddObject->GetDisplayMode(&ddsd);
- dm.width = ddsd.dwWidth;
- dm.height = ddsd.dwHeight;
- dm.bitDepth = ddsd.ddpfPixelFormat.dwRGBBitCount;
- dm.refreshRate = ddsd.dwRefreshRate;
- return ddResult;
-}
-
-HRESULT DXObject::EnumDisplayModes(DDrawDisplayMode *dm,
- DDrawDisplayMode::Callback callback,
- void *context)
-{
- DDSURFACEDESC2 ddsd;
- memset(&ddsd, 0, sizeof(ddsd));
- ddsd.dwSize = sizeof(ddsd);
- LPDDSURFACEDESC2 pDDSD;
- if (dm == NULL) {
- pDDSD = NULL;
- } else {
- ddsd.dwFlags = DDSD_WIDTH | DDSD_HEIGHT;
- ddsd.dwWidth = dm->width;
- ddsd.dwHeight = dm->height;
- ddsd.dwFlags |= DDSD_PIXELFORMAT;
- ddsd.ddpfPixelFormat.dwFlags = DDPF_RGB;
- ddsd.ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT);
- // dm->bitDepth could be BIT_DEPTH_MULTI or some other invalid value,
- // we rely on DirectDraw to reject such mode
- ddsd.ddpfPixelFormat.dwRGBBitCount = dm->bitDepth;
- if (dm->refreshRate != java_awt_DisplayMode_REFRESH_RATE_UNKNOWN) {
- ddsd.dwFlags |= DDSD_REFRESHRATE;
- ddsd.dwRefreshRate = dm->refreshRate;
- }
- pDDSD = &ddsd;
- }
-
- EnumDisplayModesParam param(callback, context);
-
- HRESULT ddResult;
- ddResult = ddObject->EnumDisplayModes(DDEDM_REFRESHRATES, pDDSD,
- ¶m, EnumCallback);
- return ddResult;
-}
-
-HRESULT DXObject::CreateD3DObject(IDirect3D7 **d3dObject)
-{
- HRESULT ddResult = ddObject->QueryInterface(IID_IDirect3D7,
- (void**)d3dObject);
- if (FAILED(ddResult)) {
- DebugPrintDirectDrawError(ddResult,
- "DXObject::CreateD3DObject: "\
- "query Direct3D7 interface failed");
- }
- return ddResult;
-}
-
-/**
- * Class DDraw
- */
-DDraw::DDraw(DXObject *dxObject) {
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::DDraw");
- lpPrimary = NULL;
- d3dContext = NULL;
- deviceUseD3D = useD3D;
- this->dxObject = dxObject;
-}
-
-DDraw::~DDraw() {
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::~DDraw");
- if (dxObject) {
- delete dxObject;
- }
- if (d3dContext) {
- delete d3dContext;
- }
-}
-
-DDraw *DDraw::CreateDDrawObject(GUID *lpGUID, HMONITOR hMonitor) {
-
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::CreateDDrawObject");
- HRESULT ddResult;
- DXObject *newDXObject;
-
- // First, try to create a DX7 object
- FnDDCreateExFunc ddCreateEx = NULL;
-
- if (getenv("NO_J2D_DX7") == NULL) {
- ddCreateEx = (FnDDCreateExFunc)
- ::GetProcAddress(hLibDDraw, "DirectDrawCreateEx");
- }
-
- if (ddCreateEx) {
-
- J2dTraceLn(J2D_TRACE_VERBOSE, " Using DX7");
- // Success - we are going to use the DX7 interfaces
- // create ddraw object
- IDirectDraw7 *ddObject;
-
- ddResult = (*ddCreateEx)(lpGUID, (void**)&ddObject, IID_IDirectDraw7, NULL);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDraw::CreateDDrawObject: "\
- "DirectDrawCreateEx failed");
- return NULL;
- }
- ddResult = ddObject->SetCooperativeLevel(NULL,
- (DDSCL_NORMAL |
- DDSCL_FPUPRESERVE));
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDraw::CreateDDrawObject: Error "\
- "setting cooperative level");
- return NULL;
- }
- newDXObject = new DXObject(ddObject, hMonitor);
-
- } else {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "DDraw::CreateDDrawObject: No DX7+, ddraw is disabled");
- return NULL;
- }
-
- return new DDraw(newDXObject);
-}
-
-BOOL DDraw::GetDDCaps(LPDDCAPS caps) {
- HRESULT ddResult;
-
- memset(caps, 0, sizeof(*caps));
- caps->dwSize = sizeof(*caps);
- ddResult = dxObject->GetCaps(caps, NULL);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDraw::GetDDCaps: dxObject->GetCaps failed");
- return FALSE;
- }
- return TRUE;
-}
-
-HRESULT DDraw::GetDDAvailableVidMem(DWORD *freeMem)
-{
- DDrawDisplayMode dm;
- HRESULT ddResult;
-
- ddResult = dxObject->GetAvailableVidMem((DDSCAPS_VIDEOMEMORY |
- DDSCAPS_OFFSCREENPLAIN),
- NULL, freeMem);
- if (*freeMem == 0 || ddResult != DD_OK) {
- // Need to check it out ourselves: allocate as much as we can
- // and return that amount
- DDSURFACEDESC ddsd;
- ZeroMemory (&ddsd, sizeof(ddsd));
- ddsd.dwSize = sizeof( ddsd );
- HRESULT ddr = dxObject->GetDisplayMode(dm);
- if (ddr != DD_OK)
- DebugPrintDirectDrawError(ddr,
- "DDraw::GetDDAvailableVidMem: GetDisplayMode failed");
- int bytesPerPixel = dm.bitDepth;
- static int maxSurfaces = 20;
- DXSurface **lpDDSOffscreenVram = (DXSurface**)
- safe_Malloc(maxSurfaces*sizeof(DXSurface*));
- DWORD dwFlags = (DWORD)(DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH);
- DWORD ddsCaps = (DWORD)(DDSCAPS_VIDEOMEMORY | DDSCAPS_OFFSCREENPLAIN);
- int size = 1024;
- int numVramSurfaces = 0;
- int bitsAllocated = 0;
- BOOL done = FALSE;
- while (!done) {
- HRESULT hResult =
- dxObject->CreateSurface(dwFlags, ddsCaps, NULL, size, size,
- &lpDDSOffscreenVram[numVramSurfaces]);
- if (hResult != DD_OK) {
- if (size > 1) {
- size >>= 1;
- } else {
- done = TRUE;
- }
- } else {
- *freeMem += size * size * bytesPerPixel;
- numVramSurfaces++;
- if (numVramSurfaces == maxSurfaces) {
- // Need to reallocate surface holder array
- int newMaxSurfaces = 2 * maxSurfaces;
- DXSurface **newSurfaceArray = (DXSurface**)
- safe_Malloc(maxSurfaces*sizeof(DXSurface*));
- for (int i= 0; i < maxSurfaces; ++i) {
- newSurfaceArray[i] = lpDDSOffscreenVram[i];
- }
- free(lpDDSOffscreenVram);
- maxSurfaces = newMaxSurfaces;
- lpDDSOffscreenVram = newSurfaceArray;
- }
- }
- }
- // Now release all the surfaces we allocated
- for (int i = 0; i < numVramSurfaces; ++i) {
- delete lpDDSOffscreenVram[i];
- }
- free(lpDDSOffscreenVram);
- }
- return ddResult;
-}
-
-
-DDrawSurface* DDraw::CreateDDOffScreenSurface(DWORD width, DWORD height,
- DWORD depth,
- jint transparency,
- DWORD surfaceTypeCaps)
-{
- HRESULT ddResult;
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::CreateDDOffScreenSurface");
-
- DXSurface *dxSurface;
- DWORD dwFlags, ddsCaps;
-
- // Create the offscreen surface
- dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH;
-
- switch (transparency) {
- case TR_BITMASK:
- J2dTraceLn(J2D_TRACE_VERBOSE, " bitmask surface");
- dwFlags |= DDSD_CKSRCBLT;
- /*FALLTHROUGH*/
- case TR_OPAQUE:
- ddsCaps = DDSCAPS_OFFSCREENPLAIN | surfaceTypeCaps;
- }
- J2dTraceLn1(J2D_TRACE_VERBOSE, " creating %s surface",
- (transparency == TR_BITMASK ? "bitmask" : "opaque"));
-
- DDrawSurface* ret = NULL;
- if (dxObject) {
- ddResult = dxObject->CreateSurface(dwFlags, ddsCaps,
- NULL /*texture pixel format*/,
- width, height, &dxSurface);
- if (ddResult == DD_OK) {
- ret = new DDrawSurface(this, dxSurface);
- } else {
- DebugPrintDirectDrawError(ddResult,
- "DDraw::CreateDDOffScreenSurface: "\
- "dxObject->CreateSurface failed");
- }
- }
-
- return ret;
-}
-
-DDrawSurface* DDraw::CreateDDPrimarySurface(DWORD backBufferCount)
-{
- HRESULT ddResult;
- DXSurface *dxSurface;
- DWORD dwFlags, ddsCaps;
- LPDIRECTDRAWSURFACE lpSurface(NULL);
- DDSURFACEDESC ddsd;
- memset(&ddsd, 0, sizeof(ddsd));
- ddsd.dwSize = sizeof(DDSURFACEDESC);
-
- J2dRlsTraceLn1(J2D_TRACE_INFO,
- "DDraw::CreateDDPrimarySurface: back-buffers=%d",
- backBufferCount);
- // create primary surface. There is one of these per ddraw object
- dwFlags = DDSD_CAPS;
- ddsCaps = DDSCAPS_PRIMARYSURFACE;
- if (backBufferCount > 0) {
- dwFlags |= DDSD_BACKBUFFERCOUNT;
- ddsCaps |= (DDSCAPS_FLIP | DDSCAPS_COMPLEX);
-
- // this is required to be able to use d3d for rendering to
- // a backbuffer
- if (deviceUseD3D) {
- ddsCaps |= DDSCAPS_3DDEVICE;
- }
- }
- DDrawSurface* ret;
- if (lpPrimary) {
-
- lpPrimary->GetExclusiveAccess();
- // REMIND: it looks like we need to release
- // d3d resources associated with this
- // surface prior to releasing the dd surfaces;
- ReleaseD3DContext();
-
- ddResult = lpPrimary->ReleaseSurface();
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDraw::CreateDDPrimarySurface: failed releasing old primary");
- }
- lpPrimary->dxSurface = NULL;
- }
- if (dxObject) {
- ddResult = dxObject->CreateSurface(dwFlags, ddsCaps, &dxSurface,
- backBufferCount);
- } else {
- if (lpPrimary) {
- lpPrimary->ReleaseExclusiveAccess();
- }
- return NULL;
- }
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDraw::CreateDDPrimarySurface: "\
- "CreateSurface failed");
- if (lpPrimary) {
- lpPrimary->ReleaseExclusiveAccess();
- }
- return NULL;
- }
-
- if (lpPrimary) {
- lpPrimary->SetNewSurface(dxSurface);
- lpPrimary->ReleaseExclusiveAccess();
- } else {
- lpPrimary = new DDrawPrimarySurface(this, dxSurface);
- }
-
- // The D3D context will be initialized when it's requested
- // by the D3DContext java class (see D3DContext.cpp:initNativeContext).
-
- ret = lpPrimary;
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "DDraw::CreateDDPrimarySurface new primary=0x%x", ret);
- return ret;
-}
-
-void
-DDraw::InitD3DContext()
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::InitD3DContext");
- // note: the first time the context initialization fails,
- // deviceUseD3D is set to FALSE, and we never attempt to
- // initialize it again later.
- // For example, if the app switches to a display mode where
- // d3d is not supported, we disable d3d, but it stays disabled
- // even when we switch back to a supported mode
- if (deviceUseD3D) {
- if (d3dContext == NULL) {
- d3dContext = D3DContext::CreateD3DContext(this, dxObject);
- } else {
- d3dContext->CreateD3DDevice();
- }
- }
-}
-
-void
-DDraw::ReleaseD3DContext()
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::ReleaseD3DContext");
- if (d3dContext != NULL) {
- d3dContext->Release3DDevice();
- }
-}
-
-DDrawClipper* DDraw::CreateDDClipper() {
- HRESULT ddResult;
-
- LPDIRECTDRAWCLIPPER pClipper;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::CreateDDClipper");
- ddResult = dxObject->CreateClipper(0, &pClipper);
-
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDraw::CreateDDClipper");
- return NULL;
- }
-
- return new DDrawClipper(pClipper);
-}
-
-BOOL DDraw::GetDDDisplayMode(DDrawDisplayMode& dm) {
- HRESULT ddResult;
-
- ddResult = dxObject->GetDisplayMode(dm);
-
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "GetDDDisplayMode");
- return FALSE;
- }
-
- return TRUE;
-}
-
-HRESULT DDraw::SetDDDisplayMode(DDrawDisplayMode& dm) {
-
- J2dTraceLn4(J2D_TRACE_INFO, "DDraw::SetDisplayMode %dx%dx%d, %d",
- dm.width, dm.height, dm.bitDepth, dm.refreshRate);
-
- HRESULT ddResult;
- // Sleep so that someone can't programatically set the display mode
- // multiple times very quickly and accidentally crash the driver
- static DWORD prevTime = 0;
- DWORD currTime = ::GetTickCount();
- DWORD timeDiff = (currTime - prevTime);
- if (timeDiff < 500) {
- ::Sleep(500 - timeDiff);
- }
- prevTime = currTime;
-
- ddResult = dxObject->SetDisplayMode(dm.width, dm.height, dm.bitDepth,
- dm.refreshRate);
-
- return ddResult;
-}
-
-/**
- * Private callback used by EnumDisplayModes
- */
-long WINAPI DXObject::EnumCallback(LPDDSURFACEDESC2 pDDSD,
- void* pContext)
-{
- EnumDisplayModesParam* pParam =
- (EnumDisplayModesParam*)pContext;
- DDrawDisplayMode::Callback callback = pParam->callback;
- void* context = pParam->context;
-
- DDrawDisplayMode displayMode(pDDSD->dwWidth, pDDSD->dwHeight,
- pDDSD->ddpfPixelFormat.dwRGBBitCount, pDDSD->dwRefreshRate);
- (*callback)(displayMode, context);
- return DDENUMRET_OK;
-}
-
-BOOL DDraw::EnumDDDisplayModes(DDrawDisplayMode* constraint,
- DDrawDisplayMode::Callback callback, void* context)
-{
- HRESULT ddResult;
-
- ddResult = dxObject->EnumDisplayModes(constraint, callback, context);
-
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDraw::EnumDisplayModes");
- return FALSE;
- }
-
- return TRUE;
-}
-
-BOOL DDraw::RestoreDDDisplayMode() {
- HRESULT ddResult;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::RestoreDDDisplayMode");
- ddResult = dxObject->RestoreDisplayMode();
-
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDraw::RestoreDDDisplayMode");
- return FALSE;
- }
- return TRUE;
-}
-
-HRESULT DDraw::SetCooperativeLevel(HWND hwnd, DWORD dwFlags)
-{
- HRESULT ddResult;
- J2dTraceLn(J2D_TRACE_INFO, "DDraw::SetCooperativeLevel");
- ddResult = dxObject->SetCooperativeLevel(hwnd, dwFlags);
- /* On some hardware (Radeon 7500 and GeForce2), attempting
- * to use the d3d device created prior to running FS|EX
- * may cause a system crash. A workaround is to restore
- * the primary surface and recreate the
- * 3d device on the restored surface.
- */
- if ((ddResult == DD_OK) && lpPrimary && d3dContext) {
-
- lpPrimary->GetExclusiveAccess();
- if (lpPrimary->IsLost() != DD_OK) {
- // Only bother with workaround if the primary has been lost
- // Note that this call may fail with DDERR_WRONGMODE if
- // the surface was created in a different mode, but we
- // do not want to propagate this (non-fatal) error.
- HRESULT res = lpPrimary->Restore();
- if (FAILED(res)) {
- DebugPrintDirectDrawError(res,
- "DDraw::SetCooperativeLevel:"
- " lpPrimary->Restore() failed");
- }
- }
- lpPrimary->ReleaseExclusiveAccess();
- }
- return ddResult;
-}
-
-
-DXSurface::DXSurface(IDirectDrawSurface7 *lpSurface)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DXSurface::DXSurface");
- this->lpSurface = lpSurface;
- this->versionID = VERSION_DX7;
- this->depthBuffer = NULL;
- memset(&ddsd, 0, sizeof(ddsd));
- ddsd.dwSize = sizeof(ddsd);
- lpSurface->GetSurfaceDesc(&ddsd);
- width = ddsd.dwWidth;
- height = ddsd.dwHeight;
-}
-
-HRESULT DXSurface::Restore() {
- J2dTraceLn(J2D_TRACE_INFO, "DXSurface::Restore");
- HRESULT resDepth = D3D_OK, resSurface;
- if (depthBuffer != NULL) {
- J2dTraceLn(J2D_TRACE_VERBOSE, " restoring depth buffer");
- resDepth = depthBuffer->Restore();
- }
- // If this is an attached backbuffer surface, we should not
- // restore it explicitly, as it will be restored implicitly with the
- // primary surface's restoration. But we did need to restore the
- // depth buffer, because it won't be restored with the primary.
- if ((ddsd.ddsCaps.dwCaps & DDSCAPS_BACKBUFFER) != 0) {
- return resDepth;
- }
- resSurface = lpSurface->Restore();
- return FAILED(resDepth) ? resDepth : resSurface;
-}
-HRESULT DXSurface::Lock(RECT *lockRect, SurfaceDataRasInfo *pRasInfo,
- DWORD dwFlags, HANDLE hEvent)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DXSurface::Lock");
- HRESULT retValue = lpSurface->Lock(lockRect, &ddsd, dwFlags, hEvent);
- if (SUCCEEDED(retValue) && pRasInfo) {
- // Someone might call Lock() just to synchronize, in which case
- // they don't care about the result and pass in NULL for pRasInfo
- pRasInfo->pixelStride = ddsd.ddpfPixelFormat.dwRGBBitCount / 8;
- pRasInfo->pixelBitOffset = ddsd.ddpfPixelFormat.dwRGBBitCount & 7;
- pRasInfo->scanStride = ddsd.lPitch;
- pRasInfo->rasBase = (void *) ddsd.lpSurface;
- }
- return retValue;
-}
-
-HRESULT DXSurface::AttachDepthBuffer(DXObject *dxObject,
- BOOL bAccelerated,
- DDPIXELFORMAT *pddpf)
-{
- HRESULT res;
-
- J2dTraceLn(J2D_TRACE_INFO, "DXSurface::AttachDepthBuffer");
- J2dTraceLn1(J2D_TRACE_VERBOSE, " bAccelerated=%d", bAccelerated);
- // we already have a depth buffer
- if (depthBuffer != NULL) {
- J2dTraceLn(J2D_TRACE_VERBOSE,
- " depth buffer already created");
- // we don't want to restore the depth buffer
- // here if it was lost, it will be restored when
- // the surface is restored.
- if (FAILED(res = depthBuffer->IsLost())) {
- J2dTraceLn(J2D_TRACE_WARNING,
- "DXSurface::AttachDepthBuffer: depth buffer is lost");
- }
- return res;
- }
-
- DWORD flags = DDSD_WIDTH|DDSD_HEIGHT|DDSD_CAPS|DDSD_PIXELFORMAT;
- DWORD caps = DDSCAPS_ZBUFFER;
- if (bAccelerated) {
- caps |= DDSCAPS_VIDEOMEMORY;
- } else {
- caps |= DDSCAPS_SYSTEMMEMORY;
- }
- if (SUCCEEDED(res =
- dxObject->CreateSurface(flags, caps, pddpf,
- GetWidth(), GetHeight(),
- (DXSurface **)&depthBuffer, 0)))
- {
- if (FAILED(res =
- lpSurface->AddAttachedSurface(depthBuffer->GetDDSurface())))
- {
- DebugPrintDirectDrawError(res, "DXSurface::AttachDepthBuffer: "\
- "failed to attach depth buffer");
- depthBuffer->Release();
- delete depthBuffer;
- depthBuffer = NULL;
- }
- return res;
- }
- DebugPrintDirectDrawError(res, "DXSurface::AttachDepthBuffer: "\
- "depth buffer creation failed");
- return res;
-}
-
-HRESULT DXSurface::GetAttachedSurface(DWORD dwCaps, DXSurface **bbSurface)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DXSurface::GetAttachedSurface");
- IDirectDrawSurface7 *lpDDSBack;
- HRESULT retValue;
- DDSCAPS2 ddsCaps;
- memset(&ddsCaps, 0, sizeof(ddsCaps));
- ddsCaps.dwCaps = dwCaps;
-
- retValue = lpSurface->GetAttachedSurface(&ddsCaps, &lpDDSBack);
- if (retValue == DD_OK) {
- *bbSurface = new DXSurface(lpDDSBack);
- }
- return retValue;
-}
-
-HRESULT DXSurface::SetClipper(DDrawClipper *pClipper)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DXSurface::SetClipper");
- // A NULL pClipper is valid; it means we want no clipper
- // on this surface
- IDirectDrawClipper *actualClipper = pClipper ?
- pClipper->GetClipper() :
- NULL;
- // Calling SetClipper(NULL) on a surface that currently does
- // not have a clipper can cause a crash on some devices
- // (e.g., Matrox G400), so only call SetClipper(NULL) if
- // there is currently a non-NULL clipper set on this surface.
- if (actualClipper || clipperSet) {
- clipperSet = (actualClipper != NULL);
- return lpSurface->SetClipper(actualClipper);
- }
- return DD_OK;
-}
-
-int DXSurface::GetSurfaceDepth()
-{
- if (FAILED(lpSurface->GetSurfaceDesc(&ddsd))) {
- // Failure: return 0 as an error indication
- return 0;
- }
- return ddsd.ddpfPixelFormat.dwRGBBitCount;
-}
-
-/**
- * Class DDrawSurface
- *
- * This class handles all operations on DirectDraw surfaces.
- * Mostly, it is a wrapper for the standard ddraw operations on
- * surfaces, but it also provides some additional functionality.
- * There is a surfaceLock CriticalSection associated with every
- * DDrawSurface which is used to make each instance MT-safe.
- * In general, ddraw itself is MT-safe, but we need to ensure
- * that accesses to our internal variables are MT-safe as well.
- * For example, we may need to recreate the primary surface
- * (during a display-mode-set operation) or release a ddraw
- * surface (due to a call to GraphicsDevice.flush()). The surfaceLock
- * enables us to do these operations without putting other threads
- * in danger of dereferencing garbage memory.
- *
- * If a surface has been released but other threads are still
- * using it, most methods simply return DD_OK and the calling
- * thread can go about its business without worrying about the
- * failure. Some methods (Lock and GetDC) return an error
- * code so that the caller does not base further operations on
- * an unsuccessful lock call.
- */
-
-DDrawSurface::DDrawSurface()
-{
- surfaceLock = new DDCriticalSection(this);
- //d3dObject = NULL;
-}
-
-DDrawSurface::DDrawSurface(DDraw *ddObject, DXSurface *dxSurface)
-{
- surfaceLock = new DDCriticalSection(this);
- CRITICAL_SECTION_ENTER(*surfaceLock);
- this->ddObject = ddObject;
- this->dxSurface = dxSurface;
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::DDrawSurface: dxSurface=0x%x", dxSurface);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
-}
-
-DDrawSurface::~DDrawSurface()
-{
- ReleaseSurface();
- delete surfaceLock;
-}
-
-/**
- * This function can only be called when the caller has exclusive
- * access to the DDrawSurface object. This is done because some
- * surfaces (e.g., the primary surface) must be released before a
- * new one can be created and the surfaceLock must be held during
- * the entire process so that no other thread can access the
- * lpSurface before the process is complete.
- */
-void DDrawSurface::SetNewSurface(DXSurface *dxSurface)
-{
- this->dxSurface = dxSurface;
-}
-
-HRESULT DDrawSurface::ReleaseSurface() {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::ReleaseSurface: dxSurface=0x%x", dxSurface);
- FlushD3DContext();
- HRESULT retValue = dxSurface->Release();
- dxSurface = NULL;
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-HRESULT DDrawSurface::SetClipper(DDrawClipper* pClipper) {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::SetClipper: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- HRESULT retValue = dxSurface->SetClipper(pClipper);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-HRESULT DDrawSurface::SetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey) {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::SetColorKey: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- HRESULT retValue = dxSurface->SetColorKey(dwFlags, lpDDColorKey);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-HRESULT DDrawSurface::GetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey) {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::GetColorKey: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DDERR_NOCOLORKEY;
- }
- HRESULT retValue = dxSurface->GetColorKey(dwFlags, lpDDColorKey);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-/**
- * NOTE: This function takes the surfaceLock critical section, but
- * does not release that lock. The
- * Unlock method for this surface MUST be called before anything
- * else can happen on the surface. This is necessary to prevent the
- * surface from being released or recreated while it is being used.
- * See also Unlock(), GetDC(), and ReleaseDC().
- */
-HRESULT DDrawSurface::Lock(LPRECT lockRect, SurfaceDataRasInfo *pRasInfo,
- DWORD dwFlags, HANDLE hEvent) {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::Lock: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- // Return error here so that caller does not assume
- // lock worked and perform operations on garbage data
- // based on that assumption
- return DDERR_INVALIDOBJECT;
- }
-
- FlushD3DContext();
- HRESULT retValue = dxSurface->Lock(lockRect, pRasInfo, dwFlags, hEvent);
- if (retValue != DD_OK) {
- // Failure should release CriticalSection: either the lock will
- // be attempted again (e.g., DDERR_SURFACEBUSY) or the lock
- // failed and DDUnlock will not be called.
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- }
-
- return retValue;
-}
-
-HRESULT DDrawSurface::Unlock(LPRECT lockRect) {
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::Unlock: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- HRESULT retValue = dxSurface->Unlock(lockRect);
- if (retValue != DD_OK && lockRect) {
- // Strange and undocumented bug using pre-DX7 interface;
- // for some reason unlocking the same rectangle as we
- // locked returns a DDERR_NOTLOCKED error, but unlocking
- // NULL (the entire surface) seems to work instead. It is
- // as if Lock(&rect) actually performs Lock(NULL) implicitly,
- // thus causing Unlock(&rect) to fail but Unlock(NULL) to
- // succeed. Trap this error specifically and try the workaround
- // of attempting to unlock the whole surface instead.
- retValue = dxSurface->Unlock(NULL);
- }
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-HRESULT DDrawSurface::Blt(LPRECT destRect, DDrawSurface* pSrc,
- LPRECT srcRect, DWORD dwFlags, LPDDBLTFX lpDDBltFx) {
- LPDIRECTDRAWSURFACE lpSrc = NULL;
- DXSurface *dxSrcSurface = NULL;
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::Blt: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- if (pSrc) {
- pSrc->GetExclusiveAccess();
- dxSrcSurface = pSrc->dxSurface;
- if (!dxSrcSurface || (dxSrcSurface->IsLost() != DD_OK)) {
- // If no src surface, then surface must have been released
- // by some other thread. If src is lost, then we should not
- // attempt this operation (causes a crash on some framebuffers).
- // Return SURFACELOST error in IsLost() case to force surface
- // restoration as necessary.
- HRESULT retError;
- if (!dxSrcSurface) {
- retError = DD_OK;
- } else {
- retError = DDERR_SURFACELOST;
- }
- pSrc->ReleaseExclusiveAccess();
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retError;
- }
- pSrc->FlushD3DContext();
- }
- FlushD3DContext();
- HRESULT retValue = dxSurface->Blt(destRect, dxSrcSurface, srcRect, dwFlags,
- lpDDBltFx);
- if (pSrc) {
- pSrc->ReleaseExclusiveAccess();
- }
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-void DDrawSurface::FlushD3DContext(BOOL bForce) {
- D3DContext *d3dContext = ddObject->GetD3dContext();
- if (d3dContext) {
- d3dContext->FlushD3DQueueForTarget(bForce ? NULL : this);
- }
-}
-
-HRESULT DDrawSurface::Flip(DDrawSurface* pDest, DWORD dwFlags) {
- J2dTraceLn2(J2D_TRACE_INFO, "DDrawSurface::Flip this=0x%x pDest=0x%x",
- this, pDest);
- CRITICAL_SECTION_ENTER(*surfaceLock);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- pDest->GetExclusiveAccess();
- if (!pDest->dxSurface) {
- pDest->ReleaseExclusiveAccess();
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- // Flush the queue unconditionally
- FlushD3DContext(TRUE);
- HRESULT retValue = dxSurface->Flip(dwFlags);
- pDest->ReleaseExclusiveAccess();
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-HRESULT DDrawSurface::IsLost() {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::IsLost: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- HRESULT retValue = dxSurface->IsLost();
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-HRESULT DDrawSurface::Restore() {
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::Restore: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- FlushD3DContext();
- HRESULT retValue = dxSurface->Restore();
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-/**
- * Returns the bit depth of the ddraw surface
- */
-int DDrawSurface::GetSurfaceDepth() {
- int retValue = 0; // default value; 0 indicates some problem getting depth
- CRITICAL_SECTION_ENTER(*surfaceLock);
- if (dxSurface) {
- retValue = dxSurface->GetSurfaceDepth();
- }
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-/**
- * As in Lock(), above, we grab the surfaceLock in this function,
- * but do not release it until ReleaseDC() is called. This is because
- * these functions must be called as a pair (they take a lock on
- * the surface inside the ddraw runtime) and the surface should not
- * be released or recreated while the DC is held. The caveat is that
- * a failure in this method causes us to release the surfaceLock here
- * because we will not (and should not) call ReleaseDC if we are returning
- * an error from GetDC.
- */
-HRESULT DDrawSurface::GetDC(HDC *pHDC) {
- *pHDC = (HDC)NULL;
- CRITICAL_SECTION_ENTER(*surfaceLock);
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::GetDC: dxSurface=0x%x", dxSurface);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DDERR_GENERIC;
- }
- FlushD3DContext();
- HRESULT ddResult = dxSurface->GetDC(pHDC);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDrawSurface::GetDC");
- if (*pHDC != (HDC)NULL) {
- // Probably cannot reach here; we got an error
- // but we also got a valid hDC. Release it and
- // return NULL. Note that releasing the DC also
- // releases the surfaceLock so we do not duplicate
- // that here
- ReleaseDC(*pHDC);
- *pHDC = (HDC)NULL;
- } else {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- }
- }
- return ddResult;
-}
-
-HRESULT DDrawSurface::ReleaseDC(HDC hDC) {
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDrawSurface::ReleaseDC: dxSurface=0x%x", dxSurface);
- if (!hDC) {
- // We should not get here, but just in case we need to trap this
- // situation and simply noop. Note that we do not release the
- // surfaceLock because we already released it when we failed to
- // get the HDC in the first place in GetDC
- J2dRlsTraceLn(J2D_TRACE_ERROR, "DDrawSurface::ReleaseDC: Null "\
- "HDC received in ReleaseDC");
- return DD_OK;
- }
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
- }
- HRESULT retValue = dxSurface->ReleaseDC(hDC);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return retValue;
-}
-
-/**
- * Class DDrawPrimarySurface
- * This sublcass of DDrawSurface handles primary-specific
- * functionality. In particular, the primary can have a
- * back buffer associated with it; DDrawPrimarySurface holds
- * the reference to that shared resource.
- */
-DDrawPrimarySurface::DDrawPrimarySurface() : DDrawSurface()
-{
- bbHolder = NULL;
-}
-
-DDrawPrimarySurface::DDrawPrimarySurface(DDraw *ddObject,
- DXSurface *dxSurface) :
- DDrawSurface(ddObject, dxSurface)
-{
- bbHolder = NULL;
-}
-
-DDrawPrimarySurface::~DDrawPrimarySurface() {
-}
-
-HRESULT DDrawPrimarySurface::ReleaseSurface() {
- J2dTraceLn(J2D_TRACE_INFO, "DDrawPrimarySurface::ReleaseSurface");
- if (bbHolder) {
- delete bbHolder;
- bbHolder = NULL;
- }
- return DDrawSurface::ReleaseSurface();
-}
-
-void DDrawPrimarySurface::SetNewSurface(DXSurface *dxSurface)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDrawPrimarySurface::SetNewSurface");
- if (bbHolder) {
- delete bbHolder;
- bbHolder = NULL;
- }
- DDrawSurface::SetNewSurface(dxSurface);
-}
-
-DDrawSurface* DDrawPrimarySurface::GetDDAttachedSurface(DWORD caps) {
- J2dTraceLn(J2D_TRACE_INFO, "DDrawPrimarySurface::GetDDAttachedSurface");
- if (!bbHolder) {
- HRESULT ddResult;
- DWORD dwCaps;
- if (caps == 0) {
- dwCaps = DDSCAPS_BACKBUFFER;
- } else {
- dwCaps = caps;
- }
-
- DXSurface *dxSurfaceBB;
-
- CRITICAL_SECTION_ENTER(*surfaceLock);
- if (!dxSurface) {
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return NULL;
- }
- ddResult = dxSurface->GetAttachedSurface(dwCaps, &dxSurfaceBB);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDrawPrimarySurface::GetDDAttachedSurface failed");
- return NULL;
- }
- bbHolder = new BackBufferHolder(dxSurfaceBB);
- }
- return new DDrawBackBufferSurface(ddObject, bbHolder);
-}
-
-/**
- * Primary restoration is different from non-primary because
- * of the d3dContext object. There is a bug (4754180) on some
- * configurations (including Radeon and GeForce2) where using
- * the d3dDevice associated with a primary that is either lost
- * or has been restored can crash the system. The solution is
- * to force a primary restoration at the appropriate time and
- * to recreate the d3d device associated with that primary.
- */
-HRESULT DDrawPrimarySurface::Restore() {
- J2dTraceLn(J2D_TRACE_INFO, "DDrawPrimarySurface::Restore");
- AwtToolkit::GetInstance().SendMessage(WM_AWT_D3D_RELEASE_DEVICE,
- (WPARAM)ddObject, NULL);
-
- J2dTraceLn(J2D_TRACE_VERBOSE, " Restoring primary surface");
- HRESULT res = DDrawSurface::Restore();
- if (SUCCEEDED(res) && bbHolder != NULL) {
- res = bbHolder->RestoreDepthBuffer();
- }
- return res;
-}
-
-
-/**
- * Class DDrawBackBufferSurface
- * This subclass handles functionality that is specific to
- * the back buffer surface. The back buffer is a different
- * type of surface than a typical ddraw surface; it is
- * created by the primary surface and restored/released
- * implicitly by similar operations on the primary.
- * There is only one back buffer per primary, so each
- * DDrawBackBufferSurface object which refers to that object
- * shares the reference to it. In order to appropriately
- * share this resource (both avoid creating too many objects
- * and avoid leaking those that we create), we use the
- * BackBufferHolder structure to contain the single ddraw
- * surface and register ourselves with that object. This
- * allows us to have multi-threaded access to the back buffer
- * because if it was somehow deleted by another thread while we
- * are still using it, then the reference to our lpSurface will
- * be nulled-out for us and we will noop operations on that
- * surface (instead of crashing due to dereferencing a released
- * resource).
- */
-DDrawBackBufferSurface::DDrawBackBufferSurface() : DDrawSurface() {
-}
-
-DDrawBackBufferSurface::DDrawBackBufferSurface(DDraw *ddObject,
- BackBufferHolder *holder) :
- DDrawSurface(ddObject, holder->GetBackBufferSurface())
-{
- J2dTraceLn(J2D_TRACE_INFO,
- "DDrawBackBufferSurface::DDrawBackBufferSurface");
- CRITICAL_SECTION_ENTER(*surfaceLock);
- // Register ourselves with the back buffer container.
- // This means that we will be updated by that container
- // if the back buffer goes away.
- bbHolder = holder;
- bbHolder->Add(this);
- CRITICAL_SECTION_LEAVE(*surfaceLock);
-}
-
-/**
- * This destructor removes us from the list of back buffers
- * that hold pointers to the one true back buffer. It also
- * nulls-out references to the ddraw and d3d objects to make
- * sure that our parent class does not attempt to release
- * those objects.
- */
-DDrawBackBufferSurface::~DDrawBackBufferSurface() {
- J2dTraceLn(J2D_TRACE_INFO,
- "DDrawBackBufferSurface::~DDrawBackBufferSurface");
- CRITICAL_SECTION_ENTER(*surfaceLock);
- if (bbHolder) {
- // Tell the back buffer container that we are no
- // longer alive; otherwise it will try to update
- // us when the back buffer dies.
- bbHolder->Remove(this);
- }
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- // Note: our parent class destructor also calls ReleaseSurface,
- // but a function called within a destructor calls only the local
- // version, not the overridden version. So we first call our
- // overridden ReleaseSurface to make sure our variables are nulled-out
- // before calling the superclass which will attempt to release
- // non-null objects.
- ReleaseSurface();
-}
-
-/**
- * Note that in this subclass' version of ReleaseSurface
- * we merely null-out the references to our objects.
- * They are shared resources and will be deleted elsewhere.
- */
-HRESULT DDrawBackBufferSurface::ReleaseSurface() {
- J2dTraceLn(J2D_TRACE_INFO, "DDrawBackBufferSurface::ReleaseSurface");
- CRITICAL_SECTION_ENTER(*surfaceLock);
- bbHolder = NULL;
- dxSurface = NULL;
- CRITICAL_SECTION_LEAVE(*surfaceLock);
- return DD_OK;
-}
-
-/**
- * Class BackBufferHolder
- * This class holds the real ddraw/d3d back buffer surfaces.
- * It also contains a list of everyone that is currently
- * sharing those resources. When the back buffer goes away
- * due to the primary being released or deleted), then
- * we tell everyone on the list that the back buffer is
- * gone (by nulling out their references to that object)
- * and thus avoid dereferencing a released resource.
- */
-BackBufferHolder::BackBufferHolder(DXSurface *backBuffer)
-{
- this->backBuffer = backBuffer;
- bbList = NULL;
-}
-
-/**
- * The back buffer is going away; iterate through our
- * list and tell all of those objects that information.
- * Then go ahead and actually release the back buffer's
- * resources.
- */
-BackBufferHolder::~BackBufferHolder()
-{
- bbLock.Enter();
- BackBufferList *bbListPtr = bbList;
- while (bbListPtr) {
- bbListPtr->backBuffer->ReleaseSurface();
- BackBufferList *bbTmp = bbListPtr;
- bbListPtr = bbListPtr->next;
- delete bbTmp;
- }
- // Note: don't release the ddraw surface; this is
- // done implicitly through releasing the primary
- //if (backBuffer3D) {
- //backBuffer3D->Release();
- //}
- bbLock.Leave();
-}
-
-/**
- * Add a new client to the list of objects sharing the
- * back buffer.
- */
-void BackBufferHolder::Add(DDrawBackBufferSurface *surf)
-{
- bbLock.Enter();
- BackBufferList *bbTmp = new BackBufferList;
- bbTmp->next = bbList;
- bbTmp->backBuffer = surf;
- bbList = bbTmp;
- bbLock.Leave();
-}
-
-/**
- * Remove a client from the sharing list. This happens when
- * a client is deleted; we need to remove it from the list
- * so that we do not later go to update a defunct client
- * in the ~BackBufferHolder() destructor.
- */
-void BackBufferHolder::Remove(DDrawBackBufferSurface *surf)
-{
- bbLock.Enter();
- BackBufferList *bbListPtr = bbList;
- BackBufferList *bbListPtrPrev = NULL;
- while (bbListPtr) {
- if (bbListPtr->backBuffer == surf) {
- BackBufferList *bbTmp = bbListPtr;
- if (!bbListPtrPrev) {
- bbList = bbListPtr->next;
- } else {
- bbListPtrPrev->next = bbTmp->next;
- }
- delete bbTmp;
- break;
- }
- bbListPtrPrev = bbListPtr;
- bbListPtr = bbListPtr->next;
- }
- bbLock.Leave();
-}
-
-HRESULT BackBufferHolder::RestoreDepthBuffer() {
- J2dTraceLn(J2D_TRACE_INFO,
- "BackBufferHolder::RestoreDepthBuffer");
- if (backBuffer != NULL) {
- // this restores the depth-buffer attached
- // to the back-buffer. The back-buffer itself is restored
- // when the primary surface is restored, but the depth buffer
- // needs to be restored manually.
- return backBuffer->Restore();
- } else {
- return D3D_OK;
- }
-}
-
-/**
- * Class DDrawClipper
- */
-DDrawClipper::DDrawClipper(LPDIRECTDRAWCLIPPER clipper) : lpClipper(clipper) {}
-
-DDrawClipper::~DDrawClipper()
-{
- if (lpClipper) {
- lpClipper->Release();
- }
-}
-
-HRESULT DDrawClipper::SetHWnd(DWORD dwFlags, HWND hwnd)
-{
- return lpClipper->SetHWnd(dwFlags, hwnd);
-}
-
-HRESULT DDrawClipper::GetClipList(LPRECT rect, LPRGNDATA rgnData,
- LPDWORD rgnSize)
-{
- return lpClipper->GetClipList(rect, rgnData, rgnSize);
-}
-
-LPDIRECTDRAWCLIPPER DDrawClipper::GetClipper()
-{
- return lpClipper;
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/ddrawObject.h b/jdk/src/windows/native/sun/java2d/windows/ddrawObject.h
deleted file mode 100644
index 16a34f82341..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/ddrawObject.h
+++ /dev/null
@@ -1,481 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#ifndef DDRAWOBJECT_H
-#define DDRAWOBJECT_H
-
-#include
-#include
-#include
-#include
-#include "Win32SurfaceData.h"
-#include "GraphicsPrimitiveMgr.h"
-
-#ifdef DEBUG
- #define DX_FUNC(func) do { \
- HRESULT ddr = (func); \
- if (FAILED(ddr)) \
- DebugPrintDirectDrawError(ddr, #func); \
- } while (0)
-#else
- #define DX_FUNC(func) do { func; } while (0)
-#endif
-
-/**
- * Class for display modes
- */
-class DDrawDisplayMode {
-
-public:
- typedef void (*Callback)(DDrawDisplayMode&, void*);
-
-public:
- DDrawDisplayMode();
- DDrawDisplayMode(DDrawDisplayMode& rhs);
- DDrawDisplayMode(jint w, jint h, jint b, jint r);
- virtual ~DDrawDisplayMode();
-
- jint width;
- jint height;
- jint bitDepth;
- jint refreshRate;
-
-};
-
-class DDrawSurface;
-class DDrawClipper;
-class DXSurface;
-
-class D3DContext;
-
-
-class DXObject {
-private:
- IDirectDraw7 *ddObject;
- HMONITOR hMonitor;
- static long WINAPI EnumCallback(LPDDSURFACEDESC2 pDDSD, void* pContext);
-
-public:
- DXObject(IDirectDraw7 *ddObject, HMONITOR hMonitor) {
- this->ddObject = ddObject;
- this->hMonitor = hMonitor;
- }
- virtual ~DXObject();
- HRESULT GetCaps(LPDDCAPS halCaps, LPDDCAPS helCaps) {
- return ddObject->GetCaps(halCaps, helCaps);
- }
- HMONITOR GetHMonitor() { return hMonitor; }
- HRESULT GetAvailableVidMem(DWORD caps, DWORD *total,
- DWORD *free);
- HRESULT CreateSurface(DWORD dwFlags,
- DWORD ddsCaps, DWORD ddsCaps2,
- LPDDPIXELFORMAT lpPf,
- int width, int height,
- DXSurface **lpDDSurface,
- int numBackBuffers);
-
- HRESULT CreateSurface(DWORD dwFlags, DWORD ddsCaps,
- LPDDPIXELFORMAT lpPf,
- int width, int height,
- DXSurface **lpDDSurface,
- int numBackBuffers = 0)
- {
- return CreateSurface(dwFlags, ddsCaps, 0, lpPf, width, height,
- lpDDSurface, numBackBuffers);
- }
- HRESULT CreateSurface(DWORD dwFlags, DWORD ddsCaps,
- DXSurface **lpDDSurface)
- {
- return CreateSurface(dwFlags, ddsCaps, NULL, 0, 0, lpDDSurface, 0);
- }
- HRESULT CreateSurface(DWORD dwFlags, DWORD ddsCaps,
- DXSurface **lpDDSurface,
- int numBackBuffers)
- {
- return CreateSurface(dwFlags, ddsCaps, NULL, 0, 0, lpDDSurface,
- numBackBuffers);
- }
-
- HRESULT CreateClipper(DWORD dwFlags,
- LPDIRECTDRAWCLIPPER FAR *lplpDDClipper)
- {
- return ddObject->CreateClipper(dwFlags, lplpDDClipper, NULL);
- }
- HRESULT GetDisplayMode(DDrawDisplayMode &dm);
- HRESULT SetDisplayMode(DWORD width, DWORD height, DWORD depth,
- DWORD refreshRate)
- {
- return ddObject->SetDisplayMode(width, height, depth, refreshRate, 0);
- }
- HRESULT EnumDisplayModes(DDrawDisplayMode *dm,
- DDrawDisplayMode::Callback callback,
- void *context);
- HRESULT RestoreDisplayMode() {
- return ddObject->RestoreDisplayMode();
- }
- HRESULT SetCooperativeLevel(HWND hWnd, DWORD dwFlags) {
- return ddObject->SetCooperativeLevel(hWnd,
- (dwFlags | DDSCL_FPUPRESERVE));
- }
- HRESULT CreateD3DObject(IDirect3D7 **d3dObject);
- /**
- * Structure for enumerating display modes; used to invoke the callback
- */
- class EnumDisplayModesParam {
- public:
- EnumDisplayModesParam(DDrawDisplayMode::Callback cb, void* ct);
- virtual ~EnumDisplayModesParam();
- DDrawDisplayMode::Callback callback;
- void* context;
- };
-};
-
-typedef HRESULT (WINAPI *FnDDCreateFunc)(GUID FAR * lpGUID,
- LPDIRECTDRAW FAR * lplpDD, IUnknown FAR * pUnkOuter);
-typedef HRESULT (WINAPI *FnDDCreateExFunc)(GUID FAR * lpGUID,
- LPVOID * lplpDD, REFIID refIID, IUnknown FAR * pUnkOuter);
-/**
- * Class for the direct draw object
- */
-class DDraw {
-
-private:
-
-public:
- DDraw(DXObject *dxObject);
- virtual ~DDraw();
-
- static DDraw *CreateDDrawObject(GUID *lpGUID, HMONITOR hMonitor);
-
- BOOL GetDDCaps(LPDDCAPS caps);
- HRESULT GetDDAvailableVidMem(DWORD *free);
- DDrawSurface* CreateDDOffScreenSurface(DWORD width, DWORD height,
- DWORD depth,
- jint transparency,
- DWORD surfaceTypeCaps);
- DDrawSurface* CreateDDPrimarySurface(DWORD backBufferCount);
- void InitD3DContext();
- void ReleaseD3DContext();
- void DisableD3D() { deviceUseD3D = FALSE; }
- BOOL IsD3DEnabled() { return deviceUseD3D; }
- D3DContext * GetD3dContext() { return d3dContext; };
-
- DDrawClipper* CreateDDClipper();
-
- BOOL GetDDDisplayMode(DDrawDisplayMode& dm);
- HRESULT SetDDDisplayMode(DDrawDisplayMode& dm);
- BOOL EnumDDDisplayModes(DDrawDisplayMode* constraint,
- DDrawDisplayMode::Callback callback, void* context);
- BOOL RestoreDDDisplayMode();
-
- HRESULT SetCooperativeLevel(HWND hwnd, DWORD dwFlags);
-
-private:
- DXObject *dxObject;
- DDrawSurface *lpPrimary;
- D3DContext *d3dContext;
- BOOL deviceUseD3D;
-};
-
-
-#define VERSION_DX7 0x00000007
-
-/**
- * DXSurface class implementating DX 7 interfaces
- * (IDirectDrawSurface7)
- */
-class DXSurface {
-public:
- IDirectDrawSurface7 *lpSurface;
- DDSURFACEDESC2 ddsd;
- DXSurface* depthBuffer;
-
-public:
- DXSurface() {
- versionID = VERSION_DX7; depthBuffer = NULL; clipperSet = FALSE;
- }
-
- DXSurface(IDirectDrawSurface7 *lpSurface);
-
- IDirectDrawSurface7 *GetDDSurface() { return lpSurface; }
- HRESULT Blt(RECT *destRect, DXSurface *lpSurfaceSrc,
- RECT *srcRect, DWORD dwFlags, LPDDBLTFX ddBltFx)
- {
- return lpSurface->Blt(destRect,
- lpSurfaceSrc ?
- lpSurfaceSrc->GetDDSurface() :
- NULL,
- srcRect, dwFlags, ddBltFx);
- }
- HRESULT Lock(RECT *lockRect, SurfaceDataRasInfo *pRasInfo,
- DWORD dwFlags, HANDLE hEvent);
- HRESULT Unlock(RECT *unlockRect) {
- return lpSurface->Unlock(unlockRect);
- }
- HRESULT Flip(DWORD dwFlags) {
- return lpSurface->Flip(NULL, dwFlags);
- }
- HRESULT IsLost() {
- HRESULT res = D3D_OK;
- if (depthBuffer != NULL) {
- res = depthBuffer->IsLost();
- }
- return FAILED(res) ? res : lpSurface->IsLost();
- }
- HRESULT Restore();
- HRESULT GetDC(HDC *hDC) {
- return lpSurface->GetDC(hDC);
- }
- HRESULT ReleaseDC(HDC hDC) {
- return lpSurface->ReleaseDC(hDC);
- }
- ULONG Release() {
- if (depthBuffer != NULL) {
- depthBuffer->Release();
- delete depthBuffer;
- depthBuffer = NULL;
- }
- return lpSurface->Release();
- }
- HRESULT SetClipper(DDrawClipper *pClipper);
- HRESULT SetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey) {
- return lpSurface->SetColorKey(dwFlags, lpDDColorKey);
- }
- HRESULT GetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey) {
- return lpSurface->GetColorKey(dwFlags, lpDDColorKey);
- }
- HRESULT GetAttachedSurface(DWORD dwCaps, DXSurface **bbSurface);
- int GetSurfaceDepth();
- HRESULT AttachDepthBuffer(DXObject* dxObject,
- BOOL bAccelerated,
- DDPIXELFORMAT* pddpf);
-
- DWORD GetWidth() { return width; }
- DWORD GetHeight() { return height; }
- DWORD GetVersionID() { return versionID; }
-
-private:
- DWORD width, height;
-protected:
- DWORD versionID;
- BOOL clipperSet;
-};
-
-
-/**
- * Class for direct draw surfaces
- */
-class DDrawSurface {
-
- friend class DDraw;
- friend class DDrawPrimarySurface;
- friend class DDrawBackBufferSurface;
-
-protected:
- DDrawSurface();
-
-public:
- virtual ~DDrawSurface();
- DDrawSurface(DDraw *ddObject, DXSurface *dxSurface);
-
-public:
- virtual void SetNewSurface(DXSurface *dxSurface);
- virtual HRESULT ReleaseSurface();
- virtual HRESULT SetClipper(DDrawClipper* pClipper);
- virtual HRESULT SetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey);
- virtual HRESULT GetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey);
- virtual HRESULT Lock(LPRECT lockRect = NULL, SurfaceDataRasInfo *pRasInfo = NULL,
- DWORD dwFlags = DDLOCK_WAIT, HANDLE hEvent = NULL);
- virtual HRESULT Unlock(LPRECT lockRect = NULL);
- virtual HRESULT Blt(LPRECT destRect, DDrawSurface* pSrc,
- LPRECT srcRect = NULL, DWORD dwFlags = DDBLT_WAIT,
- LPDDBLTFX lpDDBltFx = NULL);
- virtual HRESULT Flip(DDrawSurface* pDest, DWORD dwFlags = DDFLIP_WAIT);
- virtual HRESULT IsLost();
- /**
- * Restores the surface or the depth buffer if the surface
- * represents an attached backbuffer surface. In the latter case
- * the surface itself will be restored implicitly with the primary.
- */
- virtual HRESULT Restore();
- virtual HRESULT GetDC(HDC *hDC);
- virtual HRESULT ReleaseDC(HDC hDC);
- void GetExclusiveAccess() { CRITICAL_SECTION_ENTER(*surfaceLock); };
- void ReleaseExclusiveAccess() { CRITICAL_SECTION_LEAVE(*surfaceLock); };
- virtual DDrawSurface* GetDDAttachedSurface(DWORD caps = 0)
- { return NULL; };
- virtual DXSurface *GetDXSurface() { return dxSurface; }
- void FlushD3DContext(BOOL bForce = FALSE);
- int GetSurfaceDepth();
-
-protected:
- DDraw *ddObject;
- DXSurface *dxSurface;
- CriticalSection *surfaceLock;
-};
-
-class BackBufferHolder;
-
-/**
- * Class for direct draw primary surface
- */
-class DDrawPrimarySurface : DDrawSurface {
-
- friend class DDraw;
-
-protected:
- BackBufferHolder *bbHolder;
-
-protected:
- DDrawPrimarySurface(DDraw *ddObject, DXSurface *dxSurface);
- DDrawPrimarySurface();
-
-public:
- virtual ~DDrawPrimarySurface();
- virtual HRESULT ReleaseSurface();
- virtual void SetNewSurface(DXSurface *dxSurface);
- virtual DDrawSurface* GetDDAttachedSurface(DWORD caps = 0);
- virtual HRESULT Restore();
-};
-
-/**
- * Class for direct draw back buffer surface
- */
-class DDrawBackBufferSurface : DDrawSurface {
-
- friend class DDraw;
- friend class DDrawPrimarySurface;
-
-protected:
- DDrawPrimarySurface *lpPrimary;
- BackBufferHolder *bbHolder;
-
-protected:
- DDrawBackBufferSurface(DDraw *ddObject, BackBufferHolder *holder);
- DDrawBackBufferSurface();
-
-public:
- virtual ~DDrawBackBufferSurface();
- virtual HRESULT ReleaseSurface();
-};
-
-/**
- * Linked list holding all references to DDrawBackBufferSurface
- * objects that share a single ddraw surface. This class
- * is used by BackBufferHolder.
- */
-class BackBufferList {
-public:
- DDrawBackBufferSurface *backBuffer;
- BackBufferList *next;
-};
-
-/**
- * Class for storing the shared ddraw/d3d back buffer objects
- * and a list of all objects that use those shared surfaces.
- */
-class BackBufferHolder {
-
-public:
- BackBufferHolder(DXSurface *dxSurface);
- BackBufferHolder();
- ~BackBufferHolder();
-
- virtual void Add(DDrawBackBufferSurface *surf);
- virtual void Remove(DDrawBackBufferSurface *surf);
- DXSurface *GetBackBufferSurface() { return backBuffer; };
- HRESULT RestoreDepthBuffer();
-
-protected:
- BackBufferList *bbList; // linked list of objects that
- // share the ddraw/d3d surfaces
- DXSurface *backBuffer;
- CriticalSection bbLock; // synchronize accesses to list
-};
-
-
-#ifdef DEBUG
-void StackTrace();
-// Critical Section debugging class
-class DDCriticalSection : public CriticalSection {
-private:
- DDrawSurface* lpSurface;
- int count;
-
-public:
- DDCriticalSection(DDrawSurface* surface) : lpSurface(surface), count(0) {
- }
- void Enter() {
- ++count;
- //J2dTraceLn2(J2D_TRACE_VERBOSE,
- // "DDCriticalSection::Enter for surface 0x%x count %d\n",
- // lpSurface, count);
- CriticalSection::Enter();
- }
- void Leave() {
- //J2dTraceLn2(J2D_TRACE_VERBOSE,
- // "DDCriticalSection::Leave for surface 0x%x count %d\n",
- // lpSurface, count);
- if (count == 0) {
- //J2dTraceLn1(J2D_TRACE_VERBOSE,
- // "DDCriticalSection::Leave Invalid "\
- // "decrement in DDCriticalSection "\
- // "for surface 0x%x\n",
- // lpSurface);
- StackTrace();
- }
- CriticalSection::Leave();
- count--;
- }
-};
-#else
-#define DDCriticalSection(x) CriticalSection()
-#define StackTrace()
-#endif
-
-/**
- * Class for direct draw clippers
- */
-class DDrawClipper {
-
- friend class DDraw;
-
-private:
- DDrawClipper(LPDIRECTDRAWCLIPPER clipper);
-
-public:
- virtual ~DDrawClipper();
-
-public:
- HRESULT SetHWnd(DWORD dwFlags, HWND hwnd);
- HRESULT GetClipList(LPRECT lpRect, LPRGNDATA rgnData, LPDWORD rgnSize);
- LPDIRECTDRAWCLIPPER GetClipper();
-
-private:
- LPDIRECTDRAWCLIPPER lpClipper;
-};
-
-
-#endif DDRAWOBJECT_H
diff --git a/jdk/src/windows/native/sun/java2d/windows/ddrawUtils.cpp b/jdk/src/windows/native/sun/java2d/windows/ddrawUtils.cpp
deleted file mode 100644
index b672c3ef287..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/ddrawUtils.cpp
+++ /dev/null
@@ -1,1804 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#define INITGUID
-#include "Trace.h"
-#include "ddrawUtils.h"
-#include "ddrawObject.h"
-#include "awt_Multimon.h"
-#include "awt_MMStub.h"
-#include "dxInit.h"
-#include "WindowsFlags.h"
-#include "D3DContext.h"
-
-//
-// Globals
-//
-DDrawObjectStruct **ddInstance;
-int maxDDDevices = 0;
-int currNumDevices = 0;
-CriticalSection ddInstanceLock;
-extern BOOL isAppActive;
-extern HINSTANCE hLibDDraw; // DDraw Library handle
-extern jfieldID ddSurfacePuntedID;
-
-extern "C" void Win32OSSD_DisableDD(JNIEnv *env, Win32SDOps *wsdo);
-
-//
-// Constants
-//
-#define MAX_BUSY_ATTEMPTS 50 // Arbitrary number of times to attempt
- // an operation that returns a busy error
-
-//
-// Macros
-//
-
-/**
- * This macro is just a shortcut for the various places in the
- * code where we want to call a ddraw function and print any error
- * if the result is not equal to DD_OK. The errorString passed
- * in is for debugging/tracing purposes only.
- */
-#define DD_FUNC(func, errorString) { \
- HRESULT ddResult = func; \
- if (ddResult != DD_OK) { \
- DebugPrintDirectDrawError(ddResult, errorString); \
- } \
-}
-/**
- * Same as above, only return FALSE when done (to be used only in
- * functions that should return FALSE on a ddraw failure).
- */
-#define DD_FUNC_RETURN(func, errorString) { \
- HRESULT ddResult = func; \
- if (ddResult != DD_OK) { \
- DebugPrintDirectDrawError(ddResult, errorString); \
- return FALSE; \
- } \
-}
-
-//
-// INLINE functions
-//
-
-// Attaches the clipper object of a given surface to
-// the primary. Note that this action only happens if the
-// surface is onscreen (clipping only makes sense for onscreen windows)
-INLINE void AttachClipper(Win32SDOps *wsdo) {
- if (wsdo->window && wsdo->ddInstance->hwndFullScreen == NULL) {
- J2dTraceLn(J2D_TRACE_VERBOSE, "AttachClipper");
- HRESULT ddResult;
- ddResult = wsdo->ddInstance->clipper->SetHWnd(0, wsdo->window);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "AttachClipper");
- }
- }
-}
-
-//
-// Functions
-//
-
-/**
- * Returns the ddInstance associated with a particular HMONITOR
- */
-DDrawObjectStruct *GetDDInstanceForDevice(HMONITOR hMon)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "GetDDInstanceForDevice");
- DDrawObjectStruct *tmpDdInstance = NULL;
- ddInstanceLock.Enter();
- if (currNumDevices == 1) {
- // Non multimon situation
- if (ddInstance[0])
- {
- tmpDdInstance = ddInstance[0];
- }
- } else {
- for (int i = 0; i < currNumDevices; ++i) {
- if (ddInstance[i]
- && hMon == ddInstance[i]->hMonitor)
- {
- tmpDdInstance = ddInstance[i];
- break;
- }
- }
- }
- if (tmpDdInstance != NULL && !tmpDdInstance->accelerated) {
- // Some failure situations (see DDSetupDevice in dxInit.cpp) can cause
- // a ddInstance object to become invalid. If this happens, we should
- // not be using this ddInstance object at all, so return NULL.
- tmpDdInstance = NULL;
- }
- ddInstanceLock.Leave();
- return tmpDdInstance;
-}
-
-/**
- * Can return FALSE if there was some problem during ddraw
- * initialization for this screen, or if this screen does not
- * support some of the capabilities necessary for running ddraw
- * correctly.
- */
-BOOL DDCanCreatePrimary(HMONITOR hMon) {
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- return (useDD && ddInstance && tmpDdInstance);
-}
-
-/**
- * Can return FALSE if the device that the surfaceData object
- * resides on cannot support accelerated Blt's. Some devices
- * can perform basic ddraw Lock/Unlock commands but cannot
- * handle the ddraw Blt command.
- */
-BOOL DDCanBlt(Win32SDOps *wsdo) {
- return (useDD && wsdo->ddInstance && wsdo->ddInstance->canBlt);
-}
-
-/**
- * Can return FALSE if either ddraw is not enabled at all (problems
- * during initialization) or the device associated with the hMon object
- * cannot support the basic required capabilities (in
- * which case the ddInstance for that device will be set to NULL).
- */
-BOOL DeviceUseDDraw(HMONITOR hMon) {
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- return (useDD && tmpDdInstance && tmpDdInstance->ddObject);
-}
-
-/**
- * Can return FALSE if either ddraw is not enabled at all (problems
- * during initialization) or the device associated with the hMon object
- * cannot support the basic required capabilities (in
- * which case the ddInstance for that device will be set to NULL).
- */
-BOOL DeviceUseD3D(HMONITOR hMon) {
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- return (useDD && tmpDdInstance && tmpDdInstance->ddObject &&
- tmpDdInstance->ddObject->IsD3DEnabled());
-}
-
-/**
- * Can return FALSE if either ddraw is not enabled at all (problems
- * during initialization) or the device that the surfaceData object
- * resides on cannot support the basic required capabilities (in
- * which case the ddInstance for that device will be set to NULL).
- */
-BOOL DDUseDDraw(Win32SDOps *wsdo) {
- return (useDD && wsdo->ddInstance && wsdo->ddInstance->valid);
-}
-
-
-/**
- * Release the resources consumed by ddraw. This will be called
- * by the DllMain function when it receives a PROCESS_DETACH method,
- * meaning that the application is done with awt. We need to release
- * these ddraw resources because of potential memory leaks, but
- * more importantly, because if we don't release a primary surface
- * that has been locked and not unlocked, then we may cause
- * ddraw to be corrupted on this system until reboot.
- * IMPORTANT: because we do not use any locks around this release,
- * we assume that this function is called only during the
- * PROCESS_DETACH procedure described above. Any other situation
- * could cause unpredictable results.
- */
-void DDRelease()
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDRelease");
-
- // Note that we do not lock the ddInstanceLock CriticalSection.
- // Normally we should do that in this kind of situation (to ensure
- // that the ddInstance used in all release calls is the same on).
- // But in this case we do not want the release of a locked surface
- // to be hampered by some bad CriticalSection deadlock, so we
- // will just release ddInstance anyway.
- // Anyway, if users of this function call it properly (as
- // documented above), then there should be no problem.
- try {
- if (hLibDDraw) {
- ::FreeLibrary(hLibDDraw);
- hLibDDraw = NULL;
- }
- hLibDDraw = NULL;
- if (ddInstance) {
- for (int i = 0; i < currNumDevices; ++i) {
- ReleaseDDInstance(ddInstance[i]);
- }
- free(ddInstance);
- }
- } catch (...) {
- // Handle all exceptions by simply returning.
- // There are some cases where the OS may have already
- // released our objects for us (e.g., NT4) and we have
- // no way of knowing, but the above call into Release will
- // cause an exception to be thrown by dereferencing
- // already-released ddraw objects
- }
-}
-
-
-/**
- * Create the primary surface. Note that we do not take the ddInstanceLock
- * here; we assume that our callers are taking that lock for us.
- */
-BOOL DDCreatePrimary(Win32SDOps *wsdo) {
- J2dTraceLn(J2D_TRACE_INFO, "DDCreatePrimary");
- BOOL ret = TRUE;
-
- if (wsdo != NULL && wsdo->device != NULL) {
- HMONITOR hMon;
- hMon = (HMONITOR)wsdo->device->GetMonitor();
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- // Check if we need to recreate the primary for this device.
- // If we are in full-screen mode, we do not need to change
- // the primary unless the number of back buffers has changed.
- if (tmpDdInstance == NULL) {
- return FALSE;
- }
- if (tmpDdInstance->hwndFullScreen == NULL ||
- tmpDdInstance->context != CONTEXT_NORMAL)
- {
- ret = DDSetupDevice(tmpDdInstance,
- AwtWin32GraphicsDevice::GetDxCapsForDevice(hMon));
- tmpDdInstance->context = CONTEXT_NORMAL;
- }
- if (ret) {
- tmpDdInstance->valid = TRUE;
- }
- return ret;
- }
- return ret;
-}
-
-/**
- * Returns a DDrawSurface which should be used for performing DDraw sync.
- *
- * On systems other than Windows Vista, a primary surface is returned.
- *
- * On Windows Vista, a 1x1 scratch offscreen surface will be created and
- * maintained, because locking the primary surface will cause DWM to be
- * disabled for the run of the application.
- *
- * Note: this method must be called while ddInstance lock is held.
- * Note: a "DDINSTANCE_USABLE" non-null argument is assumed.
- */
-// 4978973: Only lock one pixel to flush; avoids GDI flicker artifacts
-// and only fill one pixel in the sync surface
-static RECT tinyRect = { 0, 0, 1, 1 };
-static DDrawSurface* DDGetSyncSurface(DDrawObjectStruct *tmpDdInstance)
-{
- static BOOL bIsVista = IS_WINVISTA;
- static DDBLTFX ddBltFx;
- DDrawSurface *lpSyncSurface;
-
- if (bIsVista) {
- lpSyncSurface = tmpDdInstance->syncSurface;
-
- if (lpSyncSurface != NULL) {
- // return the existing surface if it wasn't lost or it was
- // restored succesfully
- if (SUCCEEDED(lpSyncSurface->IsLost()) ||
- SUCCEEDED(lpSyncSurface->Restore()))
- {
- // we need to render to the sync surface so that DDraw
- // will flush the pipeline when we lock it in DDSync
- lpSyncSurface->Blt(&tinyRect, NULL, NULL,
- DDBLT_COLORFILL | DDBLT_WAIT,
- &ddBltFx);
- return lpSyncSurface;
- }
- J2dTraceLn(J2D_TRACE_WARNING, "DDGetSyncSurface: failed to restore"
- " sync surface, recreating");
- delete lpSyncSurface;
- }
-
- // this doesn't need to be done every time in the fast path
- ddBltFx.dwSize = sizeof(ddBltFx);
- ddBltFx.dwFillColor = 0xffffffff;
-
- lpSyncSurface =
- tmpDdInstance->ddObject->
- CreateDDOffScreenSurface(1, 1, 24/*ignored*/,
- TR_OPAQUE,
- DDSCAPS_VIDEOMEMORY);
- tmpDdInstance->syncSurface = lpSyncSurface;
- } else {
- lpSyncSurface = tmpDdInstance->primary;
- }
-
- return lpSyncSurface;
-}
-
-void
-DDFreeSyncSurface(DDrawObjectStruct *tmpDdInstance)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDFreeSyncSurface");
- if (tmpDdInstance != NULL && tmpDdInstance->syncSurface != NULL) {
- delete tmpDdInstance->syncSurface;
- tmpDdInstance->syncSurface = NULL;
- }
-}
-
-/**
- * Synchronize graphics pipeline by calling Lock/Unlock on primary
- * surface
- */
-void DDSync()
-{
- int attempts = 0;
- HRESULT ddResult;
- DDrawSurface *lpSyncSurface = NULL;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDSync");
- // REMIND: need to handle errors here
- ddInstanceLock.Enter();
- for (int i = 0; i < currNumDevices; ++i) {
- DDrawObjectStruct *tmpDdInstance = ddInstance[i];
-
- if (!DDINSTANCE_USABLE(tmpDdInstance)) {
- continue;
- }
- lpSyncSurface = DDGetSyncSurface(tmpDdInstance);
- if (lpSyncSurface == NULL) {
- J2dRlsTraceLn1(J2D_TRACE_ERROR,
- "DDSync: no sync surface for device %d", i);
- continue;
- }
- // Spin while busy up to some finite number of times
- do {
- ddResult = lpSyncSurface->Lock(&tinyRect, NULL,
- DDLOCK_WAIT, NULL);
- } while ((ddResult == DDERR_SURFACEBUSY) &&
- (++attempts < MAX_BUSY_ATTEMPTS));
- if (ddResult == DD_OK) {
- ddResult = lpSyncSurface->Unlock(&tinyRect);
- }
- }
- ddInstanceLock.Leave();
- J2dTraceLn(J2D_TRACE_VERBOSE, "DDSync done");
-}
-
-
-/**
- * Simple clip check against the window of the given surface data.
- * If the clip list is complex or if the clip list intersects
- * the visible region of the window then return FALSE, meaning
- * that the clipping is sufficiently complex that the caller
- * may want to find an alternative means (other than ddraw) of
- * performing an operation.
- */
-BOOL DDClipCheck(Win32SDOps *wsdo, RECT *operationRect)
-{
- static struct {
- RGNDATAHEADER rdh;
- RECT rects[1];
- } rgnData;
- unsigned long rgnSize = sizeof(rgnData);
- HRESULT ddResult;
-
- J2dTraceLn(J2D_TRACE_VERBOSE, "DDClipCheck");
-
- if (!wsdo->window) {
- // Offscreen surfaces need no clipping
- return TRUE;
- }
-
- // If ddResult not OK, could be because of a complex clipping region
- // (Our rgnData structure only has space for a simple rectangle region).
- // Thus, we return FALSE and attach the clipper object.
- DDrawObjectStruct *tmpDdInstance = wsdo->ddInstance;
- if (!DDINSTANCE_USABLE(tmpDdInstance)) {
- return FALSE;
- }
- if (wsdo->window == tmpDdInstance->hwndFullScreen) {
- // Fullscreen surfaces need no clipping
- return TRUE;
- }
- DD_FUNC(tmpDdInstance->clipper->SetHWnd(0, wsdo->window),
- "DDClipCheck: SetHWnd");
- ddResult = tmpDdInstance->clipper->GetClipList(NULL, (RGNDATA*)&rgnData,
- &rgnSize);
- if (ddResult == DDERR_REGIONTOOSMALL) {
- // Complex clipping region
- // REMIND: could be more clever here and actually check operationRect
- // against all rectangles in clipList, but this works for now.
- return FALSE;
- }
- // Check intersection of clip region with operationRect. If clip region
- // smaller, then we have a simple clip case.
- // If no operationRect, then check against entire window bounds.
- if (operationRect) {
- if (operationRect->left < rgnData.rects[0].left ||
- operationRect->top < rgnData.rects[0].top ||
- operationRect->right > rgnData.rects[0].right ||
- operationRect->bottom > rgnData.rects[0].bottom)
- {
- return FALSE;
- }
- } else {
- RECT winrect;
- ::GetWindowRect(wsdo->window, &winrect);
- if (winrect.left < rgnData.rects[0].left ||
- winrect.top < rgnData.rects[0].top ||
- winrect.right > rgnData.rects[0].right ||
- winrect.bottom > rgnData.rects[0].bottom)
- {
- return FALSE;
- }
- }
- return TRUE;
-}
-
-
-/**
- * Lock the surface.
- */
-BOOL DDLock(JNIEnv *env, Win32SDOps *wsdo, RECT *lockRect,
- SurfaceDataRasInfo *pRasInfo)
-{
- J2dTraceLn1(J2D_TRACE_INFO, "DDLock: wsdo->lpSurface=0x%x", wsdo->lpSurface);
-
- int attempts = 0;
-
- if (wsdo->gdiOpPending) {
- // This sync is really for flushing any pending GDI
- // operations. On ATI boards GdiFlush() doesn't do the trick,
- // only locking the primary works.
- DDSync();
- wsdo->gdiOpPending = FALSE;
- }
- while (attempts++ < MAX_BUSY_ATTEMPTS) {
- if (!wsdo->ddInstance->valid) {
- // If dd object became invalid, don't bother calling Lock
- // Note: This check should not be necessary because we should
- // do the right thing in any case - catch the error, try to
- // restore the surface, fai, etc. But there seem to be problems
- // with ddraw that sometimes cause it to hang in the Restore and
- // Lock calls. Better to avoid the situation as much as we can and
- // bail out early.
- J2dTraceLn(J2D_TRACE_ERROR,
- "DDLock: wsdo->ddInstance invalid");
- return FALSE;
- }
- HRESULT ddResult = wsdo->lpSurface->Lock(lockRect, pRasInfo,
- DDLOCK_WAIT, NULL);
- // Spin on the busy-type errors, else return having failed or succeeded
- switch (ddResult) {
- case DD_OK:
- return TRUE;
- case DDERR_WASSTILLDRAWING:
- case DDERR_SURFACEBUSY:
- J2dTraceLn(J2D_TRACE_WARNING, "DDLock: surface busy...");
- break;
- case DDERR_SURFACELOST:
- J2dTraceLn(J2D_TRACE_WARNING, "DDLock: surface lost");
- wsdo->RestoreSurface(env, wsdo);
- return FALSE;
- case DDERR_GENERIC:
- J2dRlsTraceLn(J2D_TRACE_ERROR, "DDLock: unexpected error");
- if (wsdo->window == NULL) {
- Win32OSSD_DisableDD(env, wsdo);
- }
- return FALSE;
- default:
- DebugPrintDirectDrawError(ddResult, "DDLock");
- return FALSE;
- }
- }
- // If we get here, then there was an error in the function and we
- // should return false
- return FALSE;
-}
-
-
-/**
- * Unlock the surface
- */
-void DDUnlock(JNIEnv *env, Win32SDOps *wsdo)
-{
- J2dTraceLn1(J2D_TRACE_INFO, "DDUnlock: wsdo->lpSurface=0x%x", wsdo->lpSurface);
-
- HRESULT ddResult = wsdo->lpSurface->Unlock(NULL);
- // Spin on the busy-type errors, else return having failed or succeeded
- switch (ddResult) {
- case DD_OK:
- return;
- case DDERR_NOTLOCKED:
- J2dTraceLn(J2D_TRACE_ERROR, "DDUnlock: Surface not locked");
- return;
- case DDERR_SURFACELOST:
- J2dTraceLn(J2D_TRACE_WARNING, "DDUnlock: Surface lost");
- wsdo->RestoreSurface(env, wsdo);
- return;
- default:
- DebugPrintDirectDrawError(ddResult, "DDUnlock");
- return;
- }
-}
-
-/**
- * Fill given surface with given color in given RECT bounds
- */
-BOOL DDColorFill(JNIEnv *env, jobject sData, Win32SDOps *wsdo,
- RECT *fillRect, jint color)
-{
- DDBLTFX ddBltFx;
- HRESULT ddResult;
- int attempts = 0;
-
- J2dTraceLn(J2D_TRACE_VERBOSE, "DDColorFill");
- J2dTraceLn5(J2D_TRACE_VERBOSE,
- " color=0x%x l=%-4d t=%-4d r=%-4d b=%-4d",
- color, fillRect->left, fillRect->top, fillRect->right,
- fillRect->bottom);
- ddBltFx.dwSize = sizeof(ddBltFx);
- ddBltFx.dwFillColor = color;
- AttachClipper(wsdo);
- while (attempts++ < MAX_BUSY_ATTEMPTS) {
- ddResult = wsdo->lpSurface->Blt(fillRect, NULL, NULL,
- DDBLT_COLORFILL | DDBLT_WAIT,
- &ddBltFx);
- // Spin on the busy-type errors, else return having failed or succeeded
- switch (ddResult) {
- case DD_OK:
- return TRUE;
- case DDERR_INVALIDRECT:
- J2dTraceLn4(J2D_TRACE_ERROR,
- "DDColorFill: Invalid rect for colorfill "\
- "l=%-4d t=%-4d r=%-4d b=%-4d",
- fillRect->left, fillRect->top,
- fillRect->right, fillRect->bottom);
- return FALSE;
- case DDERR_SURFACEBUSY:
- J2dTraceLn(J2D_TRACE_WARNING, "DDColorFill: surface busy");
- break;
- case DDERR_SURFACELOST:
- J2dTraceLn(J2D_TRACE_WARNING, "DDColorfill: surface lost");
- wsdo->RestoreSurface(env, wsdo);
- return FALSE;
- default:
- DebugPrintDirectDrawError(ddResult, "DDColorFill");
- }
- }
- J2dTraceLn(J2D_TRACE_VERBOSE, "DDColorFill done");
- return FALSE;
-}
-
-void ManageOffscreenSurfaceBlt(JNIEnv *env, Win32SDOps *wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "ManageOffscreenSurfaceBlt");
- wsdo->surfacePuntData.pixelsReadSinceBlt = 0;
- if (wsdo->surfacePuntData.numBltsSinceRead >=
- wsdo->surfacePuntData.numBltsThreshold)
- {
- if (wsdo->surfacePuntData.usingDDSystem) {
- if (wsdo->surfacePuntData.lpSurfaceVram->Blt(NULL,
- wsdo->surfacePuntData.lpSurfaceSystem,
- NULL, DDBLT_WAIT, NULL) == DD_OK)
- {
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- " Unpunting sys to VRAM: 0x%x -> 0x%x",
- wsdo->surfacePuntData.lpSurfaceVram,
- wsdo->surfacePuntData.lpSurfaceSystem);
- wsdo->lpSurface = wsdo->surfacePuntData.lpSurfaceVram;
- wsdo->surfacePuntData.usingDDSystem = FALSE;
- // Now: double our threshhold to prevent thrashing; we
- // don't want to keep punting and un-punting our surface
- wsdo->surfacePuntData.numBltsThreshold *= 2;
- // Notify the Java level that this surface has
- // been unpunted so that future copies to this surface
- // from accelerated src surfaces will do the right thing.
- jobject sdObject = env->NewLocalRef(wsdo->sdOps.sdObject);
- if (sdObject) {
- // Only bother with this optimization if the
- // reference is still valid
- env->SetBooleanField(sdObject, ddSurfacePuntedID, JNI_FALSE);
- env->DeleteLocalRef(sdObject);
- }
- }
- }
- } else {
- wsdo->surfacePuntData.numBltsSinceRead++;
- }
-}
-
-/**
- * Copy data from src to dst using src and dst rectangles
- */
-BOOL DDBlt(JNIEnv *env, Win32SDOps *wsdoSrc, Win32SDOps *wsdoDst,
- RECT *rDst, RECT *rSrc, CompositeInfo *compInfo)
-{
- int attempts = 0;
- DWORD bltFlags = DDBLT_WAIT;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDBlt");
- J2dTraceLn4(J2D_TRACE_INFO, " src rect: l=%-4d t=%-4d r=%-4d b=%-4d",
- rSrc->left, rSrc->top, rSrc->right, rSrc->bottom);
- J2dTraceLn4(J2D_TRACE_INFO, " dst rect: l=%-4d t=%-4d r=%-4d b=%-4d",
- rDst->left, rDst->top, rDst->right, rDst->bottom);
-
- // Note: the primary can only have one clipper attached to it at
- // any time. This seems weird to set it to src then dst, but this
- // works because either: both are the same window (devCopyArea),
- // neither are windows (both offscreen), or only one is a window
- // (Blt). We can't get here from a windowA -> windowB copy operation.
- AttachClipper(wsdoSrc);
- AttachClipper(wsdoDst);
-
- // Administrate system-surface punt mechanism for offscreen images
- if (!wsdoSrc->window && !wsdoSrc->surfacePuntData.disablePunts) {
- ManageOffscreenSurfaceBlt(env, wsdoSrc);
- }
- if (wsdoSrc->transparency == TR_BITMASK) {
- bltFlags |= DDBLT_KEYSRC;
- }
- while (attempts++ < MAX_BUSY_ATTEMPTS) {
- HRESULT ddResult =
- wsdoDst->lpSurface->Blt(rDst, wsdoSrc->lpSurface,
- rSrc, bltFlags, NULL);
-
- // Spin on the busy-type errors or return having failed or succeeded
- switch (ddResult) {
- case DD_OK:
- return TRUE;
- case DDERR_SURFACEBUSY:
- J2dTraceLn(J2D_TRACE_WARNING, "DDBlt: surface busy");
- break;
- case DDERR_SURFACELOST:
- /**
- * Only restore the Dst if it is truly lost; "restoring" an
- * offscreen surface simply sets a flag and throws an exception,
- * thus guaranteeing that the Src restore below will not happen.
- * So if the Src stays Lost and we keep trying to restore an un-Lost
- * Dst, then we will never actually do the restore on the Src.
- */
- if (wsdoDst->lpSurface->IsLost() != DD_OK) {
- J2dTraceLn(J2D_TRACE_WARNING, "DDBlt: dst surface lost");
- wsdoDst->RestoreSurface(env, wsdoDst);
- }
- if (wsdoSrc->lpSurface->IsLost() != DD_OK) {
- J2dTraceLn(J2D_TRACE_WARNING, "DDBlt: src surface lost");
- wsdoSrc->RestoreSurface(env, wsdoSrc);
- }
- return FALSE;
- default:
- DebugPrintDirectDrawError(ddResult, "DDBlt");
- return FALSE;
- }
- }
- return FALSE;
-}
-
-/**
- * Set the color key information for this surface. During a
- * blit operation, pixels of the specified color will not be
- * drawn (resulting in transparent areas of the image). Note
- * that the "transparency" field in the Win32SDOps structure must
- * be set to TR_BITMASK for the color key information to have an effect.
- */
-void DDSetColorKey(JNIEnv *env, Win32SDOps *wsdo, jint color)
-{
- J2dTraceLn(J2D_TRACE_VERBOSE, "DDSetColorKey");
- DDCOLORKEY ddck;
- HRESULT ddResult;
-
- ddck.dwColorSpaceLowValue = color;
- ddck.dwColorSpaceHighValue = color;
-
- ddResult = wsdo->lpSurface->SetColorKey(DDCKEY_SRCBLT, &ddck);
-
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDSetColorKey");
- }
-}
-
-
-/**
- * Swaps the surface memory of the front and back buffers.
- * Flips memory from the source surface to the destination surface.
- */
-BOOL DDFlip(JNIEnv *env, Win32SDOps *src, Win32SDOps *dest)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDFlip");
- int attempts = 0;
- while (attempts++ < MAX_BUSY_ATTEMPTS) {
- HRESULT ddResult = src->lpSurface->Flip(dest->lpSurface);
- // Spin on the busy-type errors or return having failed or succeeded
- switch (ddResult) {
- case DD_OK:
- return TRUE;
- case DDERR_SURFACEBUSY:
- J2dTraceLn(J2D_TRACE_WARNING, "DDFlip: surface busy");
- break;
- case DDERR_SURFACELOST:
- if (dest->lpSurface->IsLost() != DD_OK) {
- J2dTraceLn(J2D_TRACE_WARNING, "DDFlip: dst surface lost");
- dest->RestoreSurface(env, dest);
- }
- if (src->lpSurface->IsLost() != DD_OK) {
- J2dTraceLn(J2D_TRACE_WARNING, "DDFlip: src surface lost");
- src->RestoreSurface(env, src);
- }
- return FALSE;
- default:
- DebugPrintDirectDrawError(ddResult, "DDFlip");
- return FALSE;
- }
- }
- return FALSE;
-}
-
-
-/**
- * Mark the given ddInstance structure as invalid. This flag
- * can then be used to detect rendering with an invalid ddraw
- * object later (to avoid further ddraw errors) or to detect
- * when it is time to create a new ddraw object. Recreation
- * happens when we are asked to create a new surface but the
- * current ddInstance global structure is invalid.
- */
-void DDInvalidateDDInstance(DDrawObjectStruct *ddInst) {
- J2dTraceLn(J2D_TRACE_INFO, "DDInvalidateDDInstance");
- if (useDD) {
- if (ddInst != NULL) {
- // Invalidate given instance of ddInstance
- ddInst->valid = FALSE;
- } else {
- // Invalidate global ddInstance. This occurs at the start
- // of a display-change event.
- for (int i = 0; i < currNumDevices; ++i) {
- if (ddInstance[i] && ddInstance[i]->hwndFullScreen == NULL) {
- ddInstance[i]->valid = FALSE;
- }
- }
- }
- }
-}
-
-/**
- * Utility routine: release all elements of given ddInst structure
- * and free the memory consumed by ddInst. Note that this may be
- * called during a failed DDCreateDDObject, so any null fields were
- * not yet initialized and should not be released.
- */
-void ReleaseDDInstance(DDrawObjectStruct *ddInst)
-{
- J2dTraceLn(J2D_TRACE_INFO, "ReleaseDDInstance");
- if (ddInst) {
- if (ddInst->primary) {
- delete ddInst->primary;
- ddInst->primary = NULL;
- }
- if (ddInst->clipper) {
- delete ddInst->clipper;
- ddInst->clipper = NULL;
- }
- if (ddInst->ddObject) {
- delete ddInst->ddObject;
- ddInst->ddObject = NULL;
- }
- free(ddInst);
- }
-}
-
-/**
- * Enters full-screen exclusive mode, setting the hwnd as the screen
- */
-BOOL DDEnterFullScreen(HMONITOR hMon, HWND hwnd, HWND topLevelHwnd)
-{
- HRESULT ddResult = DD_OK;
- // Sleep so that programatically full-screen cannot be entered
- // and left multiple times quickly enough to crash the driver
- static DWORD prevTime = 0;
- DWORD currTime = ::GetTickCount();
- DWORD timeDiff = (currTime - prevTime);
- if (timeDiff < 500) {
- ::Sleep(500 - timeDiff);
- }
- prevTime = currTime;
-
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- tmpDdInstance->ddObject->SetCooperativeLevel(topLevelHwnd,
- DDSCL_FULLSCREEN | DDSCL_EXCLUSIVE);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDEnterFullScreen");
- return FALSE;
- }
- if (tmpDdInstance->primary) {
- // No clipping necessary in fullscreen mode. Elsewhere,
- // we avoid setting the clip list for the fullscreen window,
- // so we should also null-out the clipper object for the
- // primary surface in that case. Bug 4737785.
- tmpDdInstance->primary->SetClipper(NULL);
- }
- tmpDdInstance->hwndFullScreen = hwnd;
- tmpDdInstance->context = CONTEXT_ENTER_FULL_SCREEN;
-
- return TRUE;
-}
-
-
-/**
- * Exits full-screen exclusive mode
- */
-BOOL DDExitFullScreen(HMONITOR hMon, HWND hwnd)
-{
- // Sleep so that programatically full-screen cannot be entered
- // and left multiple times quickly enough to crash the driver
- static DWORD prevTime = 0;
- DWORD currTime = ::GetTickCount();
- DWORD timeDiff = (currTime - prevTime);
- if (timeDiff < 500) {
- ::Sleep(500 - timeDiff);
- }
- prevTime = currTime;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDExitFullScreen");
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- tmpDdInstance->context = CONTEXT_EXIT_FULL_SCREEN;
- if (!tmpDdInstance || !tmpDdInstance->ddObject ||
- !tmpDdInstance->ddObject->RestoreDDDisplayMode()) {
- return FALSE;
- }
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "DDExitFullScreen: Restoring cooperative level hwnd=0x%x",
- hwnd);
- HRESULT ddResult =
- tmpDdInstance->ddObject->SetCooperativeLevel(NULL, DDSCL_NORMAL);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDExitFullScreen");
- return FALSE;
- }
- if (tmpDdInstance->clipper == NULL) {
- // May not have created clipper if we were in FS mode during
- // primary creation
- tmpDdInstance->clipper = tmpDdInstance->ddObject->CreateDDClipper();
- }
- if (tmpDdInstance->clipper != NULL) {
- tmpDdInstance->primary->SetClipper(tmpDdInstance->clipper);
- }
- J2dTraceLn(J2D_TRACE_VERBOSE,
- "DDExitFullScreen: Restored cooperative level");
- tmpDdInstance->hwndFullScreen = NULL;
- tmpDdInstance->context = CONTEXT_NORMAL;
- return TRUE;
-}
-
-/**
- * Gets the current display mode; sets the values in displayMode
- */
-BOOL DDGetDisplayMode(HMONITOR hMon, DDrawDisplayMode& displayMode)
-{
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- if (tmpDdInstance && tmpDdInstance->ddObject) {
- return tmpDdInstance->ddObject->GetDDDisplayMode(displayMode);
- } else {
- return FALSE;
- }
-}
-
-/**
- * Sets the display mode to the supplied mode
- */
-BOOL DDSetDisplayMode(HMONITOR hMon, DDrawDisplayMode& displayMode)
-{
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- if (tmpDdInstance) {
- tmpDdInstance->context = CONTEXT_DISPLAY_CHANGE;
- }
- if (tmpDdInstance && tmpDdInstance->ddObject) {
- int attempts = 0;
- while (attempts++ < MAX_BUSY_ATTEMPTS) {
- HRESULT ddResult = tmpDdInstance->ddObject->SetDDDisplayMode(
- displayMode);
- // Spin on the busy-type errors or return having failed or succeeded
- switch (ddResult) {
- case DD_OK:
- return TRUE;
- case DDERR_SURFACEBUSY:
- J2dTraceLn(J2D_TRACE_WARNING,
- "DDSetDisplayMode: surface busy");
- break;
- default:
- DebugPrintDirectDrawError(ddResult, "DDSetDisplayMode");
- return FALSE;
- }
- }
- return FALSE;
- } else {
- return FALSE;
- }
-}
-
-/**
- * Enumerates all display modes, calling the supplied callback for each
- * display mode returned by the system
- */
-BOOL DDEnumDisplayModes(HMONITOR hMon, DDrawDisplayMode* constraint,
- DDrawDisplayMode::Callback callback, void* context)
-{
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- if (tmpDdInstance && tmpDdInstance->ddObject) {
- return tmpDdInstance->ddObject->EnumDDDisplayModes(
- constraint, callback, context);
- } else {
- return FALSE;
- }
-}
-
-/**
- * Attempts to restore surface. This will only succeed if the system is
- * in a state that allows the surface to be restored. If a restore
- * results in a DDERR_WRONGMODE, then the surface must be recreated
- * entirely; we do this by invalidating the surfaceData and recreating
- * it from scratch (at the Java level).
- */
-BOOL DDRestoreSurface(Win32SDOps *wsdo)
-{
- J2dTraceLn1(J2D_TRACE_INFO, "DDRestoreSurface, wsdo->lpSurface=0x%x",
- wsdo->lpSurface);
-
- DDrawObjectStruct *tmpDdInstance = wsdo->ddInstance;
- if (tmpDdInstance == NULL || !tmpDdInstance->accelerated) {
- return FALSE;
- }
- // Don't try to restore an inactive primary in full-screen mode
- if (!isAppActive && wsdo->window &&
- wsdo->window == tmpDdInstance->hwndFullScreen) {
- return FALSE;
- }
- if (wsdo->lpSurface->IsLost() == DD_OK) {
- J2dTraceLn(J2D_TRACE_VERBOSE, "DDRestoreSurface: surface memory ok");
- }
- else {
- J2dTraceLn(J2D_TRACE_WARNING,
- "DDRestoreSurface: surface memory lost, trying to restore");
- HRESULT ddResult;
- ddResult = wsdo->lpSurface->Restore();
- if (ddResult == DDERR_WRONGMODE) {
- // Strange full-screen bug; return false to avoid a hang.
- // Note that we should never get this error in full-screen mode.
- if (wsdo->window == tmpDdInstance->hwndFullScreen) {
- return FALSE;
- }
- // Wrong mode: display depth has been changed.
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "DDRestoreSurface failure: DDERR_WRONGMODE");
- if (wsdo->window) {
- /**
- * If this is a window surface, invalidate this
- * object's ddInstance and return the approriate error. The
- * surfaceData will later be invalidated, disposed, and
- * re-created with the new and correct depth information.
- * Only invalidate for windows because offscreen surfaces
- * have other means of being re-created and do not necessarily
- * mean that the ddInstance object is invalid for other surfaces
- */
- DDInvalidateDDInstance(wsdo->ddInstance);
- }
- return FALSE;
- } else if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "DDRestoreSurface");
- return FALSE;
- }
- }
- if (!tmpDdInstance->valid) {
- tmpDdInstance->valid = TRUE;
- }
- return TRUE;
-}
-
-jint DDGetAvailableMemory(HMONITOR hMon)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDGetAvailableMemory");
- DWORD dwFree;
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
- if (!useDD || !tmpDdInstance || !tmpDdInstance->valid) {
- return 0;
- }
-
- HRESULT ddResult = tmpDdInstance->ddObject->GetDDAvailableVidMem(&dwFree);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult, "GetAvailableMemory");
- }
-
- return (jint)dwFree;
-}
-
-
-/**
- * Creates either an offscreen or onscreen ddraw surface, depending
- * on the value of wsdo->window. Handles the common
- * framework of surface creation, such as ddInstance management,
- * and passes off the functionality of actual surface creation to
- * other functions. Successful creation results in a return value
- * of TRUE.
- */
-BOOL DDCreateSurface(Win32SDOps *wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDCreateSurface");
- HMONITOR hMon;
- hMon = (HMONITOR)wsdo->device->GetMonitor();
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
-
- wsdo->ddInstance = NULL; // default value in case of error
- wsdo->lpSurface = NULL; // default value in case of error
-
- if (wsdo->window) {
- if (tmpDdInstance &&
- tmpDdInstance->backBufferCount != wsdo->backBufferCount &&
- tmpDdInstance->hwndFullScreen == wsdo->window)
- {
- tmpDdInstance->context = CONTEXT_CHANGE_BUFFER_COUNT;
- tmpDdInstance->backBufferCount = wsdo->backBufferCount;
- }
- if (!tmpDdInstance || !tmpDdInstance->valid ||
- tmpDdInstance->context != CONTEXT_NORMAL
- ) {
- // Only recreate dd object on primary create. Given our current
- // model of displayChange event propagation, we can only guarantee
- // that the system has been properly prepared for a recreate when
- // we recreate a primary surface. The offscreen surfaces may
- // be recreated at any time.
- // Recreating ddraw at offscreen surface creation time has caused
- // rendering artifacts as well as unexplainable hangs in ddraw
- // calls.
- ddInstanceLock.Enter();
- BOOL success = DDCreatePrimary(wsdo);
- ddInstanceLock.Leave();
- if (!success) {
- return FALSE;
- }
- tmpDdInstance = GetDDInstanceForDevice(hMon);
- }
- // restore the primary if it's lost
- if (tmpDdInstance->primary != NULL &&
- FAILED(tmpDdInstance->primary->IsLost()) &&
- FAILED(tmpDdInstance->primary->Restore()))
- {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "DDCreateSurface: failed to restore primary surface");
- return FALSE;
- }
- // non-null window means onscreen surface. Primary already
- // exists, just need to cache a pointer to it in this wsdo
- wsdo->lpSurface = tmpDdInstance->primary;
- } else {
- if (!tmpDdInstance || !tmpDdInstance->valid) {
- // Don't recreate the ddraw object here (see note above), but
- // do fail this creation. We will come back here eventually
- // after an onscreen surface has been created (and the new
- // ddraw object to go along with it).
- return FALSE;
- }
- if (!DDCreateOffScreenSurface(wsdo, tmpDdInstance)) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "DDCreateSurface: Failed creating offscreen surface");
- return FALSE;
- }
- }
- wsdo->ddInstance = tmpDdInstance;
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- "DDCreateSurface: succeeded ddInst=0x%x wsdo->lpSurface=0x%x",
- tmpDdInstance, wsdo->lpSurface);
- return TRUE;
-}
-
-/**
- * Utility function to make sure that native and java-level
- * surface depths are matched. They can be mismatched when display-depths
- * change, either between the creation of the Java surfaceData structure
- * and the native ddraw surface, or later when a surface is automatically
- * adjusted to be the new display depth (even if it was created in a different
- * depth to begin with)
- */
-BOOL DDSurfaceDepthsCompatible(int javaDepth, int nativeDepth)
-{
- if (nativeDepth != javaDepth) {
- switch (nativeDepth) {
- case 0: // Error condition: something is wrong with the surface
- case 8:
- case 24:
- // Java and native surface depths should match exactly for
- // these cases
- return FALSE;
- break;
- case 16:
- // Java surfaceData should be 15 or 16 bits
- if (javaDepth < 15 || javaDepth > 16) {
- return FALSE;
- }
- break;
- case 32:
- // Could have this native depth for either 24- or 32-bit
- // Java surfaceData
- if (javaDepth != 24 && javaDepth != 32) {
- return FALSE;
- }
- break;
- default:
- // should not get here, but if we do something is odd, so
- // just register a failure
- return FALSE;
- }
- }
- return TRUE;
-}
-
-
-/**
- * Creates offscreen surface. Examines the display mode information
- * for the current ddraw object and uses that to create this new
- * surface.
- */
-BOOL DDCreateOffScreenSurface(Win32SDOps *wsdo,
- DDrawObjectStruct *ddInst)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDCreateOffScreenSurface");
-
- wsdo->lpSurface =
- ddInst->ddObject->CreateDDOffScreenSurface(wsdo->w, wsdo->h,
- wsdo->depth, wsdo->transparency, DDSCAPS_VIDEOMEMORY);
- if (!ddInst->primary || (ddInst->primary->IsLost() != DD_OK)) {
- if (wsdo->lpSurface) {
- delete wsdo->lpSurface;
- wsdo->lpSurface = NULL;
- }
- if (ddInst->primary) {
- // attempt to restore primary
- ddInst->primary->Restore();
- if (ddInst->primary->IsLost() == DD_OK) {
- // Primary restored: create the offscreen surface again
- wsdo->lpSurface =
- ddInst->ddObject->CreateDDOffScreenSurface(wsdo->w, wsdo->h,
- wsdo->depth, wsdo->transparency,
- DDSCAPS_VIDEOMEMORY);
- if (ddInst->primary->IsLost() != DD_OK) {
- // doubtful, but possible that it is lost again
- // If so, delete the surface and get out of here
- if (wsdo->lpSurface) {
- delete wsdo->lpSurface;
- wsdo->lpSurface = NULL;
- }
- }
- }
- }
- }
- if (wsdo->lpSurface != NULL && (wsdo->transparency != TR_TRANSLUCENT)) {
- /**
- * 4941350: Double-check that the depth of the surface just created
- * is compatible with the depth requested. Note that we ignore texture
- * (translucent) surfaces as those depths may differ between Java and
- * native representations.
- */
- int surfaceDepth = wsdo->lpSurface->GetSurfaceDepth();
- if (!DDSurfaceDepthsCompatible(wsdo->depth, surfaceDepth)) {
- J2dTraceLn2(J2D_TRACE_WARNING,
- "DDCreateOffScreenSurface: Surface depth mismatch: "\
- "intended=%d actual=%d",
- wsdo->depth, surfaceDepth);
- DDReleaseSurfaceMemory(wsdo->lpSurface);
- wsdo->lpSurface = NULL;
- }
- }
- return (wsdo->lpSurface != NULL);
-}
-
-
-/**
- * Gets an attached surface, such as a back buffer, from a parent
- * surface. Sets the lpSurface member of the wsdo supplied to
- * the attached surface.
- */
-BOOL DDGetAttachedSurface(JNIEnv *env, Win32SDOps* wsdo_parent,
- Win32SDOps* wsdo)
-{
- J2dTraceLn(J2D_TRACE_INFO, "DDGetAttachedSurface");
- HMONITOR hMon = (HMONITOR)wsdo_parent->device->GetMonitor();
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice(hMon);
-
- wsdo->ddInstance = NULL; // default value in case of error
- wsdo->lpSurface = NULL; // default value in case of error
-
- if (!tmpDdInstance || !tmpDdInstance->valid ||
- wsdo_parent->lpSurface == NULL)
- {
- J2dTraceLn2(J2D_TRACE_WARNING,
- "DDGetAttachedSurface: unable to get attached "\
- "surface for wsdo=0%x wsdo_parent=0%x", wsdo, wsdo_parent);
- return FALSE;
- }
- DDrawSurface* pNew = wsdo_parent->lpSurface->GetDDAttachedSurface();
- if (pNew == NULL) {
- return FALSE;
- }
- wsdo->lpSurface = pNew;
- wsdo->ddInstance = tmpDdInstance;
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "DDGetAttachedSurface: succeeded wsdo->lpSurface=0x%x",
- wsdo->lpSurface);
- return TRUE;
-}
-
-
-/**
- * Destroys resources associated with a surface
- */
-void DDDestroySurface(Win32SDOps *wsdo)
-{
- J2dTraceLn1(J2D_TRACE_INFO, "DDDestroySurface: wsdo->lpSurface=0x%x",
- wsdo->lpSurface);
-
- if (!wsdo->lpSurface) {
- // null surface means it was never created; simply return
- return;
- }
- if (!wsdo->window) {
- // offscreen surface
- delete wsdo->lpSurface;
- wsdo->lpSurface = NULL;
- }
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "DDDestroySurface: ddInstance->refCount=%d",
- wsdo->ddInstance->refCount);
-}
-
-/**
- * Releases ddraw resources associated with a surface. Note that
- * the DDrawSurface object is still valid, but the underlying
- * DirectDraw surface is released.
- */
-void DDReleaseSurfaceMemory(DDrawSurface *lpSurface)
-{
- J2dTraceLn1(J2D_TRACE_INFO,
- "DDReleaseSurfaceMemory: lpSurface=0x%x", lpSurface);
-
- if (!lpSurface) {
- // null surface means it was never created; simply return
- return;
- }
- HRESULT ddResult = lpSurface->ReleaseSurface();
-}
-
-/*
- * This function returns whether or not surfaces should be replaced
- * in response to a WM_DISPLAYCHANGE message. If we are a full-screen
- * application that has lost its surfaces, we do not want to replace
- * our surfaces in response to a WM_DISPLAYCHANGE.
- */
-BOOL DDCanReplaceSurfaces(HWND hwnd)
-{
- J2dTraceLn1(J2D_TRACE_VERBOSE, "DDCanReplaceSurfaces: hwnd=0x%x", hwnd);
- DDrawObjectStruct *tmpDdInstance = NULL;
- ddInstanceLock.Enter();
- for (int i = 0; i < currNumDevices; i++) {
- tmpDdInstance = ddInstance[i];
- if (DDINSTANCE_USABLE(tmpDdInstance)) {
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- " ddInstance[%d]->hwndFullScreen=0x%x",
- i, tmpDdInstance->hwndFullScreen);
- if (tmpDdInstance->hwndFullScreen != NULL &&
- tmpDdInstance->context == CONTEXT_NORMAL &&
- (tmpDdInstance->hwndFullScreen == hwnd || hwnd == NULL)) {
- ddInstanceLock.Leave();
- return FALSE;
- }
- }
- }
- ddInstanceLock.Leave();
- return TRUE;
-}
-
-/*
- * This function prints the DirectDraw error associated with
- * the given errNum
- */
-void PrintDirectDrawError(DWORD errNum, char *message)
-{
- char buffer[255];
-
- GetDDErrorString(errNum, buffer);
- printf("%s:: %s\n", message, buffer);
-}
-
-
-/*
- * This function prints the DirectDraw error associated with
- * the given errNum
- */
-void DebugPrintDirectDrawError(DWORD errNum, char *message)
-{
- char buffer[255];
-
- GetDDErrorString(errNum, buffer);
- J2dRlsTraceLn2(J2D_TRACE_ERROR, "%s: %s", message, buffer);
-}
-
-
-/*
- * This function prints the error string into the given buffer
- */
-void GetDDErrorString(DWORD errNum, char *buffer)
-{
- switch (errNum) {
- case DDERR_ALREADYINITIALIZED:
- sprintf(buffer, "DirectDraw Error: DDERR_ALREADYINITIALIZED");
- break;
- case DDERR_CANNOTATTACHSURFACE:
- sprintf(buffer, "DirectDraw Error: DDERR_CANNOTATTACHSURFACE");
- break;
- case DDERR_CANNOTDETACHSURFACE:
- sprintf(buffer, "DirectDraw Error: DDERR_CANNOTDETACHSURFACE");
- break;
- case DDERR_CURRENTLYNOTAVAIL:
- sprintf(buffer, "DirectDraw Error: DDERR_CURRENTLYNOTAVAIL");
- break;
- case DDERR_EXCEPTION:
- sprintf(buffer, "DirectDraw Error: DDERR_EXCEPTION");
- break;
- case DDERR_GENERIC:
- sprintf(buffer, "DirectDraw Error: DDERR_GENERIC");
- break;
- case DDERR_HEIGHTALIGN:
- sprintf(buffer, "DirectDraw Error: DDERR_HEIGHTALIGN");
- break;
- case DDERR_INCOMPATIBLEPRIMARY:
- sprintf(buffer, "DirectDraw Error: DDERR_INCOMPATIBLEPRIMARY");
- break;
- case DDERR_INVALIDCAPS:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDCAPS");
- break;
- case DDERR_INVALIDCLIPLIST:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDCLIPLIST");
- break;
- case DDERR_INVALIDMODE:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDMODE");
- break;
- case DDERR_INVALIDOBJECT:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDOBJECT");
- break;
- case DDERR_INVALIDPARAMS:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDPARAMS");
- break;
- case DDERR_INVALIDPIXELFORMAT:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDPIXELFORMAT");
- break;
- case DDERR_INVALIDRECT:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDRECT");
- break;
- case DDERR_LOCKEDSURFACES:
- sprintf(buffer, "DirectDraw Error: DDERR_LOCKEDSURFACES");
- break;
- case DDERR_NO3D:
- sprintf(buffer, "DirectDraw Error: DDERR_NO3D");
- break;
- case DDERR_NOALPHAHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOALPHAHW");
- break;
- case DDERR_NOCLIPLIST:
- sprintf(buffer, "DirectDraw Error: DDERR_NOCLIPLIST");
- break;
- case DDERR_NOCOLORCONVHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOCOLORCONVHW");
- break;
- case DDERR_NOCOOPERATIVELEVELSET:
- sprintf(buffer, "DirectDraw Error: DDERR_NOCOOPERATIVELEVELSET");
- break;
- case DDERR_NOCOLORKEY:
- sprintf(buffer, "DirectDraw Error: DDERR_NOCOLORKEY");
- break;
- case DDERR_NOCOLORKEYHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOCOLORKEYHW");
- break;
- case DDERR_NODIRECTDRAWSUPPORT:
- sprintf(buffer, "DirectDraw Error: DDERR_NODIRECTDRAWSUPPORT");
- break;
- case DDERR_NOEXCLUSIVEMODE:
- sprintf(buffer, "DirectDraw Error: DDERR_NOEXCLUSIVEMODE");
- break;
- case DDERR_NOFLIPHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOFLIPHW");
- break;
- case DDERR_NOGDI:
- sprintf(buffer, "DirectDraw Error: DDERR_NOGDI");
- break;
- case DDERR_NOMIRRORHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOMIRRORHW");
- break;
- case DDERR_NOTFOUND:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTFOUND");
- break;
- case DDERR_NOOVERLAYHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOOVERLAYHW");
- break;
- case DDERR_NORASTEROPHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NORASTEROPHW");
- break;
- case DDERR_NOROTATIONHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOROTATIONHW");
- break;
- case DDERR_NOSTRETCHHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOSTRETCHHW");
- break;
- case DDERR_NOT4BITCOLOR:
- sprintf(buffer, "DirectDraw Error: DDERR_NOT4BITCOLOR");
- break;
- case DDERR_NOT4BITCOLORINDEX:
- sprintf(buffer, "DirectDraw Error: DDERR_NOT4BITCOLORINDEX");
- break;
- case DDERR_NOT8BITCOLOR:
- sprintf(buffer, "DirectDraw Error: DDERR_NOT8BITCOLOR");
- break;
- case DDERR_NOTEXTUREHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTEXTUREHW");
- break;
- case DDERR_NOVSYNCHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOVSYNCHW");
- break;
- case DDERR_NOZBUFFERHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOZBUFFERHW");
- break;
- case DDERR_NOZOVERLAYHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOZOVERLAYHW");
- break;
- case DDERR_OUTOFCAPS:
- sprintf(buffer, "DirectDraw Error: DDERR_OUTOFCAPS");
- break;
- case DDERR_OUTOFMEMORY:
- sprintf(buffer, "DirectDraw Error: DDERR_OUTOFMEMORY");
- break;
- case DDERR_OUTOFVIDEOMEMORY:
- sprintf(buffer, "DirectDraw Error: DDERR_OUTOFVIDEOMEMORY");
- break;
- case DDERR_OVERLAYCANTCLIP:
- sprintf(buffer, "DirectDraw Error: DDERR_OVERLAYCANTCLIP");
- break;
- case DDERR_OVERLAYCOLORKEYONLYONEACTIVE:
- sprintf(buffer, "DirectDraw Error: DDERR_OVERLAYCOLORKEYONLYONEACTIVE");
- break;
- case DDERR_PALETTEBUSY:
- sprintf(buffer, "DirectDraw Error: DDERR_PALETTEBUSY");
- break;
- case DDERR_COLORKEYNOTSET:
- sprintf(buffer, "DirectDraw Error: DDERR_COLORKEYNOTSET");
- break;
- case DDERR_SURFACEALREADYATTACHED:
- sprintf(buffer, "DirectDraw Error: DDERR_SURFACEALREADYATTACHED");
- break;
- case DDERR_SURFACEALREADYDEPENDENT:
- sprintf(buffer, "DirectDraw Error: DDERR_SURFACEALREADYDEPENDENT");
- break;
- case DDERR_SURFACEBUSY:
- sprintf(buffer, "DirectDraw Error: DDERR_SURFACEBUSY");
- break;
- case DDERR_CANTLOCKSURFACE:
- sprintf(buffer, "DirectDraw Error: DDERR_CANTLOCKSURFACE");
- break;
- case DDERR_SURFACEISOBSCURED:
- sprintf(buffer, "DirectDraw Error: DDERR_SURFACEISOBSCURED");
- break;
- case DDERR_SURFACELOST:
- sprintf(buffer, "DirectDraw Error: DDERR_SURFACELOST");
- break;
- case DDERR_SURFACENOTATTACHED:
- sprintf(buffer, "DirectDraw Error: DDERR_SURFACENOTATTACHED");
- break;
- case DDERR_TOOBIGHEIGHT:
- sprintf(buffer, "DirectDraw Error: DDERR_TOOBIGHEIGHT");
- break;
- case DDERR_TOOBIGSIZE:
- sprintf(buffer, "DirectDraw Error: DDERR_TOOBIGSIZE");
- break;
- case DDERR_TOOBIGWIDTH:
- sprintf(buffer, "DirectDraw Error: DDERR_TOOBIGWIDTH");
- break;
- case DDERR_UNSUPPORTED:
- sprintf(buffer, "DirectDraw Error: DDERR_UNSUPPORTED");
- break;
- case DDERR_UNSUPPORTEDFORMAT:
- sprintf(buffer, "DirectDraw Error: DDERR_UNSUPPORTEDFORMAT");
- break;
- case DDERR_UNSUPPORTEDMASK:
- sprintf(buffer, "DirectDraw Error: DDERR_UNSUPPORTEDMASK");
- break;
- case DDERR_VERTICALBLANKINPROGRESS:
- sprintf(buffer, "DirectDraw Error: DDERR_VERTICALBLANKINPROGRESS");
- break;
- case DDERR_WASSTILLDRAWING:
- sprintf(buffer, "DirectDraw Error: DDERR_WASSTILLDRAWING");
- break;
- case DDERR_XALIGN:
- sprintf(buffer, "DirectDraw Error: DDERR_XALIGN");
- break;
- case DDERR_INVALIDDIRECTDRAWGUID:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDDIRECTDRAWGUID");
- break;
- case DDERR_DIRECTDRAWALREADYCREATED:
- sprintf(buffer, "DirectDraw Error: DDERR_DIRECTDRAWALREADYCREATED");
- break;
- case DDERR_NODIRECTDRAWHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NODIRECTDRAWHW");
- break;
- case DDERR_PRIMARYSURFACEALREADYEXISTS:
- sprintf(buffer, "DirectDraw Error: DDERR_PRIMARYSURFACEALREADYEXISTS");
- break;
- case DDERR_NOEMULATION:
- sprintf(buffer, "DirectDraw Error: DDERR_NOEMULATION");
- break;
- case DDERR_REGIONTOOSMALL:
- sprintf(buffer, "DirectDraw Error: DDERR_REGIONTOOSMALL");
- break;
- case DDERR_CLIPPERISUSINGHWND:
- sprintf(buffer, "DirectDraw Error: DDERR_CLIPPERISUSINGHWND");
- break;
- case DDERR_NOCLIPPERATTACHED:
- sprintf(buffer, "DirectDraw Error: DDERR_NOCLIPPERATTACHED");
- break;
- case DDERR_NOHWND:
- sprintf(buffer, "DirectDraw Error: DDERR_NOHWND");
- break;
- case DDERR_HWNDSUBCLASSED:
- sprintf(buffer, "DirectDraw Error: DDERR_HWNDSUBCLASSED");
- break;
- case DDERR_HWNDALREADYSET:
- sprintf(buffer, "DirectDraw Error: DDERR_HWNDALREADYSET");
- break;
- case DDERR_NOPALETTEATTACHED:
- sprintf(buffer, "DirectDraw Error: DDERR_NOPALETTEATTACHED");
- break;
- case DDERR_NOPALETTEHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOPALETTEHW");
- break;
- case DDERR_BLTFASTCANTCLIP:
- sprintf(buffer, "DirectDraw Error: DDERR_BLTFASTCANTCLIP");
- break;
- case DDERR_NOBLTHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOBLTHW");
- break;
- case DDERR_NODDROPSHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NODDROPSHW");
- break;
- case DDERR_OVERLAYNOTVISIBLE:
- sprintf(buffer, "DirectDraw Error: DDERR_OVERLAYNOTVISIBLE");
- break;
- case DDERR_NOOVERLAYDEST:
- sprintf(buffer, "DirectDraw Error: DDERR_NOOVERLAYDEST");
- break;
- case DDERR_INVALIDPOSITION:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDPOSITION");
- break;
- case DDERR_NOTAOVERLAYSURFACE:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTAOVERLAYSURFACE");
- break;
- case DDERR_EXCLUSIVEMODEALREADYSET:
- sprintf(buffer, "DirectDraw Error: DDERR_EXCLUSIVEMODEALREADYSET");
- break;
- case DDERR_NOTFLIPPABLE:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTFLIPPABLE");
- break;
- case DDERR_CANTDUPLICATE:
- sprintf(buffer, "DirectDraw Error: DDERR_CANTDUPLICATE");
- break;
- case DDERR_NOTLOCKED:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTLOCKED");
- break;
- case DDERR_CANTCREATEDC:
- sprintf(buffer, "DirectDraw Error: DDERR_CANTCREATEDC");
- break;
- case DDERR_NODC:
- sprintf(buffer, "DirectDraw Error: DDERR_NODC");
- break;
- case DDERR_WRONGMODE:
- sprintf(buffer, "DirectDraw Error: DDERR_WRONGMODE");
- break;
- case DDERR_IMPLICITLYCREATED:
- sprintf(buffer, "DirectDraw Error: DDERR_IMPLICITLYCREATED");
- break;
- case DDERR_NOTPALETTIZED:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTPALETTIZED");
- break;
- case DDERR_UNSUPPORTEDMODE:
- sprintf(buffer, "DirectDraw Error: DDERR_UNSUPPORTEDMODE");
- break;
- case DDERR_NOMIPMAPHW:
- sprintf(buffer, "DirectDraw Error: DDERR_NOMIPMAPHW");
- break;
- case DDERR_INVALIDSURFACETYPE:
- sprintf(buffer, "DirectDraw Error: DDERR_INVALIDSURFACETYPE");
- break;
- case DDERR_DCALREADYCREATED:
- sprintf(buffer, "DirectDraw Error: DDERR_DCALREADYCREATED");
- break;
- case DDERR_CANTPAGELOCK:
- sprintf(buffer, "DirectDraw Error: DDERR_CANTPAGELOCK");
- break;
- case DDERR_CANTPAGEUNLOCK:
- sprintf(buffer, "DirectDraw Error: DDERR_CANTPAGEUNLOCK");
- break;
- case DDERR_NOTPAGELOCKED:
- sprintf(buffer, "DirectDraw Error: DDERR_NOTPAGELOCKED");
- break;
- case D3DERR_INVALID_DEVICE:
- sprintf(buffer, "Direct3D Error: D3DERR_INVALID_DEVICE");
- break;
- case D3DERR_INITFAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_INITFAILED");
- break;
- case D3DERR_DEVICEAGGREGATED:
- sprintf(buffer, "Direct3D Error: D3DERR_DEVICEAGGREGATED");
- break;
- case D3DERR_EXECUTE_CREATE_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_CREATE_FAILED");
- break;
- case D3DERR_EXECUTE_DESTROY_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_DESTROY_FAILED");
- break;
- case D3DERR_EXECUTE_LOCK_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_LOCK_FAILED");
- break;
- case D3DERR_EXECUTE_UNLOCK_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_UNLOCK_FAILED");
- break;
- case D3DERR_EXECUTE_LOCKED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_LOCKED");
- break;
- case D3DERR_EXECUTE_NOT_LOCKED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_NOT_LOCKED");
- break;
- case D3DERR_EXECUTE_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_FAILED");
- break;
- case D3DERR_EXECUTE_CLIPPED_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_EXECUTE_CLIPPED_FAILED");
- break;
- case D3DERR_TEXTURE_NO_SUPPORT:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_NO_SUPPORT");
- break;
- case D3DERR_TEXTURE_CREATE_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_CREATE_FAILED");
- break;
- case D3DERR_TEXTURE_DESTROY_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_DESTROY_FAILED");
- break;
- case D3DERR_TEXTURE_LOCK_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_LOCK_FAILED");
- break;
- case D3DERR_TEXTURE_UNLOCK_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_UNLOCK_FAILED");
- break;
- case D3DERR_TEXTURE_LOAD_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_LOAD_FAILED");
- break;
- case D3DERR_TEXTURE_SWAP_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_SWAP_FAILED");
- break;
- case D3DERR_TEXTURE_LOCKED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_LOCKED");
- break;
- case D3DERR_TEXTURE_NOT_LOCKED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_NOT_LOCKED");
- break;
- case D3DERR_TEXTURE_GETSURF_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_GETSURF_FAILED");
- break;
- case D3DERR_MATRIX_CREATE_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATRIX_CREATE_FAILED");
- break;
- case D3DERR_MATRIX_DESTROY_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATRIX_DESTROY_FAILED");
- break;
- case D3DERR_MATRIX_SETDATA_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATRIX_SETDATA_FAILED");
- break;
- case D3DERR_MATRIX_GETDATA_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATRIX_GETDATA_FAILED");
- break;
- case D3DERR_SETVIEWPORTDATA_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_SETVIEWPORTDATA_FAILED");
- break;
- case D3DERR_INVALIDCURRENTVIEWPORT:
- sprintf(buffer, "Direct3D Error: D3DERR_INVALIDCURRENTVIEWPORT");
- break;
- case D3DERR_INVALIDPRIMITIVETYPE:
- sprintf(buffer, "Direct3D Error: D3DERR_INVALIDPRIMITIVETYPE");
- break;
- case D3DERR_INVALIDVERTEXTYPE:
- sprintf(buffer, "Direct3D Error: D3DERR_INVALIDVERTEXTYPE");
- break;
- case D3DERR_TEXTURE_BADSIZE:
- sprintf(buffer, "Direct3D Error: D3DERR_TEXTURE_BADSIZE");
- break;
- case D3DERR_INVALIDRAMPTEXTURE:
- sprintf(buffer, "Direct3D Error: D3DERR_INVALIDRAMPTEXTURE");
- break;
- case D3DERR_MATERIAL_CREATE_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATERIAL_CREATE_FAILED");
- break;
- case D3DERR_MATERIAL_DESTROY_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATERIAL_DESTROY_FAILED");
- break;
- case D3DERR_MATERIAL_SETDATA_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATERIAL_SETDATA_FAILED");
- break;
- case D3DERR_MATERIAL_GETDATA_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_MATERIAL_GETDATA_FAILED");
- break;
- case D3DERR_INVALIDPALETTE:
- sprintf(buffer, "Direct3D Error: D3DERR_INVALIDPALETTE");
- break;
- case D3DERR_ZBUFF_NEEDS_SYSTEMMEMORY:
- sprintf(buffer, "Direct3D Error: D3DERR_ZBUFF_NEEDS_SYSTEMMEMORY");
- break;
- case D3DERR_ZBUFF_NEEDS_VIDEOMEMORY:
- sprintf(buffer, "Direct3D Error: D3DERR_ZBUFF_NEEDS_VIDEOMEMORY");
- break;
- case D3DERR_SURFACENOTINVIDMEM:
- sprintf(buffer, "Direct3D Error: D3DERR_SURFACENOTINVIDMEM");
- break;
- case D3DERR_LIGHT_SET_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_LIGHT_SET_FAILED");
- break;
- case D3DERR_LIGHTHASVIEWPORT:
- sprintf(buffer, "Direct3D Error: D3DERR_LIGHTHASVIEWPORT");
- break;
- case D3DERR_LIGHTNOTINTHISVIEWPORT:
- sprintf(buffer, "Direct3D Error: D3DERR_LIGHTNOTINTHISVIEWPORT");
- break;
- case D3DERR_SCENE_IN_SCENE:
- sprintf(buffer, "Direct3D Error: D3DERR_SCENE_IN_SCENE");
- break;
- case D3DERR_SCENE_NOT_IN_SCENE:
- sprintf(buffer, "Direct3D Error: D3DERR_SCENE_NOT_IN_SCENE");
- break;
- case D3DERR_SCENE_BEGIN_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_SCENE_BEGIN_FAILED");
- break;
- case D3DERR_SCENE_END_FAILED:
- sprintf(buffer, "Direct3D Error: D3DERR_SCENE_END_FAILED");
- break;
- case D3DERR_INBEGIN:
- sprintf(buffer, "Direct3D Error: D3DERR_INBEGIN");
- break;
- case D3DERR_NOTINBEGIN:
- sprintf(buffer, "Direct3D Error: D3DERR_NOTINBEGIN");
- break;
- case D3DERR_NOVIEWPORTS:
- sprintf(buffer, "Direct3D Error: D3DERR_NOVIEWPORTS");
- break;
- case D3DERR_VIEWPORTDATANOTSET:
- sprintf(buffer, "Direct3D Error: D3DERR_VIEWPORTDATANOTSET");
- break;
- case D3DERR_VIEWPORTHASNODEVICE:
- sprintf(buffer, "Direct3D Error: D3DERR_VIEWPORTHASNODEVICE");
- break;
- case D3DERR_NOCURRENTVIEWPORT:
- sprintf(buffer, "Direct3D Error: D3DERR_NOCURRENTVIEWPORT");
- break;
- default:
- sprintf(buffer, "DirectX Error Unknown 0x%x", errNum);
- break;
- }
-
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/ddrawUtils.h b/jdk/src/windows/native/sun/java2d/windows/ddrawUtils.h
deleted file mode 100644
index 6ef73af9311..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/ddrawUtils.h
+++ /dev/null
@@ -1,135 +0,0 @@
-/*
- * Copyright 2000-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#ifndef DDRAWUTILS_H
-#define DDRAWUTILS_H
-
-
-#include
-#include
-#include
-#include "Win32SurfaceData.h"
-#include "ddrawObject.h"
-
-/**
- * Direct Draw utility functions
- */
-
-#define DDINSTANCE_USABLE(ddInst) \
- ((ddInst) && (ddInst->valid) && (ddInst->accelerated))
-
-void DDRelease();
-
-void DDReleaseSurfaceMemory(DDrawSurface *lpSurface);
-
-BOOL DDCreatePrimary(Win32SDOps *wsdo);
-
-void DDFreeSyncSurface(DDrawObjectStruct *tmpDdInstance);
-
-void DDSync();
-
-BOOL DDCanCreatePrimary(HMONITOR hMon);
-
-BOOL DDCanBlt(Win32SDOps *wsdo);
-
-BOOL DDUseDDraw(Win32SDOps *wsdo);
-
-BOOL DeviceUseDDraw(HMONITOR hMon);
-
-BOOL DeviceUseD3D(HMONITOR hMon);
-
-void DDInvalidateDDInstance(DDrawObjectStruct *ddInst);
-
-void ReleaseDDInstance(DDrawObjectStruct *ddInst);
-
-BOOL DDEnterFullScreen(HMONITOR hMon, HWND hwnd, HWND topLevelHwnd);
-
-BOOL DDExitFullScreen(HMONITOR hMon, HWND hwnd);
-
-BOOL DDGetDisplayMode(HMONITOR hMon, DDrawDisplayMode& displayMode);
-
-BOOL DDSetDisplayMode(HMONITOR hMon, DDrawDisplayMode& displayMode);
-
-BOOL DDEnumDisplayModes(HMONITOR hMon, DDrawDisplayMode* constraint,
- DDrawDisplayMode::Callback callback, void* context);
-
-BOOL DDClipCheck(Win32SDOps *wsdo, RECT *operationRect);
-
-BOOL DDLock(JNIEnv *env, Win32SDOps *wsdo, RECT *lockRect,
- SurfaceDataRasInfo *pRasInfo);
-
-void DDUnlock(JNIEnv *env, Win32SDOps *wsdo);
-
-BOOL DDColorFill(JNIEnv *env, jobject sData, Win32SDOps *wsdo,
- RECT *fillRect, jint color);
-
-BOOL DDBlt(JNIEnv *env, Win32SDOps *wsdoSrc, Win32SDOps *wsdoDst,
- RECT *rDst, RECT *rSrc, CompositeInfo *compInfo = NULL);
-
-void DDSetColorKey(JNIEnv *env, Win32SDOps *wsdo, jint color);
-
-BOOL DDFlip(JNIEnv *env, Win32SDOps *src, Win32SDOps *dest);
-
-BOOL DDRestoreSurface(Win32SDOps *wsdo);
-
-jint DDGetAvailableMemory(HMONITOR hMon);
-
-BOOL DDCreateSurface(Win32SDOps *wsdo);
-
-BOOL DDCreateOffScreenSurface(Win32SDOps *wsdo, DDrawObjectStruct *ddInst);
-
-BOOL DDGetAttachedSurface(JNIEnv *env, Win32SDOps* wsdo_parent, Win32SDOps* wsdo);
-
-void DDDestroySurface(Win32SDOps *wsdo);
-
-BOOL DDCanReplaceSurfaces(HWND hwnd);
-
-BOOL DDSurfaceDepthsCompatible(int javaDepth, int nativeDepth);
-
-void PrintDirectDrawError(DWORD errNum, char *message);
-
-void DebugPrintDirectDrawError(DWORD errNum, char *message);
-
-void GetDDErrorString(DWORD errNum, char *buffer);
-
-DDrawObjectStruct *GetDDInstanceForDevice(HMONITOR hMon);
-
-#define CLIP2RECTS_1PARAM(r1, r2, param, comp, lim) \
- do { \
- if (r1.param comp lim) { \
- r2.param += lim - r1.param; \
- r1.param = lim; \
- } \
- } while (0)
-
-#define CLIP2RECTS(r1, L, T, R, B, r2) \
- do { \
- CLIP2RECTS_1PARAM(r1, r2, left, <, L); \
- CLIP2RECTS_1PARAM(r1, r2, top, <, T); \
- CLIP2RECTS_1PARAM(r1, r2, right, >, R); \
- CLIP2RECTS_1PARAM(r1, r2, bottom, >, B); \
- } while(0)
-
-#endif DDRAWUTILS_H
diff --git a/jdk/src/windows/native/sun/java2d/windows/dxCapabilities.cpp b/jdk/src/windows/native/sun/java2d/windows/dxCapabilities.cpp
deleted file mode 100644
index 9c75d8d30e7..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/dxCapabilities.cpp
+++ /dev/null
@@ -1,108 +0,0 @@
-/*
- * Copyright 2003-2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include
-#include
-#include
-#include "dxCapabilities.h"
-
-/**
- * DxCapabilities encapsulates the DirectX capabilities of a display
- * device. Typically, we run tests at startup on each display device
- * at the current display depth. We record the results of those tests
- * in a capabilities object and also record those results in the registry.
- * The next time we run on this display device, we check whether we have
- * already recorded results for this device/depth in the registry and simply
- * use those values instead of re-running the tests. The results of the
- * tests determine which ddraw/d3d capabilities we enable/disable at runtime.
- */
-
-void DxCapabilities::Initialize(WCHAR *keyName)
-{
- this->keyName = (WCHAR*)malloc((wcslen(keyName) + 1) * sizeof(WCHAR));
- wcscpy(this->keyName, keyName);
- RegistryKey regKey(keyName, KEY_READ);
- ddCreation = regKey.GetIntValue(DD_CREATION);
- ddSurfaceCreation = regKey.GetIntValue(DD_SURFACE_CREATION);
-
- d3dCapsValidity = regKey.GetIntValue(D3D_CAPS_VALIDITY);
- d3dDeviceCaps = regKey.GetIntValue(D3D_DEVICE_CAPS);
-}
-
-WCHAR *StringForValue(int value)
-{
- switch (value) {
- case J2D_ACCEL_UNVERIFIED:
- return L"UNVERIFIED";
- break;
- case J2D_ACCEL_TESTING:
- return L"TESTING (may indicate crash during test)";
- break;
- case J2D_ACCEL_FAILURE:
- return L"FAILURE";
- break;
- case J2D_ACCEL_SUCCESS:
- return L"SUCCESS";
- break;
- default:
- return L"UNKNOWN";
- break;
- }
-}
-
-/**
- * PrintCaps is here for debugging purposes only
- */
-void DxCapabilities::PrintCaps() {
- printf(" %S: %S\n", DD_CREATION, StringForValue(ddCreation));
- printf(" %S: %S\n", DD_SURFACE_CREATION, StringForValue(ddSurfaceCreation));
- printf(" %S: %S\n", D3D_CAPS_VALIDITY, StringForValue(d3dCapsValidity));
- printf(" %S: 0x%X\n", D3D_DEVICE_CAPS, d3dDeviceCaps);
-}
-
-void DxCapabilities::SetDdCreationCap(int value) {
- ddCreation = value;
- SetCap(DD_CREATION, value);
-}
-
-void DxCapabilities::SetDdSurfaceCreationCap(int value) {
- ddSurfaceCreation = value;
- SetCap(DD_SURFACE_CREATION, value);
-}
-
-void DxCapabilities::SetD3dCapsValidity(int value) {
- d3dCapsValidity = value;
- SetCap(D3D_CAPS_VALIDITY, value);
-}
-
-void DxCapabilities::SetD3dDeviceCaps(int value) {
- d3dDeviceCaps = value;
- SetCap(D3D_DEVICE_CAPS, value);
-}
-
-void DxCapabilities::SetCap(WCHAR *capName, int value) {
- RegistryKey regKey(keyName, KEY_WRITE);
- regKey.SetIntValue(capName, value, TRUE);
-}
diff --git a/jdk/src/windows/native/sun/java2d/windows/dxCapabilities.h b/jdk/src/windows/native/sun/java2d/windows/dxCapabilities.h
deleted file mode 100644
index 6d9b9d09570..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/dxCapabilities.h
+++ /dev/null
@@ -1,67 +0,0 @@
-/*
- * Copyright 2003-2005 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#ifndef DX_CAPABILITIES_H
-#define DX_CAPABILITIES_H
-
-#include "RegistryKey.h"
-
-#define DD_CREATION L"ddCreation"
-#define DD_SURFACE_CREATION L"ddSurfaceCreation"
-#define D3D_CAPS_VALIDITY L"d3dCapsValidity"
-#define D3D_DEVICE_CAPS L"d3dDeviceCaps"
-
-class DxCapabilities {
-private:
- WCHAR *keyName;
- int ddCreation;
- int ddSurfaceCreation;
- int d3dCapsValidity;
- int d3dDeviceCaps;
-
-public:
- DxCapabilities() { keyName = NULL; }
- ~DxCapabilities() { if (keyName) free(keyName); }
- void Initialize(WCHAR *keyName);
-
- int GetDdCreationCap() { return ddCreation; }
- int GetDdSurfaceCreationCap() { return ddSurfaceCreation; }
- int GetD3dCapsValidity() { return d3dCapsValidity; }
- int GetD3dDeviceCaps() { return d3dDeviceCaps; }
-
- WCHAR *GetDeviceName() { return keyName; }
-
- void SetDdCreationCap(int value);
- void SetDdSurfaceCreationCap(int value);
- void SetD3dCapsValidity(int value);
- void SetD3dDeviceCaps(int value);
-
- void PrintCaps();
-
-private:
- void SetCap(WCHAR *capName, int value);
-};
-
-#endif DX_CAPABILITIES_H
diff --git a/jdk/src/windows/native/sun/java2d/windows/dxInit.cpp b/jdk/src/windows/native/sun/java2d/windows/dxInit.cpp
deleted file mode 100644
index 383f2a4f290..00000000000
--- a/jdk/src/windows/native/sun/java2d/windows/dxInit.cpp
+++ /dev/null
@@ -1,600 +0,0 @@
-/*
- * Copyright 2003-2006 Sun Microsystems, Inc. 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. Sun designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
- * CA 95054 USA or visit www.sun.com if you need additional information or
- * have any questions.
- */
-
-#include "dxInit.h"
-#include "ddrawUtils.h"
-#include "D3DRuntimeTest.h"
-#include "Trace.h"
-#include "RegistryKey.h"
-#include "WindowsFlags.h"
-#include "Devices.h"
-
-/**
- * This file holds the functions that handle the initialization
- * process for DirectX. This process includes checking the
- * Windows Registry for information about the system and each display device,
- * running any necessary functionality tests, and storing information
- * out to the registry depending on the test results.
- *
- * In general, startup tests should only have to execute once;
- * they will run the first time we initialize ourselves on a
- * particular display device. After that, we should just be able
- * to check the registry to see what the results of those tests were
- * and enable/disable DirectX support appropriately. Startup tests
- * may be re-run in situations where we cannot check the display
- * device information (it may fail on some OSs) or when the
- * display device we start up on is different from the devices
- * we have tested on before (eg, the user has switched video cards
- * or maybe display depths). The user may also force the tests to be re-run
- * by using the -Dsun.java2d.accelReset flag.
- */
-
-
-WCHAR *j2dAccelKey; // Name of java2d root key
-WCHAR *j2dAccelDriverKey; // Name of j2d per-device key
-int dxAcceleration; // dx acceleration ability
- // according to the Registry
-HINSTANCE hLibDDraw = NULL; // DDraw Library handle
-extern DDrawObjectStruct **ddInstance;
-extern CriticalSection ddInstanceLock;
-extern int maxDDDevices;
-extern int currNumDevices;
-extern char *javaVersion;
-
-
-BOOL CheckDDCreationCaps(DDrawObjectStruct *tmpDdInstance,
- DxCapabilities *dxCaps)
-{
- J2dTraceLn(J2D_TRACE_INFO, "CheckDDCreationCaps");
- if (dxCaps == NULL) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CheckDDCreationCaps: null dxCaps (new monitor?)");
- return FALSE;
- }
- // If we have not yet tested this configuration, test it now
- if (dxCaps->GetDdSurfaceCreationCap() == J2D_ACCEL_UNVERIFIED) {
- // First, create a non-d3d offscreen surface
- dxCaps->SetDdSurfaceCreationCap(J2D_ACCEL_TESTING);
- DDrawSurface *lpSurface =
- tmpDdInstance->ddObject->CreateDDOffScreenSurface(1, 1,
- 32, TR_OPAQUE, DDSCAPS_VIDEOMEMORY);
- if (!lpSurface) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CheckDDCreationCaps: failed to create basic "\
- "ddraw surface");
- // problems creating basic ddraw surface - log it and return FALSE
- dxCaps->SetDdSurfaceCreationCap(J2D_ACCEL_FAILURE);
- return FALSE;
- }
- // Success; log it and continue
- dxCaps->SetDdSurfaceCreationCap(J2D_ACCEL_SUCCESS);
- delete lpSurface;
- } else if (dxCaps->GetDdSurfaceCreationCap() != J2D_ACCEL_SUCCESS) {
- // we have tested and failed previously; return FALSE
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CheckDDCreationCaps: previous surface creation "\
- "failure detected");
- return FALSE;
- }
- return TRUE;
-}
-/**
- * Called from AwtWin32GraphicsEnv's initScreens() after it initializes
- * all of the display devices. This function initializes the global
- * DirectX state as well as the per-device DirectX objects. This process
- * includes:
- * - Checking native/Java flags to see what the user wants to manually
- * enable/disable
- * - Checking the registry to see if DirectX should be globally disabled
- * - Enumerating the display devices (this returns unique string IDs
- * for each display device)
- * - Checking the registry for each device to see what we have stored
- * there for this device.
- * - Enumerate the ddraw devices
- * - For each ddraw device, match it up with the associated device from
- * EnumDisplayDevices.
- * - If no registry entries exist, then run a series of tests using
- * ddraw and d3d, storing the results in the registry for this device ID
- * (and possibly color depth - test results may be bpp-specific)
- * - based on the results of the registry storage or the tests, enable
- * and disable various ddraw/d3d capabilities as appropriate.
- */
-void InitDirectX()
-{
-
- J2dRlsTraceLn(J2D_TRACE_INFO, "InitDirectX");
- // Check registry state for all display devices
- CheckRegistry();
-
- // Need to prevent multiple initializations of the DX objects/primaries.
- // CriticalSection ensures that this initialization will only happen once,
- // even if multiple threads call into this function at startup time.
- static CriticalSection initLock;
- initLock.Enter();
- static bool dxInitialized = false;
- if (dxInitialized) {
- initLock.Leave();
- return;
- }
- dxInitialized = true;
- initLock.Leave();
-
- // Check to make sure ddraw is not disabled globally
- if (useDD) {
- if (dxAcceleration == J2D_ACCEL_UNVERIFIED) {
- RegistryKey::SetIntValue(j2dAccelKey, J2D_ACCEL_DX_NAME,
- J2D_ACCEL_TESTING, TRUE);
- }
- hLibDDraw = ::LoadLibrary(TEXT("ddraw.dll"));
- if (!hLibDDraw) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "InitDirectX: Could not load library");
- SetDDEnabledFlag(NULL, FALSE);
- if (dxAcceleration == J2D_ACCEL_UNVERIFIED) {
- RegistryKey::SetIntValue(j2dAccelKey, J2D_ACCEL_DX_NAME,
- J2D_ACCEL_FAILURE, TRUE);
- }
- return;
- }
- if (dxAcceleration == J2D_ACCEL_UNVERIFIED) {
- RegistryKey::SetIntValue(j2dAccelKey, J2D_ACCEL_DX_NAME,
- J2D_ACCEL_SUCCESS, TRUE);
- }
- maxDDDevices = 1;
- ddInstance = (DDrawObjectStruct**)safe_Malloc(maxDDDevices *
- sizeof(DDrawObjectStruct*));
- memset(ddInstance, NULL, maxDDDevices * sizeof(DDrawObjectStruct*));
- if (!DDCreateObject()) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "InitDirectX: Could not create ddraw object");
- SetDDEnabledFlag(NULL, FALSE);
- }
- }
-
- if (checkRegistry) {
- // diagnostic purposes: iterate through all of the registry
- // settings we have just checked or set and print them out to
- // the console
- printf("Registry Settings:\n");
- RegistryKey::PrintValue(j2dAccelKey, J2D_ACCEL_DX_NAME,
- L" DxAcceleration");
- // Now check the registry entries for all display devices on the system
- int deviceNum = 0;
- _DISPLAY_DEVICE displayDevice;
- displayDevice.dwSize = sizeof(displayDevice);
- while (EnumDisplayDevices(NULL, deviceNum, & displayDevice, 0) &&
- deviceNum < 20) // avoid infinite loop with buggy drivers
- {
- DxCapabilities caps;
- if (displayDevice.dwFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP) {
- // We only care about actual display devices. Devices without
- // this flag could be virtual devices such as NetMeeting
- Devices::InstanceAccess devices;
- AwtWin32GraphicsDevice **devArray = devices->GetRawArray();
- int numScreens = devices->GetNumDevices();
- for (int i = 0; i < numScreens; ++i) {
- MONITOR_INFO_EXTENDED *pMonInfo =
- (PMONITOR_INFO_EXTENDED) devArray[i]->GetMonitorInfo();
- if (wcscmp(pMonInfo->strDevice,
- displayDevice.strDevName) == 0) {
- // this GraphicsDevice matches this DisplayDevice; check
- // the bit depth and grab the appropriate values from
- // the registry
- int bitDepth = devArray[i]->GetBitDepth();
- WCHAR driverKeyName[2048];
- WCHAR fullKeyName[2048];
- GetDeviceKeyName(&displayDevice, driverKeyName);
- swprintf(fullKeyName, L"%s%s\\%d", j2dAccelDriverKey,
- driverKeyName, bitDepth);
- printf(" Device\\Depth: %S\\%d\n",
- driverKeyName, bitDepth);
- caps.Initialize(fullKeyName);
- caps.PrintCaps();
- }
- }
- }
- deviceNum++;
- }
- }
-}
-
-/**
- * Utility function that derives a unique name for this display
- * device. We do this by combining the "name" and "string"
- * fields from the displayDevice structure. Note that we
- * remove '\' characters from the dev name; since we're going
- * to use this as a registry key, we do not want all those '\'
- * characters to create extra registry key levels.
- */
-void GetDeviceKeyName(_DISPLAY_DEVICE *displayDevice, WCHAR *devName)
-{
- WCHAR *strDevName = displayDevice->strDevName;
- int devNameIndex = 0;
- for (size_t i = 0; i < wcslen(strDevName); ++i) {
- if (strDevName[i] != L'\\') {
- devName[devNameIndex++] = strDevName[i];
- }
- }
- devName[devNameIndex++] = L' ';
- devName[devNameIndex] = L'\0';
- wcscat(devName, displayDevice->strDevString);
-}
-
-
-/**
- * CheckRegistry first queries the registry for whether DirectX
- * should be disabled globally. Then it enumerates the current
- * display devices and queries the registry for each unique display
- * device, putting the resulting values in the AwtWin32GraphicsDevice
- * array for each appropriate display device.
- */
-void CheckRegistry()
-{
- J2dTraceLn(J2D_TRACE_INFO, "CheckRegistry");
- if (accelReset) {
- RegistryKey::DeleteKey(j2dAccelKey);
- }
- dxAcceleration = RegistryKey::GetIntValue(j2dAccelKey, J2D_ACCEL_DX_NAME);
- if (dxAcceleration == J2D_ACCEL_TESTING ||
- dxAcceleration == J2D_ACCEL_FAILURE)
- {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CheckRegistry: previous ddraw initialization failure"\
- " detected, ddraw is disabled");
- // Disable ddraw if previous testing either crashed or failed
- SetDDEnabledFlag(NULL, FALSE);
- // Without DirectX, there is no point to the rest of the registry checks
- // so just return
- return;
- }
-
- // First, get the list of current display devices
- int deviceNum = 0; // all display devices (virtual and not)
- int numDesktopDevices = 0; // actual display devices
- _DISPLAY_DEVICE displayDevice;
- displayDevice.dwSize = sizeof(displayDevice);
- _DISPLAY_DEVICE displayDevices[20];
- while (deviceNum < 20 && // avoid infinite loop with buggy drivers
- EnumDisplayDevices(NULL, deviceNum, &displayDevice, 0))
- {
- if (displayDevice.dwFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP)
- {
- // We only care about actual display devices. Devices without
- // this flag could be virtual devices such as NetMeeting
- J2dRlsTraceLn2(J2D_TRACE_VERBOSE,
- "CheckRegistry: Found Display Device %d: %S",
- deviceNum, displayDevice.strDevString);
- displayDevices[numDesktopDevices] = displayDevice;
- ++numDesktopDevices;
- }
- deviceNum++;
- }
- // Workaround for platforms that do not have the EnumDisplayDevices function
- // (i.e., NT4): just set up a single device that has the display name that
- // has already been assigned to the first (and only) graphics device.
- if (deviceNum == 0) {
- Devices::InstanceAccess devices;
- MONITOR_INFO_EXTENDED *pMonInfo =
- (PMONITOR_INFO_EXTENDED) devices->GetDevice(0)->GetMonitorInfo();
- wcscpy(displayDevices[0].strDevName, pMonInfo->strDevice);
- wcscpy(displayDevices[0].strDevString, L"DefaultDriver");
- J2dRlsTraceLn(J2D_TRACE_VERBOSE,
- "CheckRegistry: Single Default Display Device detected");
- numDesktopDevices++;
- }
-
- // Now, check the current display devices against the list stored
- // in the registry already.
- // First, we get the current list of devices in the registry
- WCHAR subKeyNames[20][1024];
- int numSubKeys = 0;
- {
- RegistryKey hKey(j2dAccelDriverKey, KEY_ALL_ACCESS);
- DWORD buffSize = 1024;
- DWORD ret;
- while (numSubKeys < 20 && // same limit as display devices above
- ((ret = hKey.EnumerateSubKeys(numSubKeys, subKeyNames[numSubKeys],
- &buffSize)) ==
- ERROR_SUCCESS))
- {
- ++numSubKeys;
- buffSize = 1024;
- }
- }
- // Now, compare the display devices to the registry display devices
- BOOL devicesDifferent = FALSE;
- // Check that each display device is in the registry
- // Do this by checking each physical display device to see if it
- // is also in the registry. If it is, do the same for the rest of
- // the physical devices. If any device is not in the registry,
- // then there is a mis-match and we break out of the loop and
- // reset the registry.
- for (int i = 0; i < numDesktopDevices; ++i) {
- // Assume the device is not in the registry until proven otherwise
- devicesDifferent = TRUE;
- WCHAR driverName[2048];
- // Key name consists of (driver string) (driver name)
- // but we must remove the "\" characters from the driver
- // name to avoid creating too many levels
- GetDeviceKeyName(&(displayDevices[i]), driverName);
- for (int j = 0; j < numDesktopDevices; ++j) {
- if (wcscmp(driverName,
- subKeyNames[j]) == 0)
- {
- // Found a match for this device; time to move on
- devicesDifferent = FALSE;
- break;
- }
- }
- if (devicesDifferent) {
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "CheckRegistry: Display device %S not in registry",
- driverName);
- break;
- }
- }
- // Something was different in the runtime versus the registry; delete
- // the registry entries to force testing and writing the results to
- // the registry
- if (devicesDifferent) {
- for (int i = 0; i < numSubKeys; ++i) {
- WCHAR driverKeyName[2048];
- swprintf(driverKeyName, L"%s%s", j2dAccelDriverKey,
- subKeyNames[i]);
- J2dTraceLn1(J2D_TRACE_VERBOSE,
- "CheckRegistry: Deleting registry key: %S",
- driverKeyName);
- RegistryKey::DeleteKey(driverKeyName);
- }
- }
-
- // Now that we have the display devices and the registry in a good
- // start state, get or initialize the dx capabilities in the registry
- // for each display device
- for (deviceNum = 0; deviceNum < numDesktopDevices; ++deviceNum) {
- Devices::InstanceAccess devices;
- AwtWin32GraphicsDevice **devArray = devices->GetRawArray();
- int numScreens = devices->GetNumDevices();
- for (int i = 0; i < numScreens; ++i) {
- MONITOR_INFO_EXTENDED *pMonInfo =
- (PMONITOR_INFO_EXTENDED)devArray[i]->GetMonitorInfo();
- if (wcscmp(pMonInfo->strDevice,
- displayDevices[deviceNum].strDevName) == 0)
- {
- // this GraphicsDevice matches this DisplayDevice; check
- // the bit depth and grab the appropriate values from
- // the registry
- int bitDepth = devArray[i]->GetBitDepth();
- WCHAR driverKeyName[2048];
- WCHAR fullKeyName[2048];
- // Key name consists of (driver string) (driver name)
- // but we must remove the "\" characters from the driver
- // name to avoid creating too many levels
- GetDeviceKeyName(&(displayDevices[i]), driverKeyName);
- swprintf(fullKeyName, L"%s%s\\%d", j2dAccelDriverKey,
- driverKeyName, bitDepth);
- // - query registry for key with strDevString\\depth
- devArray[i]->GetDxCaps()->Initialize(fullKeyName);
- }
- }
- }
-}
-
-
-BOOL DDSetupDevice(DDrawObjectStruct *tmpDdInstance, DxCapabilities *dxCaps)
-{
- J2dRlsTraceLn(J2D_TRACE_INFO, "DDSetupDevice");
- BOOL surfaceBasics = CheckDDCreationCaps(tmpDdInstance, dxCaps);
- if (!surfaceBasics) {
- goto FAILURE;
- }
- // create primary surface. There is one of these per ddraw object.
- // A D3DContext will be attempted to be created during the creation
- // of the primary surface.
- tmpDdInstance->primary = tmpDdInstance->ddObject->CreateDDPrimarySurface(
- (DWORD)tmpDdInstance->backBufferCount);
- if (!tmpDdInstance->primary) {
- goto FAILURE;
- }
- J2dRlsTraceLn(J2D_TRACE_VERBOSE,
- "DDSetupDevice: successfully created primary surface");
- if (!tmpDdInstance->capsSet) {
- DDCAPS caps;
- tmpDdInstance->ddObject->GetDDCaps(&caps);
- tmpDdInstance->canBlt = (caps.dwCaps & DDCAPS_BLT);
- BOOL canCreateOffscreen = tmpDdInstance->canBlt &&
- (caps.dwVidMemTotal > 0);
- // Only register offscreen creation ok if we can Blt and if there
- // is available video memory. Otherwise it
- // is useless functionality. The Barco systems apparently allow
- // offscreen creation but do not allow hardware Blt's
- if ((caps.dwCaps & DDCAPS_NOHARDWARE) || !canCreateOffscreen) {
- AwtWin32GraphicsDevice::DisableOffscreenAccelerationForDevice(
- tmpDdInstance->hMonitor);
- if (caps.dwCaps & DDCAPS_NOHARDWARE) {
- // Does not have basic functionality we need; release
- // ddraw instance and return FALSE for this device.
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "DDSetupDevice: Disabling ddraw on "\
- "device: no hw support");
- goto FAILURE;
- }
- }
- tmpDdInstance->capsSet = TRUE;
- }
- // Do NOT create a clipper in full-screen mode
- if (tmpDdInstance->hwndFullScreen == NULL) {
- if (!tmpDdInstance->clipper) {
- // May have already created a clipper
- tmpDdInstance->clipper = tmpDdInstance->ddObject->CreateDDClipper();
- }
- if (tmpDdInstance->clipper != NULL) {
- if (tmpDdInstance->primary->SetClipper(tmpDdInstance->clipper)
- != DD_OK)
- {
- goto FAILURE;
- }
- } else {
- goto FAILURE;
- }
- }
- J2dRlsTraceLn(J2D_TRACE_VERBOSE,
- "DDSetupDevice: successfully setup ddraw device");
- return TRUE;
-
-FAILURE:
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "DDSetupDevice: Failed to setup ddraw device");
- AwtWin32GraphicsDevice::DisableOffscreenAccelerationForDevice(
- tmpDdInstance->hMonitor);
- ddInstanceLock.Enter();
- // Do not release the ddInstance structure here, just flag it
- // as having problems; other threads may currently be using a
- // reference to the structure and we cannot release it out from
- // under them. It will be released sometime later
- // when all DD resources are released.
- tmpDdInstance->accelerated = FALSE;
- ddInstanceLock.Leave();
- return FALSE;
-}
-
-DDrawObjectStruct *CreateDevice(GUID *lpGUID, HMONITOR hMonitor)
-{
- J2dRlsTraceLn2(J2D_TRACE_INFO, "CreateDevice: lpGUID=0x%x hMon=0x%x",
- lpGUID, hMonitor);
- DDrawObjectStruct *tmpDdInstance =
- (DDrawObjectStruct*)safe_Calloc(1, sizeof(DDrawObjectStruct));
- memset(tmpDdInstance, NULL, sizeof(DDrawObjectStruct));
- tmpDdInstance->valid = TRUE;
- tmpDdInstance->accelerated = TRUE;
- tmpDdInstance->capsSet = FALSE;
- tmpDdInstance->hMonitor = hMonitor;
- tmpDdInstance->hwndFullScreen = NULL;
- tmpDdInstance->backBufferCount = 0;
- tmpDdInstance->syncSurface = NULL;
- tmpDdInstance->context = CONTEXT_NORMAL;
- // Create ddraw object
- DxCapabilities *dxCaps =
- AwtWin32GraphicsDevice::GetDxCapsForDevice(hMonitor);
- if (dxCaps->GetDdCreationCap() == J2D_ACCEL_UNVERIFIED) {
- dxCaps->SetDdCreationCap(J2D_ACCEL_TESTING);
- } else if (dxCaps->GetDdCreationCap() != J2D_ACCEL_SUCCESS) {
- J2dRlsTraceLn(J2D_TRACE_ERROR,
- "CreateDevice: previous failure detected, "\
- "no ddraw device created");
- free(tmpDdInstance);
- return NULL;
- }
- tmpDdInstance->ddObject = DDraw::CreateDDrawObject(lpGUID, hMonitor);
- if (dxCaps->GetDdCreationCap() == J2D_ACCEL_TESTING) {
- dxCaps->SetDdCreationCap(tmpDdInstance->ddObject ? J2D_ACCEL_SUCCESS :
- J2D_ACCEL_FAILURE);
- }
- if (!tmpDdInstance->ddObject) {
- // REMIND: might want to shut down ddraw (useDD == FALSE?)
- // if this error occurs
- free(tmpDdInstance);
- return NULL;
- }
- if (DDSetupDevice(tmpDdInstance, dxCaps)) {
- return tmpDdInstance;
- } else {
- free(tmpDdInstance);
- return NULL;
- }
-}
-
-BOOL CALLBACK EnumDeviceCallback(GUID FAR* lpGUID, LPSTR szName, LPSTR szDevice,
- LPVOID lParam, HMONITOR hMonitor)
-{
- J2dTraceLn(J2D_TRACE_INFO, "EnumDeviceCallback");
- if (currNumDevices == maxDDDevices) {
- maxDDDevices *= 2;
- DDrawObjectStruct **tmpDDDevices =
- (DDrawObjectStruct**)safe_Malloc(maxDDDevices *
- sizeof(DDrawObjectStruct*));
- memset(tmpDDDevices, NULL, maxDDDevices * sizeof(DDrawObjectStruct*));
- for (int i = 0; i < currNumDevices; ++i) {
- tmpDDDevices[i] = ddInstance[i];
- }
- DDrawObjectStruct **oldDDDevices = ddInstance;
- ddInstance = tmpDDDevices;
- free(oldDDDevices);
- }
- if (hMonitor != NULL) {
- DDrawObjectStruct *tmpDdInstance;
- if (ddInstance[currNumDevices] != NULL) {
- DDFreeSyncSurface(ddInstance[currNumDevices]);
- free(ddInstance[currNumDevices]);
- }
- tmpDdInstance = CreateDevice(lpGUID, hMonitor);
- ddInstance[currNumDevices] = tmpDdInstance;
- J2dTraceLn2(J2D_TRACE_VERBOSE,
- "EnumDeviceCallback: ddInstance[%d]=0x%x",
- currNumDevices, tmpDdInstance);
- // Increment currNumDevices on success or failure; a null device
- // is perfectly fine; we may have an unaccelerated device
- // in the midst of our multimon configuration
- currNumDevices++;
- }
- return TRUE;
-}
-
-typedef HRESULT (WINAPI *FnDDEnumerateFunc)(LPDDENUMCALLBACK cb,
- LPVOID lpContext);
-
-/**
- * Create the ddraw object and the global
- * ddInstance structure. Note that we do not take the ddInstanceLock
- * here; we assume that our callers are taking that lock for us.
- */
-BOOL DDCreateObject() {
- LPDIRECTDRAWENUMERATEEXA lpDDEnum;
-
- J2dTraceLn(J2D_TRACE_INFO, "DDCreateObject");
-
- currNumDevices = 0;
- // Note that we are hardcoding this call to the ANSI version and not
- // using the ANIS-or-UNICODE macro name. This is because there is
- // apparently a problem with the UNICODE function name not being
- // implemented under the win98 MSLU. So we just use the ANSI version
- // on all flavors of Windows instead.
- lpDDEnum = (LPDIRECTDRAWENUMERATEEXA)
- GetProcAddress(hLibDDraw, "DirectDrawEnumerateExA");
- if (lpDDEnum) {
- HRESULT ddResult = (lpDDEnum)(EnumDeviceCallback,
- NULL, DDENUM_ATTACHEDSECONDARYDEVICES);
- if (ddResult != DD_OK) {
- DebugPrintDirectDrawError(ddResult,
- "DDCreateObject: EnumDeviceCallback failed");
- }
- }
- if (currNumDevices == 0) {
- // Either there was no ddEnumEx function or there was a problem during
- // enumeration; just create a device on the primary.
- ddInstance[currNumDevices++] = CreateDevice(NULL, NULL);
- }
- return TRUE;
-}
diff --git a/jdk/src/windows/native/sun/windows/Devices.cpp b/jdk/src/windows/native/sun/windows/Devices.cpp
index d0432041675..8bcee15b877 100644
--- a/jdk/src/windows/native/sun/windows/Devices.cpp
+++ b/jdk/src/windows/native/sun/windows/Devices.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 2001-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2001-2008 Sun Microsystems, Inc. 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
@@ -84,6 +84,7 @@
#include "Devices.h"
#include "Trace.h"
#include "awt_Multimon.h"
+#include "D3DPipelineManager.h"
Devices* Devices::theInstance = NULL;
CriticalSection Devices::arrayLock;
@@ -133,13 +134,12 @@ BOOL Devices::UpdateInstance(JNIEnv *env)
AwtWin32GraphicsDevice** rawDevices = newDevices->GetRawArray();
int i;
for (i = 0; i < numScreens; ++i) {
+ J2dTraceLn2(J2D_TRACE_VERBOSE, " hmon[%d]=0x%x", i, monHds[i]);
rawDevices[i] = new AwtWin32GraphicsDevice(i, monHds[i], newDevices);
}
for (i = 0; i < numScreens; ++i) {
rawDevices[i]->Initialize();
}
- free(monHds);
-
{
CriticalSection::Lock l(arrayLock);
@@ -161,12 +161,14 @@ BOOL Devices::UpdateInstance(JNIEnv *env)
"Devices::UpdateInstance: device removed: %d", i);
oldDevices->GetDevice(i)->Invalidate(env);
}
-
// Now that we have a new array in place, remove this (possibly the
// last) reference to the old instance.
oldDevices->Release();
}
+ D3DPipelineManager::HandleAdaptersChange((HMONITOR*)monHds,
+ theInstance->GetNumDevices());
}
+ free(monHds);
return TRUE;
}
diff --git a/jdk/src/windows/native/sun/windows/awt.h b/jdk/src/windows/native/sun/windows/awt.h
index 2a1d9904c9b..389aba67db3 100644
--- a/jdk/src/windows/native/sun/windows/awt.h
+++ b/jdk/src/windows/native/sun/windows/awt.h
@@ -155,6 +155,7 @@ typedef AwtObject* PDATA;
#endif
#define IS_NT (IS_WIN32 && !(::GetVersion() & 0x80000000))
#define IS_WIN2000 (IS_NT && LOBYTE(LOWORD(::GetVersion())) >= 5)
+#define IS_WIN2003 (IS_NT && LOBYTE(LOWORD(::GetVersion())) == 5 && HIBYTE(LOWORD(::GetVersion())) >= 2)
#define IS_WINXP (IS_NT && (IS_WIN2000 && HIBYTE(LOWORD(::GetVersion())) >= 1) || LOBYTE(LOWORD(::GetVersion())) > 5)
#define IS_WINVISTA (IS_NT && LOBYTE(LOWORD(::GetVersion())) >= 6)
#define IS_WIN32S (IS_WIN32 && !IS_NT && LOBYTE(LOWORD(::GetVersion())) < 4)
diff --git a/jdk/src/windows/native/sun/windows/awt_Component.cpp b/jdk/src/windows/native/sun/windows/awt_Component.cpp
index ea9ce00f0aa..5b017c97810 100644
--- a/jdk/src/windows/native/sun/windows/awt_Component.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_Component.cpp
@@ -45,7 +45,6 @@
#include "awt_Unicode.h"
#include "awt_Window.h"
#include "awt_Win32GraphicsDevice.h"
-#include "ddrawUtils.h"
#include "Hashtable.h"
#include "ComCtl32Util.h"
@@ -165,6 +164,11 @@ struct SetRectangularShapeStruct {
jint x1, x2, y1, y2;
jobject region;
};
+// Struct for _GetInsets function
+struct GetInsetsStruct {
+ jobject window;
+ RECT *insets;
+};
/************************************************************************/
//////////////////////////////////////////////////////////////////////////
@@ -414,7 +418,10 @@ BOOL AwtComponent::IsFocusable() {
jobject peer = GetPeer(env);
jobject target = env->GetObjectField(peer, AwtObject::targetID);
BOOL res = env->GetBooleanField(target, focusableID);
- res &= GetContainer()->IsFocusableWindow();
+ AwtWindow *pCont = GetContainer();
+ if (pCont) {
+ res &= pCont->IsFocusableWindow();
+ }
env->DeleteLocalRef(target);
return res;
}
@@ -1936,25 +1943,6 @@ LRESULT AwtComponent::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
*((SIZE *)lParam));
mr = mrConsume;
break;
- case WM_AWT_DD_CREATE_SURFACE:
- {
- return (LRESULT)WmDDCreateSurface((Win32SDOps*)wParam);
- }
- case WM_AWT_DD_ENTER_FULLSCREEN:
- {
- mr = WmDDEnterFullScreen((HMONITOR)wParam);
- break;
- }
- case WM_AWT_DD_EXIT_FULLSCREEN:
- {
- mr = WmDDExitFullScreen((HMONITOR)wParam);
- break;
- }
- case WM_AWT_DD_SET_DISPLAY_MODE:
- {
- mr = WmDDSetDisplayMode((HMONITOR)wParam, (DDrawDisplayMode*)lParam);
- break;
- }
case WM_UNDOCUMENTED_CLICKMENUBAR:
{
if (::IsWindow(AwtWindow::GetModalBlocker(GetHWnd()))) {
@@ -2223,7 +2211,8 @@ AwtComponent::AwtSetFocus()
}
}
- AwtFrame *owner = GetContainer()->GetOwningFrameOrDialog();
+ AwtWindow *pCont = GetContainer();
+ AwtFrame *owner = pCont ? pCont->GetOwningFrameOrDialog() : NULL;
if (owner == NULL) {
::SetFocus(hwnd);
@@ -4806,31 +4795,6 @@ jintArray AwtComponent::CreatePrintedPixels(SIZE &loc, SIZE &size) {
return pixelArray;
}
-BOOL AwtComponent::WmDDCreateSurface(Win32SDOps* wsdo) {
- return DDCreateSurface(wsdo);
-}
-
-// This method is fully implemented in AwtWindow
-MsgRouting AwtComponent::WmDDEnterFullScreen(HMONITOR monitor) {
- DASSERT(FALSE);
- return mrDoDefault;
-}
-
-// This method is fully implemented in AwtWindow
-MsgRouting AwtComponent::WmDDExitFullScreen(HMONITOR monitor) {
- DASSERT(FALSE);
- return mrDoDefault;
-}
-
-MsgRouting AwtComponent::WmDDSetDisplayMode(HMONITOR monitor,
- DDrawDisplayMode* pDisplayMode) {
-
- DDSetDisplayMode(monitor, *pDisplayMode);
-
- delete pDisplayMode;
- return mrDoDefault;
-}
-
void *
AwtComponent::GetNativeFocusOwner() {
JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
@@ -5377,7 +5341,8 @@ void AwtComponent::SynthesizeMouseMessage(JNIEnv *env, jobject mouseEvent)
MSG* msg = CreateMessage(message, wParam, MAKELPARAM(x, y), x, y);
// If the window is not focusable but if this is a focusing
// message we should skip it then and perform our own actions.
- if (((AwtWindow*)GetContainer())->IsFocusableWindow() || !ActMouseMessage(msg)) {
+ AwtWindow *pCont = GetContainer();
+ if ((pCont && pCont->IsFocusableWindow()) || !ActMouseMessage(msg)) {
PostHandleEventMessage(msg, TRUE);
} else {
delete msg;
@@ -5900,7 +5865,10 @@ void AwtComponent::_NativeHandleEvent(void *param)
AwtComponent* p = (AwtComponent*)pData;
// If the window is not focusable but if this is a focusing
// message we should skip it then and perform our own actions.
- if (((AwtWindow*)p->GetContainer())->IsFocusableWindow() || !p->ActMouseMessage(&msg)) {
+ AwtWindow *pCont = (AwtWindow*)(p->GetContainer());
+ if ((pCont && pCont->IsFocusableWindow()) ||
+ !p->ActMouseMessage(&msg))
+ {
// Create copy for local msg
MSG* pCopiedMsg = new MSG;
memmove(pCopiedMsg, &msg, sizeof(MSG));
@@ -6439,6 +6407,46 @@ AwtComponent_GetHWnd(JNIEnv *env, jlong pData)
return p->GetHWnd();
}
+static void _GetInsets(void* param)
+{
+ JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
+
+ GetInsetsStruct *gis = (GetInsetsStruct *)param;
+ jobject self = gis->window;
+
+ gis->insets->left = gis->insets->top =
+ gis->insets->right = gis->insets->bottom = 0;
+
+ PDATA pData;
+ JNI_CHECK_PEER_GOTO(self, ret);
+ AwtComponent *component = (AwtComponent *)pData;
+
+ component->GetInsets(gis->insets);
+
+ ret:
+ env->DeleteGlobalRef(self);
+ delete gis;
+}
+
+/**
+ * This method is called from the WGL pipeline when it needs to retrieve
+ * the insets associated with a ComponentPeer's C++ level object.
+ */
+void AwtComponent_GetInsets(JNIEnv *env, jobject peer, RECT *insets)
+{
+ TRY;
+
+ GetInsetsStruct *gis = new GetInsetsStruct;
+ gis->window = env->NewGlobalRef(peer);
+ gis->insets = insets;
+
+ AwtToolkit::GetInstance().InvokeFunction(_GetInsets, gis);
+ // global refs and mds are deleted in _UpdateWindow
+
+ CATCH_BAD_ALLOC;
+
+}
+
JNIEXPORT void JNICALL
Java_java_awt_Component_initIDs(JNIEnv *env, jclass cls)
{
diff --git a/jdk/src/windows/native/sun/windows/awt_Component.h b/jdk/src/windows/native/sun/windows/awt_Component.h
index 14c02700734..f73980c28b4 100644
--- a/jdk/src/windows/native/sun/windows/awt_Component.h
+++ b/jdk/src/windows/native/sun/windows/awt_Component.h
@@ -32,7 +32,7 @@
#include "awt_Brush.h"
#include "awt_Pen.h"
#include "awt_Win32GraphicsDevice.h"
-#include "Win32SurfaceData.h"
+#include "GDIWindowSurfaceData.h"
#include "java_awt_Component.h"
#include "sun_awt_windows_WComponentPeer.h"
@@ -62,8 +62,6 @@ class AwtPopupMenu;
class AwtDropTarget;
-class DDrawDisplayMode;
-
struct WmComponentSetFocusData;
/*
@@ -203,7 +201,8 @@ public:
*/
AwtComponent* GetParent();
- /* Get the component's immediate container. */
+ /* Get the component's immediate container. Note: may return NULL while
+ the component is being reparented in full-screen mode by Direct3D */
class AwtWindow* GetContainer();
/* Is a component a container? Used by above method */
@@ -600,11 +599,6 @@ public:
jintArray CreatePrintedPixels(SIZE &loc, SIZE &size);
- virtual BOOL WmDDCreateSurface(Win32SDOps* wsdo);
- virtual MsgRouting WmDDEnterFullScreen(HMONITOR monitor);
- virtual MsgRouting WmDDExitFullScreen(HMONITOR monitor);
- virtual MsgRouting WmDDSetDisplayMode(HMONITOR monitor, DDrawDisplayMode* pDisplayMode);
-
static void * GetNativeFocusOwner();
static void * GetNativeFocusedWindow();
static void ClearGlobalFocusOwner();
diff --git a/jdk/src/windows/native/sun/windows/awt_DrawingSurface.cpp b/jdk/src/windows/native/sun/windows/awt_DrawingSurface.cpp
index a69d932d03f..d1ef0568494 100644
--- a/jdk/src/windows/native/sun/windows/awt_DrawingSurface.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_DrawingSurface.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 1996-2005 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1996-2008 Sun Microsystems, Inc. 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
@@ -25,12 +25,10 @@
#define _JNI_IMPLEMENTATION_
#include "awt_DrawingSurface.h"
-#include "WindowsFlags.h"
#include "awt_Component.h"
jclass jawtVImgClass;
jclass jawtComponentClass;
-jclass jawtW32ossdClass;
jfieldID jawtPDataID;
jfieldID jawtSDataID;
jfieldID jawtSMgrID;
@@ -82,31 +80,7 @@ jint JAWTOffscreenDrawingSurfaceInfo::Init(JAWTOffscreenDrawingSurface* parent)
{
TRY;
- JNIEnv* env = parent->env;
- jobject target = parent->target;
- if (JNU_IsNull(env, target)) {
- DTRACE_PRINTLN("NULL target");
- return JAWT_LOCK_ERROR;
- }
- Win32SDOps * ops =
- (Win32SDOps *)((void*)env->GetLongField(target, jawtPDataID));
- if (ops == NULL) {
- DTRACE_PRINTLN("NULL ops");
- return JAWT_LOCK_ERROR;
- }
- ddrawSurface = ops->lpSurface;
- if (ddrawSurface == NULL) {
- DTRACE_PRINTLN("NULL lpSurface");
- return JAWT_LOCK_ERROR;
- }
- DXSurface *dxSurface = ddrawSurface->GetDXSurface();
- if (dxSurface == NULL) {
- DTRACE_PRINTLN("NULL dxSurface");
- return JAWT_LOCK_ERROR;
- }
- platformInfo = this;
- ds = parent;
- return 0;
+ return JAWT_LOCK_ERROR;
CATCH_BAD_ALLOC_RET(JAWT_LOCK_ERROR);
}
@@ -248,58 +222,12 @@ void JNICALL JAWTOffscreenDrawingSurface::FreeDSI
jint JNICALL JAWTOffscreenDrawingSurface::LockSurface
(JAWT_DrawingSurface* ds)
{
- TRY;
-
- if (ds == NULL) {
- DTRACE_PRINTLN("Drawing Surface is NULL");
- return JAWT_LOCK_ERROR;
- }
- JAWTOffscreenDrawingSurface* pds =
- static_cast(ds);
- jint val = pds->info.Init(pds);
- if ((val & JAWT_LOCK_ERROR) != 0) {
- return val;
- }
- DDrawSurface *ddrawSurface = pds->info.ddrawSurface;
- if (ddrawSurface == NULL) {
- return JAWT_LOCK_ERROR;
- }
- ddrawSurface->GetExclusiveAccess();
- DXSurface *dxSurface = ddrawSurface->GetDXSurface();
- if (!dxSurface) {
- return JAWT_LOCK_ERROR;
- }
- switch (dxSurface->GetVersionID()) {
- case VERSION_DX7:
- {
- pds->info.dx7Surface = dxSurface->GetDDSurface();
- break;
- }
- default:
- // Leave info values at default and return error
- DTRACE_PRINTLN1("unknown jawt offscreen version: %d\n",
- dxSurface->GetVersionID());
- return JAWT_LOCK_ERROR;
- }
- return 0;
-
- CATCH_BAD_ALLOC_RET(JAWT_LOCK_ERROR);
+ return JAWT_LOCK_ERROR;
}
void JNICALL JAWTOffscreenDrawingSurface::UnlockSurface
(JAWT_DrawingSurface* ds)
{
- TRY_NO_VERIFY;
-
- if (ds == NULL) {
- DTRACE_PRINTLN("Drawing Surface is NULL");
- return;
- }
- JAWTOffscreenDrawingSurface* pds =
- static_cast(ds);
- pds->info.ddrawSurface->ReleaseExclusiveAccess();
-
- CATCH_BAD_ALLOC;
}
/* C exports */
@@ -313,31 +241,9 @@ extern "C" JNIEXPORT JAWT_DrawingSurface* JNICALL DSGetDrawingSurface
if (env->IsInstanceOf(target, jawtComponentClass)) {
return new JAWTDrawingSurface(env, target);
}
- // Sharing native offscreen surfaces is disabled by default in
- // this release. Sharing is enabled via the -Dsun.java2d.offscreenSharing
- // flag.
- if (g_offscreenSharing && env->IsInstanceOf(target, jawtVImgClass)) {
- jobject sMgr, sData;
- sMgr = env->GetObjectField(target, jawtSMgrID);
- if (!sMgr) {
- return NULL;
- }
- sData = env->GetObjectField(sMgr, jawtSDataID);
- if (!sData || !(env->IsInstanceOf(sData, jawtW32ossdClass))) {
- return NULL;
- }
- return new JAWTOffscreenDrawingSurface(env, sData);
- } else {
- if (!g_offscreenSharing) {
- DTRACE_PRINTLN(
- "GetDrawingSurface target must be a Component");
- } else {
- DTRACE_PRINTLN(
- "GetDrawingSurface target must be a Component or VolatileImage");
- }
- return NULL;
- }
+ DTRACE_PRINTLN("GetDrawingSurface target must be a Component");
+ return NULL;
CATCH_BAD_ALLOC_RET(NULL);
}
diff --git a/jdk/src/windows/native/sun/windows/awt_DrawingSurface.h b/jdk/src/windows/native/sun/windows/awt_DrawingSurface.h
index 5bfd7b7353a..4027ebe722c 100644
--- a/jdk/src/windows/native/sun/windows/awt_DrawingSurface.h
+++ b/jdk/src/windows/native/sun/windows/awt_DrawingSurface.h
@@ -1,5 +1,5 @@
/*
- * Copyright 1997-2004 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1997-2008 Sun Microsystems, Inc. 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
@@ -30,7 +30,7 @@
#include
#include
#include "awt_Component.h"
-#include "ddrawObject.h"
+#include
class JAWTDrawingSurface;
class JAWTOffscreenDrawingSurface;
@@ -84,8 +84,6 @@ class JAWTOffscreenDrawingSurfaceInfo :
public:
jint Init(JAWTOffscreenDrawingSurface* parent);
-public:
- DDrawSurface *ddrawSurface;
};
/*
diff --git a/jdk/src/windows/native/sun/windows/awt_PrintJob.cpp b/jdk/src/windows/native/sun/windows/awt_PrintJob.cpp
index 5d15e72540a..0ecd0388ac0 100644
--- a/jdk/src/windows/native/sun/windows/awt_PrintJob.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_PrintJob.cpp
@@ -632,7 +632,9 @@ Java_sun_awt_windows_WPrinterJob_getDefaultPage(JNIEnv *env, jobject self,
return ;
}
- if (pDevMode->dmFields & DM_PAPERSIZE) {
+ if ((pDevMode->dmFields & DM_PAPERSIZE) ||
+ (pDevMode->dmFields & DM_PAPERWIDTH) ||
+ (pDevMode->dmFields & DM_PAPERLENGTH)) {
POINT paperSize;
RECT margins;
jint orientation = PAGEFORMAT_PORTRAIT;
diff --git a/jdk/src/windows/native/sun/windows/awt_Toolkit.cpp b/jdk/src/windows/native/sun/windows/awt_Toolkit.cpp
index 958ca8d871d..3a9dfb24c7d 100644
--- a/jdk/src/windows/native/sun/windows/awt_Toolkit.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_Toolkit.cpp
@@ -52,12 +52,13 @@
#include "CmdIDList.h"
#include "awt_new.h"
#include "awt_Unicode.h"
-#include "ddrawUtils.h"
#include "debug_trace.h"
#include "debug_mem.h"
#include "ComCtl32Util.h"
+#include "D3DPipelineManager.h"
+
#include
#include
@@ -78,11 +79,12 @@ extern BOOL windowMoveLockHeld;
extern jclass jawtVImgClass;
extern jclass jawtVSMgrClass;
extern jclass jawtComponentClass;
-extern jclass jawtW32ossdClass;
extern jfieldID jawtPDataID;
extern jfieldID jawtSDataID;
extern jfieldID jawtSMgrID;
+extern void DWMResetCompositionEnabled();
+
/************************************************************************
* Utilities
*/
@@ -268,20 +270,6 @@ extern "C" BOOL APIENTRY DllMain(HANDLE hInstance, DWORD ul_reason_for_call,
DTrace_DisableMutex();
DMem_DisableMutex();
#endif DEBUG
- // Release any resources that have not yet been released
- // Note that releasing DirectX objects is necessary for some
- // failure situations on win9x (such as the primary remaining
- // locked on application exit) but cannot be done during
- // PROCESS_DETACH on XP. On NT and win2k calling this ends up
- // in a catch() clause in the calling function, but on XP
- // the process simply hangs during the release of the ddraw
- // device object. Thus we check for NT here and do not bother
- // with the release on any NT flavored OS. Note that XP is
- // based on NT, so the IS_NT check is valid for NT4, win2k,
- // XP, and presumably XP follow-ons.
- if (!IS_NT) {
- DDRelease();
- }
break;
}
return TRUE;
@@ -465,6 +453,11 @@ BOOL AwtToolkit::Dispose() {
awt_dnd_uninitialize();
awt_clipboard_uninitialize((JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2));
+ // dispose Direct3D-related resources. This should be done
+ // before AwtObjectList::Cleanup() as the d3d will attempt to
+ // shutdown when the last of its windows is disposed of
+ D3DPipelineManager::DeleteInstance();
+
AwtObjectList::Cleanup();
AwtFont::Cleanup();
@@ -697,20 +690,6 @@ LRESULT CALLBACK AwtToolkit::WndProc(HWND hWnd, UINT message,
AwtObjectList::Cleanup();
return 0;
}
- case WM_AWT_D3D_CREATE_DEVICE: {
- DDraw *ddObject = (DDraw*)wParam;
- if (ddObject != NULL) {
- ddObject->InitD3DContext();
- }
- return 0;
- }
- case WM_AWT_D3D_RELEASE_DEVICE: {
- DDraw *ddObject = (DDraw*)wParam;
- if (ddObject != NULL) {
- ddObject->ReleaseD3DContext();
- }
- return 0;
- }
case WM_SYSCOLORCHANGE: {
jclass systemColorClass = env->FindClass("java/awt/SystemColor");
@@ -745,6 +724,17 @@ LRESULT CALLBACK AwtToolkit::WndProc(HWND hWnd, UINT message,
}
return 0;
}
+#ifndef WM_DWMCOMPOSITIONCHANGED
+#define WM_DWMCOMPOSITIONCHANGED 0x031E
+#define WM_DWMNCRENDERINGCHANGED 0x031F
+#define WM_DWMCOLORIZATIONCOLORCHANGED 0x0320
+#define WM_DWMWINDOWMAXIMIZEDCHANGED 0x0321
+#endif // WM_DWMCOMPOSITIONCHANGED
+ case WM_DWMCOMPOSITIONCHANGED: {
+ DWMResetCompositionEnabled();
+ return 0;
+ }
+
case WM_TIMER: {
// 6479820. Should check if a window is in manual resizing process: skip
// sending any MouseExit/Enter events while inside resize-loop.
@@ -901,17 +891,6 @@ LRESULT CALLBACK AwtToolkit::WndProc(HWND hWnd, UINT message,
// Reinitialize screens
initScreens(env);
- // Invalidate current DDraw object; the object must be recreated
- // when we first try to create a new DDraw surface. Note that we
- // don't recreate the ddraw object directly here because of
- // multi-threading issues; we'll just leave that to the first
- // time an object tries to create a DDraw surface under the new
- // display depth (which will happen after the displayChange event
- // propagation at the end of this case).
- if (DDCanReplaceSurfaces(NULL)) {
- DDInvalidateDDInstance(NULL);
- }
-
// Notify Java side - call WToolkit.displayChanged()
jclass clazz = env->FindClass("sun/awt/windows/WToolkit");
env->CallStaticVoidMethod(clazz, AwtToolkit::displayChangeMID);
@@ -1545,9 +1524,6 @@ Java_sun_awt_windows_WToolkit_initIDs(JNIEnv *env, jclass cls)
DASSERT(vSMgrClassLocal != 0);
jclass componentClassLocal = env->FindClass("java/awt/Component");
DASSERT(componentClassLocal != 0);
- jclass w32ossdClassLocal =
- env->FindClass("sun/java2d/windows/Win32OffScreenSurfaceData");
- DASSERT(w32ossdClassLocal != 0);
jawtSMgrID = env->GetFieldID(vImgClassLocal, "volSurfaceManager",
"Lsun/awt/image/VolatileSurfaceManager;");
DASSERT(jawtSMgrID != 0);
@@ -1560,7 +1536,6 @@ Java_sun_awt_windows_WToolkit_initIDs(JNIEnv *env, jclass cls)
// Save these classes in global references for later use
jawtVImgClass = (jclass)env->NewGlobalRef(vImgClassLocal);
jawtComponentClass = (jclass)env->NewGlobalRef(componentClassLocal);
- jawtW32ossdClass = (jclass)env->NewGlobalRef(w32ossdClassLocal);
CATCH_BAD_ALLOC;
}
@@ -1867,7 +1842,6 @@ Java_sun_awt_windows_WToolkit_nativeSync(JNIEnv *env, jobject self)
// Synchronize both GDI and DDraw
VERIFY(::GdiFlush());
- DDSync();
CATCH_BAD_ALLOC;
}
diff --git a/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.cpp b/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.cpp
index ff144bc61a0..dc4d05efed8 100644
--- a/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -43,7 +43,6 @@
#include
#include "awt_Canvas.h"
#include "awt_Win32GraphicsDevice.h"
-#include "ddrawUtils.h"
#include "java_awt_Transparency.h"
#include "java_awt_color_ColorSpace.h"
#include "sun_awt_Win32GraphicsDevice.h"
@@ -60,7 +59,6 @@ jclass AwtWin32GraphicsDevice::wToolkitClass;
jfieldID AwtWin32GraphicsDevice::dynamicColorModelID;
jfieldID AwtWin32GraphicsDevice::indexCMrgbID;
jfieldID AwtWin32GraphicsDevice::indexCMcacheID;
-jfieldID AwtWin32GraphicsDevice::accelerationEnabledID;
jmethodID AwtWin32GraphicsDevice::paletteChangedMID;
BOOL AwtWin32GraphicsDevice::primaryPalettized;
int AwtWin32GraphicsDevice::primaryIndex = 0;
@@ -455,7 +453,7 @@ void AwtWin32GraphicsDevice::SetGrayness(int grayValue)
/**
* Update our dynamic IndexedColorModel. This happens after
* a change to the system palette. Any surfaces stored in vram
- * (Win32OffScreenSurfaceData and Win32SurfaceData objects)
+ * (Win32OffScreenSurfaceData and GDIWindowSurfaceData objects)
* refer to this colorModel and use its lookup table and inverse
* lookup to calculate correct index values for rgb colors. So
* the colorModel must always reflect the current state of the
@@ -602,14 +600,6 @@ void AwtWin32GraphicsDevice::Release()
void AwtWin32GraphicsDevice::SetJavaDevice(JNIEnv *env, jobject objPtr)
{
javaDevice = env->NewWeakGlobalRef(objPtr);
- // We may have discovered earlier that the device is not
- // accelerated, but there were no way to notify the GraphicsDevice
- // about it because the DirectDraw initialization happens before those
- // are created (from the static initializer of Win32GraphicsEnvironment).
- DDrawObjectStruct *tmpDdInstance = GetDDInstanceForDevice((HMONITOR)monitor);
- if (tmpDdInstance == NULL || !tmpDdInstance->accelerated) {
- DisableOffscreenAcceleration();
- }
}
/**
@@ -619,10 +609,7 @@ void AwtWin32GraphicsDevice::SetJavaDevice(JNIEnv *env, jobject objPtr)
*/
void AwtWin32GraphicsDevice::DisableOffscreenAcceleration()
{
- JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
- if (!JNU_IsNull(env, javaDevice)) {
- env->SetBooleanField(javaDevice, accelerationEnabledID, JNI_FALSE);
- }
+ // REMIND: noop for now
}
/**
@@ -695,23 +682,6 @@ void AwtWin32GraphicsDevice::DisableOffscreenAccelerationForDevice(
}
}
-DxCapabilities *AwtWin32GraphicsDevice::GetDxCapsForDevice(MHND hMonitor)
-{
- Devices::InstanceAccess devices;
-
- if (hMonitor == NULL) {
- return devices->GetDevice(0)->GetDxCaps();
- } else {
- int devicesNum = devices->GetNumDevices();
- for (int i = 0; i < devicesNum; ++i) {
- if (devices->GetDevice(i)->GetMonitor() == hMonitor) {
- return devices->GetDevice(i)->GetDxCaps();
- }
- }
- }
- return (DxCapabilities*)NULL;
-}
-
MHND AwtWin32GraphicsDevice::GetMonitor(int deviceIndex)
{
Devices::InstanceAccess devices;
@@ -860,8 +830,6 @@ Java_sun_awt_Win32GraphicsDevice_initIDs(JNIEnv *env, jclass cls)
AwtWin32GraphicsDevice::indexCMcacheID =
env->GetFieldID(AwtWin32GraphicsDevice::indexCMClass,
"lookupcache", "[I");
- AwtWin32GraphicsDevice::accelerationEnabledID =
- env->GetFieldID(cls, "offscreenAccelerationEnabled", "Z");
/* method ids */
AwtWin32GraphicsDevice::paletteChangedMID = env->GetStaticMethodID(
@@ -871,7 +839,6 @@ Java_sun_awt_Win32GraphicsDevice_initIDs(JNIEnv *env, jclass cls)
DASSERT(AwtWin32GraphicsDevice::dynamicColorModelID);
DASSERT(AwtWin32GraphicsDevice::indexCMrgbID);
DASSERT(AwtWin32GraphicsDevice::indexCMcacheID);
- DASSERT(AwtWin32GraphicsDevice::accelerationEnabledID);
DASSERT(AwtWin32GraphicsDevice::paletteChangedMID);
@@ -1012,7 +979,7 @@ JNIEXPORT jint JNICALL Java_sun_awt_Win32GraphicsDevice_getDefaultPixIDImpl
JNIEXPORT void JNICALL
Java_sun_awt_Win32GraphicsDevice_enterFullScreenExclusive(
JNIEnv* env, jobject graphicsDevice,
- jboolean useDD, jint screen, jobject windowPeer) {
+ jint screen, jobject windowPeer) {
TRY;
@@ -1023,39 +990,12 @@ Java_sun_awt_Win32GraphicsDevice_enterFullScreenExclusive(
// with the WWindowPeer object
HWND hWnd = window->GetHWnd();
- if (useDD) {
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- /**
- * We call SendMessage with a timeout of 1000 ms because we would like
- * this call to be synchronous, but we want to avoid any possibility
- * of deadlocking here.
- */
- DWORD_PTR eventResult;
- LRESULT sendResult;
- sendResult = ::SendMessageTimeout(hWnd, WM_AWT_DD_ENTER_FULLSCREEN,
- (WPARAM)monitor, NULL,
- SMTO_NORMAL, 1000, &eventResult);
- if (sendResult == 0) {
- // Not the end of the world, but we would like to know about
- // it and fix the problem/deadlock if one exists
- int error = GetLastError();
- if (error == 0) {
- J2dTraceLn(J2D_TRACE_ERROR,
- "SendMessage(ENTER_FULLSCREEN) timed out");
- } else {
- J2dTraceLn1(J2D_TRACE_ERROR,
- "SendMessage(ENTER_FULLSCREEN) failed with error %d",
- error);
- }
- }
- } else {
- if (!::SetWindowPos(hWnd, HWND_TOPMOST, 0, 0, 0, 0,
- SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOSIZE))
- {
- J2dTraceLn1(J2D_TRACE_ERROR,
- "Error %d setting topmost attribute to fs window",
- ::GetLastError());
- }
+ if (!::SetWindowPos(hWnd, HWND_TOPMOST, 0, 0, 0, 0,
+ SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOSIZE))
+ {
+ J2dTraceLn1(J2D_TRACE_ERROR,
+ "Error %d setting topmost attribute to fs window",
+ ::GetLastError());
}
CATCH_BAD_ALLOC;
@@ -1070,7 +1010,7 @@ Java_sun_awt_Win32GraphicsDevice_enterFullScreenExclusive(
JNIEXPORT void JNICALL
Java_sun_awt_Win32GraphicsDevice_exitFullScreenExclusive(
JNIEnv* env, jobject graphicsDevice,
- jboolean useDD, jint screen, jobject windowPeer) {
+ jint screen, jobject windowPeer) {
TRY;
@@ -1081,45 +1021,18 @@ Java_sun_awt_Win32GraphicsDevice_exitFullScreenExclusive(
// with the WWindowPeer object
HWND hWnd = window->GetHWnd();
- if (useDD) {
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- /**
- * We call SendMessage with a timeout of 1000 ms because we would like
- * this call to be synchronous, but we want to avoid any possibility
- * of deadlocking here.
- */
- DWORD_PTR eventResult;
- LRESULT sendResult;
- sendResult = ::SendMessageTimeout(hWnd, WM_AWT_DD_EXIT_FULLSCREEN,
- (WPARAM)monitor, NULL,
- SMTO_NORMAL, 1000, &eventResult);
- if (sendResult == 0) {
- // Not the end of the world, but we would like to know about
- // it and fix the problem/deadlock if one exists
- int error = GetLastError();
- if (error == 0) {
- J2dTraceLn(J2D_TRACE_ERROR,
- "SendMessage(EXIT_FULLSCREEN) timed out");
- } else {
- J2dTraceLn1(J2D_TRACE_ERROR,
- "SendMessage(EXIT_FULLSCREEN) failed with error %d",
- error);
- }
- }
- } else {
- if (!::SetWindowPos(hWnd, HWND_NOTOPMOST, 0, 0, 0, 0,
- SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOSIZE))
- {
- J2dTraceLn1(J2D_TRACE_ERROR,
- "Error %d unsetting topmost attribute to fs window",
- ::GetLastError());
- }
+ if (!::SetWindowPos(hWnd, HWND_NOTOPMOST, 0, 0, 0, 0,
+ SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOSIZE))
+ {
+ J2dTraceLn1(J2D_TRACE_ERROR,
+ "Error %d unsetting topmost attribute to fs window",
+ ::GetLastError());
}
CATCH_BAD_ALLOC;
}
-static jobject CreateDisplayMode(JNIEnv* env, jint width, jint height,
+jobject CreateDisplayMode(JNIEnv* env, jint width, jint height,
jint bitDepth, jint refreshRate) {
TRY;
@@ -1146,24 +1059,6 @@ static jobject CreateDisplayMode(JNIEnv* env, jint width, jint height,
CATCH_BAD_ALLOC_RET(NULL);
}
-/*
- * Class: sun_awt_Win32GraphicsDevice
- * Method: isDDEnabledOnDeviceNative
- * Signature: (I)Z
- */
-JNIEXPORT jboolean JNICALL
-Java_sun_awt_Win32GraphicsDevice_isDDEnabledOnDeviceNative
- (JNIEnv* env, jobject graphicsDevice, jint screen)
-{
- TRY;
-
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- return DeviceUseDDraw(monitor);
-
- CATCH_BAD_ALLOC_RET(FALSE);
-}
-
-
/**
* A utility function which retrieves a DISPLAY_DEVICE information
* given a screen number.
@@ -1206,37 +1101,23 @@ Java_sun_awt_Win32GraphicsDevice_getCurrentDisplayMode
{
TRY;
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- if (DeviceUseDDraw(monitor)) {
- DDrawDisplayMode dm;
+ DEVMODE dm;
+ LPTSTR pName = NULL;
- if (!DDGetDisplayMode(monitor, dm)) {
- JNU_ThrowInternalError(env,
- "Could not get display mode");
- return NULL;
- }
+ dm.dmSize = sizeof(dm);
+ dm.dmDriverExtra = 0;
- return CreateDisplayMode(env, dm.width,
- dm.height, dm.bitDepth, dm.refreshRate);
- } else {
- DEVMODE dm;
- LPTSTR pName = NULL;
-
- dm.dmSize = sizeof(dm);
- dm.dmDriverExtra = 0;
-
- _DISPLAY_DEVICE displayDevice;
- if (GetAttachedDisplayDevice(screen, &displayDevice)) {
- pName = displayDevice.strDevName;
- }
- if (!EnumDisplaySettings(pName, ENUM_CURRENT_SETTINGS, &dm))
- {
- return NULL;
- }
-
- return CreateDisplayMode(env, dm.dmPelsWidth,
- dm.dmPelsHeight, dm.dmBitsPerPel, dm.dmDisplayFrequency);
+ _DISPLAY_DEVICE displayDevice;
+ if (GetAttachedDisplayDevice(screen, &displayDevice)) {
+ pName = displayDevice.strDevName;
}
+ if (!EnumDisplaySettings(pName, ENUM_CURRENT_SETTINGS, &dm))
+ {
+ return NULL;
+ }
+
+ return CreateDisplayMode(env, dm.dmPelsWidth,
+ dm.dmPelsHeight, dm.dmBitsPerPel, dm.dmDisplayFrequency);
CATCH_BAD_ALLOC_RET(NULL);
}
@@ -1253,96 +1134,43 @@ Java_sun_awt_Win32GraphicsDevice_configDisplayMode
{
TRY;
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- if (DeviceUseDDraw(monitor)) {
- PDATA pData;
-
- JNI_CHECK_PEER_RETURN(windowPeer);
-
- AwtWindow * window = (AwtWindow * )pData; // safe cast; we are called
- // with the WWindowPeer object
- HWND hWnd = window->GetHWnd();
-
- DDrawDisplayMode *dm = new DDrawDisplayMode(width, height, bitDepth,
- refreshRate);
-
- /**
- * We call SendMessage with a timeout of 1000 ms because we would like
- * this call to be synchronous, but we want to avoid any possibility
- * of deadlocking here.
- * Also, note that we have to free the dm object in the message
- * processing code since we may return from the SendMessageTimeout
- * before that object has been used.
- */
- DWORD_PTR eventResult;
- LRESULT sendResult;
- sendResult = ::SendMessageTimeout(hWnd, WM_AWT_DD_SET_DISPLAY_MODE,
- (WPARAM)monitor, (LPARAM)dm,
- SMTO_NORMAL, 1000, &eventResult);
- if (sendResult == 0) {
- // Not the end of the world, but we would like to know about
- // it and fix the problem/deadlock if one exists
- int error = GetLastError();
- if (error == 0) {
- J2dTraceLn(J2D_TRACE_ERROR,
- "SendMessage(SET_DISPLAY_MODE) timed out");
- } else {
- J2dTraceLn1(J2D_TRACE_ERROR,
- "SendMessage(SET_DISPLAY_MODE) failed with error %d",
- error);
- }
- } else {
- // see comment in awt_Window.cpp WmDDEnterFullScreenMode
- // Since the event to reshape the window will come to the owner
- // frame, we need to reshape the window by ourselves.
- // Note that we do not need to reshape it on the exit
- // from full screen mode, this is already handled in
- // GraphicsDevice.setFullScreenWindow .
- if (window->GetOwningFrameOrDialog() != NULL) {
- RECT* r = new RECT;
- ::SetRect(r, 0, 0, width, height);
- window->SendMessage(WM_AWT_RESHAPE_COMPONENT, 0, (LPARAM)r);
- }
- }
- } else {
DEVMODE dm;
- dm.dmSize = sizeof(dm);
- dm.dmDriverExtra = 0;
- dm.dmPelsWidth = width;
- dm.dmPelsHeight = height;
- dm.dmBitsPerPel = bitDepth;
- dm.dmDisplayFrequency = refreshRate;
- dm.dmFields = DM_PELSWIDTH | DM_PELSHEIGHT |
- DM_BITSPERPEL | DM_DISPLAYFREQUENCY;
+ dm.dmSize = sizeof(dm);
+ dm.dmDriverExtra = 0;
+ dm.dmPelsWidth = width;
+ dm.dmPelsHeight = height;
+ dm.dmBitsPerPel = bitDepth;
+ dm.dmDisplayFrequency = refreshRate;
+ dm.dmFields = DM_PELSWIDTH | DM_PELSHEIGHT |
+ DM_BITSPERPEL | DM_DISPLAYFREQUENCY;
- // ChangeDisplaySettings works only on the primary screen.
- // ChangeDisplaySettingsEx is not available on NT,
- // so it'd be nice not to break it if we can help it.
- if (screen == AwtWin32GraphicsDevice::GetDefaultDeviceIndex()) {
- if (ChangeDisplaySettings(&dm, CDS_FULLSCREEN) !=
- DISP_CHANGE_SUCCESSFUL)
- {
- JNU_ThrowInternalError(env,
- "Could not set display mode");
- }
- return;
- }
-
- // make sure the function pointer for fn_change_display_settings_ex
- // is initialized
- load_user_procs();
-
- _DISPLAY_DEVICE displayDevice;
- if (fn_change_display_settings_ex == NULL ||
- !GetAttachedDisplayDevice(screen, &displayDevice) ||
- ((*fn_change_display_settings_ex)
- (displayDevice.strDevName, &dm, NULL, CDS_FULLSCREEN, NULL) !=
- DISP_CHANGE_SUCCESSFUL))
+ // ChangeDisplaySettings works only on the primary screen.
+ // ChangeDisplaySettingsEx is not available on NT,
+ // so it'd be nice not to break it if we can help it.
+ if (screen == AwtWin32GraphicsDevice::GetDefaultDeviceIndex()) {
+ if (ChangeDisplaySettings(&dm, CDS_FULLSCREEN) !=
+ DISP_CHANGE_SUCCESSFUL)
{
JNU_ThrowInternalError(env,
"Could not set display mode");
}
+ return;
+ }
+
+ // make sure the function pointer for fn_change_display_settings_ex
+ // is initialized
+ load_user_procs();
+
+ _DISPLAY_DEVICE displayDevice;
+ if (fn_change_display_settings_ex == NULL ||
+ !GetAttachedDisplayDevice(screen, &displayDevice) ||
+ ((*fn_change_display_settings_ex)
+ (displayDevice.strDevName, &dm, NULL, CDS_FULLSCREEN, NULL) !=
+ DISP_CHANGE_SUCCESSFUL))
+ {
+ JNU_ThrowInternalError(env,
+ "Could not set display mode");
}
CATCH_BAD_ALLOC;
@@ -1355,7 +1183,7 @@ public:
jobject arrayList;
};
-static void addDisplayMode(JNIEnv* env, jobject arrayList, jint width,
+void addDisplayMode(JNIEnv* env, jobject arrayList, jint width,
jint height, jint bitDepth, jint refreshRate) {
TRY;
@@ -1383,22 +1211,6 @@ static void addDisplayMode(JNIEnv* env, jobject arrayList, jint width,
CATCH_BAD_ALLOC;
}
-static void enumDMCallback(DDrawDisplayMode& dm, void* pContext) {
-
- TRY;
-
- EnumDisplayModeParam* pParam = (EnumDisplayModeParam*)pContext;
- JNIEnv* env = pParam->env;
- jobject arrayList = pParam->arrayList;
-
- if (dm.bitDepth >= 8) {
- addDisplayMode(env, arrayList, dm.width, dm.height,
- dm.bitDepth, dm.refreshRate);
- }
-
- CATCH_BAD_ALLOC;
-}
-
/*
* Class: sun_awt_Win32GraphicsDevice
* Method: enumDisplayModes
@@ -1410,72 +1222,30 @@ JNIEXPORT void JNICALL Java_sun_awt_Win32GraphicsDevice_enumDisplayModes
TRY;
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
- if (DeviceUseDDraw(monitor)) {
- EnumDisplayModeParam param(env, arrayList);
-
- if (!DDEnumDisplayModes(monitor, NULL, enumDMCallback, ¶m)) {
- JNU_ThrowInternalError(env,
- "Could not get display modes");
- return;
- }
- } else {
- DEVMODE dm;
- LPTSTR pName = NULL;
- _DISPLAY_DEVICE displayDevice;
+ DEVMODE dm;
+ LPTSTR pName = NULL;
+ _DISPLAY_DEVICE displayDevice;
- if (GetAttachedDisplayDevice(screen, &displayDevice)) {
- pName = displayDevice.strDevName;
- }
+ if (GetAttachedDisplayDevice(screen, &displayDevice)) {
+ pName = displayDevice.strDevName;
+ }
- dm.dmSize = sizeof(dm);
- dm.dmDriverExtra = 0;
+ dm.dmSize = sizeof(dm);
+ dm.dmDriverExtra = 0;
- BOOL bContinue = TRUE;
- for (int i = 0; bContinue; i++) {
- bContinue = EnumDisplaySettings(pName, i, &dm);
- if (dm.dmBitsPerPel >= 8) {
- addDisplayMode(env, arrayList, dm.dmPelsWidth, dm.dmPelsHeight,
- dm.dmBitsPerPel, dm.dmDisplayFrequency);
- }
+ BOOL bContinue = TRUE;
+ for (int i = 0; bContinue; i++) {
+ bContinue = EnumDisplaySettings(pName, i, &dm);
+ if (dm.dmBitsPerPel >= 8) {
+ addDisplayMode(env, arrayList, dm.dmPelsWidth, dm.dmPelsHeight,
+ dm.dmBitsPerPel, dm.dmDisplayFrequency);
}
}
CATCH_BAD_ALLOC;
}
-void checkDMCallback(DDrawDisplayMode& dm, void* pContext) {
- jboolean* isCalled = (jboolean*)pContext;
- *isCalled = JNI_TRUE;
-}
-
-/*
- * Class: sun_awt_Win32GraphicsDevice
- * Method: isDisplayModeAvailable
- * Signature: (IIII)Z
- */
-JNIEXPORT jboolean JNICALL Java_sun_awt_Win32GraphicsDevice_isDisplayModeAvailable
- (JNIEnv* env, jobject graphicsDevice, jint screen, jint width, jint height,
- jint bitDepth, jint refreshRate) {
-
- TRY;
-
- DDrawDisplayMode dm(width, height, bitDepth, refreshRate);
- jboolean isCalled = JNI_FALSE;
- HMONITOR monitor = (HMONITOR)AwtWin32GraphicsDevice::GetMonitor(screen);
-
- if (!DDEnumDisplayModes(monitor, &dm, checkDMCallback, &isCalled)) {
- JNU_ThrowInternalError(env,
- "Could not get display modes");
- return JNI_FALSE;
- }
-
- return isCalled;
-
- CATCH_BAD_ALLOC_RET(JNI_FALSE);
-}
-
/*
* Class: sun_awt_Win32GraphicsDevice
* Method: makeColorModel
@@ -1502,17 +1272,3 @@ JNIEXPORT void JNICALL
Devices::InstanceAccess devices;
devices->GetDevice(screen)->SetJavaDevice(env, thisPtr);
}
-
-/*
- * Class: sun_awt_Win32GraphicsDevice
- * Method: getDeviceMemoryNative
- * Signature: (I)I
- */
-JNIEXPORT jint JNICALL
- Java_sun_awt_Win32GraphicsDevice_getDeviceMemoryNative
- (JNIEnv *env, jobject thisPtr, jint screen)
-{
- Devices::InstanceAccess devices;
- return DDGetAvailableMemory(
- (HMONITOR)devices->GetDevice(screen)->GetMonitor());
-}
diff --git a/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.h b/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.h
index 152dc9f5b34..9d3978ba324 100644
--- a/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.h
+++ b/jdk/src/windows/native/sun/windows/awt_Win32GraphicsDevice.h
@@ -1,5 +1,5 @@
/*
- * Copyright 2001-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 2001-2008 Sun Microsystems, Inc. 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
@@ -34,7 +34,6 @@ extern "C" {
#include "awt_Palette.h"
#include "awt_MMStub.h"
#include "Devices.h"
-#include "dxCapabilities.h"
class AwtPalette;
class Devices;
@@ -63,7 +62,6 @@ public:
int GetBitDepth() { return colorData->bitsperpixel; }
MHND GetMonitor() { return monitor; }
MONITOR_INFO *GetMonitorInfo() { return pMonitorInfo; }
- DxCapabilities *GetDxCaps() { return &dxCaps; }
jobject GetJavaDevice() { return javaDevice; }
int GetDeviceIndex() { return screen; }
void Release();
@@ -86,7 +84,6 @@ public:
static BOOL IsPrimaryPalettized() { return primaryPalettized; }
static int GetDefaultDeviceIndex() { return primaryIndex; }
static void DisableOffscreenAccelerationForDevice(MHND hMonitor);
- static DxCapabilities *GetDxCapsForDevice(MHND hMonitor);
static HDC GetDCFromScreen(int screen);
static int GetScreenFromMHND(MHND mon);
@@ -97,7 +94,6 @@ public:
static jfieldID dynamicColorModelID;
static jfieldID indexCMrgbID;
static jfieldID indexCMcacheID;
- static jfieldID accelerationEnabledID;
static jmethodID paletteChangedMID;
private:
@@ -112,7 +108,6 @@ private:
MONITOR_INFO *pMonitorInfo;
jobject javaDevice;
Devices *devicesArray;
- DxCapabilities dxCaps;
};
#endif AWT_WIN32GRAPHICSDEVICE_H
diff --git a/jdk/src/windows/native/sun/windows/awt_Win32GraphicsEnv.cpp b/jdk/src/windows/native/sun/windows/awt_Win32GraphicsEnv.cpp
index 89e488c3324..398e17d27d8 100644
--- a/jdk/src/windows/native/sun/windows/awt_Win32GraphicsEnv.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_Win32GraphicsEnv.cpp
@@ -1,5 +1,5 @@
/*
- * Copyright 1999-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1999-2008 Sun Microsystems, Inc. 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
@@ -31,7 +31,8 @@
#include "awt_Win32GraphicsDevice.h"
#include "Devices.h"
#include "WindowsFlags.h"
-#include "dxInit.h"
+
+BOOL DWMIsCompositionEnabled();
void initScreens(JNIEnv *env) {
@@ -39,8 +40,6 @@ void initScreens(JNIEnv *env) {
JNU_ThrowInternalError(env, "Could not update the devices array.");
return;
}
-
- InitDirectX();
}
/**
@@ -54,7 +53,7 @@ void initScreens(JNIEnv *env) {
static void
SetProcessDPIAwareProperty()
{
- typedef BOOL SetProcessDPIAwareFunc(void);
+ typedef BOOL (WINAPI SetProcessDPIAwareFunc)(void);
static BOOL bAlreadySet = FALSE;
// setHighDPIAware is set in WindowsFlags.cpp
@@ -64,7 +63,7 @@ SetProcessDPIAwareProperty()
bAlreadySet = TRUE;
- HINSTANCE hLibUser32Dll = ::LoadLibrary(TEXT("user32.dll"));
+ HMODULE hLibUser32Dll = ::LoadLibrary(TEXT("user32.dll"));
if (hLibUser32Dll != NULL) {
SetProcessDPIAwareFunc *lpSetProcessDPIAware =
@@ -77,6 +76,76 @@ SetProcessDPIAwareProperty()
}
}
+#define DWM_COMP_UNDEFINED (~(TRUE|FALSE))
+static int dwmIsCompositionEnabled = DWM_COMP_UNDEFINED;
+
+/**
+ * This function is called from toolkit event handling code when
+ * WM_DWMCOMPOSITIONCHANGED event is received
+ */
+void DWMResetCompositionEnabled() {
+ dwmIsCompositionEnabled = DWM_COMP_UNDEFINED;
+ (void)DWMIsCompositionEnabled();
+}
+
+/**
+ * Returns true if dwm composition is enabled, false if it is not applicable
+ * (if the OS is not Vista) or dwm composition is disabled.
+ *
+ * Note: since DWM composition state changes are very rare we load/unload the
+ * dll on every change.
+ */
+BOOL DWMIsCompositionEnabled() {
+ typedef HRESULT (WINAPI DwmIsCompositionEnabledFunc)(BOOL*);
+
+ // cheaper to check than whether it's vista or not
+ if (dwmIsCompositionEnabled != DWM_COMP_UNDEFINED) {
+ return (BOOL)dwmIsCompositionEnabled;
+ }
+
+ if (!IS_WINVISTA) {
+ dwmIsCompositionEnabled = FALSE;
+ return FALSE;
+ }
+
+ BOOL bRes = FALSE;
+ HMODULE hDwmApiDll = ::LoadLibrary(TEXT("dwmapi.dll"));
+
+ if (hDwmApiDll != NULL) {
+ DwmIsCompositionEnabledFunc *lpDwmIsCompEnabled =
+ (DwmIsCompositionEnabledFunc*)
+ GetProcAddress(hDwmApiDll, "DwmIsCompositionEnabled");
+ if (lpDwmIsCompEnabled != NULL) {
+ BOOL bEnabled;
+ HRESULT res = lpDwmIsCompEnabled(&bEnabled);
+ if (SUCCEEDED(res)) {
+ bRes = bEnabled;
+ J2dTraceLn1(J2D_TRACE_VERBOSE, " composition enabled: %d",bRes);
+ } else {
+ J2dTraceLn1(J2D_TRACE_ERROR,
+ "IsDWMCompositionEnabled: error %x when detecting"\
+ "if composition is enabled", res);
+ }
+ } else {
+ J2dTraceLn(J2D_TRACE_ERROR,
+ "IsDWMCompositionEnabled: no DwmIsCompositionEnabled() "\
+ "in dwmapi.dll");
+ }
+ ::FreeLibrary(hDwmApiDll);
+ } else {
+ J2dTraceLn(J2D_TRACE_ERROR,
+ "IsDWMCompositionEnabled: error opening dwmapi.dll");
+ }
+
+ dwmIsCompositionEnabled = bRes;
+
+ JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
+ JNU_CallStaticMethodByName(env, NULL,
+ "sun/awt/Win32GraphicsEnvironment",
+ "dwmCompositionChanged", "(Z)V", (jboolean)bRes);
+ return bRes;
+}
+
/*
* Class: sun_awt_Win32GraphicsEnvironment
* Method: initDisplay
@@ -89,6 +158,8 @@ Java_sun_awt_Win32GraphicsEnvironment_initDisplay(JNIEnv *env,
// This method needs to be called prior to any display-related activity
SetProcessDPIAwareProperty();
+ DWMIsCompositionEnabled();
+
initScreens(env);
}
@@ -306,3 +377,14 @@ Java_sun_awt_Win32GraphicsEnvironment_getYResolution(JNIEnv *env, jobject wge)
CATCH_BAD_ALLOC_RET(0);
}
+
+/*
+ * Class: sun_awt_Win32GraphicsEnvironment
+ * Method: isVistaOS
+ * Signature: ()Z
+ */
+JNIEXPORT jboolean JNICALL Java_sun_awt_Win32GraphicsEnvironment_isVistaOS
+ (JNIEnv *env, jclass wgeclass)
+{
+ return IS_WINVISTA;
+}
diff --git a/jdk/src/windows/native/sun/windows/awt_Window.cpp b/jdk/src/windows/native/sun/windows/awt_Window.cpp
index 2d94c779309..13b41863235 100644
--- a/jdk/src/windows/native/sun/windows/awt_Window.cpp
+++ b/jdk/src/windows/native/sun/windows/awt_Window.cpp
@@ -33,7 +33,6 @@
#include "awt_Toolkit.h"
#include "awt_Window.h"
#include "awt_dlls.h"
-#include "ddrawUtils.h"
#include "awt_Win32GraphicsDevice.h"
#include "awt_BitmapUtil.h"
#include "awt_IconCursor.h"
@@ -742,7 +741,7 @@ BOOL AwtWindow::UpdateInsets(jobject insets)
insetsChanged = !::EqualRect( &m_old_insets, &m_insets );
::CopyRect( &m_old_insets, &m_insets );
- if (insetsChanged && DDCanReplaceSurfaces(GetHWnd())) {
+ if (insetsChanged) {
// Since insets are changed we need to update the surfaceData object
// to reflect that change
env->CallVoidMethod(peer, AwtComponent::replaceSurfaceDataLaterMID);
@@ -931,50 +930,6 @@ void AwtWindow::BounceActivation(void *self) {
}
}
-MsgRouting AwtWindow::WmDDEnterFullScreen(HMONITOR monitor) {
- /**
- * DirectDraw expects to receive a top-level window. This object may
- * be an AwtWindow instance, which has an owning AwtFrame window, or
- * an AwtFrame object which does not have an owner.
- * What we want is the top-level Frame hWnd, whether we were handed a
- * top-level AwtFrame, or some owned AwtWindow. We get this by calling
- * GetTopLevelHWnd(), which returns the hwnd of a top-level AwtFrame
- * object (if this window has an owner) which we then pass
- * into DirectDraw.
- */
- HWND hWnd = GetTopLevelHWnd();
- if (!::IsWindowVisible(hWnd)) {
- // Sometimes there are problems going into fullscreen on an owner frame
- // that is not yet visible; make sure the FS window is visible first
- ::ShowWindow(hWnd, SW_SHOWNA);
- }
- /*
- * Fix for 6225472.
- * Non-focusable window should be set alwaysOnTop to overlap the taskbar.
- */
- AwtWindow* window = (AwtWindow *)AwtComponent::GetComponent(GetHWnd());
- if (window != NULL && !window->IsFocusableWindow()) {
- JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
- Java_sun_awt_windows_WWindowPeer_setAlwaysOnTopNative(env, GetPeer(env), (jboolean)TRUE);
- }
-
- DDEnterFullScreen(monitor, GetHWnd(), hWnd);
- return mrDoDefault;
-}
-
-MsgRouting AwtWindow::WmDDExitFullScreen(HMONITOR monitor) {
- HWND hWnd = GetTopLevelHWnd();
- DDExitFullScreen(monitor, hWnd);
-
- JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
- jboolean alwaysOnTop = JNU_CallMethodByName(env, NULL, GetTarget(env),
- "isAlwaysOnTop", "()Z").z;
-
- // We should restore alwaysOnTop state as it's anyway dropped here.
- Java_sun_awt_windows_WWindowPeer_setAlwaysOnTopNative(env, GetPeer(env), alwaysOnTop);
- return mrDoDefault;
-}
-
MsgRouting AwtWindow::WmCreate()
{
return mrDoDefault;
@@ -1333,10 +1288,8 @@ void AwtWindow::WindowResized()
SendComponentEvent(java_awt_event_ComponentEvent_COMPONENT_RESIZED);
// Need to replace surfaceData on resize to catch changes to
// various component-related values, such as insets
- if (DDCanReplaceSurfaces(GetHWnd())) {
- JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
- env->CallVoidMethod(m_peerObject, AwtComponent::replaceSurfaceDataLaterMID);
- }
+ JNIEnv *env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
+ env->CallVoidMethod(m_peerObject, AwtComponent::replaceSurfaceDataLaterMID);
}
BOOL CALLBACK InvalidateChildRect(HWND hWnd, LPARAM)
diff --git a/jdk/src/windows/native/sun/windows/awt_Window.h b/jdk/src/windows/native/sun/windows/awt_Window.h
index bc706d97aeb..9a252a25ce1 100644
--- a/jdk/src/windows/native/sun/windows/awt_Window.h
+++ b/jdk/src/windows/native/sun/windows/awt_Window.h
@@ -161,8 +161,6 @@ public:
virtual MsgRouting WmClose();
virtual MsgRouting WmDestroy();
virtual MsgRouting WmShowWindow(BOOL show, UINT status);
- virtual MsgRouting WmDDEnterFullScreen(HMONITOR monitor);
- virtual MsgRouting WmDDExitFullScreen(HMONITOR monitor);
virtual MsgRouting WmGetMinMaxInfo(LPMINMAXINFO lpmmi);
virtual MsgRouting WmMove(int x, int y);
virtual MsgRouting WmSize(UINT type, int w, int h);
diff --git a/jdk/src/windows/native/sun/windows/awtmsg.h b/jdk/src/windows/native/sun/windows/awtmsg.h
index 6ac2746c2d5..fe03b0c460c 100644
--- a/jdk/src/windows/native/sun/windows/awtmsg.h
+++ b/jdk/src/windows/native/sun/windows/awtmsg.h
@@ -1,5 +1,5 @@
/*
- * Copyright 1996-2006 Sun Microsystems, Inc. All Rights Reserved.
+ * Copyright 1996-2008 Sun Microsystems, Inc. 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
@@ -252,17 +252,6 @@ enum {
WM_AWT_HIDECURSOR,
WM_AWT_CREATE_PRINTED_PIXELS,
- /* Synchronize ddraw fullscreen events on Windows event thread */
- WM_AWT_DD_CREATE_SURFACE,
- WM_AWT_DD_ENTER_FULLSCREEN,
- WM_AWT_DD_EXIT_FULLSCREEN,
- WM_AWT_DD_SET_DISPLAY_MODE,
- WM_AWT_DD_RESTORE_DISPLAY_MODE,
-
- /* Synchronize D3D device creation on Windows event thread */
- WM_AWT_D3D_CREATE_DEVICE,
- WM_AWT_D3D_RELEASE_DEVICE,
-
// Tray messages
WM_AWT_TRAY_NOTIFY,
diff --git a/jdk/test/java/awt/FullScreen/BufferStrategyExceptionTest/BufferStrategyExceptionTest.java b/jdk/test/java/awt/FullScreen/BufferStrategyExceptionTest/BufferStrategyExceptionTest.java
new file mode 100644
index 00000000000..b50cf59390c
--- /dev/null
+++ b/jdk/test/java/awt/FullScreen/BufferStrategyExceptionTest/BufferStrategyExceptionTest.java
@@ -0,0 +1,134 @@
+/*
+ * Copyright 2006-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/**
+ * @test
+ * @bug 6366813 6459844
+ * @summary Tests that no exception is thrown if a frame is resized just
+ * before we create a bufferStrategy
+ * @author Dmitri.Trembovetski area=FullScreen/BufferStrategy
+ * @run main/othervm -Dsun.java2d.opengl=true BufferStrategyExceptionTest
+ * @run main/othervm BufferStrategyExceptionTest
+ */
+
+import java.awt.AWTException;
+import java.awt.BufferCapabilities;
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
+import java.awt.ImageCapabilities;
+import java.awt.image.BufferStrategy;
+import java.awt.image.BufferedImage;
+
+/**
+ * The purpose of this test is to make sure that we do not throw an
+ * IllegalStateException during the creation of BufferStrategy if
+ * a window has been resized just before our creation attempt.
+ *
+ * We test both windowed and fullscreen mode, although the exception has
+ * been observed in full screen mode only.
+ */
+public class BufferStrategyExceptionTest {
+ private static final int TEST_REPS = 20;
+
+ public static void main(String[] args) {
+ GraphicsDevice gd =
+ GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getDefaultScreenDevice();
+
+ for (int i = 0; i < TEST_REPS; i++) {
+ TestFrame f = new TestFrame();
+ f.pack();
+ f.setSize(400, 400);
+ f.setVisible(true);
+ if (i % 2 == 0) {
+ gd.setFullScreenWindow(f);
+ }
+ // generate a resize event which will invalidate the peer's
+ // surface data and hopefully cause an exception during
+ // BufferStrategy creation in TestFrame.render()
+ Dimension d = f.getSize();
+ d.width -= 5; d.height -= 5;
+ f.setSize(d);
+
+ f.render();
+ gd.setFullScreenWindow(null);
+ sleep(100);
+ f.dispose();
+ }
+ System.out.println("Test passed.");
+ }
+
+ private static void sleep(long msecs) {
+ try {
+ Thread.sleep(msecs);
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ }
+
+ private static final BufferedImage bi =
+ new BufferedImage(200, 200, BufferedImage.TYPE_INT_RGB);
+
+ static class TestFrame extends Frame {
+ TestFrame() {
+ setUndecorated(true);
+ setIgnoreRepaint(true);
+ setSize(400, 400);
+ }
+
+ public void render() {
+ ImageCapabilities imgBackBufCap = new ImageCapabilities(true);
+ ImageCapabilities imgFrontBufCap = new ImageCapabilities(true);
+ BufferCapabilities bufCap =
+ new BufferCapabilities(imgFrontBufCap,
+ imgBackBufCap, BufferCapabilities.FlipContents.COPIED);
+ try {
+
+ createBufferStrategy(2, bufCap);
+ } catch (AWTException ex) {
+ createBufferStrategy(2);
+ }
+
+ BufferStrategy bs = getBufferStrategy();
+ do {
+ Graphics g = bs.getDrawGraphics();
+ g.setColor(Color.green);
+ g.fillRect(0, 0, getWidth(), getHeight());
+
+ g.setColor(Color.red);
+ g.drawString("Rendering test", 20, 20);
+
+ g.drawImage(bi, 50, 50, null);
+
+ g.dispose();
+ bs.show();
+ } while (bs.contentsLost()||bs.contentsRestored());
+ }
+ }
+
+}
diff --git a/jdk/test/java/awt/FullScreen/MultimonFullscreenTest/MultimonFullscreenTest.java b/jdk/test/java/awt/FullScreen/MultimonFullscreenTest/MultimonFullscreenTest.java
new file mode 100644
index 00000000000..7e615068c62
--- /dev/null
+++ b/jdk/test/java/awt/FullScreen/MultimonFullscreenTest/MultimonFullscreenTest.java
@@ -0,0 +1,387 @@
+/*
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/**
+ * @test
+ * @bug 5041219
+ * @bug 5101561
+ * @bug 5035272
+ * @bug 5096011
+ * @bug 5101712
+ * @bug 5098624
+ * @summary Here are a few assertions worth verification:
+ * - the fullscreen window is positioned at 0,0
+ * - the fs window appears on the correct screen
+ * - if the exclusive FS mode is supported, no other widndow should
+ * overlap the fs window (including the taskbar).
+ * You could, however, alt+tab out of a fullscreen window, or at least
+ * minimize it (if you've entered the fs mode with a Window, you'll need
+ * to minimize the owner frame).
+ * Note that there may be issues with FS exclusive mode with ddraw and
+ * multiple fullscreen windows (one per device).
+ * - if display mode is supported that it did change
+ * - that the original display mode is restored once
+ * the ws window is disposed
+ * All of the above should work with and w/o DirectDraw
+ * (-Dsun.java2d.noddraw=true) on windows, and w/ and w/o opengl on X11
+ * (-Dsun.java2d.opengl=True).
+ * @run main/manual/othervm -Dsun.java2d.pmoffscreen=true MultimonFullscreenTest
+ * @run main/manual/othervm -Dsun.java2d.pmoffscreen=false MultimonFullscreenTest
+ * @run main/manual/othervm -Dsun.java2d.d3d=True MultimonFullscreenTest
+ * @run main/manual/othervm -Dsun.java2d.noddraw=true MultimonFullscreenTest
+ * @run main/manual/othervm -Dsun.java2d.opengl=True MultimonFullscreenTest
+ */
+
+import java.awt.Button;
+import java.awt.Checkbox;
+import java.awt.CheckboxGroup;
+import java.awt.Color;
+import java.awt.Component;
+import java.awt.Dialog;
+import java.awt.DisplayMode;
+import java.awt.Font;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
+import java.awt.GridLayout;
+import java.awt.Panel;
+import java.awt.Rectangle;
+import java.awt.Window;
+import java.awt.event.ActionEvent;
+import java.awt.event.ActionListener;
+import java.awt.event.ItemEvent;
+import java.awt.event.ItemListener;
+import java.awt.event.MouseAdapter;
+import java.awt.event.MouseEvent;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.image.BufferStrategy;
+import java.util.HashMap;
+import java.util.Random;
+
+/**
+ */
+
+public class MultimonFullscreenTest extends Frame implements ActionListener {
+ GraphicsDevice defDev = GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getDefaultScreenDevice();
+ GraphicsDevice gd[] = GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getScreenDevices();
+ HashMap deviceMap;
+
+ private static boolean dmChange = false;
+ static boolean setNullOnDispose = false;
+ static boolean useFSFrame = true;
+ static boolean useFSWindow = false;
+ static boolean useFSDialog = false;
+ static boolean useBS = false;
+ static boolean runRenderLoop = false;
+ static boolean addHWChildren = false;
+ static volatile boolean done = true;
+
+ public MultimonFullscreenTest(String title) {
+ super(title);
+ addWindowListener(new WindowAdapter() {
+ public void windowClosing(WindowEvent e) {
+ System.exit(0);
+ }
+ });
+ Panel p = new Panel();
+ deviceMap = new HashMap(gd.length);
+ int num = 0;
+ for (GraphicsDevice dev : gd) {
+ Button b;
+ if (dev == defDev) {
+ b = new Button("Primary screen: " + num);
+ System.out.println("Primary Dev : " + dev + " Bounds: " +
+ dev.getDefaultConfiguration().getBounds());
+ } else {
+ b = new Button("Secondary screen " + num);
+ System.out.println("Secondary Dev : " + dev + " Bounds: " +
+ dev.getDefaultConfiguration().getBounds());
+ }
+ b.addActionListener(this);
+ p.add(b);
+ deviceMap.put(b, dev);
+ num++;
+ }
+ add("South", p);
+ Panel p1 = new Panel();
+ p1.setLayout(new GridLayout(2,0));
+ Checkbox cb = new Checkbox("Change DM on entering FS");
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ dmChange = ((Checkbox)e.getSource()).getState();
+ }
+ });
+ p1.add(cb);
+// cb = new Checkbox("Exit FS on window dispose");
+// cb.addItemListener(new ItemListener() {
+// public void itemStateChanged(ItemEvent e) {
+// setNullOnDispose = ((Checkbox)e.getSource()).getState();
+// }
+// });
+// p1.add(cb);
+ CheckboxGroup cbg = new CheckboxGroup();
+ cb = new Checkbox("Use Frame to enter FS", cbg, true);
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ useFSFrame = true;
+ useFSWindow = false;
+ useFSDialog = false;
+ }
+ });
+ p1.add(cb);
+ cb = new Checkbox("Use Window to enter FS", cbg, false);
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ useFSFrame = false;
+ useFSWindow = true;
+ useFSDialog = false;
+ }
+ });
+ p1.add(cb);
+ cb = new Checkbox("Use Dialog to enter FS", cbg, false);
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ useFSFrame = false;
+ useFSWindow = false;
+ useFSDialog = true;
+ }
+ });
+ p1.add(cb);
+ cb = new Checkbox("Run render loop");
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ runRenderLoop = ((Checkbox)e.getSource()).getState();
+ }
+ });
+ p1.add(cb);
+ cb = new Checkbox("Use BufferStrategy in render loop");
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ useBS = ((Checkbox)e.getSource()).getState();
+ }
+ });
+ p1.add(cb);
+ cb = new Checkbox("Add Children to FS window");
+ cb.addItemListener(new ItemListener() {
+ public void itemStateChanged(ItemEvent e) {
+ addHWChildren = ((Checkbox)e.getSource()).getState();
+ }
+ });
+ p1.add(cb);
+ add("North", p1);
+
+ pack();
+ setVisible(true);
+ }
+
+ Font f = new Font("Dialog", Font.BOLD, 24);
+ Random rnd = new Random();
+ public void renderDimensions(Graphics g, Rectangle rectWndBounds,
+ GraphicsConfiguration gc) {
+ g.setColor(new Color(rnd.nextInt(0xffffff)));
+ g.fillRect(0, 0, rectWndBounds.width, rectWndBounds.height);
+
+ g.setColor(new Color(rnd.nextInt(0xffffff)));
+ Rectangle rectStrBounds;
+
+ g.setFont(f);
+
+ rectStrBounds = g.getFontMetrics().
+ getStringBounds(rectWndBounds.toString(), g).getBounds();
+ rectStrBounds.height += 30;
+ g.drawString(rectWndBounds.toString(), 50, rectStrBounds.height);
+ int oldHeight = rectStrBounds.height;
+ String isFSupported = "Exclusive Fullscreen mode supported: " +
+ gc.getDevice().isFullScreenSupported();
+ rectStrBounds = g.getFontMetrics().
+ getStringBounds(isFSupported, g).getBounds();
+ rectStrBounds.height += (10 + oldHeight);
+ g.drawString(isFSupported, 50, rectStrBounds.height);
+
+ oldHeight = rectStrBounds.height;
+ String isDMChangeSupported = "Display Mode Change supported: " +
+ gc.getDevice().isDisplayChangeSupported();
+ rectStrBounds = g.getFontMetrics().
+ getStringBounds(isDMChangeSupported, g).getBounds();
+ rectStrBounds.height += (10 + oldHeight);
+ g.drawString(isDMChangeSupported, 50, rectStrBounds.height);
+
+ oldHeight = rectStrBounds.height;
+ String usingBS = "Using BufferStrategy: " + useBS;
+ rectStrBounds = g.getFontMetrics().
+ getStringBounds(usingBS, g).getBounds();
+ rectStrBounds.height += (10 + oldHeight);
+ g.drawString(usingBS, 50, rectStrBounds.height);
+
+ final String m_strQuitMsg = "Double-click to dispose FullScreen Window";
+ rectStrBounds = g.getFontMetrics().
+ getStringBounds(m_strQuitMsg, g).getBounds();
+ g.drawString(m_strQuitMsg,
+ (rectWndBounds.width - rectStrBounds.width) / 2,
+ (rectWndBounds.height - rectStrBounds.height) / 2);
+
+
+ }
+
+ public void actionPerformed(ActionEvent ae) {
+ GraphicsDevice dev = deviceMap.get(ae.getSource());
+ System.err.println("Setting FS on device:"+dev);
+ final Window fsWindow;
+
+ if (useFSWindow) {
+ fsWindow = new Window(this, dev.getDefaultConfiguration()) {
+ public void paint(Graphics g) {
+ renderDimensions(g, getBounds(),
+ this.getGraphicsConfiguration());
+ }
+ };
+ } else if (useFSDialog) {
+ fsWindow = new Dialog((Frame)null, "FS Dialog on device "+dev, false,
+ dev.getDefaultConfiguration());
+ fsWindow.add(new Component() {
+ public void paint(Graphics g) {
+ renderDimensions(g, getBounds(),
+ this.getGraphicsConfiguration());
+ }
+ });
+ } else {
+ fsWindow = new Frame("FS Frame on device "+dev,
+ dev.getDefaultConfiguration())
+ {
+ public void paint(Graphics g) {
+ renderDimensions(g, getBounds(),
+ this.getGraphicsConfiguration());
+ }
+ };
+ if (addHWChildren) {
+ fsWindow.add("South", new Panel() {
+ public void paint(Graphics g) {
+ g.setColor(Color.red);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ }
+ });
+ fsWindow.add("North", new Button("Button, sucka!"));
+ }
+ }
+ fsWindow.addMouseListener(new MouseAdapter() {
+ public void mouseClicked(MouseEvent e) {
+ if (e.getClickCount() > 1) {
+ done = true;
+ fsWindow.dispose();
+ }
+ }
+ });
+
+ fsWindow.addWindowListener(new WindowHandler());
+ dev.setFullScreenWindow(fsWindow);
+ if (dmChange && dev.isDisplayChangeSupported()) {
+ DisplayMode dms[] = dev.getDisplayModes();
+ DisplayMode myDM = null;
+ for (DisplayMode dm : dms) {
+ if (dm.getWidth() == 800 && dm.getHeight() == 600 &&
+ (dm.getBitDepth() >= 16 ||
+ dm.getBitDepth() == DisplayMode.BIT_DEPTH_MULTI) &&
+ (dm.getRefreshRate() >= 60 ||
+ dm.getRefreshRate() == DisplayMode.REFRESH_RATE_UNKNOWN))
+ {
+ myDM = dm;
+ break;
+ }
+ }
+ if (myDM != null) {
+ System.err.println("Setting Display Mode: "+
+ myDM.getWidth() + "x" + myDM.getHeight() + "x" +
+ myDM.getBitDepth() + "@" + myDM.getRefreshRate() +
+ "Hz on device" + dev);
+ dev.setDisplayMode(myDM);
+ } else {
+ System.err.println("Can't find suitable display mode.");
+ }
+ }
+ done = false;
+ if (runRenderLoop) {
+ Thread updateThread = new Thread(new Runnable() {
+ public void run() {
+ BufferStrategy bs = null;
+ if (useBS) {
+ fsWindow.createBufferStrategy(2);
+ bs = fsWindow.getBufferStrategy();
+ }
+ while (!done) {
+ if (useBS) {
+ Graphics g = bs.getDrawGraphics();
+ renderDimensions(g, fsWindow.getBounds(),
+ fsWindow.getGraphicsConfiguration());
+ bs.show();
+ } else {
+ fsWindow.repaint();
+ }
+ try {
+ Thread.sleep(1000);
+ } catch (InterruptedException e) {}
+ }
+ if (useBS) {
+ bs.dispose();
+ }
+ }
+ });
+ updateThread.start();
+ }
+ }
+
+ public static void main(String args[]) {
+ for (String s : args) {
+ if (s.equalsIgnoreCase("-dm")) {
+ System.err.println("Do Display Change after entering FS mode");
+ dmChange = true;
+ } else if (s.equalsIgnoreCase("-usewindow")) {
+ System.err.println("Using Window to enter FS mode");
+ useFSWindow = true;
+ } else if (s.equalsIgnoreCase("-setnull")) {
+ System.err.println("Setting null FS window on dispose");
+ setNullOnDispose = true;
+ } else {
+ System.err.println("Usage: MultimonFullscreenTest " +
+ "[-dm][-usewindow][-setnull]");
+ }
+
+ }
+ MultimonFullscreenTest fs =
+ new MultimonFullscreenTest("Test Full Screen");
+ }
+ class WindowHandler extends WindowAdapter {
+ public void windowClosing(WindowEvent we) {
+ done = true;
+ Window w = (Window)we.getSource();
+ if (setNullOnDispose) {
+ w.getGraphicsConfiguration().getDevice().setFullScreenWindow(null);
+ }
+ w.dispose();
+ }
+ }
+}
diff --git a/jdk/test/java/awt/FullScreen/NoResizeEventOnDMChangeTest/NoResizeEventOnDMChangeTest.java b/jdk/test/java/awt/FullScreen/NoResizeEventOnDMChangeTest/NoResizeEventOnDMChangeTest.java
new file mode 100644
index 00000000000..2e6ccdfb024
--- /dev/null
+++ b/jdk/test/java/awt/FullScreen/NoResizeEventOnDMChangeTest/NoResizeEventOnDMChangeTest.java
@@ -0,0 +1,216 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 6646411
+ * @summary Tests that full screen window and its children receive resize
+ event when display mode changes
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm NoResizeEventOnDMChangeTest
+ * @run main/othervm -Dsun.java2d.d3d=false NoResizeEventOnDMChangeTest
+ */
+
+import java.awt.Canvas;
+import java.awt.Color;
+import java.awt.Component;
+import java.awt.DisplayMode;
+import java.awt.EventQueue;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
+import java.awt.Window;
+import java.awt.event.ComponentAdapter;
+import java.awt.event.ComponentEvent;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+
+public class NoResizeEventOnDMChangeTest {
+ public static void main(String[] args) {
+ final GraphicsDevice gd = GraphicsEnvironment.
+ getLocalGraphicsEnvironment().getDefaultScreenDevice();
+
+ if (!gd.isFullScreenSupported()) {
+ System.out.println("Full screen not supported, test passed");
+ return;
+ }
+
+ DisplayMode dm = gd.getDisplayMode();
+ final DisplayMode dms[] = new DisplayMode[2];
+ for (DisplayMode dm1 : gd.getDisplayModes()) {
+ if (dm1.getWidth() != dm.getWidth() ||
+ dm1.getHeight() != dm.getHeight())
+ {
+ dms[0] = dm1;
+ break;
+ }
+ }
+ if (dms[0] == null) {
+ System.out.println("Test Passed: all DMs have same dimensions");
+ return;
+ }
+ dms[1] = dm;
+
+ Frame f = new Frame() {
+ @Override
+ public void paint(Graphics g) {
+ g.setColor(Color.red);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ g.setColor(Color.green);
+ g.drawRect(0, 0, getWidth()-1, getHeight()-1);
+ }
+ };
+ f.setUndecorated(true);
+ testFSWindow(gd, dms, f);
+
+ Window w = new Window(f) {
+ @Override
+ public void paint(Graphics g) {
+ g.setColor(Color.magenta);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ g.setColor(Color.cyan);
+ g.drawRect(0, 0, getWidth()-1, getHeight()-1);
+ }
+ };
+ testFSWindow(gd, dms, w);
+ System.out.println("Test Passed.");
+ }
+
+ private static void testFSWindow(final GraphicsDevice gd,
+ final DisplayMode dms[],
+ final Window fsWin)
+ {
+ System.out.println("Testing FS window: "+fsWin);
+ Component c = new Canvas() {
+ @Override
+ public void paint(Graphics g) {
+ g.setColor(Color.blue);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ g.setColor(Color.magenta);
+ g.drawRect(0, 0, getWidth()-1, getHeight()-1);
+ g.setColor(Color.red);
+ g.drawString("FS Window : " + fsWin, 50, 50);
+ DisplayMode dm =
+ getGraphicsConfiguration().getDevice().getDisplayMode();
+ g.drawString("Display Mode: " +
+ dm.getWidth() + "x" + dm.getHeight(), 50, 75);
+ }
+ };
+ fsWin.add("Center", c);
+ fsWin.addWindowListener(new WindowAdapter() {
+ @Override
+ public void windowClosing(WindowEvent e) {
+ fsWin.dispose();
+ if (fsWin.getOwner() != null) {
+ fsWin.getOwner().dispose();
+ }
+ }
+ });
+
+ try {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ gd.setFullScreenWindow(fsWin);
+ }
+ });
+ } catch (Exception ex) {}
+
+ sleep(1000);
+
+ final ResizeEventChecker r1 = new ResizeEventChecker();
+ final ResizeEventChecker r2 = new ResizeEventChecker();
+
+ if (gd.isDisplayChangeSupported()) {
+ fsWin.addComponentListener(r1);
+ c.addComponentListener(r2);
+ for (final DisplayMode dm1 : dms) {
+ try {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ System.err.printf("----------- Setting DM %dx%d:\n",
+ dm1.getWidth(), dm1.getHeight());
+ try {
+ gd.setDisplayMode(dm1);
+ r1.incDmChanges();
+ r2.incDmChanges();
+ } catch (IllegalArgumentException iae) {}
+ }
+ });
+ } catch (Exception ex) {}
+ for (int i = 0; i < 3; i++) {
+ fsWin.repaint();
+ sleep(1000);
+ }
+ }
+ fsWin.removeComponentListener(r1);
+ c.removeComponentListener(r2);
+ }
+ try {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ gd.setFullScreenWindow(null);
+ fsWin.dispose();
+ if (fsWin.getOwner() != null) {
+ fsWin.getOwner().dispose();
+ }
+ }
+ });
+ } catch (Exception ex) {}
+
+ System.out.printf("FS Window: resizes=%d, dm changes=%d\n",
+ r1.getResizes(), r1.getDmChanges());
+ System.out.printf("Component: resizes=%d, dm changes=%d\n",
+ r2.getResizes(), r2.getDmChanges());
+ if (r1.getResizes() < r1.getDmChanges()) {
+ throw new RuntimeException("FS Window didn't receive all resizes!");
+ }
+ if (r2.getResizes() < r2.getDmChanges()) {
+ throw new RuntimeException("Component didn't receive all resizes!");
+ }
+ }
+
+ static void sleep(long ms) {
+ try {
+ Thread.sleep(ms);
+ } catch (InterruptedException ex) {}
+ }
+ static class ResizeEventChecker extends ComponentAdapter {
+ int dmChanges;
+ int resizes;
+
+ @Override
+ public synchronized void componentResized(ComponentEvent e) {
+ System.out.println("Received resize event for "+e.getSource());
+ resizes++;
+ }
+ public synchronized int getResizes() {
+ return resizes;
+ }
+ public synchronized void incDmChanges() {
+ dmChanges++;
+ }
+ public synchronized int getDmChanges() {
+ return dmChanges;
+ }
+ }
+}
diff --git a/jdk/test/java/awt/FullScreen/SetFSWindow/FSFrame.java b/jdk/test/java/awt/FullScreen/SetFSWindow/FSFrame.java
new file mode 100644
index 00000000000..c4d423114f7
--- /dev/null
+++ b/jdk/test/java/awt/FullScreen/SetFSWindow/FSFrame.java
@@ -0,0 +1,204 @@
+/*
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6240507 6662642
+ * @summary verify that isFullScreenSupported and getFullScreenWindow work
+ * correctly with and without a SecurityManager. Note that the test may fail
+ * on older Gnome versions (see bug 6500686).
+ * @run main FSFrame
+ * @run main/othervm -Dsun.java2d.noddraw=true FSFrame
+ * @author cheth
+ */
+
+import java.awt.*;
+import java.awt.image.*;
+import java.applet.*;
+import java.io.File;
+import java.io.IOException;
+import java.lang.reflect.InvocationTargetException;
+import javax.imageio.ImageIO;
+
+public class FSFrame extends Frame implements Runnable {
+
+ // Don't start the test until the window is visible
+ boolean visible = false;
+ Robot robot = null;
+ static volatile boolean done = false;
+
+ public void paint(Graphics g) {
+ if (!visible && getWidth() != 0 && getHeight() != 0) {
+ visible = true;
+ try {
+ GraphicsDevice gd = getGraphicsConfiguration().getDevice();
+ robot = new Robot(gd);
+ } catch (Exception e) {
+ System.out.println("Problem creating robot: cannot verify FS " +
+ "window display");
+ }
+ }
+ g.setColor(Color.green);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ }
+
+ @Override
+ public void update(Graphics g) {
+ paint(g);
+ }
+
+ boolean checkColor(int x, int y, BufferedImage bImg) {
+ int pixelColor;
+ int correctColor = Color.green.getRGB();
+ pixelColor = bImg.getRGB(x, y);
+ if (pixelColor != correctColor) {
+ System.out.println("FAILURE: pixelColor " +
+ Integer.toHexString(pixelColor) +
+ " != correctColor " +
+ Integer.toHexString(correctColor) +
+ " at coordinates (" + x + ", " + y + ")");
+ return false;
+ }
+ return true;
+ }
+
+ void checkFSDisplay(boolean fsSupported) {
+ GraphicsConfiguration gc = getGraphicsConfiguration();
+ GraphicsDevice gd = gc.getDevice();
+ Rectangle r = gc.getBounds();
+ Insets in = null;
+ if (!fsSupported) {
+ in = Toolkit.getDefaultToolkit().getScreenInsets(gc);
+ r = new Rectangle(in.left, in.top,
+ r.width - (in.left + in.right),
+ r.height - (in.top + in.bottom));
+ }
+ BufferedImage bImg = robot.createScreenCapture(r);
+ // Check that all four corners and middle pixel match the window's
+ // fill color
+ if (robot == null) {
+ return;
+ }
+ boolean colorCorrect = true;
+ colorCorrect &= checkColor(0, 0, bImg);
+ colorCorrect &= checkColor(0, bImg.getHeight() - 1, bImg);
+ colorCorrect &= checkColor(bImg.getWidth() - 1, 0, bImg);
+ colorCorrect &= checkColor(bImg.getWidth() - 1, bImg.getHeight() - 1, bImg);
+ colorCorrect &= checkColor(bImg.getWidth() / 2, bImg.getHeight() / 2, bImg);
+ if (!colorCorrect) {
+ System.err.println("Test failed for mode: fsSupported="+fsSupported);
+ if (in != null) {
+ System.err.println("screen insets : " + in);
+ }
+ System.err.println("screen shot rect: " + r);
+ String name = "FSFrame_fs_"+
+ (fsSupported?"supported":"not_supported")+".png";
+ try {
+ ImageIO.write(bImg, "png", new File(name));
+ System.out.println("Dumped screen shot to "+name);
+ } catch (IOException ex) {}
+ throw new Error("Some pixel colors not correct; FS window may not" +
+ " have been displayed correctly");
+ }
+ }
+
+ void checkFSFunctionality(boolean withSecurity) {
+ GraphicsDevice gd = getGraphicsConfiguration().getDevice();
+ if (withSecurity) {
+ SecurityManager sm = new SecurityManager();
+ System.setSecurityManager(sm);
+ }
+ try {
+ // None of these should throw an exception
+ final boolean fs = gd.isFullScreenSupported();
+ System.out.println("FullscreenSupported: " + (fs ? "yes" : "no"));
+ gd.setFullScreenWindow(this);
+ try {
+ // Give the system time to set the FS window and display it
+ // properly
+ Thread.sleep(2000);
+ } catch (Exception e) {}
+ if (!withSecurity) {
+ // See if FS window got displayed correctly
+ try {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ repaint();
+ checkFSDisplay(fs);
+ }
+ });
+ } catch (InvocationTargetException ex) {
+ ex.printStackTrace();
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ }
+ // reset window
+ gd.setFullScreenWindow(null);
+ try {
+ // Give the system time to set the FS window and display it
+ // properly
+ Thread.sleep(2000);
+ } catch (Exception e) {}
+ } catch (SecurityException e) {
+ e.printStackTrace();
+ throw new Error("Failure: should not get an exception when " +
+ "calling isFSSupported or setFSWindow");
+ }
+ }
+
+ public void run() {
+ boolean firstTime = true;
+ while (!done) {
+ if (visible) {
+ checkFSFunctionality(false);
+ checkFSFunctionality(true);
+ done = true;
+ } else {
+ // sleep while we wait
+ try {
+ // Give the system time to set the FS window and display it
+ // properly
+ Thread.sleep(100);
+ } catch (Exception e) {}
+ }
+ }
+ System.out.println("PASS");
+ }
+
+ public static void main(String args[]) {
+ FSFrame frame = new FSFrame();
+ frame.setUndecorated(true);
+ Thread t = new Thread(frame);
+ frame.setSize(500, 500);
+ frame.setVisible(true);
+ t.start();
+ while (!done) {
+ try {
+ // Do not exit the main thread until the test is finished
+ Thread.sleep(1000);
+ } catch (Exception e) {}
+ }
+ frame.dispose();
+ }
+}
diff --git a/jdk/test/java/awt/Multiscreen/DeviceIdentificationTest/DeviceIdentificationTest.java b/jdk/test/java/awt/Multiscreen/DeviceIdentificationTest/DeviceIdentificationTest.java
new file mode 100644
index 00000000000..a3062bcef86
--- /dev/null
+++ b/jdk/test/java/awt/Multiscreen/DeviceIdentificationTest/DeviceIdentificationTest.java
@@ -0,0 +1,168 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/**
+ * @test
+ * @bug 6614214
+ * @summary Verifies that we enter the fs mode on the correct screen.
+ * Here is how to test: start the test on on a multi-screen system.
+ * Verify that the display is correctly tracked by dragging the frame back
+ * and forth between screens. Then verify that the correct device enters
+ * the full-screen mode - when "Enter FS mode" is pressed it should enter on
+ * the device where the frame is.
+ *
+ * Then change the order of the monitors in the DisplayProperties dialog,
+ * (while the app is running) and see that it still works.
+ * Restart the app, verify again.
+ *
+ * Now change the primary monitor on the system and verify with the
+ * app running, as well as after restarting it that we still enter the
+ * fs mode on the right device.
+ *
+ * @run main/manual/othervm DeviceIdentificationTest
+ * @run main/manual/othervm -Dsun.java2d.noddraw=true DeviceIdentificationTest
+ * @run main/manual/othervm -Dsun.java2d.opengl=True DeviceIdentificationTest
+ */
+
+import java.awt.Button;
+import java.awt.Color;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
+import java.awt.Panel;
+import java.awt.event.ActionEvent;
+import java.awt.event.ActionListener;
+import java.awt.event.ComponentAdapter;
+import java.awt.event.ComponentEvent;
+import java.awt.event.MouseAdapter;
+import java.awt.event.MouseEvent;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+
+public class DeviceIdentificationTest {
+
+ public static void main(String args[]) {
+ final Frame f = new Frame("DeviceIdentificationTest");
+ f.addWindowListener(new WindowAdapter() {
+ public void windowClosing(WindowEvent e) {
+ f.dispose();
+ }
+ });
+ f.addComponentListener(new ComponentAdapter() {
+ public void componentMoved(ComponentEvent e) {
+ f.setTitle("Currently on: "+
+ f.getGraphicsConfiguration().getDevice());
+ }
+ });
+
+ Panel p = new Panel();
+ Button b = new Button("Print Current Devices");
+ b.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ GraphicsDevice gds[] =
+ GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getScreenDevices();
+ int i = 0;
+ System.err.println("--- Devices: ---");
+ for (GraphicsDevice gd : gds) {
+ System.err.println("Device["+i+"]= "+ gd);
+ System.err.println(" bounds = "+
+ gd.getDefaultConfiguration().getBounds());
+ i++;
+ }
+ System.err.println("-------------------");
+ }
+ });
+ p.add(b);
+
+ b = new Button("Print My Device");
+ b.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ GraphicsConfiguration gc = f.getGraphicsConfiguration();
+ GraphicsDevice gd = gc.getDevice();
+ System.err.println("--- My Device ---");
+ System.err.println("Device = "+ gd);
+ System.err.println(" bounds = "+
+ gd.getDefaultConfiguration().getBounds());
+ }
+ });
+ p.add(b);
+
+ b = new Button("Create FS Frame on my Device");
+ b.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ GraphicsConfiguration gc = f.getGraphicsConfiguration();
+ final GraphicsDevice gd = gc.getDevice();
+ System.err.println("--- Creating FS Frame on Device ---");
+ System.err.println("Device = "+ gd);
+ System.err.println(" bounds = "+
+ gd.getDefaultConfiguration().getBounds());
+ final Frame fsf = new Frame("Full-screen Frame on dev"+gd, gc) {
+ public void paint(Graphics g) {
+ g.setColor(Color.green);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ g.setColor(Color.red);
+ g.drawString("FS on device: "+gd, 200, 200);
+ g.drawString("Click to exit Full-screen.", 200, 250);
+ }
+ };
+ fsf.setUndecorated(true);
+ fsf.addMouseListener(new MouseAdapter() {
+ public void mouseClicked(MouseEvent e) {
+ gd.setFullScreenWindow(null);
+ fsf.dispose();
+ }
+ });
+ gd.setFullScreenWindow(fsf);
+ }
+ });
+ p.add(b);
+ f.add("North", p);
+
+ p = new Panel();
+ b = new Button("Test Passed");
+ b.setBackground(Color.green);
+ b.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ System.out.println("Test Passed");
+ f.dispose();
+ }
+ });
+ p.add(b);
+ b = new Button("Test Failed");
+ b.setBackground(Color.red);
+ b.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ System.out.println("Test FAILED");
+ f.dispose();
+ throw new RuntimeException("Test FAILED");
+ }
+ });
+ p.add(b);
+ f.add("South", p);
+
+ f.pack();
+ f.setVisible(true);
+ }
+}
diff --git a/jdk/test/java/awt/font/TextLayout/VisibleAdvance.java b/jdk/test/java/awt/font/TextLayout/VisibleAdvance.java
index 446d16dba15..224b2386125 100644
--- a/jdk/test/java/awt/font/TextLayout/VisibleAdvance.java
+++ b/jdk/test/java/awt/font/TextLayout/VisibleAdvance.java
@@ -1,3 +1,27 @@
+/*
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+
import java.awt.*;
import java.awt.font.*;
import java.awt.geom.*;
diff --git a/jdk/test/java/awt/image/MemoryLeakTest/MemoryLeakTest.java b/jdk/test/java/awt/image/MemoryLeakTest/MemoryLeakTest.java
new file mode 100644
index 00000000000..ace1bb4979e
--- /dev/null
+++ b/jdk/test/java/awt/image/MemoryLeakTest/MemoryLeakTest.java
@@ -0,0 +1,175 @@
+/*
+ * Copyright 1998-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/* @test
+ @bug 4078566 6658398
+ @summary Test for a memory leak in Image.
+ @run main/manual MemoryLeakTest
+*/
+
+import java.applet.Applet;
+import java.lang.*;
+import java.awt.*;
+import java.awt.event.*;
+
+class Globals {
+ static boolean testPassed=false;
+ static Thread mainThread=null;
+}
+
+public class MemoryLeakTest extends Applet {
+
+public static void main(String args[]) throws Exception {
+ new TestDialog(new Frame(), "MemoryLeakTest").start();
+ new MemoryLeak().start();
+ Globals.mainThread = Thread.currentThread();
+ try {
+ Thread.sleep(300000);
+ } catch (InterruptedException e) {
+ if (!Globals.testPassed)
+ throw new Exception("MemoryLeakTest failed.");
+ }
+}
+
+}
+
+class TestDialog extends Dialog
+ implements ActionListener {
+
+TextArea output;
+Button passButton;
+Button failButton;
+String name;
+
+public TestDialog(Frame frame, String name)
+{
+ super(frame, name + " Pass/Fail Dialog");
+ this.name = name;
+ output = new TextArea(11, 50);
+ add("North", output);
+ output.append("Do the following steps on Solaris only.\n");
+ output.append("Maximize and minimize the Memory Leak Test window.\n");
+ output.append("Execute the following after minimize.\n");
+ output.append(" ps -al | egrep -i 'java|PPID'\n");
+ output.append("Examine the size of the process under SZ.\n");
+ output.append("Maximize and minimize the Memory Leak Test window again.\n");
+ output.append("Execute the following after minimize.\n");
+ output.append(" ps -al | egrep -i 'java|PPID'\n");
+ output.append("Examine the size of the process under SZ.\n");
+ output.append("If the two SZ values are the same, plus or minus one,\n");
+ output.append("then click Pass, else click Fail.");
+ Panel buttonPanel = new Panel();
+ passButton = new Button("Pass");
+ failButton = new Button("Fail");
+ passButton.addActionListener(this);
+ failButton.addActionListener(this);
+ buttonPanel.add(passButton);
+ buttonPanel.add(failButton);
+ add("South", buttonPanel);
+ pack();
+}
+
+public void start()
+{
+ show();
+}
+
+public void actionPerformed(ActionEvent event)
+{
+ if ( event.getSource() == passButton ) {
+ Globals.testPassed = true;
+ System.err.println(name + " Passed.");
+ }
+ else if ( event.getSource() == failButton ) {
+ Globals.testPassed = false;
+ System.err.println(name + " Failed.");
+ }
+ this.dispose();
+ if (Globals.mainThread != null)
+ Globals.mainThread.interrupt();
+}
+
+}
+
+
+class MemoryLeak extends Frame implements ComponentListener
+{
+private Image osImage;
+
+public MemoryLeak()
+{
+ super("Memory Leak Test");
+ setSize(200, 200);
+ addComponentListener(this);
+}
+
+public static void main(String args[])
+{
+ new MemoryLeak().start();
+}
+
+public void start()
+{
+ show();
+}
+
+public void paint(Graphics g) {
+ if (osImage != null) {
+ g.drawImage(osImage, 0, 0, this);
+ }
+}
+
+public void update(Graphics g)
+{
+ paint(g);
+}
+
+public void componentResized(ComponentEvent e)
+{
+ Image oldimage = osImage;
+ osImage = createImage(getSize().width, getSize().height);
+ Graphics g = osImage.getGraphics();
+ if (oldimage != null) {
+ g.drawImage(oldimage, 0, 0, getSize().width, getSize().height, this);
+ oldimage.flush();
+ } else {
+ g.setColor(Color.blue);
+ g.drawLine(0, 0, getSize().width, getSize().height);
+ }
+ g.dispose();
+}
+
+public void componentMoved(ComponentEvent e) {}
+
+public void componentShown(ComponentEvent e)
+{
+ osImage = createImage(getSize().width, getSize().height);
+ Graphics g = osImage.getGraphics();
+ g.setColor(Color.blue);
+ g.drawLine(0, 0, getSize().width, getSize().height);
+ g.dispose();
+}
+
+public void componentHidden(ComponentEvent e) {}
+
+}
diff --git a/jdk/test/java/awt/print/PrinterJob/PrintAWTImage.java b/jdk/test/java/awt/print/PrinterJob/PrintAWTImage.java
new file mode 100644
index 00000000000..0cb3b4eb86b
--- /dev/null
+++ b/jdk/test/java/awt/print/PrinterJob/PrintAWTImage.java
@@ -0,0 +1,90 @@
+/*
+ * Copyright 1999-2003 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/**
+ * @test
+ * @bug 4257262 6708509
+ * @summary Image should be sent to printer.
+* @run main/manual PrintAWTImage
+ */
+
+import java.awt.*;
+import java.awt.event.*;
+import java.awt.print.*;
+
+
+public class PrintAWTImage extends Frame
+ implements ActionListener, Printable {
+
+ public Image imgJava;
+
+
+ public static void main(String args[]) {
+ PrintAWTImage f = new PrintAWTImage();
+ f.show();
+ }
+
+ public PrintAWTImage() {
+
+ Button printButton = new Button("Print");
+ setLayout(new FlowLayout());
+ add(printButton);
+ printButton.addActionListener(this);
+
+ addWindowListener(new WindowAdapter() {
+ public void windowClosing(WindowEvent e) {
+ System.exit(0);
+ }
+ });
+
+ pack();
+ }
+
+ public void actionPerformed(ActionEvent e) {
+
+ PrinterJob pj = PrinterJob.getPrinterJob();
+
+ if (pj != null && pj.printDialog()) {
+ pj.setPrintable(this);
+ try {
+ pj.print();
+ } catch (PrinterException pe) {
+ } finally {
+ System.err.println("PRINT RETURNED");
+ }
+ }
+ }
+
+
+ public int print(Graphics g, PageFormat pgFmt, int pgIndex) {
+ if (pgIndex > 0)
+ return Printable.NO_SUCH_PAGE;
+
+ Graphics2D g2d = (Graphics2D)g;
+ g2d.translate(pgFmt.getImageableX(), pgFmt.getImageableY());
+ Image imgJava = Toolkit.getDefaultToolkit().getImage("duke.gif");
+ g2d.drawImage(imgJava, 0, 0, this);
+
+ return Printable.PAGE_EXISTS;
+ }
+
+}
diff --git a/jdk/test/java/awt/print/PrinterJob/duke.gif b/jdk/test/java/awt/print/PrinterJob/duke.gif
new file mode 100644
index 00000000000..ed32e0ff79b
Binary files /dev/null and b/jdk/test/java/awt/print/PrinterJob/duke.gif differ
diff --git a/jdk/test/java/util/Currency/ValidateISO4217.java b/jdk/test/java/util/Currency/ValidateISO4217.java
index 0e7e7d15462..ed1f2bc259d 100644
--- a/jdk/test/java/util/Currency/ValidateISO4217.java
+++ b/jdk/test/java/util/Currency/ValidateISO4217.java
@@ -90,7 +90,7 @@ public class ValidateISO4217 {
/* Codes that are obsolete, do not have related country */
static final String otherCodes =
- "ADP-AFA-ATS-AYM-BEF-BGL-BOV-BYB-CLF-DEM-ESP-FIM-FRF-GRD-GWP-IEP-ITL-LUF-MGF-MXV-NLG-PTE-RUR-SDD-SIT-SRG-TPE-TRL-VEF-USN-USS-XAG-XAU-XBA-XBB-XBC-XBD-XDR-XFO-XFU-XPD-XPT-XTS-XXX-YUM-ZWN";
+ "ADP-AFA-ATS-AYM-BEF-BGL-BOV-BYB-CLF-CYP-DEM-ESP-FIM-FRF-GRD-GWP-IEP-ITL-LUF-MGF-MTL-MXV-NLG-PTE-RUR-SDD-SIT-SRG-TPE-TRL-VEF-USN-USS-XAG-XAU-XBA-XBB-XBC-XBD-XDR-XFO-XFU-XPD-XPT-XTS-XXX-YUM-ZWN";
static boolean err = false;
diff --git a/jdk/test/java/util/Currency/tablea1.txt b/jdk/test/java/util/Currency/tablea1.txt
index cd2601101cf..69684358b9c 100644
--- a/jdk/test/java/util/Currency/tablea1.txt
+++ b/jdk/test/java/util/Currency/tablea1.txt
@@ -1,12 +1,12 @@
#
#
-# Based on BSi's ISO4217 data - "TABLE A1.doc" + amendments up until MA139.doc
-# (As of 18 June 2007)
+# Based on BSi's ISO4217 data - "TABLE A1.doc" + amendments up until MA140.doc
+# (As of 24 September 2007)
#
# Version
FILEVERSION=1
-DATAVERSION=139
+DATAVERSION=140
# ISO 4217 currency data
AF AFN 971 2
@@ -68,7 +68,7 @@ CR CRC 188 2
CI XOF 952 0
HR HRK 191 2
CU CUP 192 2
-CY CYP 196 2
+CY EUR 978 2
CZ CZK 203 2
DK DKK 208 2
DJ DJF 262 0
@@ -149,7 +149,7 @@ MW MWK 454 2
MY MYR 458 2
MV MVR 462 2
ML XOF 952 0
-MT MTL 470 2
+MT EUR 978 2
MH USD 840 2
MQ EUR 978 2
MR MRO 478 2
diff --git a/jdk/test/sun/java2d/DirectX/AccelPaintsTest/AccelPaintsTest.java b/jdk/test/sun/java2d/DirectX/AccelPaintsTest/AccelPaintsTest.java
new file mode 100644
index 00000000000..f58ee5a9ea9
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/AccelPaintsTest/AccelPaintsTest.java
@@ -0,0 +1,180 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6659345
+ * @summary Tests that various paints work correctly when preceeded by a
+ * textured operaiton.
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm AccelPaintsTest
+ * @run main/othervm -Dsun.java2d.opengl=True AccelPaintsTest
+ */
+
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.EventQueue;
+import java.awt.GradientPaint;
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsEnvironment;
+import java.awt.LinearGradientPaint;
+import java.awt.MultipleGradientPaint.CycleMethod;
+import java.awt.Paint;
+import java.awt.RadialGradientPaint;
+import java.awt.Rectangle;
+import java.awt.Shape;
+import java.awt.TexturePaint;
+import java.awt.Transparency;
+import java.awt.geom.Rectangle2D;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+import java.io.File;
+import java.io.IOException;
+import java.lang.reflect.InvocationTargetException;
+import javax.imageio.ImageIO;
+import javax.swing.JFrame;
+import javax.swing.JPanel;
+
+public class AccelPaintsTest extends JPanel {
+ BufferedImage bi =
+ new BufferedImage(80, 100, BufferedImage.TYPE_INT_ARGB_PRE);
+
+ RadialGradientPaint rgp =
+ new RadialGradientPaint(100, 100, 100, new float[] {0f, 0.2f, 0.6f, 1f},
+ new Color[] { Color.red,
+ Color.yellow,
+ Color.blue,
+ Color.green},
+ CycleMethod.REFLECT);
+ LinearGradientPaint lgp =
+ new LinearGradientPaint(30, 30, 120, 130, new float[] {0f, 0.2f, 0.6f, 1f},
+ new Color[] {Color.red,
+ Color.yellow,
+ Color.blue,
+ Color.green});
+ GradientPaint gp =
+ new GradientPaint(30, 30, Color.red, 120, 130, Color.yellow, true);
+
+ TexturePaint tp =
+ new TexturePaint(bi, new Rectangle2D.Float(30, 30, 120, 130));
+
+
+ public AccelPaintsTest() {
+ Graphics g = bi.getGraphics();
+ g.setColor(Color.blue);
+ g.fillRect(0, 0, bi.getWidth(), bi.getHeight());
+
+ setPreferredSize(new Dimension(250, 4*120));
+ }
+
+ private void renderWithPaint(Graphics2D g2d, Paint p) {
+ g2d.drawImage(bi, 130, 30, null);
+
+ g2d.setPaint(p);
+ g2d.fillRect(30, 30, 80, 100);
+ }
+
+ private void render(Graphics2D g2d) {
+ renderWithPaint(g2d, rgp);
+ g2d.translate(0, 100);
+
+ renderWithPaint(g2d, lgp);
+ g2d.translate(0, 100);
+
+ renderWithPaint(g2d, gp);
+ g2d.translate(0, 100);
+
+ renderWithPaint(g2d, tp);
+ g2d.translate(0, 100);
+ }
+
+ private void test() {
+ GraphicsConfiguration gc =
+ GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getDefaultScreenDevice().getDefaultConfiguration();
+ if (gc.getColorModel().getPixelSize() < 16) {
+ System.out.println("<16 bit depth detected, test passed");
+ return;
+ }
+
+ VolatileImage vi =
+ gc.createCompatibleVolatileImage(250, 4*120, Transparency.OPAQUE);
+ BufferedImage res;
+ do {
+ vi.validate(gc);
+ Graphics2D g2d = vi.createGraphics();
+ g2d.setColor(Color.white);
+ g2d.fillRect(0, 0, vi.getWidth(), vi.getHeight());
+
+ render(g2d);
+
+ res = vi.getSnapshot();
+ } while (vi.contentsLost());
+
+ for (int y = 0; y < bi.getHeight(); y++) {
+ for (int x = 0; x < bi.getWidth(); x++) {
+ if (res.getRGB(x, y) == Color.black.getRGB()) {
+ System.err.printf("Test FAILED: found black at %d,%d\n",
+ x, y);
+ try {
+ String fileName = "AccelPaintsTest.png";
+ ImageIO.write(res, "png", new File(fileName));
+ System.err.println("Dumped rendering to " + fileName);
+ } catch (IOException e) {}
+ throw new RuntimeException("Test FAILED: found black");
+ }
+ }
+ }
+ }
+
+ protected void paintComponent(Graphics g) {
+ super.paintComponent(g);
+ Graphics2D g2d = (Graphics2D)g;
+
+ render(g2d);
+ }
+
+ public static void main(String[] args)
+ throws InterruptedException, InvocationTargetException
+ {
+
+ if (args.length > 0 && args[0].equals("-show")) {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ JFrame f = new JFrame("RadialGradientTest");
+ f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
+ AccelPaintsTest t = new AccelPaintsTest();
+ f.add(t);
+ f.pack();
+ f.setVisible(true);
+ }
+ });
+ } else {
+ AccelPaintsTest t = new AccelPaintsTest();
+ t.test();
+ System.out.println("Test Passed.");
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/AcceleratedScaleTest/AcceleratedScaleTest.java b/jdk/test/sun/java2d/DirectX/AcceleratedScaleTest/AcceleratedScaleTest.java
new file mode 100644
index 00000000000..654d03ed722
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/AcceleratedScaleTest/AcceleratedScaleTest.java
@@ -0,0 +1,135 @@
+/*
+ * Copyright 2006-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+import java.awt.Color;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsEnvironment;
+import java.awt.RenderingHints;
+import java.awt.Toolkit;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+import java.io.File;
+import java.io.IOException;
+import javax.imageio.ImageIO;
+
+/**
+ * @test
+ * @bug 6429665
+ * @bug 6588884
+ * @summary Tests that the transform is correctly handled
+ * @author Dmitri.Trembovetski area=Graphics
+ * @run main AcceleratedScaleTest
+ * @run main/othervm -Dsun.java2d.d3d=true AcceleratedScaleTest
+ * @run main/othervm -Dsun.java2d.opengl=true AcceleratedScaleTest
+ */
+public class AcceleratedScaleTest {
+ private static final int IMAGE_SIZE = 200;
+ private static VolatileImage destVI;
+
+ private static void initVI(GraphicsConfiguration gc) {
+ int res;
+ if (destVI == null) {
+ res = VolatileImage.IMAGE_INCOMPATIBLE;
+ } else {
+ res = destVI.validate(gc);
+ }
+ if (res == VolatileImage.IMAGE_INCOMPATIBLE) {
+ if (destVI != null) destVI.flush();
+ destVI = gc.createCompatibleVolatileImage(IMAGE_SIZE, IMAGE_SIZE);
+ destVI.validate(gc);
+ res = VolatileImage.IMAGE_RESTORED;
+ }
+ if (res == VolatileImage.IMAGE_RESTORED) {
+ Graphics vig = destVI.getGraphics();
+ vig.setColor(Color.red);
+ vig.fillRect(0, 0, destVI.getWidth(), destVI.getHeight());
+ vig.dispose();
+ }
+ }
+
+ public static void main(String[] args) {
+ Frame f = new Frame();
+ f.pack();
+ GraphicsConfiguration gc = f.getGraphicsConfiguration();
+ if (gc.getColorModel().getPixelSize() < 16) {
+ System.out.printf("Bit depth: %d . Test considered passed.",
+ gc.getColorModel().getPixelSize());
+ f.dispose();
+ return;
+ }
+
+ BufferedImage bi =
+ new BufferedImage(IMAGE_SIZE/4, IMAGE_SIZE/4,
+ BufferedImage.TYPE_INT_RGB);
+ Graphics2D g = (Graphics2D)bi.getGraphics();
+ g.setColor(Color.red);
+ g.fillRect(0, 0, bi.getWidth(), bi.getHeight());
+ BufferedImage snapshot;
+ do {
+ initVI(gc);
+ g = (Graphics2D)destVI.getGraphics();
+ // "accelerate" BufferedImage
+ for (int i = 0; i < 5; i++) {
+ g.drawImage(bi, 0, 0, null);
+ }
+ g.setColor(Color.white);
+ g.fillRect(0, 0, destVI.getWidth(), destVI.getHeight());
+
+ // this will force the use of Transform primitive instead of
+ // Scale (the latter doesn't do bilinear filtering required by
+ // VALUE_RENDER_QUALITY, which triggers the bug in D3D pipeline
+ g.setRenderingHint(RenderingHints.KEY_RENDERING,
+ RenderingHints.VALUE_RENDER_QUALITY);
+ g.drawImage(bi, 0, 0, destVI.getWidth(), destVI.getHeight(), null);
+ g.fillRect(0, 0, destVI.getWidth(), destVI.getHeight());
+
+ g.drawImage(bi, 0, 0, destVI.getWidth(), destVI.getHeight(), null);
+
+ snapshot = destVI.getSnapshot();
+ } while (destVI.contentsLost());
+
+ f.dispose();
+ int whitePixel = Color.white.getRGB();
+ for (int y = 0; y < snapshot.getHeight(); y++) {
+ for (int x = 0; x < snapshot.getWidth(); x++) {
+ if (snapshot.getRGB(x, y) == whitePixel) {
+ System.out.printf("Found untouched pixel at %dx%d\n", x, y);
+ System.out.println("Dumping the dest. image to " +
+ "AcceleratedScaleTest_dst.png");
+ try {
+ ImageIO.write(snapshot, "png",
+ new File("AcceleratedScaleTest_dst.png"));
+ } catch (IOException ex) {
+ ex.printStackTrace();
+ }
+ throw new RuntimeException("Test failed.");
+ }
+ }
+ }
+ System.out.println("Test Passed.");
+ }
+
+}
diff --git a/jdk/test/sun/java2d/DirectX/IAEforEmptyFrameTest/IAEforEmptyFrameTest.java b/jdk/test/sun/java2d/DirectX/IAEforEmptyFrameTest/IAEforEmptyFrameTest.java
new file mode 100644
index 00000000000..ffe9eb2de68
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/IAEforEmptyFrameTest/IAEforEmptyFrameTest.java
@@ -0,0 +1,50 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6668439
+ * @summary Verifies that no exceptions are thrown when frame is resized to 0x0
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm IAEforEmptyFrameTest
+ * @run main/othervm -Dsun.java2d.d3d=false IAEforEmptyFrameTest
+ */
+import javax.swing.JFrame;
+
+public class IAEforEmptyFrameTest {
+ public static void main(String[] args) {
+ JFrame f = null;
+ try {
+ f = new JFrame("IAEforEmptyFrameTest");
+ f.setUndecorated(true);
+ f.setBounds(100, 100, 320, 240);
+ f.setVisible(true);
+ try { Thread.sleep(1000); } catch (Exception z) {}
+ f.setBounds(0, 0, 0, 0);
+ try { Thread.sleep(1000); } catch (Exception z) {}
+ f.dispose();
+ } finally {
+ f.dispose();
+ };
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/InfiniteValidationLoopTest/InfiniteValidationLoopTest.java b/jdk/test/sun/java2d/DirectX/InfiniteValidationLoopTest/InfiniteValidationLoopTest.java
new file mode 100644
index 00000000000..a520a388056
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/InfiniteValidationLoopTest/InfiniteValidationLoopTest.java
@@ -0,0 +1,116 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6648018
+ * @summary Tests that we don't run into infinite validation loop when copying
+ a VolatileImage to the screen
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm InfiniteValidationLoopTest
+ * @run main/othervm -Dsun.java2d.d3d=false InfiniteValidationLoopTest
+ */
+import java.awt.Color;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsConfiguration;
+import static java.awt.image.VolatileImage.*;
+import java.awt.image.VolatileImage;
+import java.util.concurrent.CountDownLatch;
+
+public class InfiniteValidationLoopTest extends Frame {
+ private static volatile boolean failed = false;
+ private static final int LOOP_THRESHOLD = 50;
+ private static volatile CountDownLatch latch;
+ private VolatileImage vi;
+
+ public InfiniteValidationLoopTest() {
+ super("InfiniteValidationLoopTest");
+ }
+
+ @Override
+ public void paint(Graphics g) {
+ try {
+ runTest(g);
+ } finally {
+ latch.countDown();
+ }
+ }
+
+ private void runTest(Graphics g) {
+ int status = IMAGE_OK;
+ int count1 = 0;
+ do {
+ GraphicsConfiguration gc = getGraphicsConfiguration();
+ int count2 = 0;
+ while (vi == null || (status = vi.validate(gc)) != IMAGE_OK) {
+ if (++count2 > LOOP_THRESHOLD) {
+ System.err.println("Infinite loop detected: count2="+count2);
+ failed = true;
+ return;
+ }
+ if (vi == null || status == IMAGE_INCOMPATIBLE) {
+ if (vi != null) { vi.flush(); vi = null; }
+ vi = gc.createCompatibleVolatileImage(100, 100);
+ continue;
+ }
+ if (status == IMAGE_RESTORED) {
+ Graphics gg = vi.getGraphics();
+ gg.setColor(Color.green);
+ gg.fillRect(0, 0, vi.getWidth(), vi.getHeight());
+ break;
+ }
+ }
+ g.drawImage(vi, getInsets().left, getInsets().top, null);
+ if (++count1 > LOOP_THRESHOLD) {
+ System.err.println("Infinite loop detected: count1="+count1);
+ failed = true;
+ return;
+ }
+ } while (vi.contentsLost());
+ }
+
+ public static void main(String[] args) {
+ latch = new CountDownLatch(1);
+ InfiniteValidationLoopTest t1 = new InfiniteValidationLoopTest();
+ t1.pack();
+ t1.setSize(200, 200);
+ t1.setVisible(true);
+ try { latch.await(); } catch (InterruptedException ex) {}
+ t1.dispose();
+
+ latch = new CountDownLatch(1);
+ t1 = new InfiniteValidationLoopTest();
+ t1.pack();
+ t1.setSize(50, 50);
+ t1.setVisible(true);
+ try { latch.await(); } catch (InterruptedException ex) {}
+ t1.dispose();
+
+ if (failed) {
+ throw new
+ RuntimeException("Failed: infinite validattion loop detected");
+ }
+ System.out.println("Test PASSED");
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/OnScreenRenderingResizeTest/OnScreenRenderingResizeTest.java b/jdk/test/sun/java2d/DirectX/OnScreenRenderingResizeTest/OnScreenRenderingResizeTest.java
new file mode 100644
index 00000000000..65439f90961
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/OnScreenRenderingResizeTest/OnScreenRenderingResizeTest.java
@@ -0,0 +1,209 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 6664068 6666931
+ * @summary Tests that resizing a window to which a tight loop is rendering
+ * doesn't produce artifacts or crashes
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm OnScreenRenderingResizeTest
+ * @run main/othervm -Dsun.java2d.d3d=false OnScreenRenderingResizeTest
+ */
+
+import java.awt.AWTException;
+import java.awt.Color;
+import java.awt.EventQueue;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.Insets;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.Robot;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+import java.io.File;
+import java.io.IOException;
+import javax.imageio.ImageIO;
+
+public class OnScreenRenderingResizeTest {
+
+ private static volatile boolean done = false;
+ private static volatile boolean nocheck = false;
+
+ private static final int FRAME_W = 256;
+ private static final int FRAME_H = 256;
+ private static final int IMAGE_W = 128;
+ private static final int IMAGE_H = 128;
+ private static long RUN_TIME = 1000*20;
+
+ private static final Color renderColor = Color.green;
+ private static final Color bgColor = Color.white;
+
+ public static void main(String[] args) {
+
+ for (String arg : args) {
+ if ("-inf".equals(arg)) {
+ System.err.println("Test will run indefinitely");
+ RUN_TIME = Long.MAX_VALUE;
+ } else if ("-nocheck".equals(arg)) {
+ System.err.println("Test will not check rendering results");
+ nocheck = true;
+ } else {
+ System.err.println("Usage: OnScreenRenderingResizeTest [-inf][-nocheck]");
+ }
+ }
+
+ BufferedImage output =
+ new BufferedImage(IMAGE_W, IMAGE_H, BufferedImage.TYPE_INT_RGB);
+ output.setAccelerationPriority(0.0f);
+ Graphics g = output.getGraphics();
+ g.setColor(renderColor);
+ g.fillRect(0, 0, output.getWidth(), output.getHeight());
+
+ final Frame frame = new Frame("OnScreenRenderingResizeTest") {
+ public void paint(Graphics g) {}
+ public void update(Graphics g) {}
+ };
+ frame.setBackground(bgColor);
+ frame.pack();
+ frame.setSize(FRAME_W, FRAME_H);
+ frame.addWindowListener(new WindowAdapter() {
+ public void windowClosing(WindowEvent e) {
+ done = true;
+ }
+ });
+ try {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ frame.setVisible(true);
+ }
+ });
+ // wait for Vista's effects to complete
+ Thread.sleep(2000);
+ } catch (Exception ex) {
+ ex.printStackTrace();
+ }
+
+ GraphicsConfiguration gc = frame.getGraphicsConfiguration();
+ int maxW = gc.getBounds().width /2;
+ int maxH = gc.getBounds().height/2;
+ int minW = frame.getWidth();
+ int minH = frame.getHeight();
+ int incW = 10, incH = 10, cnt = 0;
+ Robot robot = null;
+ if (!nocheck && gc.getColorModel().getPixelSize() > 8) {
+ try {
+ robot = new Robot();
+ } catch (AWTException ex) {
+ System.err.println("Robot creation failed, continuing.");
+ }
+ } else {
+ System.err.println("No screen rendering checks.");
+ }
+
+ VolatileImage vi = gc.createCompatibleVolatileImage(512, 512);
+ vi.validate(gc);
+
+ long timeStarted = System.currentTimeMillis();
+ while (!done && (System.currentTimeMillis() - timeStarted) < RUN_TIME) {
+
+ if (++cnt > 100) {
+ int w = frame.getWidth() + incW;
+ int h = frame.getHeight() + incH;
+ if (w < minW || w > maxW ) {
+ incW = -incW;
+ }
+ if (h < minH || h > maxH ) {
+ incH = -incH;
+ }
+ frame.setSize(w, h);
+ cnt = 0;
+ }
+
+ // try to put the device into non-default state, for example,
+ // this operation below will set the transform
+ vi.validate(gc);
+ Graphics2D vig = (Graphics2D)vi.getGraphics();
+ vig.rotate(30.0f, vi.getWidth()/2, vi.getHeight()/2);
+ vig.drawImage(output, 0, 0,
+ vi.getWidth(), vi.getHeight(), null);
+
+ Insets in = frame.getInsets();
+ frame.getGraphics().drawImage(output, in.left, in.top, null);
+ if (cnt == 90 && robot != null) {
+ // area where we blitted to should be either white or green
+ Point p = frame.getLocationOnScreen();
+ p.move(in.left+10, in.top+10);
+ BufferedImage bi =
+ robot.createScreenCapture(
+ new Rectangle(p.x, p.y, IMAGE_W/2, IMAGE_H/2));
+ int accepted1[] = { Color.white.getRGB(), Color.green.getRGB()};
+ checkBI(bi, accepted1);
+
+ // the are where we didn't render should stay white
+ p = frame.getLocationOnScreen();
+ p.move(in.left, in.top+IMAGE_H+5);
+ bi = robot.createScreenCapture(
+ new Rectangle(p.x, p.y,
+ frame.getWidth()-in.left-in.right,
+ frame.getHeight()-in.top-in.bottom-5-IMAGE_H));
+ int accepted2[] = { Color.white.getRGB() };
+ checkBI(bi, accepted1);
+ }
+
+ Thread.yield();
+ }
+ frame.dispose();
+ System.out.println("Test Passed");
+ }
+
+ private static void checkBI(BufferedImage bi, int accepted[]) {
+ for (int x = 0; x < bi.getWidth(); x++) {
+ for (int y = 0; y < bi.getHeight(); y++) {
+ int pix = bi.getRGB(x, y);
+ boolean found = false;
+ for (int acc : accepted) {
+ if (pix == acc) {
+ found = true;
+ break;
+ }
+ }
+ if (!found) {
+ try {
+ String name = "OnScreenRenderingResizeTest.png";
+ ImageIO.write(bi, "png", new File(name));
+ System.out.println("Screen shot file: " + name);
+ } catch (IOException ex) {}
+
+ throw new
+ RuntimeException("Test failed at " + x + "-" + y +
+ " rgb=0x" + Integer.toHexString(pix));
+ }
+ }
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/OverriddenInsetsTest/OverriddenInsetsTest.java b/jdk/test/sun/java2d/DirectX/OverriddenInsetsTest/OverriddenInsetsTest.java
new file mode 100644
index 00000000000..658f42ac939
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/OverriddenInsetsTest/OverriddenInsetsTest.java
@@ -0,0 +1,150 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6694230
+ * @summary Tests that components overriding getInsets paint correctly
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm OverriddenInsetsTest
+ * @run main/othervm -Dsun.java2d.opengl=True OverriddenInsetsTest
+ */
+
+import java.awt.Color;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsEnvironment;
+import java.awt.Insets;
+import java.awt.Panel;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.Robot;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.image.BufferedImage;
+import java.io.File;
+import java.io.IOException;
+import java.util.concurrent.CountDownLatch;
+import javax.imageio.ImageIO;
+
+public class OverriddenInsetsTest {
+
+ public static final Insets INSETS1 = new Insets(25,25,0,0);
+ public static final Insets INSETS2 = new Insets(100,100,0,0);
+ static final CountDownLatch lock = new CountDownLatch(1);
+ static boolean failed = false;
+
+ public static void main(String[] args) {
+
+ if (GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getDefaultScreenDevice().getDefaultConfiguration().
+ getColorModel().getPixelSize() < 16)
+ {
+ System.out.println("<16 bit mode detected, test passed");
+ }
+
+ final Frame f = new Frame("OverriddenInsetsTest");
+ f.setSize(260,260);
+
+ f.addWindowListener(new WindowAdapter() {
+ public void windowClosing(WindowEvent e) {
+ f.setVisible(false);
+ System.exit(0);
+ }
+ });
+
+ f.setBackground(Color.gray);
+ Panel p1 = new Panel() {
+ public Insets getInsets() {
+ return INSETS1;
+ }
+ };
+ p1.setLayout(null);
+ p1.setSize(250, 250);
+
+ Panel p = new Panel(){
+ @Override
+ public Insets getInsets() {
+ return INSETS2;
+ }
+
+ public void paint(Graphics g) {
+ // make sure Vista is done with its effects
+ try {
+ Thread.sleep(2000);
+ } catch (InterruptedException ex) {}
+ g.setColor(Color.red);
+ g.drawRect(0,0,getWidth()-1,getHeight()-1 );
+ g.setColor(Color.blue);
+ g.fillRect(0,0,getWidth()/2,getHeight()/2);
+
+ Point p = getLocationOnScreen();
+ try {
+ Robot r = new Robot();
+ BufferedImage bi =
+ r.createScreenCapture(new
+ Rectangle(p.x, p.y, getWidth()/2, getHeight()/2));
+ for (int y = 0; y < bi.getHeight(); y++) {
+ for (int x = 0; x < bi.getWidth(); x++) {
+ if (bi.getRGB(x, y) != Color.blue.getRGB()) {
+ failed = true;
+ System.err.printf("Test failed at %d %d c=%x\n",
+ x, y, bi.getRGB(x, y));
+ String name = "OverriddenInsetsTest_res.png";
+ try {
+ ImageIO.write(bi, "png", new File(name));
+ System.out.println("Dumped res to: "+name);
+ } catch (IOException e) {}
+ return;
+ }
+ }
+ }
+ } catch (Exception e) {
+ failed = true;
+ } finally {
+ lock.countDown();
+ }
+ }
+ };
+ p.setSize(200, 200);
+
+ p1.add(p);
+ p.setLocation(50, 50);
+ f.add(p1);
+ f.setVisible(true);
+
+ try {
+ lock.await();
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ if (args.length <= 0 || !"-show".equals(args[0])) {
+ f.dispose();
+ }
+
+ if (failed) {
+ throw new RuntimeException("Test FAILED.");
+ }
+ System.out.println("Test PASSED");
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/RenderingToCachedGraphicsTest/RenderingToCachedGraphicsTest.java b/jdk/test/sun/java2d/DirectX/RenderingToCachedGraphicsTest/RenderingToCachedGraphicsTest.java
new file mode 100644
index 00000000000..e5f44ab0d46
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/RenderingToCachedGraphicsTest/RenderingToCachedGraphicsTest.java
@@ -0,0 +1,174 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6648018 6652662
+ * @summary Verifies that rendering to a cached onscreen Graphics works
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm RenderingToCachedGraphicsTest
+ * @run main/othervm -Dsun.java2d.d3d=false RenderingToCachedGraphicsTest
+ */
+import java.awt.Canvas;
+import java.awt.Color;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsEnvironment;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.Robot;
+import java.awt.Toolkit;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.image.BufferStrategy;
+import java.awt.image.BufferedImage;
+import static java.awt.image.VolatileImage.*;
+import java.awt.image.VolatileImage;
+import java.io.File;
+import java.util.concurrent.CountDownLatch;
+import javax.imageio.ImageIO;
+
+public class RenderingToCachedGraphicsTest extends Frame {
+ private static volatile boolean failed = false;
+ private static volatile CountDownLatch latch;
+ private Graphics cachedGraphics;
+ private Canvas renderCanvas;
+
+ public RenderingToCachedGraphicsTest() {
+ super("Test starts in 2 seconds");
+ renderCanvas = new Canvas() {
+ @Override
+ public void paint(Graphics g) {
+ if (getWidth() < 100 || getHeight() < 100) {
+ repaint();
+ return;
+ }
+ // wait for a bit so that Vista's Window manager's animation
+ // effects on window's appearance are completed (6652662)
+ try { Thread.sleep(2000); } catch (InterruptedException ex) {}
+
+ try {
+ runTest();
+ } catch (Throwable t) {
+ failed = true;
+ } finally {
+ latch.countDown();
+ }
+ }
+ @Override
+ public void update(Graphics g) {}
+ };
+
+ add("Center", renderCanvas);
+ }
+
+ private void runTest() {
+ // this will cause screen update manager to dump the accelerated surface
+ // for this canvas
+ renderCanvas.createBufferStrategy(2);
+ BufferStrategy bs = renderCanvas.getBufferStrategy();
+ do {
+ Graphics bsg = bs.getDrawGraphics();
+ bsg.setColor(Color.blue);
+ bsg.fillRect(0, 0,
+ renderCanvas.getWidth(), renderCanvas.getHeight());
+ } while (bs.contentsLost() || bs.contentsRestored());
+
+ // grab the "unaccelerated" onscreen surface
+ cachedGraphics = renderCanvas.getGraphics();
+ cachedGraphics.setColor(Color.red);
+ cachedGraphics.fillRect(0, 0, getWidth(), getHeight());
+
+ bs.dispose();
+ bs = null;
+ // now the update manager should be able to accelerate onscreen
+ // rendering to it again
+
+ cachedGraphics.setColor(Color.green);
+ // this causes restoration of the new accelerated onscreen surface
+ // (it is created in "lost" state)
+ cachedGraphics.fillRect(0, 0,
+ renderCanvas.getWidth(),
+ renderCanvas.getHeight());
+ Toolkit.getDefaultToolkit().sync();
+ // and now we should be able to render to it
+ cachedGraphics.fillRect(0, 0,
+ renderCanvas.getWidth(),
+ renderCanvas.getHeight());
+ Toolkit.getDefaultToolkit().sync();
+
+ Robot robot = null;
+ try {
+ robot = new Robot();
+ } catch (Exception e) {
+ e.printStackTrace();
+ failed = true;
+ return;
+ }
+
+ Point p = renderCanvas.getLocationOnScreen();
+ Rectangle r = new Rectangle(p.x, p.y,
+ renderCanvas.getWidth(),
+ renderCanvas.getHeight());
+ BufferedImage bi = robot.createScreenCapture(r);
+ for (int y = 0; y < bi.getHeight(); y++) {
+ for (int x = 0; x < bi.getWidth(); x++) {
+ if (bi.getRGB(x, y) != Color.green.getRGB()) {
+ System.err.println("Colors mismatch!");
+ String name = "RenderingToCachedGraphicsTest.png";
+ try {
+ ImageIO.write(bi, "png", new File(name));
+ System.err.println("Dumped grabbed image to: "+name);
+ } catch (Exception e) {}
+ failed = true;
+ return;
+ }
+ }
+ }
+ }
+
+ public static void main(String[] args) {
+ int depth = GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getDefaultScreenDevice().getDefaultConfiguration().
+ getColorModel().getPixelSize();
+ if (depth < 16) {
+ System.out.println("Test PASSED (depth < 16bit)");
+ return;
+ }
+
+ latch = new CountDownLatch(1);
+ RenderingToCachedGraphicsTest t1 = new RenderingToCachedGraphicsTest();
+ t1.pack();
+ t1.setSize(300, 300);
+ t1.setVisible(true);
+
+ try { latch.await(); } catch (InterruptedException ex) {}
+ t1.dispose();
+
+ if (failed) {
+ throw new
+ RuntimeException("Failed: rendering didn't show up");
+ }
+ System.out.println("Test PASSED");
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/StrikeDisposalCrashTest/StrikeDisposalCrashTest.java b/jdk/test/sun/java2d/DirectX/StrikeDisposalCrashTest/StrikeDisposalCrashTest.java
new file mode 100644
index 00000000000..6d0e0794105
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/StrikeDisposalCrashTest/StrikeDisposalCrashTest.java
@@ -0,0 +1,109 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/**
+ * @test
+ * @bug 6705443
+ * @summary tests that we don't crash during exit if font strikes were disposed
+ * during the lifetime of the application
+ *
+ * @run main/othervm -Dsun.java2d.font.reftype=weak StrikeDisposalCrashTest
+ * @run main/othervm -Dsun.java2d.font.reftype=weak -Dsun.java2d.noddraw=true StrikeDisposalCrashTest
+ * @run main/othervm -Dsun.java2d.font.reftype=weak -Dsun.java2d.opengl=True StrikeDisposalCrashTest
+ */
+
+import java.awt.Font;
+import java.awt.Frame;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
+import java.awt.RenderingHints;
+import java.awt.Toolkit;
+import java.awt.image.VolatileImage;
+
+public class StrikeDisposalCrashTest {
+
+ public static void main(String[] args) {
+ System.setProperty("sun.java2d.font.reftype", "weak");
+
+ GraphicsDevice gd[] =
+ GraphicsEnvironment.getLocalGraphicsEnvironment().getScreenDevices();
+
+ Frame frames[] = new Frame[gd.length];
+ for (int i = 0; i < frames.length; i++) {
+ GraphicsConfiguration gc = gd[i].getDefaultConfiguration();
+ Frame f = new Frame("Frame on "+gc, gc);
+ f.setSize(100, 100);
+ f.setLocation(gc.getBounds().x, gc.getBounds().y);
+ f.pack();
+ frames[i] = f;
+ }
+
+ Font f1 = new Font("Dialog", Font.PLAIN, 10);
+ Font f2 = new Font("Dialog", Font.ITALIC, 12);
+
+ for (int i = 0; i < frames.length/2; i++) {
+ // making sure the glyphs are cached in the accel. cache on
+ // one frame, then the other
+ renderText(frames[i], f1);
+ renderText(frames[frames.length -1 - i], f1);
+
+ // and now the other way around, with different glyphs
+ renderText(frames[frames.length -1 - i], f2);
+ renderText(frames[i], f2);
+ }
+
+ // try to force strike disposal (note that we have to use
+ // -Dsun.java2d.font.reftype=weak to facilitate the disposal)
+
+ System.gc();
+ System.runFinalization();
+ System.gc();
+ System.runFinalization();
+
+ for (Frame f : frames) {
+ f.dispose();
+ }
+
+ System.err.println("Exiting. If the test crashed after this it FAILED");
+ }
+
+ private static final String text =
+ "The quick brown fox jumps over the lazy dog 1234567890";
+ private static void renderText(Frame frame, Font f1) {
+ VolatileImage vi = frame.createVolatileImage(256, 32);
+ vi.validate(frame.getGraphicsConfiguration());
+
+ Graphics2D g = vi.createGraphics();
+ g.setFont(f1);
+ g.drawString(text, 0, vi.getHeight()/2);
+ g.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
+ RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
+ g.drawString(text, 0, vi.getHeight()/2);
+ g.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
+ RenderingHints.VALUE_TEXT_ANTIALIAS_LCD_HRGB);
+ g.drawString(text, 0, vi.getHeight()/2);
+ Toolkit.getDefaultToolkit().sync();
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/SwingOnScreenScrollingTest/SwingOnScreenScrollingTest.java b/jdk/test/sun/java2d/DirectX/SwingOnScreenScrollingTest/SwingOnScreenScrollingTest.java
new file mode 100644
index 00000000000..5f2d5cb056d
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/SwingOnScreenScrollingTest/SwingOnScreenScrollingTest.java
@@ -0,0 +1,146 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 6630702
+ * @summary Tests that scrolling after paint() is performed correctly.
+ * This is really only applicable to Vista
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm SwingOnScreenScrollingTest
+ * run main/othervm -Dsun.java2d.opengl=True SwingOnScreenScrollingTest
+ */
+
+import java.awt.AWTException;
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.EventQueue;
+import java.awt.Graphics;
+import java.awt.GraphicsEnvironment;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.Robot;
+import java.awt.image.BufferedImage;
+import java.io.File;
+import java.lang.reflect.InvocationTargetException;
+import javax.imageio.ImageIO;
+import javax.swing.JFrame;
+import javax.swing.JPanel;
+import javax.swing.JScrollPane;
+
+public class SwingOnScreenScrollingTest extends JPanel {
+
+ static JScrollPane pane;
+ static SwingOnScreenScrollingTest test;
+
+ public SwingOnScreenScrollingTest() {
+ }
+
+ public static void main(String[] args) {
+ int size = GraphicsEnvironment.
+ getLocalGraphicsEnvironment().
+ getDefaultScreenDevice().
+ getDefaultConfiguration().getColorModel().getPixelSize();
+ if (size < 16) {
+ System.err.println("<16 bit display mode detected. Test PASSED");
+ return;
+ }
+
+ final JFrame f = new JFrame("SwingOnScreenScrollingTest");
+ try {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ test = new SwingOnScreenScrollingTest();
+ pane = new JScrollPane(test);
+ f.add(pane);
+ f.pack();
+ f.setSize(100, 200);
+ f.setVisible(true);
+ }
+ });
+ } catch (InvocationTargetException ex) {
+ ex.printStackTrace();
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ try {
+ Thread.sleep(500);
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ EventQueue.invokeLater(new Runnable() {
+ public void run() {
+ BufferedImage bi;
+ bi = new BufferedImage(100, 300,
+ BufferedImage.TYPE_INT_RGB);
+ Graphics gg = bi.getGraphics();
+ test.paint(gg);
+ for (int y = 80; y < 200; y +=10) {
+ test.scrollRectToVisible(new Rectangle(0, y, 100, 100));
+ try {
+ Thread.sleep(200);
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ }
+ Point p = pane.getViewport().getLocationOnScreen();
+ Robot r = null;
+ try {
+ r = new Robot();
+ } catch (AWTException ex) {
+ throw new RuntimeException(ex);
+ }
+ bi = r.createScreenCapture(new Rectangle(p.x+5, p.y+5, 30, 30));
+ for (int y = 0; y < bi.getHeight(); y++) {
+ for (int x = 0; x < bi.getHeight(); x++) {
+ int rgb = bi.getRGB(x, y);
+ if (bi.getRGB(x, y) != Color.red.getRGB()) {
+ System.err.printf("Test Failed at (%d,%d) c=0x%x\n",
+ x, y, rgb);
+ try {
+ String name =
+ "SwingOnScreenScrollingTest_out.png";
+ ImageIO.write(bi, "png", new File(name));
+ System.err.println("Wrote grabbed image to "+
+ name);
+ } catch (Throwable ex) {}
+ throw new RuntimeException("Test failed");
+ }
+ }
+ }
+ System.out.println("Test PASSED.");
+ f.dispose();
+ }
+ });
+ }
+
+ protected void paintComponent(Graphics g) {
+ g.setColor(Color.green);
+ g.fillRect(0, 0, getWidth(), 100);
+ g.setColor(Color.red);
+ g.fillRect(0, 100, getWidth(), getHeight()-100);
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(100, 300);
+ }
+}
diff --git a/jdk/test/sun/java2d/DirectX/TransformedPaintTest/TransformedPaintTest.java b/jdk/test/sun/java2d/DirectX/TransformedPaintTest/TransformedPaintTest.java
new file mode 100644
index 00000000000..3929a59c869
--- /dev/null
+++ b/jdk/test/sun/java2d/DirectX/TransformedPaintTest/TransformedPaintTest.java
@@ -0,0 +1,219 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 6689025
+ * @summary Tests that transformed Paints are rendered correctly
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @run main/othervm TransformedPaintTest
+ * @run main/othervm -Dsun.java2d.opengl=True TransformedPaintTest
+ */
+
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.EventQueue;
+import java.awt.GradientPaint;
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsEnvironment;
+import java.awt.LinearGradientPaint;
+import java.awt.MultipleGradientPaint.CycleMethod;
+import java.awt.Paint;
+import java.awt.RadialGradientPaint;
+import java.awt.TexturePaint;
+import java.awt.geom.Rectangle2D;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+import java.io.File;
+import java.io.IOException;
+import java.lang.reflect.InvocationTargetException;
+import javax.imageio.ImageIO;
+import javax.swing.JFrame;
+import javax.swing.JPanel;
+
+public class TransformedPaintTest {
+
+ private static enum PaintType {
+ COLOR,
+ GRADIENT,
+ LINEAR_GRADIENT,
+ RADIAL_GRADIENT,
+ TEXTURE
+ };
+
+ private static final int CELL_SIZE = 100;
+ private static final int R_WIDTH = 3 * CELL_SIZE;
+ private static final int R_HEIGHT = PaintType.values().length * CELL_SIZE;
+
+ private Paint createPaint(PaintType type, int startx, int starty,
+ int w, int h)
+ {
+ // make sure that the blue color doesn't show up when filling a
+ // w by h rect
+ w++; h++;
+
+ int endx = startx + w;
+ int endy = starty + h;
+ Rectangle2D.Float r = new Rectangle2D.Float(startx, starty, w, h);
+ switch (type) {
+ case COLOR: return Color.red;
+ case GRADIENT: return
+ new GradientPaint(startx, starty, Color.red,
+ endx, endy, Color.green);
+ case LINEAR_GRADIENT: return
+ new LinearGradientPaint(startx, starty, endx, endy,
+ new float[] { 0.0f, 0.999f, 1.0f },
+ new Color[] { Color.red, Color.green, Color.blue });
+ case RADIAL_GRADIENT: return
+ new RadialGradientPaint(startx, starty,
+ (float)Math.sqrt(w * w + h * h),
+ new float[] { 0.0f, 0.999f, 1.0f },
+ new Color[] { Color.red, Color.green, Color.blue },
+ CycleMethod.NO_CYCLE);
+ case TEXTURE: {
+ BufferedImage bi =
+ new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
+ Graphics2D g = (Graphics2D) bi.getGraphics();
+ g.setPaint(createPaint(PaintType.LINEAR_GRADIENT, 0, 0, w, h));
+ g.fillRect(0, 0, w, h);
+ return new TexturePaint(bi, r);
+ }
+ }
+ return Color.green;
+ }
+
+ private void renderLine(PaintType type, Graphics2D g,
+ int startx, int starty, int w, int h)
+ {
+ Paint p = createPaint(type, startx, starty, w, h);
+ g.setPaint(p);
+
+ // first, no transform
+ g.fillRect(startx, starty, w, h);
+
+ // translation only
+ g.translate(w, 0);
+ g.fillRect(startx, starty, w, h);
+ g.translate(-w, 0);
+
+ // complex transform
+ g.translate(startx + w*2, starty);
+ g.rotate(Math.toRadians(90), w/2, h/2);
+ g.translate(-startx, -starty);
+ g.fillRect(startx, starty, w, h);
+ }
+
+ private void render(Graphics2D g, int w, int h) {
+ int paintTypes = PaintType.values().length;
+ int ystep = h / paintTypes;
+ int y = 0;
+
+ for (PaintType type : PaintType.values()) {
+ renderLine(type, (Graphics2D)g.create(),
+ 0, y, h / paintTypes, h / paintTypes);
+ y += ystep;
+ }
+ }
+
+ private void checkBI(BufferedImage bi) {
+ for (int y = 0; y < bi.getHeight(); y++) {
+ for (int x = 0; x < bi.getWidth(); x++) {
+ if (bi.getRGB(x, y) == Color.blue.getRGB()) {
+ try {
+ String fileName = "TransformedPaintTest_res.png";
+ ImageIO.write(bi, "png", new File(fileName));
+ System.err.println("Dumped image to: " + fileName);
+ } catch (IOException ex) {}
+ throw new RuntimeException("Test failed, blue color found");
+ }
+ }
+ }
+ }
+
+ private void runTest() {
+ GraphicsConfiguration gc = GraphicsEnvironment.
+ getLocalGraphicsEnvironment().getDefaultScreenDevice().
+ getDefaultConfiguration();
+
+ if (gc.getColorModel().getPixelSize() < 16) {
+ System.out.println("8-bit desktop depth found, test passed");
+ return;
+ }
+
+ VolatileImage vi = gc.createCompatibleVolatileImage(R_WIDTH, R_HEIGHT);
+ BufferedImage bi = null;
+ do {
+ vi.validate(gc);
+ Graphics2D g = vi.createGraphics();
+ render(g, vi.getWidth(), vi.getHeight());
+ bi = vi.getSnapshot();
+ } while (vi.contentsLost());
+
+ checkBI(bi);
+ System.out.println("Test PASSED.");
+ }
+
+ private static void showFrame(final TransformedPaintTest t) {
+ JFrame f = new JFrame("TransformedPaintTest");
+ f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
+ final BufferedImage bi =
+ new BufferedImage(R_WIDTH, R_HEIGHT, BufferedImage.TYPE_INT_RGB);
+ JPanel p = new JPanel() {
+ @Override
+ protected void paintComponent(Graphics g) {
+ super.paintComponent(g);
+ Graphics2D g2d = (Graphics2D) g;
+ t.render(g2d, R_WIDTH, R_HEIGHT);
+ t.render(bi.createGraphics(), R_WIDTH, R_HEIGHT);
+ g2d.drawImage(bi, R_WIDTH + 5, 0, null);
+
+ g.setColor(Color.black);
+ g.drawString("Rendered to Back Buffer", 10, 20);
+ g.drawString("Rendered to BufferedImage", R_WIDTH + 15, 20);
+ }
+ };
+ p.setPreferredSize(new Dimension(2 * R_WIDTH + 5, R_HEIGHT));
+ f.add(p);
+ f.pack();
+ f.setVisible(true);
+ }
+
+ public static void main(String[] args) throws
+ InterruptedException, InvocationTargetException
+ {
+ boolean show = (args.length > 0 && "-show".equals(args[0]));
+
+ final TransformedPaintTest t = new TransformedPaintTest();
+ if (show) {
+ EventQueue.invokeAndWait(new Runnable() {
+ public void run() {
+ showFrame(t);
+ }
+ });
+ } else {
+ t.runTest();
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/GdiRendering/InsetClipping.java b/jdk/test/sun/java2d/GdiRendering/InsetClipping.java
new file mode 100644
index 00000000000..fbaf71956dd
--- /dev/null
+++ b/jdk/test/sun/java2d/GdiRendering/InsetClipping.java
@@ -0,0 +1,112 @@
+/*
+ * Copyright 2003-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/**
+ * @test
+ * @bug 4873505 6588884
+ * @author cheth
+ * @summary verifies that drawImage behaves the bounds of a complex
+ * clip shape. This was a problem with our GDI renderer on Windows, where
+ * we would ignore the window insets.
+ * @run main InsetClipping
+*/
+
+/**
+ * This test works by setting up a clip area that equals the visible area
+ * of the Frame. When we perform any rendering operation to that window,
+ * we should not see the results of the operation because they should be
+ * clipped out. We create an Image with one color (red) and use a
+ * different background fill color (blue). We fill the area with the
+ * background color, then set the clip, then draw the image; if we detect
+ * the image color at pixel (0, 0) then we did not clip correctly and the
+ * test fails.
+ */
+
+import java.awt.*;
+import java.awt.geom.*;
+import java.awt.image.*;
+
+
+public class InsetClipping extends Frame {
+ BufferedImage image;
+ Area area;
+ static boolean painted = false;
+ static Color imageColor = Color.red;
+ static Color fillColor = Color.blue;
+
+ public InsetClipping() {
+
+ image = new BufferedImage( 300, 300,BufferedImage.TYPE_INT_RGB);
+ Graphics g2 = image.createGraphics();
+ g2.setColor(imageColor);
+ g2.fillRect(0,0, 300,300);
+ }
+
+ public void paint(Graphics g) {
+ Insets insets = getInsets();
+ area = new Area( new Rectangle(0,0, getWidth(), getHeight()));
+ area.subtract(new Area(new Rectangle(insets.left, insets.top,
+ getWidth() - insets.right,
+ getHeight() - insets.bottom)));
+ g.setColor(fillColor);
+ g.fillRect(0, 0, getWidth(), getHeight());
+ g.setClip(area);
+ g.drawImage(image, 0, 0, null);
+ painted = true;
+ }
+
+ public static void main(String args[]) {
+ InsetClipping clipTest = new InsetClipping();
+ clipTest.setSize(300, 300);
+ clipTest.setVisible(true);
+ while (!painted) {
+ try {
+ Thread.sleep(100);
+ } catch (Exception e) {}
+ }
+ try {
+ Robot robot = new Robot();
+ Point clientLoc = clipTest.getLocationOnScreen();
+ Insets insets = clipTest.getInsets();
+ clientLoc.x += insets.left;
+ clientLoc.y += insets.top;
+ BufferedImage clientPixels =
+ robot.createScreenCapture(new Rectangle(clientLoc.x,
+ clientLoc.y,
+ clientLoc.x + 2,
+ clientLoc.y + 2));
+ try {
+ Thread.sleep(2000);
+ } catch (Exception e) {}
+ int pixelVal = clientPixels.getRGB(0, 0);
+ clipTest.dispose();
+ if ((new Color(pixelVal)).equals(fillColor)) {
+ System.out.println("Passed");
+ } else {
+ throw new Error("Failed: incorrect color in pixel (0, 0)");
+ }
+ } catch (Exception e) {
+ System.out.println("Problems creating Robot");
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/OpenGL/DrawBufImgOp.java b/jdk/test/sun/java2d/OpenGL/DrawBufImgOp.java
new file mode 100644
index 00000000000..321d8c0a5ea
--- /dev/null
+++ b/jdk/test/sun/java2d/OpenGL/DrawBufImgOp.java
@@ -0,0 +1,483 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 6514990
+ * @summary Verifies that calling
+ * Graphics2D.drawImage(BufferedImage, BufferedImageOp, x, y) to an
+ * OpenGL-accelerated destination produces the same results when performed
+ * in software via BufferedImageOp.filter().
+ * @run main/othervm -Dsun.java2d.opengl=True DrawBufImgOp -ignore
+ * @author campbelc
+ */
+
+import java.awt.*;
+import java.awt.image.*;
+import java.io.File;
+import javax.imageio.ImageIO;
+
+/**
+ * REMIND: This testcase was originally intended to automatically compare
+ * the results of the software BufferedImageOp implementations against
+ * the OGL-accelerated codepaths. However, there are just too many open
+ * bugs in the mediaLib-based codepaths (see below), which means that
+ * creating the reference image may cause crashes or exceptions,
+ * and even if we work around those cases using the "-ignore" flag,
+ * the visual results of the reference image are often buggy as well
+ * (so the comparison will fail even though the OGL results are correct).
+ * Therefore, for now we will run the testcase with the "-ignore" flag
+ * but without the "-compare" flag, so at least it will be checking for
+ * any exceptions/crashes in the OGL code. When we fix all of the
+ * outstanding bugs with the software codepaths, we can remove the
+ * "-ignore" flag and maybe even restore the "-compare" flag. In the
+ * meantime, it stil functions well as a manual testcase (with either
+ * the "-show" or "-dump" options).
+ */
+public class DrawBufImgOp extends Canvas {
+
+ private static final int TESTW = 600;
+ private static final int TESTH = 500;
+ private static boolean done;
+
+ /*
+ * If true, skips tests that are known to trigger bugs (which in
+ * turn may cause crashes, exceptions, or other artifacts).
+ */
+ private static boolean ignore;
+
+ // Test both pow2 and non-pow2 sized images
+ private static final int[] srcSizes = { 32, 17 };
+ private static final int[] srcTypes = {
+ BufferedImage.TYPE_INT_RGB,
+ BufferedImage.TYPE_INT_ARGB,
+ BufferedImage.TYPE_INT_ARGB_PRE,
+ BufferedImage.TYPE_INT_BGR,
+ BufferedImage.TYPE_3BYTE_BGR,
+ BufferedImage.TYPE_4BYTE_ABGR,
+ BufferedImage.TYPE_USHORT_565_RGB,
+ BufferedImage.TYPE_BYTE_GRAY,
+ BufferedImage.TYPE_USHORT_GRAY,
+ };
+
+ private static final RescaleOp
+ rescale1band, rescale3band, rescale4band;
+ private static final LookupOp
+ lookup1bandbyte, lookup3bandbyte, lookup4bandbyte;
+ private static final LookupOp
+ lookup1bandshort, lookup3bandshort, lookup4bandshort;
+ private static final ConvolveOp
+ convolve3x3zero, convolve5x5zero, convolve7x7zero;
+ private static final ConvolveOp
+ convolve3x3noop, convolve5x5noop, convolve7x7noop;
+
+ static {
+ rescale1band = new RescaleOp(0.5f, 10.0f, null);
+ rescale3band = new RescaleOp(
+ new float[] { 0.6f, 0.4f, 0.6f },
+ new float[] { 10.0f, -3.0f, 5.0f },
+ null);
+ rescale4band = new RescaleOp(
+ new float[] { 0.6f, 0.4f, 0.6f, 0.9f },
+ new float[] { -1.0f, 5.0f, 3.0f, 1.0f },
+ null);
+
+ // REMIND: we should probably test non-zero offsets, but that
+ // would require massaging the source image data to avoid going
+ // outside the lookup table array bounds
+ int offset = 0;
+ {
+ byte invert[] = new byte[256];
+ byte halved[] = new byte[256];
+ for (int j = 0; j < 256 ; j++) {
+ invert[j] = (byte) (255-j);
+ halved[j] = (byte) (j / 2);
+ }
+ ByteLookupTable lut1 = new ByteLookupTable(offset, invert);
+ lookup1bandbyte = new LookupOp(lut1, null);
+ ByteLookupTable lut3 =
+ new ByteLookupTable(offset,
+ new byte[][] {invert, halved, invert});
+ lookup3bandbyte = new LookupOp(lut3, null);
+ ByteLookupTable lut4 =
+ new ByteLookupTable(offset,
+ new byte[][] {invert, halved, invert, halved});
+ lookup4bandbyte = new LookupOp(lut4, null);
+ }
+
+ {
+ short invert[] = new short[256];
+ short halved[] = new short[256];
+ for (int j = 0; j < 256 ; j++) {
+ invert[j] = (short) ((255-j) * 255);
+ halved[j] = (short) ((j / 2) * 255);
+ }
+ ShortLookupTable lut1 = new ShortLookupTable(offset, invert);
+ lookup1bandshort = new LookupOp(lut1, null);
+ ShortLookupTable lut3 =
+ new ShortLookupTable(offset,
+ new short[][] {invert, halved, invert});
+ lookup3bandshort = new LookupOp(lut3, null);
+ ShortLookupTable lut4 =
+ new ShortLookupTable(offset,
+ new short[][] {invert, halved, invert, halved});
+ lookup4bandshort = new LookupOp(lut4, null);
+ }
+
+ // 3x3 blur
+ float[] data3 = {
+ 0.1f, 0.1f, 0.1f,
+ 0.1f, 0.2f, 0.1f,
+ 0.1f, 0.1f, 0.1f,
+ };
+ Kernel k3 = new Kernel(3, 3, data3);
+
+ // 5x5 edge
+ float[] data5 = {
+ -1.0f, -1.0f, -1.0f, -1.0f, -1.0f,
+ -1.0f, -1.0f, -1.0f, -1.0f, -1.0f,
+ -1.0f, -1.0f, 24.0f, -1.0f, -1.0f,
+ -1.0f, -1.0f, -1.0f, -1.0f, -1.0f,
+ -1.0f, -1.0f, -1.0f, -1.0f, -1.0f,
+ };
+ Kernel k5 = new Kernel(5, 5, data5);
+
+ // 7x7 blur
+ float[] data7 = {
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f, 0.02f,
+ };
+ Kernel k7 = new Kernel(7, 7, data7);
+
+ convolve3x3zero = new ConvolveOp(k3, ConvolveOp.EDGE_ZERO_FILL, null);
+ convolve5x5zero = new ConvolveOp(k5, ConvolveOp.EDGE_ZERO_FILL, null);
+ convolve7x7zero = new ConvolveOp(k7, ConvolveOp.EDGE_ZERO_FILL, null);
+
+ convolve3x3noop = new ConvolveOp(k3, ConvolveOp.EDGE_NO_OP, null);
+ convolve5x5noop = new ConvolveOp(k5, ConvolveOp.EDGE_NO_OP, null);
+ convolve7x7noop = new ConvolveOp(k7, ConvolveOp.EDGE_NO_OP, null);
+ }
+
+ public void paint(Graphics g) {
+ synchronized (this) {
+ if (done) {
+ return;
+ }
+ }
+
+ VolatileImage vimg = createVolatileImage(TESTW, TESTH);
+ vimg.validate(getGraphicsConfiguration());
+
+ Graphics2D g2d = vimg.createGraphics();
+ renderTest(g2d);
+ g2d.dispose();
+
+ g.drawImage(vimg, 0, 0, null);
+
+ Toolkit.getDefaultToolkit().sync();
+
+ synchronized (this) {
+ done = true;
+ notifyAll();
+ }
+ }
+
+ /*
+ * foreach source image size (once with pow2, once with non-pow2)
+ *
+ * foreach BufferedImage type
+ *
+ * RescaleOp (1 band)
+ * RescaleOp (3 bands, if src has 3 bands)
+ * RescaleOp (4 bands, if src has 4 bands)
+ *
+ * foreach LookupTable type (once with ByteLUT, once with ShortLUT)
+ * LookupOp (1 band)
+ * LookupOp (3 bands, if src has 3 bands)
+ * LookupOp (4 bands, if src has 4 bands)
+ *
+ * foreach edge condition (once with ZERO_FILL, once with EDGE_NO_OP)
+ * ConvolveOp (3x3)
+ * ConvolveOp (5x5)
+ * ConvolveOp (7x7)
+ */
+ private void renderTest(Graphics2D g2d) {
+ g2d.setColor(Color.white);
+ g2d.fillRect(0, 0, TESTW, TESTH);
+
+ int yorig = 2;
+ int xinc = 34;
+ int yinc = srcSizes[0] + srcSizes[1] + 2 + 2;
+
+ for (int srcType : srcTypes) {
+ int y = yorig;
+
+ for (int srcSize : srcSizes) {
+ int x = 2;
+ System.out.printf("type=%d size=%d\n", srcType, srcSize);
+
+ BufferedImage srcImg = makeSourceImage(srcSize, srcType);
+ ColorModel srcCM = srcImg.getColorModel();
+
+ // RescaleOp
+ g2d.drawImage(srcImg, rescale1band, x, y);
+ x += xinc;
+ // REMIND: 3-band RescaleOp.filter() throws IAE for images
+ // that contain an alpha channel (bug to be filed)
+ if (srcCM.getNumColorComponents() == 3 &&
+ !(ignore && srcCM.hasAlpha()))
+ {
+ g2d.drawImage(srcImg, rescale3band, x, y);
+ }
+ x += xinc;
+ if (srcCM.getNumComponents() == 4) {
+ g2d.drawImage(srcImg, rescale4band, x, y);
+ }
+ x += xinc;
+
+ // LookupOp
+ // REMIND: Our LUTs are only 256 elements long, so won't
+ // currently work with USHORT_GRAY data
+ if (srcType != BufferedImage.TYPE_USHORT_GRAY) {
+ g2d.drawImage(srcImg, lookup1bandbyte, x, y);
+ x += xinc;
+ if (srcCM.getNumColorComponents() == 3) {
+ g2d.drawImage(srcImg, lookup3bandbyte, x, y);
+ }
+ x += xinc;
+ if (srcCM.getNumComponents() == 4) {
+ g2d.drawImage(srcImg, lookup4bandbyte, x, y);
+ }
+ x += xinc;
+
+ // REMIND: LookupOp.createCompatibleDestImage() throws
+ // IAE for 3BYTE_BGR/4BYTE_ABGR (bug to be filed)
+ if (!(ignore &&
+ (srcType == BufferedImage.TYPE_3BYTE_BGR ||
+ srcType == BufferedImage.TYPE_4BYTE_ABGR)))
+ {
+ g2d.drawImage(srcImg, lookup1bandshort, x, y);
+ x += xinc;
+ // REMIND: 3-band LookupOp.filter() throws IAE for
+ // images that contain an alpha channel
+ // (bug to be filed)
+ if (srcCM.getNumColorComponents() == 3 &&
+ !(ignore && srcCM.hasAlpha()))
+ {
+ g2d.drawImage(srcImg, lookup3bandshort, x, y);
+ }
+ x += xinc;
+ if (srcCM.getNumComponents() == 4) {
+ g2d.drawImage(srcImg, lookup4bandshort, x, y);
+ }
+ x += xinc;
+ } else {
+ x += 3*xinc;
+ }
+ } else {
+ x += 6*xinc;
+ }
+
+ // ConvolveOp
+ // REMIND: ConvolveOp.filter() throws ImagingOpException
+ // for 3BYTE_BGR (see 4957775)
+ if (srcType != BufferedImage.TYPE_3BYTE_BGR) {
+ g2d.drawImage(srcImg, convolve3x3zero, x, y);
+ x += xinc;
+ g2d.drawImage(srcImg, convolve5x5zero, x, y);
+ x += xinc;
+ g2d.drawImage(srcImg, convolve7x7zero, x, y);
+ x += xinc;
+
+ g2d.drawImage(srcImg, convolve3x3noop, x, y);
+ x += xinc;
+ g2d.drawImage(srcImg, convolve5x5noop, x, y);
+ x += xinc;
+ g2d.drawImage(srcImg, convolve7x7noop, x, y);
+ x += xinc;
+ } else {
+ x += 6*xinc;
+ }
+
+ y += srcSize + 2;
+ }
+
+ yorig += yinc;
+ }
+ }
+
+ private BufferedImage makeSourceImage(int size, int type) {
+ int s2 = size/2;
+ BufferedImage img = new BufferedImage(size, size, type);
+ Graphics2D g2d = img.createGraphics();
+ g2d.setComposite(AlphaComposite.Src);
+ g2d.setColor(Color.orange);
+ g2d.fillRect(0, 0, size, size);
+ g2d.setColor(Color.red);
+ g2d.fillRect(0, 0, s2, s2);
+ g2d.setColor(Color.green);
+ g2d.fillRect(s2, 0, s2, s2);
+ g2d.setColor(Color.blue);
+ g2d.fillRect(0, s2, s2, s2);
+ g2d.setColor(new Color(255, 255, 0, 128));
+ g2d.fillRect(s2, s2, s2, s2);
+ g2d.setColor(Color.pink);
+ g2d.fillOval(s2-3, s2-3, 6, 6);
+ g2d.dispose();
+ return img;
+ }
+
+ public BufferedImage makeReferenceImage() {
+ BufferedImage img = new BufferedImage(TESTW, TESTH,
+ BufferedImage.TYPE_INT_RGB);
+ Graphics2D g2d = img.createGraphics();
+ renderTest(g2d);
+ g2d.dispose();
+ return img;
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(TESTW, TESTH);
+ }
+
+ private static void compareImages(BufferedImage refImg,
+ BufferedImage testImg,
+ int tolerance)
+ {
+ int x1 = 0;
+ int y1 = 0;
+ int x2 = refImg.getWidth();
+ int y2 = refImg.getHeight();
+
+ for (int y = y1; y < y2; y++) {
+ for (int x = x1; x < x2; x++) {
+ Color expected = new Color(refImg.getRGB(x, y));
+ Color actual = new Color(testImg.getRGB(x, y));
+ if (!isSameColor(expected, actual, tolerance)) {
+ throw new RuntimeException("Test failed at x="+x+" y="+y+
+ " (expected="+expected+
+ " actual="+actual+
+ ")");
+ }
+ }
+ }
+ }
+
+ private static boolean isSameColor(Color c1, Color c2, int e) {
+ int r1 = c1.getRed();
+ int g1 = c1.getGreen();
+ int b1 = c1.getBlue();
+ int r2 = c2.getRed();
+ int g2 = c2.getGreen();
+ int b2 = c2.getBlue();
+ int rmin = Math.max(r2-e, 0);
+ int gmin = Math.max(g2-e, 0);
+ int bmin = Math.max(b2-e, 0);
+ int rmax = Math.min(r2+e, 255);
+ int gmax = Math.min(g2+e, 255);
+ int bmax = Math.min(b2+e, 255);
+ if (r1 >= rmin && r1 <= rmax &&
+ g1 >= gmin && g1 <= gmax &&
+ b1 >= bmin && b1 <= bmax)
+ {
+ return true;
+ }
+ return false;
+ }
+
+ public static void main(String[] args) throws Exception {
+ boolean show = false;
+ boolean dump = false;
+ boolean compare = false;
+
+ for (String arg : args) {
+ if (arg.equals("-show")) {
+ show = true;
+ } else if (arg.equals("-dump")) {
+ dump = true;
+ } else if (arg.equals("-compare")) {
+ compare = true;
+ } else if (arg.equals("-ignore")) {
+ ignore = true;
+ }
+ }
+
+ DrawBufImgOp test = new DrawBufImgOp();
+ Frame frame = new Frame();
+ frame.add(test);
+ frame.pack();
+ frame.setVisible(true);
+
+ // Wait until the component's been painted
+ synchronized (test) {
+ while (!done) {
+ try {
+ test.wait();
+ } catch (InterruptedException e) {
+ throw new RuntimeException("Failed: Interrupted");
+ }
+ }
+ }
+
+ GraphicsConfiguration gc = frame.getGraphicsConfiguration();
+ if (gc.getColorModel() instanceof IndexColorModel) {
+ System.out.println("IndexColorModel detected: " +
+ "test considered PASSED");
+ frame.dispose();
+ return;
+ }
+
+ // Grab the screen region
+ BufferedImage capture = null;
+ try {
+ Robot robot = new Robot();
+ Point pt1 = test.getLocationOnScreen();
+ Rectangle rect = new Rectangle(pt1.x, pt1.y, TESTW, TESTH);
+ capture = robot.createScreenCapture(rect);
+ } catch (Exception e) {
+ throw new RuntimeException("Problems creating Robot");
+ } finally {
+ if (!show) {
+ frame.dispose();
+ }
+ }
+
+ // Compare the images (allow for +/- 1 bit differences in color comps)
+ if (dump || compare) {
+ BufferedImage ref = test.makeReferenceImage();
+ if (dump) {
+ ImageIO.write(ref, "png",
+ new File("DrawBufImgOp.ref.png"));
+ ImageIO.write(capture, "png",
+ new File("DrawBufImgOp.cap.png"));
+ }
+ if (compare) {
+ test.compareImages(ref, capture, 1);
+ }
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/SunGraphics2D/DrawImageBilinear.java b/jdk/test/sun/java2d/SunGraphics2D/DrawImageBilinear.java
new file mode 100644
index 00000000000..5e8831862c8
--- /dev/null
+++ b/jdk/test/sun/java2d/SunGraphics2D/DrawImageBilinear.java
@@ -0,0 +1,205 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 5009033 6603000 6666362
+ * @summary Verifies that images transformed with bilinear filtering do not
+ * leave artifacts at the edges.
+ * @run main/othervm DrawImageBilinear
+ * @run main/othervm -Dsun.java2d.opengl=True DrawImageBilinear
+ * @author campbelc
+ */
+
+import java.awt.Canvas;
+import java.awt.Color;
+import java.awt.Component;
+import java.awt.Dimension;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.RenderingHints;
+import java.awt.Robot;
+import java.awt.Toolkit;
+import java.awt.image.BufferedImage;
+import java.awt.image.IndexColorModel;
+import java.awt.image.VolatileImage;
+
+public class DrawImageBilinear extends Canvas {
+
+ private static final int SIZE = 5;
+
+ private static boolean done;
+ private BufferedImage bimg1, bimg2;
+ private VolatileImage vimg;
+ private static volatile BufferedImage capture;
+ private static void doCapture(Component test) {
+ // Grab the screen region
+ try {
+ Robot robot = new Robot();
+ Point pt1 = test.getLocationOnScreen();
+ Rectangle rect =
+ new Rectangle(pt1.x, pt1.y, test.getWidth(), test.getHeight());
+ capture = robot.createScreenCapture(rect);
+ } catch (Exception e) {
+ e.printStackTrace();
+ }
+ }
+
+ private void renderPattern(Graphics g) {
+ g.setColor(Color.red);
+ g.fillRect(0, 0, SIZE, SIZE);
+ //g.setColor(Color.green);
+ //g.drawRect(0, 0, SIZE-1, SIZE-1);
+ g.dispose();
+ }
+
+ public void paint(Graphics g) {
+ Graphics2D g2d = (Graphics2D)g;
+
+ if (bimg1 == null) {
+ bimg1 = (BufferedImage)createImage(SIZE, SIZE);
+ bimg1.setAccelerationPriority(0.0f);
+ renderPattern(bimg1.createGraphics());
+
+ bimg2 = (BufferedImage)createImage(SIZE, SIZE);
+ renderPattern(bimg2.createGraphics());
+
+ vimg = createVolatileImage(SIZE, SIZE);
+ vimg.validate(getGraphicsConfiguration());
+ renderPattern(vimg.createGraphics());
+ }
+
+ do {
+ g2d.setColor(Color.white);
+ g2d.fillRect(0, 0, getWidth(), getHeight());
+
+ g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION,
+ RenderingHints.VALUE_INTERPOLATION_BILINEAR);
+
+ // first time will be a sw->surface blit
+ g2d.drawImage(bimg1, 10, 10, 40, 40, null);
+
+ // second time will be a texture->surface blit
+ g2d.drawImage(bimg2, 80, 10, 40, 40, null);
+ g2d.drawImage(bimg2, 80, 10, 40, 40, null);
+
+ // third time will be a pbuffer->surface blit
+ if (vimg.validate(getGraphicsConfiguration()) != VolatileImage.IMAGE_OK) {
+ renderPattern(vimg.createGraphics());
+ }
+ g2d.drawImage(vimg, 150, 10, 40, 40, null);
+
+ Toolkit.getDefaultToolkit().sync();
+ } while (vimg.contentsLost());
+
+ synchronized (this) {
+ if (!done) {
+ doCapture(this);
+ done = true;
+ }
+ notifyAll();
+ }
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(200, 100);
+ }
+
+ private static void testRegion(BufferedImage bi,
+ Rectangle affectedRegion)
+ {
+ int x1 = affectedRegion.x;
+ int y1 = affectedRegion.y;
+ int x2 = x1 + affectedRegion.width;
+ int y2 = y1 + affectedRegion.height;
+
+ for (int y = y1; y < y2; y++) {
+ for (int x = x1; x < x2; x++) {
+ int actual = bi.getRGB(x, y);
+ if ((actual != 0xfffe0000) && (actual != 0xffff0000)) {
+ throw new RuntimeException("Test failed at x="+x+" y="+y+
+ " (expected=0xffff0000"+
+ " actual=0x"+
+ Integer.toHexString(actual) +
+ ")");
+ }
+ }
+ }
+ }
+
+ public static void main(String[] args) {
+ boolean show = false;
+ for (String arg : args) {
+ if ("-show".equals(arg)) {
+ show = true;
+ }
+ }
+
+ DrawImageBilinear test = new DrawImageBilinear();
+ Frame frame = new Frame();
+ frame.add(test);
+ frame.pack();
+ frame.setVisible(true);
+
+ // Wait until the component's been painted
+ synchronized (test) {
+ while (!done) {
+ try {
+ test.wait();
+ } catch (InterruptedException e) {
+ throw new RuntimeException("Failed: Interrupted");
+ }
+ }
+ }
+
+ GraphicsConfiguration gc = frame.getGraphicsConfiguration();
+ if (gc.getColorModel() instanceof IndexColorModel) {
+ System.out.println("IndexColorModel detected: " +
+ "test considered PASSED");
+ frame.dispose();
+ return;
+ }
+
+ if (!show) {
+ frame.dispose();
+ }
+ if (capture == null) {
+ throw new RuntimeException("Failed: capture is null");
+ }
+
+ // Test background color
+ int pixel = capture.getRGB(5, 5);
+ if (pixel != 0xffffffff) {
+ throw new RuntimeException("Failed: Incorrect color for " +
+ "background");
+ }
+
+ // Test pixels
+ testRegion(capture, new Rectangle(10, 10, 40, 40));
+ testRegion(capture, new Rectangle(80, 10, 40, 40));
+ testRegion(capture, new Rectangle(150, 10, 40, 40));
+ }
+}
diff --git a/jdk/test/sun/java2d/SunGraphics2D/PolyVertTest.java b/jdk/test/sun/java2d/SunGraphics2D/PolyVertTest.java
new file mode 100644
index 00000000000..73138973b9e
--- /dev/null
+++ b/jdk/test/sun/java2d/SunGraphics2D/PolyVertTest.java
@@ -0,0 +1,661 @@
+/*
+ * Copyright 2002-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 4678208 4771101 6328481 6588884
+ * @summary verify the pixelization of degenerate polylines and polygons
+ * @run main PolyVertTest
+ * @run main/othervm -Dsun.java2d.d3d=True PolyVertTest -hwonly
+ * @run main/othervm -Dsun.java2d.opengl=True PolyVertTest -hwonly
+ */
+
+import java.awt.*;
+import java.awt.event.*;
+import java.awt.geom.*;
+import java.awt.image.*;
+
+public class PolyVertTest {
+ static int TESTWIDTH;
+ static int TESTHEIGHT;
+ static final int REG_TEST_WIDTH = 10;
+ static final int REG_TEST_HEIGHT = 10;
+ static final int FULL_TEST_WIDTH = 50;
+ static final int FULL_TEST_HEIGHT = 200;
+
+ static final int FRINGE = 2;
+ static final int GREEN = Color.green.getRGB();
+ static final int RED = Color.red.getRGB();
+
+ static BufferedImage refImg;
+ static BufferedImage errorImg;
+ static Graphics errorG;
+ static Component testCanvas;
+
+ static int totalbadpixels;
+ static int totalfuzzypixels;
+ static int numbadtests;
+ static int numfuzzytests;
+ static int numframes;
+ static int fuzzystarty;
+
+ static boolean counting;
+ static boolean showerrors;
+ static boolean showresults;
+ static boolean fringe;
+ static boolean forceerror;
+ static boolean fulltest = true;
+ static boolean hwonly;
+
+ static WindowListener windowCloser = new WindowAdapter() {
+ public void windowClosing(WindowEvent e) {
+ e.getWindow().hide();
+ if (--numframes <= 0) {
+ System.exit(0);
+ }
+ }
+ };
+
+ public PolyVertTest() {
+ /*
+ setBackground(Color.white);
+ setForeground(Color.black);
+ */
+ }
+
+ static int polypts[][][] = {
+ {
+ // void polygon (no points)
+ {}, {},
+ },
+ {
+ // one point
+ { 0 }, { 0 },
+ },
+ {
+ // two points
+ { 0, 5 }, { 0, 0 },
+ { 0, 0, 6, 1,
+ 10, 0, 6, 1,
+ 20, 0, 6, 1 },
+ { 0, 0, 6, 1,
+ 10, 0, 1, 1, 15, 0, 1, 1,
+ 20, 0, 1, 1, 25, 0, 1, 1 },
+ { 10, 0, 1, 1,
+ 20, 0, 1, 1 },
+ },
+ {
+ // open triangle
+ { 0, 5, 5 }, { 0, 0, 5 },
+
+ { 0, 0, 6, 1, 5, 1, 1, 5,
+
+ 10, 0, 6, 1, 15, 1, 1, 5, 11, 1, 1, 1,
+ 12, 2, 1, 1, 13, 3, 1, 1, 14, 4, 1, 1,
+
+ 20, 0, 6, 1, 25, 1, 1, 5, 21, 1, 1, 1,
+ 22, 2, 1, 1, 23, 3, 1, 1, 24, 4, 1, 1 },
+
+ { 0, 0, 6, 1, 5, 1, 1, 5,
+
+ 10, 0, 6, 1, 15, 1, 1, 5, 11, 1, 1, 1,
+ 12, 2, 1, 1, 13, 3, 1, 1, 14, 4, 1, 1,
+
+ 20, 0, 6, 1, 25, 1, 1, 5, 21, 1, 1, 1,
+ 22, 2, 1, 1, 23, 3, 1, 1, 24, 4, 1, 1 },
+
+ { 10, 0, 1, 1,
+ 20, 0, 1, 1 },
+ },
+ {
+ // closed triangle
+ { 0, 5, 5, 0 }, { 0, 0, 5, 0 },
+
+ { 0, 0, 6, 1, 5, 1, 1, 5, 1, 1, 1, 1,
+ 2, 2, 1, 1, 3, 3, 1, 1, 4, 4, 1, 1,
+
+ 10, 0, 6, 1, 15, 1, 1, 5, 11, 1, 1, 1,
+ 12, 2, 1, 1, 13, 3, 1, 1, 14, 4, 1, 1,
+
+ 20, 0, 6, 1, 25, 1, 1, 5, 21, 1, 1, 1,
+ 22, 2, 1, 1, 23, 3, 1, 1, 24, 4, 1, 1 },
+
+ { 1, 0, 5, 1, 5, 1, 1, 5, 1, 1, 1, 1,
+ 2, 2, 1, 1, 3, 3, 1, 1, 4, 4, 1, 1,
+
+ 10, 0, 6, 1, 15, 1, 1, 5, 11, 1, 1, 1,
+ 12, 2, 1, 1, 13, 3, 1, 1, 14, 4, 1, 1,
+
+ 20, 0, 6, 1, 25, 1, 1, 5, 21, 1, 1, 1,
+ 22, 2, 1, 1, 23, 3, 1, 1, 24, 4, 1, 1 },
+
+ { 0, 0, 1, 1,
+ 10, 0, 1, 1,
+ 20, 0, 1, 1 },
+ },
+ {
+ // empty line
+ { 0, 0 }, { 0, 0 },
+ { 0, 0, 1, 1,
+ 10, 0, 1, 1,
+ 20, 0, 1, 1 },
+ },
+ {
+ // empty triangle
+ { 0, 0, 0 }, { 0, 0, 0 },
+ { 0, 0, 1, 1,
+ 10, 0, 1, 1,
+ 20, 0, 1, 1 },
+ },
+ };
+
+ public static void render(Graphics2D g2d) {
+ g2d.setColor(Color.white);
+ g2d.fillRect(0, 0, TESTWIDTH, TESTHEIGHT);
+ g2d.setColor(Color.black);
+
+ if (forceerror) {
+ g2d.fillRect(2, 2, 2, 2);
+ g2d.fillRect(15, 5, 1, 1);
+ }
+
+ if (!fulltest) {
+ g2d.draw(new Rectangle2D.Double(5, 5, 0, 0));
+ return;
+ }
+
+ g2d.drawLine(10, 10, 10, 10);
+ g2d.draw(new Line2D.Double(20, 10, 20, 10));
+
+ g2d.drawRect(10, 20, 0, 0);
+ g2d.draw(new Rectangle2D.Double(20, 20, 0, 0));
+
+ g2d.setXORMode(Color.white);
+
+ g2d.drawLine(10, 30, 10, 30);
+ g2d.draw(new Line2D.Double(20, 30, 20, 30));
+
+ g2d.drawRect(10, 40, 0, 0);
+ g2d.draw(new Rectangle2D.Double(20, 40, 0, 0));
+
+ g2d.setPaintMode();
+
+ int y = 50;
+ for (int i = 0; i < polypts.length; i++) {
+ int data[][] = polypts[i];
+ int xpoints[] = data[0];
+ int ypoints[] = data[1];
+ int npoints = xpoints.length;
+ g2d.translate(10, y);
+ g2d.drawPolyline(xpoints, ypoints, npoints);
+ g2d.translate(10, 0);
+ g2d.drawPolygon(xpoints, ypoints, npoints);
+ g2d.translate(10, 0);
+ g2d.draw(new Polygon(xpoints, ypoints, npoints));
+ g2d.translate(-30, -y);
+ y += 10;
+ }
+ g2d.setXORMode(Color.white);
+ for (int i = 0; i < polypts.length; i++) {
+ int data[][] = polypts[i];
+ int xpoints[] = data[0];
+ int ypoints[] = data[1];
+ int npoints = xpoints.length;
+ g2d.translate(10, y);
+ g2d.drawPolyline(xpoints, ypoints, npoints);
+ g2d.translate(10, 0);
+ g2d.drawPolygon(xpoints, ypoints, npoints);
+ g2d.translate(10, 0);
+ g2d.draw(new Polygon(xpoints, ypoints, npoints));
+ g2d.translate(-30, -y);
+ y += 10;
+ }
+ g2d.setPaintMode();
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(500, 500);
+ }
+
+ public static void usage(int exitcode) {
+ System.err.println("usage: java PolyVertTest []*");
+ System.err.println(" -usage "+
+ "print this usage summary");
+ System.err.println(" -count "+
+ "run all tests and accumulate error counts");
+ System.err.println(" -forceerror "+
+ "force at least one error in each test");
+ System.err.println(" -fringe "+
+ "draw a yellow fringe around problems");
+ System.err.println(" -showerrors "+
+ "display results window for tests with problems");
+ System.err.println(" -showresults "+
+ "display results window for all tests");
+ System.err.println(" -quicktest "+
+ "only run test cases reported in bug reports");
+ System.err.println(" -fulltest "+
+ "run full suite of test cases for a 'unit test'");
+ System.err.println(" -hwonly "+
+ "only run tests for screen and VolatileImage");
+ System.exit(exitcode);
+ }
+
+ public static void main(String argv[]) {
+ for (int i = 0; i < argv.length; i++) {
+ String arg = argv[i];
+ if (arg.equalsIgnoreCase("-count")) {
+ counting = true;
+ } else if (arg.equalsIgnoreCase("-forceerror")) {
+ forceerror = true;
+ } else if (arg.equalsIgnoreCase("-fringe")) {
+ fringe = true;
+ } else if (arg.equalsIgnoreCase("-showerrors")) {
+ showerrors = true;
+ } else if (arg.equalsIgnoreCase("-showresults")) {
+ showresults = true;
+ } else if (arg.equalsIgnoreCase("-quicktest")) {
+ fulltest = false;
+ } else if (arg.equalsIgnoreCase("-fulltest")) {
+ fulltest = true;
+ } else if (arg.equalsIgnoreCase("-hwonly")) {
+ hwonly = true;
+ } else if (arg.equalsIgnoreCase("-usage")) {
+ usage(0);
+ } else {
+ System.err.println("unknown option: "+arg);
+ usage(1);
+ }
+ }
+
+ if (fulltest) {
+ TESTWIDTH = FULL_TEST_WIDTH;
+ TESTHEIGHT = FULL_TEST_HEIGHT;
+ } else {
+ TESTWIDTH = REG_TEST_WIDTH;
+ TESTHEIGHT = REG_TEST_HEIGHT;
+ }
+
+ // Prevents premature exit by the WindowAdapter if the user
+ // closes the last visible results window before we've
+ // finished our tests.
+ numframes++;
+
+ makeReferenceImage();
+ testScreen();
+ testVolatileImage();
+ if (!hwonly) {
+ testBufferedImage();
+ testOffscreen();
+ testCompatibleImages();
+ }
+ if (totalfuzzypixels > 0) {
+ System.err.println(totalfuzzypixels+" fuzzy pixels found in "+
+ numfuzzytests+" tests");
+ }
+ if (totalbadpixels > 0) {
+ throw new RuntimeException(totalbadpixels+" bad pixels found in "+
+ numbadtests+" tests");
+ }
+ System.out.println("Test done - no bad pixels found");
+
+ --numframes;
+
+ if (counting || ((showresults || showerrors) && numframes == 0)) {
+ System.exit(0);
+ }
+ }
+
+ public static void makeReferenceImage() {
+ refImg = new BufferedImage(TESTWIDTH, TESTHEIGHT,
+ BufferedImage.TYPE_INT_RGB);
+ Graphics g = refImg.getGraphics();
+
+ g.setColor(Color.white);
+ g.fillRect(0, 0, TESTWIDTH, TESTHEIGHT);
+
+ g.setColor(Color.black);
+
+ if (!fulltest) {
+ g.fillRect(5, 5, 1, 1);
+ g.dispose();
+ return;
+ }
+
+ for (int y = 10; y < 50; y += 10) {
+ g.fillRect(10, y, 1, 1);
+ g.fillRect(20, y, 1, 1);
+ }
+ int y = 50;
+ for (int i = 0; i < polypts.length; i++) {
+ int data[][] = polypts[i];
+ g.translate(10, y);
+ if (data.length > 2) {
+ int rectvals[] = data[2];
+ for (int j = 0; j < rectvals.length; j += 4) {
+ g.fillRect(rectvals[j+0], rectvals[j+1],
+ rectvals[j+2], rectvals[j+3]);
+ }
+ }
+ g.translate(-10, -y);
+ y += 10;
+ }
+ fuzzystarty = y;
+ for (int i = 0; i < polypts.length; i++) {
+ int data[][] = polypts[i];
+ g.translate(10, y);
+ if (data.length > 2) {
+ int rectvals[] = data.length > 3 ? data[3] : data[2];
+ for (int j = 0; j < rectvals.length; j += 4) {
+ g.fillRect(rectvals[j+0], rectvals[j+1],
+ rectvals[j+2], rectvals[j+3]);
+ }
+ }
+ g.translate(-10, -y);
+ y += 10;
+ }
+ g.dispose();
+ }
+
+ public static void initerrorbuf() {
+ if (errorImg == null) {
+ droperrorbuf();
+ errorImg = new BufferedImage(TESTWIDTH, TESTHEIGHT,
+ BufferedImage.TYPE_INT_RGB);
+ }
+ if (errorG == null) {
+ errorG = errorImg.getGraphics();
+ }
+ errorG.setColor(Color.green);
+ errorG.fillRect(0, 0, TESTWIDTH, TESTHEIGHT);
+ errorG.setColor(Color.red);
+ }
+
+ public static void droperrorbuf() {
+ errorImg = null;
+ if (errorG != null) {
+ errorG.dispose();
+ }
+ errorG = null;
+ }
+
+ public static void test(Image img, String name) {
+ Graphics2D g2d = (Graphics2D) img.getGraphics();
+ render(g2d);
+ g2d.dispose();
+ verify(img, name);
+ }
+
+ public static void test(BufferedImage bimg, String name) {
+ Graphics2D g2d = bimg.createGraphics();
+ render(g2d);
+ g2d.dispose();
+ verify(bimg, name);
+ }
+
+ public static void verify(Image img, String name) {
+ BufferedImage bimg;
+ if (img instanceof BufferedImage) {
+ bimg = (BufferedImage) img;
+ } else {
+ bimg = new BufferedImage(TESTWIDTH, TESTHEIGHT,
+ BufferedImage.TYPE_INT_RGB);
+ Graphics g = bimg.getGraphics();
+ g.drawImage(img, 0, 0, null);
+ g.dispose();
+ }
+ verify(bimg, name);
+ }
+
+ public static boolean isFuzzyPixel(int X, int Y) {
+ int ytrans = fuzzystarty;
+ if (!fulltest || Y < ytrans) {
+ return false;
+ }
+ for (int i = 0; i < polypts.length; i++) {
+ int data[][] = polypts[i];
+ if (data.length > 4) {
+ int rectvals[] = data[4];
+ for (int j = 0; j < rectvals.length; j += 4) {
+ int rectx = rectvals[j+0] + 10;
+ int recty = rectvals[j+1] + ytrans;
+ int rectw = rectvals[j+2];
+ int recth = rectvals[j+3];
+ if (X >= rectx && Y >= recty &&
+ X < rectx + rectw && Y < recty + recth)
+ {
+ return true;
+ }
+ }
+ }
+ ytrans += 10;
+ }
+ return false;
+ }
+
+ public static void verify(BufferedImage bimg, String name) {
+ int numbadpixels = 0;
+ int numfuzzypixels = 0;
+ for (int y = 0; y < TESTHEIGHT; y++) {
+ for (int x = 0; x < TESTWIDTH; x++) {
+ if (refImg.getRGB(x, y) != bimg.getRGB(x, y)) {
+ boolean isfuzzy = isFuzzyPixel(x, y);
+ if (showerrors || showresults) {
+ if (errorG == null) {
+ initerrorbuf();
+ }
+ errorG.setColor(isfuzzy ? Color.blue : Color.red);
+ errorG.fillRect(x, y, 1, 1);
+ } else if (!counting && !isfuzzy) {
+ throw new RuntimeException("Error at "+x+", "+y+
+ " while testing: "+name);
+ }
+ if (isfuzzy) {
+ numfuzzypixels++;
+ } else {
+ numbadpixels++;
+ }
+ }
+ }
+ }
+ if (numbadpixels > 0 || numfuzzypixels > 0) {
+ if (numbadpixels > 0) {
+ totalbadpixels += numbadpixels;
+ numbadtests++;
+ }
+ if (numfuzzypixels > 0) {
+ totalfuzzypixels += numfuzzypixels;
+ numfuzzytests++;
+ }
+ System.out.println(numbadpixels+" bad pixels and "+
+ numfuzzypixels+" questionable pixels "+
+ "found while testing "+name);
+ if (showerrors || showresults) {
+ displaydiffs(bimg, name);
+ }
+ } else if (showresults) {
+ if (errorG == null) {
+ initerrorbuf();
+ }
+ displaydiffs(bimg, name);
+ }
+ }
+
+ public static void displaydiffs(BufferedImage bimg, String name) {
+ if (fringe) {
+ errorG.setColor(Color.yellow);
+ for (int y = 0; y < TESTHEIGHT; y++) {
+ for (int x = 0; x < TESTWIDTH; x++) {
+ if (errorImg.getRGB(x, y) == RED) {
+ for (int iy = y-FRINGE; iy <= y+FRINGE; iy++) {
+ for (int ix = x-FRINGE; ix <= x+FRINGE; ix++) {
+ if (ix >= 0 && ix < TESTWIDTH &&
+ iy >= 0 && iy < TESTHEIGHT &&
+ errorImg.getRGB(ix, iy) == GREEN)
+ {
+ errorG.fillRect(ix, iy, 1, 1);
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ Frame f = new Frame("Results for "+name);
+ f.setLayout(new BorderLayout());
+ f.addWindowListener(windowCloser);
+ ++numframes;
+ Panel p = new Panel();
+ p.add(new ImageCanvas(bimg));
+ p.add(new ImageCanvas(errorImg));
+ p.add(new ImageCanvas(refImg));
+ f.add(p, "Center");
+ droperrorbuf();
+ f.pack();
+ f.show();
+ }
+
+ public static void testBufferedImage() {
+ testBufferedImage(BufferedImage.TYPE_INT_RGB, "IntXrgb");
+ testBufferedImage(BufferedImage.TYPE_INT_ARGB, "IntArgb");
+ testBufferedImage(BufferedImage.TYPE_3BYTE_BGR, "ThreeByte");
+ testBufferedImage(BufferedImage.TYPE_4BYTE_ABGR, "FourByte");
+ testBufferedImage(BufferedImage.TYPE_USHORT_555_RGB, "UShort555");
+ testBufferedImage(BufferedImage.TYPE_BYTE_GRAY, "ByteGray");
+ testBufferedImage(BufferedImage.TYPE_BYTE_INDEXED, "Indexed");
+ }
+
+ public static void testBufferedImage(int type, String name) {
+ BufferedImage bimg = new BufferedImage(TESTWIDTH, TESTHEIGHT, type);
+ test(bimg, name);
+ }
+
+ public static void testScreen() {
+ Frame f = new Frame("PolyVertTest");
+ TestCanvas child = new TestCanvas();
+ testCanvas = child;
+ f.add(child);
+ f.pack();
+ f.show();
+ BufferedImage bimg = child.getImage();
+ f.hide();
+ verify(bimg, "Screen");
+ }
+
+ public static void testOffscreen() {
+ Image img = testCanvas.createImage(TESTWIDTH, TESTHEIGHT);
+ test(img, "Offscreen");
+ }
+
+ public static void testCompatibleImages() {
+ GraphicsEnvironment genv =
+ GraphicsEnvironment.getLocalGraphicsEnvironment();
+ GraphicsDevice gdevs[] = genv.getScreenDevices();
+ for (int i = 0; i < gdevs.length; i++) {
+ testCompatibleImages(gdevs[i]);
+ }
+ }
+
+ public static void testCompatibleImages(GraphicsDevice gdev) {
+ GraphicsConfiguration gconfigs[] = gdev.getConfigurations();
+ for (int i = 0; i < gconfigs.length; i++) {
+ testCompatibleImages(gconfigs[i]);
+ }
+ }
+
+ public static void testCompatibleImages(GraphicsConfiguration gconfig) {
+ test(gconfig.createCompatibleImage(TESTWIDTH, TESTHEIGHT),
+ gconfig+".createCompat()");
+ test(gconfig.createCompatibleImage(TESTWIDTH, TESTHEIGHT,
+ Transparency.OPAQUE),
+ gconfig+".createCompat(OPAQUE)");
+ test(gconfig.createCompatibleImage(TESTWIDTH, TESTHEIGHT,
+ Transparency.BITMASK),
+ gconfig+".createCompat(BITMASK)");
+ test(gconfig.createCompatibleImage(TESTWIDTH, TESTHEIGHT,
+ Transparency.TRANSLUCENT),
+ gconfig+".createCompat(TRANSLUCENT)");
+ test(gconfig.createCompatibleVolatileImage(TESTWIDTH, TESTHEIGHT),
+ gconfig+".createCompatVolatile()");
+ }
+
+ public static void testVolatileImage() {
+ Image img = testCanvas.createVolatileImage(TESTWIDTH, TESTHEIGHT);
+ test(img, "Volatile");
+ }
+
+ public static class ImageCanvas extends Canvas {
+ BufferedImage bimg;
+
+ public ImageCanvas(BufferedImage bimg) {
+ this.bimg = bimg;
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(bimg.getWidth(), bimg.getHeight());
+ }
+
+ public void paint(Graphics g) {
+ g.drawImage(bimg, 0, 0, null);
+ }
+ }
+
+ public static class TestCanvas extends Canvas {
+ BufferedImage bimg;
+
+ public Dimension getPreferredSize() {
+ return new Dimension(TESTWIDTH, TESTHEIGHT);
+ }
+
+ public void paint(Graphics g) {
+ if (bimg != null ||
+ getWidth() < TESTWIDTH ||
+ getHeight() < TESTHEIGHT)
+ {
+ return;
+ }
+ render((Graphics2D) g);
+ Toolkit.getDefaultToolkit().sync();
+ Point p = getLocationOnScreen();
+ Rectangle r = new Rectangle(p.x, p.y, TESTWIDTH, TESTHEIGHT);
+ try {
+ bimg = new Robot().createScreenCapture(r);
+ } catch (AWTException e) {
+ e.printStackTrace();
+ }
+ synchronized (this) {
+ notifyAll();
+ }
+ }
+
+ public synchronized BufferedImage getImage() {
+ while (bimg == null) {
+ try {
+ wait();
+ } catch (InterruptedException e) {
+ return null;
+ }
+ }
+ return bimg;
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/SunGraphics2D/SimplePrimQuality.java b/jdk/test/sun/java2d/SunGraphics2D/SimplePrimQuality.java
new file mode 100644
index 00000000000..8b7b11b25f5
--- /dev/null
+++ b/jdk/test/sun/java2d/SunGraphics2D/SimplePrimQuality.java
@@ -0,0 +1,281 @@
+/*
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ * @test
+ * @bug 4832224 6322584 6328478 6328481 6322580 6588884 6587863
+ * @summary Verifies that the pixelization of simple primitives (drawLine,
+ * fillRect, drawRect, fill, draw) with the OGL pipeline enabled
+ * matches that produced by our software loops. (The primitives tested here
+ * are simple enough that the OGL results should match the software results
+ * exactly.) There is some overlap with PolyVertTest as we test both
+ * solid and XOR rendering here, but this testcase is a bit simpler and
+ * more appropriate for quick OGL testing. This test is also useful for
+ * comparing quality between our X11/GDI and software pipelines.
+ * @run main/othervm SimplePrimQuality
+ * @run main/othervm -Dsun.java2d.opengl=True SimplePrimQuality
+ * @author campbelc
+ */
+
+import java.awt.*;
+import java.awt.geom.*;
+import java.awt.image.*;
+import java.io.File;
+import java.io.IOException;
+import javax.imageio.ImageIO;
+
+public class SimplePrimQuality extends Canvas {
+
+ private static final int SIZE = 300;
+ private static boolean done;
+ private static boolean testVI;
+ private static volatile BufferedImage capture;
+ private static void doCapture(Component test) {
+ // Grab the screen region
+ try {
+ Robot robot = new Robot();
+ Point pt1 = test.getLocationOnScreen();
+ Rectangle rect =
+ new Rectangle(pt1.x, pt1.y, test.getWidth(), test.getHeight());
+ capture = robot.createScreenCapture(rect);
+ } catch (Exception e) {
+ e.printStackTrace();
+ }
+ }
+
+ private static final int[][] rpts = {
+ {2, 0, 0, 0},
+ {12, 0, 1, 0},
+ {22, 0, 0, 1},
+ {32, 0, 1, 1},
+ {42, 0, 2, 1},
+ {52, 0, 1, 2},
+ {62, 0, 2, 2},
+ {72, 0, 5, 5},
+ {82, 0, 10, 10},
+ {97, 0, 15, 15},
+ };
+
+ private void drawLine(Graphics2D g, int x, int y, int dx, int dy) {
+ g.drawLine(x, y, x + dx, y + dy);
+ }
+
+ private void drawLines(Graphics2D g, int s) {
+ drawLine(g, 2, 0, 0, 0);
+ drawLine(g, 12, 0, 0, s);
+ drawLine(g, 22, 0, s, 0);
+ drawLine(g, 32, 0, s, s);
+ drawLine(g, 42, 0, 0, -s);
+ drawLine(g, 52, 0, -s, 0);
+ drawLine(g, 62, 0, -s, -s);
+ drawLine(g, 72, 0, -s, s);
+ drawLine(g, 82, 0, s, -s);
+ }
+
+ private void fillRects(Graphics2D g) {
+ for (int i = 0; i < rpts.length; i++) {
+ g.fillRect(rpts[i][0], rpts[i][1], rpts[i][2], rpts[i][3]);
+ }
+ }
+
+ private void drawRects(Graphics2D g) {
+ for (int i = 0; i < rpts.length; i++) {
+ g.drawRect(rpts[i][0], rpts[i][1], rpts[i][2], rpts[i][3]);
+ }
+ }
+
+ private void fillOvals(Graphics2D g) {
+ for (int i = 0; i < rpts.length; i++) {
+ // use fill() instead of fillOval(), since the former is more
+ // likely to be consistent with our software loops when the
+ // OGL pipeline cannot be enabled
+ g.fill(new Ellipse2D.Float(rpts[i][0], rpts[i][1],
+ rpts[i][2], rpts[i][3]));
+ }
+ }
+
+ private void drawOvals(Graphics2D g) {
+ for (int i = 0; i < rpts.length; i++) {
+ // use draw() instead of drawOval(), since the former is more
+ // likely to be consistent with our software loops when the
+ // OGL pipeline cannot be enabled
+ g.draw(new Ellipse2D.Float(rpts[i][0], rpts[i][1],
+ rpts[i][2], rpts[i][3]));
+ }
+ }
+
+ private void renderShapes(Graphics2D g) {
+ // drawLine tests...
+ g.translate(0, 5);
+ drawLines(g, 1);
+ g.translate(0, 10);
+ drawLines(g, 4);
+
+ // fillRect tests...
+ g.translate(0, 10);
+ fillRects(g);
+
+ // drawRect tests...
+ g.translate(0, 20);
+ drawRects(g);
+
+ // fillOval tests...
+ g.translate(0, 20);
+ fillOvals(g);
+
+ // drawOval tests...
+ g.translate(0, 20);
+ drawOvals(g);
+ }
+
+ private void renderTest(Graphics2D g, int w, int h) {
+ // on the left side, render the shapes in solid mode
+ g.setColor(Color.black);
+ g.fillRect(0, 0, w, h);
+ g.setColor(Color.green);
+ renderShapes(g);
+
+ // on the right side, render the shapes in XOR mode
+ g.setTransform(AffineTransform.getTranslateInstance(SIZE/2, 0));
+ g.setXORMode(Color.black);
+ renderShapes(g);
+ g.setTransform(AffineTransform.getTranslateInstance(SIZE/2, 0));
+ renderShapes(g);
+ }
+
+ public void paint(Graphics g) {
+
+ Graphics2D g2d = (Graphics2D)g;
+ renderTest(g2d, SIZE, SIZE);
+
+ Toolkit.getDefaultToolkit().sync();
+
+ synchronized (this) {
+ if (!done) {
+ doCapture(this);
+ done = true;
+ }
+ notifyAll();
+ }
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(SIZE, SIZE);
+ }
+
+ public static void main(String[] args) {
+ boolean show = false;
+ for (String arg : args) {
+ if (arg.equals("-testvi")) {
+ System.out.println("Testing VolatileImage, not screen");
+ testVI = true;
+ } else if (arg.equals("-show")) {
+ show = true;
+ }
+ }
+
+ SimplePrimQuality test = new SimplePrimQuality();
+ Frame frame = new Frame();
+ frame.add(test);
+ frame.pack();
+ frame.setVisible(true);
+
+ // Wait until the component's been painted
+ synchronized (test) {
+ while (!done) {
+ try {
+ test.wait();
+ } catch (InterruptedException e) {
+ throw new RuntimeException("Failed: Interrupted");
+ }
+ }
+ }
+
+ // REMIND: We will allow this test to pass silently on Windows
+ // (when OGL is not enabled) until we fix the GDI pipeline so that
+ // its stroked/filled GeneralPaths match our software loops (see
+ // 6322554). This check should be removed when 6322554 is fixed.
+ GraphicsConfiguration gc = frame.getGraphicsConfiguration();
+ if (gc.getClass().getSimpleName().startsWith("Win")) {
+ System.out.println("GDI pipeline detected: " +
+ "test considered PASSED");
+ frame.dispose();
+ return;
+ }
+
+
+ if (testVI) {
+ // render to a VI instead of the screen
+ VolatileImage vi = frame.createVolatileImage(SIZE, SIZE);
+ do {
+ vi.validate(frame.getGraphicsConfiguration());
+ Graphics2D g1 = vi.createGraphics();
+ test.renderTest(g1, SIZE, SIZE);
+ g1.dispose();
+ capture = vi.getSnapshot();
+ } while (vi.contentsLost());
+ frame.dispose();
+ }
+
+ if (!show) {
+ frame.dispose();
+ }
+ if (capture == null) {
+ throw new RuntimeException("Error capturing the rendering");
+ }
+
+ // Create reference image
+ int w = SIZE, h = SIZE;
+ BufferedImage refimg = new BufferedImage(w, h,
+ BufferedImage.TYPE_INT_RGB);
+ Graphics2D g = refimg.createGraphics();
+ test.renderTest(g, w, h);
+ g.dispose();
+
+ // Test pixels
+ for (int y = 0; y < h; y++) {
+ for (int x = 0; x < w; x++) {
+ int actual = capture.getRGB(x, y);
+ int expected = refimg.getRGB(x, y);
+ if (actual != expected) {
+ String expectedName = "SimplePrimQuality_expected.png";
+ String actualName = "SimplePrimQuality_actual.png";
+ try {
+ System.out.println("Writing expected image to: "+
+ expectedName);
+ ImageIO.write(refimg, "png", new File(expectedName));
+ System.out.println("Writing actual image to: "+
+ actualName);
+ ImageIO.write(capture, "png", new File(actualName));
+ } catch (IOException ex) {}
+ throw new RuntimeException("Test failed at x="+x+" y="+y+
+ " (expected="+
+ Integer.toHexString(expected) +
+ " actual="+
+ Integer.toHexString(actual) +
+ ")");
+ }
+ }
+ }
+ }
+}
diff --git a/jdk/test/sun/java2d/SunGraphics2D/SourceClippingBlitTest/SourceClippingBlitTest.java b/jdk/test/sun/java2d/SunGraphics2D/SourceClippingBlitTest/SourceClippingBlitTest.java
new file mode 100644
index 00000000000..3bd1109bb62
--- /dev/null
+++ b/jdk/test/sun/java2d/SunGraphics2D/SourceClippingBlitTest/SourceClippingBlitTest.java
@@ -0,0 +1,295 @@
+/*
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/*
+ @test
+ @bug 6244574
+ @bug 6258142
+ @bug 6395165
+ @bug 6588884
+ @summary Tests that source is clipped correctly when blitting
+ different types of images to the screen
+ @author Dmitri.Trembovetski: area=Graphics2D
+ @run main SourceClippingBlitTest
+*/
+
+import java.awt.AWTException;
+import java.awt.Canvas;
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.GraphicsConfiguration;
+import java.awt.Image;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.Robot;
+import java.awt.Toolkit;
+import java.awt.Transparency;
+import java.awt.event.ComponentAdapter;
+import java.awt.event.ComponentEvent;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+
+public class SourceClippingBlitTest extends Canvas {
+ static final int TESTW = 300;
+ static final int TESTH = 300;
+ static final int IMAGEW = 50;
+ static final int IMAGEH = 50;
+
+ static final Rectangle IMAGE_BOUNDS = new Rectangle(0, 0, IMAGEW, IMAGEH);
+ static Robot robot;
+ static private boolean showErrors;
+
+ private static final Object lock = new Object();
+ private static volatile boolean done = false;
+
+ BufferedImage grabbedBI;
+
+ public static void main(String[] args) {
+ // allow user to override the properties if needed
+ if (System.getProperty("sun.java2d.pmoffscreen") == null) {
+ System.setProperty("sun.java2d.pmoffscreen", "true");
+ }
+
+ if (args.length > 0 && args[0].equals("-showerrors")) {
+ showErrors = true;
+ }
+
+ try {
+ robot = new Robot();
+ } catch (AWTException e) {
+ throw new RuntimeException(e);
+ }
+
+ Frame f = new Frame(SourceClippingBlitTest.class.getName());
+ final SourceClippingBlitTest test = new SourceClippingBlitTest();
+ f.add(test);
+ f.addWindowListener(new WindowAdapter() {
+ public void windowActivated(WindowEvent e) {
+ if (!done) {
+ test.runTests();
+ }
+ }
+
+ });
+ f.pack();
+ f.setLocation(100, 100);
+ f.setVisible(true);
+ synchronized (lock) {
+ while (!done) {
+ try {
+ lock.wait();
+ } catch (InterruptedException ex) {
+ ex.printStackTrace();
+ }
+ }
+ }
+ if (!showErrors) {
+ f.dispose();
+ }
+ }
+
+ public Dimension getPreferredSize() {
+ return new Dimension(TESTW, TESTH);
+ }
+
+ public void paint(Graphics g) {
+ if (showErrors && done && grabbedBI != null) {
+ g.drawImage(grabbedBI, 0, 0, null);
+ }
+ }
+
+ public void runTests() {
+ GraphicsConfiguration gc = getGraphicsConfiguration();
+ for (Image srcIm :
+ new Image[] {
+ getBufferedImage(gc, IMAGEW, IMAGEH,
+ BufferedImage.TYPE_INT_RGB, true),
+ getBufferedImage(gc, IMAGEW, IMAGEH,
+ BufferedImage.TYPE_INT_RGB, false),
+ // commented out due to 6593406
+// getBMImage(gc, IMAGEW, IMAGEH),
+// getBufferedImage(gc, IMAGEW, IMAGEH,
+// BufferedImage.TYPE_INT_ARGB, true),
+// getBufferedImage(gc, IMAGEW, IMAGEH,
+// BufferedImage.TYPE_INT_ARGB, false),
+ getVImage(gc, IMAGEW, IMAGEH),
+ })
+ {
+ System.out.println("Testing source: " + srcIm);
+ // wiggle the source and dest rectangles
+ try {
+ for (int locationVar = -10; locationVar < 20; locationVar += 10)
+ {
+ for (int sizeVar = -10; sizeVar < 20; sizeVar += 10) {
+ Rectangle srcRect = (Rectangle)IMAGE_BOUNDS.clone();
+ srcRect.translate(locationVar, locationVar);
+ srcRect.grow(sizeVar, sizeVar);
+
+ Rectangle dstRect =
+ new Rectangle(sizeVar, sizeVar,
+ srcRect.width, srcRect.height);
+ System.out.println("testing blit rect src: " + srcRect);
+ System.out.println(" dst: " + dstRect);
+ render(getGraphics(), srcIm, srcRect, dstRect);
+ test(srcRect, dstRect);
+ }
+ }
+ System.out.println("Test passed.");
+ } finally {
+ synchronized (lock) {
+ done = true;
+ lock.notifyAll();
+ }
+ }
+ }
+ }
+
+ public void render(Graphics g, Image image,
+ Rectangle srcRect, Rectangle dstRect)
+ {
+ int w = getWidth();
+ int h = getHeight();
+ g.setColor(Color.green);
+ g.fillRect(0, 0, w, h);
+
+ int bltWidth = srcRect.width;
+ int bltHeight = srcRect.height;
+ VolatileImage vi = null;
+ if (image instanceof VolatileImage) {
+ vi = (VolatileImage)image;
+ }
+ do {
+ if (vi != null) {
+ GraphicsConfiguration gc = getGraphicsConfiguration();
+ if (vi.validate(gc) != VolatileImage.IMAGE_OK) {
+ initImage(gc, vi);
+ }
+ }
+ g.drawImage(image,
+ dstRect.x, dstRect.y,
+ dstRect.x + bltWidth, dstRect.y + bltHeight,
+ srcRect.x, srcRect.y,
+ srcRect.x + bltWidth, srcRect.y + bltHeight,
+ Color.red,
+ null);
+ } while (vi != null && vi.contentsLost());
+ }
+
+
+ public void test(Rectangle srcRect, Rectangle dstRect) {
+ int w = getWidth();
+ int h = getHeight();
+ Toolkit.getDefaultToolkit().sync();
+ Point p = getLocationOnScreen();
+ grabbedBI = robot.createScreenCapture(new Rectangle(p.x, p.y, w, h));
+
+ // calculate the destination rectangle
+ Rectangle srcBounds = srcRect.intersection(IMAGE_BOUNDS);
+ int trX = dstRect.x - srcRect.x;
+ int trY = dstRect.y - srcRect.y;
+ Rectangle newDstRect = (Rectangle)dstRect.clone();
+ newDstRect.translate(-trX, -trY);
+ Rectangle.intersect(newDstRect, srcBounds, newDstRect);
+ newDstRect.translate(trX, trY);
+ Rectangle.intersect(newDstRect, new Rectangle(0, 0, w, h), newDstRect);
+
+ System.out.println("calculated dest rect:" + newDstRect);
+
+ // we do implicit clipping of the destination surface
+ // by only checking pixels within its bounds
+ for (int y = 0; y < h; y++) {
+ for (int x = 0; x < w; x++) {
+ int rgb = 0;
+ if (newDstRect.contains(x, y)) {
+ rgb = Color.red.getRGB();
+ } else {
+ rgb = Color.green.getRGB();
+ }
+ if (grabbedBI.getRGB(x, y) != rgb) {
+ String msg1 = "Test failed at x="+x+" y="+y;
+ System.out.println(msg1);
+ System.out.println(" expected: "+Integer.toHexString(rgb)+
+ " got:"+Integer.toHexString(grabbedBI.getRGB(x, y)));
+ throw new RuntimeException(msg1);
+ }
+ }
+ }
+ System.out.println("subtest passed");
+ }
+
+ static VolatileImage dstImage;
+ static void initImage(GraphicsConfiguration gc, Image image) {
+ Graphics g = image.getGraphics();
+ g.setColor(Color.RED);
+ int w = image.getWidth(null);
+ int h = image.getHeight(null);
+ g.fillRect(0, 0, w, h);
+ g.dispose();
+
+ // need to 'accelerate' the image
+ if (dstImage == null) {
+ dstImage =
+ gc.createCompatibleVolatileImage(TESTW, TESTH,
+ Transparency.OPAQUE);
+ }
+ dstImage.validate(gc);
+ g = dstImage.getGraphics();
+ g.drawImage(image, 0, 0, null);
+ g.drawImage(image, 0, 0, null);
+ g.drawImage(image, 0, 0, null);
+ }
+
+ static VolatileImage getVImage(GraphicsConfiguration gc,
+ int w, int h)
+ {
+ VolatileImage image =
+ gc.createCompatibleVolatileImage(w, h, Transparency.OPAQUE);
+ image.validate(gc);
+ initImage(gc, image);
+ return image;
+ }
+
+ static Image getBMImage(GraphicsConfiguration gc,
+ int w, int h)
+ {
+ Image image =
+ gc.createCompatibleImage(w, h, Transparency.BITMASK);
+ initImage(gc, image);
+ return image;
+ }
+
+ static Image getBufferedImage(GraphicsConfiguration gc,
+ int w, int h, int type, boolean acceleratable)
+ {
+ BufferedImage image = new BufferedImage(w, h, type);
+ if (!acceleratable) {
+ image.setAccelerationPriority(0.0f);
+ }
+ initImage(gc, image);
+ return image;
+ }
+}
diff --git a/jdk/test/sun/java2d/X11SurfaceData/SharedMemoryPixmapsTest/SharedMemoryPixmapsTest.java b/jdk/test/sun/java2d/X11SurfaceData/SharedMemoryPixmapsTest/SharedMemoryPixmapsTest.java
new file mode 100644
index 00000000000..0b85955a26d
--- /dev/null
+++ b/jdk/test/sun/java2d/X11SurfaceData/SharedMemoryPixmapsTest/SharedMemoryPixmapsTest.java
@@ -0,0 +1,156 @@
+/*
+ * Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+import java.awt.AWTException;
+import java.awt.Color;
+import java.awt.Component;
+import java.awt.Dimension;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.Robot;
+import java.awt.Toolkit;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+
+/*
+ *
+ *
+ * Tests that the use of shared memory pixmaps isn't broken:
+ * create a VolatileImage, fill it with red color, copy it to the screen
+ * make sure the pixels on the screen are red.
+ *
+ * Note that we force the use of shared memory pixmaps in the shell script.
+ *
+ * @author Dmitri.Trembovetski
+ */
+
+public class SharedMemoryPixmapsTest {
+ static final int IMAGE_SIZE = 100;
+ static boolean show = false;
+ final Frame testFrame;
+ /** Creates a new instance of SharedMemoryPixmapsTest */
+ public SharedMemoryPixmapsTest() {
+ testFrame = new Frame("SharedMemoryPixmapsTest");
+ testFrame.add(new TestComponent());
+ testFrame.pack();
+ testFrame.setVisible(true);
+ }
+
+ public static void main(String[] args) {
+ for (String s : args) {
+ if ("-show".equals(s)) {
+ show = true;
+ } else {
+ System.err.println("Usage: SharedMemoryPixmapsTest [-show]");
+ }
+ }
+ new SharedMemoryPixmapsTest();
+ }
+
+ private class TestComponent extends Component {
+ VolatileImage vi = null;
+ boolean tested = false;
+
+ void initVI() {
+ int res;
+ if (vi == null) {
+ res = VolatileImage.IMAGE_INCOMPATIBLE;
+ } else {
+ res = vi.validate(getGraphicsConfiguration());
+ }
+ if (res == VolatileImage.IMAGE_INCOMPATIBLE) {
+ if (vi != null) vi.flush();
+ vi = createVolatileImage(IMAGE_SIZE, IMAGE_SIZE);
+ vi.validate(getGraphicsConfiguration());
+ res = VolatileImage.IMAGE_RESTORED;
+ }
+ if (res == VolatileImage.IMAGE_RESTORED) {
+ Graphics vig = vi.getGraphics();
+ vig.setColor(Color.red);
+ vig.fillRect(0, 0, vi.getWidth(), vi.getHeight());
+ vig.dispose();
+ }
+ }
+
+ @Override
+ public synchronized void paint(Graphics g) {
+ do {
+ g.setColor(Color.green);
+ g.fillRect(0, 0, getWidth(), getHeight());
+
+ initVI();
+ g.drawImage(vi, 0, 0, null);
+ } while (vi.contentsLost());
+
+ Toolkit.getDefaultToolkit().sync();
+ if (!tested) {
+ if (testRendering()) {
+ System.err.println("Test Passed");
+ } else {
+ System.err.println("Test Failed");
+ }
+ tested = true;
+ }
+ if (!show) {
+ testFrame.setVisible(false);
+ testFrame.dispose();
+ }
+ }
+
+ private boolean testRendering() throws RuntimeException {
+ Robot r = null;
+ try {
+ r = new Robot();
+ } catch (AWTException ex) {
+ ex.printStackTrace();
+ throw new RuntimeException("Can't create Robot");
+ }
+ Point p = getLocationOnScreen();
+ BufferedImage b =
+ r.createScreenCapture(new Rectangle(p, getPreferredSize()));
+ for (int y = 0; y < b.getHeight(); y++) {
+ for (int x = 0; x < b.getWidth(); x++) {
+ if (b.getRGB(x, y) != Color.red.getRGB()) {
+ System.err.println("Incorrect pixel" + " at "
+ + x + "x" + y + " : " +
+ Integer.toHexString(b.getRGB(x, y)));
+ if (show) {
+ return false;
+ }
+ System.err.println("Test Failed");
+ System.exit(1);
+ }
+ }
+ }
+ return true;
+ }
+
+ @Override
+ public Dimension getPreferredSize() {
+ return new Dimension(IMAGE_SIZE, IMAGE_SIZE);
+ }
+ }
+
+}
diff --git a/jdk/test/sun/java2d/X11SurfaceData/SharedMemoryPixmapsTest/SharedMemoryPixmapsTest.sh b/jdk/test/sun/java2d/X11SurfaceData/SharedMemoryPixmapsTest/SharedMemoryPixmapsTest.sh
new file mode 100644
index 00000000000..2bc9fd10c29
--- /dev/null
+++ b/jdk/test/sun/java2d/X11SurfaceData/SharedMemoryPixmapsTest/SharedMemoryPixmapsTest.sh
@@ -0,0 +1,51 @@
+#
+# Copyright 2005-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+# CA 95054 USA or visit www.sun.com if you need additional information or
+# have any questions.
+#
+
+#!/bin/sh
+# @test
+# @bug 6363434 6588884
+# @summary Verify that shared memory pixmaps are not broken
+# by filling a VolatileImage with red color and copying it
+# to the screen.
+# Note that we force the use of shared memory pixmaps.
+# @author Dmitri.Trembovetski
+
+echo "TESTJAVA=${TESTJAVA}"
+echo "TESTSRC=${TESTSRC}"
+echo "TESTCLASSES=${TESTCLASSES}"
+cd ${TESTSRC}
+${TESTJAVA}/bin/javac -d ${TESTCLASSES} SharedMemoryPixmapsTest.java
+cd ${TESTCLASSES}
+
+NO_J2D_DGA=true
+J2D_PIXMAPS=shared
+export NO_J2D_DGA J2D_PIXMAPS
+
+${TESTJAVA}/bin/java SharedMemoryPixmapsTest
+
+if [ $? -ne 0 ]; then
+ echo "Test failed!"
+ exit 1
+fi
+
+exit 0
diff --git a/jdk/test/sun/java2d/pipe/MutableColorTest/MutableColorTest.java b/jdk/test/sun/java2d/pipe/MutableColorTest/MutableColorTest.java
new file mode 100644
index 00000000000..30e7f949fe4
--- /dev/null
+++ b/jdk/test/sun/java2d/pipe/MutableColorTest/MutableColorTest.java
@@ -0,0 +1,169 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+
+/**
+ * @test
+ * @bug 6613860 6691934
+ * @summary Tests that the pipelines can handle (in somewhat limited
+ * manner) mutable Colors
+ *
+ * @run main/othervm MutableColorTest
+ * @run main/othervm -Dsun.java2d.noddraw=true MutableColorTest
+ * @run main/othervm -Dsun.java2d.opengl=True MutableColorTest
+ */
+
+import java.awt.Color;
+import java.awt.Graphics2D;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsEnvironment;
+import java.awt.Image;
+import java.awt.Transparency;
+import java.awt.geom.Ellipse2D;
+import java.awt.image.BufferedImage;
+import java.awt.image.VolatileImage;
+import java.io.File;
+import java.io.IOException;
+import javax.imageio.ImageIO;
+
+public class MutableColorTest {
+
+ static Image bmImage;
+ static Image argbImage;
+
+ static class EvilColor extends Color {
+ Color colors[] = { Color.red, Color.green, Color.blue };
+ int currentIndex = 0;
+ EvilColor() {
+ super(Color.red.getRGB());
+ }
+
+ @Override
+ public int getRGB() {
+ return colors[currentIndex].getRGB();
+ }
+ void nextColor() {
+ currentIndex++;
+ }
+ }
+
+ private static int testImage(Image im,
+ boolean doClip, boolean doTx)
+ {
+ int w = im.getWidth(null);
+ int h = im.getHeight(null);
+ Graphics2D g = (Graphics2D)im.getGraphics();
+ EvilColor evilColor = new EvilColor();
+ g.setColor(evilColor);
+ g.fillRect(0, 0, w, h);
+ g.dispose();
+
+ evilColor.nextColor();
+
+ g = (Graphics2D)im.getGraphics();
+
+ if (doTx) {
+ g.rotate(Math.PI/2.0, w/2, h/2);
+ }
+ g.setColor(evilColor);
+ g.fillRect(0, 0, w, h);
+ if (doClip) {
+ g.clip(new Ellipse2D.Float(0, 0, w, h));
+ }
+ g.fillRect(0, h/3, w, h/3);
+
+ // tests native BlitBg loop
+ g.drawImage(bmImage, 0, 2*h/3, evilColor, null);
+ // tests General BlitBg loop
+ g.drawImage(argbImage, 0, 2*h/3+h/3/2, evilColor, null);
+
+ return evilColor.getRGB();
+ }
+
+ private static void testResult(final String desc,
+ final BufferedImage snapshot,
+ final int evilColor) {
+ for (int y = 0; y < snapshot.getHeight(); y++) {
+ for (int x = 0; x < snapshot.getWidth(); x++) {
+ int snapRGB = snapshot.getRGB(x, y);
+ if (snapRGB != evilColor) {
+ System.err.printf("Wrong RGB for %s at (%d,%d): 0x%x " +
+ "instead of 0x%x\n", desc, x, y, snapRGB, evilColor);
+ String fileName = "MutableColorTest_"+desc+".png";
+ try {
+ ImageIO.write(snapshot, "png", new File(fileName));
+ System.err.println("Dumped snapshot to "+fileName);
+ } catch (IOException ex) {}
+ throw new RuntimeException("Test FAILED.");
+ }
+ }
+ }
+ }
+
+ public static void main(String[] args) {
+ GraphicsConfiguration gc =
+ GraphicsEnvironment.getLocalGraphicsEnvironment().
+ getDefaultScreenDevice().getDefaultConfiguration();
+
+ bmImage = gc.createCompatibleImage(64, 64, Transparency.BITMASK);
+ argbImage = gc.createCompatibleImage(64, 64, Transparency.TRANSLUCENT);
+
+ if (gc.getColorModel().getPixelSize() > 8) {
+ VolatileImage vi =
+ gc.createCompatibleVolatileImage(64, 64, Transparency.OPAQUE);
+ do {
+ if (vi.validate(gc) == VolatileImage.IMAGE_INCOMPATIBLE) {
+ vi = gc.createCompatibleVolatileImage(64, 64,
+ Transparency.OPAQUE);
+ vi.validate(gc);
+ }
+
+ int color = testImage(vi, false, false);
+ testResult("vi_noclip_notx", vi.getSnapshot(), color);
+
+ color = testImage(vi, true, true);
+ testResult("vi_clip_tx", vi.getSnapshot(), color);
+
+ color = testImage(vi, true, false);
+ testResult("vi_clip_notx", vi.getSnapshot(), color);
+
+ color = testImage(vi, false, true);
+ testResult("vi_noclip_tx", vi.getSnapshot(), color);
+ } while (vi.contentsLost());
+ }
+
+ BufferedImage bi = new BufferedImage(64, 64, BufferedImage.TYPE_INT_RGB);
+ int color = testImage(bi, false, false);
+ testResult("bi_noclip_notx", bi, color);
+
+ color = testImage(bi, true, true);
+ testResult("bi_clip_tx", bi, color);
+
+ color = testImage(bi, true, false);
+ testResult("bi_clip_notx", bi, color);
+
+ color = testImage(bi, false, true);
+ testResult("bi_noclip_tx", bi, color);
+
+ System.err.println("Test passed.");
+ }
+}
diff --git a/jdk/test/sun/java2d/pipe/hw/RSLAPITest/RSLAPITest.java b/jdk/test/sun/java2d/pipe/hw/RSLAPITest/RSLAPITest.java
new file mode 100644
index 00000000000..52ddd3b2196
--- /dev/null
+++ b/jdk/test/sun/java2d/pipe/hw/RSLAPITest/RSLAPITest.java
@@ -0,0 +1,308 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 6635805 6653780 6667607
+ * @summary Tests that the resource sharing layer API is not broken
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @compile -XDignore.symbol.file=true RSLAPITest.java
+ * @run main/othervm RSLAPITest
+ * @run main/othervm -Dsun.java2d.noddraw=true RSLAPITest
+ * @run main/othervm -Dsun.java2d.opengl=True RSLAPITest
+ */
+
+import java.awt.Graphics;
+import java.awt.GraphicsConfiguration;
+import java.awt.GraphicsDevice;
+import java.awt.GraphicsEnvironment;
+import java.awt.Rectangle;
+import java.awt.Transparency;
+import java.awt.image.VolatileImage;
+import java.util.HashSet;
+import sun.java2d.DestSurfaceProvider;
+import sun.java2d.Surface;
+import sun.java2d.pipe.BufferedContext;
+import sun.java2d.pipe.RenderQueue;
+import sun.java2d.pipe.hw.AccelDeviceEventListener;
+import sun.java2d.pipe.hw.AccelGraphicsConfig;
+import sun.java2d.pipe.hw.AccelSurface;
+import static java.awt.Transparency.*;
+import java.lang.reflect.Field;
+import static sun.java2d.pipe.hw.AccelSurface.*;
+import static sun.java2d.pipe.hw.ContextCapabilities.*;
+
+public class RSLAPITest {
+ private static volatile boolean failed = false;
+
+ public static void main(String[] args) {
+ GraphicsEnvironment ge =
+ GraphicsEnvironment.getLocalGraphicsEnvironment();
+ GraphicsDevice gd = ge.getDefaultScreenDevice();
+ GraphicsConfiguration gc = gd.getDefaultConfiguration();
+ testGC(gc);
+
+ if (failed) {
+ throw new RuntimeException("Test FAILED. See err output for more");
+ }
+
+ System.out.println("Test PASSED.");
+ }
+
+ private static void testInvalidType(AccelSurface surface, int type) {
+ long ret = surface.getNativeResource(type);
+ System.out.printf(" getNativeResource(%d)=0x%x\n", type, ret);
+ if (ret != 0l) {
+ System.err.printf(
+ "FAILED: surface.getNativeResource(%d) returned" +
+ " 0x%s. It should have have returned 0L\n",
+ type, ret);
+ failed = true;
+ }
+ }
+
+ private static void testD3DDeviceResourceField(final AccelSurface surface) {
+ try {
+ Class d3dc = Class.forName("sun.java2d.d3d.D3DSurfaceData");
+ if (d3dc.isInstance(surface)) {
+ Field f = d3dc.getDeclaredField("D3D_DEVICE_RESOURCE");
+ f.setAccessible(true);
+ int d3dDR = (Integer)f.get(null);
+
+ System.out.printf(
+ " getNativeResource(D3D_DEVICE_RESOURCE)=0x%x\n",
+ surface.getNativeResource(d3dDR));
+ }
+ } catch (ClassNotFoundException e) {}
+ catch (IllegalAccessException e) {}
+ catch (NoSuchFieldException e) {
+ System.err.println("Failed: D3DSurfaceData.D3D_DEVICE_RESOURCE" +
+ " field not found!");
+ failed = true;
+ }
+ }
+
+ private static void printSurface(Surface s) {
+ if (s instanceof AccelSurface) {
+ final AccelSurface surface = (AccelSurface) s;
+ System.out.println(" Accel Surface: ");
+ System.out.println(" type=" + surface.getType());
+ System.out.println(" bounds=" + surface.getBounds());
+ System.out.println(" nativeBounds=" + surface.getNativeBounds());
+ System.out.println(" isSurfaceLost=" + surface.isSurfaceLost());
+ System.out.println(" isValid=" + surface.isValid());
+ RenderQueue rq = surface.getContext().getRenderQueue();
+ rq.lock();
+ try {
+ rq.flushAndInvokeNow(new Runnable() {
+ public void run() {
+ System.out.printf(" getNativeResource(TEXTURE)=0x%x\n",
+ surface.getNativeResource(TEXTURE));
+ System.out.printf(" getNativeResource(RT_TEXTURE)=0x%x\n",
+ surface.getNativeResource(RT_TEXTURE));
+ System.out.printf(" getNativeResource(RT_PLAIN)=0x%x\n",
+ surface.getNativeResource(RT_PLAIN));
+ System.out.printf(
+ " getNativeResource(FLIP_BACKBUFFER)=0x%x\n",
+ surface.getNativeResource(FLIP_BACKBUFFER));
+
+ testD3DDeviceResourceField(surface);
+
+ testInvalidType(surface, -1);
+ testInvalidType(surface, -150);
+ testInvalidType(surface, 300);
+ testInvalidType(surface, Integer.MAX_VALUE);
+ testInvalidType(surface, Integer.MIN_VALUE);
+ }
+ });
+ } finally {
+ rq.unlock();
+ }
+ } else {
+ System.out.println("null accelerated surface");
+ }
+ }
+
+ private static void printAGC(AccelGraphicsConfig agc) {
+ System.out.println("Accelerated Graphics Config: " + agc);
+ System.out.println("Capabilities:");
+ System.out.printf("AGC caps: 0x%x\n",
+ agc.getContextCapabilities().getCaps());
+ System.out.println(agc.getContextCapabilities());
+ }
+
+ private static void testGC(GraphicsConfiguration gc) {
+ if (!(gc instanceof AccelGraphicsConfig)) {
+ System.out.println("Test passed: no hw accelerated configs found.");
+ return;
+ }
+ System.out.println("AccelGraphicsConfig exists, testing.");
+ AccelGraphicsConfig agc = (AccelGraphicsConfig) gc;
+ printAGC(agc);
+
+ testContext(agc);
+
+ VolatileImage vi = gc.createCompatibleVolatileImage(10, 10);
+ vi.validate(gc);
+ if (vi instanceof DestSurfaceProvider) {
+ System.out.println("Passed: VI is DestSurfaceProvider");
+ Surface s = ((DestSurfaceProvider) vi).getDestSurface();
+ if (s instanceof AccelSurface) {
+ System.out.println("Passed: Obtained Accel Surface");
+ printSurface((AccelSurface) s);
+ }
+ Graphics g = vi.getGraphics();
+ if (g instanceof DestSurfaceProvider) {
+ System.out.println("Passed: VI graphics is " +
+ "DestSurfaceProvider");
+ printSurface(((DestSurfaceProvider) g).getDestSurface());
+ }
+ } else {
+ System.out.println("VI is not DestSurfaceProvider");
+ }
+ testVICreation(agc, CAPS_RT_TEXTURE_ALPHA, TRANSLUCENT, RT_TEXTURE);
+ testVICreation(agc, CAPS_RT_TEXTURE_OPAQUE, OPAQUE, RT_TEXTURE);
+ testVICreation(agc, CAPS_RT_PLAIN_ALPHA, TRANSLUCENT, RT_PLAIN);
+ testVICreation(agc, agc.getContextCapabilities().getCaps(), OPAQUE,
+ TEXTURE);
+ testForNPEDuringCreation(agc);
+ }
+
+ private static void testVICreation(AccelGraphicsConfig agc, int cap,
+ int transparency, int type)
+ {
+ int caps = agc.getContextCapabilities().getCaps();
+ int w = 11, h = 17;
+
+ VolatileImage vi =
+ agc.createCompatibleVolatileImage(w, h, transparency, type);
+ if ((cap & caps) != 0) {
+ if (vi == null) {
+ System.out.printf("Failed: cap=%d is supported but " +
+ "image wasn't created\n", cap);
+ throw new RuntimeException("Failed: image wasn't created " +
+ "for supported cap");
+ } else {
+ if (!(vi instanceof DestSurfaceProvider)) {
+ throw new RuntimeException("Failed: created VI is not " +
+ "DestSurfaceProvider");
+ }
+ Surface s = ((DestSurfaceProvider) vi).getDestSurface();
+ if (s instanceof AccelSurface) {
+ AccelSurface as = (AccelSurface) s;
+ printSurface(as);
+ if (as.getType() != type) {
+ throw new RuntimeException("Failed: returned VI is" +
+ " of incorrect type: " + as.getType() +
+ " requested type=" + type);
+ } else {
+ System.out.printf("Passed: VI of type %d was " +
+ "created for cap=%d\n", type, cap);
+ }
+ if (as.getType() == TEXTURE) {
+ boolean ex = false;
+ try {
+ Graphics g = vi.getGraphics();
+ g.dispose();
+ } catch (UnsupportedOperationException e) {
+ ex = true;
+ }
+ if (!ex) {
+ throw new RuntimeException("Failed: " +
+ "texture.getGraphics() didn't throw exception");
+ } else {
+ System.out.println("Passed: VI.getGraphics()" +
+ " threw exception for texture-based VI");
+ }
+ }
+ } else {
+ System.out.printf("Passed: VI of type %d was " +
+ "created for cap=%d but accel surface is null\n",
+ type, cap);
+ }
+ }
+ } else {
+ if (vi != null) {
+ throw new RuntimeException("Failed: created VI for " +
+ "unsupported cap=" + cap);
+ }
+ }
+ }
+
+ private static void testContext(final AccelGraphicsConfig agc) {
+ BufferedContext c = agc.getContext();
+ final AccelDeviceEventListener l = new AccelDeviceEventListener() {
+ public void onDeviceDispose() {
+ System.out.println("onDeviceDispose invoked");
+ agc.removeDeviceEventListener(this);
+ }
+ public void onDeviceReset() {
+ System.out.println("onDeviceReset invoked");
+ }
+ };
+ agc.addDeviceEventListener(l);
+
+ RenderQueue rq = c.getRenderQueue();
+ rq.lock();
+ try {
+ c.saveState();
+ rq.flushNow();
+ c.restoreState();
+ rq.flushNow();
+ System.out.println("Passed: Save/Restore");
+ } finally {
+ rq.unlock();
+ }
+ }
+
+ private static void testForNPEDuringCreation(AccelGraphicsConfig agc) {
+ int iterations = 100;
+ HashSet vis = new HashSet();
+ GraphicsConfiguration gc = (GraphicsConfiguration)agc;
+ Rectangle r = gc.getBounds();
+ long ram = gc.getDevice().getAvailableAcceleratedMemory();
+ if (ram > 0) {
+ // guesstimate the number of iterations needed to exhaust vram
+ int i = 2 *
+ (int)(ram / (r.width * r.height * gc.getColorModel().getPixelSize()/8));
+ iterations = Math.max(iterations, i);
+ System.err.println("iterations="+iterations);
+ }
+ for (int i = 0; i < iterations; i++) {
+ VolatileImage vi =
+ agc.createCompatibleVolatileImage(r.width, r.height,
+ Transparency.OPAQUE,
+ AccelSurface.RT_PLAIN);
+ if (vi == null) {
+ break;
+ }
+ vis.add(vi);
+ }
+ for (VolatileImage vi : vis) {
+ vi.flush();
+ }
+ vis = null;
+
+ System.out.println("Passed: testing for possible NPEs " +
+ "during VI creation");
+ }
+}
diff --git a/jdk/test/sun/java2d/pipe/hw/VSyncedBufferStrategyTest/VSyncedBufferStrategyTest.java b/jdk/test/sun/java2d/pipe/hw/VSyncedBufferStrategyTest/VSyncedBufferStrategyTest.java
new file mode 100644
index 00000000000..cec37046c79
--- /dev/null
+++ b/jdk/test/sun/java2d/pipe/hw/VSyncedBufferStrategyTest/VSyncedBufferStrategyTest.java
@@ -0,0 +1,338 @@
+/*
+ * Copyright 2007-2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
+ * CA 95054 USA or visit www.sun.com if you need additional information or
+ * have any questions.
+ */
+/*
+ * @test
+ * @bug 6678218 6681745 6691737
+ * @summary Tests that v-synced BufferStrategies works (if vsync is supported)
+ * @author Dmitri.Trembovetski@sun.com: area=Graphics
+ * @compile -XDignore.symbol.file=true VSyncedBufferStrategyTest.java
+ * @run main/manual/othervm VSyncedBufferStrategyTest
+ * @run main/manual/othervm -Dsun.java2d.opengl=True VSyncedBufferStrategyTest
+ */
+
+import java.awt.AWTException;
+import java.awt.BufferCapabilities;
+import java.awt.BufferCapabilities.FlipContents;
+import java.awt.Button;
+import java.awt.Canvas;
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.EventQueue;
+import java.awt.FlowLayout;
+import java.awt.Font;
+import java.awt.Frame;
+import java.awt.Graphics;
+import java.awt.HeadlessException;
+import java.awt.ImageCapabilities;
+import java.awt.Panel;
+import java.awt.event.ActionEvent;
+import java.awt.event.ActionListener;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+import java.awt.image.BufferStrategy;
+import java.util.concurrent.CountDownLatch;
+import javax.swing.JButton;
+import javax.swing.JFrame;
+import javax.swing.JPanel;
+import javax.swing.JScrollPane;
+import javax.swing.JTextArea;
+
+public class VSyncedBufferStrategyTest extends Canvas implements Runnable {
+
+ private static final int BLOCK_W = 50;
+ private static final int BLOCK_H = 200;
+
+ BufferStrategy bs;
+ Thread renderThread;
+
+ int blockX = 10;
+ int blockY = 10;
+
+ private volatile boolean done = false;
+ private volatile boolean requestVSync;
+ private boolean currentBSVSynced;
+
+ public VSyncedBufferStrategyTest(boolean requestVSync) {
+ this.requestVSync = requestVSync;
+ this.currentBSVSynced = !requestVSync;
+ renderThread = new Thread(this);
+ renderThread.start();
+ }
+
+ private static final BufferCapabilities defaultBC =
+ new BufferCapabilities(
+ new ImageCapabilities(true),
+ new ImageCapabilities(true),
+ null);
+
+ private void createBS(boolean requestVSync) {
+ if (bs != null && requestVSync == currentBSVSynced) {
+ return;
+ }
+
+ BufferCapabilities bc = defaultBC;
+ if (requestVSync) {
+ bc = new sun.java2d.pipe.hw.ExtendedBufferCapabilities(
+ new ImageCapabilities(true),
+ new ImageCapabilities(true),
+ FlipContents.COPIED,
+ sun.java2d.pipe.hw.ExtendedBufferCapabilities.VSyncType.VSYNC_ON);
+ }
+ try {
+ createBufferStrategy(2, bc);
+ } catch (AWTException e) {
+ System.err.println("Warning: cap is not supported: "+bc);
+ e.printStackTrace();
+ createBufferStrategy(2);
+ }
+ currentBSVSynced = requestVSync;
+ bs = getBufferStrategy();
+ String s =
+ getParent() instanceof Frame ?
+ ((Frame)getParent()).getTitle() : "parent";
+ System.out.println("Created BS for \"" + s + "\" frame, bs="+bs);
+ }
+
+ @Override
+ public void paint(Graphics g) {
+ }
+ @Override
+ public void update(Graphics g) {
+ }
+
+ @Override
+ public void run() {
+ while (!isShowing()) {
+ try { Thread.sleep(5); } catch (InterruptedException e) {}
+ }
+ try { Thread.sleep(2000); } catch (InterruptedException e) {}
+
+ try {
+ while (!done && isShowing()) {
+ createBS(requestVSync);
+ do {
+ step();
+ Graphics g = bs.getDrawGraphics();
+ render(g);
+ if (!bs.contentsRestored()) {
+ bs.show();
+ }
+ } while (bs.contentsLost());
+ Thread.yield();
+ }
+ } catch (Throwable e) {
+ // since we're not bothering with proper synchronization, exceptions
+ // may be thrown when the frame is closed
+ if (isShowing()) {
+ throw new RuntimeException(e);
+ }
+ }
+ }
+
+ int inc = 5;
+ private void step() {
+ blockX += inc;
+ if (blockX > getWidth() - BLOCK_W - 10) {
+ inc = -inc;
+ blockX += inc;
+ }
+ if (blockX < 10) {
+ inc = -inc;
+ blockX += inc;
+ }
+ }
+
+ private void render(Graphics g) {
+ g.setColor(Color.white);
+ g.fillRect(0, 0, getWidth(), getHeight());
+
+ g.setColor(Color.black);
+ g.fillRect(blockX, blockY, BLOCK_W, BLOCK_H);
+ }
+
+ private void setRequestVSync(boolean reqVSync) {
+ requestVSync = reqVSync;
+ }
+
+ @Override
+ public Dimension getPreferredSize() {
+ return new Dimension(BLOCK_W*10+20, BLOCK_H+20);
+ }
+
+ private static int frameNum = 0;
+ private static Frame createAndShowBSFrame() {
+ final Frame f = new Frame("Not V-Synced");
+
+ int myNum;
+ synchronized (VSyncedBufferStrategyTest.class) {
+ myNum = frameNum++;
+ }
+
+ final VSyncedBufferStrategyTest component =
+ new VSyncedBufferStrategyTest(false);
+ f.setIgnoreRepaint(true);
+ f.add("Center", component);
+
+ Panel p = new Panel();
+
+ Button b = new Button("Request VSync");
+ b.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ f.setTitle("Possibly V-Synced");
+ component.setRequestVSync(true);
+ }
+ });
+ p.add(b);
+
+ b = new Button("Relinquish VSync");
+ b.addActionListener(new ActionListener() {
+ int inc = 1;
+ public void actionPerformed(ActionEvent e) {
+ f.setTitle("Not V-Synced");
+ component.setRequestVSync(false);
+ f.setSize(f.getWidth()+inc, f.getHeight());
+ inc = -inc;
+ }
+ });
+ p.add(b);
+
+ f.add("South", p);
+
+ f.pack();
+ f.setLocation(10, myNum * f.getHeight());
+ f.setVisible(true);
+ f.addWindowListener(new WindowAdapter() {
+ @Override
+ public void windowClosing(WindowEvent e) {
+ component.done = true;
+ f.dispose();
+ }
+ @Override
+ public void windowClosed(WindowEvent e) {
+ component.done = true;
+ }
+ });
+
+ return f;
+ }
+
+ private static final String description =
+ "Tests that v-synced BufferStrategy works. Note that it in some\n" +
+ "cases the v-sync can not be enabled, and it is accepted.\n" +
+ "The following however is true: only one buffer strategy at a time can\n"+
+ "be created v-synced. In order for other BS to become v-synced, the one\n"+
+ "that currently is v-synched (or its window) needs to be disposed.\n" +
+ "Try the following scenarios:\n" +
+ " - click the \"Request VSync\" button in one of the frames. If the\n"+
+ " behavior of the animation changes - the animation becomes smooth\n" +
+ " it had successfully created a v-synced BS. Note that the animation\n" +
+ " in other frames may also become smoother - this is a side-effect\n"+
+ " of one of the BS-es becoming v-synched\n" +
+ " - click the \"Relinquish VSync\" button on the same frame. If the\n"+
+ " behavior changes to the original (tearing)- it had successfully\n" +
+ " created a non-vsynced strategy.\n" +
+ " - next, try making another one v-synced. It should succeed.\n" +
+ " - next, try making another one v-synced - while there's already\n" +
+ " a v-synced frame. It should not succeed - meaning, it shouldn't\n" +
+ " appear to become smoother, and the behavior of the current v-synced\n" +
+ " frame shouldn't change.\n" +
+ "\n" +
+ "If there aren't any BufferStrategy-related exceptions or other\n" +
+ "issues, and the scenarios worked, the test passed, otherwise it\n"+
+ "failed.\n";
+
+ private static void createAndShowDescGUI(final Frame f3, final Frame f1,
+ final Frame f2)
+ throws HeadlessException, RuntimeException
+ {
+ final JFrame desc =
+ new JFrame("VSyncedBufferStrategyTest - Description");
+ desc.addWindowListener(new WindowAdapter() {
+
+ @Override
+ public void windowClosing(WindowEvent e) {
+ f1.dispose();
+ f2.dispose();
+ f3.dispose();
+ l.countDown();
+ }
+ });
+ JPanel p = new JPanel();
+ JButton bPassed = new JButton("Passed");
+ bPassed.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ desc.dispose();
+ f1.dispose();
+ f2.dispose();
+ f3.dispose();
+ l.countDown();
+ }
+ });
+ JButton bFailed = new JButton("Failed");
+ bFailed.addActionListener(new ActionListener() {
+ public void actionPerformed(ActionEvent e) {
+ failed = true;
+ desc.dispose();
+ f1.dispose();
+ f2.dispose();
+ f3.dispose();
+ l.countDown();
+ }
+ });
+ p.setLayout(new FlowLayout());
+ p.add(bPassed);
+ p.add(bFailed);
+ JTextArea ta = new JTextArea(24, 75);
+ ta.setFont(new Font(Font.MONOSPACED, Font.PLAIN, 12));
+ ta.setEditable(false);
+ ta.setText(description);
+ desc.add("Center", new JScrollPane(ta));
+ desc.add("South", p);
+ desc.pack();
+ desc.setLocation(BLOCK_W*10+50, 0);
+ desc.setVisible(true);
+ }
+
+ private static void createTestFrames() {
+ Frame f1 = createAndShowBSFrame();
+ Frame f2 = createAndShowBSFrame();
+ Frame f3 = createAndShowBSFrame();
+ createAndShowDescGUI(f1, f2, f3);
+ }
+
+ static boolean failed = false;
+ static CountDownLatch l = new CountDownLatch(1);
+ public static void main(String[] args) throws Exception {
+ EventQueue.invokeLater(new Runnable() {
+ public void run() {
+ createTestFrames();
+ }
+ });
+ l.await();
+ if (failed) {
+ throw new RuntimeException("Test FAILED");
+ }
+ System.out.println("Test PASSED");
+ }
+
+}
diff --git a/jdk/test/sun/text/resources/LocaleData b/jdk/test/sun/text/resources/LocaleData
index 5c4e65760ec..f885bbeb2db 100644
--- a/jdk/test/sun/text/resources/LocaleData
+++ b/jdk/test/sun/text/resources/LocaleData
@@ -5514,3 +5514,7 @@ CurrencyNames/ar_SD/SDG=\u062c.\u0633.\u200f
# bug 6531593
FormatData/is_IS/NumberPatterns/1=#,##0. \u00A4;-#,##0. \u00A4
+# bug 6509039
+FormatData/sv/AmPmMarkers/0=fm
+FormatData/sv/AmPmMarkers/1=em
+
diff --git a/jdk/test/sun/text/resources/LocaleDataTest.java b/jdk/test/sun/text/resources/LocaleDataTest.java
index 70fd1828a28..ff36327ea77 100644
--- a/jdk/test/sun/text/resources/LocaleDataTest.java
+++ b/jdk/test/sun/text/resources/LocaleDataTest.java
@@ -31,6 +31,7 @@
* 5102005 5074431 6182685 6208712 6277020 6245766 6351682 6386647 6379382
* 6414459 6455680 6498742 6558863 6488119 6547501 6497154 6558856 6481177
* 6379214 6485516 6486607 4225362 4494727 6533691 6531591 6531593 6570259
+ * 6509039
* @summary Verify locale data
*
*/