mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-31 16:32:18 +01:00 
			
		
		
		
	Spotted by CI: https://gitlab.gnome.org/pwithnall/glib/-/jobs/4968919 Signed-off-by: Philip Withnall <pwithnall@gnome.org> Helps: #3405
		
			
				
	
	
		
			240 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			240 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GMODULE - GLIB wrapper code for dynamic module loading
 | |
|  * Copyright (C) 1998, 2000 Tim Janik
 | |
|  *
 | |
|  * Win32 GMODULE implementation
 | |
|  * Copyright (C) 1998 Tor Lillqvist
 | |
|  *
 | |
|  * SPDX-License-Identifier: LGPL-2.1-or-later
 | |
|  *
 | |
|  * This library is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU Lesser General Public
 | |
|  * License as published by the Free Software Foundation; either
 | |
|  * version 2.1 of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This library is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the GNU
 | |
|  * Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public
 | |
|  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Modified by the GLib Team and others 1997-2000.  See the AUTHORS
 | |
|  * file for a list of people on the GLib Team.  See the ChangeLog
 | |
|  * files for a list of changes.  These files are distributed with
 | |
|  * GLib at ftp://ftp.gtk.org/pub/gtk/. 
 | |
|  */
 | |
| 
 | |
| /* 
 | |
|  * MT safe
 | |
|  */
 | |
| #include "config.h"
 | |
| 
 | |
| #include <glib.h>
 | |
| #include <stdio.h>
 | |
| #include <windows.h>
 | |
| 
 | |
| #include <tlhelp32.h>
 | |
| 
 | |
| #ifdef G_WITH_CYGWIN
 | |
| #include <sys/cygwin.h>
 | |
| #endif
 | |
| 
 | |
| static void G_GNUC_PRINTF (2, 3)
 | |
| set_error (GError      **error,
 | |
|            const gchar  *format,
 | |
|            ...)
 | |
| {
 | |
|   gchar *win32_error;
 | |
|   gchar *detail;
 | |
|   gchar *message;
 | |
|   va_list args;
 | |
| 
 | |
|   win32_error = g_win32_error_message ((gint) GetLastError ());
 | |
| 
 | |
|   va_start (args, format);
 | |
|   detail = g_strdup_vprintf (format, args);
 | |
|   va_end (args);
 | |
| 
 | |
|   message = g_strconcat (detail, win32_error, NULL);
 | |
| 
 | |
|   g_module_set_error (message);
 | |
|   g_set_error_literal (error, G_MODULE_ERROR, G_MODULE_ERROR_FAILED, message);
 | |
| 
 | |
|   g_free (message);
 | |
|   g_free (detail);
 | |
|   g_free (win32_error);
 | |
| }
 | |
| 
 | |
| /* --- functions --- */
 | |
| static gpointer
 | |
| _g_module_open (const gchar *file_name,
 | |
| 		gboolean     bind_lazy,
 | |
| 		gboolean     bind_local,
 | |
|                 GError     **error)
 | |
| {
 | |
|   HINSTANCE handle;
 | |
|   wchar_t *wfilename;
 | |
|   DWORD old_mode;
 | |
|   BOOL success;
 | |
| #ifdef G_WITH_CYGWIN
 | |
|   gchar tmp[MAX_PATH];
 | |
| 
 | |
|   cygwin_conv_to_win32_path(file_name, tmp);
 | |
|   file_name = tmp;
 | |
| #endif
 | |
|   wfilename = g_utf8_to_utf16 (file_name, -1, NULL, NULL, NULL);
 | |
| 
 | |
|   /* suppress error dialog */
 | |
|   success = SetThreadErrorMode (SEM_NOOPENFILEERRORBOX | SEM_FAILCRITICALERRORS, &old_mode);
 | |
|   if (!success)
 | |
|     set_error (error, "");
 | |
| 
 | |
|   /* When building for UWP, load app asset DLLs instead of filesystem DLLs.
 | |
|    * Needs MSVC, Windows 8 and newer, and is only usable from apps. */
 | |
| #if _WIN32_WINNT >= 0x0602 && defined(G_WINAPI_ONLY_APP)
 | |
|   handle = LoadPackagedLibrary (wfilename, 0);
 | |
| #else
 | |
|   handle = LoadLibraryW (wfilename);
 | |
| #endif
 | |
| 
 | |
|   if (success)
 | |
|     SetThreadErrorMode (old_mode, NULL);
 | |
|   g_free (wfilename);
 | |
|       
 | |
|   if (!handle)
 | |
|     set_error (error, "'%s': ", file_name);
 | |
| 
 | |
|   return handle;
 | |
| }
 | |
| 
 | |
| static gint dummy;
 | |
| static gpointer null_module_handle = &dummy;
 | |
|   
 | |
| static gpointer
 | |
| _g_module_self (void)
 | |
| {
 | |
|   return null_module_handle;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _g_module_close (gpointer handle)
 | |
| {
 | |
|   if (handle != null_module_handle)
 | |
|     if (!FreeLibrary (handle))
 | |
|       set_error (NULL, "");
 | |
| }
 | |
| 
 | |
| static gpointer
 | |
| find_in_any_module_using_toolhelp (const gchar *symbol_name)
 | |
| {
 | |
|   HANDLE snapshot; 
 | |
|   MODULEENTRY32 me32;
 | |
| 
 | |
|   gpointer p = NULL;
 | |
| 
 | |
|   /* Under UWP, Module32Next and Module32First are not available since we're
 | |
|    * not allowed to search in the address space of arbitrary loaded DLLs */
 | |
| #if !defined(G_WINAPI_ONLY_APP)
 | |
|   /* https://docs.microsoft.com/en-us/windows/win32/api/tlhelp32/nf-tlhelp32-createtoolhelp32snapshot#remarks
 | |
|    * If the function fails with ERROR_BAD_LENGTH, retry the function until it succeeds. */
 | |
|   while (TRUE)
 | |
|     {
 | |
|       snapshot = CreateToolhelp32Snapshot (TH32CS_SNAPMODULE, 0);
 | |
|       if (snapshot == INVALID_HANDLE_VALUE && GetLastError () == ERROR_BAD_LENGTH)
 | |
|         {
 | |
|           g_thread_yield ();
 | |
|           continue;
 | |
|         }
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|   if (snapshot == INVALID_HANDLE_VALUE)
 | |
|     return NULL;
 | |
| 
 | |
|   me32.dwSize = sizeof (me32);
 | |
|   p = NULL;
 | |
|   if (Module32First (snapshot, &me32))
 | |
|     {
 | |
|       do {
 | |
| 	if ((p = GetProcAddress (me32.hModule, symbol_name)) != NULL)
 | |
| 	  break;
 | |
|       } while (Module32Next (snapshot, &me32));
 | |
|     }
 | |
| 
 | |
|   CloseHandle (snapshot);
 | |
| #endif
 | |
| 
 | |
|   return p;
 | |
| }
 | |
| 
 | |
| static gpointer
 | |
| find_in_any_module (const gchar *symbol_name)
 | |
| {
 | |
|   gpointer result;
 | |
| 
 | |
|   if ((result = find_in_any_module_using_toolhelp (symbol_name)) == NULL)
 | |
|     return NULL;
 | |
|   else
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static gpointer
 | |
| _g_module_symbol (gpointer     handle,
 | |
| 		  const gchar *symbol_name)
 | |
| {
 | |
|   gpointer p;
 | |
|   
 | |
|   if (handle == null_module_handle)
 | |
|     {
 | |
|       if ((p = GetProcAddress (GetModuleHandle (NULL), symbol_name)) == NULL)
 | |
| 	p = find_in_any_module (symbol_name);
 | |
|     }
 | |
|   else
 | |
|     p = GetProcAddress (handle, symbol_name);
 | |
| 
 | |
|   if (!p)
 | |
|     set_error (NULL, "");
 | |
| 
 | |
|   return p;
 | |
| }
 | |
| 
 | |
| static gchar*
 | |
| _g_module_build_path (const gchar *directory,
 | |
| 		      const gchar *module_name)
 | |
| {
 | |
|   size_t k;
 | |
| 
 | |
|   k = strlen (module_name);
 | |
|     
 | |
|   if (directory && *directory)
 | |
|     if (k > 4 && g_ascii_strcasecmp (module_name + k - 4, ".dll") == 0)
 | |
|       return g_strconcat (directory, G_DIR_SEPARATOR_S, module_name, NULL);
 | |
| #ifdef G_WITH_CYGWIN
 | |
|     else if (strncmp (module_name, "lib", 3) == 0 || strncmp (module_name, "cyg", 3) == 0)
 | |
|       return g_strconcat (directory, G_DIR_SEPARATOR_S, module_name, ".dll", NULL);
 | |
|     else
 | |
|       return g_strconcat (directory, G_DIR_SEPARATOR_S, "cyg", module_name, ".dll", NULL);
 | |
| #else
 | |
|     else if (strncmp (module_name, "lib", 3) == 0)
 | |
|       return g_strconcat (directory, G_DIR_SEPARATOR_S, module_name, ".dll", NULL);
 | |
|     else
 | |
|       return g_strconcat (directory, G_DIR_SEPARATOR_S, "lib", module_name, ".dll", NULL);
 | |
| #endif
 | |
|   else if (k > 4 && g_ascii_strcasecmp (module_name + k - 4, ".dll") == 0)
 | |
|     return g_strdup (module_name);
 | |
| #ifdef G_WITH_CYGWIN
 | |
|   else if (strncmp (module_name, "lib", 3) == 0 || strncmp (module_name, "cyg", 3) == 0)
 | |
|     return g_strconcat (module_name, ".dll", NULL);
 | |
|   else
 | |
|     return g_strconcat ("cyg", module_name, ".dll", NULL);
 | |
| #else
 | |
|   else if (strncmp (module_name, "lib", 3) == 0)
 | |
|     return g_strconcat (module_name, ".dll", NULL);
 | |
|   else
 | |
|     return g_strconcat ("lib", module_name, ".dll", NULL);
 | |
| #endif
 | |
| }
 |