wined3d: Proper CheckDeviceType / CheckDepthStencilMatch support.
diff --git a/dlls/wined3d/directx.c b/dlls/wined3d/directx.c
index 75951ea..13fcf51 100644
--- a/dlls/wined3d/directx.c
+++ b/dlls/wined3d/directx.c
@@ -1308,129 +1308,50 @@
     return WINED3D_OK;
 }
 
-static BOOL IWineD3DImpl_IsGLXFBConfigCompatibleWithRenderFmt(int iPixelFormat, WINED3DFORMAT Format) {
-#if 0 /* This code performs a strict test between the format and the current X11  buffer depth, which may give the best performance */
-  int gl_test;
-  int rb, gb, bb, ab, type, buf_sz;
+static BOOL IWineD3DImpl_IsPixelFormatCompatibleWithRenderFmt(const WineD3D_PixelFormat *cfg, WINED3DFORMAT Format) {
+    short redSize, greenSize, blueSize, alphaSize, colorBits;
 
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_RED_SIZE,   &rb);
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_GREEN_SIZE, &gb);
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_BLUE_SIZE,  &bb);
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_ALPHA_SIZE, &ab);
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_RENDER_TYPE, &type);
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_BUFFER_SIZE, &buf_sz);
+    if(!cfg)
+        return FALSE;
 
-  switch (Format) {
-  case WINED3DFMT_X8R8G8B8:
-  case WINED3DFMT_R8G8B8:
-    if (8 == rb && 8 == gb && 8 == bb) return TRUE;
-    break;
-  case WINED3DFMT_A8R8G8B8:
-    if (8 == rb && 8 == gb && 8 == bb && 8 == ab) return TRUE;
-    break;
-  case WINED3DFMT_A2R10G10B10:
-    if (10 == rb && 10 == gb && 10 == bb && 2 == ab) return TRUE;
-    break;
-  case WINED3DFMT_X1R5G5B5:
-    if (5 == rb && 5 == gb && 5 == bb) return TRUE;
-    break;
-  case WINED3DFMT_A1R5G5B5:
-    if (5 == rb && 5 == gb && 5 == bb && 1 == ab) return TRUE;
-    break;
-  case WINED3DFMT_X4R4G4B4:
-    if (16 == buf_sz && 4 == rb && 4 == gb && 4 == bb) return TRUE;
-    break;
-  case WINED3DFMT_R5G6B5:
-    if (5 == rb && 6 == gb && 5 == bb) return TRUE;
-    break;
-  case WINED3DFMT_R3G3B2:
-    if (3 == rb && 3 == gb && 2 == bb) return TRUE;
-    break;
-  case WINED3DFMT_A8P8:
-    if (type & GLX_COLOR_INDEX_BIT && 8 == buf_sz && 8 == ab) return TRUE;
-    break;
-  case WINED3DFMT_P8:
-    if (type & GLX_COLOR_INDEX_BIT && 8 == buf_sz) return TRUE;
-    break;
-  default:
-    break;
-  }
-  return FALSE;
-#else /* Most of the time performance is less of an issue than compatibility, this code allows for most common opengl/d3d formats */
-switch (Format) {
-  case WINED3DFMT_X8R8G8B8:
-  case WINED3DFMT_R8G8B8:
-  case WINED3DFMT_A8R8G8B8:
-  case WINED3DFMT_A2R10G10B10:
-  case WINED3DFMT_X1R5G5B5:
-  case WINED3DFMT_A1R5G5B5:
-  case WINED3DFMT_R5G6B5:
-  case WINED3DFMT_R3G3B2:
-  case WINED3DFMT_A8P8:
-  case WINED3DFMT_P8:
-return TRUE;
-  default:
-    break;
-  }
-return FALSE;
-#endif
+    if(!getColorBits(Format, &redSize, &greenSize, &blueSize, &alphaSize, &colorBits)) {
+        ERR("Unable to check compatibility for Format=%s\n", debug_d3dformat(Format));
+        return FALSE;
+    }
+
+    if(cfg->redSize < redSize)
+        return FALSE;
+
+    if(cfg->greenSize < greenSize)
+        return FALSE;
+
+    if(cfg->blueSize < blueSize)
+        return FALSE;
+
+    if(cfg->alphaSize < alphaSize)
+        return FALSE;
+
+    return TRUE;
 }
 
-static BOOL IWineD3DImpl_IsGLXFBConfigCompatibleWithDepthFmt(int iPixelFormat, WINED3DFORMAT Format) {
-#if 0/* This code performs a strict test between the format and the current X11  buffer depth, which may give the best performance */
-  int gl_test;
-  int db, sb;
+static BOOL IWineD3DImpl_IsPixelFormatCompatibleWithDepthFmt(const WineD3D_PixelFormat *cfg, WINED3DFORMAT Format) {
+    short depthSize, stencilSize;
 
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_DEPTH_SIZE, &db);
-  gl_test = glXGetFBConfigAttrib(display, cfgs, GLX_STENCIL_SIZE, &sb);
+    if(!cfg)
+        return FALSE;
 
-  switch (Format) {
-  case WINED3DFMT_D16:
-  case WINED3DFMT_D16_LOCKABLE:
-    if (16 == db) return TRUE;
-    break;
-  case WINED3DFMT_D32:
-    if (32 == db) return TRUE;
-    break;
-  case WINED3DFMT_D15S1:
-    if (15 == db) return TRUE;
-    break;
-  case WINED3DFMT_D24S8:
-    if (24 == db && 8 == sb) return TRUE;
-    break;
-  case WINED3DFMT_D24FS8:
-    if (24 == db && 8 == sb) return TRUE;
-    break;
-  case WINED3DFMT_D24X8:
-    if (24 == db) return TRUE;
-    break;
-  case WINED3DFMT_D24X4S4:
-    if (24 == db && 4 == sb) return TRUE;
-    break;
-  case WINED3DFMT_D32F_LOCKABLE:
-    if (32 == db) return TRUE;
-    break;
-  default:
-    break;
-  }
-  return FALSE;
-#else /* Most of the time performance is less of an issue than compatibility, this code allows for most common opengl/d3d formats */
-  switch (Format) {
-  case WINED3DFMT_D16:
-  case WINED3DFMT_D16_LOCKABLE:
-  case WINED3DFMT_D32:
-  case WINED3DFMT_D15S1:
-  case WINED3DFMT_D24S8:
-  case WINED3DFMT_D24FS8:
-  case WINED3DFMT_D24X8:
-  case WINED3DFMT_D24X4S4:
-  case WINED3DFMT_D32F_LOCKABLE:
+    if(!getDepthStencilBits(Format, &depthSize, &stencilSize)) {
+        ERR("Unable to check compatibility for Format=%s\n", debug_d3dformat(Format));
+        return FALSE;
+    }
+
+    if(cfg->depthSize < depthSize)
+        return FALSE;
+
+    if(cfg->stencilSize < stencilSize)
+        return FALSE;
+
     return TRUE;
-  default:
-    break;
-  }
-  return FALSE;
-#endif
 }
 
 static HRESULT WINAPI IWineD3DImpl_CheckDepthStencilMatch(IWineD3D *iface, UINT Adapter, WINED3DDEVTYPE DeviceType,
@@ -1438,6 +1359,8 @@
                                                    WINED3DFORMAT RenderTargetFormat,
                                                    WINED3DFORMAT DepthStencilFormat) {
     IWineD3DImpl *This = (IWineD3DImpl *)iface;
+    int nCfgs;
+    WineD3D_PixelFormat *cfgs;
     int it;
 
     WARN_(d3d_caps)("(%p)-> (STUB) (Adptr:%d, DevType:(%x,%s), AdptFmt:(%x,%s), RendrTgtFmt:(%x,%s), DepthStencilFmt:(%x,%s))\n",
@@ -1452,9 +1375,11 @@
         return WINED3DERR_INVALIDCALL;
     }
 
-    for (it = 0; it < Adapters[Adapter].nCfgs; ++it) {
-        if (IWineD3DImpl_IsGLXFBConfigCompatibleWithRenderFmt(it, RenderTargetFormat)) {
-            if (IWineD3DImpl_IsGLXFBConfigCompatibleWithDepthFmt(it, DepthStencilFormat)) {
+    cfgs = Adapters[Adapter].cfgs;
+    nCfgs = Adapters[Adapter].nCfgs;
+    for (it = 0; it < nCfgs; ++it) {
+        if (IWineD3DImpl_IsPixelFormatCompatibleWithRenderFmt(&cfgs[it], RenderTargetFormat)) {
+            if (IWineD3DImpl_IsPixelFormatCompatibleWithDepthFmt(&cfgs[it], DepthStencilFormat)) {
                 TRACE_(d3d_caps)("(%p) : Formats matched\n", This);
                 return WINED3D_OK;
             }
@@ -1502,6 +1427,7 @@
 
     IWineD3DImpl *This = (IWineD3DImpl *)iface;
     int nCfgs = 0;
+    WineD3D_PixelFormat *cfgs;
     int it;
     HRESULT hr = WINED3DERR_NOTAVAILABLE;
 
@@ -1518,23 +1444,20 @@
         return WINED3DERR_INVALIDCALL;
     }
 
-    /* TODO: Store in adapter structure */
-    if (WineD3D_CreateFakeGLContext()) {
-        nCfgs = DescribePixelFormat(wined3d_fake_gl_context_hdc, 0, 0, NULL);
-        for (it = 0; it < nCfgs; ++it) {
-            if (IWineD3DImpl_IsGLXFBConfigCompatibleWithRenderFmt(it, DisplayFormat)) {
-                hr = WINED3D_OK;
-                TRACE_(d3d_caps)("OK\n");
-                break ;
-            }
+    cfgs = Adapters[Adapter].cfgs;
+    nCfgs = Adapters[Adapter].nCfgs;
+    for (it = 0; it < nCfgs; ++it) {
+        if (IWineD3DImpl_IsPixelFormatCompatibleWithRenderFmt(&cfgs[it], DisplayFormat)) {
+            hr = WINED3D_OK;
+            TRACE_(d3d_caps)("OK\n");
+            break ;
         }
-
-        if(hr != WINED3D_OK)
-            ERR("unsupported format %s\n", debug_d3dformat(DisplayFormat));
-        WineD3D_ReleaseFakeGLContext();
     }
 
     if(hr != WINED3D_OK)
+        ERR("unsupported format %s\n", debug_d3dformat(DisplayFormat));
+
+    if(hr != WINED3D_OK)
         TRACE_(d3d_caps)("returning something different from WINED3D_OK\n");
 
     return hr;
@@ -2541,6 +2464,7 @@
     return IUnknown_Release(volumeParent);
 }
 
+#define PUSH1(att)        attribs[nAttribs++] = (att);
 #define GLINFO_LOCATION (Adapters[0].gl_info)
 BOOL InitAdapters(void) {
     BOOL ret;
@@ -2576,8 +2500,43 @@
         Adapters[0].description = "Direct3D HAL";
 
         if (WineD3D_CreateFakeGLContext()) {
+            int iPixelFormat;
+            int attribs[8];
+            int values[8];
+            int nAttribs = 0;
+            int res;
+            WineD3D_PixelFormat *cfgs;
+
             attribute = WGL_NUMBER_PIXEL_FORMATS_ARB;
             GL_EXTCALL(wglGetPixelFormatAttribivARB(wined3d_fake_gl_context_hdc, 0, 0, 1, &attribute, &Adapters[0].nCfgs));
+
+            Adapters[0].cfgs = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, Adapters[0].nCfgs *sizeof(WineD3D_PixelFormat));
+            cfgs = Adapters[0].cfgs;
+            PUSH1(WGL_RED_BITS_ARB)
+            PUSH1(WGL_GREEN_BITS_ARB)
+            PUSH1(WGL_BLUE_BITS_ARB)
+            PUSH1(WGL_ALPHA_BITS_ARB)
+            PUSH1(WGL_DEPTH_BITS_ARB)
+            PUSH1(WGL_STENCIL_BITS_ARB)
+
+            for(iPixelFormat=1; iPixelFormat<=Adapters[0].nCfgs; iPixelFormat++) {
+                res = GL_EXTCALL(wglGetPixelFormatAttribivARB(wined3d_fake_gl_context_hdc, iPixelFormat, 0, nAttribs, attribs, values));
+
+                if(!res)
+                    continue;
+
+                /* Cache the pixel format */
+                cfgs->iPixelFormat = iPixelFormat;
+                cfgs->redSize = values[0];
+                cfgs->greenSize = values[1];
+                cfgs->blueSize = values[2];
+                cfgs->alphaSize = values[3];
+                cfgs->depthSize = values[4];
+                cfgs->stencilSize = values[5];
+
+                TRACE("iPixelFormat=%d, RGBA=%d/%d/%d/%d, depth=%d, stencil=%d\n", cfgs->iPixelFormat, cfgs->redSize, cfgs->greenSize, cfgs->blueSize, cfgs->alphaSize, cfgs->depthSize, cfgs->stencilSize);
+                cfgs++;
+            }
             WineD3D_ReleaseFakeGLContext();
         }
 
@@ -2590,6 +2549,7 @@
 
     return TRUE;
 }
+#undef PUSH1
 #undef GLINFO_LOCATION
 
 /**********************************************************
diff --git a/dlls/wined3d/wined3d_private.h b/dlls/wined3d/wined3d_private.h
index c996b23..3ceb767 100644
--- a/dlls/wined3d/wined3d_private.h
+++ b/dlls/wined3d/wined3d_private.h
@@ -573,6 +573,13 @@
 /* The default light parameters */
 extern const WINED3DLIGHT WINED3D_default_light;
 
+typedef struct WineD3D_PixelFormat
+{
+    int iPixelFormat; /* WGL pixel format */
+    int redSize, greenSize, blueSize, alphaSize;
+    int depthSize, stencilSize;
+} WineD3D_PixelFormat;
+
 /* The adapter structure */
 typedef struct GLPixelFormatDesc GLPixelFormatDesc;
 struct WineD3DAdapter
@@ -582,6 +589,7 @@
     const char              *driver;
     const char              *description;
     int                     nCfgs;
+    WineD3D_PixelFormat     *cfgs;
 };
 
 extern BOOL InitAdapters(void);