services: Remove dependency on service name being determined correctly in order for the two ends on the control pipe to meet up.

Instead use the mechanism used by native which uses 
\\.\pipe\net\NtControlPipeN.

Also remove the service startup event set by StartServiceCtrlDispatcher 
as we wait for the service status to change instead.
This commit is contained in:
Rob Shearman 2008-03-28 17:07:48 +00:00 committed by Alexandre Julliard
parent 9a6fc01d84
commit b8348b95a6
2 changed files with 89 additions and 88 deletions

View File

@ -421,23 +421,42 @@ static inline LONG service_write_values( HKEY hKey,
/****************************************************************************** /******************************************************************************
* Service IPC functions * Service IPC functions
*/ */
static LPWSTR service_get_pipe_name(LPCWSTR service) static LPWSTR service_get_pipe_name(void)
{ {
static const WCHAR prefix[] = { '\\','\\','.','\\','p','i','p','e','\\', static const WCHAR format[] = { '\\','\\','.','\\','p','i','p','e','\\',
'_','_','w','i','n','e','s','e','r','v','i','c','e','_',0}; 'n','e','t','\\','N','t','C','o','n','t','r','o','l','P','i','p','e','%','u',0};
static const WCHAR service_current_key_str[] = { 'S','Y','S','T','E','M','\\',
'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\',
'C','o','n','t','r','o','l','\\',
'S','e','r','v','i','c','e','C','u','r','r','e','n','t',0};
LPWSTR name; LPWSTR name;
DWORD len; DWORD len;
HKEY service_current_key;
DWORD service_current;
LONG ret;
DWORD type;
len = sizeof prefix + strlenW(service)*sizeof(WCHAR); ret = RegOpenKeyExW(HKEY_LOCAL_MACHINE, service_current_key_str, 0,
name = HeapAlloc(GetProcessHeap(), 0, len); KEY_QUERY_VALUE, &service_current_key);
strcpyW(name, prefix); if (ret != ERROR_SUCCESS)
strcatW(name, service); return NULL;
len = sizeof(service_current);
ret = RegQueryValueExW(service_current_key, NULL, NULL, &type,
(BYTE *)&service_current, &len);
RegCloseKey(service_current_key);
if (ret != ERROR_SUCCESS || type != REG_DWORD)
return NULL;
len = sizeof(format)/sizeof(WCHAR) + 10 /* strlenW("4294967295") */;
name = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
if (!name)
return NULL;
snprintfW(name, len, format, service_current);
return name; return name;
} }
static HANDLE service_open_pipe(LPCWSTR service) static HANDLE service_open_pipe(void)
{ {
LPWSTR szPipe = service_get_pipe_name( service ); LPWSTR szPipe = service_get_pipe_name();
HANDLE handle = INVALID_HANDLE_VALUE; HANDLE handle = INVALID_HANDLE_VALUE;
do { do {
@ -453,26 +472,6 @@ static HANDLE service_open_pipe(LPCWSTR service)
return handle; return handle;
} }
/******************************************************************************
* service_get_event_handle
*/
static HANDLE service_get_event_handle(LPCWSTR service)
{
static const WCHAR prefix[] = {
'_','_','w','i','n','e','s','e','r','v','i','c','e','_',0};
LPWSTR name;
DWORD len;
HANDLE handle;
len = sizeof prefix + strlenW(service)*sizeof(WCHAR);
name = HeapAlloc(GetProcessHeap(), 0, len);
strcpyW(name, prefix);
strcatW(name, service);
handle = CreateEventW(NULL, TRUE, FALSE, name);
HeapFree(GetProcessHeap(), 0, name);
return handle;
}
/****************************************************************************** /******************************************************************************
* service_thread * service_thread
* *
@ -596,22 +595,18 @@ static BOOL service_handle_control(HANDLE pipe, service_data *service,
static DWORD WINAPI service_control_dispatcher(LPVOID arg) static DWORD WINAPI service_control_dispatcher(LPVOID arg)
{ {
service_data *service = arg; service_data *service = arg;
HANDLE pipe, event; HANDLE pipe;
TRACE("%p %s\n", service, debugstr_w(service->name)); TRACE("%p %s\n", service, debugstr_w(service->name));
pipe = service_open_pipe(service->name); pipe = service_open_pipe();
if (pipe==INVALID_HANDLE_VALUE) if (pipe==INVALID_HANDLE_VALUE)
{
ERR("failed to create pipe for %s, error = %d\n", ERR("failed to create pipe for %s, error = %d\n",
debugstr_w(service->name), GetLastError()); debugstr_w(service->name), GetLastError());
return 0;
/* let the process who started us know we've tried to create a pipe */ }
event = service_get_event_handle(service->name);
SetEvent(event);
CloseHandle(event);
if (pipe==INVALID_HANDLE_VALUE) return 0;
/* dispatcher loop */ /* dispatcher loop */
while (1) while (1)

View File

@ -652,44 +652,54 @@ DWORD svcctl_QueryServiceStatusEx(
return ERROR_SUCCESS; return ERROR_SUCCESS;
} }
static HANDLE service_get_event_handle(LPCWSTR service) /* only one service started at a time, so there is no race on the registry
* value here */
static LPWSTR service_get_pipe_name(void)
{ {
static const WCHAR prefix[] = { static const WCHAR format[] = { '\\','\\','.','\\','p','i','p','e','\\',
'_','_','w','i','n','e','s','e','r','v','i','c','e','_',0}; 'n','e','t','\\','N','t','C','o','n','t','r','o','l','P','i','p','e','%','u',0};
static const WCHAR service_current_key_str[] = { 'S','Y','S','T','E','M','\\',
'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\',
'C','o','n','t','r','o','l','\\',
'S','e','r','v','i','c','e','C','u','r','r','e','n','t',0};
LPWSTR name; LPWSTR name;
DWORD len; DWORD len;
HANDLE handle; HKEY service_current_key;
DWORD service_current = -1;
LONG ret;
DWORD type;
len = sizeof prefix + strlenW(service)*sizeof(WCHAR); ret = RegCreateKeyExW(HKEY_LOCAL_MACHINE, service_current_key_str, 0,
name = HeapAlloc(GetProcessHeap(), 0, len); NULL, REG_OPTION_VOLATILE, KEY_SET_VALUE | KEY_QUERY_VALUE, NULL,
strcpyW(name, prefix); &service_current_key, NULL);
strcatW(name, service); if (ret != ERROR_SUCCESS)
handle = CreateEventW(NULL, TRUE, FALSE, name); return NULL;
HeapFree(GetProcessHeap(), 0, name); len = sizeof(service_current);
return handle; ret = RegQueryValueExW(service_current_key, NULL, NULL, &type,
} (BYTE *)&service_current, &len);
if ((ret == ERROR_SUCCESS && type == REG_DWORD) || ret == ERROR_FILE_NOT_FOUND)
static LPWSTR service_get_pipe_name(LPCWSTR service) {
{ service_current++;
static const WCHAR prefix[] = { '\\','\\','.','\\','p','i','p','e','\\', RegSetValueExW(service_current_key, NULL, 0, REG_DWORD,
'_','_','w','i','n','e','s','e','r','v','i','c','e','_',0}; (BYTE *)&service_current, sizeof(service_current));
LPWSTR name; }
DWORD len; RegCloseKey(service_current_key);
if ((ret != ERROR_SUCCESS || type != REG_DWORD) && (ret != ERROR_FILE_NOT_FOUND))
len = sizeof prefix + strlenW(service)*sizeof(WCHAR); return NULL;
name = HeapAlloc(GetProcessHeap(), 0, len); len = sizeof(format)/sizeof(WCHAR) + 10 /* strlenW("4294967295") */;
strcpyW(name, prefix); name = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
strcatW(name, service); if (!name)
return NULL;
snprintfW(name, len, format, service_current);
return name; return name;
} }
static DWORD service_start_process(struct service_entry *service_entry) static DWORD service_start_process(struct service_entry *service_entry, HANDLE *process)
{ {
PROCESS_INFORMATION pi; PROCESS_INFORMATION pi;
STARTUPINFOW si; STARTUPINFOW si;
LPWSTR path = NULL; LPWSTR path = NULL;
DWORD size, ret; DWORD size;
HANDLE handles[2];
BOOL r; BOOL r;
lock_services(); lock_services();
@ -714,8 +724,6 @@ static DWORD service_start_process(struct service_entry *service_entry)
ExpandEnvironmentStringsW(service_entry->config.lpBinaryPathName,path,size); ExpandEnvironmentStringsW(service_entry->config.lpBinaryPathName,path,size);
} }
/* wait for the process to start and set an event or terminate */
handles[0] = service_get_event_handle( service_entry->name );
ZeroMemory(&si, sizeof(STARTUPINFOW)); ZeroMemory(&si, sizeof(STARTUPINFOW));
si.cb = sizeof(STARTUPINFOW); si.cb = sizeof(STARTUPINFOW);
if (!(service_entry->config.dwServiceType & SERVICE_INTERACTIVE_PROCESS)) if (!(service_entry->config.dwServiceType & SERVICE_INTERACTIVE_PROCESS))
@ -729,36 +737,30 @@ static DWORD service_start_process(struct service_entry *service_entry)
r = CreateProcessW(NULL, path, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi); r = CreateProcessW(NULL, path, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi);
HeapFree(GetProcessHeap(),0,path); HeapFree(GetProcessHeap(),0,path);
if (!r) if (!r)
{
CloseHandle( handles[0] );
return GetLastError(); return GetLastError();
}
handles[1] = pi.hProcess; lock_services();
ret = WaitForMultipleObjectsEx(2, handles, FALSE, 30000, FALSE);
CloseHandle( pi.hThread );
CloseHandle( pi.hProcess );
CloseHandle( handles[0] );
if(ret != WAIT_OBJECT_0)
return ERROR_SERVICE_REQUEST_TIMEOUT;
/* FIXME */
service_entry->status.dwCurrentState = SERVICE_START_PENDING; service_entry->status.dwCurrentState = SERVICE_START_PENDING;
service_entry->status.dwProcessId = pi.dwProcessId; service_entry->status.dwProcessId = pi.dwProcessId;
unlock_services();
*process = pi.hProcess;
CloseHandle( pi.hThread );
return ERROR_SUCCESS; return ERROR_SUCCESS;
} }
static DWORD service_wait_for_startup(struct service_entry *service_entry) static DWORD service_wait_for_startup(struct service_entry *service_entry, HANDLE process_handle)
{ {
WINE_TRACE("%p\n", service_entry); WINE_TRACE("%p\n", service_entry);
for (;;) for (;;)
{ {
DWORD dwCurrentStatus; DWORD dwCurrentStatus;
DWORD ret = WaitForSingleObject(service_entry->status_changed_event, 20000); HANDLE handles[2] = { service_entry->status_changed_event, process_handle };
if (ret == WAIT_TIMEOUT) DWORD ret;
ret = WaitForMultipleObjects(sizeof(handles)/sizeof(handles[0]), handles, FALSE, 20000);
if (ret != WAIT_OBJECT_0)
return ERROR_SERVICE_REQUEST_TIMEOUT; return ERROR_SERVICE_REQUEST_TIMEOUT;
lock_services(); lock_services();
dwCurrentStatus = service_entry->status.dwCurrentState; dwCurrentStatus = service_entry->status.dwCurrentState;
@ -785,6 +787,7 @@ static BOOL service_send_start_message(HANDLE pipe, LPCWSTR *argv, DWORD argc)
WINE_TRACE("%p %p %d\n", pipe, argv, argc); WINE_TRACE("%p %p %d\n", pipe, argv, argc);
/* FIXME: this can block so should be done in another thread */
r = ConnectNamedPipe(pipe, NULL); r = ConnectNamedPipe(pipe, NULL);
if (!r && GetLastError() != ERROR_PIPE_CONNECTED) if (!r && GetLastError() != ERROR_PIPE_CONNECTED)
{ {
@ -907,6 +910,7 @@ DWORD svcctl_StartServiceW(
struct sc_service *service; struct sc_service *service;
DWORD err; DWORD err;
LPWSTR name; LPWSTR name;
HANDLE process_handle = NULL;
WINE_TRACE("(%p, %d, %p)\n", hService, dwNumServiceArgs, lpServiceArgVectors); WINE_TRACE("(%p, %d, %p)\n", hService, dwNumServiceArgs, lpServiceArgVectors);
@ -928,7 +932,7 @@ DWORD svcctl_StartServiceW(
if (!service->service_entry->status_changed_event) if (!service->service_entry->status_changed_event)
service->service_entry->status_changed_event = CreateEventW(NULL, TRUE, FALSE, NULL); service->service_entry->status_changed_event = CreateEventW(NULL, TRUE, FALSE, NULL);
name = service_get_pipe_name(service->service_entry->name); name = service_get_pipe_name();
service->service_entry->control_pipe = CreateNamedPipeW(name, PIPE_ACCESS_DUPLEX, service->service_entry->control_pipe = CreateNamedPipeW(name, PIPE_ACCESS_DUPLEX,
PIPE_TYPE_BYTE|PIPE_WAIT, 1, 256, 256, 10000, NULL ); PIPE_TYPE_BYTE|PIPE_WAIT, 1, 256, 256, 10000, NULL );
HeapFree(GetProcessHeap(), 0, name); HeapFree(GetProcessHeap(), 0, name);
@ -940,9 +944,7 @@ DWORD svcctl_StartServiceW(
return GetLastError(); return GetLastError();
} }
err = service_start_process(service->service_entry); err = service_start_process(service->service_entry, &process_handle);
unlock_service_database();
if (err == ERROR_SUCCESS) if (err == ERROR_SUCCESS)
{ {
@ -954,9 +956,13 @@ DWORD svcctl_StartServiceW(
WINE_TRACE("returning %d\n", err); WINE_TRACE("returning %d\n", err);
if (err == ERROR_SUCCESS) if (err == ERROR_SUCCESS)
err = service_wait_for_startup(service->service_entry); err = service_wait_for_startup(service->service_entry, process_handle);
if (process_handle)
CloseHandle(process_handle);
ReleaseMutex(service->service_entry->control_mutex); ReleaseMutex(service->service_entry->control_mutex);
unlock_service_database();
return err; return err;
} }