Windows2003-3790/sdktools/psapi/driver.c
2020-09-30 16:53:55 +02:00

420 lines
10 KiB
C

#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <windows.h>
#include "psapi.h"
#include <stddef.h>
BOOL
FindDeviceDriver(
LPVOID ImageBase,
PRTL_PROCESS_MODULE_INFORMATION Module
)
/*++
Routine Description:
This function retrieves the full pathname of the executable file
from which the specified module was loaded. The function copies the
null-terminated filename into the buffer pointed to by the
lpFilename parameter.
Routine Description:
ImageBase - Identifies the driver whose executable file name is being
requested.
Return Value:
A return value of FALSE indicates an error and extended
error status is available using the GetLastError function.
Arguments:
--*/
{
NTSTATUS Status;
DWORD cbModuleInformation, cbModuleInformationNew, NumberOfModules;
PRTL_PROCESS_MODULES pModuleInformation;
DWORD i, ReturnedLength;
//
// Set the buffer length and pointer to a fixed size for the first pass
//
cbModuleInformation = sizeof (RTL_PROCESS_MODULES) + 0x400;
pModuleInformation = NULL;
while (1) {
pModuleInformation = LocalAlloc (LMEM_FIXED, cbModuleInformation);
if (pModuleInformation == NULL) {
SetLastError (ERROR_NO_SYSTEM_RESOURCES);
return FALSE;
}
Status = NtQuerySystemInformation (SystemModuleInformation,
pModuleInformation,
cbModuleInformation,
&ReturnedLength);
NumberOfModules = pModuleInformation->NumberOfModules;
if (NT_SUCCESS(Status)) {
break;
} else {
LocalFree (pModuleInformation);
if (Status == STATUS_INFO_LENGTH_MISMATCH) {
ASSERT (cbModuleInformation >= sizeof (RTL_PROCESS_MODULES));
cbModuleInformationNew = FIELD_OFFSET (RTL_PROCESS_MODULES, Modules) +
NumberOfModules * sizeof (RTL_PROCESS_MODULE_INFORMATION);
ASSERT (cbModuleInformationNew >= sizeof (RTL_PROCESS_MODULES));
ASSERT (cbModuleInformationNew > cbModuleInformation);
if (cbModuleInformationNew <= cbModuleInformation) {
SetLastError (RtlNtStatusToDosError (Status));
return FALSE;
}
cbModuleInformation = cbModuleInformationNew;
} else {
SetLastError (RtlNtStatusToDosError (Status));
return FALSE;
}
}
}
for (i = 0; i < NumberOfModules; i++) {
if (pModuleInformation->Modules[i].ImageBase == ImageBase) {
*Module = pModuleInformation->Modules[i];
LocalFree (pModuleInformation);
return TRUE;
}
}
LocalFree (pModuleInformation);
SetLastError (ERROR_INVALID_HANDLE);
return FALSE;
}
BOOL
WINAPI
EnumDeviceDrivers(
LPVOID *lpImageBase,
DWORD cb,
LPDWORD lpcbNeeded
)
{
NTSTATUS Status;
DWORD cbModuleInformation, cbModuleInformationNew, NumberOfModules;
PRTL_PROCESS_MODULES pModuleInformation;
DWORD cpvMax;
DWORD i, ReturnedLength;
//
// Set the buffer length and pointer to a fixed size for the first pass
//
cbModuleInformation = sizeof (RTL_PROCESS_MODULES) + 0x400;
pModuleInformation = NULL;
while (1) {
pModuleInformation = LocalAlloc (LMEM_FIXED, cbModuleInformation);
if (pModuleInformation == NULL) {
SetLastError (ERROR_NO_SYSTEM_RESOURCES);
return FALSE;
}
Status = NtQuerySystemInformation (SystemModuleInformation,
pModuleInformation,
cbModuleInformation,
&ReturnedLength);
NumberOfModules = pModuleInformation->NumberOfModules;
if (NT_SUCCESS(Status)) {
break;
} else {
LocalFree (pModuleInformation);
if (Status == STATUS_INFO_LENGTH_MISMATCH) {
ASSERT (cbModuleInformation >= sizeof (RTL_PROCESS_MODULES));
cbModuleInformationNew = FIELD_OFFSET (RTL_PROCESS_MODULES, Modules) +
NumberOfModules * sizeof (RTL_PROCESS_MODULE_INFORMATION);
ASSERT (cbModuleInformationNew >= sizeof (RTL_PROCESS_MODULES));
ASSERT (cbModuleInformationNew > cbModuleInformation);
if (cbModuleInformationNew <= cbModuleInformation) {
SetLastError (RtlNtStatusToDosError (Status));
return FALSE;
}
cbModuleInformation = cbModuleInformationNew;
} else {
SetLastError (RtlNtStatusToDosError (Status));
return FALSE;
}
}
}
cpvMax = cb / sizeof(LPVOID);
for (i = 0; i < NumberOfModules; i++) {
if (i == cpvMax) {
break;
}
try {
lpImageBase[i] = pModuleInformation->Modules[i].ImageBase;
} except (EXCEPTION_EXECUTE_HANDLER) {
LocalFree (pModuleInformation);
SetLastError (RtlNtStatusToDosError (GetExceptionCode ()));
return FALSE;
}
}
try {
*lpcbNeeded = NumberOfModules * sizeof(LPVOID);
} except (EXCEPTION_EXECUTE_HANDLER) {
LocalFree (pModuleInformation);
SetLastError (RtlNtStatusToDosError (GetExceptionCode ()));
return FALSE;
}
LocalFree (pModuleInformation);
return TRUE;
}
DWORD
WINAPI
GetDeviceDriverFileNameW(
LPVOID ImageBase,
LPWSTR lpFilename,
DWORD nSize
)
/*++
Routine Description:
This function retrieves the full pathname of the executable file
from which the specified module was loaded. The function copies the
null-terminated filename into the buffer pointed to by the
lpFilename parameter.
Routine Description:
ImageBase - Identifies the driver whose executable file name is being
requested.
lpFilename - Points to the buffer that is to receive the filename.
nSize - Specifies the maximum number of characters to copy. If the
filename is longer than the maximum number of characters
specified by the nSize parameter, it is truncated.
Return Value:
The return value specifies the actual length of the string copied to
the buffer. A return value of zero indicates an error and extended
error status is available using the GetLastError function.
Arguments:
--*/
{
LPSTR lpstr;
DWORD cch;
DWORD cchT;
lpstr = (LPSTR) LocalAlloc (LMEM_FIXED, nSize);
if (lpstr == NULL) {
return(0);
}
cchT = cch = GetDeviceDriverFileNameA (ImageBase, lpstr, nSize);
if (!cch) {
LocalFree((HLOCAL) lpstr);
return 0;
}
if (cchT < nSize) {
//
// Include NULL terminator
//
cchT++;
}
if (!MultiByteToWideChar(CP_ACP, 0, lpstr, cchT, lpFilename, nSize)) {
cch = 0;
}
LocalFree((HLOCAL) lpstr);
return(cch);
}
DWORD
WINAPI
GetDeviceDriverFileNameA(
LPVOID ImageBase,
LPSTR lpFilename,
DWORD nSize
)
{
RTL_PROCESS_MODULE_INFORMATION Module;
DWORD cchT;
DWORD cch;
if (!FindDeviceDriver(ImageBase, &Module)) {
return(0);
}
cch = cchT = (DWORD) (strlen(Module.FullPathName) + 1);
if ( nSize < cch ) {
cch = nSize;
}
CopyMemory(lpFilename, Module.FullPathName, cch);
if (cch == cchT) {
cch--;
}
return(cch);
}
DWORD
WINAPI
GetDeviceDriverBaseNameW(
LPVOID ImageBase,
LPWSTR lpFilename,
DWORD nSize
)
/*++
Routine Description:
This function retrieves the full pathname of the executable file
from which the specified module was loaded. The function copies the
null-terminated filename into the buffer pointed to by the
lpFilename parameter.
Routine Description:
ImageBase - Identifies the driver whose executable file name is being
requested.
lpFilename - Points to the buffer that is to receive the filename.
nSize - Specifies the maximum number of characters to copy. If the
filename is longer than the maximum number of characters
specified by the nSize parameter, it is truncated.
Return Value:
The return value specifies the actual length of the string copied to
the buffer. A return value of zero indicates an error and extended
error status is available using the GetLastError function.
Arguments:
--*/
{
LPSTR lpstr;
DWORD cch;
DWORD cchT;
lpstr = (LPSTR) LocalAlloc(LMEM_FIXED, nSize);
if (lpstr == NULL) {
return(0);
}
cchT = cch = GetDeviceDriverBaseNameA(ImageBase, lpstr, nSize);
if (!cch) {
LocalFree((HLOCAL) lpstr);
return 0;
}
if (cchT < nSize) {
//
// Include NULL terminator
//
cchT++;
}
if (!MultiByteToWideChar(CP_ACP, 0, lpstr, cchT, lpFilename, nSize)) {
cch = 0;
}
LocalFree((HLOCAL) lpstr);
return(cch);
}
DWORD
WINAPI
GetDeviceDriverBaseNameA(
LPVOID ImageBase,
LPSTR lpFilename,
DWORD nSize
)
{
RTL_PROCESS_MODULE_INFORMATION Module;
DWORD cchT;
DWORD cch;
if (!FindDeviceDriver(ImageBase, &Module)) {
return(0);
}
cch = cchT = (DWORD) (strlen(Module.FullPathName + Module.OffsetToFileName) + 1);
if ( nSize < cch ) {
cch = nSize;
}
CopyMemory(lpFilename, Module.FullPathName + Module.OffsetToFileName, cch);
if (cch == cchT) {
cch--;
}
return(cch);
}