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:
parent
c0977fcbec
commit
d01f58a3f2
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue