mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-26 16:38:11 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			393 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			393 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* Copyright (C) 2020 MariaDB Corporation
 | |
| 
 | |
|    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; version 2 of the License.
 | |
| 
 | |
|    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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
 | |
| */
 | |
| 
 | |
| /*
 | |
|   main() function for the server on Windows is implemented here.
 | |
|   The core functionality is implemented elsewhere, in mysqld_main(), and running as
 | |
|   service is done here.
 | |
| 
 | |
|   Main tasks of the service are
 | |
| 
 | |
|   1. Report current status back to service control manager. Here we're
 | |
|   providing callbacks so code outside of winmain.cc can call it
 | |
|   (via mysqld_set_service_status_callback())
 | |
| 
 | |
|   2. React to notification, the only one we care about is the "stop"
 | |
|   notification. we initiate shutdown, when instructed.
 | |
| 
 | |
|   Note that our service might not be too Windows-friendly, as it might take
 | |
|   a while to startup (recovery), and a while to shut down(innodb cleanups).
 | |
| 
 | |
|   Most of the code more of less standard service stuff, taken from Microsoft
 | |
|   docs examples.
 | |
| 
 | |
|   Notable oddity in running services, is that we do not know for sure,
 | |
|   whether we should run as a service or not (there is no --service parameter that
 | |
|   would tell).Heuristics are used, and if the last command line argument is
 | |
|   valid service name, we try to run as service, but fallback to usual process
 | |
|   if this fails.
 | |
| 
 | |
|   As an example, even if mysqld.exe is started  with command line like "mysqld.exe --help",
 | |
|   it is entirely possible that mysqld.exe run as service "--help".
 | |
| 
 | |
|   Apart from that, now deprecated and obsolete service registration/removal functionality is
 | |
|   still provided (mysqld.exe --install/--remove)
 | |
| */
 | |
| 
 | |
| #include <my_global.h>
 | |
| #include <mysqld.h>
 | |
| #include <log.h>
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <windows.h>
 | |
| #include <string>
 | |
| #include <cassert>
 | |
| #include <winservice.h>
 | |
| 
 | |
| static SERVICE_STATUS svc_status{SERVICE_WIN32_OWN_PROCESS};
 | |
| static SERVICE_STATUS_HANDLE svc_status_handle;
 | |
| static char *svc_name;
 | |
| 
 | |
| static char **save_argv;
 | |
| static int save_argc;
 | |
| 
 | |
| static int install_service(int argc, char **argv, const char *name);
 | |
| static int remove_service(const char *name);
 | |
| 
 | |
| /*
 | |
|   Report service status to SCM. This function is indirectly invoked
 | |
|   by the server to report state transitions.
 | |
| 
 | |
|   1. from START_PENDING to SERVICE_RUNNING, when we start accepting user connections
 | |
|   2. from SERVICE_RUNNING to STOP_PENDING, when we start shutdown
 | |
|   3. from STOP_PENDING to SERVICE_STOPPED, in mysqld_exit()
 | |
|      sometimes also START_PENDING to SERVICE_STOPPED, on startup errors
 | |
| */
 | |
| static void report_svc_status(DWORD current_state, DWORD exit_code, DWORD wait_hint)
 | |
| {
 | |
|   if (!svc_status_handle)
 | |
|     return;
 | |
| 
 | |
|   static DWORD check_point= 1;
 | |
|   if (current_state != (DWORD)-1)
 | |
|     svc_status.dwCurrentState= current_state;
 | |
|   svc_status.dwWaitHint= wait_hint;
 | |
| 
 | |
|   if (exit_code)
 | |
|   {
 | |
|     svc_status.dwWin32ExitCode= ERROR_SERVICE_SPECIFIC_ERROR;
 | |
|     svc_status.dwServiceSpecificExitCode= exit_code;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     svc_status.dwWin32ExitCode= 0;
 | |
|   }
 | |
| 
 | |
|   if (current_state == SERVICE_START_PENDING)
 | |
|     svc_status.dwControlsAccepted= 0;
 | |
|   else
 | |
|     svc_status.dwControlsAccepted= SERVICE_ACCEPT_STOP|SERVICE_ACCEPT_SHUTDOWN;
 | |
| 
 | |
|   if ((current_state == SERVICE_RUNNING) || (current_state == SERVICE_STOPPED))
 | |
|     svc_status.dwCheckPoint= 0;
 | |
|   else
 | |
|     svc_status.dwCheckPoint= check_point++;
 | |
| 
 | |
|   SetServiceStatus(svc_status_handle, &svc_status);
 | |
| }
 | |
| 
 | |
| /* Report unexpected errors. */
 | |
| static void svc_report_event(const char *svc_name, const char *command)
 | |
| {
 | |
|   char buffer[80];
 | |
|   sprintf_s(buffer, "mariadb service %s, %s failed with %d",
 | |
|       svc_name, command, GetLastError());
 | |
|   OutputDebugString(buffer);
 | |
| }
 | |
| 
 | |
| /*
 | |
|   Service control function.
 | |
|   Reacts to service stop, initiates shutdown.
 | |
| */
 | |
| static void WINAPI svc_ctrl_handle(DWORD cntrl)
 | |
| {
 | |
|   switch (cntrl)
 | |
|   {
 | |
|   case SERVICE_CONTROL_SHUTDOWN:
 | |
|   case SERVICE_CONTROL_STOP:
 | |
|     sql_print_information(
 | |
|       "Windows service \"%s\":  received %s",
 | |
|       svc_name,
 | |
|       cntrl == SERVICE_CONTROL_STOP? "SERVICE_CONTROL_STOP": "SERVICE_CONTROL_SHUTDOWN");
 | |
| 
 | |
|     /* The below will also set the status to STOP_PENDING. */
 | |
|     mysqld_win_initiate_shutdown();
 | |
|     break;
 | |
| 
 | |
|   case SERVICE_CONTROL_INTERROGATE:
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* Service main routine, mainly runs mysqld_main() */
 | |
| static void WINAPI svc_main(DWORD svc_argc, char **svc_argv)
 | |
| {
 | |
|   /* Register the handler function for the service */
 | |
|   char *name= svc_argv[0];
 | |
| 
 | |
|   svc_status_handle= RegisterServiceCtrlHandler(name, svc_ctrl_handle);
 | |
|   if (!svc_status_handle)
 | |
|   {
 | |
|     svc_report_event(name, "RegisterServiceCtrlHandler");
 | |
|     return;
 | |
|   }
 | |
|   report_svc_status(SERVICE_START_PENDING, NO_ERROR, 0);
 | |
| 
 | |
|   /* Make server report service status via our callback.*/
 | |
|   mysqld_set_service_status_callback(report_svc_status);
 | |
| 
 | |
|   /* This would add service name entry to load_defaults.*/
 | |
|   mysqld_win_set_service_name(name);
 | |
| 
 | |
|   /*
 | |
|    Do not pass the service name parameter (last on the command line)
 | |
|    to mysqld_main(), it is unaware of it.
 | |
|   */
 | |
|   save_argv[save_argc - 1]= 0;
 | |
|   mysqld_main(save_argc - 1, save_argv);
 | |
| }
 | |
| 
 | |
| /*
 | |
|   This start the service. Sometimes it will fail, because
 | |
|   currently we do not know for sure whether we run as service or not.
 | |
|   If this fails, the fallback is to run as normal process.
 | |
| */
 | |
| static int run_as_service(char *name)
 | |
| {
 | |
|   SERVICE_TABLE_ENTRY stb[]= {{name, svc_main}, {0, 0}};
 | |
|   if (!StartServiceCtrlDispatcher(stb))
 | |
|   {
 | |
|     assert(GetLastError() == ERROR_FAILED_SERVICE_CONTROLLER_CONNECT);
 | |
|     return -1;
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|   Check for valid existing service name.
 | |
|   Part of our guesswork, whether we run as service or not.
 | |
| */
 | |
| static bool is_existing_service(const char *name)
 | |
| {
 | |
|   if (strchr(name, '\\') || strchr(name, '/'))
 | |
|   {
 | |
|     /* Invalid characters in service name */
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   SC_HANDLE sc_service= 0, scm= 0;
 | |
|   bool ret= ((scm= OpenSCManager(0, 0, SC_MANAGER_ENUMERATE_SERVICE)) != 0) &&
 | |
|        ((sc_service= OpenService(scm, name, SERVICE_QUERY_STATUS)) != 0);
 | |
| 
 | |
|   if (sc_service)
 | |
|     CloseServiceHandle(sc_service);
 | |
|   if (scm)
 | |
|     CloseServiceHandle(scm);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /*
 | |
|   If service name is not given to --install/--remove
 | |
|   it is assumed to be "MySQL" (traditional handling)
 | |
| */
 | |
| static const char *get_svc_name(const char *arg)
 | |
| {
 | |
|   return arg ? arg : "MySQL";
 | |
| }
 | |
| 
 | |
| /*
 | |
|   Disable CPU throttling for the process.
 | |
| 
 | |
|   Windows 11 heuristics misdetects server as a background process and runs it
 | |
|   on "efficiency" cores, in hybrid  architectures such as Alder Lake (12th
 | |
|   generation Intel Core).This results in serious performance degradation.
 | |
| */
 | |
| void disable_cpu_throttling()
 | |
| {
 | |
| #ifdef PROCESS_POWER_THROTTLING_EXECUTION_SPEED
 | |
|   PROCESS_POWER_THROTTLING_STATE power_throttling{};
 | |
|   power_throttling.Version= PROCESS_POWER_THROTTLING_CURRENT_VERSION;
 | |
|   power_throttling.ControlMask= PROCESS_POWER_THROTTLING_EXECUTION_SPEED;
 | |
|   power_throttling.StateMask= 0;
 | |
|   SetProcessInformation(GetCurrentProcess(), ProcessPowerThrottling,
 | |
|                         &power_throttling, sizeof(power_throttling));
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /*
 | |
|   Main function on Windows.
 | |
|   Runs mysqld as normal process, or as a service.
 | |
| 
 | |
|   Plus, the obsolete functionality to register/remove services.
 | |
| */
 | |
| __declspec(dllexport) int mysqld_win_main(int argc, char **argv)
 | |
| {
 | |
|   save_argv= argv;
 | |
|   save_argc= argc;
 | |
| 
 | |
|   disable_cpu_throttling();
 | |
|    /*
 | |
|      If no special arguments are given, service name is nor present
 | |
|      run as normal program.
 | |
|    */
 | |
|   if (argc == 1)
 | |
|     return mysqld_main(argc, argv);
 | |
| 
 | |
|   auto cmd= argv[1];
 | |
| 
 | |
|   /* Handle install/remove */
 | |
|   if (!strcmp(cmd, "--install") || !strcmp(cmd, "--install-manual"))
 | |
|     return install_service(argc, argv, get_svc_name(argv[2]));
 | |
| 
 | |
|   if (!strcmp(cmd, "--remove"))
 | |
|     return remove_service(get_svc_name(argv[2]));
 | |
| 
 | |
|   /* Try to run as service, and fallback to mysqld_main(), if this fails */
 | |
|   svc_name= argv[argc - 1];
 | |
|   if (is_existing_service(svc_name) && !run_as_service(svc_name))
 | |
|     return 0;
 | |
|   svc_name= 0;
 | |
| 
 | |
|   /* Run as normal program.*/
 | |
|   return mysqld_main(argc, argv);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|   Register/remove services functionality.
 | |
|   This is kept for backward compatibility only, and is
 | |
|   superseeded by much more versatile mysql_install_db.exe
 | |
| 
 | |
|   "mysqld --remove=svc" has no advantage over
 | |
|   OS own "sc delete svc"
 | |
| */
 | |
| static void ATTRIBUTE_NORETURN die(const char *func, const char *name)
 | |
| {
 | |
|   DWORD err= GetLastError();
 | |
|   fprintf(stderr, "FATAL ERROR : %s failed (%lu)\n", func, err);
 | |
|   switch (err)
 | |
|   {
 | |
|   case ERROR_SERVICE_EXISTS:
 | |
|     fprintf(stderr, "Service %s already exists.\n", name);
 | |
|     break;
 | |
|   case ERROR_SERVICE_DOES_NOT_EXIST:
 | |
|     fprintf(stderr, "Service %s does not exist.\n", name);
 | |
|     break;
 | |
|   case ERROR_ACCESS_DENIED:
 | |
|     fprintf(stderr, "Access is denied. "
 | |
|         "Make sure to run as elevated admin user.\n");
 | |
|     break;
 | |
|   case ERROR_INVALID_NAME:
 | |
|     fprintf(stderr, "Invalid service name '%s'\n", name);
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
|   exit(1);
 | |
| }
 | |
| 
 | |
| static inline std::string quoted(const char *src)
 | |
| {
 | |
|   std::string s;
 | |
|   s.append("\"").append(src).append("\"");
 | |
|   return s;
 | |
| }
 | |
| 
 | |
| static int install_service(int argc, char **argv, const char *name)
 | |
| {
 | |
|   std::string cmdline;
 | |
| 
 | |
|   char path[MAX_PATH];
 | |
|   auto nSize = GetModuleFileName(0, path, sizeof(path));
 | |
| 
 | |
|   if (nSize == (DWORD) sizeof(path) && GetLastError() == ERROR_INSUFFICIENT_BUFFER)
 | |
|     die("GetModuleName", name);
 | |
| 
 | |
|   cmdline.append(quoted(path));
 | |
| 
 | |
|   const char *user= 0;
 | |
|   // mysqld --install[-manual] name ...[--local-service]
 | |
|   if (argc > 2)
 | |
|   {
 | |
|     for (int i= 3; argv[i]; i++)
 | |
|     {
 | |
|       if (!strcmp(argv[i], "--local-service"))
 | |
|         user= "NT AUTHORITY\\LocalService";
 | |
|       else
 | |
|       {
 | |
|         cmdline.append(" ").append(quoted(argv[i]));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   cmdline.append(" ").append(quoted(name));
 | |
| 
 | |
|   DWORD start_type;
 | |
|   if (!strcmp(argv[1], "--install-manual"))
 | |
|     start_type= SERVICE_DEMAND_START;
 | |
|   else
 | |
|     start_type= SERVICE_AUTO_START;
 | |
| 
 | |
|   SC_HANDLE scm, sc_service;
 | |
|   if (!(scm= OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE)))
 | |
|     die("OpenSCManager", name);
 | |
| 
 | |
|   if (!(sc_service= CreateService(
 | |
|       scm, name, name, SERVICE_ALL_ACCESS,
 | |
|       SERVICE_WIN32_OWN_PROCESS, start_type, SERVICE_ERROR_NORMAL,
 | |
|       cmdline.c_str(), 0, 0, 0, user, 0)))
 | |
|     die("CreateService", name);
 | |
| 
 | |
|   char description[]= "MariaDB database server";
 | |
|   SERVICE_DESCRIPTION sd= {description};
 | |
|   ChangeServiceConfig2(sc_service, SERVICE_CONFIG_DESCRIPTION, &sd);
 | |
| 
 | |
|   CloseServiceHandle(sc_service);
 | |
|   CloseServiceHandle(scm);
 | |
| 
 | |
|   printf("Service '%s' successfully installed.\n", name);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static int remove_service(const char *name)
 | |
| {
 | |
|   SC_HANDLE scm, sc_service;
 | |
| 
 | |
|   if (!(scm= OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE)))
 | |
|     die("OpenSCManager", name);
 | |
| 
 | |
|   if (!(sc_service= OpenService(scm, name, DELETE)))
 | |
|     die("OpenService", name);
 | |
| 
 | |
|   if (!DeleteService(sc_service))
 | |
|     die("DeleteService", name);
 | |
| 
 | |
|   CloseServiceHandle(sc_service);
 | |
|   CloseServiceHandle(scm);
 | |
| 
 | |
|   printf("Service '%s' successfully deleted.\n", name);
 | |
|   return 0;
 | |
| }
 | 
