/* * COPYRIGHT: See COPYING in the top level directory * PROJECT: ReactOS system libraries * FILE: lib/advapi32/service/scm.c * PURPOSE: Service control manager functions * PROGRAMMER: Emanuele Aliberti * Eric Kohl * UPDATE HISTORY: * 19990413 EA created * 19990515 EA */ /* INCLUDES ******************************************************************/ #include #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(advapi); /* FUNCTIONS *****************************************************************/ handle_t __RPC_USER SVCCTL_HANDLEA_bind(SVCCTL_HANDLEA szMachineName) { handle_t hBinding = NULL; UCHAR *pszStringBinding; RPC_STATUS status; TRACE("SVCCTL_HANDLEA_bind() called\n"); status = RpcStringBindingComposeA(NULL, (UCHAR *)"ncacn_np", (UCHAR *)szMachineName, (UCHAR *)"\\pipe\\ntsvcs", NULL, (UCHAR **)&pszStringBinding); if (status != RPC_S_OK) { ERR("RpcStringBindingCompose returned 0x%x\n", status); return NULL; } /* Set the binding handle that will be used to bind to the server. */ status = RpcBindingFromStringBindingA(pszStringBinding, &hBinding); if (status != RPC_S_OK) { ERR("RpcBindingFromStringBinding returned 0x%x\n", status); } status = RpcStringFreeA(&pszStringBinding); if (status != RPC_S_OK) { ERR("RpcStringFree returned 0x%x\n", status); } return hBinding; } void __RPC_USER SVCCTL_HANDLEA_unbind(SVCCTL_HANDLEA szMachineName, handle_t hBinding) { RPC_STATUS status; TRACE("SVCCTL_HANDLEA_unbind() called\n"); status = RpcBindingFree(&hBinding); if (status != RPC_S_OK) { ERR("RpcBindingFree returned 0x%x\n", status); } } handle_t __RPC_USER SVCCTL_HANDLEW_bind(SVCCTL_HANDLEW szMachineName) { handle_t hBinding = NULL; LPWSTR pszStringBinding; RPC_STATUS status; TRACE("SVCCTL_HANDLEW_bind() called\n"); status = RpcStringBindingComposeW(NULL, L"ncacn_np", szMachineName, L"\\pipe\\ntsvcs", NULL, &pszStringBinding); if (status != RPC_S_OK) { ERR("RpcStringBindingCompose returned 0x%x\n", status); return NULL; } /* Set the binding handle that will be used to bind to the server. */ status = RpcBindingFromStringBindingW(pszStringBinding, &hBinding); if (status != RPC_S_OK) { ERR("RpcBindingFromStringBinding returned 0x%x\n", status); } status = RpcStringFreeW(&pszStringBinding); if (status != RPC_S_OK) { ERR("RpcStringFree returned 0x%x\n", status); } return hBinding; } void __RPC_USER SVCCTL_HANDLEW_unbind(SVCCTL_HANDLEW szMachineName, handle_t hBinding) { RPC_STATUS status; TRACE("SVCCTL_HANDLEW_unbind() called\n"); status = RpcBindingFree(&hBinding); if (status != RPC_S_OK) { ERR("RpcBindingFree returned 0x%x\n", status); } } DWORD ScmRpcStatusToWinError(RPC_STATUS Status) { switch (Status) { case RPC_S_INVALID_BINDING: case RPC_X_SS_IN_NULL_CONTEXT: return ERROR_INVALID_HANDLE; case RPC_X_ENUM_VALUE_OUT_OF_RANGE: case RPC_X_BYTE_COUNT_TOO_SMALL: return ERROR_INVALID_PARAMETER; case RPC_X_NULL_REF_POINTER: return ERROR_INVALID_ADDRESS; default: return (DWORD)Status; } } /********************************************************************** * ChangeServiceConfig2A * * @implemented */ BOOL WINAPI ChangeServiceConfig2A(SC_HANDLE hService, DWORD dwInfoLevel, LPVOID lpInfo) { SC_RPC_CONFIG_INFOA Info; DWORD dwError; TRACE("ChangeServiceConfig2A() called\n"); /* Fill relevent field of the Info structure */ Info.dwInfoLevel = dwInfoLevel; switch (dwInfoLevel) { case SERVICE_CONFIG_DESCRIPTION: Info.psd = (LPSERVICE_DESCRIPTIONA)&lpInfo; Info.lpDescription = ((LPSERVICE_DESCRIPTIONA)lpInfo)->lpDescription; //HACK break; case SERVICE_CONFIG_FAILURE_ACTIONS: Info.psfa = (LPSERVICE_FAILURE_ACTIONSA)lpInfo; break; default: WARN("Unknown info level 0x%lx\n", dwInfoLevel); SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (lpInfo == NULL) return TRUE; RpcTryExcept { dwError = RChangeServiceConfig2A((SC_RPC_HANDLE)hService, Info); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RChangeServiceConfig2A() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * ChangeServiceConfig2W * * @implemented */ BOOL WINAPI ChangeServiceConfig2W(SC_HANDLE hService, DWORD dwInfoLevel, LPVOID lpInfo) { SC_RPC_CONFIG_INFOW Info; DWORD dwError; TRACE("ChangeServiceConfig2W() called\n"); /* Fill relevent field of the Info structure */ Info.dwInfoLevel = dwInfoLevel; switch (dwInfoLevel) { case SERVICE_CONFIG_DESCRIPTION: Info.psd = (LPSERVICE_DESCRIPTIONW)&lpInfo; break; case SERVICE_CONFIG_FAILURE_ACTIONS: Info.psfa = (LPSERVICE_FAILURE_ACTIONSW)&lpInfo; break; default: WARN("Unknown info level 0x%lx\n", dwInfoLevel); SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (lpInfo == NULL) return TRUE; RpcTryExcept { dwError = RChangeServiceConfig2W((SC_RPC_HANDLE)hService, Info); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RChangeServiceConfig2W() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * ChangeServiceConfigA * * @implemented */ BOOL WINAPI ChangeServiceConfigA(SC_HANDLE hService, DWORD dwServiceType, DWORD dwStartType, DWORD dwErrorControl, LPCSTR lpBinaryPathName, LPCSTR lpLoadOrderGroup, LPDWORD lpdwTagId, LPCSTR lpDependencies, LPCSTR lpServiceStartName, LPCSTR lpPassword, LPCSTR lpDisplayName) { DWORD dwError; DWORD dwDependenciesLength = 0; DWORD dwLength; LPSTR lpStr; TRACE("ChangeServiceConfigA() called\n"); /* Calculate the Dependencies length*/ if (lpDependencies != NULL) { lpStr = (LPSTR)lpDependencies; while (*lpStr) { dwLength = strlen(lpStr) + 1; dwDependenciesLength += dwLength; lpStr = lpStr + dwLength; } dwDependenciesLength++; } /* FIXME: Encrypt the password */ RpcTryExcept { /* Call to services.exe using RPC */ dwError = RChangeServiceConfigA((SC_RPC_HANDLE)hService, dwServiceType, dwStartType, dwErrorControl, (LPSTR)lpBinaryPathName, (LPSTR)lpLoadOrderGroup, lpdwTagId, (LPSTR)lpDependencies, dwDependenciesLength, (LPSTR)lpServiceStartName, NULL, /* FIXME: lpPassword */ 0, /* FIXME: dwPasswordLength */ (LPSTR)lpDisplayName); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RChangeServiceConfigA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * ChangeServiceConfigW * * @implemented */ BOOL WINAPI ChangeServiceConfigW(SC_HANDLE hService, DWORD dwServiceType, DWORD dwStartType, DWORD dwErrorControl, LPCWSTR lpBinaryPathName, LPCWSTR lpLoadOrderGroup, LPDWORD lpdwTagId, LPCWSTR lpDependencies, LPCWSTR lpServiceStartName, LPCWSTR lpPassword, LPCWSTR lpDisplayName) { DWORD dwError; DWORD dwDependenciesLength = 0; DWORD dwLength; LPWSTR lpStr; TRACE("ChangeServiceConfigW() called\n"); /* Calculate the Dependencies length*/ if (lpDependencies != NULL) { lpStr = (LPWSTR)lpDependencies; while (*lpStr) { dwLength = wcslen(lpStr) + 1; dwDependenciesLength += dwLength; lpStr = lpStr + dwLength; } dwDependenciesLength++; } /* FIXME: Encrypt the password */ RpcTryExcept { /* Call to services.exe using RPC */ dwError = RChangeServiceConfigW((SC_RPC_HANDLE)hService, dwServiceType, dwStartType, dwErrorControl, (LPWSTR)lpBinaryPathName, (LPWSTR)lpLoadOrderGroup, lpdwTagId, (LPBYTE)lpDependencies, dwDependenciesLength, (LPWSTR)lpServiceStartName, NULL, /* FIXME: lpPassword */ 0, /* FIXME: dwPasswordLength */ (LPWSTR)lpDisplayName); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RChangeServiceConfigW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * CloseServiceHandle * * @implemented */ BOOL WINAPI CloseServiceHandle(SC_HANDLE hSCObject) { DWORD dwError; TRACE("CloseServiceHandle() called\n"); if (!hSCObject) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = RCloseServiceHandle((LPSC_RPC_HANDLE)&hSCObject); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError) { TRACE("RCloseServiceHandle() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("CloseServiceHandle() done\n"); return TRUE; } /********************************************************************** * ControlService * * @implemented */ BOOL WINAPI ControlService(SC_HANDLE hService, DWORD dwControl, LPSERVICE_STATUS lpServiceStatus) { DWORD dwError; TRACE("ControlService(%x, %x, %p)\n", hService, dwControl, lpServiceStatus); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RControlService((SC_RPC_HANDLE)hService, dwControl, lpServiceStatus); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RControlService() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("ControlService() done\n"); return TRUE; } /********************************************************************** * ControlServiceEx * * @unimplemented */ BOOL WINAPI ControlServiceEx(IN SC_HANDLE hService, IN DWORD dwControl, IN DWORD dwInfoLevel, IN OUT PVOID pControlParams) { FIXME("ControlServiceEx(0x%p, 0x%x, 0x%x, 0x%p) UNIMPLEMENTED!\n", hService, dwControl, dwInfoLevel, pControlParams); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /********************************************************************** * CreateServiceA * * @implemented */ SC_HANDLE WINAPI CreateServiceA(SC_HANDLE hSCManager, LPCSTR lpServiceName, LPCSTR lpDisplayName, DWORD dwDesiredAccess, DWORD dwServiceType, DWORD dwStartType, DWORD dwErrorControl, LPCSTR lpBinaryPathName, LPCSTR lpLoadOrderGroup, LPDWORD lpdwTagId, LPCSTR lpDependencies, LPCSTR lpServiceStartName, LPCSTR lpPassword) { SC_HANDLE hService = NULL; DWORD dwDependenciesLength = 0; DWORD dwError; DWORD dwLength; LPSTR lpStr; TRACE("CreateServiceA() called\n"); TRACE("%p %s %s\n", hSCManager, lpServiceName, lpDisplayName); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return NULL; } /* Calculate the Dependencies length*/ if (lpDependencies != NULL) { lpStr = (LPSTR)lpDependencies; while (*lpStr) { dwLength = strlen(lpStr) + 1; dwDependenciesLength += dwLength; lpStr = lpStr + dwLength; } dwDependenciesLength++; } /* FIXME: Encrypt the password */ RpcTryExcept { /* Call to services.exe using RPC */ dwError = RCreateServiceA((SC_RPC_HANDLE)hSCManager, (LPSTR)lpServiceName, (LPSTR)lpDisplayName, dwDesiredAccess, dwServiceType, dwStartType, dwErrorControl, (LPSTR)lpBinaryPathName, (LPSTR)lpLoadOrderGroup, lpdwTagId, (LPBYTE)lpDependencies, dwDependenciesLength, (LPSTR)lpServiceStartName, NULL, /* FIXME: lpPassword */ 0, /* FIXME: dwPasswordLength */ (SC_RPC_HANDLE *)&hService); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RCreateServiceA() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } return hService; } /********************************************************************** * CreateServiceW * * @implemented */ SC_HANDLE WINAPI CreateServiceW(SC_HANDLE hSCManager, LPCWSTR lpServiceName, LPCWSTR lpDisplayName, DWORD dwDesiredAccess, DWORD dwServiceType, DWORD dwStartType, DWORD dwErrorControl, LPCWSTR lpBinaryPathName, LPCWSTR lpLoadOrderGroup, LPDWORD lpdwTagId, LPCWSTR lpDependencies, LPCWSTR lpServiceStartName, LPCWSTR lpPassword) { SC_HANDLE hService = NULL; DWORD dwDependenciesLength = 0; DWORD dwError; DWORD dwLength; LPWSTR lpStr; TRACE("CreateServiceW() called\n"); TRACE("%p %S %S\n", hSCManager, lpServiceName, lpDisplayName); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return NULL; } /* Calculate the Dependencies length*/ if (lpDependencies != NULL) { lpStr = (LPWSTR)lpDependencies; while (*lpStr) { dwLength = wcslen(lpStr) + 1; dwDependenciesLength += dwLength; lpStr = lpStr + dwLength; } dwDependenciesLength++; dwDependenciesLength *= sizeof(WCHAR); } /* FIXME: Encrypt the password */ RpcTryExcept { /* Call to services.exe using RPC */ dwError = RCreateServiceW((SC_RPC_HANDLE)hSCManager, lpServiceName, lpDisplayName, dwDesiredAccess, dwServiceType, dwStartType, dwErrorControl, lpBinaryPathName, lpLoadOrderGroup, lpdwTagId, (LPBYTE)lpDependencies, dwDependenciesLength, lpServiceStartName, NULL, /* FIXME: lpPassword */ 0, /* FIXME: dwPasswordLength */ (SC_RPC_HANDLE *)&hService); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RCreateServiceW() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } return hService; } /********************************************************************** * DeleteService * * @implemented */ BOOL WINAPI DeleteService(SC_HANDLE hService) { DWORD dwError; TRACE("DeleteService(%x)\n", hService); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RDeleteService((SC_RPC_HANDLE)hService); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RDeleteService() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * EnumDependentServicesA * * @implemented */ BOOL WINAPI EnumDependentServicesA(SC_HANDLE hService, DWORD dwServiceState, LPENUM_SERVICE_STATUSA lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned) { LPENUM_SERVICE_STATUSA lpStatusPtr; DWORD dwError; DWORD dwCount; TRACE("EnumServicesStatusA() called\n"); RpcTryExcept { dwError = REnumDependentServicesA((SC_RPC_HANDLE)hService, dwServiceState, (LPBYTE)lpServices, cbBufSize, pcbBytesNeeded, lpServicesReturned); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError == ERROR_SUCCESS || dwError == ERROR_MORE_DATA) { lpStatusPtr = (LPENUM_SERVICE_STATUSA)lpServices; for (dwCount = 0; dwCount < *lpServicesReturned; dwCount++) { if (lpStatusPtr->lpServiceName) lpStatusPtr->lpServiceName = (LPSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpServiceName); if (lpStatusPtr->lpDisplayName) lpStatusPtr->lpDisplayName = (LPSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpDisplayName); lpStatusPtr++; } } if (dwError != ERROR_SUCCESS) { TRACE("REnumDependentServicesA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("EnumDependentServicesA() done\n"); return TRUE; } /********************************************************************** * EnumDependentServicesW * * @implemented */ BOOL WINAPI EnumDependentServicesW(SC_HANDLE hService, DWORD dwServiceState, LPENUM_SERVICE_STATUSW lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned) { LPENUM_SERVICE_STATUSW lpStatusPtr; DWORD dwError; DWORD dwCount; TRACE("EnumServicesStatusW() called\n"); RpcTryExcept { dwError = REnumDependentServicesW((SC_RPC_HANDLE)hService, dwServiceState, (LPBYTE)lpServices, cbBufSize, pcbBytesNeeded, lpServicesReturned); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError == ERROR_SUCCESS || dwError == ERROR_MORE_DATA) { lpStatusPtr = (LPENUM_SERVICE_STATUSW)lpServices; for (dwCount = 0; dwCount < *lpServicesReturned; dwCount++) { if (lpStatusPtr->lpServiceName) lpStatusPtr->lpServiceName = (LPWSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpServiceName); if (lpStatusPtr->lpDisplayName) lpStatusPtr->lpDisplayName = (LPWSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpDisplayName); lpStatusPtr++; } } if (dwError != ERROR_SUCCESS) { TRACE("REnumDependentServicesW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("EnumDependentServicesW() done\n"); return TRUE; } /********************************************************************** * EnumServiceGroupW * * @unimplemented */ BOOL WINAPI EnumServiceGroupW( SC_HANDLE hSCManager, DWORD dwServiceType, DWORD dwServiceState, LPENUM_SERVICE_STATUSW lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned, LPDWORD lpResumeHandle, LPCWSTR lpGroup) { FIXME("EnumServiceGroupW is unimplemented\n"); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /********************************************************************** * EnumServicesStatusA * * @implemented */ BOOL WINAPI EnumServicesStatusA(SC_HANDLE hSCManager, DWORD dwServiceType, DWORD dwServiceState, LPENUM_SERVICE_STATUSA lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned, LPDWORD lpResumeHandle) { LPENUM_SERVICE_STATUSA lpStatusPtr; DWORD dwError; DWORD dwCount; TRACE("EnumServicesStatusA() called\n"); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (dwServiceType != SERVICE_DRIVER && dwServiceType != SERVICE_WIN32) { if (pcbBytesNeeded && lpServicesReturned) { *pcbBytesNeeded = 0; *lpServicesReturned = 0; } SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (dwServiceState != SERVICE_ACTIVE && dwServiceState != SERVICE_INACTIVE && dwServiceState != SERVICE_STATE_ALL) { if (pcbBytesNeeded) *pcbBytesNeeded = 0; if (lpServicesReturned) *lpServicesReturned = 0; SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (!pcbBytesNeeded || !lpServicesReturned) { SetLastError(ERROR_INVALID_ADDRESS); return FALSE; } if (!lpServices && cbBufSize != 0) { SetLastError(ERROR_INVALID_ADDRESS); return FALSE; } RpcTryExcept { dwError = REnumServicesStatusA((SC_RPC_HANDLE)hSCManager, dwServiceType, dwServiceState, (LPBYTE)lpServices, cbBufSize, pcbBytesNeeded, lpServicesReturned, lpResumeHandle); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError == ERROR_SUCCESS || dwError == ERROR_MORE_DATA) { lpStatusPtr = (LPENUM_SERVICE_STATUSA)lpServices; for (dwCount = 0; dwCount < *lpServicesReturned; dwCount++) { if (lpStatusPtr->lpServiceName) lpStatusPtr->lpServiceName = (LPSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpServiceName); if (lpStatusPtr->lpDisplayName) lpStatusPtr->lpDisplayName = (LPSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpDisplayName); lpStatusPtr++; } } if (dwError != ERROR_SUCCESS) { TRACE("REnumServicesStatusA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("EnumServicesStatusA() done\n"); return TRUE; } /********************************************************************** * EnumServicesStatusW * * @implemented */ BOOL WINAPI EnumServicesStatusW(SC_HANDLE hSCManager, DWORD dwServiceType, DWORD dwServiceState, LPENUM_SERVICE_STATUSW lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned, LPDWORD lpResumeHandle) { LPENUM_SERVICE_STATUSW lpStatusPtr; DWORD dwError; DWORD dwCount; TRACE("EnumServicesStatusW() called\n"); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } RpcTryExcept { dwError = REnumServicesStatusW((SC_RPC_HANDLE)hSCManager, dwServiceType, dwServiceState, (LPBYTE)lpServices, cbBufSize, pcbBytesNeeded, lpServicesReturned, lpResumeHandle); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError == ERROR_SUCCESS || dwError == ERROR_MORE_DATA) { lpStatusPtr = (LPENUM_SERVICE_STATUSW)lpServices; for (dwCount = 0; dwCount < *lpServicesReturned; dwCount++) { if (lpStatusPtr->lpServiceName) lpStatusPtr->lpServiceName = (LPWSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpServiceName); if (lpStatusPtr->lpDisplayName) lpStatusPtr->lpDisplayName = (LPWSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpDisplayName); lpStatusPtr++; } } if (dwError != ERROR_SUCCESS) { TRACE("REnumServicesStatusW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("EnumServicesStatusW() done\n"); return TRUE; } /********************************************************************** * EnumServicesStatusExA * * @implemented */ BOOL WINAPI EnumServicesStatusExA(SC_HANDLE hSCManager, SC_ENUM_TYPE InfoLevel, DWORD dwServiceType, DWORD dwServiceState, LPBYTE lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned, LPDWORD lpResumeHandle, LPCSTR pszGroupName) { LPENUM_SERVICE_STATUS_PROCESSA lpStatusPtr; DWORD dwError; DWORD dwCount; TRACE("EnumServicesStatusExA() called\n"); if (InfoLevel != SC_ENUM_PROCESS_INFO) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } RpcTryExcept { dwError = REnumServicesStatusExA((SC_RPC_HANDLE)hSCManager, InfoLevel, dwServiceType, dwServiceState, (LPBYTE)lpServices, cbBufSize, pcbBytesNeeded, lpServicesReturned, lpResumeHandle, (LPSTR)pszGroupName); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError == ERROR_SUCCESS || dwError == ERROR_MORE_DATA) { lpStatusPtr = (LPENUM_SERVICE_STATUS_PROCESSA)lpServices; for (dwCount = 0; dwCount < *lpServicesReturned; dwCount++) { if (lpStatusPtr->lpServiceName) lpStatusPtr->lpServiceName = (LPSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpServiceName); if (lpStatusPtr->lpDisplayName) lpStatusPtr->lpDisplayName = (LPSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpDisplayName); lpStatusPtr++; } } if (dwError != ERROR_SUCCESS) { TRACE("REnumServicesStatusExA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("EnumServicesStatusExA() done\n"); return TRUE; } /********************************************************************** * EnumServicesStatusExW * * @implemented */ BOOL WINAPI EnumServicesStatusExW(SC_HANDLE hSCManager, SC_ENUM_TYPE InfoLevel, DWORD dwServiceType, DWORD dwServiceState, LPBYTE lpServices, DWORD cbBufSize, LPDWORD pcbBytesNeeded, LPDWORD lpServicesReturned, LPDWORD lpResumeHandle, LPCWSTR pszGroupName) { LPENUM_SERVICE_STATUS_PROCESSW lpStatusPtr; DWORD dwError; DWORD dwCount; TRACE("EnumServicesStatusExW() called\n"); RpcTryExcept { dwError = REnumServicesStatusExW((SC_RPC_HANDLE)hSCManager, InfoLevel, dwServiceType, dwServiceState, (LPBYTE)lpServices, cbBufSize, pcbBytesNeeded, lpServicesReturned, lpResumeHandle, (LPWSTR)pszGroupName); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError == ERROR_SUCCESS || dwError == ERROR_MORE_DATA) { lpStatusPtr = (LPENUM_SERVICE_STATUS_PROCESSW)lpServices; for (dwCount = 0; dwCount < *lpServicesReturned; dwCount++) { if (lpStatusPtr->lpServiceName) lpStatusPtr->lpServiceName = (LPWSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpServiceName); if (lpStatusPtr->lpDisplayName) lpStatusPtr->lpDisplayName = (LPWSTR)((ULONG_PTR)lpServices + (ULONG_PTR)lpStatusPtr->lpDisplayName); lpStatusPtr++; } } if (dwError != ERROR_SUCCESS) { TRACE("REnumServicesStatusExW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } TRACE("EnumServicesStatusExW() done\n"); return TRUE; } /********************************************************************** * GetServiceDisplayNameA * * @implemented */ BOOL WINAPI GetServiceDisplayNameA(SC_HANDLE hSCManager, LPCSTR lpServiceName, LPSTR lpDisplayName, LPDWORD lpcchBuffer) { DWORD dwError; LPSTR lpNameBuffer; CHAR szEmptyName[] = ""; TRACE("GetServiceDisplayNameA() called\n"); TRACE("%p %s %p %p\n", hSCManager, debugstr_a(lpServiceName), lpDisplayName, lpcchBuffer); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (!lpDisplayName || *lpcchBuffer < sizeof(CHAR)) { lpNameBuffer = szEmptyName; *lpcchBuffer = sizeof(CHAR); } else { lpNameBuffer = lpDisplayName; } RpcTryExcept { dwError = RGetServiceDisplayNameA((SC_RPC_HANDLE)hSCManager, lpServiceName, lpNameBuffer, lpcchBuffer); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { /* HACK: because of a problem with rpcrt4, rpcserver is hacked to return 6 for ERROR_SERVICE_DOES_NOT_EXIST */ dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RGetServiceDisplayNameA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * GetServiceDisplayNameW * * @implemented */ BOOL WINAPI GetServiceDisplayNameW(SC_HANDLE hSCManager, LPCWSTR lpServiceName, LPWSTR lpDisplayName, LPDWORD lpcchBuffer) { DWORD dwError; LPWSTR lpNameBuffer; WCHAR szEmptyName[] = L""; TRACE("GetServiceDisplayNameW() called\n"); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (!lpDisplayName || *lpcchBuffer < sizeof(WCHAR)) { lpNameBuffer = szEmptyName; *lpcchBuffer = sizeof(WCHAR); } else { lpNameBuffer = lpDisplayName; } RpcTryExcept { dwError = RGetServiceDisplayNameW((SC_RPC_HANDLE)hSCManager, lpServiceName, lpNameBuffer, lpcchBuffer); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RGetServiceDisplayNameW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * GetServiceKeyNameA * * @implemented */ BOOL WINAPI GetServiceKeyNameA(SC_HANDLE hSCManager, LPCSTR lpDisplayName, LPSTR lpServiceName, LPDWORD lpcchBuffer) { DWORD dwError; LPSTR lpNameBuffer; CHAR szEmptyName[] = ""; TRACE("GetServiceKeyNameA() called\n"); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (!lpServiceName || *lpcchBuffer < sizeof(CHAR)) { lpNameBuffer = szEmptyName; *lpcchBuffer = sizeof(CHAR); } else { lpNameBuffer = lpServiceName; } RpcTryExcept { dwError = RGetServiceKeyNameA((SC_RPC_HANDLE)hSCManager, lpDisplayName, lpNameBuffer, lpcchBuffer); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RGetServiceKeyNameA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * GetServiceKeyNameW * * @implemented */ BOOL WINAPI GetServiceKeyNameW(SC_HANDLE hSCManager, LPCWSTR lpDisplayName, LPWSTR lpServiceName, LPDWORD lpcchBuffer) { DWORD dwError; LPWSTR lpNameBuffer; WCHAR szEmptyName[] = L""; TRACE("GetServiceKeyNameW() called\n"); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } if (!lpServiceName || *lpcchBuffer < sizeof(WCHAR)) { lpNameBuffer = szEmptyName; *lpcchBuffer = sizeof(WCHAR); } else { lpNameBuffer = lpServiceName; } RpcTryExcept { dwError = RGetServiceKeyNameW((SC_RPC_HANDLE)hSCManager, lpDisplayName, lpNameBuffer, lpcchBuffer); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RGetServiceKeyNameW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * LockServiceDatabase * * @implemented */ SC_LOCK WINAPI LockServiceDatabase(SC_HANDLE hSCManager) { SC_LOCK hLock; DWORD dwError; TRACE("LockServiceDatabase(%x)\n", hSCManager); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RLockServiceDatabase((SC_RPC_HANDLE)hSCManager, (SC_RPC_LOCK *)&hLock); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RLockServiceDatabase() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } TRACE("hLock = %p\n", hLock); return hLock; } static VOID WaitForSCManager(VOID) { HANDLE hEvent; TRACE("WaitForSCManager() called\n"); /* Try to open the existing event */ hEvent = OpenEventW(SYNCHRONIZE, FALSE, L"SvcctrlStartEvent_A3752DX"); if (hEvent == NULL) { if (GetLastError() != ERROR_FILE_NOT_FOUND) return; /* Try to create a new event */ hEvent = CreateEventW(NULL, TRUE, FALSE, L"SvcctrlStartEvent_A3752DX"); if (hEvent == NULL) { /* Try to open the existing event again */ hEvent = OpenEventW(SYNCHRONIZE, FALSE, L"SvcctrlStartEvent_A3752DX"); if (hEvent == NULL) return; } } /* Wait for 3 minutes */ WaitForSingleObject(hEvent, 180000); CloseHandle(hEvent); TRACE("ScmWaitForSCManager() done\n"); } /********************************************************************** * OpenSCManagerA * * @implemented */ SC_HANDLE WINAPI OpenSCManagerA(LPCSTR lpMachineName, LPCSTR lpDatabaseName, DWORD dwDesiredAccess) { SC_HANDLE hScm = NULL; DWORD dwError; TRACE("OpenSCManagerA(%s, %s, %lx)\n", lpMachineName, lpDatabaseName, dwDesiredAccess); WaitForSCManager(); RpcTryExcept { /* Call to services.exe using RPC */ dwError = ROpenSCManagerA((LPSTR)lpMachineName, (LPSTR)lpDatabaseName, dwDesiredAccess, (SC_RPC_HANDLE *)&hScm); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("ROpenSCManagerA() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } TRACE("hScm = %p\n", hScm); return hScm; } /********************************************************************** * OpenSCManagerW * * @implemented */ SC_HANDLE WINAPI OpenSCManagerW(LPCWSTR lpMachineName, LPCWSTR lpDatabaseName, DWORD dwDesiredAccess) { SC_HANDLE hScm = NULL; DWORD dwError; TRACE("OpenSCManagerW(%S, %S, %lx)\n", lpMachineName, lpDatabaseName, dwDesiredAccess); WaitForSCManager(); RpcTryExcept { /* Call to services.exe using RPC */ dwError = ROpenSCManagerW((LPWSTR)lpMachineName, (LPWSTR)lpDatabaseName, dwDesiredAccess, (SC_RPC_HANDLE *)&hScm); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("ROpenSCManagerW() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } TRACE("hScm = %p\n", hScm); return hScm; } /********************************************************************** * OpenServiceA * * @implemented */ SC_HANDLE WINAPI OpenServiceA(SC_HANDLE hSCManager, LPCSTR lpServiceName, DWORD dwDesiredAccess) { SC_HANDLE hService = NULL; DWORD dwError; TRACE("OpenServiceA(%p, %s, %lx)\n", hSCManager, lpServiceName, dwDesiredAccess); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return NULL; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = ROpenServiceA((SC_RPC_HANDLE)hSCManager, (LPSTR)lpServiceName, dwDesiredAccess, (SC_RPC_HANDLE *)&hService); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("ROpenServiceA() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } TRACE("hService = %p\n", hService); return hService; } /********************************************************************** * OpenServiceW * * @implemented */ SC_HANDLE WINAPI OpenServiceW(SC_HANDLE hSCManager, LPCWSTR lpServiceName, DWORD dwDesiredAccess) { SC_HANDLE hService = NULL; DWORD dwError; TRACE("OpenServiceW(%p, %S, %lx)\n", hSCManager, lpServiceName, dwDesiredAccess); if (!hSCManager) { SetLastError(ERROR_INVALID_HANDLE); return NULL; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = ROpenServiceW((SC_RPC_HANDLE)hSCManager, (LPWSTR)lpServiceName, dwDesiredAccess, (SC_RPC_HANDLE *)&hService); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("ROpenServiceW() failed (Error %lu)\n", dwError); SetLastError(dwError); return NULL; } TRACE("hService = %p\n", hService); return hService; } /********************************************************************** * QueryServiceConfigA * * @implemented */ BOOL WINAPI QueryServiceConfigA(SC_HANDLE hService, LPQUERY_SERVICE_CONFIGA lpServiceConfig, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceConfigA(%p, %p, %lu, %p)\n", hService, lpServiceConfig, cbBufSize, pcbBytesNeeded); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceConfigA((SC_RPC_HANDLE)hService, (LPBYTE)lpServiceConfig, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceConfigA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } /* Adjust the pointers */ if (lpServiceConfig->lpBinaryPathName) lpServiceConfig->lpBinaryPathName = (LPSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpBinaryPathName); if (lpServiceConfig->lpLoadOrderGroup) lpServiceConfig->lpLoadOrderGroup = (LPSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpLoadOrderGroup); if (lpServiceConfig->lpDependencies) lpServiceConfig->lpDependencies = (LPSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpDependencies); if (lpServiceConfig->lpServiceStartName) lpServiceConfig->lpServiceStartName = (LPSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpServiceStartName); if (lpServiceConfig->lpDisplayName) lpServiceConfig->lpDisplayName = (LPSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpDisplayName); TRACE("QueryServiceConfigA() done\n"); return TRUE; } /********************************************************************** * QueryServiceConfigW * * @implemented */ BOOL WINAPI QueryServiceConfigW(SC_HANDLE hService, LPQUERY_SERVICE_CONFIGW lpServiceConfig, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceConfigW(%p, %p, %lu, %p)\n", hService, lpServiceConfig, cbBufSize, pcbBytesNeeded); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceConfigW((SC_RPC_HANDLE)hService, (LPBYTE)lpServiceConfig, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceConfigW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } /* Adjust the pointers */ if (lpServiceConfig->lpBinaryPathName) lpServiceConfig->lpBinaryPathName = (LPWSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpBinaryPathName); if (lpServiceConfig->lpLoadOrderGroup) lpServiceConfig->lpLoadOrderGroup = (LPWSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpLoadOrderGroup); if (lpServiceConfig->lpDependencies) lpServiceConfig->lpDependencies = (LPWSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpDependencies); if (lpServiceConfig->lpServiceStartName) lpServiceConfig->lpServiceStartName = (LPWSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpServiceStartName); if (lpServiceConfig->lpDisplayName) lpServiceConfig->lpDisplayName = (LPWSTR)((ULONG_PTR)lpServiceConfig + (ULONG_PTR)lpServiceConfig->lpDisplayName); TRACE("QueryServiceConfigW() done\n"); return TRUE; } /********************************************************************** * QueryServiceConfig2A * * @implemented */ BOOL WINAPI QueryServiceConfig2A(SC_HANDLE hService, DWORD dwInfoLevel, LPBYTE lpBuffer, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceConfig2A(hService %p, dwInfoLevel %lu, lpBuffer %p, cbBufSize %lu, pcbBytesNeeded %p)\n", hService, dwInfoLevel, lpBuffer, cbBufSize, pcbBytesNeeded); if (dwInfoLevel != SERVICE_CONFIG_DESCRIPTION && dwInfoLevel != SERVICE_CONFIG_FAILURE_ACTIONS) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if ((lpBuffer == NULL && cbBufSize != 0) || pcbBytesNeeded == NULL) { SetLastError(ERROR_INVALID_ADDRESS); return FALSE; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceConfig2A((SC_RPC_HANDLE)hService, dwInfoLevel, lpBuffer, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceConfig2A() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } switch (dwInfoLevel) { case SERVICE_CONFIG_DESCRIPTION: { LPSERVICE_DESCRIPTIONA lpPtr = (LPSERVICE_DESCRIPTIONA)lpBuffer; if (lpPtr->lpDescription != NULL) lpPtr->lpDescription = (LPSTR)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpDescription); } break; case SERVICE_CONFIG_FAILURE_ACTIONS: { LPSERVICE_FAILURE_ACTIONSA lpPtr = (LPSERVICE_FAILURE_ACTIONSA)lpBuffer; if (lpPtr->lpRebootMsg != NULL) lpPtr->lpRebootMsg = (LPSTR)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpRebootMsg); if (lpPtr->lpCommand != NULL) lpPtr->lpCommand = (LPSTR)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpCommand); if (lpPtr->lpsaActions != NULL) lpPtr->lpsaActions = (SC_ACTION*)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpsaActions); } break; default: ERR("Unknown info level 0x%lx\n", dwInfoLevel); SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } TRACE("QueryServiceConfig2A() done\n"); return TRUE; } /********************************************************************** * QueryServiceConfig2W * * @implemented */ BOOL WINAPI QueryServiceConfig2W(SC_HANDLE hService, DWORD dwInfoLevel, LPBYTE lpBuffer, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceConfig2W(%p, %lu, %p, %lu, %p)\n", hService, dwInfoLevel, lpBuffer, cbBufSize, pcbBytesNeeded); if (dwInfoLevel != SERVICE_CONFIG_DESCRIPTION && dwInfoLevel != SERVICE_CONFIG_FAILURE_ACTIONS) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } if ((lpBuffer == NULL && cbBufSize != 0) || pcbBytesNeeded == NULL) { SetLastError(ERROR_INVALID_ADDRESS); return FALSE; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceConfig2W((SC_RPC_HANDLE)hService, dwInfoLevel, lpBuffer, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceConfig2W() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } switch (dwInfoLevel) { case SERVICE_CONFIG_DESCRIPTION: { LPSERVICE_DESCRIPTIONW lpPtr = (LPSERVICE_DESCRIPTIONW)lpBuffer; if (lpPtr->lpDescription != NULL) lpPtr->lpDescription = (LPWSTR)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpDescription); } break; case SERVICE_CONFIG_FAILURE_ACTIONS: { LPSERVICE_FAILURE_ACTIONSW lpPtr = (LPSERVICE_FAILURE_ACTIONSW)lpBuffer; if (lpPtr->lpRebootMsg != NULL) lpPtr->lpRebootMsg = (LPWSTR)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpRebootMsg); if (lpPtr->lpCommand != NULL) lpPtr->lpCommand = (LPWSTR)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpCommand); if (lpPtr->lpsaActions != NULL) lpPtr->lpsaActions = (SC_ACTION*)((UINT_PTR)lpPtr + (UINT_PTR)lpPtr->lpsaActions); } break; default: WARN("Unknown info level 0x%lx\n", dwInfoLevel); SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } TRACE("QueryServiceConfig2W() done\n"); return TRUE; } /********************************************************************** * QueryServiceLockStatusA * * @implemented */ BOOL WINAPI QueryServiceLockStatusA(SC_HANDLE hSCManager, LPQUERY_SERVICE_LOCK_STATUSA lpLockStatus, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceLockStatusA() called\n"); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceLockStatusA((SC_RPC_HANDLE)hSCManager, lpLockStatus, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceLockStatusA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } if (lpLockStatus->lpLockOwner != NULL) { lpLockStatus->lpLockOwner = (LPSTR)((UINT_PTR)lpLockStatus + (UINT_PTR)lpLockStatus->lpLockOwner); } TRACE("QueryServiceLockStatusA() done\n"); return TRUE; } /********************************************************************** * QueryServiceLockStatusW * * @implemented */ BOOL WINAPI QueryServiceLockStatusW(SC_HANDLE hSCManager, LPQUERY_SERVICE_LOCK_STATUSW lpLockStatus, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceLockStatusW() called\n"); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceLockStatusW((SC_RPC_HANDLE)hSCManager, lpLockStatus, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceLockStatusW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } if (lpLockStatus->lpLockOwner != NULL) { lpLockStatus->lpLockOwner = (LPWSTR)((UINT_PTR)lpLockStatus + (UINT_PTR)lpLockStatus->lpLockOwner); } TRACE("QueryServiceLockStatusW() done\n"); return TRUE; } /********************************************************************** * QueryServiceObjectSecurity * * @implemented */ BOOL WINAPI QueryServiceObjectSecurity(SC_HANDLE hService, SECURITY_INFORMATION dwSecurityInformation, PSECURITY_DESCRIPTOR lpSecurityDescriptor, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceObjectSecurity(%p, %lu, %p)\n", hService, dwSecurityInformation, lpSecurityDescriptor); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceObjectSecurity((SC_RPC_HANDLE)hService, dwSecurityInformation, (LPBYTE)lpSecurityDescriptor, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("QueryServiceObjectSecurity() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * SetServiceObjectSecurity * * @implemented */ BOOL WINAPI SetServiceObjectSecurity(SC_HANDLE hService, SECURITY_INFORMATION dwSecurityInformation, PSECURITY_DESCRIPTOR lpSecurityDescriptor) { PSECURITY_DESCRIPTOR SelfRelativeSD = NULL; ULONG Length; NTSTATUS Status; DWORD dwError; Length = 0; Status = RtlMakeSelfRelativeSD(lpSecurityDescriptor, SelfRelativeSD, &Length); if (Status != STATUS_BUFFER_TOO_SMALL) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } SelfRelativeSD = HeapAlloc(GetProcessHeap(), 0, Length); if (SelfRelativeSD == NULL) { SetLastError(ERROR_NOT_ENOUGH_MEMORY); return FALSE; } Status = RtlMakeSelfRelativeSD(lpSecurityDescriptor, SelfRelativeSD, &Length); if (!NT_SUCCESS(Status)) { HeapFree(GetProcessHeap(), 0, SelfRelativeSD); SetLastError(RtlNtStatusToDosError(Status)); return FALSE; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = RSetServiceObjectSecurity((SC_RPC_HANDLE)hService, dwSecurityInformation, (LPBYTE)SelfRelativeSD, Length); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; HeapFree(GetProcessHeap(), 0, SelfRelativeSD); if (dwError != ERROR_SUCCESS) { TRACE("RServiceObjectSecurity() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * QueryServiceStatus * * @implemented */ BOOL WINAPI QueryServiceStatus(SC_HANDLE hService, LPSERVICE_STATUS lpServiceStatus) { DWORD dwError; TRACE("QueryServiceStatus(%p, %p)\n", hService, lpServiceStatus); if (!hService) { SetLastError(ERROR_INVALID_HANDLE); return FALSE; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceStatus((SC_RPC_HANDLE)hService, lpServiceStatus); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceStatus() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * QueryServiceStatusEx * * @implemented */ BOOL WINAPI QueryServiceStatusEx(SC_HANDLE hService, SC_STATUS_TYPE InfoLevel, LPBYTE lpBuffer, DWORD cbBufSize, LPDWORD pcbBytesNeeded) { DWORD dwError; TRACE("QueryServiceStatusEx() called\n"); if (InfoLevel != SC_STATUS_PROCESS_INFO) { SetLastError(ERROR_INVALID_LEVEL); return FALSE; } RpcTryExcept { /* Call to services.exe using RPC */ dwError = RQueryServiceStatusEx((SC_RPC_HANDLE)hService, InfoLevel, lpBuffer, cbBufSize, pcbBytesNeeded); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RQueryServiceStatusEx() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * StartServiceA * * @implemented */ BOOL WINAPI StartServiceA(SC_HANDLE hService, DWORD dwNumServiceArgs, LPCSTR *lpServiceArgVectors) { DWORD dwError; RpcTryExcept { dwError = RStartServiceA((SC_RPC_HANDLE)hService, dwNumServiceArgs, (LPSTRING_PTRSA)lpServiceArgVectors); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RStartServiceA() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * StartServiceW * * @implemented */ BOOL WINAPI StartServiceW(SC_HANDLE hService, DWORD dwNumServiceArgs, LPCWSTR *lpServiceArgVectors) { DWORD dwError; RpcTryExcept { dwError = RStartServiceW((SC_RPC_HANDLE)hService, dwNumServiceArgs, (LPSTRING_PTRSW)lpServiceArgVectors); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RStartServiceW() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * UnlockServiceDatabase * * @implemented */ BOOL WINAPI UnlockServiceDatabase(SC_LOCK ScLock) { DWORD dwError; TRACE("UnlockServiceDatabase(%x)\n", ScLock); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RUnlockServiceDatabase((LPSC_RPC_LOCK)&ScLock); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("RUnlockServiceDatabase() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /********************************************************************** * NotifyBootConfigStatus * * @implemented */ BOOL WINAPI NotifyBootConfigStatus(BOOL BootAcceptable) { DWORD dwError; TRACE("NotifyBootConfigStatus()\n"); RpcTryExcept { /* Call to services.exe using RPC */ dwError = RNotifyBootConfigStatus(NULL, BootAcceptable); } RpcExcept(EXCEPTION_EXECUTE_HANDLER) { dwError = ScmRpcStatusToWinError(RpcExceptionCode()); } RpcEndExcept; if (dwError != ERROR_SUCCESS) { TRACE("NotifyBootConfigStatus() failed (Error %lu)\n", dwError); SetLastError(dwError); return FALSE; } return TRUE; } /* EOF */