Windows-Server-2003/base/ntsetup/hwlog/hwlog.c

1976 lines
55 KiB
C

/*++
Copyright (c) Microsoft Corporation
Module Name:
ntsetup\hwlog\hwlog.c
Abstract:
Logging some aspects of the hardware configuration to winnt32.log / setupact.log.
Esp. disk drive by connection, and map drive letters to disk drives.
Author:
Jay Krell (JayKrell) April 2001, May 2001
Revision History:
Environment:
winnt32.dll -- Win9x ANSI (down to Win95gold) or NT Unicode
libcmt statically linked in, _tcs* ok
actually only built for Unicode/NT, and does nothing
if run on less than Windows 2000
setup.exe -newsetup -- guimode setup
--*/
/* Platform notes
Win95:
apparently no setupapi.dll (redist)
apparently no cfgmgr32.dll
no kernel32.dll::GetVolumeNameForVolumeMountPoint
NT 3.1
no kernel32.dll::GetVolumeNameForVolumeMountPoint
NT 3.51
apparently no setupapi.dll
apparently no cfgmgr32.dll
no kernel32.dll::GetVolumeNameForVolumeMountPoint
NT4
no kernel32.dll::GetVolumeNameForVolumeMountPoint
setupapi.dll
has SetupDiGetClassDevs
does not have SetupDiGetClassDevsEx
does not have SetupDiEnumDeviceInterfaces
does not have SetupDiGetDeviceInterfaceDetail
cfgmgr32.dll
has the functions we call
Win2000, WinXp:
has all the functions we call
Win98, Win98se:
no kernel32.dll::GetVolumeNameForVolumeMountPoint
setupapi.dll, has everything we use
has SetupDiGetClassDevs
has SetupDiGetClassDevsEx
has SetupDiEnumDeviceInterfaces
has SetupDiGetDeviceInterfaceDetail
cfgmgr32.dll, has everything we use
has CM_Get_Parent_Ex
has CM_Connect_MachineA
has CM_Get_DevNode_Registry_Property_ExA
existing versions of
winnt32a.dll
not statically dependent on setupapi.dll
not statically dependent on cfgmgr32.dll
winnt32u.dll
not statically dependent on setupapi.dll
statically dependent on cfgmgr32.dll, all functions exported on NT4 and Win98
CM_Get_Device_ID_List_SizeW
CM_Get_Device_ID_ListW
CM_Get_DevNode_Registry_PropertyW
CM_Locate_DevNodeW
conclusions
works on Win2000 and WinXp
maybe some of it works on Win98, Win98se, Win9me
maybe can be changed slightly to work on NT4
cannot easily work on NT 3 or Win95
but GetVolumeNameForVolumeMountPoint and the DeviceIoControl might prevent it
dynamically link it "all"
*/
#define STANDALONE 0
#define DYNLINK 1
#if STANDALONE
#define UNICODE
#define _UNICODE
#if !defined(_WIN32_WINNT)
#define _WIN32_WINNT 0x0501
#endif
// from windows.h, but we want this before nt.h
#if !defined(_X86_) && !defined(_IA64_) && !defined(_AMD64_) && defined(_M_IX86)
#define _X86_
#endif
#if !defined(_X86_) && !defined(_IA64_) && !defined(_AMD64_) && defined(_M_AMD64)
#define _AMD64_
#endif
#if !defined(_IA64_) && !defined(_X86_) && !defined(_M_IX86) && !defined(_AMD64_) && defined(_M_IA64)
#define _IA64_
#endif
#include "io.h"
#if !defined(_WIN64)
typedef unsigned long ULONG_PTR; // vc6 compatibility
typedef unsigned long DWORD_PTR; // vc6 compatibility
#endif
#endif
#include <stdio.h>
#include "nt.h"
#include "ntrtl.h"
#include "nturtl.h"
#include "windows.h"
#include "cfgmgr32.h"
#include "objbase.h"
#include "initguid.h"
#include "devguid.h"
#include "setupapi.h"
#include "winioctl.h"
#include <stdlib.h>
#include "tchar.h"
#include <stdarg.h>
#include <stdlib.h>
#include "hwlog.h"
typedef CONST VOID* PCVOID;
struct _SP_LINKAGE;
typedef struct _SP_LINKAGE SP_LINKAGE, *PSP_LINKAGE;
typedef CONST SP_LINKAGE* PCSP_LINKAGE;
typedef struct _SP_MACHINE {
PCTSTR Name; // for setupapi.dll functions
HMACHINE Handle; // for cfgmgr32.dll functions
} SP_MACHINE, *PSP_MACHINE;
typedef CONST SP_MACHINE* PCSP_MACHINE;
typedef struct _SP_LOG_HARDWARE {
SP_MACHINE Machine;
HANDLE LogFile;
#if !DYNLINK
CONST
#endif
SP_LINKAGE* Linkage;
BOOL (WINAPI* SetupLogError)(PCTSTR MessageString, LogSeverity);
BOOL (__cdecl * SetuplogError)(
IN LogSeverity Severity,
IN LPCTSTR MessageString,
IN UINT MessageId, OPTIONAL
...
) OPTIONAL;
} SP_LOG_HARDWARE, *PSP_LOG_HARDWARE;
typedef CONST SP_LOG_HARDWARE* PCSP_LOG_HARDWARE;
#if STANDALONE || defined(UNICODE)
#define QUASH_SIMPLE_PHYSICAL_DEVICE_OBJECT_NAMES 1
#define INDENT_FACTOR 2
#define UNAVAILABLE_VERBOSE 1
#if 0 /* this is more like a small tree, one device per line, indenting */
#define ONE_PROPERTY_PER_LINE 0
#define INDENT_CHILDREN 1
#define NUMBER_CHILDREN 0
#define DESCRIPTION_DASH_PHYSICAL_DEVICE_OBJECT 0
#else
#define ONE_PROPERTY_PER_LINE 1
#define INDENT_CHILDREN 0
#define NUMBER_CHILDREN 1
#define DESCRIPTION_DASH_PHYSICAL_DEVICE_OBJECT 1
#endif
#define ONE_DEVICE_PER_LINE (!ONE_PROPERTY_PER_LINE)
#if !defined(DBG)
#define DBG 1
#endif
#define NUMBER_OF(x) (sizeof(x)/sizeof((x)[0]))
#if !defined(ISNT)
#if STANDALONE || !defined(UNICODE)
#define ISNT() ((GetVersion() & 0x80000000) == 0)
#else
#define ISNT() TRUE
#endif
#endif
#define SP_FILE_SHARE_DELETE() (ISNT() ? FILE_SHARE_DELETE : 0)
// RTL_* from ntdef.h
#define FIELD_TYPE(type, field) (((type*)0)->field)
#define BITS_OF_FIELD(type, field) (BITS_OF(FIELD_TYPE(type, field)))
#define BITS_OF(sizeOfArg) (sizeof(sizeOfArg) * 8)
#define xPASTE(x,y) x##y
#define PASTE(x,y) xPASTE(x,y)
typedef struct _SP_LINKAGE
{
#if DYNLINK
#define U(x) union { PCSTR PASTE(Name, __LINE__); x; }
#else
#define U(x) x
#endif
PCSTR Kernel32Dll;
U(BOOL (WINAPI* GetVolumeNameForVolumeMountPoint)(PCTSTR lpszVolumeMountPoint, PTSTR lpszVolumeName, DWORD cchBufferLength));
PCSTR Setupapidll;
U(BOOL (WINAPI* SetupDiEnumDeviceInterfaces)(HDEVINFO DeviceInfoSet, PSP_DEVINFO_DATA DeviceInfoData, CONST GUID* InterfaceClassGuid, DWORD MemberIndex, PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData));
U(HDEVINFO (WINAPI* SetupDiGetClassDevsEx)(CONST GUID* ClassGuid, PCTSTR Enumerator, HWND hwndParent, DWORD Flags, HDEVINFO DeviceInfoSet, PCTSTR MachineName, PVOID Reserved));
U(BOOL (WINAPI* SetupDiGetDeviceInterfaceDetail)(HDEVINFO DeviceInfoSet, PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData, PSP_DEVICE_INTERFACE_DETAIL_DATA DeviceInterfaceDetailData, DWORD DeviceInterfaceDetailDataSize, PDWORD RequiredSize, PSP_DEVINFO_DATA DeviceInfoData));
U(CONFIGRET (WINAPI* CM_Get_Parent_Ex)(OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags, IN HMACHINE hMachine));
U(CONFIGRET (WINAPI* CM_Connect_Machine)(IN PCTSTR UNCServerName, OUT PHMACHINE phMachine));
U(CONFIGRET (WINAPI* CM_Get_DevNode_Registry_Property_Ex)(
IN DEVINST dnDevInst, IN ULONG ulProperty,
OUT PULONG pulRegDataType OPTIONAL, OUT PVOID Buffer OPTIONAL,
IN OUT PULONG pulLength, IN ULONG ulFlags, IN HMACHINE hMachine));
#undef U
} SP_LINKAGE;
#if defined(UNICODE)
#define T "W"
#else
#define T "A"
#endif
#if !DYNLINK
CONST
#endif
static SP_LINKAGE SpLinkage =
{
#if DYNLINK
//
// note: lowercase => .dll name
//
"kernel32.dll",
"GetVolumeNameForVolumeMountPoint" T,
"setupapi.dll",
"SetupDiEnumDeviceInterfaces",
"SetupDiGetClassDevsEx" T,
"SetupDiGetDeviceInterfaceDetail" T,
"CM_Get_Parent_Ex",
"CM_Connect_Machine" T,
"CM_Get_DevNode_Registry_Property_Ex" T
#undef T
#else
NULL, // kernel32
GetVolumeNameForVolumeMountPoint,
NULL, // setupapi
SetupDiEnumDeviceInterfaces,
SetupDiGetClassDevsEx,
SetupDiGetDeviceInterfaceDetail,
CM_Get_Parent_Ex,
CM_Connect_Machine,
CM_Get_DevNode_Registry_Property_Ex
#endif
};
BOOL
SpDoDynlink(
PSP_LOG_HARDWARE This
)
{
#if DYNLINK
SIZE_T i;
FARPROC* rgproc = (FARPROC*)This->Linkage;
PCSTR* rgpsz = (PCSTR*)This->Linkage;
HMODULE DllHandle;
for (i = 0 ; i != sizeof(SpLinkage)/sizeof(PVOID) ; ++i)
{
if (islower(rgpsz[i][0]))
{
if ((DllHandle = LoadLibraryA(rgpsz[i])) == NULL)
return FALSE;
}
else if ((rgproc[i] = GetProcAddress(DllHandle, rgpsz[i])) == NULL)
{
return FALSE;
}
}
#endif
return TRUE;
}
PCTSTR
SpGetSpacesString(
SIZE_T n
);
PCTSTR
SpGetDashesString(
SIZE_T n
);
PVOLUME_DISK_EXTENTS
SpGetVolumeDiskExtents(
HANDLE DeviceFileHandle
);
//
// need a downlevel static .lib version of ntdll.dll..
//
typedef struct SP_STRING {
PTSTR Chars;
SIZE_T Length;
SIZE_T MaximumLength;
} SP_STRING, *PSP_STRING;
typedef CONST SP_STRING* PCSP_STRING;
#define SpStringLength(s) ((s)->Length)
#define SpInitString(s, t) \
((s)->MaximumLength = sizeof((s)->Chars[0]) + ((s)->Length = lstrlen((s)->Chars = (PTSTR)t)))
VOID
SpHwDebugLog(
PSP_LOG_HARDWARE This,
PCTSTR Format,
...
);
VOID SpStringAppendFormatVa(PSP_STRING Buffer, PCSP_STRING Format, va_list va)
{
if (Buffer->MaximumLength > Buffer->Length + 1)
{
_vsntprintf(
Buffer->Chars + Buffer->Length,
Buffer->MaximumLength - Buffer->Length,
Format->Chars,
va
);
}
Buffer->Chars[Buffer->MaximumLength - 1] = 0;
}
VOID SpStringAppendFormat(PSP_STRING Buffer, PCSP_STRING Format, ...)
{
va_list va;
va_start(va, Format);
SpStringAppendFormatVa(Buffer, Format, va);
va_end(va);
}
VOID SpStringAppend(PSP_STRING s, PCSP_STRING t)
{
SP_STRING Format;
Format.Chars = TEXT("%s");
Format.Length = 2;
SpStringAppendFormat(s, &Format, t);
}
VOID SpStringCopy(PSP_STRING s, PCSP_STRING t)
{
s->Chars[0] = 0;
SpStringAppend(s, t);
}
VOID SpStringFormat(PSP_STRING s, PCSP_STRING Format, ...)
{
va_list va;
va_start(va, Format);
s->Chars[0] = 0;
SpStringAppendFormatVa(s, Format, va);
va_end(va);
}
PVOID SpMalloc(SIZE_T n)
{
return HeapAlloc(GetProcessHeap(), 0, n);
}
VOID SpFree(PVOID p)
{
HeapFree(GetProcessHeap(), 0, p);
}
PVOID SpRealloc(PVOID p, SIZE_T n)
{
return HeapReAlloc(GetProcessHeap(), 0, p, n);
}
VOID SpStringFree(PTSTR s) { SpFree(s); }
VOID SpRemoveTrailingChars(PSP_STRING s, PCTSTR ch)
{
while (s->Length != 0
&& (s->Chars[s->Length - 1] == ch[0] || s->Chars[s->Length - 1] == ch[1])
&& !(s->Chars[s->Length - 1] = 0)
&& (s->Length -= 1)
)
{
// nothing
}
}
VOID SpEnsureTrailingChar(PSP_STRING s, TCHAR ch)
{
if (s->Length == 0
|| s->Chars[s->Length - 1] != ch)
{
s->Length += 1;
s->Chars[s->Length - 1] = ch;
s->Chars[s->Length] = 0;
}
}
//
// for now, let's hope that device numbers stay in the range 0-63
//
typedef struct SP_DEVICE_NUMBERS {
ULONGLONG Bitset;
} SP_DEVICE_NUMBERS, *PSP_DEVICE_NUMBERS;
typedef CONST SP_DEVICE_NUMBERS* PCSP_DEVICE_NUMBERS;
typedef struct SP_VOLUME
{
PTSTR GuidVolumeNamePointer; // if this is null, use GuidVolumeNameBuffer
TCHAR GuidVolumeNameBuffer[64]; // \\?\{guid}
#define SP_VOLUME_GET_NAME(v) (((v)->GuidVolumeNamePointer != NULL) ? (v)->GuidVolumeNamePointer : (v)->GuidVolumeNameBuffer)
//
// DiskNumbers are gotten via DeviceIoControl(STORAGE_DEVICE_NUMBER) (Win2K)
// and DeviceIoControl(IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS) (Whistler)
//
ULONG DeviceType;
SP_DEVICE_NUMBERS DeviceNumbers;
TCHAR DriveLetter;
} SP_VOLUME, *PSP_VOLUME;
typedef CONST SP_VOLUME* PCSP_VOLUME;
//
// most systems are limited to 24 volumes, C-Z, unless they use
// mount points, and setup is unlikely to be affected by those volumes
//
typedef struct _SP_VOLUMES {
SP_VOLUME Entries[24];
} SP_VOLUMES, *PSP_VOLUMES;
typedef CONST SP_VOLUMES* PCSP_VOLUMES;
#define SP_PROPERTY_QUIET_UNAVAILABLE (0x00000001)
typedef struct _SP_DEVICE_PROPERTY_CONST {
//ULONG SetupapiInteger;
ULONG ConfigManagerInteger;
PCTSTR Name;
ULONG Flags;
} SP_DEVICE_PROPERTY_CONST, *PSP_DEVICE_PROPERTY_CONST;
typedef CONST SP_DEVICE_PROPERTY_CONST* PCSP_DEVICE_PROPERTY_CONST;
#if ONE_DEVICE_PER_LINE
//CONST static TCHAR FriendlyNameString[] = TEXT("FriendlyName");
//CONST static TCHAR DescriptionString[] = TEXT("Description");
CONST static TCHAR PhysicalDeviceObjectNameString[] = TEXT("");
CONST static TCHAR HardwareIdString[] = TEXT("");
CONST static TCHAR LowerFiltersString[] = TEXT("");
CONST static TCHAR UpperFiltersString[] = TEXT("");
//CONST static TCHAR FlagsString[] = TEXT("");
CONST static TCHAR LocationInformationString[] = TEXT("");
#else
//CONST static TCHAR FriendlyNameString[] = TEXT("FriendlyName");
//CONST static TCHAR DescriptionString[] = TEXT("Description");
CONST static TCHAR PhysicalDeviceObjectNameString[] = TEXT("PhysicalDeviceObjectName");
CONST static TCHAR HardwareIdString[] = TEXT("HardwareId");
CONST static TCHAR LowerFiltersString[] = TEXT("LowerFilters");
CONST static TCHAR UpperFiltersString[] = TEXT("UpperFilters");
//CONST static TCHAR FlagsString[] = TEXT("Flags");
CONST static TCHAR LocationInformationString[] = TEXT("Location");
#endif
#define SETUPAPI_PROPERTY_NUMBER(x) /* nothing */
#define DESCRIPTION 0
#define PHYSICAL_DEVICE_OBJECT 1
#define FRIENDLY_NAME 2
#define HARDWARE_ID 3
#define FIRST_GENERIC_PROPERTY 4
CONST static SP_DEVICE_PROPERTY_CONST DevicePropertyMetaInfo[] =
{
{ SETUPAPI_PROPERTY_NUMBER(SPDRP_DEVICEDESC) CM_DRP_DEVICEDESC, TEXT(""), SP_PROPERTY_QUIET_UNAVAILABLE },
{ SETUPAPI_PROPERTY_NUMBER(SPDRP_PHYSICAL_DEVICE_OBJECT_NAME) CM_DRP_PHYSICAL_DEVICE_OBJECT_NAME, PhysicalDeviceObjectNameString },
{ SETUPAPI_PROPERTY_NUMBER(SPDRP_FRIENDLYNAME) CM_DRP_FRIENDLYNAME, TEXT(""), SP_PROPERTY_QUIET_UNAVAILABLE },
{ SETUPAPI_PROPERTY_NUMBER(SPDRP_HARDWAREID) CM_DRP_HARDWAREID, HardwareIdString },
{ SETUPAPI_PROPERTY_NUMBER(x) CM_DRP_LOCATION_INFORMATION, LocationInformationString, SP_PROPERTY_QUIET_UNAVAILABLE },
{ SETUPAPI_PROPERTY_NUMBER(SPDRP_LOWERFILTERS) CM_DRP_LOWERFILTERS, LowerFiltersString, SP_PROPERTY_QUIET_UNAVAILABLE },
{ SETUPAPI_PROPERTY_NUMBER(SPDRP_UPPERFILTERS) CM_DRP_UPPERFILTERS, UpperFiltersString, SP_PROPERTY_QUIET_UNAVAILABLE },
//{ SETUPAPI_PROPERTY_NUMBER(x) CM_DRP_CONFIGFLAGS, FlagsString },
//{ SETUPAPI_PROPERTY_NUMBER(x) CM_DRP_CAPABILITIES, TEXT("Capabilities") },
//{ SETUPAPI_PROPERTY_NUMBER(x) CM_DRP_UI_NUMBER, TEXT("UI Number") },
//{ SETUPAPI_PROPERTY_NUMBER(x) CM_DRP_CHARACTERISTICS, TEXT("Characteristics") },
//{ SETUPAPI_PROPERTY_NUMBER(x) CM_DRP_ADDRESS, TEXT("Address") },
};
typedef struct _SP_DEVICE_PROPERTY {
PCSP_DEVICE_PROPERTY_CONST Const;
TCHAR Value[256];
ULONG Type;
} SP_DEVICE_PROPERTY, *PSP_DEVICE_PROPERTY;
typedef CONST SP_DEVICE_PROPERTY* PCSP_DEVICE_PROPERTY;
typedef struct _SP_DEVICE_CLASS {
CONST GUID* Guid;
ULONG IsInterface;
//PCTSTR Name;
} SP_DEVICE_CLASS, *PSP_DEVICE_CLASS;
typedef CONST SP_DEVICE_CLASS* PCSP_DEVICE_CLASS;
/*
CONST static TCHAR VolumesString[] = TEXT("Volumes");
CONST static TCHAR DisksString[] = TEXT("Disks");
CONST static TCHAR CDROMsString[] = TEXT("CDROMs");
CONST static TCHAR PartitionsString[] = TEXT("Partitions");
*/
CONST static SP_DEVICE_CLASS DeviceClasses[] =
{
{ &GUID_DEVINTERFACE_CDROM, DIGCF_DEVICEINTERFACE, /*CDROMsString*/ },
{ &GUID_DEVINTERFACE_DISK, DIGCF_DEVICEINTERFACE, /*DisksString*/ },
{ &GUID_DEVINTERFACE_PARTITION, DIGCF_DEVICEINTERFACE, /*PartitionsString*/ },
// The information this adds is not very useful.
//{ &GUID_DEVINTERFACE_VOLUME, DIGCF_DEVICEINTERFACE, /*VolumesString*/ },
};
typedef struct _SP_DEVICE {
PCSP_DEVICE_CLASS Class;
ULONG DevInst;
ULONG DeviceType;
SP_DEVICE_NUMBERS DeviceNumbers;
//BOOL IsLeaf;
SIZE_T NumberOfParents;
ULONG ParentDevInsts[MAX_DEVICE_ID_LEN];
TCHAR DevicePath[MAX_PATH];
TCHAR GuidVolumePath[64];
//SP_DEVICE_PROPERTY Properties[NUMBER_OF(DevicePropertyMetaInfo)];
} SP_DEVICE, *PSP_DEVICE;
typedef CONST SP_DEVICE* PCSP_DEVICE;
#define SP_IS_PATH_SEPERATOR(ch) ((ch) == '\\' || (ch) == '/')
SIZE_T
SpStringLengthWithoutTrailingPathSeperators(
PCTSTR s
)
{
SIZE_T Length;
if (s == NULL || *s == 0)
return 0;
Length = _tcslen(s);
s += Length - 1;
while (SP_IS_PATH_SEPERATOR(s[Length]))
Length -= 1;
return Length;
}
int __cdecl SpCompareVolume(CONST VOID* v1, CONST VOID* v2)
{
CONST PCSP_VOLUME p1 = (PCSP_VOLUME)v1;
CONST PCSP_VOLUME p2 = (PCSP_VOLUME)v2;
CONST PCTSTR s1 = (p1 != NULL) ? SP_VOLUME_GET_NAME(p1) : TEXT("");
CONST SIZE_T len1 = SpStringLengthWithoutTrailingPathSeperators(s1);
CONST PCTSTR s2 = (p1 != NULL) ? SP_VOLUME_GET_NAME(p2) : TEXT("");
CONST SIZE_T len2 = SpStringLengthWithoutTrailingPathSeperators(s2);
return _tcsicmp(s1, s2);
}
VOID SpSortVolumes(PSP_VOLUMES Volumes)
{
qsort(
Volumes->Entries,
NUMBER_OF(Volumes->Entries),
sizeof(Volumes->Entries[0]),
SpCompareVolume
);
}
PCSP_VOLUME
SpFindVolume(
PSP_VOLUMES Volumes,
PCTSTR VolumeGuidPath
)
{
CONST SP_VOLUME VolumeKey = { (PTSTR)VolumeGuidPath };
PCSP_VOLUME VolumeFound =
(PCSP_VOLUME)
bsearch(
&VolumeKey,
Volumes->Entries,
NUMBER_OF(Volumes->Entries),
sizeof(Volumes->Entries[0]),
SpCompareVolume
);
return VolumeFound;
}
#if 0
void
SpCovertRNToN(
PTSTR Buffer
)
{
PTSTR p;
PTSTR q;
for (p = q = Buffer ; *p != 0 && *(p + 1) != 0 ; )
{
if (*p == '\r' && *(p + 1) == '\n')
{
*q++ = '\n';
p += 2;
}
else
{
*q++ = *p++;
}
}
if (*p != 0)
*q++ = *p++;
*q++ = 0;
}
void
SpCovertNToRN(
PTSTR Buffer
)
{
// determine size
// heap alloc
// convert
}
#endif
VOID
SpHwDebugLog(
PSP_LOG_HARDWARE This,
PCTSTR Format,
...
)
{
va_list va;
TCHAR BufferT[500];
#ifdef UNICODE
CHAR BufferA[500];
#endif
SIZE_T Length;
BOOLEAN Newline = FALSE;
BufferT[0] = 0;
va_start(va, Format);
FormatMessage(
FORMAT_MESSAGE_FROM_STRING,
Format,
0,
0,
BufferT,
NUMBER_OF(BufferT),
&va
);
BufferT[NUMBER_OF(BufferT) - 1] = 0;
Length = lstrlen(BufferT);
if (Length != 0)
{
#if 0
Newline = (BufferT[Length - 1] == '\n' || BufferT[Length - 1] == '\r');
if (Newline)
{
while (Length != 0 && (BufferT[Length - 1] == '\n' || BufferT[Length - 1] == '\r'))
Length -= 1;
if (Length != 0)
BufferT[Length] = 0;
else
Newline = FALSE;
}
#endif
if (Newline)
lstrcat(BufferT, TEXT("\r\n"));
if (This->LogFile != NULL)
{
DWORD BytesWritten;
if (Newline)
lstrcat(BufferT, TEXT("\r\n"));
#ifdef UNICODE
WideCharToMultiByte(
CP_ACP,
0,
BufferT,
-1,
BufferA,
sizeof(BufferA) - 1,
NULL,
NULL
);
WriteFile(This->LogFile, &BufferA, lstrlenA(BufferA), &BytesWritten, NULL);
#else
WriteFile(This->LogFile, &BufferT, lstrlen(BufferT), &BytesWritten, NULL);
#endif
}
if (This->SetupLogError != NULL)
{
This->SetupLogError(BufferT, LogSevInformation);
}
if (This->SetuplogError != NULL)
{
This->SetuplogError(
LogSevInformation,
TEXT("%1"),
0,
BufferT,
(PVOID)NULL,
(PVOID)NULL
);
}
}
va_end(va);
}
#define SpHwLog SpHwDebugLog
#if 0
VOID PrependString(PTSTR s, PCTSTR t)
{
SIZE_T slen = _tcslen(s);
SIZE_T tlen = _tcslen(t);
MoveMemory(s + tlen, s, (slen + 1) * sizeof(*s));
MoveMemory(s, t, tlen);
}
#endif
#define SP_CLOSE_HANDLE(h) \
do { if ((h) != NULL && (h) != INVALID_HANDLE_VALUE) { CloseHandle(h); h = INVALID_HANDLE_VALUE; } } while(0)
#define SP_FREE(p) \
do { if ((p) != NULL) { SpFree(p); (p) = NULL; } } while(0)
PVOLUME_DISK_EXTENTS
SpGetVolumeDiskExtents(
HANDLE DeviceFileHandle
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
//
// This pattern is iffy, but it is used elsewhere, and the implementation of
// IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS is iffy too, and not quite as documented.
// See drivers\vsm\vsmio\nt\voldev.c.
//
struct {
VOLUME_DISK_EXTENTS VolumeDiskExtents;
DISK_EXTENT DiskExtents[4];
} StackDiskExtents;
PVOLUME_DISK_EXTENTS HeapDiskExtents = NULL;
PVOLUME_DISK_EXTENTS DiskExtents = NULL;
PVOLUME_DISK_EXTENTS ResultDiskExtents = NULL;
DWORD BytesReturned;
BOOL Success = FALSE;
DWORD Size;
DiskExtents = &StackDiskExtents.VolumeDiskExtents;
Size = sizeof(StackDiskExtents);
//
// loop in case it is changing
//
while (!Success)
{
Success =
DeviceIoControl(
DeviceFileHandle,
IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS,
NULL,
0,
DiskExtents,
Size,
&BytesReturned,
NULL
);
if (!Success) {
if (GetLastError() != ERROR_MORE_DATA)
goto Exit;
Size = sizeof(VOLUME_DISK_EXTENTS) + DiskExtents->NumberOfDiskExtents * sizeof(DISK_EXTENT);
SP_FREE(HeapDiskExtents);
HeapDiskExtents = (PVOLUME_DISK_EXTENTS)SpMalloc(Size);
if (HeapDiskExtents == NULL) {
SetLastError(ERROR_OUTOFMEMORY);
goto Exit;
}
DiskExtents = HeapDiskExtents;
} else {
if (DiskExtents->NumberOfDiskExtents == 0) {
/* nothing */
}
else if (DiskExtents == HeapDiskExtents) {
ResultDiskExtents = HeapDiskExtents;
HeapDiskExtents = NULL;
} else if (DiskExtents == &StackDiskExtents.VolumeDiskExtents) {
// VOLUME_DISK_EXTENTS includes an array of one DISK_EXTENT at the end.
ASSERT(BytesReturned == (sizeof(VOLUME_DISK_EXTENTS) + (DiskExtents->NumberOfDiskExtents - 1) * sizeof(DISK_EXTENT)));
ResultDiskExtents = (PVOLUME_DISK_EXTENTS)SpMalloc(BytesReturned);
if (ResultDiskExtents == NULL) {
SetLastError(ERROR_OUTOFMEMORY);
goto Exit;
}
ASSERT(BytesReturned <= sizeof(StackDiskExtents));
CopyMemory(ResultDiskExtents, &StackDiskExtents, BytesReturned);
} else {
ASSERT(FALSE && "DiskExtents != HeapDiskExtents, StackDiskExtents");
}
}
}
Exit:
SP_FREE(HeapDiskExtents);
return ResultDiskExtents;
}
PTSTR
SpDeviceTypeToString(
ULONG i,
PTSTR s
)
{
//
// this is a partial list from public\ddk\inc\devioctl.h
//
PCTSTR t = NULL;
s[0] = 0;
switch (i)
{
case FILE_DEVICE_CD_ROM: t = TEXT("CDROM"); break;
case FILE_DEVICE_CD_ROM_FILE_SYSTEM: t = TEXT("CDROM File System"); break;
case FILE_DEVICE_CONTROLLER: t = TEXT("Device Controller"); break;
case FILE_DEVICE_DFS: t = TEXT("Distributed File System"); break;
case FILE_DEVICE_DISK: t = TEXT("Disk"); break;
case FILE_DEVICE_DISK_FILE_SYSTEM: t = TEXT("Disk File System"); break;
case FILE_DEVICE_FILE_SYSTEM: t = TEXT("File System"); break;
case FILE_DEVICE_NETWORK_FILE_SYSTEM: t = TEXT("Network File System"); break;
case FILE_DEVICE_TAPE: t = TEXT("Tape"); break;
case FILE_DEVICE_TAPE_FILE_SYSTEM: t = TEXT("Tape File System"); break;
case FILE_DEVICE_VIRTUAL_DISK: t = TEXT("Virtual Disk"); break;
case FILE_DEVICE_NETWORK_REDIRECTOR: t = TEXT("Network Redirector"); break;
case FILE_DEVICE_MASS_STORAGE: t = TEXT("Mass Storage"); break;
case FILE_DEVICE_SMB: t = TEXT("SMB"); break;
case FILE_DEVICE_CHANGER: t = TEXT("Changer"); break;
case FILE_DEVICE_ACPI: t = TEXT("ACPI"); break;
case FILE_DEVICE_DFS_FILE_SYSTEM: t = TEXT("DFS File System"); break;
case FILE_DEVICE_DFS_VOLUME: t = TEXT("DFS Volume"); break;
default:
_stprintf(s, TEXT("Other (%ld)"), i);
break;
}
if (t != NULL)
_tcscpy(s, t);
return s;
}
VOID
SpGetDeviceNumbersAndType(
PSP_LOG_HARDWARE This,
PCTSTR DevicePath,
PSP_DEVICE_NUMBERS DeviceNumbers,
PULONG DeviceType
)
{
PVOLUME_DISK_EXTENTS VolumeDiskExtents = NULL;
STORAGE_DEVICE_NUMBER StorageDeviceNumber = { 0 };
DWORD Error = 0;
DWORD DeviceIoControlBytesReturned = 0;
CONST static TCHAR Function[] = TEXT("SpGetDeviceNumbersAndType");
HANDLE DeviceFileHandle = INVALID_HANDLE_VALUE;
#if 0
SpHwDebugLog(
This,
TEXT("%1: DeviceIoControl(%2)\n"),
Function,
DevicePath
);
#endif
DeviceFileHandle =
CreateFile(
DevicePath,
GENERIC_READ,
FILE_SHARE_READ | FILE_SHARE_READ | SP_FILE_SHARE_DELETE(),
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL
);
if (DeviceFileHandle == INVALID_HANDLE_VALUE
) {
Error = GetLastError();
if (Error != ERROR_FILE_NOT_FOUND) {
SpHwDebugLog(
This,
TEXT("%1: CreateFile(%2) warning %3!lu!\r\n"),
Function,
DevicePath,
Error
);
}
goto Exit;
}
if (!DeviceIoControl(
DeviceFileHandle,
IOCTL_STORAGE_GET_DEVICE_NUMBER,
NULL,
0,
&StorageDeviceNumber,
sizeof(StorageDeviceNumber),
&DeviceIoControlBytesReturned,
NULL
)) {
Error = GetLastError();
if (Error != ERROR_INVALID_FUNCTION
&& Error != ERROR_FILE_NOT_FOUND
&& Error != ERROR_INVALID_PARAMETER // dynamic disk
)
{
SpHwDebugLog(
This,
TEXT("%1: DeviceIoControl(%2) warning %3!lu!\r\n"),
Function,
DevicePath,
Error
);
}
if (Error == ERROR_INVALID_PARAMETER)
{
// dynamic disk
*DeviceType = FILE_DEVICE_DISK;
}
} else if (DeviceIoControlBytesReturned < sizeof(StorageDeviceNumber)) {
SpHwDebugLog(
This,
TEXT("%1: DeviceIoControl size mismatch (%4!lu!, %5!lu!)"),
Function,
(ULONG)DeviceIoControlBytesReturned,
(ULONG)sizeof(StorageDeviceNumber)
);
} else {
CONST ULONG DeviceNumber = StorageDeviceNumber.DeviceNumber;
#if 0
{
TCHAR DeviceTypeString[64];
SpHwDebugLog(
This,
TEXT("%1: DeviceType of %2 is %3 (%4!lu!)\r\n"),
Function,
DevicePath,
SpDeviceTypeToString(StorageDeviceNumber.DeviceType, DeviceTypeString),
StorageDeviceNumber.DeviceType
);
}
#endif
*DeviceType = StorageDeviceNumber.DeviceType;
if (DeviceNumber > 63) {
SpHwDebugLog(
This,
TEXT("%1: DeviceNumber out of range (%2!lu!)\r\n"),
Function,
StorageDeviceNumber.DeviceNumber
);
}
else {
#if 0
SpHwDebugLog(
This,
TEXT("%1 disk number %2!lu!\r\n"),
DevicePath,
DeviceNumber
);
#endif
DeviceNumbers->Bitset |= (1ui64 << DeviceNumber);
}
}
VolumeDiskExtents = (PVOLUME_DISK_EXTENTS) SpGetVolumeDiskExtents(DeviceFileHandle);
if (VolumeDiskExtents != NULL) {
SIZE_T i;
CONST SIZE_T VolumeDiskExtents_NumberOfDiskExtents = VolumeDiskExtents->NumberOfDiskExtents;
for (i = 0 ; i != VolumeDiskExtents_NumberOfDiskExtents ; ++i) {
CONST SIZE_T DiskNumber = VolumeDiskExtents->Extents[i].DiskNumber;
if (DiskNumber > 63) {
SpHwDebugLog(
This,
TEXT("%1: DiskNumber out of range (%2!lu!)\r\n"),
Function,
DiskNumber
);
}
else {
DeviceNumbers->Bitset |= (1ui64 << DiskNumber);
}
}
}
Exit:
SP_CLOSE_HANDLE(DeviceFileHandle);
SpFree(VolumeDiskExtents);
}
VOID
SpCollectVolumeInformation(
PSP_LOG_HARDWARE This,
PSP_VOLUMES Volumes
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
TCHAR DriveLetter;
TCHAR GuidVolumeNameBuffer[NUMBER_OF(Volumes->Entries[0].GuidVolumeNameBuffer)];
CONST static TCHAR Function[] = TEXT("SpCollectVolumeInformation");
DWORD Error;
SP_STRING GuidVolumeNameString;
ZeroMemory(Volumes, sizeof(*Volumes));
for (DriveLetter = 'C' ; DriveLetter <= 'Z' ; DriveLetter++) {
CONST PSP_VOLUME Volume = &Volumes->Entries[DriveLetter - 'C'];
CONST TCHAR DriveLetterPath[] = { DriveLetter, ':', '\\', 0 };
CONST TCHAR DeviceDriveLetterPath[] = { '\\', '\\', '.', '\\', DriveLetter, ':', 0 };
/* This looks interesting. WindowsXp only.
if (GetVolumePathNamesForVolumeName(DriveLetterPath, VolumeName, NUMBER_OF(VolumeName)))
{
}
*/
GuidVolumeNameBuffer[0] = 0;
if (!This->Linkage->GetVolumeNameForVolumeMountPoint(
DriveLetterPath, GuidVolumeNameBuffer, NUMBER_OF(GuidVolumeNameBuffer) - 1)
)
{
Error = GetLastError();
if (Error != ERROR_FILE_NOT_FOUND
&& Error != ERROR_PATH_NOT_FOUND
) {
SpHwDebugLog(
This,
TEXT("%1: GetVolumeNameForVolumeMountPoint(%2) warning %3!lu!\r\n"),
Function,
DriveLetterPath,
Error
);
}
continue;
}
SpInitString(&GuidVolumeNameString, GuidVolumeNameBuffer);
SpRemoveTrailingChars(&GuidVolumeNameString, TEXT("\\/"));
Volume->DriveLetter = DriveLetter;
CopyMemory(&Volume->GuidVolumeNameBuffer, &GuidVolumeNameBuffer, sizeof(GuidVolumeNameBuffer));
SpGetDeviceNumbersAndType(
This,
Volume->GuidVolumeNameBuffer,
&Volume->DeviceNumbers,
&Volume->DeviceType
);
}
qsort(Volumes->Entries, NUMBER_OF(Volumes->Entries), sizeof(Volumes->Entries[0]), SpCompareVolume);
//Exit:
;
}
BOOLEAN
SpCollectDeviceProperties(
PSP_LOG_HARDWARE This,
ULONG DevInst,
SIZE_T NumberOfProperties,
PCSP_DEVICE_PROPERTY_CONST InArray,
PSP_DEVICE_PROPERTY OutArray
)
{
BOOLEAN Success = FALSE;
SIZE_T PropertyIndex = 0;
CONFIGRET ConfigRet = 0;
CONST static TCHAR Function[] = TEXT("SpCollectDeviceProperties");
SP_STRING ValueString;
SP_STRING Whitespace;
SpInitString(&Whitespace, TEXT("\r\n\v\t "));
for (PropertyIndex = 0 ; PropertyIndex != NumberOfProperties ; PropertyIndex += 1)
{
PCSP_DEVICE_PROPERTY_CONST In = &InArray[PropertyIndex];
PSP_DEVICE_PROPERTY Out = &OutArray[PropertyIndex];
ULONG PropertyBufferSize = sizeof(Out->Value);
//
// zero out two chars due to multi_sz
//
Out->Value[0] = 0;
Out->Value[1] = 0;
ConfigRet =
This->Linkage->CM_Get_DevNode_Registry_Property_Ex(
DevInst,
In->ConfigManagerInteger,
&Out->Type,
Out->Value,
&PropertyBufferSize,
0,
This->Machine.Handle
);
if (ConfigRet == CR_BUFFER_SMALL)
{
//
// zero out two chars due to multi_sz
//
Out->Value[0] = 0;
Out->Value[1] = 0;
#if DBG
SpHwDebugLog(
This,
TEXT("%1: Buffer too small, property %2\r\n"),
Function,
DevicePropertyMetaInfo[PropertyIndex].Name
);
#endif
}
#if UNAVAILABLE_VERBOSE
if (ConfigRet == CR_NO_SUCH_VALUE
&& (DevicePropertyMetaInfo[PropertyIndex].Flags & SP_PROPERTY_QUIET_UNAVAILABLE) == 0
)
{
_tcscpy(Out->Value, TEXT("<unavailable>"));
ConfigRet = CR_SUCCESS;
}
#endif
if (Out->Type == REG_SZ)
{
SpInitString(&ValueString, Out->Value);
SpRemoveTrailingChars(&ValueString, TEXT("\r\n"));
}
#if QUASH_SIMPLE_PHYSICAL_DEVICE_OBJECT_NAMES
/* of the form \Device\12345678 */
if (ConfigRet == CR_SUCCESS
&& In->ConfigManagerInteger == CM_DRP_PHYSICAL_DEVICE_OBJECT_NAME
)
{
CONST static TCHAR SimpleDeviceName[] = TEXT("\\Device\\12345678");
if (lstrlen(Out->Value) == NUMBER_OF(SimpleDeviceName) - 1)
{
SIZE_T i;
for (i = 0 ; i != NUMBER_OF(SimpleDeviceName) - 1 ; ++i)
{
if (iswdigit(SimpleDeviceName[i]) && iswdigit(Out->Value[i]))
{
// ok
}
else if (SimpleDeviceName[i] != Out->Value[i])
{
break;
}
}
if (i == NUMBER_OF(SimpleDeviceName) - 1)
{
Out->Value[0] = 0;
Out->Value[1] = 0;
}
}
}
#endif
if (ConfigRet != CR_SUCCESS
&& ConfigRet != CR_NO_SUCH_VALUE
&& ConfigRet != CR_INVALID_PROPERTY
&& ConfigRet != CR_BUFFER_SMALL
)
goto Exit;
Out->Const = In; // connect it back to the meta info
}
Success = TRUE;
Exit:;
return Success;
}
BOOL
SpGrowArray(
IN OUT PVOID* Array,
IN SIZE_T SizeOfElement,
IN OUT PSIZE_T NumberOfElements,
IN OUT PSIZE_T NumberAllocated
)
{
ASSERT(*NumberOfElements < *NumberAllocated);
*NumberOfElements += 1;
if (*NumberOfElements >= *NumberAllocated) {
PVOID Next = SpRealloc(*Array, SizeOfElement * *NumberAllocated * 2);
if (Next == NULL) {
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
return FALSE;
}
*Array = Next;
*NumberAllocated *= 2;
}
return TRUE;
}
VOID
SpCollectDeviceInformation(
PSP_LOG_HARDWARE This,
PCSP_DEVICE_CLASS DeviceClasses,
SIZE_T NumberOfDeviceClasses,
PCSP_DEVICE_PROPERTY_CONST DevicePropertyMetaInfo, // the size of this is assumed
PSP_DEVICE* OutDevices,
SIZE_T* OutNumberOfDevices
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
CONST static TCHAR Function[] = TEXT("SpCollectDeviceInformation");
SIZE_T DeviceClassIndex = 0;
ULONG DeviceInClassIndex = 0;
SIZE_T NumberOfDevices = 0;
SIZE_T NumberOfDevicesAllocated = 64;
HANDLE DeviceInfoHandle = NULL;
BOOL Success = FALSE;
SP_DEVICE_INTERFACE_DATA SetupDeviceInterfaceData = { sizeof(SetupDeviceInterfaceData) };
DWORD Error = 0;
PSP_DEVICE Devices = NULL;
ULONG DevInst = 0;
ULONG ParentIndex = 0;
SP_STRING String;
struct
{
SP_DEVICE_INTERFACE_DETAIL_DATA Base;
TCHAR Buffer[MAX_PATH];
} DetailAndBuffer;
SP_DEVINFO_DATA SetupDeviceInfoData = { sizeof(SetupDeviceInfoData) };
CONFIGRET ConfigRet = 0;
if (OutDevices != NULL)
*OutDevices = NULL;
if (OutNumberOfDevices != NULL)
*OutNumberOfDevices = 0;
if (OutDevices == NULL || OutNumberOfDevices == NULL)
goto Exit;
DetailAndBuffer.Base.cbSize = sizeof(DetailAndBuffer.Base);
Devices = (PSP_DEVICE)SpMalloc(NumberOfDevicesAllocated * sizeof(*Devices));
if (Devices == NULL)
goto Exit;
ZeroMemory(Devices, NumberOfDevicesAllocated * sizeof(*Devices));
//
// loop over device class (disk, volume, cdrom, etc.)
//
for (DeviceClassIndex = 0 ; DeviceClassIndex != NumberOfDeviceClasses ; ++DeviceClassIndex) {
DeviceInfoHandle =
This->Linkage->SetupDiGetClassDevsEx(
DeviceClasses[DeviceClassIndex].Guid,
NULL,
NULL,
DIGCF_PRESENT | DeviceClasses[DeviceClassIndex].IsInterface,
NULL,
This->Machine.Name,
NULL
);
if (DeviceInfoHandle == INVALID_HANDLE_VALUE)
{
Error = GetLastError();
continue;
}
//
// loop over devices in the class
//
Success = TRUE;
for (DeviceInClassIndex = 0 ; Success ; ++DeviceInClassIndex) {
PSP_DEVICE Device = &Devices[NumberOfDevices];
Success =
This->Linkage->SetupDiEnumDeviceInterfaces(
DeviceInfoHandle,
NULL,//&SetupDeviceInfoData,
DeviceClasses[DeviceClassIndex].Guid,
DeviceInClassIndex,
&SetupDeviceInterfaceData
);
if (!Success)
Error = GetLastError();
if (!Success && Error == ERROR_NO_MORE_ITEMS) {
break;
}
if (!Success) {
break;
}
//
// get the devinst and the device path
//
DevInst = SetupDeviceInfoData.DevInst;
Success =
This->Linkage->SetupDiGetDeviceInterfaceDetail(
DeviceInfoHandle,
&SetupDeviceInterfaceData,
&DetailAndBuffer.Base,
sizeof(DetailAndBuffer),
NULL, /* required size */
&SetupDeviceInfoData
);
if (!Success) {
break;
}
//Device->IsLeaf = TRUE;
Device->DevInst = SetupDeviceInfoData.DevInst;
_tcscpy(Device->DevicePath, DetailAndBuffer.Base.DevicePath);
#if 0
SpHwDebugLog(
This,
TEXT("%1: %2\r\n"),
Function,
Device->DevicePath
);
#endif
#if 1
SpInitString(&String, Device->DevicePath);
SpEnsureTrailingChar(&String, '\\');
if (!This->Linkage->GetVolumeNameForVolumeMountPoint(Device->DevicePath,
Device->GuidVolumePath, NUMBER_OF(Device->GuidVolumePath) - 1)
) {
Error = GetLastError();
if (Error != ERROR_FILE_NOT_FOUND
&& Error != ERROR_PATH_NOT_FOUND
&& Error != ERROR_NOT_READY
&& Error != ERROR_INVALID_FUNCTION
)
{
SpHwDebugLog(
This,
TEXT("%1: GetVolumeNameForVolumeMountPoint(%2) warning %3!lu!\r\n"),
Function,
Device->DevicePath,
Error
);
continue;
}
}
//SpHwDebugLog(This, TEXT("%1: GetVolumeNameForVolumeMountPoint(%2) : %3\r\n"), Function, Device->DevicePath, Device->GuidVolumePath);
SpRemoveTrailingChars(&String, TEXT("\\/"));
SpInitString(&String, Device->GuidVolumePath);
SpRemoveTrailingChars(&String, TEXT("\\/"));
#endif
//
// this is how we match up the devices to the drive letters
//
SpGetDeviceNumbersAndType(
This,
Device->DevicePath,
&Device->DeviceNumbers,
&Device->DeviceType
);
//
// get the parent devinsts
//
for (
(ConfigRet = CR_SUCCESS), (ParentIndex = 0);
(ConfigRet == CR_SUCCESS)
&& ParentIndex < NUMBER_OF(Device->ParentDevInsts);
ParentIndex += 1
)
{
ULONG ChildIndex = (ParentIndex == 0 ? Device->DevInst : Device->ParentDevInsts[ParentIndex - 1]);
ConfigRet =
This->Linkage->CM_Get_Parent_Ex(
&Device->ParentDevInsts[ParentIndex],
ChildIndex,
0,
This->Machine.Handle
);
}
// the last one is never interesting, err.. two
if (ParentIndex != 0)
ParentIndex -= 1;
if (ParentIndex != 0)
ParentIndex -= 1;
Device->NumberOfParents = ParentIndex;
//
// get the properties
// we should do this here, but not if we don't also get the parent properties
//
//SpCollectDeviceProperties(Device, DevicePropertyMetaInfo);
//
// grow the array of devices if necessary
//
if (!SpGrowArray(&Devices, sizeof(Devices[0]), &NumberOfDevices, &NumberOfDevicesAllocated))
goto Exit;
//
// We should probably the properties of the parents here,
// but our data structures are not very good, and this would be inefficient
// We must avoid O(n^2) behavior, because people really do have machines
// with many disks, like 100.
//
// We should keep sorted arrays of devinsts.
//
}
}
if (!Success) {
Error = GetLastError();
}
//SpChangeParentDevInstsToIndices(Devices, NumberOfDevices);
*OutDevices = Devices;
Devices = NULL;
*OutNumberOfDevices = NumberOfDevices;
Exit:
SpFree(Devices);
}
VOID
SpFillStaticString(
PTSTR s,
SIZE_T n,
TCHAR ch
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
ULONGLONG chch;
ULONGLONG* pchch;
SIZE_T i;
SIZE_T m;
if (s[n - 2] != 0)
return;
if (sizeof(ULONGLONG) <= sizeof(TCHAR))
{
for (i = 0 ; i != n - 1 ; i += 1)
{
s[i] = ch;
}
return;
}
chch = 0;
for (i = 0 ; i != sizeof(ULONGLONG)/sizeof(TCHAR) ; i += 1)
{
chch <<= BITS_OF(ch);
chch |= ch;
}
m = (n - 1) / (sizeof(ULONGLONG)/sizeof(TCHAR));
pchch = (ULONGLONG*)s;
for (i = 0 ; i < m ; i += 1)
{
pchch[i] = chch;
}
m *= (sizeof(ULONGLONG)/sizeof(TCHAR));
for (i = m ; i < n - 1 ; i += 1)
{
s[i] = ch;
}
#if DBG
for (i = 0 ; i != n - 1 ; i += 1)
{
ASSERT(s[i] == ch);
}
ASSERT(s[i] == 0);
#endif
}
PCTSTR
SpGetDashesString(
SIZE_T n
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
static union {
ULONGLONG Ulonglongs[1 + 128 / (sizeof(ULONGLONG)/sizeof(TCHAR))];
TCHAR Tchars[128];
} u;
SpFillStaticString(u.Tchars, NUMBER_OF(u.Tchars), '-');
n *= 2;
if (n > NUMBER_OF(u.Tchars))
n = NUMBER_OF(u.Tchars);
return &u.Tchars[NUMBER_OF(u.Tchars) - n];
}
PCTSTR
SpGetSpacesString(
SIZE_T n
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
static union {
ULONGLONG Ulonglongs[1 + 128 / (sizeof(ULONGLONG)/sizeof(TCHAR))];
TCHAR Tchars[128];
} u;
SpFillStaticString(u.Tchars, NUMBER_OF(u.Tchars), ' ');
n *= INDENT_FACTOR;
if (n > NUMBER_OF(u.Tchars))
n = NUMBER_OF(u.Tchars);
return &u.Tchars[NUMBER_OF(u.Tchars) - 1 - n];
}
PCTSTR
SpGetFirstMultipleString(
PCTSTR s
)
{
//
// the first string being empty is odd case
// it is the only string that can be empty
//
if (*s == 0 && *(s + 1) != 0)
s += 1;
return s;
}
PCTSTR
SpGetNextMultipleString(
PCTSTR s
)
{
s += lstrlen(s) + 1;
return s;
}
SIZE_T
SpMultipleStringCount(
PCTSTR s
)
{
SIZE_T i = 0;
if (*s != 0 || *(s + 1) != 0)
{
do
{
i += 1;
s += lstrlen(s) + 1;
} while (*s != 0);
}
return i;
}
VOID
SpLogDeviceProperties(
PSP_LOG_HARDWARE This,
SIZE_T NumberOfProperties,
PCSP_DEVICE_PROPERTY PropertyData,
ULONG Indent
)
{
ULONG PropertyType;
SIZE_T PropertyIndex;
//
// we save away this string, so we that can indent the rest of the lines to account for it
//
TCHAR NumberString[BITS_OF(Indent)];
NumberString[0] = 0;
#if NUMBER_CHILDREN
_stprintf(NumberString, TEXT("%lu. "), Indent + 1);
#endif
#if ONE_DEVICE_PER_LINE
SpHwLog(This, TEXT("%1"), SpGetSpacesString(Indent));
#endif
for (PropertyIndex = 0 ; PropertyIndex != NumberOfProperties ; PropertyIndex += 1)
{
PCTSTR Name = PropertyData[PropertyIndex].Const->Name;
if (PropertyData[PropertyIndex].Value[0] != 0
&& (PropertyIndex == 0
//
// friendly name sometimes == description
// general fix: don't print adjacent equal values,
// unless they are both unavailable (which don't generally have anymore)
//
|| _tcsicmp(PropertyData[PropertyIndex].Value, PropertyData[PropertyIndex - 1].Value) != 0
|| _tcsicmp(PropertyData[PropertyIndex].Value, TEXT("<unavailable>")) == 0
))
{
#if DESCRIPTION_DASH_PHYSICAL_DEVICE_OBJECT
if (PropertyIndex == PHYSICAL_DEVICE_OBJECT)
{
// nothing
}
else
#endif
{
#if NUMBER_CHILDREN
SpHwLog(This, TEXT("%1"), NumberString);
#endif
#if INDENT_CHILDREN && !ONE_DEVICE_PER_LINE
SpHwLog(This, TEXT("%1"), SpGetSpacesString(Indent));
#endif
}
//
// only print "PhysicalDeviceObject" if there was no description
//
#if DESCRIPTION_DASH_PHYSICAL_DEVICE_OBJECT
if (PropertyIndex == PHYSICAL_DEVICE_OBJECT
&& (
PropertyData[DESCRIPTION].Value[0] != 0
&& _tcsicmp(PropertyData[DESCRIPTION].Value, TEXT("<unavailable>")) != 0)
)
{
// nothing
}
else
#endif
{
if (Name != NULL && Name[0] != 0)
{
//SpHwLog(This, TEXT("%1 = "), Name);
SpHwLog(This, TEXT("%1: "), Name);
}
}
PropertyType = PropertyData[PropertyIndex].Type;
// if one device per line, shrink hardware id to just first item
// also compress its formating if it only contains one element
#if ONE_PROPERTY_PER_LINE
if (PropertyType == REG_MULTI_SZ
&& SpMultipleStringCount(PropertyData[PropertyIndex].Value) < 2
)
#endif
{
PropertyType = REG_SZ;
}
switch (PropertyType)
{
case REG_MULTI_SZ:
{
PCTSTR Value;
for (
Value = SpGetFirstMultipleString(PropertyData[PropertyIndex].Value);
*Value != 0;
Value = SpGetNextMultipleString(Value)
)
{
#if INDENT_CHILDREN
SpHwLog(
This,
TEXT("\r\n%1%2"),
SpGetSpacesString(Indent + 2),
Value
);
#else
SpHwLog(
This,
TEXT("\r\n%1 %2"),
NumberString,
Value
);
#endif
}
}
case REG_SZ:
SpHwLog(This, TEXT("%1"), PropertyData[PropertyIndex].Value);
break;
}
#if DESCRIPTION_DASH_PHYSICAL_DEVICE_OBJECT
if (PropertyIndex == DESCRIPTION
&& PropertyData[PHYSICAL_DEVICE_OBJECT].Value[0] != 0
)
{
SpHwLog(This, TEXT(" - "));
}
else
#endif
{
#if ONE_PROPERTY_PER_LINE
SpHwLog(This, TEXT("\r\n"));
#endif
#if ONE_DEVICE_PER_LINE
SpHwLog(This, TEXT(" "));
#endif
}
if (NumberString[0] != 0 && NumberString[0] != ' ')
{
//
// convert to spaces for the rest of the lines
//
SIZE_T i;
for (i = 0 ; NumberString[i] != 0 ; ++i)
{
NumberString[i] = ' ';
}
}
}
}
#if ONE_DEVICE_PER_LINE
SpHwLog(This, TEXT("\r\n"));
#endif
}
VOID
SpLogDeviceTree(
PSP_LOG_HARDWARE This,
PCSP_DEVICE Device,
ULONG Indent
)
{
ULONG ParentIndex;
SP_DEVICE_PROPERTY PropertyData[NUMBER_OF(DevicePropertyMetaInfo)];
if (!SpCollectDeviceProperties(This, Device->DevInst, NUMBER_OF(PropertyData), DevicePropertyMetaInfo, PropertyData))
return;
#if NUMBER_CHILDREN
Indent = 1;
#endif
SpLogDeviceProperties(This, NUMBER_OF(PropertyData), PropertyData, Indent - 1);
for (ParentIndex = 0 ; ParentIndex < Device->NumberOfParents ; ParentIndex += 1)
{
if (!SpCollectDeviceProperties(This, Device->ParentDevInsts[ParentIndex], NUMBER_OF(PropertyData), DevicePropertyMetaInfo, PropertyData))
break;
SpLogDeviceProperties(This, NUMBER_OF(PropertyData), PropertyData, Indent + ParentIndex);
}
#if NUMBER_CHILDREN
SpHwLog(This, TEXT("\r\n"));
#endif
}
VOID
SpLogVolumeAndDeviceInformation(
PSP_LOG_HARDWARE This,
PCSP_VOLUMES Volumes,
PCSP_DEVICE Devices,
SIZE_T NumberOfDevices
)
{
ULONG Indent = 0;
SIZE_T VolumeIndex = 0;
SIZE_T DiskNumber = 0;
SIZE_T DeviceIndex = 0;
for (DeviceIndex = 0; DeviceIndex != NumberOfDevices ; DeviceIndex += 1)
{
PCSP_DEVICE Device = &Devices[DeviceIndex];
for (DiskNumber = 0 ; DiskNumber != BITS_OF_FIELD(SP_DEVICE_NUMBERS, Bitset); DiskNumber += 1)
{
if ((Device->DeviceNumbers.Bitset & (1ui64 << DiskNumber)) != 0)
{
ULONG DriveLetters = 0;
LONG NumberOfDriveLetters = 0;
for (VolumeIndex = 0 ; VolumeIndex != NUMBER_OF(Volumes->Entries) ; ++VolumeIndex)
{
CONST PCSP_VOLUME Volume = &Volumes->Entries[VolumeIndex];
if (Volume->DriveLetter != 0
&& (Volume->DeviceNumbers.Bitset & (1ui64 << DiskNumber)) != 0
&& Volume->DeviceType == Device->DeviceType
)
{
DriveLetters |= (1UL << (Volume->DriveLetter - 'C'));
NumberOfDriveLetters += 1;
}
}
if (DriveLetters != 0)
{
SIZE_T i;
SpHwLog(
This,
TEXT("%1"),
SpGetSpacesString(Indent)
);
SpHwLog(
This,
TEXT("%1: "),
(NumberOfDriveLetters == 1)
? TEXT("Volume")
: TEXT("Volumes")
);
for (i = 'C' ; i <= 'Z' ; i += 1)
{
if ((DriveLetters & (1UL << (i - 'C'))) != 0)
SpHwLog(
This,
TEXT("%1!c!:\\ "),
i
);
}
SpHwLog(This, TEXT("%1"), TEXT("\r\n\r\n"));
}
#if 1
SpHwLog(
This,
TEXT("Device Path: %1%2"),
Device->DevicePath,
TEXT("\r\n\r\n")
);
#endif
//SpHwLog(This, TEXT("\r\n"));
SpLogDeviceTree(This, Device, Indent + 1);
//SpHwLog(This, TEXT("\r\n\r\n"));
SpHwLog(
This,
TEXT("%1\r\n\r\n"),
SpGetDashesString(100)
);
}
}
}
}
VOID
SpLogHardware(
PSP_LOG_HARDWARE_IN In
)
{
SP_VOLUMES Volumes = { 0 };
PSP_DEVICE Devices = NULL;
SIZE_T NumberOfDevices = 0;
CONFIGRET ConfigRet = 0;
SP_MACHINE Machine = { 0 };
SP_LOG_HARDWARE This = { 0 };
This.Machine.Name = In->MachineName;
This.LogFile = In->LogFile;
This.Linkage = &SpLinkage;
This.SetupLogError = In->SetupLogError;
This.SetuplogError = In->SetuplogError;
if (!SpDoDynlink(&This))
{
SpHwLog(&This, TEXT("downlevel, deviceinfo not logged\r\n"));
return;
}
ConfigRet = This.Linkage->CM_Connect_Machine(This.Machine.Name, &This.Machine.Handle);
if (ConfigRet != CR_SUCCESS)
goto Exit;
SpCollectVolumeInformation(
&This,
&Volumes
);
SpCollectDeviceInformation(
&This,
DeviceClasses,
NUMBER_OF(DeviceClasses),
DevicePropertyMetaInfo,
&Devices,
&NumberOfDevices
);
if (Devices != NULL && NumberOfDevices != 0)
{
SpLogVolumeAndDeviceInformation(
&This,
&Volumes,
Devices,
NumberOfDevices
);
SpFree(Devices);
}
Exit:
;
}
#if STANDALONE
void Main(
PSP_LOG_HARDWARE_IN Parameters
)
{
#if STANDALONE == 1
Parameters->LogFile = (HANDLE)_get_osfhandle(_fileno(stdout));
#elif STANDALONE == 2
SetupOpenLog(FALSE);
Parameters->SetupLogError = SetupLogError;
#endif
SpLogHardware(Parameters);
#if STANDALONE == 2
SetupCloseLog();
#endif
}
#ifdef UNICODE
int __cdecl _wmain(int argc, WCHAR** argv)
{
SP_LOG_HARDWARE_IN Parameters = { 0 };
Parameters.MachineName = (argc > 1) ? argv[1] : NULL;
Main(&Parameters);
return 0;
}
#endif
int __cdecl main(int argc, CHAR** argv)
{
SP_LOG_HARDWARE_IN Parameters = { 0 };
#ifdef UNICODE
WCHAR** argvw = NULL;
argvw = CommandLineToArgvW(GetCommandLineW(), &argc);
Parameters.MachineName = (argc > 1) ? argvw[1] : NULL;
#else
Parameters.MachineName = (argc > 1) ? argv[1] : NULL;
#endif
Main(&Parameters);
return 0;
}
#endif
#else
VOID
SpLogHardware(
PSP_LOG_HARDWARE_IN In
)
{
}
#endif