2 * OpenGL function forwarding to the display driver
4 * Copyright (c) 1999 Lionel Ulmer
5 * Copyright (c) 2005 Raphael Junqueira
6 * Copyright (c) 2006 Roderick Colenbrander
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
24 #include "wine/port.h"
36 #include "gdi_private.h"
37 #include "wine/debug.h"
39 WINE_DEFAULT_DEBUG_CHANNEL(wgl
);
41 static HDC default_hdc
= 0;
43 typedef struct opengl_context
48 /* We route all wgl functions from opengl32.dll through gdi32.dll to
49 * the display driver. Various wgl calls have a hDC as one of their parameters.
50 * Using get_dc_ptr we get access to the functions exported by the driver.
51 * Some functions don't receive a hDC. This function creates a global hdc and
52 * if there's already a global hdc, it returns it.
54 static DC
* OPENGL_GetDefaultDC(void)
57 default_hdc
= CreateDCA("DISPLAY", NULL
, NULL
, NULL
);
59 return get_dc_ptr(default_hdc
);
62 /***********************************************************************
63 * wglCopyContext (OPENGL32.@)
65 BOOL WINAPI
wglCopyContext(HGLRC hglrcSrc
, HGLRC hglrcDst
, UINT mask
)
69 OPENGL_Context ctx
= (OPENGL_Context
)hglrcSrc
;
71 TRACE("hglrcSrc: (%p), hglrcDst: (%p), mask: %#x\n", hglrcSrc
, hglrcDst
, mask
);
72 /* If no context is set, this call doesn't have a purpose */
73 if(!hglrcSrc
|| !hglrcDst
)
76 /* Retrieve the HDC associated with the context to access the display driver */
77 dc
= get_dc_ptr(ctx
->hdc
);
78 if (!dc
) return FALSE
;
80 if (!dc
->funcs
->pwglCopyContext
) FIXME(" :stub\n");
81 else ret
= dc
->funcs
->pwglCopyContext(hglrcSrc
, hglrcDst
, mask
);
87 /***********************************************************************
88 * wglCreateContext (OPENGL32.@)
90 HGLRC WINAPI
wglCreateContext(HDC hdc
)
93 DC
* dc
= get_dc_ptr( hdc
);
100 if (!dc
->funcs
->pwglCreateContext
) FIXME(" :stub\n");
101 else ret
= dc
->funcs
->pwglCreateContext(dc
->physDev
);
103 release_dc_ptr( dc
);
107 /***********************************************************************
108 * wglCreateContextAttribsARB
110 static HGLRC WINAPI
wglCreateContextAttribsARB(HDC hdc
, HGLRC hShareContext
, const int *attributeList
)
113 DC
* dc
= get_dc_ptr( hdc
);
120 if (!dc
->funcs
->pwglCreateContextAttribsARB
) FIXME(" :stub\n");
121 else ret
= dc
->funcs
->pwglCreateContextAttribsARB(dc
->physDev
, hShareContext
, attributeList
);
123 release_dc_ptr( dc
);
127 /***********************************************************************
128 * wglDeleteContext (OPENGL32.@)
130 BOOL WINAPI
wglDeleteContext(HGLRC hglrc
)
134 OPENGL_Context ctx
= (OPENGL_Context
)hglrc
;
136 TRACE("hglrc: (%p)\n", hglrc
);
139 SetLastError(ERROR_INVALID_HANDLE
);
143 /* Retrieve the HDC associated with the context to access the display driver */
144 dc
= get_dc_ptr(ctx
->hdc
);
147 SetLastError(ERROR_INVALID_HANDLE
);
151 if (!dc
->funcs
->pwglDeleteContext
) FIXME(" :stub\n");
152 else ret
= dc
->funcs
->pwglDeleteContext(hglrc
);
154 release_dc_ptr( dc
);
158 /***********************************************************************
159 * wglGetCurrentContext (OPENGL32.@)
161 HGLRC WINAPI
wglGetCurrentContext(void)
163 HGLRC ret
= NtCurrentTeb()->glContext
;
164 TRACE(" returning %p\n", ret
);
168 /***********************************************************************
169 * wglGetCurrentDC (OPENGL32.@)
171 HDC WINAPI
wglGetCurrentDC(void)
173 OPENGL_Context ctx
= (OPENGL_Context
)wglGetCurrentContext();
175 TRACE(" found context: %p\n", ctx
);
179 /* Retrieve the current DC from the active context */
180 TRACE(" returning hdc: %p\n", ctx
->hdc
);
184 /***********************************************************************
187 static HDC WINAPI
wglGetPbufferDCARB(void *pbuffer
)
191 /* Create a device context to associate with the pbuffer */
192 HDC hdc
= CreateDCA("DISPLAY", NULL
, NULL
, NULL
);
193 DC
*dc
= get_dc_ptr(hdc
);
195 TRACE("(%p)\n", pbuffer
);
199 /* The display driver has to do the rest of the work because
200 * we need access to lowlevel datatypes which we can't access here
202 if (!dc
->funcs
->pwglGetPbufferDCARB
) FIXME(" :stub\n");
203 else ret
= dc
->funcs
->pwglGetPbufferDCARB(dc
->physDev
, pbuffer
);
205 TRACE("(%p), hdc=%p\n", pbuffer
, ret
);
207 release_dc_ptr( dc
);
211 /***********************************************************************
212 * wglMakeCurrent (OPENGL32.@)
214 BOOL WINAPI
wglMakeCurrent(HDC hdc
, HGLRC hglrc
)
219 /* When the context hglrc is NULL, the HDC is ignored and can be NULL.
220 * In that case use the global hDC to get access to the driver. */
223 if( hdc
== NULL
&& !wglGetCurrentContext() )
225 WARN( "Current context is NULL\n");
226 SetLastError( ERROR_INVALID_HANDLE
);
229 dc
= OPENGL_GetDefaultDC();
232 dc
= get_dc_ptr( hdc
);
234 TRACE("hdc: (%p), hglrc: (%p)\n", hdc
, hglrc
);
236 if (!dc
) return FALSE
;
239 if (!dc
->funcs
->pwglMakeCurrent
) FIXME(" :stub\n");
240 else ret
= dc
->funcs
->pwglMakeCurrent(dc
->physDev
,hglrc
);
242 release_dc_ptr( dc
);
246 /***********************************************************************
247 * wglMakeContextCurrentARB
249 static BOOL WINAPI
wglMakeContextCurrentARB(HDC hDrawDC
, HDC hReadDC
, HGLRC hglrc
)
255 TRACE("hDrawDC: (%p), hReadDC: (%p) hglrc: (%p)\n", hDrawDC
, hReadDC
, hglrc
);
257 /* Both hDrawDC and hReadDC need to be valid */
258 DrawDC
= get_dc_ptr( hDrawDC
);
259 if (!DrawDC
) return FALSE
;
261 ReadDC
= get_dc_ptr( hReadDC
);
263 release_dc_ptr( DrawDC
);
269 if (!DrawDC
->funcs
->pwglMakeContextCurrentARB
) FIXME(" :stub\n");
270 else ret
= DrawDC
->funcs
->pwglMakeContextCurrentARB(DrawDC
->physDev
, ReadDC
->physDev
, hglrc
);
272 release_dc_ptr( DrawDC
);
273 release_dc_ptr( ReadDC
);
277 /**************************************************************************************
278 * WINE-specific wglSetPixelFormat which can set the iPixelFormat multiple times
281 static BOOL WINAPI
wglSetPixelFormatWINE(HDC hdc
, int iPixelFormat
, const PIXELFORMATDESCRIPTOR
*ppfd
)
284 DC
* dc
= get_dc_ptr( hdc
);
286 TRACE("(%p,%d,%p)\n", hdc
, iPixelFormat
, ppfd
);
291 if (!dc
->funcs
->pwglSetPixelFormatWINE
) FIXME(" :stub\n");
292 else bRet
= dc
->funcs
->pwglSetPixelFormatWINE(dc
->physDev
, iPixelFormat
, ppfd
);
294 release_dc_ptr( dc
);
298 /***********************************************************************
299 * wglShareLists (OPENGL32.@)
301 BOOL WINAPI
wglShareLists(HGLRC hglrc1
, HGLRC hglrc2
)
305 OPENGL_Context ctx
= (OPENGL_Context
)hglrc1
;
307 TRACE("hglrc1: (%p); hglrc: (%p)\n", hglrc1
, hglrc2
);
308 if(ctx
== NULL
|| hglrc2
== NULL
)
311 /* Retrieve the HDC associated with the context to access the display driver */
312 dc
= get_dc_ptr(ctx
->hdc
);
313 if (!dc
) return FALSE
;
315 if (!dc
->funcs
->pwglShareLists
) FIXME(" :stub\n");
316 else ret
= dc
->funcs
->pwglShareLists(hglrc1
, hglrc2
);
318 release_dc_ptr( dc
);
322 /***********************************************************************
323 * wglUseFontBitmapsA (OPENGL32.@)
325 BOOL WINAPI
wglUseFontBitmapsA(HDC hdc
, DWORD first
, DWORD count
, DWORD listBase
)
328 DC
* dc
= get_dc_ptr( hdc
);
330 TRACE("(%p, %d, %d, %d)\n", hdc
, first
, count
, listBase
);
332 if (!dc
) return FALSE
;
334 if (!dc
->funcs
->pwglUseFontBitmapsA
) FIXME(" :stub\n");
335 else ret
= dc
->funcs
->pwglUseFontBitmapsA(dc
->physDev
, first
, count
, listBase
);
337 release_dc_ptr( dc
);
341 /***********************************************************************
342 * wglUseFontBitmapsW (OPENGL32.@)
344 BOOL WINAPI
wglUseFontBitmapsW(HDC hdc
, DWORD first
, DWORD count
, DWORD listBase
)
347 DC
* dc
= get_dc_ptr( hdc
);
349 TRACE("(%p, %d, %d, %d)\n", hdc
, first
, count
, listBase
);
351 if (!dc
) return FALSE
;
353 if (!dc
->funcs
->pwglUseFontBitmapsW
) FIXME(" :stub\n");
354 else ret
= dc
->funcs
->pwglUseFontBitmapsW(dc
->physDev
, first
, count
, listBase
);
356 release_dc_ptr( dc
);
360 /***********************************************************************
361 * Internal wglGetProcAddress for retrieving WGL extensions
363 PROC WINAPI
wglGetProcAddress(LPCSTR func
)
371 TRACE("func: '%s'\n", func
);
373 /* Retrieve the global hDC to get access to the driver. */
374 dc
= OPENGL_GetDefaultDC();
375 if (!dc
) return NULL
;
377 if (!dc
->funcs
->pwglGetProcAddress
) FIXME(" :stub\n");
378 else ret
= dc
->funcs
->pwglGetProcAddress(func
);
380 release_dc_ptr( dc
);
382 /* At the moment we implement one WGL extension which requires a HDC. When we
383 * are looking up this call and when the Extension is available (that is the case
384 * when a non-NULL value is returned by wglGetProcAddress), we return the address
385 * of a wrapper function which will handle the HDC->PhysDev conversion.
387 if(ret
&& strcmp(func
, "wglCreateContextAttribsARB") == 0)
388 return (PROC
)wglCreateContextAttribsARB
;
389 else if(ret
&& strcmp(func
, "wglMakeContextCurrentARB") == 0)
390 return (PROC
)wglMakeContextCurrentARB
;
391 else if(ret
&& strcmp(func
, "wglGetPbufferDCARB") == 0)
392 return (PROC
)wglGetPbufferDCARB
;
393 else if(ret
&& strcmp(func
, "wglSetPixelFormatWINE") == 0)
394 return (PROC
)wglSetPixelFormatWINE
;