mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-30 04:26:45 +03:00 
			
		
		
		
	Bug#33031 app linked to libmysql.lib crash if run as service in vista under localsystem There are some problems using DllMain hook functions on Windows that automatically do global and per-thread initialization for libmysqld.dll 1)per-thread initialization(DLL_THREAD_ATTACH) MySQL internally counts number of active threads that and causes a delay in in my_end() if not all threads are exited. But,there are threads that can be started either by Windows internally (often in TCP/IP scenarios) or by user himself - those threads are not necessarily using libmysql.dll functionality, but nonetheless the contribute to the count of open threads. 2)process-initialization (DLL_PROCESS_ATTACH) my_init() calls WSAStartup that itself loads DLLs and can lead to a deadlock in Windows loader. Fix is to remove dll initialization code from libmysql.dll in general case. I still leave an environment variable LIBMYSQL_DLLINIT, which if set to any value will cause the old behavior (DLL init hooks will be called). This env.variable exists only to prevent breakage of existing Windows-only applications that don't do mysql_thread_init() and work ok today. Use of LIBMYSQL_DLLINIT is discouraged and it will be removed in 6.0
		
			
				
	
	
		
			126 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			126 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Copyright (C) 2000-2004 MySQL AB
 | |
| 
 | |
|    This program is free software; you can redistribute it and/or modify
 | |
|    it under the terms of the GNU General Public License as published by
 | |
|    the Free Software Foundation.
 | |
| 
 | |
|    There are special exceptions to the terms and conditions of the GPL as it
 | |
|    is applied to this software. View the full text of the exception in file
 | |
|    EXCEPTIONS-CLIENT in the directory of this software distribution.
 | |
| 
 | |
|    This program 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 General Public License for more details.
 | |
| 
 | |
|    You should have received a copy of the GNU General Public License
 | |
|    along with this program; if not, write to the Free Software
 | |
|    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
 | |
| 
 | |
| /*
 | |
| ** Handling initialization of the dll library
 | |
| */
 | |
| 
 | |
| #include <my_global.h>
 | |
| #include <my_sys.h>
 | |
| #include <my_pthread.h>
 | |
| 
 | |
| static my_bool libmysql_inited=0;
 | |
| 
 | |
| void libmysql_init(void)
 | |
| {
 | |
|   if (libmysql_inited)
 | |
|     return;
 | |
|   libmysql_inited=1;
 | |
|   my_init();
 | |
|   {
 | |
|     DBUG_ENTER("libmysql_init");
 | |
| #ifdef LOG_ALL
 | |
|     DBUG_PUSH("d:t:S:O,c::\\tmp\\libmysql.log");
 | |
| #else
 | |
|     if (getenv("LIBMYSQL_LOG") != NULL)
 | |
|       DBUG_PUSH(getenv("LIBMYSQL_LOG"));
 | |
| #endif
 | |
|     DBUG_VOID_RETURN;
 | |
|   }
 | |
| }
 | |
| 
 | |
| #ifdef __WIN__
 | |
| 
 | |
| static int inited=0,threads=0;
 | |
| HINSTANCE NEAR s_hModule;	/* Saved module handle */
 | |
| DWORD main_thread;
 | |
| 
 | |
| BOOL APIENTRY LibMain(HANDLE hInst,DWORD ul_reason_being_called,
 | |
| 		      LPVOID lpReserved)
 | |
| {
 | |
|   switch (ul_reason_being_called) {
 | |
|   case DLL_PROCESS_ATTACH:	/* case of libentry call in win 3.x */
 | |
|     if (!inited++)
 | |
|     {
 | |
|       s_hModule=hInst;
 | |
|       libmysql_init();
 | |
|       main_thread=GetCurrentThreadId();
 | |
|     }
 | |
|     break;
 | |
|   case DLL_THREAD_ATTACH:
 | |
|     threads++;
 | |
|     my_thread_init();
 | |
|     break;
 | |
|   case DLL_PROCESS_DETACH:	/* case of wep call in win 3.x */
 | |
|      if (!--inited)		/* Safety */
 | |
|      {
 | |
|        /* my_thread_init() */	/* This may give extra safety */
 | |
|        my_end(0);
 | |
|      }
 | |
|     break;
 | |
|   case DLL_THREAD_DETACH:
 | |
|     /* Main thread will free by my_end() */
 | |
|     threads--;
 | |
|     if (main_thread != GetCurrentThreadId())
 | |
|       my_thread_end();
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   } /* switch */
 | |
| 
 | |
|   return TRUE;
 | |
| 
 | |
|   UNREFERENCED_PARAMETER(lpReserved);
 | |
| } /* LibMain */
 | |
| 
 | |
| 
 | |
| static BOOL do_libmain;
 | |
| int __stdcall DllMain(HANDLE hInst,DWORD ul_reason_being_called,LPVOID lpReserved)
 | |
| {
 | |
|   /*
 | |
|     Unless environment variable LIBMYSQL_DLLINIT is set, do nothing.
 | |
|     The environment variable is checked once, during the first call to DllMain()
 | |
|     (in DLL_PROCESS_ATTACH hook).
 | |
|   */
 | |
|   if (ul_reason_being_called == DLL_PROCESS_ATTACH)
 | |
|     do_libmain = (getenv("LIBMYSQL_DLLINIT") != NULL);
 | |
|   if (do_libmain)
 | |
|     return LibMain(hInst,ul_reason_being_called,lpReserved);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| #elif defined(WINDOWS)
 | |
| 
 | |
| /****************************************************************************
 | |
| **	This routine is called by LIBSTART.ASM at module load time.  All it
 | |
| **	does in this sample is remember the DLL module handle.	The module
 | |
| **	handle is needed if you want to do things like load stuff from the
 | |
| **	resource file (for instance string resources).
 | |
| ****************************************************************************/
 | |
| 
 | |
| int _export FAR PASCAL libmain(HANDLE hModule,short cbHeapSize,
 | |
| 			       UCHAR FAR *lszCmdLine)
 | |
| {
 | |
|   s_hModule = hModule;
 | |
|   libmysql_init();
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| #endif
 |