wgl: Add initial WGL_ARB_create_context support.

This commit is contained in:
Roderick Colenbrander 2009-11-16 23:24:24 +01:00 committed by Alexandre Julliard
parent 3323ae9dd6
commit 41f061ea1b
6 changed files with 158 additions and 3 deletions

View File

@ -198,6 +198,7 @@ static struct graphics_driver *create_driver( HMODULE module )
/* OpenGL32 */ /* OpenGL32 */
GET_FUNC(wglCreateContext); GET_FUNC(wglCreateContext);
GET_FUNC(wglCreateContextAttribsARB);
GET_FUNC(wglDeleteContext); GET_FUNC(wglDeleteContext);
GET_FUNC(wglGetProcAddress); GET_FUNC(wglGetProcAddress);
GET_FUNC(wglGetPbufferDCARB); GET_FUNC(wglGetPbufferDCARB);

View File

@ -199,6 +199,7 @@ typedef struct tagDC_FUNCS
/* OpenGL32 */ /* OpenGL32 */
BOOL (CDECL *pwglCopyContext)(HGLRC, HGLRC, UINT); BOOL (CDECL *pwglCopyContext)(HGLRC, HGLRC, UINT);
HGLRC (CDECL *pwglCreateContext)(PHYSDEV); HGLRC (CDECL *pwglCreateContext)(PHYSDEV);
HGLRC (CDECL *pwglCreateContextAttribsARB)(PHYSDEV, HGLRC, const int*);
BOOL (CDECL *pwglDeleteContext)(HGLRC); BOOL (CDECL *pwglDeleteContext)(HGLRC);
PROC (CDECL *pwglGetProcAddress)(LPCSTR); PROC (CDECL *pwglGetProcAddress)(LPCSTR);
HDC (CDECL *pwglGetPbufferDCARB)(PHYSDEV, void*); HDC (CDECL *pwglGetPbufferDCARB)(PHYSDEV, void*);

View File

@ -104,6 +104,25 @@ HGLRC WINAPI wglCreateContext(HDC hdc)
return ret; return ret;
} }
/***********************************************************************
* wglCreateContextAttribsARB
*/
HGLRC WINAPI wglCreateContextAttribsARB(HDC hdc, HGLRC hShareContext, const int *attributeList)
{
HGLRC ret = 0;
DC * dc = get_dc_ptr( hdc );
TRACE("(%p)\n",hdc);
if (!dc) return 0;
update_dc( dc );
if (!dc->funcs->pwglCreateContextAttribsARB) FIXME(" :stub\n");
else ret = dc->funcs->pwglCreateContextAttribsARB(dc->physDev, hShareContext, attributeList);
release_dc_ptr( dc );
return ret;
}
/*********************************************************************** /***********************************************************************
* wglDeleteContext (OPENGL32.@) * wglDeleteContext (OPENGL32.@)
@ -358,7 +377,9 @@ PROC WINAPI wglGetProcAddress(LPCSTR func)
* when a non-NULL value is returned by wglGetProcAddress), we return the address * when a non-NULL value is returned by wglGetProcAddress), we return the address
* of a wrapper function which will handle the HDC->PhysDev conversion. * of a wrapper function which will handle the HDC->PhysDev conversion.
*/ */
if(ret && strcmp(func, "wglMakeContextCurrentARB") == 0) if(ret && strcmp(func, "wglCreateContextAttribsARB") == 0)
return (PROC)wglCreateContextAttribsARB;
else if(ret && strcmp(func, "wglMakeContextCurrentARB") == 0)
return (PROC)wglMakeContextCurrentARB; return (PROC)wglMakeContextCurrentARB;
else if(ret && strcmp(func, "wglGetPbufferDCARB") == 0) else if(ret && strcmp(func, "wglGetPbufferDCARB") == 0)
return (PROC)wglGetPbufferDCARB; return (PROC)wglGetPbufferDCARB;

View File

@ -591,7 +591,7 @@ static void test_opengl3(HDC hdc)
HGLRC gl3Ctx; HGLRC gl3Ctx;
DWORD error; DWORD error;
gl3Ctx = pwglCreateContextAttribsARB(hdc, (HGLRC)0xdeadbeef, 0); gl3Ctx = pwglCreateContextAttribsARB(hdc, (HGLRC)0xdeadbeef, 0);
ok(gl3Ctx == 0, "pwglCreateContextAttribsARB using an invalid shareList passed\n"); todo_wine ok(gl3Ctx == 0, "pwglCreateContextAttribsARB using an invalid shareList passed\n");
error = GetLastError(); error = GetLastError();
todo_wine ok(error == ERROR_INVALID_OPERATION, "Expected ERROR_INVALID_OPERATION, got error=%x\n", error); todo_wine ok(error == ERROR_INVALID_OPERATION, "Expected ERROR_INVALID_OPERATION, got error=%x\n", error);
wglDeleteContext(gl3Ctx); wglDeleteContext(gl3Ctx);

View File

@ -105,8 +105,11 @@ typedef struct wine_glcontext {
BOOL do_escape; BOOL do_escape;
BOOL has_been_current; BOOL has_been_current;
BOOL sharing; BOOL sharing;
BOOL gl3_context;
XVisualInfo *vis; XVisualInfo *vis;
WineGLPixelFormat *fmt; WineGLPixelFormat *fmt;
int numAttribs; /* This is needed for delaying wglCreateContextAttribsARB */
int attribList[16]; /* This is needed for delaying wglCreateContextAttribsARB */
GLXContext ctx; GLXContext ctx;
HDC read_hdc; HDC read_hdc;
Drawable drawables[2]; Drawable drawables[2];
@ -240,6 +243,7 @@ MAKE_FUNCPTR(glXQueryDrawable)
MAKE_FUNCPTR(glXGetCurrentReadDrawable) MAKE_FUNCPTR(glXGetCurrentReadDrawable)
/* GLX Extensions */ /* GLX Extensions */
static GLXContext (*pglXCreateContextAttribsARB)(Display *dpy, GLXFBConfig config, GLXContext share_context, Bool direct, const int *attrib_list);
static void* (*pglXGetProcAddressARB)(const GLubyte *); static void* (*pglXGetProcAddressARB)(const GLubyte *);
static int (*pglXSwapIntervalSGI)(int); static int (*pglXSwapIntervalSGI)(int);
@ -274,6 +278,12 @@ MAKE_FUNCPTR(glFinish)
MAKE_FUNCPTR(glFlush) MAKE_FUNCPTR(glFlush)
#undef MAKE_FUNCPTR #undef MAKE_FUNCPTR
static int GLXErrorHandler(Display *dpy, XErrorEvent *event, void *arg)
{
/* In the future we might want to find the exact X or GLX error to report back to the app */
return 1;
}
static BOOL infoInitialized = FALSE; static BOOL infoInitialized = FALSE;
static BOOL X11DRV_WineGL_InitOpenglInfo(void) static BOOL X11DRV_WineGL_InitOpenglInfo(void)
{ {
@ -442,6 +452,8 @@ static BOOL has_opengl(void)
the associated extension is available (and if a driver reports the extension the associated extension is available (and if a driver reports the extension
is available but fails to provide the functions, it's quite broken) */ is available but fails to provide the functions, it's quite broken) */
#define LOAD_FUNCPTR(f) p##f = (void*)pglXGetProcAddressARB((const unsigned char*)#f) #define LOAD_FUNCPTR(f) p##f = (void*)pglXGetProcAddressARB((const unsigned char*)#f)
/* ARB GLX Extension */
LOAD_FUNCPTR(glXCreateContextAttribsARB);
/* NV GLX Extension */ /* NV GLX Extension */
LOAD_FUNCPTR(glXAllocateMemoryNV); LOAD_FUNCPTR(glXAllocateMemoryNV);
LOAD_FUNCPTR(glXFreeMemoryNV); LOAD_FUNCPTR(glXFreeMemoryNV);
@ -1066,7 +1078,14 @@ static GLXContext create_glxcontext(Display *display, Wine_GLContext *context, G
/* We use indirect rendering for rendering to bitmaps. See get_formats for a comment about this. */ /* We use indirect rendering for rendering to bitmaps. See get_formats for a comment about this. */
BOOL indirect = (context->fmt->dwFlags & PFD_DRAW_TO_BITMAP) ? FALSE : TRUE; BOOL indirect = (context->fmt->dwFlags & PFD_DRAW_TO_BITMAP) ? FALSE : TRUE;
if(context->vis) if(context->gl3_context)
{
if(context->numAttribs)
ctx = pglXCreateContextAttribsARB(gdi_display, context->fmt->fbconfig, shareList, indirect, context->attribList);
else
ctx = pglXCreateContextAttribsARB(gdi_display, context->fmt->fbconfig, shareList, indirect, NULL);
}
else if(context->vis)
ctx = pglXCreateContext(gdi_display, context->vis, shareList, indirect); ctx = pglXCreateContext(gdi_display, context->vis, shareList, indirect);
else /* Create a GLX Context for a pbuffer */ else /* Create a GLX Context for a pbuffer */
ctx = pglXCreateNewContext(gdi_display, context->fmt->fbconfig, context->fmt->render_type, shareList, TRUE); ctx = pglXCreateNewContext(gdi_display, context->fmt->fbconfig, context->fmt->render_type, shareList, TRUE);
@ -2171,6 +2190,94 @@ static void WINAPI X11DRV_wglFlush(void)
if (ctx) ExtEscape(ctx->hdc, X11DRV_ESCAPE, sizeof(code), (LPSTR)&code, 0, NULL ); if (ctx) ExtEscape(ctx->hdc, X11DRV_ESCAPE, sizeof(code), (LPSTR)&code, 0, NULL );
} }
/**
* X11DRV_wglCreateContextAttribsARB
*
* WGL_ARB_create_context: wglCreateContextAttribsARB
*/
HGLRC X11DRV_wglCreateContextAttribsARB(X11DRV_PDEVICE *physDev, HGLRC hShareContext, const int* attribList)
{
Wine_GLContext *ret;
WineGLPixelFormat *fmt;
int hdcPF = physDev->current_pf;
int fmt_count = 0;
TRACE("(%p %p %p)\n", physDev, hShareContext, attribList);
if (!has_opengl()) return 0;
fmt = ConvertPixelFormatWGLtoGLX(gdi_display, hdcPF, TRUE /* Offscreen */, &fmt_count);
/* wglCreateContextAttribsARB supports ALL pixel formats, so also offscreen ones.
* If this fails something is very wrong on the system. */
if(!fmt)
{
ERR("Cannot get FB Config for iPixelFormat %d, expect problems!\n", hdcPF);
SetLastError(ERROR_INVALID_PIXEL_FORMAT);
return NULL;
}
wine_tsx11_lock();
ret = alloc_context();
wine_tsx11_unlock();
ret->hdc = physDev->hdc;
ret->fmt = fmt;
ret->vis = NULL; /* glXCreateContextAttribsARB requires a fbconfig instead of a visual */
ret->gl3_context = TRUE;
ret->numAttribs = 0;
if(attribList)
{
int *pAttribList = (int*)attribList;
int *pContextAttribList = &ret->attribList[0];
/* attribList consists of pairs {token, value] terminated with 0 */
while(pAttribList[0] != 0)
{
TRACE("%#x %#x\n", pAttribList[0], pAttribList[1]);
switch(pAttribList[0])
{
case WGL_CONTEXT_MAJOR_VERSION_ARB:
pContextAttribList[0] = GLX_CONTEXT_MAJOR_VERSION_ARB;
pContextAttribList[1] = pAttribList[1];
break;
case WGL_CONTEXT_MINOR_VERSION_ARB:
pContextAttribList[0] = GLX_CONTEXT_MINOR_VERSION_ARB;
pContextAttribList[1] = pAttribList[1];
break;
case WGL_CONTEXT_LAYER_PLANE_ARB:
break;
case WGL_CONTEXT_FLAGS_ARB:
pContextAttribList[0] = GLX_CONTEXT_FLAGS_ARB;
pContextAttribList[1] = pAttribList[1];
break;
default:
ERR("Unhandled attribList pair: %#x %#x\n", pAttribList[0], pAttribList[1]);
}
ret->numAttribs++;
pAttribList += 2;
pContextAttribList += 2;
}
}
wine_tsx11_lock();
X11DRV_expect_error(gdi_display, GLXErrorHandler, NULL);
ret->ctx = create_glxcontext(gdi_display, ret, NULL);
XSync(gdi_display, False);
if(X11DRV_check_error() || !ret->ctx)
{
/* In the future we should convert the GLX error to a win32 one here if needed */
ERR("Context creation failed\n");
free_context(ret);
wine_tsx11_unlock();
return NULL;
}
wine_tsx11_unlock();
TRACE(" creating context %p\n", ret);
return (HGLRC) ret;
}
/** /**
* X11DRV_wglGetExtensionsStringARB * X11DRV_wglGetExtensionsStringARB
* *
@ -3322,6 +3429,14 @@ static const WineGLExtension WGL_internal_functions =
}; };
static const WineGLExtension WGL_ARB_create_context =
{
"WGL_ARB_create_context",
{
{ "wglCreateContextAttribsARB", X11DRV_wglCreateContextAttribsARB },
}
};
static const WineGLExtension WGL_ARB_extensions_string = static const WineGLExtension WGL_ARB_extensions_string =
{ {
"WGL_ARB_extensions_string", "WGL_ARB_extensions_string",
@ -3422,6 +3537,11 @@ static void X11DRV_WineGL_LoadExtensions(void)
/* ARB Extensions */ /* ARB Extensions */
if(glxRequireExtension("GLX_ARB_create_context"))
{
register_extension(&WGL_ARB_create_context);
}
if(glxRequireExtension("GLX_ARB_fbconfig_float")) if(glxRequireExtension("GLX_ARB_fbconfig_float"))
{ {
register_extension_string("WGL_ARB_pixel_format_float"); register_extension_string("WGL_ARB_pixel_format_float");
@ -3693,6 +3813,17 @@ HGLRC CDECL X11DRV_wglCreateContext(X11DRV_PDEVICE *physDev) {
return NULL; return NULL;
} }
/**
* X11DRV_wglCreateContextAttribsARB
*
* WGL_ARB_create_context: wglCreateContextAttribsARB
*/
HGLRC X11DRV_wglCreateContextAttribsARB(X11DRV_PDEVICE *physDev, HGLRC hShareContext, const int* attribList)
{
opengl_error();
return NULL;
}
/** /**
* X11DRV_wglDeleteContext * X11DRV_wglDeleteContext
* *

View File

@ -133,6 +133,7 @@
# OpenGL # OpenGL
@ cdecl wglCopyContext(long long long) X11DRV_wglCopyContext @ cdecl wglCopyContext(long long long) X11DRV_wglCopyContext
@ cdecl wglCreateContext(ptr) X11DRV_wglCreateContext @ cdecl wglCreateContext(ptr) X11DRV_wglCreateContext
@ cdecl wglCreateContextAttribsARB(ptr long ptr) X11DRV_wglCreateContextAttribsARB
@ cdecl wglDeleteContext(long) X11DRV_wglDeleteContext @ cdecl wglDeleteContext(long) X11DRV_wglDeleteContext
@ cdecl wglGetProcAddress(str) X11DRV_wglGetProcAddress @ cdecl wglGetProcAddress(str) X11DRV_wglGetProcAddress
@ cdecl wglGetPbufferDCARB(ptr ptr) X11DRV_wglGetPbufferDCARB @ cdecl wglGetPbufferDCARB(ptr ptr) X11DRV_wglGetPbufferDCARB