xref: /reactos/dll/opengl/opengl32/icdload.c (revision cdf90707)
1 /*
2  * COPYRIGHT:            See COPYING in the top level directory
3  * PROJECT:              ReactOS kernel
4  * FILE:                 lib/opengl32/icdload.c
5  * PURPOSE:              OpenGL32 lib, ICD dll loader
6  */
7 
8 #include "opengl32.h"
9 
10 #include <winreg.h>
11 
12 WINE_DEFAULT_DEBUG_CHANNEL(opengl32);
13 
14 typedef struct
15 {
16     DWORD Version;          /*!< Driver interface version */
17     DWORD DriverVersion;    /*!< Driver version */
18     WCHAR DriverName[256];  /*!< Driver name */
19 } Drv_Opengl_Info, *pDrv_Opengl_Info;
20 
21 typedef enum
22 {
23     OGL_CD_NOT_QUERIED,
24     OGL_CD_NONE,
25     OGL_CD_ROSSWI,
26     OGL_CD_CUSTOM_ICD
27 } CUSTOM_DRIVER_STATE;
28 
29 static CRITICAL_SECTION icdload_cs = {NULL, -1, 0, 0, 0, 0};
30 static struct ICD_Data* ICD_Data_List = NULL;
31 static const WCHAR OpenGLDrivers_Key[] = L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\OpenGLDrivers";
32 static const WCHAR CustomDrivers_Key[] = L"SOFTWARE\\ReactOS\\OpenGL";
33 static Drv_Opengl_Info CustomDrvInfo;
34 static CUSTOM_DRIVER_STATE CustomDriverState = OGL_CD_NOT_QUERIED;
35 
36 static void APIENTRY wglSetCurrentValue(PVOID value)
37 {
38     IntSetCurrentICDPrivate(value);
39 }
40 
41 static PVOID APIENTRY wglGetCurrentValue()
42 {
43     return IntGetCurrentICDPrivate();
44 }
45 
46 static DHGLRC APIENTRY wglGetDHGLRC(struct wgl_context* context)
47 {
48     return context->dhglrc;
49 }
50 
51 /* GDI entry points (win32k) */
52 extern INT APIENTRY GdiDescribePixelFormat(HDC hdc, INT ipfd, UINT cjpfd, PPIXELFORMATDESCRIPTOR ppfd);
53 extern BOOL APIENTRY GdiSetPixelFormat(HDC hdc, INT ipfd);
54 extern BOOL APIENTRY GdiSwapBuffers(HDC hdc);
55 
56 /* Retrieves the ICD data (driver version + relevant DLL entry points) for a device context */
57 struct ICD_Data* IntGetIcdData(HDC hdc)
58 {
59     int ret;
60     DWORD dwInput, dwValueType, Version, DriverVersion, Flags;
61     Drv_Opengl_Info DrvInfo;
62     pDrv_Opengl_Info pDrvInfo;
63     struct ICD_Data* data;
64     HKEY OglKey = NULL;
65     HKEY DrvKey, CustomKey;
66     WCHAR DllName[MAX_PATH];
67     BOOL (WINAPI *DrvValidateVersion)(DWORD);
68     void (WINAPI *DrvSetCallbackProcs)(int nProcs, PROC* pProcs);
69 
70     /* The following code is ReactOS specific and allows us to easily load an arbitrary ICD:
71      * It checks HKCU\Software\ReactOS\OpenGL for a custom ICD and will always load it
72      * no matter what driver the DC is associated with. It can also force using the
73      * built-in Software Implementation*/
74     if(CustomDriverState == OGL_CD_NOT_QUERIED)
75     {
76         /* Only do this once so there's not any significant performance penalty */
77         CustomDriverState = OGL_CD_NONE;
78         memset(&CustomDrvInfo, 0, sizeof(Drv_Opengl_Info));
79 
80         ret = RegOpenKeyExW(HKEY_CURRENT_USER, CustomDrivers_Key, 0, KEY_READ, &CustomKey);
81         if(ret != ERROR_SUCCESS)
82             goto custom_end;
83 
84         dwInput = sizeof(CustomDrvInfo.DriverName);
85         ret = RegQueryValueExW(CustomKey, L"", 0, &dwValueType, (LPBYTE)CustomDrvInfo.DriverName, &dwInput);
86         RegCloseKey(CustomKey);
87 
88         if((ret != ERROR_SUCCESS) || (dwValueType != REG_SZ) || !wcslen(CustomDrvInfo.DriverName))
89             goto custom_end;
90 
91         if(!_wcsicmp(CustomDrvInfo.DriverName, L"ReactOS Software Implementation"))
92         {
93             /* Always announce the fact that we're forcing ROSSWI */
94             ERR("Forcing ReactOS Software Implementation\n");
95             CustomDriverState = OGL_CD_ROSSWI;
96             return NULL;
97         }
98 
99         ret = RegOpenKeyExW(HKEY_LOCAL_MACHINE, OpenGLDrivers_Key, 0, KEY_READ, &OglKey);
100         if(ret != ERROR_SUCCESS)
101             goto custom_end;
102 
103         ret = RegOpenKeyExW(OglKey, CustomDrvInfo.DriverName, 0, KEY_READ, &OglKey);
104         if(ret != ERROR_SUCCESS)
105             goto custom_end;
106 
107         dwInput = sizeof(CustomDrvInfo.Version);
108         ret = RegQueryValueExW(OglKey, L"Version", 0, &dwValueType, (LPBYTE)&CustomDrvInfo.Version, &dwInput);
109         if((ret != ERROR_SUCCESS) || (dwValueType != REG_DWORD))
110             goto custom_end;
111 
112         dwInput = sizeof(DriverVersion);
113         ret = RegQueryValueExW(OglKey, L"DriverVersion", 0, &dwValueType, (LPBYTE)&CustomDrvInfo.DriverVersion, &dwInput);
114         CustomDriverState = OGL_CD_CUSTOM_ICD;
115 
116         /* Always announce the fact that we're overriding the default driver */
117         ERR("Overriding the default OGL ICD with %S\n", CustomDrvInfo.DriverName);
118 
119 custom_end:
120         if(OglKey)
121             RegCloseKey(OglKey);
122         RegCloseKey(CustomKey);
123     }
124 
125     /* If there's a custom ICD or ROSSWI was requested use it, otherwise proceed as usual */
126     if(CustomDriverState == OGL_CD_CUSTOM_ICD)
127     {
128         pDrvInfo = &CustomDrvInfo;
129     }
130     else if(CustomDriverState == OGL_CD_ROSSWI)
131     {
132         return NULL;
133     }
134     else
135     {
136         /* First, see if the driver supports this */
137         dwInput = OPENGL_GETINFO;
138         ret = ExtEscape(hdc, QUERYESCSUPPORT, sizeof(DWORD), (LPCSTR)&dwInput, 0, NULL);
139 
140         /* Driver doesn't support opengl */
141         if(ret <= 0)
142             return NULL;
143 
144         /* Query for the ICD DLL name and version */
145         dwInput = 0;
146         ret = ExtEscape(hdc, OPENGL_GETINFO, sizeof(DWORD), (LPCSTR)&dwInput, sizeof(DrvInfo), (LPSTR)&DrvInfo);
147 
148         if(ret <= 0)
149         {
150             ERR("Driver claims to support OPENGL_GETINFO escape code, but doesn't.\n");
151             return NULL;
152         }
153 
154         pDrvInfo = &DrvInfo;
155     }
156 
157     /* Protect the list while we are loading*/
158     EnterCriticalSection(&icdload_cs);
159 
160     /* Search for it in the list of already loaded modules */
161     data = ICD_Data_List;
162     while(data)
163     {
164         if(!_wcsicmp(data->DriverName, pDrvInfo->DriverName))
165         {
166             /* Found it */
167             TRACE("Found already loaded %p.\n", data);
168             LeaveCriticalSection(&icdload_cs);
169             return data;
170         }
171         data = data->next;
172     }
173 
174     /* It was still not loaded, look for it in the registry */
175     ret = RegOpenKeyExW(HKEY_LOCAL_MACHINE, OpenGLDrivers_Key, 0, KEY_READ, &OglKey);
176     if(ret != ERROR_SUCCESS)
177     {
178         ERR("Failed to open the OpenGLDrivers key.\n");
179         goto end;
180     }
181     ret = RegOpenKeyExW(OglKey, pDrvInfo->DriverName, 0, KEY_READ, &DrvKey);
182     if(ret != ERROR_SUCCESS)
183     {
184         /* Some driver installer just provide the DLL name, like the Matrox G400 */
185         TRACE("No driver subkey for %S, trying to get DLL name directly.\n", pDrvInfo->DriverName);
186         dwInput = sizeof(DllName);
187         ret = RegQueryValueExW(OglKey, pDrvInfo->DriverName, 0, &dwValueType, (LPBYTE)DllName, &dwInput);
188         if((ret != ERROR_SUCCESS) || (dwValueType != REG_SZ))
189         {
190             ERR("Unable to get ICD DLL name!\n");
191             RegCloseKey(OglKey);
192             goto end;
193         }
194         Version = DriverVersion = Flags = 0;
195         TRACE("DLL name is %S.\n", DllName);
196     }
197     else
198     {
199         /* The driver have a subkey for the ICD */
200         TRACE("Querying details from registry for %S.\n", pDrvInfo->DriverName);
201         dwInput = sizeof(DllName);
202         ret = RegQueryValueExW(DrvKey, L"Dll", 0, &dwValueType, (LPBYTE)DllName, &dwInput);
203         if((ret != ERROR_SUCCESS) || (dwValueType != REG_SZ))
204         {
205             ERR("Unable to get ICD DLL name!.\n");
206             RegCloseKey(DrvKey);
207             RegCloseKey(OglKey);
208             goto end;
209         }
210 
211         dwInput = sizeof(Version);
212         ret = RegQueryValueExW(DrvKey, L"Version", 0, &dwValueType, (LPBYTE)&Version, &dwInput);
213         if((ret != ERROR_SUCCESS) || (dwValueType != REG_DWORD))
214         {
215             WARN("No version in driver subkey\n");
216         }
217         else if(Version != pDrvInfo->Version)
218         {
219             ERR("Version mismatch between registry (%lu) and display driver (%lu).\n", Version, pDrvInfo->Version);
220             RegCloseKey(DrvKey);
221             RegCloseKey(OglKey);
222             goto end;
223         }
224 
225         dwInput = sizeof(DriverVersion);
226         ret = RegQueryValueExW(DrvKey, L"DriverVersion", 0, &dwValueType, (LPBYTE)&DriverVersion, &dwInput);
227         if((ret != ERROR_SUCCESS) || (dwValueType != REG_DWORD))
228         {
229             WARN("No driver version in driver subkey\n");
230         }
231         else if(DriverVersion != pDrvInfo->DriverVersion)
232         {
233             ERR("Driver version mismatch between registry (%lu) and display driver (%lu).\n", DriverVersion, pDrvInfo->DriverVersion);
234             RegCloseKey(DrvKey);
235             RegCloseKey(OglKey);
236             goto end;
237         }
238 
239         dwInput = sizeof(Flags);
240         ret = RegQueryValueExW(DrvKey, L"Flags", 0, &dwValueType, (LPBYTE)&Flags, &dwInput);
241         if((ret != ERROR_SUCCESS) || (dwValueType != REG_DWORD))
242         {
243             WARN("No driver version in driver subkey\n");
244             Flags = 0;
245         }
246 
247         /* We're done */
248         RegCloseKey(DrvKey);
249         TRACE("DLL name is %S, Version %lx, DriverVersion %lx, Flags %lx.\n", DllName, Version, DriverVersion, Flags);
250     }
251     /* No need for this anymore */
252     RegCloseKey(OglKey);
253 
254     /* So far so good, allocate data */
255     data = HeapAlloc(GetProcessHeap(), 0, sizeof(*data));
256     if(!data)
257     {
258         ERR("Unable to allocate ICD data!\n");
259         goto end;
260     }
261 
262     /* Load the library */
263     data->hModule = LoadLibraryW(DllName);
264     if(!data->hModule)
265     {
266         ERR("Could not load the ICD DLL: %S.\n", DllName);
267         HeapFree(GetProcessHeap(), 0, data);
268         data = NULL;
269         goto end;
270     }
271 
272     /*
273      * Validate version, if needed.
274      * Some drivers (at least VBOX), initialize stuff upon this call.
275      */
276     DrvValidateVersion = (void*)GetProcAddress(data->hModule, "DrvValidateVersion");
277     if(DrvValidateVersion)
278     {
279         if(!DrvValidateVersion(pDrvInfo->DriverVersion))
280         {
281             ERR("DrvValidateVersion failed!.\n");
282             goto fail;
283         }
284     }
285 
286     /* Pass the callbacks */
287     DrvSetCallbackProcs = (void*)GetProcAddress(data->hModule, "DrvSetCallbackProcs");
288     if(DrvSetCallbackProcs)
289     {
290         PROC callbacks[] = {
291             (PROC)wglSetCurrentValue,
292             (PROC)wglGetCurrentValue,
293             (PROC)wglGetDHGLRC};
294         DrvSetCallbackProcs(ARRAYSIZE(callbacks), callbacks);
295     }
296 
297     /* Get the DLL exports */
298 #define DRV_LOAD(x) do                                  \
299 {                                                       \
300     data->x = (void*)GetProcAddress(data->hModule, #x); \
301     if(!data->x) {                                      \
302         ERR("%S lacks " #x "!\n", DllName);             \
303         goto fail;                                      \
304     }                                                   \
305 } while(0)
306     DRV_LOAD(DrvCopyContext);
307     DRV_LOAD(DrvCreateContext);
308     DRV_LOAD(DrvCreateLayerContext);
309     DRV_LOAD(DrvDeleteContext);
310     DRV_LOAD(DrvDescribeLayerPlane);
311     DRV_LOAD(DrvDescribePixelFormat);
312     DRV_LOAD(DrvGetLayerPaletteEntries);
313     DRV_LOAD(DrvGetProcAddress);
314     DRV_LOAD(DrvReleaseContext);
315     DRV_LOAD(DrvRealizeLayerPalette);
316     DRV_LOAD(DrvSetContext);
317     DRV_LOAD(DrvSetLayerPaletteEntries);
318     DRV_LOAD(DrvSetPixelFormat);
319     DRV_LOAD(DrvShareLists);
320     DRV_LOAD(DrvSwapBuffers);
321     DRV_LOAD(DrvSwapLayerBuffers);
322 #undef DRV_LOAD
323 
324     /* Let's see if GDI should handle this instead of the ICD DLL */
325     // FIXME: maybe there is a better way
326     if (GdiDescribePixelFormat(hdc, 0, 0, NULL) != 0)
327     {
328         /* GDI knows what to do with that. Override */
329         TRACE("Forwarding WGL calls to win32k!\n");
330         data->DrvDescribePixelFormat = GdiDescribePixelFormat;
331         data->DrvSetPixelFormat = GdiSetPixelFormat;
332         data->DrvSwapBuffers = GdiSwapBuffers;
333     }
334 
335     /* Copy the DriverName */
336     wcscpy(data->DriverName, pDrvInfo->DriverName);
337 
338     /* Push the list */
339     data->next = ICD_Data_List;
340     ICD_Data_List = data;
341 
342     TRACE("Returning %p.\n", data);
343     TRACE("ICD driver %S (%S) successfully loaded.\n", pDrvInfo->DriverName, DllName);
344 
345 end:
346     /* Unlock and return */
347     LeaveCriticalSection(&icdload_cs);
348     return data;
349 
350 fail:
351     LeaveCriticalSection(&icdload_cs);
352     FreeLibrary(data->hModule);
353     HeapFree(GetProcessHeap(), 0, data);
354     return NULL;
355 }
356 
357 void IntDeleteAllICDs(void)
358 {
359     struct ICD_Data* data;
360 
361     EnterCriticalSection(&icdload_cs);
362 
363     while (ICD_Data_List != NULL)
364     {
365         data = ICD_Data_List;
366         ICD_Data_List = data->next;
367 
368         FreeLibrary(data->hModule);
369         HeapFree(GetProcessHeap(), 0, data);
370     }
371 }
372