sorry, forgot to change one var
[reactos.git] / reactos / lib / advapi32 / service / scm.c
index 9d1e6ef..60dc0cf 100644 (file)
@@ -12,7 +12,7 @@
 
 /* INCLUDES ******************************************************************/
 
-#include "advapi32.h"
+#include <advapi32.h>
 #include "svcctl_c.h"
 
 #define NDEBUG
@@ -25,37 +25,37 @@ handle_t BindingHandle = NULL;
 static VOID
 HandleBind(VOID)
 {
-  LPWSTR pszStringBinding;
-  RPC_STATUS status;
+    LPWSTR pszStringBinding;
+    RPC_STATUS status;
 
-  if (BindingHandle != NULL)
-    return;
+    if (BindingHandle != NULL)
+        return;
 
-  status = RpcStringBindingComposeW(NULL,
-                                    L"ncacn_np",
-                                    NULL,
-                                    L"\\pipe\\ntsvcs",
-                                    NULL,
-                                    &pszStringBinding);
-  if (status)
-  {
-    DPRINT1("RpcStringBindingCompose returned 0x%x\n", status);
-    return;
-  }
+    status = RpcStringBindingComposeW(NULL,
+                                      L"ncacn_np",
+                                      NULL,
+                                      L"\\pipe\\ntsvcs",
+                                      NULL,
+                                      &pszStringBinding);
+    if (status)
+    {
+        DPRINT1("RpcStringBindingCompose returned 0x%x\n", status);
+        return;
+    }
 
-  /* Set the binding handle that will be used to bind to the server. */
-  status = RpcBindingFromStringBindingW(pszStringBinding,
-                                        &BindingHandle);
-  if (status)
-  {
-    DPRINT1("RpcBindingFromStringBinding returned 0x%x\n", status);
-  }
+    /* Set the binding handle that will be used to bind to the server. */
+    status = RpcBindingFromStringBindingW(pszStringBinding,
+                                          &BindingHandle);
+    if (status)
+    {
+        DPRINT1("RpcBindingFromStringBinding returned 0x%x\n", status);
+    }
 
-  status = RpcStringFreeW(&pszStringBinding);
-  if (status)
-  {
-    DPRINT1("RpcStringFree returned 0x%x\n", status);
-  }
+    status = RpcStringFreeW(&pszStringBinding);
+    if (status)
+    {
+        DPRINT1("RpcStringFree returned 0x%x\n", status);
+    }
 }
 
 
@@ -63,16 +63,16 @@ HandleBind(VOID)
 static VOID
 HandleUnbind(VOID)
 {
-  RPC_STATUS status;
+    RPC_STATUS status;
 
-  if (BindingHandle == NULL)
-    return;
+    if (BindingHandle == NULL)
+        return;
 
-  status = RpcBindingFree(&BindingHandle);
-  if (status)
-  {
-    DPRINT1("RpcBindingFree returned 0x%x\n", status);
-  }
+    status = RpcBindingFree(&BindingHandle);
+    if (status)
+    {
+        DPRINT1("RpcBindingFree returned 0x%x\n", status);
+    }
 }
 #endif
 
@@ -106,26 +106,68 @@ ChangeServiceConfigA(
 /**********************************************************************
  *  ChangeServiceConfigW
  *
- * @unimplemented
+ * @implemented
  */
-BOOL
-STDCALL
-ChangeServiceConfigW(
-    SC_HANDLE   hService,
-    DWORD       dwServiceType,
-    DWORD       dwStartType,
-    DWORD       dwErrorControl,
-    LPCWSTR     lpBinaryPathName,
-    LPCWSTR     lpLoadOrderGroup,
-    LPDWORD     lpdwTagId,
-    LPCWSTR     lpDependencies,
-    LPCWSTR     lpServiceStartName,
-    LPCWSTR     lpPassword,
-    LPCWSTR     lpDisplayName)
+BOOL STDCALL
+ChangeServiceConfigW(SC_HANDLE hService,
+                     DWORD dwServiceType,
+                     DWORD dwStartType,
+                     DWORD dwErrorControl,
+                     LPCWSTR lpBinaryPathName,
+                     LPCWSTR lpLoadOrderGroup,
+                     LPDWORD lpdwTagId,
+                     LPCWSTR lpDependencies,
+                     LPCWSTR lpServiceStartName,
+                     LPCWSTR lpPassword,
+                     LPCWSTR lpDisplayName)
 {
-    DPRINT1("ChangeServiceConfigW is unimplemented\n");
-    SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
-    return FALSE;
+    DWORD dwError;
+    DWORD dwDependenciesLength = 0;
+    DWORD dwLength;
+    LPWSTR lpStr;
+
+    DPRINT("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 */
+
+    HandleBind();
+
+    /* Call to services.exe using RPC */
+    dwError = ScmrChangeServiceConfigW(BindingHandle,
+                                       (unsigned int)hService,
+                                       dwServiceType,
+                                       dwStartType,
+                                       dwErrorControl,
+                                       (LPWSTR)lpBinaryPathName,
+                                       (LPWSTR)lpLoadOrderGroup,
+                                       lpdwTagId,
+                                       (LPWSTR)lpDependencies,
+                                       dwDependenciesLength,
+                                       (LPWSTR)lpServiceStartName,
+                                       NULL,              /* FIXME: lpPassword */
+                                       0,                 /* FIXME: dwPasswordLength */
+                                       (LPWSTR)lpDisplayName);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrChangeServiceConfigW() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
+
+    return TRUE;
 }
 
 
@@ -137,25 +179,25 @@ ChangeServiceConfigW(
 BOOL STDCALL
 CloseServiceHandle(SC_HANDLE hSCObject)
 {
-  DWORD dwError;
+    DWORD dwError;
 
-  DPRINT("CloseServiceHandle() called\n");
+    DPRINT("CloseServiceHandle() called\n");
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrCloseServiceHandle(BindingHandle,
-                                   (unsigned int)hSCObject);
-  if (dwError)
-  {
-    DPRINT1("ScmrCloseServiceHandle() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return FALSE;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrCloseServiceHandle(BindingHandle,
+                                     (unsigned int)hSCObject);
+    if (dwError)
+    {
+        DPRINT1("ScmrCloseServiceHandle() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
 
-  DPRINT("CloseServiceHandle() done\n");
+    DPRINT("CloseServiceHandle() done\n");
 
-  return TRUE;
+    return TRUE;
 }
 
 
@@ -165,40 +207,59 @@ CloseServiceHandle(SC_HANDLE hSCObject)
  * @unimplemented
  */
 BOOL STDCALL
-ControlService(SC_HANDLE        hService,
-               DWORD            dwControl,
+ControlService(SC_HANDLE hService,
+               DWORD dwControl,
                LPSERVICE_STATUS lpServiceStatus)
 {
-  DWORD dwError;
+    DWORD dwError;
 
-  DPRINT("ControlService(%x, %x, %p)\n",
-         hService, dwControl, lpServiceStatus);
+    DPRINT("ControlService(%x, %x, %p)\n",
+           hService, dwControl, lpServiceStatus);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrControlService(BindingHandle,
-                               (unsigned int)hService,
-                               dwControl,
-                               lpServiceStatus);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrControlService() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return FALSE;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrControlService(BindingHandle,
+                                 (unsigned int)hService,
+                                 dwControl,
+                                 lpServiceStatus);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrControlService() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
 
-  DPRINT("ControlService() done\n");
+    DPRINT("ControlService() done\n");
 
-  return TRUE;
+    return TRUE;
 }
 
 
 /**********************************************************************
- *  CreateServiceA
+ *  ControlServiceEx
  *
  * @unimplemented
  */
+BOOL STDCALL
+ControlServiceEx(IN SC_HANDLE hService,
+                 IN DWORD dwControl,
+                 IN DWORD dwInfoLevel,
+                 IN OUT PVOID pControlParams)
+{
+    DPRINT1("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
 STDCALL
 CreateServiceA(
@@ -216,35 +277,191 @@ CreateServiceA(
     LPCSTR      lpServiceStartName,
     LPCSTR      lpPassword)
 {
-    DPRINT1("CreateServiceA is unimplemented, but returning INVALID_HANDLE_VALUE instead of NULL\n");
-    return INVALID_HANDLE_VALUE;
+    SC_HANDLE RetVal = NULL;
+    LPWSTR lpServiceNameW = NULL;
+    LPWSTR lpDisplayNameW = NULL;
+    LPWSTR lpBinaryPathNameW = NULL;
+    LPWSTR lpLoadOrderGroupW = NULL;
+    LPWSTR lpDependenciesW = NULL;
+    LPWSTR lpServiceStartNameW = NULL;
+    LPWSTR lpPasswordW = NULL;
+    DWORD dwDependenciesLength = 0;
+    DWORD dwLength;
+    LPSTR lpStr;
+
+    int len = MultiByteToWideChar(CP_ACP, 0, lpServiceName, -1, NULL, 0);
+    lpServiceNameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
+    if (!lpServiceNameW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpServiceName, -1, lpServiceNameW, len);
+
+    len = MultiByteToWideChar(CP_ACP, 0, lpDisplayName, -1, NULL, 0);
+    lpDisplayNameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
+    if (!lpDisplayNameW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpDisplayName, -1, lpDisplayNameW, len);
+
+    len = MultiByteToWideChar(CP_ACP, 0, lpBinaryPathName, -1, NULL, 0);
+    lpBinaryPathNameW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
+    if (!lpBinaryPathNameW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpDisplayName, -1, lpBinaryPathNameW, len);
+
+    len = MultiByteToWideChar(CP_ACP, 0, lpLoadOrderGroup, -1, NULL, 0);
+    lpLoadOrderGroupW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
+    if (!lpLoadOrderGroupW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpLoadOrderGroup, -1, lpLoadOrderGroupW, len);
+
+    if (lpDependencies != NULL)
+    {
+        lpStr = (LPSTR)lpDependencies;
+        while (*lpStr)
+        {
+            dwLength = strlen(lpStr) + 1;
+            dwDependenciesLength += dwLength;
+            lpStr = lpStr + dwLength;
+        }
+        dwDependenciesLength++;
+    }
+
+    lpDependenciesW = HeapAlloc(GetProcessHeap(), 0, dwDependenciesLength * sizeof(WCHAR));
+    if (!lpDependenciesW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpDependencies, -1, lpDependenciesW, dwDependenciesLength);
+
+    len = MultiByteToWideChar(CP_ACP, 0, lpServiceStartName, -1, NULL, 0);
+    lpServiceStartName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
+    if (!lpServiceStartNameW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpServiceStartName, -1, lpServiceStartNameW, len);
+
+    len = MultiByteToWideChar(CP_ACP, 0, lpPassword, -1, NULL, 0);
+    lpPasswordW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
+    if (!lpPasswordW)
+    {
+        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
+        goto cleanup;
+    }
+    MultiByteToWideChar(CP_ACP, 0, lpPassword, -1, lpPasswordW, len);
+
+    RetVal = CreateServiceW(hSCManager,
+                   lpServiceNameW,
+                   lpDisplayNameW,
+                   dwDesiredAccess,
+                   dwServiceType,
+                   dwStartType,
+                   dwErrorControl,
+                   lpBinaryPathNameW,
+                   lpLoadOrderGroupW,
+                   lpdwTagId,
+                   lpDependenciesW,
+                   lpServiceStartNameW,
+                   lpPasswordW);
+
+
+cleanup:
+    HeapFree(GetProcessHeap(), 0, lpServiceNameW);
+    HeapFree(GetProcessHeap(), 0, lpDisplayNameW);
+    HeapFree(GetProcessHeap(), 0, lpBinaryPathNameW);
+    HeapFree(GetProcessHeap(), 0, lpLoadOrderGroupW);
+    HeapFree(GetProcessHeap(), 0, lpDependenciesW);
+    HeapFree(GetProcessHeap(), 0, lpServiceStartNameW);
+    HeapFree(GetProcessHeap(), 0, lpPasswordW);
+
+    return RetVal;
 }
 
 
 /**********************************************************************
  *  CreateServiceW
  *
- * @unimplemented
+ * @implemented
  */
-SC_HANDLE
-STDCALL
-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 STDCALL
+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)
 {
-    DPRINT1("CreateServiceW is unimplemented, but returning INVALID_HANDLE_VALUE instead of NULL\n");
-    return INVALID_HANDLE_VALUE;
+    SC_HANDLE hService = NULL;
+    DWORD dwError;
+    DWORD dwDependenciesLength = 0;
+    DWORD dwLength;
+    LPWSTR lpStr;
+
+    DPRINT1("CreateServiceW() 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 */
+
+    HandleBind();
+
+    /* Call to services.exe using RPC */
+    dwError = ScmrCreateServiceW(BindingHandle,
+                                 (unsigned int)hSCManager,
+                                 (LPWSTR)lpServiceName,
+                                 (LPWSTR)lpDisplayName,
+                                 dwDesiredAccess,
+                                 dwServiceType,
+                                 dwStartType,
+                                 dwErrorControl,
+                                 (LPWSTR)lpBinaryPathName,
+                                 (LPWSTR)lpLoadOrderGroup,
+                                 lpdwTagId,
+                                 (LPWSTR)lpDependencies,
+                                 dwDependenciesLength,
+                                 (LPWSTR)lpServiceStartName,
+                                 NULL,              /* FIXME: lpPassword */
+                                 0,                 /* FIXME: dwPasswordLength */
+                                 (unsigned int *)&hService);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrCreateServiceW() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return INVALID_HANDLE_VALUE;
+    }
+
+    return hService;
 }
 
 
@@ -256,23 +473,23 @@ CreateServiceW(
 BOOL STDCALL
 DeleteService(SC_HANDLE hService)
 {
-  DWORD dwError;
+    DWORD dwError;
 
-  DPRINT("DeleteService(%x)\n", hService);
+    DPRINT("DeleteService(%x)\n", hService);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrDeleteService(BindingHandle,
-                              (unsigned int)hService);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrDeleteService() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return FALSE;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrDeleteService(BindingHandle,
+                                (unsigned int)hService);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrDeleteService() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
 
-  return TRUE;
+    return TRUE;
 }
 
 
@@ -458,19 +675,35 @@ GetServiceDisplayNameA(
 /**********************************************************************
  *  GetServiceDisplayNameW
  *
- * @unimplemented
+ * @implemented
  */
-BOOL
-STDCALL
-GetServiceDisplayNameW(
-    SC_HANDLE   hSCManager,
-    LPCWSTR     lpServiceName,
-    LPWSTR      lpDisplayName,
-    LPDWORD     lpcchBuffer)
+BOOL STDCALL
+GetServiceDisplayNameW(SC_HANDLE hSCManager,
+                       LPCWSTR lpServiceName,
+                       LPWSTR lpDisplayName,
+                       LPDWORD lpcchBuffer)
 {
-    DPRINT1("GetServiceDisplayNameW is unimplemented\n");
-    SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
-    return FALSE;
+    DWORD dwError;
+
+    DPRINT("GetServiceDisplayNameW() called\n");
+
+    HandleBind();
+
+    dwError = ScmrGetServiceDisplayNameW(BindingHandle,
+                                         (unsigned int)hSCManager,
+                                         (LPWSTR)lpServiceName,
+                                         lpDisplayName,
+                                         lpcchBuffer);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrGetServiceDisplayNameW() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
+
+    (*lpcchBuffer)--;
+
+    return TRUE;
 }
 
 
@@ -496,19 +729,35 @@ GetServiceKeyNameA(
 /**********************************************************************
  *  GetServiceKeyNameW
  *
- * @unimplemented
+ * @implemented
  */
-BOOL
-STDCALL
-GetServiceKeyNameW(
-    SC_HANDLE   hSCManager,
-    LPCWSTR     lpDisplayName,
-    LPWSTR      lpServiceName,
-    LPDWORD     lpcchBuffer)
+BOOL STDCALL
+GetServiceKeyNameW(SC_HANDLE hSCManager,
+                   LPCWSTR lpDisplayName,
+                   LPWSTR lpServiceName,
+                   LPDWORD lpcchBuffer)
 {
-    DPRINT1("GetServiceKeyNameW is unimplemented\n");
-    SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
-    return FALSE;
+    DWORD dwError;
+
+    DPRINT("GetServiceKeyNameW() called\n");
+
+    HandleBind();
+
+    dwError = ScmrGetServiceKeyNameW(BindingHandle,
+                                     (unsigned int)hSCManager,
+                                     (LPWSTR)lpDisplayName,
+                                     lpServiceName,
+                                     lpcchBuffer);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrGetServiceKeyNameW() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
+
+    (*lpcchBuffer)--;
+
+    return TRUE;
 }
 
 
@@ -520,67 +769,67 @@ GetServiceKeyNameW(
 SC_LOCK STDCALL
 LockServiceDatabase(SC_HANDLE hSCManager)
 {
-  SC_LOCK hLock;
-  DWORD dwError;
+    SC_LOCK hLock;
+    DWORD dwError;
 
-  DPRINT("LockServiceDatabase(%x)\n", hSCManager);
+    DPRINT("LockServiceDatabase(%x)\n", hSCManager);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrLockServiceDatabase(BindingHandle,
-                                    (unsigned int)hSCManager,
-                                    (unsigned int *)&hLock);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrLockServiceDatabase() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return NULL;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrLockServiceDatabase(BindingHandle,
+                                      (unsigned int)hSCManager,
+                                      (unsigned int *)&hLock);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrLockServiceDatabase() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return NULL;
+    }
 
-  DPRINT("hLock = %p\n", hLock);
+    DPRINT("hLock = %p\n", hLock);
 
-  return hLock;
+    return hLock;
 }
 
 
 static VOID
 WaitForSCManager(VOID)
 {
-  HANDLE hEvent;
-
-  DPRINT("WaitForSCManager() called\n");
+    HANDLE hEvent;
 
-  /* Try to open the existing event */
-  hEvent = OpenEventW(SYNCHRONIZE,
-                      FALSE,
-                      L"SvcctrlStartEvent_A3725DX");
-  if (hEvent == NULL)
-  {
-    if (GetLastError() != ERROR_FILE_NOT_FOUND)
-      return;
+    DPRINT("WaitForSCManager() called\n");
 
-    /* Try to create a new event */
-    hEvent = CreateEventW(NULL,
-                          TRUE,
-                          FALSE,
-                          L"SvcctrlStartEvent_A3725DX");
+    /* Try to open the existing event */
+    hEvent = OpenEventW(SYNCHRONIZE,
+                        FALSE,
+                        L"SvcctrlStartEvent_A3725DX");
     if (hEvent == NULL)
     {
-      /* Try to open the existing event again */
-      hEvent = OpenEventW(SYNCHRONIZE,
-                          FALSE,
-                          L"SvcctrlStartEvent_A3725DX");
-      if (hEvent == NULL)
-        return;
+        if (GetLastError() != ERROR_FILE_NOT_FOUND)
+            return;
+
+        /* Try to create a new event */
+        hEvent = CreateEventW(NULL,
+                              TRUE,
+                              FALSE,
+                              L"SvcctrlStartEvent_A3725DX");
+        if (hEvent == NULL)
+        {
+            /* Try to open the existing event again */
+            hEvent = OpenEventW(SYNCHRONIZE,
+                                FALSE,
+                                L"SvcctrlStartEvent_A3725DX");
+            if (hEvent == NULL)
+                return;
+        }
     }
-  }
 
-  /* Wait for 3 minutes */
-  WaitForSingleObject(hEvent, 180000);
-  CloseHandle(hEvent);
+    /* Wait for 3 minutes */
+    WaitForSingleObject(hEvent, 180000);
+    CloseHandle(hEvent);
 
-  DPRINT("ScmWaitForSCManager() done\n");
+    DPRINT("ScmWaitForSCManager() done\n");
 }
 
 
@@ -594,32 +843,32 @@ OpenSCManagerA(LPCSTR lpMachineName,
                LPCSTR lpDatabaseName,
                DWORD dwDesiredAccess)
 {
-  SC_HANDLE hScm = NULL;
-  DWORD dwError;
+    SC_HANDLE hScm = NULL;
+    DWORD dwError;
 
-  DPRINT("OpenSCManagerA(%s, %s, %lx)\n",
-         lpMachineName, lpDatabaseName, dwDesiredAccess);
+    DPRINT("OpenSCManagerA(%s, %s, %lx)\n",
+           lpMachineName, lpDatabaseName, dwDesiredAccess);
 
-  WaitForSCManager();
+    WaitForSCManager();
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrOpenSCManagerA(BindingHandle,
-                               (LPSTR)lpMachineName,
-                               (LPSTR)lpDatabaseName,
-                               dwDesiredAccess,
-                               (unsigned int*)&hScm);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrOpenSCManagerA() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return NULL;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrOpenSCManagerA(BindingHandle,
+                                 (LPSTR)lpMachineName,
+                                 (LPSTR)lpDatabaseName,
+                                 dwDesiredAccess,
+                                 (unsigned int*)&hScm);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrOpenSCManagerA() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return NULL;
+    }
 
-  DPRINT("hScm = %p\n", hScm);
+    DPRINT("hScm = %p\n", hScm);
 
-  return hScm;
+    return hScm;
 }
 
 
@@ -633,32 +882,32 @@ OpenSCManagerW(LPCWSTR lpMachineName,
                LPCWSTR lpDatabaseName,
                DWORD dwDesiredAccess)
 {
-  SC_HANDLE hScm = NULL;
-  DWORD dwError;
+    SC_HANDLE hScm = NULL;
+    DWORD dwError;
 
-  DPRINT("OpenSCManagerW(%S, %S, %lx)\n",
-         lpMachineName, lpDatabaseName, dwDesiredAccess);
+    DPRINT("OpenSCManagerW(%S, %S, %lx)\n",
+           lpMachineName, lpDatabaseName, dwDesiredAccess);
 
-  WaitForSCManager();
+    WaitForSCManager();
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrOpenSCManagerW(BindingHandle,
-                               (LPWSTR)lpMachineName,
-                               (LPWSTR)lpDatabaseName,
-                               dwDesiredAccess,
-                               (unsigned int*)&hScm);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrOpenSCManagerW() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return NULL;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrOpenSCManagerW(BindingHandle,
+                                 (LPWSTR)lpMachineName,
+                                 (LPWSTR)lpDatabaseName,
+                                 dwDesiredAccess,
+                                 (unsigned int*)&hScm);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrOpenSCManagerW() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return NULL;
+    }
 
-  DPRINT("hScm = %p\n", hScm);
+    DPRINT("hScm = %p\n", hScm);
 
-  return hScm;
+    return hScm;
 }
 
 
@@ -672,30 +921,30 @@ OpenServiceA(SC_HANDLE hSCManager,
              LPCSTR lpServiceName,
              DWORD dwDesiredAccess)
 {
-  SC_HANDLE hService = NULL;
-  DWORD dwError;
+    SC_HANDLE hService = NULL;
+    DWORD dwError;
 
-  DPRINT("OpenServiceA(%p, %s, %lx)\n",
-         hSCManager, lpServiceName, dwDesiredAccess);
+    DPRINT("OpenServiceA(%p, %s, %lx)\n",
+           hSCManager, lpServiceName, dwDesiredAccess);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrOpenServiceA(BindingHandle,
-                             (unsigned int)hSCManager,
-                             (LPSTR)lpServiceName,
-                             dwDesiredAccess,
-                             (unsigned int*)&hService);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrOpenServiceA() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return NULL;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrOpenServiceA(BindingHandle,
+                               (unsigned int)hSCManager,
+                               (LPSTR)lpServiceName,
+                               dwDesiredAccess,
+                               (unsigned int*)&hService);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrOpenServiceA() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return NULL;
+    }
 
-  DPRINT("hService = %p\n", hService);
+    DPRINT("hService = %p\n", hService);
 
-  return hService;
+    return hService;
 }
 
 
@@ -709,30 +958,30 @@ OpenServiceW(SC_HANDLE hSCManager,
              LPCWSTR lpServiceName,
              DWORD dwDesiredAccess)
 {
-  SC_HANDLE hService = NULL;
-  DWORD dwError;
+    SC_HANDLE hService = NULL;
+    DWORD dwError;
 
-  DPRINT("OpenServiceW(%p, %S, %lx)\n",
-         hSCManager, lpServiceName, dwDesiredAccess);
+    DPRINT("OpenServiceW(%p, %S, %lx)\n",
+           hSCManager, lpServiceName, dwDesiredAccess);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrOpenServiceW(BindingHandle,
-                             (unsigned int)hSCManager,
-                             (LPWSTR)lpServiceName,
-                             dwDesiredAccess,
-                             (unsigned int*)&hService);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrOpenServiceW() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return NULL;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrOpenServiceW(BindingHandle,
+                               (unsigned int)hSCManager,
+                               (LPWSTR)lpServiceName,
+                               dwDesiredAccess,
+                               (unsigned int*)&hService);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrOpenServiceW() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return NULL;
+    }
 
-  DPRINT("hService = %p\n", hService);
+    DPRINT("hService = %p\n", hService);
 
-  return hService;
+    return hService;
 }
 
 
@@ -769,8 +1018,17 @@ QueryServiceConfigW(
     LPDWORD                 pcbBytesNeeded)
 {
     DPRINT1("QueryServiceConfigW is unimplemented\n");
-    SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
-    return FALSE;
+    if (lpServiceConfig && cbBufSize >= sizeof(QUERY_SERVICE_CONFIGW))
+    {
+        memset(lpServiceConfig, 0, *pcbBytesNeeded);
+        return TRUE;
+    }
+    else
+    {
+        *pcbBytesNeeded = sizeof(QUERY_SERVICE_CONFIGW);
+        SetLastError(ERROR_INSUFFICIENT_BUFFER);
+        return FALSE;
+    }
 }
 
 
@@ -841,25 +1099,25 @@ BOOL STDCALL
 QueryServiceStatus(SC_HANDLE hService,
                    LPSERVICE_STATUS lpServiceStatus)
 {
-  DWORD dwError;
+    DWORD dwError;
 
-  DPRINT("QueryServiceStatus(%p, %p)\n",
-         hService, lpServiceStatus);
+    DPRINT("QueryServiceStatus(%p, %p)\n",
+           hService, lpServiceStatus);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrQueryServiceStatus(BindingHandle,
-                                   (unsigned int)hService,
-                                   lpServiceStatus);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrQueryServiceStatus() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return FALSE;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrQueryServiceStatus(BindingHandle,
+                                     (unsigned int)hService,
+                                     lpServiceStatus);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrQueryServiceStatus() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
 
-  return TRUE;
+    return TRUE;
 }
 
 
@@ -912,9 +1170,10 @@ StartServiceW(
     DWORD       dwNumServiceArgs,
     LPCWSTR     *lpServiceArgVectors)
 {
-    DPRINT1("StartServiceW is unimplemented\n");
-    SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
-    return FALSE;
+    DPRINT1("StartServiceW is unimplemented, but returns success...\n");
+    //SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
+    //return FALSE;
+    return TRUE;
 }
 
 
@@ -926,34 +1185,63 @@ StartServiceW(
 BOOL STDCALL
 UnlockServiceDatabase(SC_LOCK ScLock)
 {
-  DWORD dwError;
+    DWORD dwError;
 
-  DPRINT("UnlockServiceDatabase(%x)\n", hSCManager);
+    DPRINT("UnlockServiceDatabase(%x)\n", ScLock);
 
-  HandleBind();
+    HandleBind();
 
-  /* Call to services.exe using RPC */
-  dwError = ScmrUnlockServiceDatabase(BindingHandle,
-                                      (unsigned int)ScLock);
-  if (dwError != ERROR_SUCCESS)
-  {
-    DPRINT1("ScmrUnlockServiceDatabase() failed (Error %lu)\n", dwError);
-    SetLastError(dwError);
-    return FALSE;
-  }
+    /* Call to services.exe using RPC */
+    dwError = ScmrUnlockServiceDatabase(BindingHandle,
+                                        (unsigned int)ScLock);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("ScmrUnlockServiceDatabase() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
 
-  return TRUE;
+    return TRUE;
+}
+
+
+/**********************************************************************
+ *  NotifyBootConfigStatus
+ *
+ * @implemented
+ */
+BOOL STDCALL
+NotifyBootConfigStatus(BOOL BootAcceptable)
+{
+    DWORD dwError;
+
+    DPRINT1("NotifyBootConfigStatus()\n");
+
+    HandleBind();
+
+    /* Call to services.exe using RPC */
+    dwError = ScmrNotifyBootConfigStatus(BindingHandle,
+                                         BootAcceptable);
+    if (dwError != ERROR_SUCCESS)
+    {
+        DPRINT1("NotifyBootConfigStatus() failed (Error %lu)\n", dwError);
+        SetLastError(dwError);
+        return FALSE;
+    }
+
+    return TRUE;
 }
 
 
 void __RPC_FAR * __RPC_USER midl_user_allocate(size_t len)
 {
-  return GlobalAlloc(GPTR,len);
+    return HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, len);
 }
 
+
 void __RPC_USER midl_user_free(void __RPC_FAR * ptr)
 {
-  GlobalFree(ptr);
+    HeapFree(GetProcessHeap(), 0, ptr);
 }
 
 /* EOF */