From: Stefan Leichter Subject: [1/2] wtsapi32: partial implementation of WTSEnumerateProcessesW (2nd try) Message-Id: <201401072350.29009.Stefan.Leichter@camline.com> Date: Tue, 7 Jan 2014 23:50:28 +0100 --- dlls/wtsapi32/wtsapi32.c | 71 ++++++++++++++++++++++++++++++++++++++++++++-- include/wtsapi32.h | 2 ++ 2 files changed, 70 insertions(+), 3 deletions(-) diff --git a/dlls/wtsapi32/wtsapi32.c b/dlls/wtsapi32/wtsapi32.c index 898b95d..96b2231 100644 --- a/dlls/wtsapi32/wtsapi32.c +++ b/dlls/wtsapi32/wtsapi32.c @@ -18,8 +18,11 @@ #include "config.h" #include #include +#include "ntstatus.h" +#define WIN32_NO_STATUS #include "windef.h" #include "winbase.h" +#include "winternl.h" #include "wtsapi32.h" #include "wine/debug.h" @@ -84,14 +87,76 @@ BOOL WINAPI WTSEnumerateProcessesA(HANDLE hServer, DWORD Reserved, DWORD Version BOOL WINAPI WTSEnumerateProcessesW(HANDLE hServer, DWORD Reserved, DWORD Version, PWTS_PROCESS_INFOW* ppProcessInfo, DWORD* pCount) { - FIXME("Stub %p 0x%08x 0x%08x %p %p\n", hServer, Reserved, Version, - ppProcessInfo, pCount); + SYSTEM_PROCESS_INFORMATION *spi; + ULONG size = 100 * sizeof(SYSTEM_PROCESS_INFORMATION); + void *buf = NULL; + NTSTATUS status; + DWORD i,lcount = 0, strsize = 0; + WCHAR *name; + WTS_PROCESS_INFOW *linfo; - if (!ppProcessInfo || !pCount) return FALSE; + if (!ppProcessInfo || !pCount || Reserved != 0 || Version != 1) + { + SetLastError(ERROR_INVALID_PARAMETER); + return FALSE; + } *pCount = 0; *ppProcessInfo = NULL; + if (hServer != WTS_CURRENT_SERVER_HANDLE) + { + FIXME("Only WTS_CURRENT_SERVER_HANDLE is impemented\n"); + return FALSE; + } + + do { + size *= 2; + HeapFree(GetProcessHeap(), 0, buf); + buf = HeapAlloc(GetProcessHeap(), 0, size); + if (!buf) + return FALSE; + + status = NtQuerySystemInformation(SystemProcessInformation, buf, size, NULL); + } while(status == STATUS_INFO_LENGTH_MISMATCH); + + if (status != STATUS_SUCCESS) + { + HeapFree(GetProcessHeap(), 0, buf); + SetLastError(RtlNtStatusToDosError(status)); + return FALSE; + } + + spi = buf; + + do { + lcount++; + strsize += spi->ProcessName.MaximumLength; + spi = (SYSTEM_PROCESS_INFORMATION *)(((PCHAR)spi) + spi->NextEntryOffset); + } while(spi->NextEntryOffset != 0); + + linfo = HeapAlloc(GetProcessHeap(), 0, lcount * sizeof(WTS_PROCESS_INFOW) + strsize * sizeof(WCHAR)); + if (!linfo) + return FALSE; + + name = (WCHAR*) &linfo[lcount]; + + for (i = 0, spi = buf; i < lcount; i++) + { + linfo[i].SessionId = 0; + linfo[i].ProcessId = HandleToUlong(spi->UniqueProcessId); + linfo[i].pProcessName = name; + linfo[i].pUserSid = NULL; + memcpy( name, spi->ProcessName.Buffer, spi->ProcessName.MaximumLength ); + + name += spi->ProcessName.MaximumLength; + spi = (SYSTEM_PROCESS_INFORMATION *)(((PCHAR)spi) + spi->NextEntryOffset); + } + *ppProcessInfo = linfo; + *pCount = lcount; + FIXME("pUserSid not filled\n"); + + HeapFree(GetProcessHeap(), 0, buf); return TRUE; } diff --git a/include/wtsapi32.h b/include/wtsapi32.h index f4bab56..d00a699 100644 --- a/include/wtsapi32.h +++ b/include/wtsapi32.h @@ -135,6 +135,8 @@ typedef struct _WTS_SERVER_INFOW DECL_WINELIB_TYPE_AW(WTS_SERVER_INFO) DECL_WINELIB_TYPE_AW(PWTS_SERVER_INFO) +#define WTS_CURRENT_SERVER_HANDLE ((HANDLE)NULL) + void WINAPI WTSCloseServer(HANDLE); BOOL WINAPI WTSConnectSessionA(ULONG, ULONG, PSTR, BOOL); BOOL WINAPI WTSConnectSessionW(ULONG, ULONG, PWSTR, BOOL);