8339341: SurfaceManager cacheMap retains strong references

Reviewed-by: jdv, prr
This commit is contained in:
Nikita Gubarkov 2024-10-17 20:54:18 +00:00 committed by Alexey Ushakov
parent 12551ae64a
commit fe83b7d596
18 changed files with 121 additions and 123 deletions

View File

@ -31,20 +31,27 @@ import java.awt.Transparency;
import java.awt.geom.AffineTransform; import java.awt.geom.AffineTransform;
import java.awt.image.ColorModel; import java.awt.image.ColorModel;
import sun.awt.image.SurfaceManager;
import sun.java2d.SurfaceData; import sun.java2d.SurfaceData;
import sun.lwawt.LWGraphicsConfig; import sun.lwawt.LWGraphicsConfig;
import sun.lwawt.macosx.CFRetainedResource; import sun.lwawt.macosx.CFRetainedResource;
public abstract class CGraphicsConfig extends GraphicsConfiguration public abstract class CGraphicsConfig extends GraphicsConfiguration
implements LWGraphicsConfig { implements LWGraphicsConfig, SurfaceManager.ProxiedGraphicsConfig {
private final CGraphicsDevice device; private final CGraphicsDevice device;
private ColorModel colorModel; private ColorModel colorModel;
private final SurfaceManager.ProxyCache surfaceDataProxyCache = new SurfaceManager.ProxyCache();
protected CGraphicsConfig(CGraphicsDevice device) { protected CGraphicsConfig(CGraphicsDevice device) {
this.device = device; this.device = device;
} }
@Override
public SurfaceManager.ProxyCache getSurfaceDataProxyCache() {
return surfaceDataProxyCache;
}
@Override @Override
public final Rectangle getBounds() { public final Rectangle getBounds() {
return device.getBounds(); return device.getBounds();

View File

@ -29,7 +29,6 @@ import sun.awt.CGraphicsConfig;
import sun.awt.CGraphicsDevice; import sun.awt.CGraphicsDevice;
import sun.awt.image.OffScreenImage; import sun.awt.image.OffScreenImage;
import sun.awt.image.SunVolatileImage; import sun.awt.image.SunVolatileImage;
import sun.awt.image.SurfaceManager;
import sun.java2d.Disposer; import sun.java2d.Disposer;
import sun.java2d.DisposerRecord; import sun.java2d.DisposerRecord;
import sun.java2d.Surface; import sun.java2d.Surface;
@ -70,7 +69,7 @@ import static sun.java2d.pipe.hw.ContextCapabilities.*;
import static sun.java2d.metal.MTLContext.MTLContextCaps.CAPS_EXT_BIOP_SHADER; import static sun.java2d.metal.MTLContext.MTLContextCaps.CAPS_EXT_BIOP_SHADER;
public final class MTLGraphicsConfig extends CGraphicsConfig public final class MTLGraphicsConfig extends CGraphicsConfig
implements AccelGraphicsConfig, SurfaceManager.ProxiedGraphicsConfig implements AccelGraphicsConfig
{ {
private static ImageCapabilities imageCaps = new MTLImageCaps(); private static ImageCapabilities imageCaps = new MTLImageCaps();
@ -112,11 +111,6 @@ public final class MTLGraphicsConfig extends CGraphicsConfig
new MTLGCDisposerRecord(pConfigInfo)); new MTLGCDisposerRecord(pConfigInfo));
} }
@Override
public Object getProxyKey() {
return this;
}
public SurfaceData createManagedSurface(int w, int h, int transparency) { public SurfaceData createManagedSurface(int w, int h, int transparency) {
return MTLSurfaceData.createData(this, w, h, return MTLSurfaceData.createData(this, w, h,
getColorModel(transparency), getColorModel(transparency),

View File

@ -156,7 +156,7 @@ public abstract class MTLSurfaceData extends SurfaceData
super(getCustomSurfaceType(type), cm); super(getCustomSurfaceType(type), cm);
this.graphicsConfig = gc; this.graphicsConfig = gc;
this.type = type; this.type = type;
setBlitProxyKey(gc.getProxyKey()); setBlitProxyCache(gc.getSurfaceDataProxyCache());
// TEXTURE shouldn't be scaled, it is used for managed BufferedImages. // TEXTURE shouldn't be scaled, it is used for managed BufferedImages.
scale = type == TEXTURE ? 1 : gc.getDevice().getScaleFactor(); scale = type == TEXTURE ? 1 : gc.getDevice().getScaleFactor();

View File

@ -105,11 +105,6 @@ public final class CGLGraphicsConfig extends CGraphicsConfig
new CGLGCDisposerRecord(pConfigInfo)); new CGLGCDisposerRecord(pConfigInfo));
} }
@Override
public Object getProxyKey() {
return this;
}
@Override @Override
public SurfaceData createManagedSurface(int w, int h, int transparency) { public SurfaceData createManagedSurface(int w, int h, int transparency) {
return CGLSurfaceData.createData(this, w, h, return CGLSurfaceData.createData(this, w, h,

View File

@ -31,8 +31,11 @@ import java.awt.Image;
import java.awt.ImageCapabilities; import java.awt.ImageCapabilities;
import java.awt.image.BufferedImage; import java.awt.image.BufferedImage;
import java.awt.image.VolatileImage; import java.awt.image.VolatileImage;
import java.lang.ref.WeakReference;
import java.util.Collections;
import java.util.Iterator; import java.util.Iterator;
import java.util.concurrent.ConcurrentHashMap; import java.util.Map;
import java.util.WeakHashMap;
import sun.java2d.InvalidPipeException; import sun.java2d.InvalidPipeException;
import sun.java2d.SurfaceData; import sun.java2d.SurfaceData;
@ -89,47 +92,14 @@ public abstract class SurfaceManager {
imgaccessor.setSurfaceManager(img, mgr); imgaccessor.setSurfaceManager(img, mgr);
} }
private volatile ConcurrentHashMap<Object,Object> cacheMap;
/** /**
* Return an arbitrary cached object for an arbitrary cache key. * This map holds references to SurfaceDataProxy per given ProxyCache.
* Other objects can use this mechanism to store cached data about * Unlike ProxyCache, which contains SurfaceDataProxy objects per given SurfaceManager,
* the source image that will let them save time when using or * this map does not prevent contained proxies from being garbage collected.
* manipulating the image in the future. * Therefore, ProxyCache can be considered an "owning" container for the SurfaceDataProxy objects,
* <p> * and this map is just a weak mapping for the bookkeeping purposes.
* Note that the cache is maintained as a simple Map with no
* attempts to keep it up to date or invalidate it so any data
* stored here must either not be dependent on the state of the
* image or it must be individually tracked to see if it is
* outdated or obsolete.
* <p>
* The SurfaceData object of the primary (destination) surface
* has a StateTracker mechanism which can help track the validity
* and "currentness" of any data stored here.
* For convenience and expediency an object stored as cached
* data may implement the FlushableCacheData interface specified
* below so that it may be notified immediately if the flush()
* method is ever called.
*/ */
public Object getCacheData(Object key) { private final Map<ProxyCache, WeakReference<SurfaceDataProxy>> weakCache = new WeakHashMap<>(2);
return (cacheMap == null) ? null : cacheMap.get(key);
}
/**
* Store an arbitrary cached object for an arbitrary cache key.
* See the getCacheData() method for notes on tracking the
* validity of data stored using this mechanism.
*/
public void setCacheData(Object key, Object value) {
if (cacheMap == null) {
synchronized (this) {
if (cacheMap == null) {
cacheMap = new ConcurrentHashMap<>(2);
}
}
}
cacheMap.put(key, value);
}
/** /**
* Returns the main SurfaceData object that "owns" the pixels for * Returns the main SurfaceData object that "owns" the pixels for
@ -202,12 +172,10 @@ public abstract class SurfaceManager {
tmpGc = GraphicsEnvironment.getLocalGraphicsEnvironment(). tmpGc = GraphicsEnvironment.getLocalGraphicsEnvironment().
getDefaultScreenDevice().getDefaultConfiguration(); getDefaultScreenDevice().getDefaultConfiguration();
} }
if (tmpGc instanceof ProxiedGraphicsConfig) { if (tmpGc instanceof ProxiedGraphicsConfig pgc) {
Object proxyKey = ProxyCache cache = pgc.getSurfaceDataProxyCache();
((ProxiedGraphicsConfig) tmpGc).getProxyKey(); if (cache != null) {
if (proxyKey != null) { SurfaceDataProxy sdp = cache.get(SurfaceManager.this);
SurfaceDataProxy sdp =
(SurfaceDataProxy) getCacheData(proxyKey);
return (sdp != null && sdp.isAccelerated()); return (sdp != null && sdp.isAccelerated());
} }
} }
@ -222,13 +190,51 @@ public abstract class SurfaceManager {
* Implementing this interface facilitates the default * Implementing this interface facilitates the default
* implementation of getImageCapabilities() above. * implementation of getImageCapabilities() above.
*/ */
public static interface ProxiedGraphicsConfig { public interface ProxiedGraphicsConfig {
/** /**
* Return the key that destination surfaces created on the * Return the cache that destination surfaces created on the
* given GraphicsConfiguration use to store SurfaceDataProxy * given GraphicsConfiguration use to store SurfaceDataProxy
* objects for their cached copies. * objects for their cached copies.
*/ */
public Object getProxyKey(); ProxyCache getSurfaceDataProxyCache();
}
public static class ProxyCache {
private final Map<SurfaceManager, SurfaceDataProxy> map = Collections.synchronizedMap(new WeakHashMap<>());
/**
* Return a cached SurfaceDataProxy object for a given SurfaceManager.
* <p>
* Note that the cache is maintained as a simple Map with no
* attempts to keep it up to date or invalidate it so any data
* stored here must either not be dependent on the state of the
* image or it must be individually tracked to see if it is
* outdated or obsolete.
* <p>
* The SurfaceData object of the primary (destination) surface
* has a StateTracker mechanism which can help track the validity
* and "currentness" of any data stored here.
* For convenience and expediency an object stored as cached
* data may implement the FlushableCacheData interface specified
* below so that it may be notified immediately if the flush()
* method is ever called.
*/
public SurfaceDataProxy get(SurfaceManager manager) {
return map.get(manager);
}
/**
* Store a cached SurfaceDataProxy object for a given SurfaceManager.
* See the get() method for notes on tracking the
* validity of data stored using this mechanism.
*/
public void put(SurfaceManager manager, SurfaceDataProxy proxy) {
synchronized (manager.weakCache) { // Synchronize on weakCache first!
manager.weakCache.put(this, new WeakReference<>(proxy));
map.put(manager, proxy);
}
}
} }
/** /**
@ -244,15 +250,13 @@ public abstract class SurfaceManager {
flush(false); flush(false);
} }
synchronized void flush(boolean deaccelerate) { void flush(boolean deaccelerate) {
if (cacheMap != null) { synchronized (weakCache) {
Iterator<Object> i = cacheMap.values().iterator(); Iterator<WeakReference<SurfaceDataProxy>> i = weakCache.values().iterator();
while (i.hasNext()) { while (i.hasNext()) {
Object o = i.next(); SurfaceDataProxy sdp = i.next().get();
if (o instanceof FlushableCacheData) { if (sdp == null || sdp.flush(deaccelerate)) {
if (((FlushableCacheData) o).flush(deaccelerate)) { i.remove();
i.remove();
}
} }
} }
} }

View File

@ -113,7 +113,7 @@ public abstract class SurfaceData
private static native void initIDs(); private static native void initIDs();
private Object blitProxyKey; private SurfaceManager.ProxyCache blitProxyCache;
private StateTrackableDelegate stateDelegate; private StateTrackableDelegate stateDelegate;
static { static {
@ -143,23 +143,21 @@ public abstract class SurfaceData
} }
/** /**
* Subclasses can set a "blit proxy key" which will be used * Subclasses can set a "blit proxy cache" which will be used
* along with the SurfaceManager.getCacheData() mechanism to * along with the SurfaceManager to
* store acceleration-compatible cached copies of source images. * store acceleration-compatible cached copies of source images.
* This key is a "tag" used to identify which cached copies * The getSourceSurfaceData() method uses this cache to manage
* are compatible with this destination SurfaceData. * copies of a source image as described below.
* The getSourceSurfaceData() method uses this key to manage
* cached copies of a source image as described below.
* <p> * <p>
* The Object used as this key should be as unique as it needs * The cache used should be as unique as it needs
* to be to ensure that multiple acceleratible destinations can * to be to ensure that multiple acceleratible destinations can
* each store their cached copies separately under different keys * each store their cached copies separately into different caches
* without interfering with each other or getting back the wrong * without interfering with each other or getting back the wrong
* cached copy. * cached copy.
* <p> * <p>
* Many acceleratable SurfaceData objects can use their own * Many GraphicsConfiguration implementations have their own
* GraphicsConfiguration as their proxy key as the GC object will * cache as the GC object is
* typically be unique to a given screen and pixel format, but * typically unique to a given screen and pixel format, but
* other rendering destinations may have more or less stringent * other rendering destinations may have more or less stringent
* sharing requirements. For instance, X11 pixmaps can be * sharing requirements. For instance, X11 pixmaps can be
* shared on a given screen by any GraphicsConfiguration that * shared on a given screen by any GraphicsConfiguration that
@ -168,14 +166,14 @@ public abstract class SurfaceData
* a different cached proxy for each would be a waste. One can * a different cached proxy for each would be a waste. One can
* imagine platforms where a single cached copy can be created * imagine platforms where a single cached copy can be created
* and shared across all screens and pixel formats - such * and shared across all screens and pixel formats - such
* implementations could use a single heavily shared key Object. * implementations could use a single heavily shared cache object.
*/ */
protected void setBlitProxyKey(Object key) { protected void setBlitProxyCache(SurfaceManager.ProxyCache cache) {
// Caching is effectively disabled if we never have a proxy key // Caching is effectively disabled if we never have a proxy cache
// since the getSourceSurfaceData() method only does caching // since the getSourceSurfaceData() method only does caching
// if the key is not null. // if the cache is not null.
if (SurfaceDataProxy.isCachingAllowed()) { if (SurfaceDataProxy.isCachingAllowed()) {
this.blitProxyKey = key; this.blitProxyCache = cache;
} }
} }
@ -192,7 +190,7 @@ public abstract class SurfaceData
* appropriate SurfaceDataProxy instance. * appropriate SurfaceDataProxy instance.
* The parameters describe the type of imaging operation being performed. * The parameters describe the type of imaging operation being performed.
* <p> * <p>
* If a blitProxyKey was supplied by the subclass then it is * If a blitProxyCache was supplied by the subclass then it is
* used to potentially override the choice of source SurfaceData. * used to potentially override the choice of source SurfaceData.
* The outline of this process is: * The outline of this process is:
* <ol> * <ol>
@ -201,8 +199,8 @@ public abstract class SurfaceData
* <li> destSD gets the SurfaceManager of the source Image * <li> destSD gets the SurfaceManager of the source Image
* and first retrieves the default SD from it using * and first retrieves the default SD from it using
* getPrimarySurfaceData() * getPrimarySurfaceData()
* <li> destSD uses its "blit proxy key" (if set) to look for * <li> destSD uses its "blit proxy cache" (if set) to look for
* some cached data stored in the source SurfaceManager * some cached data corresponding to the the source SurfaceManager
* <li> If the cached data is null then makeProxyFor() is used * <li> If the cached data is null then makeProxyFor() is used
* to create some cached data which is stored back in the * to create some cached data which is stored back in the
* source SurfaceManager under the same key for future uses. * source SurfaceManager under the same key for future uses.
@ -219,18 +217,15 @@ public abstract class SurfaceData
{ {
SurfaceManager srcMgr = SurfaceManager.getManager(img); SurfaceManager srcMgr = SurfaceManager.getManager(img);
SurfaceData srcData = srcMgr.getPrimarySurfaceData(); SurfaceData srcData = srcMgr.getPrimarySurfaceData();
if (img.getAccelerationPriority() > 0.0f && if (img.getAccelerationPriority() > 0.0f && blitProxyCache != null) {
blitProxyKey != null) SurfaceDataProxy sdp = blitProxyCache.get(srcMgr);
{
SurfaceDataProxy sdp =
(SurfaceDataProxy) srcMgr.getCacheData(blitProxyKey);
if (sdp == null || !sdp.isValid()) { if (sdp == null || !sdp.isValid()) {
if (srcData.getState() == State.UNTRACKABLE) { if (srcData.getState() == State.UNTRACKABLE) {
sdp = SurfaceDataProxy.UNCACHED; sdp = SurfaceDataProxy.UNCACHED;
} else { } else {
sdp = makeProxyFor(srcData); sdp = makeProxyFor(srcData);
} }
srcMgr.setCacheData(blitProxyKey, sdp); blitProxyCache.put(srcMgr, sdp);
} }
srcData = sdp.replaceData(srcData, txtype, comp, bgColor); srcData = sdp.replaceData(srcData, txtype, comp, bgColor);
} }

View File

@ -232,7 +232,7 @@ public abstract class OGLSurfaceData extends SurfaceData
super(getCustomSurfaceType(type), cm); super(getCustomSurfaceType(type), cm);
this.graphicsConfig = gc; this.graphicsConfig = gc;
this.type = type; this.type = type;
setBlitProxyKey(gc.getProxyKey()); setBlitProxyCache(gc.getSurfaceDataProxyCache());
} }
@Override @Override

View File

@ -180,8 +180,8 @@ public class X11GraphicsConfig extends GraphicsConfiguration
} }
@Override @Override
public Object getProxyKey() { public SurfaceManager.ProxyCache getSurfaceDataProxyCache() {
return device.getProxyKeyFor(getSurfaceType()); return device.getProxyCacheFor(getSurfaceType());
} }
/** /**

View File

@ -36,10 +36,13 @@ import java.awt.Window;
import java.security.AccessController; import java.security.AccessController;
import java.security.PrivilegedAction; import java.security.PrivilegedAction;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.Map;
import java.util.Objects; import java.util.Objects;
import sun.awt.image.SurfaceManager;
import sun.awt.util.ThreadGroupUtils; import sun.awt.util.ThreadGroupUtils;
import sun.java2d.SunGraphicsEnvironment; import sun.java2d.SunGraphicsEnvironment;
import sun.java2d.loops.SurfaceType; import sun.java2d.loops.SurfaceType;
@ -62,7 +65,7 @@ public final class X11GraphicsDevice extends GraphicsDevice
* therefore methods, which is using this id should be ready to it. * therefore methods, which is using this id should be ready to it.
*/ */
private volatile int screen; private volatile int screen;
HashMap<SurfaceType, Object> x11ProxyKeyMap = new HashMap<>(); Map<SurfaceType, SurfaceManager.ProxyCache> x11ProxyCacheMap = Collections.synchronizedMap(new HashMap<>());
private static AWTPermission fullScreenExclusivePermission; private static AWTPermission fullScreenExclusivePermission;
private static Boolean xrandrExtSupported; private static Boolean xrandrExtSupported;
@ -86,15 +89,8 @@ public final class X11GraphicsDevice extends GraphicsDevice
return screen; return screen;
} }
public Object getProxyKeyFor(SurfaceType st) { public SurfaceManager.ProxyCache getProxyCacheFor(SurfaceType st) {
synchronized (x11ProxyKeyMap) { return x11ProxyCacheMap.computeIfAbsent(st, unused -> new SurfaceManager.ProxyCache());
Object o = x11ProxyKeyMap.get(st);
if (o == null) {
o = new Object();
x11ProxyKeyMap.put(st, o);
}
return o;
}
} }
/** /**

View File

@ -72,6 +72,7 @@ public final class GLXGraphicsConfig
private long pConfigInfo; private long pConfigInfo;
private ContextCapabilities oglCaps; private ContextCapabilities oglCaps;
private final OGLContext context; private final OGLContext context;
private final SurfaceManager.ProxyCache surfaceDataProxyCache = new SurfaceManager.ProxyCache();
private static native long getGLXConfigInfo(int screennum, int visualnum); private static native long getGLXConfigInfo(int screennum, int visualnum);
private static native int getOGLCapabilities(long configInfo); private static native int getOGLCapabilities(long configInfo);
@ -89,8 +90,8 @@ public final class GLXGraphicsConfig
} }
@Override @Override
public Object getProxyKey() { public SurfaceManager.ProxyCache getSurfaceDataProxyCache() {
return this; return surfaceDataProxyCache;
} }
@Override @Override

View File

@ -433,7 +433,7 @@ public abstract class X11SurfaceData extends XSurfaceData {
this.depth = cm.getPixelSize(); this.depth = cm.getPixelSize();
initOps(peer, graphicsConfig, depth); initOps(peer, graphicsConfig, depth);
if (isAccelerationEnabled()) { if (isAccelerationEnabled()) {
setBlitProxyKey(gc.getProxyKey()); setBlitProxyCache(gc.getSurfaceDataProxyCache());
} }
} }

View File

@ -34,6 +34,8 @@ import sun.java2d.SurfaceData;
public class XRGraphicsConfig extends X11GraphicsConfig implements public class XRGraphicsConfig extends X11GraphicsConfig implements
SurfaceManager.ProxiedGraphicsConfig { SurfaceManager.ProxiedGraphicsConfig {
private final SurfaceManager.ProxyCache surfaceDataProxyCache = new SurfaceManager.ProxyCache();
private XRGraphicsConfig(X11GraphicsDevice device, int visualnum, private XRGraphicsConfig(X11GraphicsDevice device, int visualnum,
int depth, int colormap, boolean doubleBuffer) { int depth, int colormap, boolean doubleBuffer) {
super(device, visualnum, depth, colormap, doubleBuffer); super(device, visualnum, depth, colormap, doubleBuffer);
@ -53,7 +55,8 @@ public class XRGraphicsConfig extends X11GraphicsConfig implements
doubleBuffer); doubleBuffer);
} }
public Object getProxyKey() { @Override
return this; public SurfaceManager.ProxyCache getSurfaceDataProxyCache() {
return surfaceDataProxyCache;
} }
} }

View File

@ -291,8 +291,7 @@ public abstract class XRSurfaceData extends XSurfaceData {
this.solidloops = graphicsConfig.getSolidLoops(sType); this.solidloops = graphicsConfig.getSolidLoops(sType);
this.depth = depth; this.depth = depth;
initOps(peer, graphicsConfig, depth); initOps(peer, graphicsConfig, depth);
setBlitProxyCache(gc.getSurfaceDataProxyCache());
setBlitProxyKey(gc.getProxyKey());
} }
protected XRSurfaceData(XRBackend renderQueue) { protected XRSurfaceData(XRBackend renderQueue) {

View File

@ -113,8 +113,8 @@ public class Win32GraphicsConfig extends GraphicsConfiguration
} }
@Override @Override
public Object getProxyKey() { public SurfaceManager.ProxyCache getSurfaceDataProxyCache() {
return device; return device.surfaceDataProxyCache;
} }
/** /**

View File

@ -41,6 +41,7 @@ import java.awt.image.ColorModel;
import java.awt.peer.WindowPeer; import java.awt.peer.WindowPeer;
import java.util.ArrayList; import java.util.ArrayList;
import sun.awt.image.SurfaceManager;
import sun.awt.windows.WWindowPeer; import sun.awt.windows.WWindowPeer;
import sun.java2d.SunGraphicsEnvironment; import sun.java2d.SunGraphicsEnvironment;
import sun.java2d.opengl.WGLGraphicsConfig; import sun.java2d.opengl.WGLGraphicsConfig;
@ -90,6 +91,8 @@ public class Win32GraphicsDevice extends GraphicsDevice implements
private float scaleX; private float scaleX;
private float scaleY; private float scaleY;
final SurfaceManager.ProxyCache surfaceDataProxyCache = new SurfaceManager.ProxyCache();
static { static {
// 4455041 - Even when ddraw is disabled, ddraw.dll is loaded when // 4455041 - Even when ddraw is disabled, ddraw.dll is loaded when

View File

@ -256,7 +256,7 @@ public class D3DSurfaceData extends SurfaceData implements AccelSurface {
} else { } else {
initSurface(); initSurface();
} }
setBlitProxyKey(gc.getProxyKey()); setBlitProxyCache(gc.getSurfaceDataProxyCache());
} }
@Override @Override

View File

@ -73,6 +73,7 @@ public final class WGLGraphicsConfig
private ContextCapabilities oglCaps; private ContextCapabilities oglCaps;
private final OGLContext context; private final OGLContext context;
private Object disposerReferent = new Object(); private Object disposerReferent = new Object();
private final SurfaceManager.ProxyCache surfaceDataProxyCache = new SurfaceManager.ProxyCache();
public static native int getDefaultPixFmt(int screennum); public static native int getDefaultPixFmt(int screennum);
private static native boolean initWGL(); private static native boolean initWGL();
@ -99,8 +100,8 @@ public final class WGLGraphicsConfig
} }
@Override @Override
public Object getProxyKey() { public SurfaceManager.ProxyCache getSurfaceDataProxyCache() {
return this; return surfaceDataProxyCache;
} }
@Override @Override

View File

@ -273,7 +273,7 @@ public class GDIWindowSurfaceData extends SurfaceData {
scaleX = gd.getDefaultScaleX(); scaleX = gd.getDefaultScaleX();
scaleY = gd.getDefaultScaleY(); scaleY = gd.getDefaultScaleY();
initOps(peer, depth, rMask, gMask, bMask, gd.getScreen()); initOps(peer, depth, rMask, gMask, bMask, gd.getScreen());
setBlitProxyKey(graphicsConfig.getProxyKey()); setBlitProxyCache(graphicsConfig.getSurfaceDataProxyCache());
} }
@Override @Override