xref: /reactos/drivers/storage/class/classpnp/debug.c (revision 3e1f4074)
1 /*++
2 
3 Copyright (C) Microsoft Corporation, 1991 - 2010
4 
5 Module Name:
6 
7     debug.c
8 
9 Abstract:
10 
11     CLASSPNP debug code and data
12 
13 Environment:
14 
15     kernel mode only
16 
17 Notes:
18 
19 
20 Revision History:
21 
22 --*/
23 
24 
25 #include "classp.h"
26 #include "debug.h"
27 
28 #ifdef DEBUG_USE_WPP
29 #include "debug.tmh"
30 #endif
31 
32 #if DBG
33 
34     //
35     // default to not breaking in for lost irps, five minutes before we even
36     // bother checking for lost irps, using standard debug print macros, and
37     // using a 64k debug print buffer
38     //
39 
40     #ifndef     CLASS_GLOBAL_BREAK_ON_LOST_IRPS
41         #error "CLASS_GLOBAL_BREAK_ON_LOST_IRPS undefined"
42         #define CLASS_GLOBAL_BREAK_ON_LOST_IRPS 0
43     #endif   // CLASS_GLOBAL_BREAK_ON_LOST_IRPS
44 
45     #ifndef     CLASS_GLOBAL_SECONDS_TO_WAIT_FOR_SYNCHRONOUS_SRB
46         #error "CLASS_GLOBAL_SECONDS_TO_WAIT_FOR_SYNCHRONOUS_SRB undefined"
47         #define CLASS_GLOBAL_SECONDS_TO_WAIT_FOR_SYNCHRONOUS_SRB 300
48     #endif   // CLASS_GLOBAL_SECONDS_TO_WAIT_FOR_SYNCHRONOUS_SRB
49 
50     #ifndef     CLASS_GLOBAL_BUFFERED_DEBUG_PRINT
51         #error "CLASS_GLOBAL_BUFFERED_DEBUG_PRINT undefined"
52         #define CLASS_GLOBAL_BUFFERED_DEBUG_PRINT 0
53     #endif   // CLASS_GLOBAL_BUFFERED_DEBUG_PRINT
54 
55     #ifndef     CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFER_SIZE
56         #error "CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFER_SIZE undefined"
57         #define CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFER_SIZE 512
58     #endif   // CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFER_SIZE
59 
60     #ifndef     CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFERS
61         #error "CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFERS undefined"
62         #define CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFERS 512
63     #endif   // CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFERS
64 
65 #ifdef _MSC_VER
66     #pragma data_seg("NONPAGE")
67 #endif
68 
69 
70 
71     CLASSPNP_GLOBALS ClasspnpGlobals;
72 
73     //
74     // the low sixteen bits are used to see if the debug level is high enough
75     // the high sixteen bits are used to singly enable debug levels 1-16
76     //
77     LONG ClassDebug = 0x00000000;
78 
79     BOOLEAN DebugTrapOnWarn = FALSE;
80 
81     //
82     // Used to track callers when we receive an access and the disk
83     // is powered down.
84     //
85     ULONG DiskSpinupIndex = 0;
86     DISK_SPINUP_TRACES DiskSpinupTraces[NUMBER_OF_DISK_SPINUP_TRACES];
87 
88     VOID ClasspInitializeDebugGlobals()
89     {
90         KIRQL irql;
91 
92         if (InterlockedCompareExchange(&ClasspnpGlobals.Initializing, 1, 0) == 0) {
93 
94             KeInitializeSpinLock(&ClasspnpGlobals.SpinLock);
95 
96             KeAcquireSpinLock(&ClasspnpGlobals.SpinLock, &irql);
97 
98             TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_INIT, "CLASSPNP.SYS => Initializing ClasspnpGlobals...\n"));
99 
100             ClasspnpGlobals.Buffer = NULL;
101             ClasspnpGlobals.Index = (ULONG)-1;
102             ClasspnpGlobals.BreakOnLostIrps = CLASS_GLOBAL_BREAK_ON_LOST_IRPS;
103             ClasspnpGlobals.EachBufferSize = CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFER_SIZE;
104             ClasspnpGlobals.NumberOfBuffers = CLASS_GLOBAL_BUFFERED_DEBUG_PRINT_BUFFERS;
105             ClasspnpGlobals.SecondsToWaitForIrps = CLASS_GLOBAL_SECONDS_TO_WAIT_FOR_SYNCHRONOUS_SRB;
106 
107             //
108             // this should be the last item set
109             //
110 
111             ClasspnpGlobals.UseBufferedDebugPrint = CLASS_GLOBAL_BUFFERED_DEBUG_PRINT;
112 
113             KeReleaseSpinLock(&ClasspnpGlobals.SpinLock, irql);
114 
115             InterlockedExchange(&ClasspnpGlobals.Initialized, 1);
116 
117         }
118     }
119 
120     /*++////////////////////////////////////////////////////////////////////////////
121 
122     ClassDebugPrint()
123 
124     Routine Description:
125 
126         Debug print for all class drivers, NOOP on FRE versions.
127         Allows printing to a debug buffer (with auto fallback to kdprint) by
128         properly setting the Globals in classpnp on CHK versions.
129 
130     Arguments:
131 
132         Debug print level, or from 0 to 3 for legacy drivers.
133 
134     Return Value:
135 
136         None
137 
138     --*/
139     VOID ClassDebugPrint(_In_ CLASS_DEBUG_LEVEL DebugPrintLevel, _In_z_ PCCHAR DebugMessage, ...)
140     {
141         va_list ap;
142         va_start(ap, DebugMessage);
143 
144         if ((DebugPrintLevel <= (ClassDebug & 0x0000ffff)) ||
145             ((1 << (DebugPrintLevel + 15)) & ClassDebug)) {
146 
147             if (ClasspnpGlobals.UseBufferedDebugPrint &&
148                 ClasspnpGlobals.Buffer == NULL) {
149 
150                 //
151                 // this double-check prevents always taking
152                 // a spinlock just to ensure we have a buffer
153                 //
154 
155                 KIRQL irql;
156 
157                 KeAcquireSpinLock(&ClasspnpGlobals.SpinLock, &irql);
158                 if (ClasspnpGlobals.Buffer == NULL) {
159 
160                     SIZE_T bufferSize;
161                     if (NT_SUCCESS(
162                             RtlSIZETMult(ClasspnpGlobals.NumberOfBuffers,
163                                          ClasspnpGlobals.EachBufferSize,
164                                          &bufferSize))) {
165 
166                         DbgPrintEx(DPFLTR_CLASSPNP_ID, DPFLTR_ERROR_LEVEL,
167                                    "ClassDebugPrint: Allocating %x bytes for "
168                                    "classdebugprint buffer\n", (ULONG)bufferSize);
169                         ClasspnpGlobals.Index = (ULONG)-1;
170                         ClasspnpGlobals.Buffer =
171                             ExAllocatePoolWithTag(NonPagedPoolNx, bufferSize, 'bDcS');
172                         DbgPrintEx(DPFLTR_CLASSPNP_ID, DPFLTR_ERROR_LEVEL,
173                                    "ClassDebugPrint: Allocated buffer at %p\n",
174                                    ClasspnpGlobals.Buffer);
175 
176                         if (ClasspnpGlobals.Buffer) {
177                             RtlZeroMemory(ClasspnpGlobals.Buffer, bufferSize);
178                         }
179                     }
180 
181                 }
182                 KeReleaseSpinLock(&ClasspnpGlobals.SpinLock, irql);
183 
184             }
185 
186             if (ClasspnpGlobals.UseBufferedDebugPrint &&
187                 ClasspnpGlobals.Buffer != NULL) {
188 
189                 //
190                 // we never free the buffer, so once it exists,
191                 // we can just print to it with immunity
192                 //
193 
194                 ULONG index;
195                 PUCHAR buffer;
196                 NTSTATUS status;
197                 index = InterlockedIncrement((volatile LONG *)&ClasspnpGlobals.Index);
198                 index %= ClasspnpGlobals.NumberOfBuffers;
199                 index *= (ULONG)ClasspnpGlobals.EachBufferSize;
200 
201                 buffer = ClasspnpGlobals.Buffer;
202                 buffer += index;
203 
204                 RtlZeroMemory(buffer, ClasspnpGlobals.EachBufferSize);
205 
206                 status = RtlStringCchVPrintfA((NTSTRSAFE_PSTR)buffer, ClasspnpGlobals.EachBufferSize, DebugMessage, ap);
207                 if (!NT_SUCCESS(status))
208                 {
209                     *buffer = 0; // force-null on failure
210                 }
211 
212             } else {
213 
214                 //
215                 // either we could not allocate a buffer for debug prints
216                 // or buffered debug prints are disabled
217                 //
218 
219                 vDbgPrintEx(DPFLTR_CLASSPNP_ID, DPFLTR_INFO_LEVEL, DebugMessage, ap);
220 
221             }
222 
223         }
224 
225         va_end(ap);
226 
227     }
228 
229 
230     /*
231      *  DbgCheckReturnedPkt
232      *
233      *      Check a completed TRANSFER_PACKET for all sorts of error conditions
234      *      and warn/trap appropriately.
235      */
236     VOID DbgCheckReturnedPkt(TRANSFER_PACKET *Pkt)
237     {
238         PCDB pCdb = ClasspTransferPacketGetCdb(Pkt);
239 
240         NT_ASSERT(SrbGetOriginalRequest(Pkt->Srb) == Pkt->Irp);
241         NT_ASSERT(SrbGetDataBuffer(Pkt->Srb) == Pkt->BufPtrCopy);
242         NT_ASSERT(SrbGetDataTransferLength(Pkt->Srb) <= Pkt->BufLenCopy);
243         NT_ASSERT(!Pkt->Irp->CancelRoutine);
244 
245         if (SRB_STATUS(Pkt->Srb->SrbStatus) == SRB_STATUS_PENDING){
246             TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_RW, "SRB completed with status PENDING in packet %ph: (op=%s srbstat=%s(%xh), irpstat=%xh)",
247                         Pkt,
248                         DBGGETSCSIOPSTR(Pkt->Srb),
249                         DBGGETSRBSTATUSSTR(Pkt->Srb),
250                         (ULONG)Pkt->Srb->SrbStatus,
251                         Pkt->Irp->IoStatus.Status));
252         }
253         else if (SRB_STATUS(Pkt->Srb->SrbStatus) == SRB_STATUS_SUCCESS){
254             /*
255              *  Make sure SRB and IRP status match.
256              */
257             if (!NT_SUCCESS(Pkt->Irp->IoStatus.Status)){
258                 TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_RW, "SRB and IRP status don't match in packet %ph: (op=%s srbstat=%s(%xh), irpstat=%xh)",
259                             Pkt,
260                             DBGGETSCSIOPSTR(Pkt->Srb),
261                             DBGGETSRBSTATUSSTR(Pkt->Srb),
262                             (ULONG)Pkt->Srb->SrbStatus,
263                             Pkt->Irp->IoStatus.Status));
264             }
265 
266             if (Pkt->Irp->IoStatus.Information != SrbGetDataTransferLength(Pkt->Srb)){
267                 TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_RW, "SRB and IRP result transfer lengths don't match in succeeded packet %ph: (op=%s, SrbStatus=%s, Srb.DataTransferLength=%xh, Irp->IoStatus.Information=%Ixh).",
268                             Pkt,
269                             DBGGETSCSIOPSTR(Pkt->Srb),
270                             DBGGETSRBSTATUSSTR(Pkt->Srb),
271                             SrbGetDataTransferLength(Pkt->Srb),
272                             Pkt->Irp->IoStatus.Information));
273             }
274         }
275         else {
276             if (NT_SUCCESS(Pkt->Irp->IoStatus.Status)){
277                 TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_RW, "SRB and IRP status don't match in packet %ph: (op=%s srbstat=%s(%xh), irpstat=%xh)",
278                             Pkt,
279                             DBGGETSCSIOPSTR(Pkt->Srb),
280                             DBGGETSRBSTATUSSTR(Pkt->Srb),
281                             (ULONG)Pkt->Srb->SrbStatus,
282                             Pkt->Irp->IoStatus.Status));
283             }
284             TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_RW, "Packet %ph failed (op=%s srbstat=%s(%xh), irpstat=%xh, sense=%s/%s/%s)",
285                             Pkt,
286                             DBGGETSCSIOPSTR(Pkt->Srb),
287                             DBGGETSRBSTATUSSTR(Pkt->Srb),
288                             (ULONG)Pkt->Srb->SrbStatus,
289                             Pkt->Irp->IoStatus.Status,
290                             DBGGETSENSECODESTR(Pkt->Srb),
291                             DBGGETADSENSECODESTR(Pkt->Srb),
292                             DBGGETADSENSEQUALIFIERSTR(Pkt->Srb)));
293 
294             /*
295              *  If the SRB failed with underrun or overrun, then the actual
296              *  transferred length should be returned in both SRB and IRP.
297              *  (SRB's only have an error status for overrun, so it's overloaded).
298              */
299             if ((SRB_STATUS(Pkt->Srb->SrbStatus) == SRB_STATUS_DATA_OVERRUN) &&
300                (Pkt->Irp->IoStatus.Information != SrbGetDataTransferLength(Pkt->Srb))){
301                 TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_RW, "SRB and IRP result transfer lengths don't match in failed packet %ph: (op=%s, SrbStatus=%s, Srb.DataTransferLength=%xh, Irp->IoStatus.Information=%Ixh).",
302                             Pkt,
303                             DBGGETSCSIOPSTR(Pkt->Srb),
304                             DBGGETSRBSTATUSSTR(Pkt->Srb),
305                             SrbGetDataTransferLength(Pkt->Srb),
306                             Pkt->Irp->IoStatus.Information));
307             }
308         }
309 
310         /*
311          *  If the port driver returned STATUS_INSUFFICIENT_RESOURCES,
312          *  make sure this is also the InternalStatus in the SRB so that we process it correctly.
313          */
314         if (Pkt->Irp->IoStatus.Status == STATUS_INSUFFICIENT_RESOURCES){
315             NT_ASSERT(SRB_STATUS(Pkt->Srb->SrbStatus) == SRB_STATUS_INTERNAL_ERROR);
316             NT_ASSERT(SrbGetSystemStatus(Pkt->Srb) == STATUS_INSUFFICIENT_RESOURCES);
317         }
318 
319         /*
320          *  Some miniport drivers have been caught changing the SCSI operation
321          *  code in the SRB.  This is absolutely disallowed as it breaks our error handling.
322          */
323         switch (pCdb->CDB10.OperationCode){
324             case SCSIOP_MEDIUM_REMOVAL:
325             case SCSIOP_MODE_SENSE:
326             case SCSIOP_READ_CAPACITY:
327             case SCSIOP_READ:
328             case SCSIOP_WRITE:
329             case SCSIOP_START_STOP_UNIT:
330             case SCSIOP_READ_CAPACITY16:
331             case SCSIOP_READ16:
332             case SCSIOP_WRITE16:
333                 break;
334             default:
335                 TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_RW, "Miniport illegally changed Srb.Cdb.OperationCode in packet %ph failed (op=%s srbstat=%s(%xh), irpstat=%xh, sense=%s/%s/%s)",
336                                 Pkt,
337                                 DBGGETSCSIOPSTR(Pkt->Srb),
338                                 DBGGETSRBSTATUSSTR(Pkt->Srb),
339                                 (ULONG)Pkt->Srb->SrbStatus,
340                                 Pkt->Irp->IoStatus.Status,
341                                 DBGGETSENSECODESTR(Pkt->Srb),
342                                 DBGGETADSENSECODESTR(Pkt->Srb),
343                                 DBGGETADSENSEQUALIFIERSTR(Pkt->Srb)));
344                 break;
345         }
346 
347     }
348 
349 
350     VOID DbgLogSendPacket(TRANSFER_PACKET *Pkt)
351     {
352         PFUNCTIONAL_DEVICE_EXTENSION fdoExt = Pkt->Fdo->DeviceExtension;
353         PCLASS_PRIVATE_FDO_DATA fdoData = fdoExt->PrivateFdoData;
354         KIRQL oldIrql;
355 
356         if (Pkt->OriginalIrp){
357             Pkt->DbgOriginalIrpCopy = *Pkt->OriginalIrp;
358             if (Pkt->OriginalIrp->MdlAddress){
359                 Pkt->DbgMdlCopy = *Pkt->OriginalIrp->MdlAddress;
360             }
361         }
362 
363         KeQueryTickCount(&Pkt->DbgTimeSent);
364         Pkt->DbgTimeReturned.QuadPart = 0L;
365 
366         KeAcquireSpinLock(&fdoData->SpinLock, &oldIrql);
367         fdoData->DbgPacketLogs[fdoData->DbgPacketLogNextIndex] = *Pkt;
368         fdoData->DbgPacketLogNextIndex++;
369         fdoData->DbgPacketLogNextIndex %= DBG_NUM_PACKET_LOG_ENTRIES;
370         KeReleaseSpinLock(&fdoData->SpinLock, oldIrql);
371     }
372 
373     VOID DbgLogReturnPacket(TRANSFER_PACKET *Pkt)
374     {
375         PFUNCTIONAL_DEVICE_EXTENSION fdoExt = Pkt->Fdo->DeviceExtension;
376         PCLASS_PRIVATE_FDO_DATA fdoData = fdoExt->PrivateFdoData;
377         KIRQL oldIrql;
378 
379         KeQueryTickCount(&Pkt->DbgTimeReturned);
380 
381         #if 0
382             // ISSUE: there are some problems with this check (e.g. multiproc), so don't include it yet
383             if (Pkt->OriginalIrp){
384                 /*
385                  *  No one should have touched the original irp while the packet was outstanding,
386                  *  except for a couple fields that we ourselves update during the transfer
387                  *  or that are allowed to change;
388                  *  make those couple fields the same and then to a bytewise compare
389                  */
390                 ULONG lenSame;
391 
392                 Pkt->DbgOriginalIrpCopy.IoStatus.Status = Pkt->OriginalIrp->IoStatus.Status;
393                 Pkt->DbgOriginalIrpCopy.IoStatus.Information = Pkt->OriginalIrp->IoStatus.Information;
394                 Pkt->DbgOriginalIrpCopy.Tail.Overlay.DriverContext[0] = Pkt->OriginalIrp->Tail.Overlay.DriverContext[0];
395                 Pkt->DbgOriginalIrpCopy.ThreadListEntry = Pkt->OriginalIrp->ThreadListEntry;
396                 Pkt->DbgOriginalIrpCopy.Cancel = Pkt->OriginalIrp->Cancel;
397 
398                 lenSame = (ULONG)RtlCompareMemory(Pkt->OriginalIrp, &Pkt->DbgOriginalIrpCopy, sizeof(IRP));
399                 NT_ASSERT(lenSame == sizeof(IRP));
400             }
401         #endif
402 
403         KeAcquireSpinLock(&fdoData->SpinLock, &oldIrql);
404         fdoData->DbgPacketLogs[fdoData->DbgPacketLogNextIndex] = *Pkt;
405         fdoData->DbgPacketLogNextIndex++;
406         fdoData->DbgPacketLogNextIndex %= DBG_NUM_PACKET_LOG_ENTRIES;
407         KeReleaseSpinLock(&fdoData->SpinLock, oldIrql);
408     }
409 
410 
411     /*++////////////////////////////////////////////////////////////////////////////
412 
413     DbgSafeInc()
414 
415     Routine Description:
416 
417         Safely increments a ULONG. If the increment would result in an overflow,
418         the value is unchanged.
419 
420     Arguments:
421 
422         A pointer to the value to be incremented.
423 
424     --*/
425     static VOID DbgSafeInc(PULONG pValue)
426     {
427         ULONG incrementResult;
428         if(NT_SUCCESS(RtlULongAdd(*pValue, 1, &incrementResult))) {
429             *pValue = incrementResult;
430         } else {
431             //
432             // Leave *pValue unchanged (i.e. at ULONG_MAX).
433             //
434         }
435     }
436 
437     VOID DbgLogFlushInfo(PCLASS_PRIVATE_FDO_DATA FdoData, BOOLEAN IsIO, BOOLEAN IsFUA, BOOLEAN IsFlush)
438     {
439 
440         /*
441          *  Reset all FUA/Flush logging fields.
442          */
443         if (FdoData->DbgInitFlushLogging){
444             FdoData->DbgNumIORequests = 0;
445             FdoData->DbgNumFUAs = 0;
446             FdoData->DbgNumFlushes = 0;
447             FdoData->DbgIOsSinceFUA = 0;
448             FdoData->DbgIOsSinceFlush = 0;
449             FdoData->DbgAveIOsToFUA = 0;
450             FdoData->DbgAveIOsToFlush = 0;
451             FdoData->DbgMaxIOsToFUA = 0;
452             FdoData->DbgMaxIOsToFlush = 0;
453             FdoData->DbgMinIOsToFUA = 0xffffffff;
454             FdoData->DbgMinIOsToFlush = 0xffffffff;
455             FdoData->DbgInitFlushLogging = FALSE;
456         }
457 
458         //
459         // Using DbgSafeInc for all increments (instead of ++) guarantees
460         // that there will be no overflow hence no division by 0. All counters
461         // are capped at ULONG_MAX.
462         //
463 
464         if (IsIO){
465             DbgSafeInc(&FdoData->DbgNumIORequests);
466             DbgSafeInc(&FdoData->DbgIOsSinceFlush);
467             if (IsFUA){
468                 if (FdoData->DbgNumFUAs > 0){
469                     FdoData->DbgMinIOsToFUA = min(FdoData->DbgMinIOsToFUA, FdoData->DbgIOsSinceFUA);
470                 }
471                 DbgSafeInc(&FdoData->DbgNumFUAs);
472                 FdoData->DbgAveIOsToFUA =  FdoData->DbgNumIORequests/FdoData->DbgNumFUAs;
473                 FdoData->DbgIOsSinceFUA = 0;
474             }
475             else {
476                 DbgSafeInc(&FdoData->DbgIOsSinceFUA);
477                 FdoData->DbgMaxIOsToFUA = max(FdoData->DbgMaxIOsToFUA, FdoData->DbgIOsSinceFUA);
478             }
479             FdoData->DbgMaxIOsToFlush = max(FdoData->DbgMaxIOsToFlush, FdoData->DbgIOsSinceFlush);
480         }
481         else if (IsFlush){
482             if (FdoData->DbgNumFlushes > 0){
483                 FdoData->DbgMinIOsToFlush = min(FdoData->DbgMinIOsToFlush, FdoData->DbgIOsSinceFlush);
484             }
485             DbgSafeInc(&FdoData->DbgNumFlushes);
486             FdoData->DbgAveIOsToFlush =  FdoData->DbgNumIORequests/FdoData->DbgNumFlushes;
487             FdoData->DbgIOsSinceFlush = 0;
488         }
489 
490     }
491 
492 
493     /*++////////////////////////////////////////////////////////////////////////////
494 
495     SnapDiskStartup()
496 
497     Routine Description:
498 
499         This function will attempt to record the caller responsible for spinning
500         up the disk.
501 
502     Arguments:
503 
504         NONE.
505 
506     Return Value:
507 
508         NONE.
509 
510     --*/
511     VOID
512     SnapDiskStartup(
513         VOID
514     )
515     {
516         ULONG       Index;
517         PDISK_SPINUP_TRACES Entry;
518         LARGE_INTEGER   SpinUpTime;
519 
520 #ifdef _MSC_VER
521 #pragma warning(push)
522 #pragma warning(disable:4210) // nonstandard extension used : function given file scope
523 #endif
524         extern NTSYSAPI USHORT NTAPI RtlCaptureStackBackTrace(
525            _In_         ULONG FramesToSkip,
526            _In_         ULONG FramesToCapture,
527            _Out_writes_to_(FramesToCapture, return) PVOID * BackTrace,
528            _Out_opt_    PULONG BackTraceHash );
529 #ifdef _MSC_VER
530 #pragma warning(pop)
531 #endif
532 
533         //
534         // Grab the current count, then mod it so that it
535         // becomes an index into the DiskSpinupTraces array.
536         //
537         Index = InterlockedIncrement( (volatile LONG *)&DiskSpinupIndex );
538         Index = Index & (NUMBER_OF_DISK_SPINUP_TRACES - 1);
539         Entry = &DiskSpinupTraces[Index];
540 
541         //
542         // Timestamp the instance.
543         //
544         KeQueryTickCount(&SpinUpTime);
545         SpinUpTime.QuadPart = (SpinUpTime.QuadPart * KeQueryTimeIncrement())/(10000000);
546 
547 
548         //
549         // Ask the kernel to read back up our stack by
550         // DISK_SPINUP_BACKTRACE_LENGTH frames.
551         //
552         Entry->TimeStamp.QuadPart = SpinUpTime.QuadPart;
553         RtlZeroMemory( &Entry->StackTrace[0], DISK_SPINUP_BACKTRACE_LENGTH * sizeof(PVOID) );
554         RtlCaptureStackBackTrace( 5,                           // stacks to skip
555                                   DISK_SPINUP_BACKTRACE_LENGTH, // buffer size
556                                   Entry->StackTrace,
557                                   &Index );
558     }
559 
560 #else
561 
562     // We have to keep this in the retail build for legacy.
563     VOID ClassDebugPrint(_In_ CLASS_DEBUG_LEVEL DebugPrintLevel, _In_z_ PCCHAR DebugMessage, ...)
564     {
565         UNREFERENCED_PARAMETER(DebugPrintLevel);
566         UNREFERENCED_PARAMETER(DebugMessage);
567     }
568 
569 #endif
570 
571     char *DbgGetIoctlStr(ULONG ioctl)
572     {
573         char *ioctlStr = "?";
574 
575         switch (ioctl){
576 
577             #undef MAKE_CASE
578             #define MAKE_CASE(ioctlCode) case ioctlCode: ioctlStr = #ioctlCode; break;
579 
580             MAKE_CASE(IOCTL_STORAGE_CHECK_VERIFY)
581             MAKE_CASE(IOCTL_STORAGE_CHECK_VERIFY2)
582             MAKE_CASE(IOCTL_STORAGE_MEDIA_REMOVAL)
583             MAKE_CASE(IOCTL_STORAGE_EJECT_MEDIA)
584             MAKE_CASE(IOCTL_STORAGE_LOAD_MEDIA)
585             MAKE_CASE(IOCTL_STORAGE_LOAD_MEDIA2)
586             MAKE_CASE(IOCTL_STORAGE_RESERVE)
587             MAKE_CASE(IOCTL_STORAGE_RELEASE)
588             MAKE_CASE(IOCTL_STORAGE_PERSISTENT_RESERVE_IN)
589             MAKE_CASE(IOCTL_STORAGE_PERSISTENT_RESERVE_OUT)
590             MAKE_CASE(IOCTL_STORAGE_FIND_NEW_DEVICES)
591             MAKE_CASE(IOCTL_STORAGE_EJECTION_CONTROL)
592             MAKE_CASE(IOCTL_STORAGE_MCN_CONTROL)
593             MAKE_CASE(IOCTL_STORAGE_GET_MEDIA_TYPES)
594             MAKE_CASE(IOCTL_STORAGE_GET_MEDIA_TYPES_EX)
595             MAKE_CASE(IOCTL_STORAGE_GET_MEDIA_SERIAL_NUMBER)
596             MAKE_CASE(IOCTL_STORAGE_GET_HOTPLUG_INFO)
597             MAKE_CASE(IOCTL_STORAGE_RESET_BUS)
598             MAKE_CASE(IOCTL_STORAGE_RESET_DEVICE)
599             MAKE_CASE(IOCTL_STORAGE_GET_DEVICE_NUMBER)
600             MAKE_CASE(IOCTL_STORAGE_PREDICT_FAILURE)
601             MAKE_CASE(IOCTL_STORAGE_QUERY_PROPERTY)
602             MAKE_CASE(OBSOLETE_IOCTL_STORAGE_RESET_BUS)
603             MAKE_CASE(OBSOLETE_IOCTL_STORAGE_RESET_DEVICE)
604         }
605 
606         return ioctlStr;
607     }
608 
609     char *DbgGetScsiOpStr(PSTORAGE_REQUEST_BLOCK_HEADER Srb)
610     {
611         PCDB pCdb = SrbGetCdb(Srb);
612         char *scsiOpStr = "?";
613 
614         if (pCdb) {
615 
616             switch (pCdb->CDB6GENERIC.OperationCode){
617 
618                 #undef MAKE_CASE
619                 #define MAKE_CASE(scsiOpCode) case scsiOpCode: scsiOpStr = #scsiOpCode; break;
620 
621                 MAKE_CASE(SCSIOP_TEST_UNIT_READY)
622                 MAKE_CASE(SCSIOP_REWIND)    // aka SCSIOP_REZERO_UNIT
623                 MAKE_CASE(SCSIOP_REQUEST_BLOCK_ADDR)
624                 MAKE_CASE(SCSIOP_REQUEST_SENSE)
625                 MAKE_CASE(SCSIOP_FORMAT_UNIT)
626                 MAKE_CASE(SCSIOP_READ_BLOCK_LIMITS)
627                 MAKE_CASE(SCSIOP_INIT_ELEMENT_STATUS)   // aka SCSIOP_REASSIGN_BLOCKS
628                 MAKE_CASE(SCSIOP_RECEIVE)       // aka SCSIOP_READ6
629                 MAKE_CASE(SCSIOP_SEND)  // aka SCSIOP_WRITE6, SCSIOP_PRINT
630                 MAKE_CASE(SCSIOP_SLEW_PRINT)    // aka SCSIOP_SEEK6, SCSIOP_TRACK_SELECT
631                 MAKE_CASE(SCSIOP_SEEK_BLOCK)
632                 MAKE_CASE(SCSIOP_PARTITION)
633                 MAKE_CASE(SCSIOP_READ_REVERSE)
634                 MAKE_CASE(SCSIOP_FLUSH_BUFFER)      // aka SCSIOP_WRITE_FILEMARKS
635                 MAKE_CASE(SCSIOP_SPACE)
636                 MAKE_CASE(SCSIOP_INQUIRY)
637                 MAKE_CASE(SCSIOP_VERIFY6)
638                 MAKE_CASE(SCSIOP_RECOVER_BUF_DATA)
639                 MAKE_CASE(SCSIOP_MODE_SELECT)
640                 MAKE_CASE(SCSIOP_RESERVE_UNIT)
641                 MAKE_CASE(SCSIOP_RELEASE_UNIT)
642                 MAKE_CASE(SCSIOP_COPY)
643                 MAKE_CASE(SCSIOP_ERASE)
644                 MAKE_CASE(SCSIOP_MODE_SENSE)
645                 MAKE_CASE(SCSIOP_START_STOP_UNIT)   // aka SCSIOP_STOP_PRINT, SCSIOP_LOAD_UNLOAD
646                 MAKE_CASE(SCSIOP_RECEIVE_DIAGNOSTIC)
647                 MAKE_CASE(SCSIOP_SEND_DIAGNOSTIC)
648                 MAKE_CASE(SCSIOP_MEDIUM_REMOVAL)
649                 MAKE_CASE(SCSIOP_READ_FORMATTED_CAPACITY)
650                 MAKE_CASE(SCSIOP_READ_CAPACITY)
651                 MAKE_CASE(SCSIOP_READ)
652                 MAKE_CASE(SCSIOP_WRITE)
653                 MAKE_CASE(SCSIOP_SEEK)  // aka SCSIOP_LOCATE, SCSIOP_POSITION_TO_ELEMENT
654                 MAKE_CASE(SCSIOP_WRITE_VERIFY)
655                 MAKE_CASE(SCSIOP_VERIFY)
656                 MAKE_CASE(SCSIOP_SEARCH_DATA_HIGH)
657                 MAKE_CASE(SCSIOP_SEARCH_DATA_EQUAL)
658                 MAKE_CASE(SCSIOP_SEARCH_DATA_LOW)
659                 MAKE_CASE(SCSIOP_SET_LIMITS)
660                 MAKE_CASE(SCSIOP_READ_POSITION)
661                 MAKE_CASE(SCSIOP_SYNCHRONIZE_CACHE)
662                 MAKE_CASE(SCSIOP_COMPARE)
663                 MAKE_CASE(SCSIOP_COPY_COMPARE)
664                 MAKE_CASE(SCSIOP_WRITE_DATA_BUFF)
665                 MAKE_CASE(SCSIOP_READ_DATA_BUFF)
666                 MAKE_CASE(SCSIOP_CHANGE_DEFINITION)
667                 MAKE_CASE(SCSIOP_READ_SUB_CHANNEL)
668                 MAKE_CASE(SCSIOP_READ_TOC)
669                 MAKE_CASE(SCSIOP_READ_HEADER)
670                 MAKE_CASE(SCSIOP_PLAY_AUDIO)
671                 MAKE_CASE(SCSIOP_GET_CONFIGURATION)
672                 MAKE_CASE(SCSIOP_PLAY_AUDIO_MSF)
673                 MAKE_CASE(SCSIOP_PLAY_TRACK_INDEX)
674                 MAKE_CASE(SCSIOP_PLAY_TRACK_RELATIVE)
675                 MAKE_CASE(SCSIOP_GET_EVENT_STATUS)
676                 MAKE_CASE(SCSIOP_PAUSE_RESUME)
677                 MAKE_CASE(SCSIOP_LOG_SELECT)
678                 MAKE_CASE(SCSIOP_LOG_SENSE)
679                 MAKE_CASE(SCSIOP_STOP_PLAY_SCAN)
680                 MAKE_CASE(SCSIOP_READ_DISK_INFORMATION)
681                 MAKE_CASE(SCSIOP_READ_TRACK_INFORMATION)
682                 MAKE_CASE(SCSIOP_RESERVE_TRACK_RZONE)
683                 MAKE_CASE(SCSIOP_SEND_OPC_INFORMATION)
684                 MAKE_CASE(SCSIOP_MODE_SELECT10)
685                 MAKE_CASE(SCSIOP_MODE_SENSE10)
686                 MAKE_CASE(SCSIOP_CLOSE_TRACK_SESSION)
687                 MAKE_CASE(SCSIOP_READ_BUFFER_CAPACITY)
688                 MAKE_CASE(SCSIOP_SEND_CUE_SHEET)
689                 MAKE_CASE(SCSIOP_PERSISTENT_RESERVE_IN)
690                 MAKE_CASE(SCSIOP_PERSISTENT_RESERVE_OUT)
691                 MAKE_CASE(SCSIOP_REPORT_LUNS)
692                 MAKE_CASE(SCSIOP_BLANK)
693                 MAKE_CASE(SCSIOP_SEND_KEY)
694                 MAKE_CASE(SCSIOP_REPORT_KEY)
695                 MAKE_CASE(SCSIOP_MOVE_MEDIUM)
696                 MAKE_CASE(SCSIOP_LOAD_UNLOAD_SLOT)  // aka SCSIOP_EXCHANGE_MEDIUM
697                 MAKE_CASE(SCSIOP_SET_READ_AHEAD)
698                 MAKE_CASE(SCSIOP_READ_DVD_STRUCTURE)
699                 MAKE_CASE(SCSIOP_REQUEST_VOL_ELEMENT)
700                 MAKE_CASE(SCSIOP_SEND_VOLUME_TAG)
701                 MAKE_CASE(SCSIOP_READ_ELEMENT_STATUS)
702                 MAKE_CASE(SCSIOP_READ_CD_MSF)
703                 MAKE_CASE(SCSIOP_SCAN_CD)
704                 MAKE_CASE(SCSIOP_SET_CD_SPEED)
705                 MAKE_CASE(SCSIOP_PLAY_CD)
706                 MAKE_CASE(SCSIOP_MECHANISM_STATUS)
707                 MAKE_CASE(SCSIOP_READ_CD)
708                 MAKE_CASE(SCSIOP_SEND_DVD_STRUCTURE)
709                 MAKE_CASE(SCSIOP_INIT_ELEMENT_RANGE)
710                 MAKE_CASE(SCSIOP_READ16)
711                 MAKE_CASE(SCSIOP_WRITE16)
712                 MAKE_CASE(SCSIOP_VERIFY16)
713                 MAKE_CASE(SCSIOP_SYNCHRONIZE_CACHE16)
714                 MAKE_CASE(SCSIOP_READ_CAPACITY16)
715             }
716         }
717 
718         return scsiOpStr;
719     }
720 
721 
722     char *DbgGetSrbStatusStr(PSTORAGE_REQUEST_BLOCK_HEADER Srb)
723     {
724         char *srbStatStr = "?";
725 
726         switch (Srb->SrbStatus){
727 
728             #undef MAKE_CASE
729             #define MAKE_CASE(srbStat) \
730                         case srbStat: \
731                             srbStatStr = #srbStat; \
732                             break; \
733                         case srbStat|SRB_STATUS_QUEUE_FROZEN: \
734                             srbStatStr = #srbStat "|SRB_STATUS_QUEUE_FROZEN"; \
735                             break; \
736                         case srbStat|SRB_STATUS_AUTOSENSE_VALID: \
737                             srbStatStr = #srbStat "|SRB_STATUS_AUTOSENSE_VALID"; \
738                             break; \
739                         case srbStat|SRB_STATUS_QUEUE_FROZEN|SRB_STATUS_AUTOSENSE_VALID: \
740                             srbStatStr = #srbStat "|SRB_STATUS_QUEUE_FROZEN|SRB_STATUS_AUTOSENSE_VALID"; \
741                             break;
742 
743             MAKE_CASE(SRB_STATUS_PENDING)
744             MAKE_CASE(SRB_STATUS_SUCCESS)
745             MAKE_CASE(SRB_STATUS_ABORTED)
746             MAKE_CASE(SRB_STATUS_ABORT_FAILED)
747             MAKE_CASE(SRB_STATUS_ERROR)
748             MAKE_CASE(SRB_STATUS_BUSY)
749             MAKE_CASE(SRB_STATUS_INVALID_REQUEST)
750             MAKE_CASE(SRB_STATUS_INVALID_PATH_ID)
751             MAKE_CASE(SRB_STATUS_NO_DEVICE)
752             MAKE_CASE(SRB_STATUS_TIMEOUT)
753             MAKE_CASE(SRB_STATUS_SELECTION_TIMEOUT)
754             MAKE_CASE(SRB_STATUS_COMMAND_TIMEOUT)
755             MAKE_CASE(SRB_STATUS_MESSAGE_REJECTED)
756             MAKE_CASE(SRB_STATUS_BUS_RESET)
757             MAKE_CASE(SRB_STATUS_PARITY_ERROR)
758             MAKE_CASE(SRB_STATUS_REQUEST_SENSE_FAILED)
759             MAKE_CASE(SRB_STATUS_NO_HBA)
760             MAKE_CASE(SRB_STATUS_DATA_OVERRUN)
761             MAKE_CASE(SRB_STATUS_UNEXPECTED_BUS_FREE)
762             MAKE_CASE(SRB_STATUS_PHASE_SEQUENCE_FAILURE)
763             MAKE_CASE(SRB_STATUS_BAD_SRB_BLOCK_LENGTH)
764             MAKE_CASE(SRB_STATUS_REQUEST_FLUSHED)
765             MAKE_CASE(SRB_STATUS_INVALID_LUN)
766             MAKE_CASE(SRB_STATUS_INVALID_TARGET_ID)
767             MAKE_CASE(SRB_STATUS_BAD_FUNCTION)
768             MAKE_CASE(SRB_STATUS_ERROR_RECOVERY)
769             MAKE_CASE(SRB_STATUS_NOT_POWERED)
770             MAKE_CASE(SRB_STATUS_INTERNAL_ERROR)
771         }
772 
773         return srbStatStr;
774     }
775 
776 
777     char *DbgGetSenseCodeStr(PSTORAGE_REQUEST_BLOCK_HEADER Srb)
778     {
779         char *senseCodeStr = "?";
780 
781         if (Srb->SrbStatus & SRB_STATUS_AUTOSENSE_VALID){
782 
783             PVOID senseData;
784             UCHAR senseCode;
785             BOOLEAN validSense;
786 
787             senseData = SrbGetSenseInfoBuffer(Srb);
788             NT_ASSERT(senseData);
789 
790             validSense = ScsiGetSenseKeyAndCodes(senseData,
791                                                  SrbGetSenseInfoBufferLength(Srb),
792                                                  SCSI_SENSE_OPTIONS_FIXED_FORMAT_IF_UNKNOWN_FORMAT_INDICATED,
793                                                  &senseCode,
794                                                  NULL,
795                                                  NULL);
796             if (validSense) {
797                 switch (senseCode){
798 
799                     #undef MAKE_CASE
800                     #define MAKE_CASE(snsCod) case snsCod: senseCodeStr = #snsCod; break;
801 
802                     MAKE_CASE(SCSI_SENSE_NO_SENSE)
803                     MAKE_CASE(SCSI_SENSE_RECOVERED_ERROR)
804                     MAKE_CASE(SCSI_SENSE_NOT_READY)
805                     MAKE_CASE(SCSI_SENSE_MEDIUM_ERROR)
806                     MAKE_CASE(SCSI_SENSE_HARDWARE_ERROR)
807                     MAKE_CASE(SCSI_SENSE_ILLEGAL_REQUEST)
808                     MAKE_CASE(SCSI_SENSE_UNIT_ATTENTION)
809                     MAKE_CASE(SCSI_SENSE_DATA_PROTECT)
810                     MAKE_CASE(SCSI_SENSE_BLANK_CHECK)
811                     MAKE_CASE(SCSI_SENSE_UNIQUE)
812                     MAKE_CASE(SCSI_SENSE_COPY_ABORTED)
813                     MAKE_CASE(SCSI_SENSE_ABORTED_COMMAND)
814                     MAKE_CASE(SCSI_SENSE_EQUAL)
815                     MAKE_CASE(SCSI_SENSE_VOL_OVERFLOW)
816                     MAKE_CASE(SCSI_SENSE_MISCOMPARE)
817                     MAKE_CASE(SCSI_SENSE_RESERVED)
818                 }
819             }
820         }
821 
822         return senseCodeStr;
823     }
824 
825 
826     char *DbgGetAdditionalSenseCodeStr(PSTORAGE_REQUEST_BLOCK_HEADER Srb)
827     {
828         char *adSenseCodeStr = "?";
829 
830         if (Srb->SrbStatus & SRB_STATUS_AUTOSENSE_VALID){
831             PVOID senseData;
832             UCHAR adSenseCode;
833             BOOLEAN validSense;
834 
835             senseData = SrbGetSenseInfoBuffer(Srb);
836             NT_ASSERT(senseData);
837 
838             validSense = ScsiGetSenseKeyAndCodes(senseData,
839                                                  SrbGetSenseInfoBufferLength(Srb),
840                                                  SCSI_SENSE_OPTIONS_FIXED_FORMAT_IF_UNKNOWN_FORMAT_INDICATED,
841                                                  NULL,
842                                                  &adSenseCode,
843                                                  NULL);
844 
845             if (validSense) {
846                 switch (adSenseCode){
847 
848                     #undef MAKE_CASE
849                     #define MAKE_CASE(adSnsCod) case adSnsCod: adSenseCodeStr = #adSnsCod; break;
850 
851                     MAKE_CASE(SCSI_ADSENSE_NO_SENSE)
852                     MAKE_CASE(SCSI_ADSENSE_LUN_NOT_READY)
853                     MAKE_CASE(SCSI_ADSENSE_TRACK_ERROR)
854                     MAKE_CASE(SCSI_ADSENSE_SEEK_ERROR)
855                     MAKE_CASE(SCSI_ADSENSE_REC_DATA_NOECC)
856                     MAKE_CASE(SCSI_ADSENSE_REC_DATA_ECC)
857                     MAKE_CASE(SCSI_ADSENSE_ILLEGAL_COMMAND)
858                     MAKE_CASE(SCSI_ADSENSE_ILLEGAL_BLOCK)
859                     MAKE_CASE(SCSI_ADSENSE_INVALID_CDB)
860                     MAKE_CASE(SCSI_ADSENSE_INVALID_LUN)
861                     MAKE_CASE(SCSI_ADSENSE_WRITE_PROTECT)   // aka SCSI_ADWRITE_PROTECT
862                     MAKE_CASE(SCSI_ADSENSE_MEDIUM_CHANGED)
863                     MAKE_CASE(SCSI_ADSENSE_BUS_RESET)
864                     MAKE_CASE(SCSI_ADSENSE_INVALID_MEDIA)
865                     MAKE_CASE(SCSI_ADSENSE_NO_MEDIA_IN_DEVICE)
866                     MAKE_CASE(SCSI_ADSENSE_POSITION_ERROR)
867                     MAKE_CASE(SCSI_ADSENSE_OPERATOR_REQUEST)
868                     MAKE_CASE(SCSI_ADSENSE_FAILURE_PREDICTION_THRESHOLD_EXCEEDED)
869                     MAKE_CASE(SCSI_ADSENSE_COPY_PROTECTION_FAILURE)
870                     MAKE_CASE(SCSI_ADSENSE_VENDOR_UNIQUE)
871                     MAKE_CASE(SCSI_ADSENSE_MUSIC_AREA)
872                     MAKE_CASE(SCSI_ADSENSE_DATA_AREA)
873                     MAKE_CASE(SCSI_ADSENSE_VOLUME_OVERFLOW)
874                 }
875             }
876         }
877 
878         return adSenseCodeStr;
879     }
880 
881 
882     char *DbgGetAdditionalSenseCodeQualifierStr(PSTORAGE_REQUEST_BLOCK_HEADER Srb)
883     {
884         char *adSenseCodeQualStr = "?";
885 
886         if (Srb->SrbStatus & SRB_STATUS_AUTOSENSE_VALID){
887             PVOID senseData;
888             UCHAR adSenseCode;
889             UCHAR adSenseCodeQual;
890             BOOLEAN validSense;
891 
892             senseData = SrbGetSenseInfoBuffer(Srb);
893             NT_ASSERT(senseData);
894 
895             validSense = ScsiGetSenseKeyAndCodes(senseData,
896                                                  SrbGetSenseInfoBufferLength(Srb),
897                                                  SCSI_SENSE_OPTIONS_FIXED_FORMAT_IF_UNKNOWN_FORMAT_INDICATED,
898                                                  NULL,
899                                                  &adSenseCode,
900                                                  &adSenseCodeQual);
901             if (validSense) {
902                 switch (adSenseCode){
903 
904                     #undef MAKE_CASE
905                     #define MAKE_CASE(adSnsCodQual) case adSnsCodQual: adSenseCodeQualStr = #adSnsCodQual; break;
906 
907                     case SCSI_ADSENSE_LUN_NOT_READY:
908                         switch (adSenseCodeQual){
909                             MAKE_CASE(SCSI_SENSEQ_CAUSE_NOT_REPORTABLE)
910                             MAKE_CASE(SCSI_SENSEQ_BECOMING_READY)
911                             MAKE_CASE(SCSI_SENSEQ_INIT_COMMAND_REQUIRED)
912                             MAKE_CASE(SCSI_SENSEQ_MANUAL_INTERVENTION_REQUIRED)
913                             MAKE_CASE(SCSI_SENSEQ_FORMAT_IN_PROGRESS)
914                             MAKE_CASE(SCSI_SENSEQ_REBUILD_IN_PROGRESS)
915                             MAKE_CASE(SCSI_SENSEQ_RECALCULATION_IN_PROGRESS)
916                             MAKE_CASE(SCSI_SENSEQ_OPERATION_IN_PROGRESS)
917                             MAKE_CASE(SCSI_SENSEQ_LONG_WRITE_IN_PROGRESS)
918                         }
919                         break;
920                     case SCSI_ADSENSE_NO_SENSE:
921                         switch (adSenseCodeQual){
922                             MAKE_CASE(SCSI_SENSEQ_FILEMARK_DETECTED)
923                             MAKE_CASE(SCSI_SENSEQ_END_OF_MEDIA_DETECTED)
924                             MAKE_CASE(SCSI_SENSEQ_SETMARK_DETECTED)
925                             MAKE_CASE(SCSI_SENSEQ_BEGINNING_OF_MEDIA_DETECTED)
926                         }
927                         break;
928                     case SCSI_ADSENSE_ILLEGAL_BLOCK:
929                         switch (adSenseCodeQual){
930                             MAKE_CASE(SCSI_SENSEQ_ILLEGAL_ELEMENT_ADDR)
931                         }
932                         break;
933                     case SCSI_ADSENSE_POSITION_ERROR:
934                         switch (adSenseCodeQual){
935                             MAKE_CASE(SCSI_SENSEQ_DESTINATION_FULL)
936                             MAKE_CASE(SCSI_SENSEQ_SOURCE_EMPTY)
937                         }
938                         break;
939                     case SCSI_ADSENSE_INVALID_MEDIA:
940                         switch (adSenseCodeQual){
941                             MAKE_CASE(SCSI_SENSEQ_INCOMPATIBLE_MEDIA_INSTALLED)
942                             MAKE_CASE(SCSI_SENSEQ_UNKNOWN_FORMAT)
943                             MAKE_CASE(SCSI_SENSEQ_INCOMPATIBLE_FORMAT)
944                             MAKE_CASE(SCSI_SENSEQ_CLEANING_CARTRIDGE_INSTALLED)
945                         }
946                         break;
947                     case SCSI_ADSENSE_OPERATOR_REQUEST:
948                         switch (adSenseCodeQual){
949                             MAKE_CASE(SCSI_SENSEQ_STATE_CHANGE_INPUT)
950                             MAKE_CASE(SCSI_SENSEQ_MEDIUM_REMOVAL)
951                             MAKE_CASE(SCSI_SENSEQ_WRITE_PROTECT_ENABLE)
952                             MAKE_CASE(SCSI_SENSEQ_WRITE_PROTECT_DISABLE)
953                         }
954                         break;
955                     case SCSI_ADSENSE_COPY_PROTECTION_FAILURE:
956                         switch (adSenseCodeQual){
957                             MAKE_CASE(SCSI_SENSEQ_AUTHENTICATION_FAILURE)
958                             MAKE_CASE(SCSI_SENSEQ_KEY_NOT_PRESENT)
959                             MAKE_CASE(SCSI_SENSEQ_KEY_NOT_ESTABLISHED)
960                             MAKE_CASE(SCSI_SENSEQ_READ_OF_SCRAMBLED_SECTOR_WITHOUT_AUTHENTICATION)
961                             MAKE_CASE(SCSI_SENSEQ_MEDIA_CODE_MISMATCHED_TO_LOGICAL_UNIT)
962                             MAKE_CASE(SCSI_SENSEQ_LOGICAL_UNIT_RESET_COUNT_ERROR)
963                         }
964                         break;
965                 }
966             }
967         }
968 
969         return adSenseCodeQualStr;
970     }
971 
972 
973