2000-05-12 20:18:14 +00:00
|
|
|
/* Window-specific OpenGL functions implementation.
|
|
|
|
|
|
|
|
Copyright (c) 1999 Lionel Ulmer
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2000-05-23 21:15:06 +00:00
|
|
|
#include "wine/exception.h"
|
|
|
|
|
2000-05-12 20:18:14 +00:00
|
|
|
#include "config.h"
|
|
|
|
#include "debugtools.h"
|
|
|
|
#include "gdi.h"
|
|
|
|
#include "dc.h"
|
|
|
|
#include "windef.h"
|
2000-05-23 21:15:06 +00:00
|
|
|
#include "winerror.h"
|
2000-05-12 20:18:14 +00:00
|
|
|
#include "wine_gl.h"
|
|
|
|
#include "x11drv.h"
|
2000-05-23 01:20:08 +00:00
|
|
|
#include "x11font.h"
|
2000-05-12 20:18:14 +00:00
|
|
|
|
|
|
|
#include "wgl.h"
|
|
|
|
#include "opengl_ext.h"
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
DEFAULT_DEBUG_CHANNEL(opengl);
|
2000-05-12 20:18:14 +00:00
|
|
|
|
2000-05-23 21:15:06 +00:00
|
|
|
static int XGLErrorFlag = 0;
|
|
|
|
static int XGLErrorHandler(Display *dpy, XErrorEvent *event) {
|
|
|
|
XGLErrorFlag = 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* filter for page-fault exceptions */
|
|
|
|
static WINE_EXCEPTION_FILTER(page_fault)
|
|
|
|
{
|
|
|
|
return EXCEPTION_EXECUTE_HANDLER;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglCreateContext
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
HGLRC WINAPI wglCreateContext(HDC hdc) {
|
|
|
|
DC * dc = DC_GetDCPtr( hdc );
|
|
|
|
X11DRV_PDEVICE *physDev;
|
|
|
|
XVisualInfo *vis;
|
|
|
|
GLXContext ret;
|
|
|
|
|
|
|
|
TRACE("(%08x)\n", hdc);
|
|
|
|
|
|
|
|
if (dc == NULL) {
|
|
|
|
ERR("Null DC !!!\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
physDev = (X11DRV_PDEVICE *)dc->physDev;
|
|
|
|
|
|
|
|
/* First, get the visual for the choosen pixel format */
|
|
|
|
vis = physDev->visuals[physDev->current_pf - 1];
|
|
|
|
|
|
|
|
if (vis == NULL) {
|
|
|
|
ERR("NULL visual !!!\n");
|
|
|
|
/* Need to set errors here */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ENTER_GL();
|
|
|
|
ret = glXCreateContext(display, vis, NULL, True);
|
|
|
|
LEAVE_GL();
|
|
|
|
|
|
|
|
return (HGLRC) ret;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglCreateLayerContext
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
HGLRC WINAPI wglCreateLayerContext(HDC hdc,
|
|
|
|
int iLayerPlane) {
|
|
|
|
FIXME("(%08x,%d): stub !\n", hdc, iLayerPlane);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglCopyContext
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglCopyContext(HGLRC hglrcSrc,
|
|
|
|
HGLRC hglrcDst,
|
|
|
|
UINT mask) {
|
|
|
|
FIXME("(%p,%p,%d)\n", hglrcSrc, hglrcDst, mask);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglDeleteContext
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglDeleteContext(HGLRC hglrc) {
|
2000-05-23 21:15:06 +00:00
|
|
|
int (*WineXHandler)(Display *, XErrorEvent *);
|
|
|
|
BOOL ret = TRUE;
|
|
|
|
|
2000-05-23 01:20:08 +00:00
|
|
|
TRACE("(%p)\n", hglrc);
|
2000-05-23 21:15:06 +00:00
|
|
|
|
2000-05-23 01:20:08 +00:00
|
|
|
ENTER_GL();
|
2000-05-23 21:15:06 +00:00
|
|
|
/* A game (Half Life not to name it) deletes twice the same context. To prevent
|
|
|
|
crashes, run with our own error function enabled */
|
|
|
|
XSync(display, False);
|
|
|
|
XGLErrorFlag = 0;
|
|
|
|
WineXHandler = XSetErrorHandler(XGLErrorHandler);
|
|
|
|
__TRY {
|
|
|
|
glXDestroyContext(display, (GLXContext) hglrc);
|
|
|
|
XSync(display, False);
|
|
|
|
XFlush(display);
|
|
|
|
}
|
|
|
|
__EXCEPT(page_fault) {
|
|
|
|
XGLErrorFlag = 1;
|
|
|
|
}
|
|
|
|
__ENDTRY
|
|
|
|
|
|
|
|
XSetErrorHandler(WineXHandler);
|
|
|
|
if (XGLErrorFlag) {
|
|
|
|
TRACE("Error deleting context !\n");
|
|
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
|
|
ret = FALSE;
|
|
|
|
}
|
2000-05-23 01:20:08 +00:00
|
|
|
LEAVE_GL();
|
|
|
|
|
2000-05-23 21:15:06 +00:00
|
|
|
return ret;
|
2000-05-12 20:18:14 +00:00
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglDescribeLayerPlane
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglDescribeLayerPlane(HDC hdc,
|
|
|
|
int iPixelFormat,
|
|
|
|
int iLayerPlane,
|
|
|
|
UINT nBytes,
|
|
|
|
LPLAYERPLANEDESCRIPTOR plpd) {
|
|
|
|
FIXME("(%08x,%d,%d,%d,%p)\n", hdc, iPixelFormat, iLayerPlane, nBytes, plpd);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglGetCurrentContext
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
HGLRC WINAPI wglGetCurrentContext(void) {
|
|
|
|
GLXContext ret;
|
|
|
|
|
|
|
|
TRACE("()\n");
|
|
|
|
|
|
|
|
ENTER_GL();
|
|
|
|
ret = glXGetCurrentContext();
|
|
|
|
LEAVE_GL();
|
|
|
|
|
|
|
|
TRACE(" returning %p\n", ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglGetCurrentDC
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
HDC WINAPI wglGetCurrentDC(void) {
|
|
|
|
GLXContext ret;
|
|
|
|
|
|
|
|
ENTER_GL();
|
|
|
|
ret = glXGetCurrentContext();
|
|
|
|
LEAVE_GL();
|
|
|
|
|
|
|
|
if (ret == NULL) {
|
|
|
|
TRACE("() no current context -> returning NULL\n");
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
FIXME("()\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglGetLayerPaletteEntries
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
int WINAPI wglGetLayerPaletteEntries(HDC hdc,
|
|
|
|
int iLayerPlane,
|
|
|
|
int iStart,
|
|
|
|
int cEntries,
|
|
|
|
const COLORREF *pcr) {
|
|
|
|
FIXME("(): stub !\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int compar(const void *elt_a, const void *elt_b) {
|
|
|
|
return strcmp(((OpenGL_extension *) elt_a)->name,
|
|
|
|
((OpenGL_extension *) elt_b)->name);
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglGetProcAddress
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
void* WINAPI wglGetProcAddress(LPCSTR lpszProc) {
|
|
|
|
void *local_func;
|
|
|
|
static HMODULE hm = 0;
|
|
|
|
|
|
|
|
TRACE("(%s)\n", lpszProc);
|
|
|
|
|
|
|
|
if (hm == 0)
|
|
|
|
hm = GetModuleHandleA("opengl32");
|
|
|
|
|
|
|
|
/* First, look if it's not already defined in the 'standard' OpenGL functions */
|
|
|
|
if ((local_func = GetProcAddress(hm, lpszProc)) != NULL) {
|
|
|
|
TRACE("Found function in 'standard' OpenGL functions (%p)\n", local_func);
|
|
|
|
return local_func;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* After that, look at the extensions defined in the Linux OpenGL library */
|
|
|
|
if ((local_func = glXGetProcAddressARB(lpszProc)) == NULL) {
|
|
|
|
char buf[256];
|
|
|
|
void *ret = NULL;
|
|
|
|
|
|
|
|
/* Remove the 3 last letters (EXT, ARB, ...).
|
|
|
|
|
|
|
|
I know that some extensions have more than 3 letters (MESA, NV,
|
|
|
|
INTEL, ...), but this is only a stop-gap measure to fix buggy
|
|
|
|
OpenGL drivers (moreover, it is only useful for old 1.0 apps
|
|
|
|
that query the glBindTextureEXT extension).
|
|
|
|
*/
|
|
|
|
strncpy(buf, lpszProc, strlen(lpszProc) - 3);
|
|
|
|
buf[strlen(lpszProc) - 3] = '\0';
|
|
|
|
TRACE("Extension not found in the Linux OpenGL library, checking against libGL bug with %s..\n", buf);
|
|
|
|
|
|
|
|
ret = GetProcAddress(hm, buf);
|
|
|
|
if (ret != NULL) {
|
|
|
|
TRACE("Found function in main OpenGL library (%p) !\n", ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
OpenGL_extension ext;
|
|
|
|
OpenGL_extension *ret;
|
|
|
|
|
|
|
|
ext.name = (char *) lpszProc;
|
|
|
|
ret = (OpenGL_extension *) bsearch(&ext, extension_registry,
|
|
|
|
extension_registry_size, sizeof(OpenGL_extension), compar);
|
|
|
|
|
|
|
|
if (ret != NULL) {
|
|
|
|
TRACE("Returning function (%p)\n", ret->func);
|
|
|
|
*(ret->func_ptr) = local_func;
|
|
|
|
|
|
|
|
return ret->func;
|
|
|
|
} else {
|
|
|
|
ERR("Extension defined in the OpenGL library but NOT in opengl_ext.c... Please report (lionel.ulmer@free.fr) !\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglMakeCurrent
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglMakeCurrent(HDC hdc,
|
|
|
|
HGLRC hglrc) {
|
|
|
|
BOOL ret;
|
|
|
|
|
|
|
|
TRACE("(%08x,%p)\n", hdc, hglrc);
|
|
|
|
|
2000-05-23 21:15:06 +00:00
|
|
|
if (hglrc == NULL) {
|
|
|
|
ENTER_GL();
|
|
|
|
ret = glXMakeCurrent(display,
|
|
|
|
None,
|
|
|
|
NULL);
|
|
|
|
LEAVE_GL();
|
|
|
|
} else {
|
|
|
|
DC * dc = DC_GetDCPtr( hdc );
|
|
|
|
|
|
|
|
if (dc == NULL) {
|
|
|
|
ERR("Null DC !!!\n");
|
|
|
|
ret = FALSE;
|
|
|
|
} else {
|
|
|
|
X11DRV_PDEVICE *physDev;
|
|
|
|
|
|
|
|
physDev =(X11DRV_PDEVICE *)dc->physDev;
|
|
|
|
|
|
|
|
ENTER_GL();
|
|
|
|
ret = glXMakeCurrent(display,
|
|
|
|
physDev->drawable,
|
|
|
|
(GLXContext) hglrc);
|
|
|
|
LEAVE_GL();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TRACE("Returning %s\n", (ret ? "True" : "False"));
|
2000-05-12 20:18:14 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglRealizeLayerPalette
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglRealizeLayerPalette(HDC hdc,
|
|
|
|
int iLayerPlane,
|
|
|
|
BOOL bRealize) {
|
|
|
|
FIXME("()\n");
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglSetLayerPaletteEntries
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
int WINAPI wglSetLayerPaletteEntries(HDC hdc,
|
|
|
|
int iLayerPlane,
|
|
|
|
int iStart,
|
|
|
|
int cEntries,
|
|
|
|
const COLORREF *pcr) {
|
|
|
|
FIXME("(): stub !\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglShareLists
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglShareLists(HGLRC hglrc1,
|
|
|
|
HGLRC hglrc2) {
|
|
|
|
FIXME("(): stub !\n");
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* wglSwapLayerBuffers
|
|
|
|
*/
|
2000-05-12 20:18:14 +00:00
|
|
|
BOOL WINAPI wglSwapLayerBuffers(HDC hdc,
|
|
|
|
UINT fuPlanes) {
|
|
|
|
FIXME("(): stub !\n");
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
2000-05-23 23:38:32 +00:00
|
|
|
* wglUseFontBitmapsA
|
2000-05-18 00:07:53 +00:00
|
|
|
*/
|
2000-05-23 01:20:08 +00:00
|
|
|
BOOL WINAPI wglUseFontBitmapsA(HDC hdc,
|
|
|
|
DWORD first,
|
|
|
|
DWORD count,
|
|
|
|
DWORD listBase) {
|
|
|
|
DC * dc = DC_GetDCPtr( hdc );
|
|
|
|
X11DRV_PDEVICE *physDev = (X11DRV_PDEVICE *)dc->physDev;
|
|
|
|
fontObject* pfo = XFONT_GetFontObject( physDev->font );
|
|
|
|
Font fid = pfo->fs->fid;
|
2000-05-12 20:18:14 +00:00
|
|
|
|
2000-05-23 01:20:08 +00:00
|
|
|
TRACE("(%08x, %ld, %ld, %ld)\n", hdc, first, count, listBase);
|
|
|
|
|
|
|
|
ENTER_GL();
|
|
|
|
/* I assume that the glyphs are at the same position for X and for Windows */
|
|
|
|
glXUseXFont(fid, first, count, listBase);
|
|
|
|
LEAVE_GL();
|
|
|
|
|
|
|
|
return TRUE;
|
2000-05-12 20:18:14 +00:00
|
|
|
}
|
|
|
|
|
2000-05-18 00:07:53 +00:00
|
|
|
/***********************************************************************
|
2000-05-23 23:38:32 +00:00
|
|
|
* wglUseFontOutlinesA
|
2000-05-18 00:07:53 +00:00
|
|
|
*/
|
2000-05-23 01:20:08 +00:00
|
|
|
BOOL WINAPI wglUseFontOutlinesA(HDC hdc,
|
|
|
|
DWORD first,
|
|
|
|
DWORD count,
|
|
|
|
DWORD listBase,
|
|
|
|
FLOAT deviation,
|
|
|
|
FLOAT extrusion,
|
|
|
|
int format,
|
|
|
|
LPGLYPHMETRICSFLOAT lpgmf) {
|
2000-05-12 20:18:14 +00:00
|
|
|
FIXME("(): stub !\n");
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This is for brain-dead applications that use OpenGL functions before even
|
|
|
|
creating a rendering context.... */
|
|
|
|
DECL_GLOBAL_CONSTRUCTOR(OpenGL_create_default_context) {
|
|
|
|
int num;
|
|
|
|
XVisualInfo template;
|
|
|
|
XVisualInfo *vis;
|
|
|
|
GLXContext cx;
|
|
|
|
|
|
|
|
ENTER_GL();
|
|
|
|
template.visualid = XVisualIDFromVisual(visual);
|
|
|
|
vis = XGetVisualInfo(display, VisualIDMask, &template, &num);
|
|
|
|
cx=glXCreateContext(display, vis, 0, GL_TRUE);
|
|
|
|
glXMakeCurrent(display, X11DRV_GetXRootWindow(), cx);
|
|
|
|
LEAVE_GL();
|
|
|
|
}
|