2020-04-04 20:30:40 +08:00
/*
* PROJECT : ReactOS Kernel
* COPYRIGHT : GPL - 2.0 + ( https : //spdx.org/licenses/GPL-2.0+)
* FILE : ntoskrnl / io / debug . c
* PURPOSE : Useful functions for debugging IO and PNP managers
* PROGRAMMERS : Copyright 2020 Vadim Galyant < vgal @ rambler . ru >
*/
# include <ntoskrnl.h>
# include "pnpio.h"
# define NDEBUG
# include <debug.h>
/* GLOBALS ********************************************************************/
extern PDEVICE_NODE IopRootDeviceNode ;
/* FUNCTIONS ******************************************************************/
/* CmResource */
2021-06-11 20:29:21 +08:00
/* PipDumpCmResourceDescriptor() displays information about a Cm Descriptor
2020-04-04 20:30:40 +08:00
DebugLevel : 0 - always dump
1 - dump if not defined NDEBUG
*/
VOID
NTAPI
PipDumpCmResourceDescriptor (
_In_ PCM_PARTIAL_RESOURCE_DESCRIPTOR Descriptor ,
_In_ ULONG DebugLevel )
{
PAGED_CODE ( ) ;
if ( DebugLevel ! = 0 )
{
# ifdef NDEBUG
return ;
# endif
}
if ( Descriptor = = NULL )
{
DPRINT1 ( " Dump CmDescriptor: Descriptor == NULL \n " ) ;
return ;
}
switch ( Descriptor - > Type )
{
case CmResourceTypePort :
DPRINT1 ( " [%p:%X:%X] IO: Start %X:%X, Len %X \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . Port . Start . HighPart , Descriptor - > u . Port . Start . LowPart , Descriptor - > u . Port . Length ) ;
break ;
case CmResourceTypeInterrupt :
DPRINT1 ( " [%p:%X:%X] INT: Lev %X Vec %X Aff %IX \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . Interrupt . Level , Descriptor - > u . Interrupt . Vector , Descriptor - > u . Interrupt . Affinity ) ;
break ;
case CmResourceTypeMemory :
DPRINT1 ( " [%p:%X:%X] MEM: %X:%X Len %X \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . Memory . Start . HighPart , Descriptor - > u . Memory . Start . LowPart , Descriptor - > u . Memory . Length ) ;
break ;
case CmResourceTypeDma :
DPRINT1 ( " [%p:%X:%X] DMA: Channel %X Port %X \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . Dma . Channel , Descriptor - > u . Dma . Port ) ;
break ;
case CmResourceTypeDeviceSpecific :
DPRINT1 ( " [%p:%X:%X] DAT: DataSize %X \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . DeviceSpecificData . DataSize ) ;
break ;
case CmResourceTypeBusNumber :
DPRINT1 ( " [%p:%X:%X] BUS: Start %X Len %X Reserv %X \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . BusNumber . Start , Descriptor - > u . BusNumber . Length , Descriptor - > u . BusNumber . Reserved ) ;
break ;
case CmResourceTypeDevicePrivate :
DPRINT1 ( " [%p:%X:%X] PVT: D[0] %X D[1] %X D[2] %X \n " , Descriptor , Descriptor - > ShareDisposition , Descriptor - > Flags , Descriptor - > u . DevicePrivate . Data [ 0 ] , Descriptor - > u . DevicePrivate . Data [ 1 ] , Descriptor - > u . DevicePrivate . Data [ 2 ] ) ;
break ;
default :
DPRINT1 ( " [%p] Unknown type %X \n " , Descriptor , Descriptor - > Type ) ;
break ;
}
}
/* PipGetNextCmPartialDescriptor() return poiner to next a Cm Descriptor */
PCM_PARTIAL_RESOURCE_DESCRIPTOR
NTAPI
PipGetNextCmPartialDescriptor (
_In_ PCM_PARTIAL_RESOURCE_DESCRIPTOR CmDescriptor )
{
PCM_PARTIAL_RESOURCE_DESCRIPTOR NextDescriptor ;
/* Assume the descriptors are the fixed size ones */
NextDescriptor = CmDescriptor + 1 ;
/* But check if this is actually a variable-sized descriptor */
if ( CmDescriptor - > Type = = CmResourceTypeDeviceSpecific )
{
/* Add the size of the variable section as well */
NextDescriptor = ( PVOID ) ( ( ULONG_PTR ) NextDescriptor +
CmDescriptor - > u . DeviceSpecificData . DataSize ) ;
}
/* Now the correct pointer has been computed, return it */
return NextDescriptor ;
}
2021-06-11 20:29:21 +08:00
/* PipDumpCmResourceList() displays information about a Cm List
2020-04-04 20:30:40 +08:00
DebugLevel : 0 - always dump
1 - dump if not defined NDEBUG
*/
VOID
NTAPI
PipDumpCmResourceList (
_In_ PCM_RESOURCE_LIST CmResource ,
_In_ ULONG DebugLevel )
{
PCM_FULL_RESOURCE_DESCRIPTOR FullList ;
PCM_PARTIAL_RESOURCE_DESCRIPTOR Descriptor ;
ULONG ix ;
ULONG jx ;
PAGED_CODE ( ) ;
if ( DebugLevel ! = 0 )
{
# ifdef NDEBUG
return ;
# endif
}
DPRINT1 ( " Dump CmList: CmResource %p \n " , CmResource ) ;
if ( CmResource = = NULL )
{
DPRINT1 ( " PipDumpCmResourceList: CmResource == NULL \n " ) ;
return ;
}
if ( CmResource - > Count = = 0 )
{
DPRINT1 ( " PipDumpCmResourceList: CmResource->Count == 0 \n " ) ;
return ;
}
DPRINT1 ( " FullList Count %x \n " , CmResource - > Count ) ;
FullList = & CmResource - > List [ 0 ] ;
for ( ix = 0 ; ix < CmResource - > Count ; ix + + )
{
DPRINT1 ( " List #%X Iface %X Bus #%X Ver.%X Rev.%X Count %X \n " ,
ix ,
FullList - > InterfaceType ,
FullList - > BusNumber ,
FullList - > PartialResourceList . Version ,
FullList - > PartialResourceList . Revision ,
FullList - > PartialResourceList . Count ) ;
Descriptor = FullList - > PartialResourceList . PartialDescriptors ;
for ( jx = 0 ; jx < FullList - > PartialResourceList . Count ; jx + + )
{
PipDumpCmResourceDescriptor ( Descriptor , DebugLevel ) ;
Descriptor = PipGetNextCmPartialDescriptor ( Descriptor ) ;
}
FullList = ( PCM_FULL_RESOURCE_DESCRIPTOR ) Descriptor ;
}
}
/* IoResource */
2021-06-11 20:29:21 +08:00
/* PipDumpIoResourceDescriptor() displays information about a Io Descriptor
2020-04-04 20:30:40 +08:00
DebugLevel : 0 - always dump
1 - dump if not defined NDEBUG
*/
VOID
NTAPI
PipDumpIoResourceDescriptor (
_In_ PIO_RESOURCE_DESCRIPTOR Descriptor ,
_In_ ULONG DebugLevel )
{
PAGED_CODE ( ) ;
if ( DebugLevel ! = 0 )
{
# ifdef NDEBUG
return ;
# endif
}
if ( Descriptor = = NULL )
{
DPRINT1 ( " DumpResourceDescriptor: Descriptor == 0 \n " ) ;
return ;
}
switch ( Descriptor - > Type )
{
case CmResourceTypeNull :
DPRINT1 ( " [%p:%X:%X] O: Len %X Align %X Min %I64X, Max %I64X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . Generic . Length , Descriptor - > u . Generic . Alignment , Descriptor - > u . Generic . MinimumAddress . QuadPart , Descriptor - > u . Generic . MaximumAddress . QuadPart ) ;
break ;
case CmResourceTypePort :
DPRINT1 ( " [%p:%X:%X] IO: Min %X:%X, Max %X:%X, Align %X Len %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . Port . MinimumAddress . HighPart , Descriptor - > u . Port . MinimumAddress . LowPart , Descriptor - > u . Port . MaximumAddress . HighPart , Descriptor - > u . Port . MaximumAddress . LowPart , Descriptor - > u . Port . Alignment , Descriptor - > u . Port . Length ) ;
break ;
case CmResourceTypeInterrupt :
DPRINT1 ( " [%p:%X:%X] INT: Min %X Max %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . Interrupt . MinimumVector , Descriptor - > u . Interrupt . MaximumVector ) ;
break ;
case CmResourceTypeMemory :
DPRINT1 ( " [%p:%X:%X] MEM: Min %X:%X, Max %X:%X, Align %X Len %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . Memory . MinimumAddress . HighPart , Descriptor - > u . Memory . MinimumAddress . LowPart , Descriptor - > u . Memory . MaximumAddress . HighPart , Descriptor - > u . Memory . MaximumAddress . LowPart , Descriptor - > u . Memory . Alignment , Descriptor - > u . Memory . Length ) ;
break ;
case CmResourceTypeDma :
DPRINT1 ( " [%p:%X:%X] DMA: Min %X Max %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . Dma . MinimumChannel , Descriptor - > u . Dma . MaximumChannel ) ;
break ;
case CmResourceTypeBusNumber :
DPRINT1 ( " [%p:%X:%X] BUS: Min %X Max %X Len %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . BusNumber . MinBusNumber , Descriptor - > u . BusNumber . MaxBusNumber , Descriptor - > u . BusNumber . Length ) ;
break ;
case CmResourceTypeConfigData :
DPRINT1 ( " [%p:%X:%X] CFG: Priority %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . ConfigData . Priority ) ;
break ;
case CmResourceTypeDevicePrivate :
DPRINT1 ( " [%p:%X:%X] DAT: %X %X %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > u . DevicePrivate . Data [ 0 ] , Descriptor - > u . DevicePrivate . Data [ 1 ] , Descriptor - > u . DevicePrivate . Data [ 2 ] ) ;
break ;
default :
DPRINT1 ( " [%p:%X:%X]. Unknown type %X \n " , Descriptor , Descriptor - > Option , Descriptor - > ShareDisposition , Descriptor - > Type ) ;
break ;
}
}
2021-06-11 20:29:21 +08:00
/* PipDumpResourceRequirementsList() displays information about a Io List
2020-04-04 20:30:40 +08:00
DebugLevel : 0 - always dump
1 - dump if not defined NDEBUG
*/
VOID
NTAPI
PipDumpResourceRequirementsList (
_In_ PIO_RESOURCE_REQUIREMENTS_LIST IoResource ,
_In_ ULONG DebugLevel )
{
PIO_RESOURCE_LIST AltList ;
PIO_RESOURCE_DESCRIPTOR Descriptor ;
ULONG ix ;
ULONG jx ;
PAGED_CODE ( ) ;
if ( DebugLevel ! = 0 )
{
# ifdef NDEBUG
return ;
# endif
}
if ( IoResource = = NULL )
{
DPRINT1 ( " PipDumpResourceRequirementsList: IoResource == 0 \n " ) ;
return ;
}
DPRINT1 ( " Dump RequirementsList: IoResource %p \n " , IoResource ) ;
DPRINT1 ( " Interface %X Bus %X Slot %X AlternativeLists %X \n " ,
IoResource - > InterfaceType ,
IoResource - > BusNumber ,
IoResource - > SlotNumber ,
IoResource - > AlternativeLists ) ;
AltList = & IoResource - > List [ 0 ] ;
if ( IoResource - > AlternativeLists < 1 )
{
DPRINT1 ( " PipDumpResourceRequirementsList: AlternativeLists < 1 \n " ) ;
return ;
}
for ( ix = 0 ; ix < IoResource - > AlternativeLists ; ix + + )
{
DPRINT1 ( " AltList %p, AltList->Count %X \n " , AltList , AltList - > Count ) ;
for ( jx = 0 ; jx < AltList - > Count ; jx + + )
{
Descriptor = & AltList - > Descriptors [ jx ] ;
PipDumpIoResourceDescriptor ( Descriptor , DebugLevel ) ;
}
AltList = ( PIO_RESOURCE_LIST ) ( AltList - > Descriptors + AltList - > Count ) ;
DPRINT1 ( " End Descriptors %p \n " , AltList ) ;
}
}
/* Gets the name of the device node state. */
PWSTR
NTAPI
PipGetDeviceNodeStateName (
_In_ PNP_DEVNODE_STATE State )
{
switch ( State )
{
case DeviceNodeUnspecified :
return L " DeviceNodeUnspecified " ;
case DeviceNodeUninitialized :
return L " DeviceNodeUninitialized " ;
case DeviceNodeInitialized :
return L " DeviceNodeInitialized " ;
case DeviceNodeDriversAdded :
return L " DeviceNodeDriversAdded " ;
case DeviceNodeResourcesAssigned :
return L " DeviceNodeResourcesAssigned " ;
case DeviceNodeStartPending :
return L " DeviceNodeStartPending " ;
case DeviceNodeStartCompletion :
return L " DeviceNodeStartCompletion " ;
case DeviceNodeStartPostWork :
return L " DeviceNodeStartPostWork " ;
case DeviceNodeStarted :
return L " DeviceNodeStarted " ;
case DeviceNodeQueryStopped :
return L " DeviceNodeQueryStopped " ;
case DeviceNodeStopped :
return L " DeviceNodeStopped " ;
case DeviceNodeRestartCompletion :
return L " DeviceNodeRestartCompletion " ;
case DeviceNodeEnumeratePending :
return L " DeviceNodeEnumeratePending " ;
case DeviceNodeEnumerateCompletion :
return L " DeviceNodeEnumerateCompletion " ;
case DeviceNodeAwaitingQueuedDeletion :
return L " DeviceNodeAwaitingQueuedDeletion " ;
case DeviceNodeAwaitingQueuedRemoval :
return L " DeviceNodeAwaitingQueuedRemoval " ;
case DeviceNodeQueryRemoved :
return L " DeviceNodeQueryRemoved " ;
case DeviceNodeRemovePendingCloses :
return L " DeviceNodeRemovePendingCloses " ;
case DeviceNodeRemoved :
return L " DeviceNodeRemoved " ;
case DeviceNodeDeletePendingCloses :
return L " DeviceNodeDeletePendingCloses " ;
case DeviceNodeDeleted :
return L " DeviceNodeDeleted " ;
default :
break ;
}
if ( State ! = MaxDeviceNodeState )
{
DPRINT1 ( " PipGetDeviceNodeStateName: Unknown State %X \n " , State ) ;
}
return L " " ;
}
/* Dump list arbiters for the device node */
VOID
NTAPI
PipDumpArbiters (
_In_ PDEVICE_NODE DeviceNode ,
_In_ ULONG NodeLevel )
{
DPRINT ( " Level %X DevNode %p for PDO %p \n " , NodeLevel , DeviceNode , DeviceNode - > PhysicalDeviceObject ) ;
/* Definitions needed for arbiter */
UNIMPLEMENTED ;
}
/* PipDumpDeviceNode() displays information about a device node
Flags is bitmask parametr :
1 - traversal of all children nodes
2 - dump allocated resources and boot configuration resources
4 - dump resources required
8 - dump translated resources
DebugLevel : 0 - always dump
1 - dump if not defined NDEBUG
*/
VOID
NTAPI
PipDumpDeviceNode (
_In_ PDEVICE_NODE DeviceNode ,
_In_ ULONG NodeLevel ,
_In_ ULONG Flags ,
_In_ ULONG DebugLevel )
{
PDEVICE_NODE ChildDeviceNode ;
if ( DebugLevel ! = 0 )
{
# ifdef NDEBUG
return ;
# endif
}
DPRINT1 ( " * Level %X DevNode %p for PDO %p \n " , NodeLevel , DeviceNode , DeviceNode - > PhysicalDeviceObject ) ;
DPRINT1 ( " Instance %wZ \n " , & DeviceNode - > InstancePath ) ;
if ( DeviceNode - > ServiceName . Length )
DPRINT1 ( " Service %wZ \n " , & DeviceNode - > ServiceName ) ;
#if 0
/* It is not used yet */
DPRINT1 ( " State %X %S \n " , DeviceNode - > State , PipGetDeviceNodeStateName ( DeviceNode - > State ) ) ;
DPRINT1 ( " Prev State %X %S \n " , DeviceNode - > PreviousState , PipGetDeviceNodeStateName ( DeviceNode - > PreviousState ) ) ;
# endif
if ( DeviceNode - > Problem )
DPRINT1 ( " Problem %X \n " , DeviceNode - > Problem ) ;
#if 0
/* It is not implemeted yet */
PipDumpArbiters ( DeviceNode , NodeLevel ) ;
# endif
/* Allocated resources and Boot configuration (reported by IRP_MN_QUERY_RESOURCES)*/
if ( Flags & PIP_DUMP_FL_RES_ALLOCATED )
{
if ( DeviceNode - > ResourceList )
{
DPRINT1 ( " ---------- ResourceList ---------- \n " ) ;
PipDumpCmResourceList ( DeviceNode - > ResourceList , DebugLevel ) ;
}
if ( DeviceNode - > BootResources )
{
DPRINT1 ( " ---------- BootResources ---------- \n " ) ;
PipDumpCmResourceList ( DeviceNode - > BootResources , DebugLevel ) ;
}
}
/* Resources required (reported by IRP_MN_FILTER_RESOURCE_REQUIREMENTS) */
if ( Flags & PIP_DUMP_FL_RES_REQUIREMENTS )
{
if ( DeviceNode - > ResourceRequirements )
{
DPRINT1 ( " ---------- ResourceRequirements ---------- \n " ) ;
PipDumpResourceRequirementsList ( DeviceNode - > ResourceRequirements , DebugLevel ) ;
}
}
/* Translated resources (AllocatedResourcesTranslated) */
if ( Flags & PIP_DUMP_FL_RES_TRANSLATED )
{
if ( DeviceNode - > ResourceListTranslated )
{
DPRINT1 ( " ---------- ResourceListTranslated ---------- \n " ) ;
PipDumpCmResourceList ( DeviceNode - > ResourceListTranslated , DebugLevel ) ;
}
}
/* Traversal of all children nodes */
if ( Flags & PIP_DUMP_FL_ALL_NODES )
{
for ( ChildDeviceNode = DeviceNode - > Child ;
ChildDeviceNode ! = NULL ;
ChildDeviceNode = ChildDeviceNode - > Sibling )
{
/* Recursive call */
PipDumpDeviceNode ( ChildDeviceNode , ( NodeLevel + 1 ) , Flags , DebugLevel ) ;
}
}
}
/* PipDumpDeviceNodes() displays information about a node(s) in the device tree
If DeviceNode is NULL , then the dump starts at the beginning of the device tree .
Flags is bitmask parametr :
1 - traversal of all children nodes
2 - dump allocated resources and boot configuration resources
4 - dump resources required
8 - dump translated resources
DebugLevel : 0 - always dump
1 - dump if not defined NDEBUG
See also : Windows DDK - > General Driver Information - >
Kernel - Mode Driver Architecture - > Design Guide - > Plug and Play - >
Introduction to Plug and Play - > Hardware Resources
*/
VOID
NTAPI
PipDumpDeviceNodes (
_In_ PDEVICE_NODE DeviceNode ,
_In_ ULONG Flags ,
_In_ ULONG DebugLevel )
{
if ( DebugLevel ! = 0 )
{
# ifdef NDEBUG
return ;
# endif
}
DPRINT1 ( " PipDumpDeviceNodes: DeviceNode %X, Flags %X Level %X \n " , DeviceNode , Flags , DebugLevel ) ;
if ( DeviceNode = = NULL )
{
DeviceNode = IopRootDeviceNode ;
}
PipDumpDeviceNode ( DeviceNode , 0 , Flags , DebugLevel ) ;
}
/* EOF */