Commit dee2fc09 authored by Roderick Colenbrander's avatar Roderick Colenbrander Committed by Alexandre Julliard

wgl: Rewrite ChoosePixelFormat.

parent ae98679e
...@@ -1009,9 +1009,18 @@ static int ConvertPixelFormatGLXtoWGL(Display *display, int fmt_id) ...@@ -1009,9 +1009,18 @@ static int ConvertPixelFormatGLXtoWGL(Display *display, int fmt_id)
*/ */
int X11DRV_ChoosePixelFormat(X11DRV_PDEVICE *physDev, int X11DRV_ChoosePixelFormat(X11DRV_PDEVICE *physDev,
const PIXELFORMATDESCRIPTOR *ppfd) { const PIXELFORMATDESCRIPTOR *ppfd) {
WineGLPixelFormat *fmt; WineGLPixelFormat *fmt = NULL;
int ret = 0; int ret = 0;
int nPixelFormats;
int value = 0; int value = 0;
int i = 0;
int bestFormat = -1;
int bestColor = -1;
int bestAlpha = -1;
int bestDepth = -1;
int bestStencil = -1;
int bestAux = -1;
int score;
if (!has_opengl()) { if (!has_opengl()) {
ERR("No libGL on this box - disabling OpenGL support !\n"); ERR("No libGL on this box - disabling OpenGL support !\n");
...@@ -1025,19 +1034,15 @@ int X11DRV_ChoosePixelFormat(X11DRV_PDEVICE *physDev, ...@@ -1025,19 +1034,15 @@ int X11DRV_ChoosePixelFormat(X11DRV_PDEVICE *physDev,
} }
wine_tsx11_lock(); wine_tsx11_lock();
if(!visual) { ConvertPixelFormatWGLtoGLX(gdi_display, 0, FALSE /* offscreen */, &nPixelFormats);
ERR("Can't get an opengl visual!\n"); for(i=0; i<nPixelFormats; i++)
goto choose_exit; {
}
fmt = ConvertPixelFormatWGLtoGLX(gdi_display, 1 /* main visual */, FALSE /* offscreen */, &value);
/* In case an fbconfig was found, check if it matches to the requirements of the ppfd */
if(!fmt) {
ERR("Can't find a matching FBCONFIG_ID for VISUAL_ID 0x%lx!\n", visual->visualid);
} else {
int dwFlags = 0; int dwFlags = 0;
int iPixelType = 0; int iPixelType = 0;
int value = 0; int alpha=0, color=0, depth=0, stencil=0, aux=0;
fmt = ConvertPixelFormatWGLtoGLX(gdi_display, i+1 /* 1-based index */, FALSE /* offscreen */, &value);
score = 0;
/* Pixel type */ /* Pixel type */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_RENDER_TYPE, &value); pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_RENDER_TYPE, &value);
...@@ -1046,67 +1051,137 @@ int X11DRV_ChoosePixelFormat(X11DRV_PDEVICE *physDev, ...@@ -1046,67 +1051,137 @@ int X11DRV_ChoosePixelFormat(X11DRV_PDEVICE *physDev,
else else
iPixelType = PFD_TYPE_COLORINDEX; iPixelType = PFD_TYPE_COLORINDEX;
if (ppfd->iPixelType != iPixelType) { if (ppfd->iPixelType != iPixelType)
TRACE("pixel type mismatch\n"); {
goto choose_exit; TRACE("pixel type mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* Doublebuffer */ /* Doublebuffer:
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_DOUBLEBUFFER, &value); if (value) dwFlags |= PFD_DOUBLEBUFFER; * Ignore doublebuffer when PFD_DOUBLEBUFFER_DONTCARE is set. Further if the flag
if (!(ppfd->dwFlags & PFD_DOUBLEBUFFER_DONTCARE) && (ppfd->dwFlags & PFD_DOUBLEBUFFER)) { * is not set, we must return a format without double buffering. */
if (!(dwFlags & PFD_DOUBLEBUFFER)) { pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_DOUBLEBUFFER, &value);
TRACE("dbl buffer mismatch\n"); if (value) dwFlags |= PFD_DOUBLEBUFFER;
goto choose_exit; if (!(ppfd->dwFlags & PFD_DOUBLEBUFFER_DONTCARE) && ((ppfd->dwFlags^dwFlags) & PFD_DOUBLEBUFFER))
} {
TRACE("dbl buffer mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* Stereo */ /* Stereo:
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_STEREO, &value); if (value) dwFlags |= PFD_STEREO; * Ignore stereo when PFD_STEREO_DONTCARE is set. Further if the flag
if (!(ppfd->dwFlags & PFD_STEREO_DONTCARE) && (ppfd->dwFlags & PFD_STEREO)) { * is not set, we must return a format without stereo support. */
if (!(dwFlags & PFD_STEREO)) { pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_STEREO, &value);
TRACE("stereo mismatch\n"); if (value) dwFlags |= PFD_STEREO;
goto choose_exit; if (!(ppfd->dwFlags & PFD_STEREO_DONTCARE) && ((ppfd->dwFlags^dwFlags) & PFD_STEREO))
{
TRACE("stereo mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_BUFFER_SIZE, &color); /* cColorBits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_ALPHA_SIZE, &alpha); /* cAlphaBits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_DEPTH_SIZE, &depth); /* cDepthBits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_STENCIL_SIZE, &stencil); /* cStencilBits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_AUX_BUFFERS, &aux); /* cAuxBuffers */
/* Below we will do a number of checks to select the 'best' pixelformat.
* We assume the precedence cColorBits > cAlphaBits > cDepthBits > cStencilBits -> cAuxBuffers.
* The code works by trying to match the most important options as close as possible.
* When a reasonable format is found, we will try to match more options. */
/* Color bits */
if( ((ppfd->cColorBits > bestColor) && (color > bestColor)) ||
((color >= ppfd->cColorBits) && (color < bestColor)) )
{
bestAlpha = alpha;
bestColor = color;
bestDepth = depth;
bestStencil = stencil;
bestAux = aux;
bestFormat = i;
continue;
} else if(bestColor != color) { /* Do further checks if the format is compatible */
TRACE("color mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* Alpha bits */ /* Alpha bits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_ALPHA_SIZE, &value); if( ((ppfd->cAlphaBits > bestAlpha) && (alpha > bestAlpha)) ||
if (ppfd->iPixelType==PFD_TYPE_RGBA && ppfd->cAlphaBits && !value) { ((alpha >= ppfd->cAlphaBits) && (alpha < bestAlpha)) )
TRACE("alpha mismatch\n"); {
goto choose_exit; bestAlpha = alpha;
bestColor = color;
bestDepth = depth;
bestStencil = stencil;
bestAux = aux;
bestFormat = i;
continue;
} else if(bestAlpha != alpha) {
TRACE("alpha mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* Depth bits */ /* Depth bits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_DEPTH_SIZE, &value); if( ((ppfd->cDepthBits > bestDepth) && (depth > bestDepth)) ||
if (ppfd->cDepthBits && !value) { ((depth >= ppfd->cDepthBits) && (depth < bestDepth)) )
TRACE("depth mismatch\n"); {
goto choose_exit; bestAlpha = alpha;
bestColor = color;
bestDepth = depth;
bestStencil = stencil;
bestAux = aux;
bestFormat = i;
continue;
} else if(bestDepth != depth) {
TRACE("depth mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* Stencil bits */ /* Stencil bits */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_STENCIL_SIZE, &value); if( ((ppfd->cStencilBits > bestStencil) && (stencil > bestStencil)) ||
if (ppfd->cStencilBits && !value) { ((stencil >= ppfd->cStencilBits) && (stencil < bestStencil)) )
TRACE("stencil mismatch\n"); {
goto choose_exit; bestAlpha = alpha;
bestColor = color;
bestDepth = depth;
bestStencil = stencil;
bestAux = aux;
bestFormat = i;
continue;
} else if(bestStencil != stencil) {
TRACE("stencil mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* Aux buffers */ /* Aux buffers */
pglXGetFBConfigAttrib(gdi_display, fmt->fbconfig, GLX_AUX_BUFFERS, &value); if( ((ppfd->cAuxBuffers > bestAux) && (aux > bestAux)) ||
if (ppfd->cAuxBuffers && !value) { ((aux >= ppfd->cAuxBuffers) && (aux < bestAux)) )
TRACE("aux mismatch\n"); {
goto choose_exit; bestAlpha = alpha;
bestColor = color;
bestDepth = depth;
bestStencil = stencil;
bestAux = aux;
bestFormat = i;
continue;
} else if(bestAux != aux) {
TRACE("aux mismatch for iPixelFormat=%d\n", i+1);
continue;
} }
/* When we pass all the checks we have found a matching format :) */
ret = 1;
TRACE("Successfully found a matching mode, returning index: %d\n", ret);
} }
choose_exit: if(bestFormat == -1) {
if(!ret)
TRACE("No matching mode was found returning 0\n"); TRACE("No matching mode was found returning 0\n");
ret = 0;
}
else {
ret = bestFormat+1; /* the return value should be a 1-based index */
TRACE("Successfully found a matching mode, returning index: %d %x\n", ret, WineGLPixelFormatList[bestFormat-1].fmt_id);
}
wine_tsx11_unlock(); wine_tsx11_unlock();
return ret; return ret;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment