advapi32: Reimplement EnumServicesStatusW() on top of EnumServicesStatusExW().

Signed-off-by: Zebediah Figura <zfigura@codeweavers.com>
Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
Zebediah Figura 2020-04-27 22:35:34 -05:00 committed by Alexandre Julliard
parent c0977fcbec
commit d01f58a3f2
2 changed files with 57 additions and 75 deletions

View File

@ -1749,109 +1749,91 @@ done:
* EnumServicesStatusW [ADVAPI32.@] * EnumServicesStatusW [ADVAPI32.@]
*/ */
BOOL WINAPI BOOL WINAPI
EnumServicesStatusW( SC_HANDLE hmngr, DWORD type, DWORD state, LPENUM_SERVICE_STATUSW EnumServicesStatusW( SC_HANDLE manager, DWORD type, DWORD state, ENUM_SERVICE_STATUSW *status,
services, DWORD size, LPDWORD needed, LPDWORD returned, DWORD size, DWORD *ret_size, DWORD *ret_count, DWORD *resume_handle )
LPDWORD resume_handle )
{ {
DWORD err, i, offset, buflen, count, total_size = 0; ENUM_SERVICE_STATUS_PROCESSW *status_ex;
struct enum_service_status *entry; DWORD alloc_size, count, i;
const WCHAR *str; WCHAR *p;
BYTE *buf;
TRACE("%p 0x%x 0x%x %p %u %p %p %p\n", hmngr, type, state, services, size, needed, TRACE("%p 0x%x 0x%x %p %u %p %p %p\n", manager, type, state, status, size,
returned, resume_handle); ret_size, ret_count, resume_handle);
if (!hmngr) if (!manager)
{ {
SetLastError( ERROR_INVALID_HANDLE ); SetLastError( ERROR_INVALID_HANDLE );
return FALSE; return FALSE;
} }
if (!needed || !returned)
if (!ret_size || !ret_count)
{ {
SetLastError( ERROR_INVALID_ADDRESS ); SetLastError( ERROR_INVALID_PARAMETER );
return FALSE; return FALSE;
} }
/* make sure we pass a valid pointer */ *ret_size = 0;
buflen = max( size, sizeof(*services) ); *ret_count = 0;
if (!(buf = heap_alloc( buflen ))) if (!EnumServicesStatusExW( manager, SC_ENUM_PROCESS_INFO, type, state,
NULL, 0, &alloc_size, &count, resume_handle, NULL )
&& GetLastError() != ERROR_MORE_DATA)
return FALSE;
if (!(status_ex = heap_alloc( alloc_size )))
{ {
SetLastError( ERROR_NOT_ENOUGH_MEMORY ); SetLastError( ERROR_NOT_ENOUGH_MEMORY );
return FALSE; return FALSE;
} }
__TRY if (!EnumServicesStatusExW( manager, SC_ENUM_PROCESS_INFO, type, state, (BYTE *)status_ex,
alloc_size, &alloc_size, &count, resume_handle, NULL )
&& GetLastError() != ERROR_MORE_DATA)
{ {
err = svcctl_EnumServicesStatusW( hmngr, type, state, buf, buflen, needed, &count, resume_handle ); heap_free( status_ex );
}
__EXCEPT(rpc_filter)
{
err = map_exception_code( GetExceptionCode() );
}
__ENDTRY
*returned = 0;
if (err != ERROR_SUCCESS)
{
/* double the needed size to fit the potentially larger ENUM_SERVICE_STATUSW */
if (err == ERROR_MORE_DATA) *needed *= 2;
heap_free( buf );
SetLastError( err );
return FALSE; return FALSE;
} }
entry = (struct enum_service_status *)buf;
for (i = 0; i < count; i++) for (i = 0; i < count; i++)
{ {
total_size += sizeof(*services); *ret_size += sizeof(ENUM_SERVICE_STATUSW);
if (entry->service_name) *ret_size += (strlenW( status_ex[i].lpServiceName ) + 1) * sizeof(WCHAR);
{ if (status_ex[i].lpDisplayName)
str = (const WCHAR *)(buf + entry->service_name); *ret_size += (strlenW( status_ex[i].lpDisplayName ) + 1) * sizeof(WCHAR);
total_size += (strlenW( str ) + 1) * sizeof(WCHAR);
} if (*ret_size <= size)
if (entry->display_name) ++*ret_count;
{
str = (const WCHAR *)(buf + entry->display_name);
total_size += (strlenW( str ) + 1) * sizeof(WCHAR);
}
entry++;
} }
if (total_size > size) p = (WCHAR *)(status + *ret_count);
for (i = 0; i < *ret_count; i++)
{
strcpyW( p, status_ex[i].lpServiceName );
status[i].lpServiceName = (WCHAR *)p;
p += strlenW( p ) + 1;
if (status_ex[i].lpDisplayName)
{
strcpyW( p, status_ex[i].lpDisplayName );
status[i].lpDisplayName = (WCHAR *)p;
p += strlenW( p ) + 1;
}
else status[i].lpDisplayName = NULL;
status[i].ServiceStatus.dwServiceType = status_ex[i].ServiceStatusProcess.dwServiceType;
status[i].ServiceStatus.dwCurrentState = status_ex[i].ServiceStatusProcess.dwCurrentState;
status[i].ServiceStatus.dwControlsAccepted = status_ex[i].ServiceStatusProcess.dwControlsAccepted;
status[i].ServiceStatus.dwWin32ExitCode = status_ex[i].ServiceStatusProcess.dwWin32ExitCode;
status[i].ServiceStatus.dwServiceSpecificExitCode = status_ex[i].ServiceStatusProcess.dwServiceSpecificExitCode;
status[i].ServiceStatus.dwCheckPoint = status_ex[i].ServiceStatusProcess.dwCheckPoint;
status[i].ServiceStatus.dwWaitHint = status_ex[i].ServiceStatusProcess.dwWaitHint;
}
heap_free( status_ex );
if (*ret_size > size)
{ {
heap_free( buf );
*needed = total_size;
SetLastError( ERROR_MORE_DATA ); SetLastError( ERROR_MORE_DATA );
return FALSE; return FALSE;
} }
offset = count * sizeof(*services); *ret_size = 0;
entry = (struct enum_service_status *)buf;
for (i = 0; i < count; i++)
{
DWORD str_size;
str = (const WCHAR *)(buf + entry->service_name);
str_size = (strlenW( str ) + 1) * sizeof(WCHAR);
services[i].lpServiceName = (WCHAR *)((char *)services + offset);
memcpy( services[i].lpServiceName, str, str_size );
offset += str_size;
if (!entry->display_name) services[i].lpDisplayName = NULL;
else
{
str = (const WCHAR *)(buf + entry->display_name);
str_size = (strlenW( str ) + 1) * sizeof(WCHAR);
services[i].lpDisplayName = (WCHAR *)((char *)services + offset);
memcpy( services[i].lpDisplayName, str, str_size );
offset += str_size;
}
services[i].ServiceStatus = entry->service_status;
entry++;
}
heap_free( buf );
*needed = 0;
*returned = count;
return TRUE; return TRUE;
} }

View File

@ -1339,7 +1339,7 @@ static void test_enum_svc(void)
services, bufsize, &needed, &returned, &resume); services, bufsize, &needed, &returned, &resume);
ok(ret, "Expected success, got error %u\n", GetLastError()); ok(ret, "Expected success, got error %u\n", GetLastError());
ok(needed == 0, "Expected needed buffer to be 0 as we are done\n"); ok(needed == 0, "Expected needed buffer to be 0 as we are done\n");
ok(returned == missing, "Expected %u services to be returned\n", missing); todo_wine ok(returned == missing, "Expected %u services to be returned\n", missing);
ok(resume == 0, "Expected the resume handle to be 0\n"); ok(resume == 0, "Expected the resume handle to be 0\n");
HeapFree(GetProcessHeap(), 0, services); HeapFree(GetProcessHeap(), 0, services);