mirror of
https://github.com/reactos/reactos.git
synced 2024-12-29 10:35:28 +00:00
9167 lines
323 KiB
C
9167 lines
323 KiB
C
/*++
|
|
|
|
Copyright (C) Microsoft Corporation, 1991 - 2010
|
|
|
|
Module Name:
|
|
|
|
utils.c
|
|
|
|
Abstract:
|
|
|
|
SCSI class driver routines
|
|
|
|
Environment:
|
|
|
|
kernel mode only
|
|
|
|
Notes:
|
|
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
#include "classp.h"
|
|
#include "debug.h"
|
|
#include <ntiologc.h>
|
|
|
|
|
|
#ifdef DEBUG_USE_WPP
|
|
#include "utils.tmh"
|
|
#endif
|
|
|
|
//
|
|
// Constant value used in firmware upgrade process.
|
|
//
|
|
#define FIRMWARE_ACTIVATE_TIMEOUT_VALUE 30
|
|
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE, ClassGetDeviceParameter)
|
|
#pragma alloc_text(PAGE, ClassScanForSpecial)
|
|
#pragma alloc_text(PAGE, ClassSetDeviceParameter)
|
|
#pragma alloc_text(PAGE, ClasspMyStringMatches)
|
|
#pragma alloc_text(PAGE, ClasspDeviceCopyOffloadProperty)
|
|
#pragma alloc_text(PAGE, ClasspValidateOffloadSupported)
|
|
#pragma alloc_text(PAGE, ClasspValidateOffloadInputParameters)
|
|
#endif
|
|
|
|
// custom string match -- careful!
|
|
BOOLEAN ClasspMyStringMatches(_In_opt_z_ PCHAR StringToMatch, _In_z_ PCHAR TargetString)
|
|
{
|
|
ULONG length; // strlen returns an int, not size_t (!)
|
|
PAGED_CODE();
|
|
NT_ASSERT(TargetString);
|
|
// if no match requested, return TRUE
|
|
if (StringToMatch == NULL) {
|
|
return TRUE;
|
|
}
|
|
// cache the string length for efficiency
|
|
length = (ULONG)strlen(StringToMatch);
|
|
// ZERO-length strings may only match zero-length strings
|
|
if (length == 0) {
|
|
return (strlen(TargetString) == 0);
|
|
}
|
|
// strncmp returns zero if the strings match
|
|
return (strncmp(StringToMatch, TargetString, length) == 0);
|
|
}
|
|
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassGetDeviceParameter(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_opt_ PWSTR SubkeyName,
|
|
_In_ PWSTR ParameterName,
|
|
_Inout_ PULONG ParameterValue // also default value
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
RTL_QUERY_REGISTRY_TABLE queryTable[2] = {0};
|
|
HANDLE deviceParameterHandle = NULL;
|
|
HANDLE deviceSubkeyHandle = NULL;
|
|
ULONG defaultParameterValue;
|
|
|
|
PAGED_CODE();
|
|
|
|
//
|
|
// open the given parameter
|
|
//
|
|
|
|
status = IoOpenDeviceRegistryKey(FdoExtension->LowerPdo,
|
|
PLUGPLAY_REGKEY_DEVICE,
|
|
KEY_READ,
|
|
&deviceParameterHandle);
|
|
|
|
if (NT_SUCCESS(status) && (SubkeyName != NULL)) {
|
|
|
|
UNICODE_STRING subkeyName;
|
|
OBJECT_ATTRIBUTES objectAttributes = {0};
|
|
|
|
RtlInitUnicodeString(&subkeyName, SubkeyName);
|
|
InitializeObjectAttributes(&objectAttributes,
|
|
&subkeyName,
|
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
|
deviceParameterHandle,
|
|
NULL);
|
|
|
|
status = ZwOpenKey(&deviceSubkeyHandle,
|
|
KEY_READ,
|
|
&objectAttributes);
|
|
if (!NT_SUCCESS(status)) {
|
|
ZwClose(deviceParameterHandle);
|
|
}
|
|
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
defaultParameterValue = *ParameterValue;
|
|
|
|
queryTable->Flags = RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED | RTL_QUERY_REGISTRY_TYPECHECK;
|
|
queryTable->Name = ParameterName;
|
|
queryTable->EntryContext = ParameterValue;
|
|
queryTable->DefaultType = (REG_DWORD << RTL_QUERY_REGISTRY_TYPECHECK_SHIFT) | REG_NONE;
|
|
queryTable->DefaultData = NULL;
|
|
queryTable->DefaultLength = 0;
|
|
|
|
status = RtlQueryRegistryValues(RTL_REGISTRY_HANDLE,
|
|
(PWSTR)(SubkeyName ?
|
|
deviceSubkeyHandle :
|
|
deviceParameterHandle),
|
|
queryTable,
|
|
NULL,
|
|
NULL);
|
|
if (!NT_SUCCESS(status)) {
|
|
*ParameterValue = defaultParameterValue; // use default value
|
|
}
|
|
|
|
//
|
|
// close what we open
|
|
//
|
|
|
|
if (SubkeyName) {
|
|
ZwClose(deviceSubkeyHandle);
|
|
}
|
|
|
|
ZwClose(deviceParameterHandle);
|
|
}
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
|
|
//
|
|
// Windows 2000 SP3 uses the driver-specific key, so look in there
|
|
//
|
|
|
|
status = IoOpenDeviceRegistryKey(FdoExtension->LowerPdo,
|
|
PLUGPLAY_REGKEY_DRIVER,
|
|
KEY_READ,
|
|
&deviceParameterHandle);
|
|
|
|
if (NT_SUCCESS(status) && (SubkeyName != NULL)) {
|
|
|
|
UNICODE_STRING subkeyName;
|
|
OBJECT_ATTRIBUTES objectAttributes = {0};
|
|
|
|
RtlInitUnicodeString(&subkeyName, SubkeyName);
|
|
InitializeObjectAttributes(&objectAttributes,
|
|
&subkeyName,
|
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
|
deviceParameterHandle,
|
|
NULL);
|
|
|
|
status = ZwOpenKey(&deviceSubkeyHandle, KEY_READ, &objectAttributes);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
ZwClose(deviceParameterHandle);
|
|
}
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
defaultParameterValue = *ParameterValue;
|
|
|
|
queryTable->Flags = RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED | RTL_QUERY_REGISTRY_TYPECHECK;
|
|
queryTable->Name = ParameterName;
|
|
queryTable->EntryContext = ParameterValue;
|
|
queryTable->DefaultType = (REG_DWORD << RTL_QUERY_REGISTRY_TYPECHECK_SHIFT) | REG_NONE;
|
|
queryTable->DefaultData = NULL;
|
|
queryTable->DefaultLength = 0;
|
|
|
|
status = RtlQueryRegistryValues(RTL_REGISTRY_HANDLE,
|
|
(PWSTR)(SubkeyName ?
|
|
deviceSubkeyHandle :
|
|
deviceParameterHandle),
|
|
queryTable,
|
|
NULL,
|
|
NULL);
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
//
|
|
// Migrate the value over to the device-specific key
|
|
//
|
|
|
|
ClassSetDeviceParameter(FdoExtension, SubkeyName, ParameterName, *ParameterValue);
|
|
|
|
} else {
|
|
|
|
//
|
|
// Use the default value
|
|
//
|
|
|
|
*ParameterValue = defaultParameterValue;
|
|
}
|
|
|
|
if (SubkeyName) {
|
|
ZwClose(deviceSubkeyHandle);
|
|
}
|
|
|
|
ZwClose(deviceParameterHandle);
|
|
}
|
|
}
|
|
|
|
return;
|
|
|
|
} // end ClassGetDeviceParameter()
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
NTSTATUS
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassSetDeviceParameter(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_opt_ PWSTR SubkeyName,
|
|
_In_ PWSTR ParameterName,
|
|
_In_ ULONG ParameterValue)
|
|
{
|
|
NTSTATUS status;
|
|
HANDLE deviceParameterHandle = NULL;
|
|
HANDLE deviceSubkeyHandle = NULL;
|
|
|
|
PAGED_CODE();
|
|
|
|
//
|
|
// open the given parameter
|
|
//
|
|
|
|
status = IoOpenDeviceRegistryKey(FdoExtension->LowerPdo,
|
|
PLUGPLAY_REGKEY_DEVICE,
|
|
KEY_READ | KEY_WRITE,
|
|
&deviceParameterHandle);
|
|
|
|
if (NT_SUCCESS(status) && (SubkeyName != NULL)) {
|
|
|
|
UNICODE_STRING subkeyName;
|
|
OBJECT_ATTRIBUTES objectAttributes;
|
|
|
|
RtlInitUnicodeString(&subkeyName, SubkeyName);
|
|
InitializeObjectAttributes(&objectAttributes,
|
|
&subkeyName,
|
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
|
deviceParameterHandle,
|
|
NULL);
|
|
|
|
status = ZwCreateKey(&deviceSubkeyHandle,
|
|
KEY_READ | KEY_WRITE,
|
|
&objectAttributes,
|
|
0, NULL, 0, NULL);
|
|
if (!NT_SUCCESS(status)) {
|
|
ZwClose(deviceParameterHandle);
|
|
}
|
|
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
status = RtlWriteRegistryValue(
|
|
RTL_REGISTRY_HANDLE,
|
|
(PWSTR) (SubkeyName ?
|
|
deviceSubkeyHandle :
|
|
deviceParameterHandle),
|
|
ParameterName,
|
|
REG_DWORD,
|
|
&ParameterValue,
|
|
sizeof(ULONG));
|
|
|
|
//
|
|
// close what we open
|
|
//
|
|
|
|
if (SubkeyName) {
|
|
ZwClose(deviceSubkeyHandle);
|
|
}
|
|
|
|
ZwClose(deviceParameterHandle);
|
|
}
|
|
|
|
return status;
|
|
|
|
} // end ClassSetDeviceParameter()
|
|
|
|
|
|
/*
|
|
* ClassScanForSpecial
|
|
*
|
|
* This routine was written to simplify scanning for special
|
|
* hardware based upon id strings. it does not check the registry.
|
|
*/
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassScanForSpecial(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_ CLASSPNP_SCAN_FOR_SPECIAL_INFO DeviceList[],
|
|
_In_ PCLASS_SCAN_FOR_SPECIAL_HANDLER Function)
|
|
{
|
|
PSTORAGE_DEVICE_DESCRIPTOR deviceDescriptor;
|
|
PUCHAR vendorId;
|
|
PUCHAR productId;
|
|
PUCHAR productRevision;
|
|
UCHAR nullString[] = "";
|
|
|
|
PAGED_CODE();
|
|
NT_ASSERT(DeviceList);
|
|
NT_ASSERT(Function);
|
|
|
|
deviceDescriptor = FdoExtension->DeviceDescriptor;
|
|
|
|
if (DeviceList == NULL) {
|
|
return;
|
|
}
|
|
if (Function == NULL) {
|
|
return;
|
|
}
|
|
|
|
//
|
|
// SCSI sets offsets to -1, ATAPI sets to 0. check for both.
|
|
//
|
|
|
|
if (deviceDescriptor->VendorIdOffset != 0 &&
|
|
deviceDescriptor->VendorIdOffset != -1) {
|
|
vendorId = ((PUCHAR)deviceDescriptor);
|
|
vendorId += deviceDescriptor->VendorIdOffset;
|
|
} else {
|
|
vendorId = nullString;
|
|
}
|
|
if (deviceDescriptor->ProductIdOffset != 0 &&
|
|
deviceDescriptor->ProductIdOffset != -1) {
|
|
productId = ((PUCHAR)deviceDescriptor);
|
|
productId += deviceDescriptor->ProductIdOffset;
|
|
} else {
|
|
productId = nullString;
|
|
}
|
|
if (deviceDescriptor->ProductRevisionOffset != 0 &&
|
|
deviceDescriptor->ProductRevisionOffset != -1) {
|
|
productRevision = ((PUCHAR)deviceDescriptor);
|
|
productRevision += deviceDescriptor->ProductRevisionOffset;
|
|
} else {
|
|
productRevision = nullString;
|
|
}
|
|
|
|
//
|
|
// loop while the device list is valid (not null-filled)
|
|
//
|
|
|
|
for (;(DeviceList->VendorId != NULL ||
|
|
DeviceList->ProductId != NULL ||
|
|
DeviceList->ProductRevision != NULL);DeviceList++) {
|
|
|
|
if (ClasspMyStringMatches(DeviceList->VendorId, (PCHAR)vendorId) &&
|
|
ClasspMyStringMatches(DeviceList->ProductId, (PCHAR)productId) &&
|
|
ClasspMyStringMatches(DeviceList->ProductRevision, (PCHAR)productRevision)
|
|
) {
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_INIT, "ClasspScanForSpecialByInquiry: Found matching "
|
|
"controller Ven: %s Prod: %s Rev: %s\n",
|
|
(PCSZ)vendorId, (PCSZ)productId, (PCSZ)productRevision));
|
|
|
|
//
|
|
// pass the context to the call back routine and exit
|
|
//
|
|
|
|
(Function)(FdoExtension, DeviceList->Data);
|
|
|
|
//
|
|
// for CHK builds, try to prevent wierd stacks by having a debug
|
|
// print here. it's a hack, but i know of no other way to prevent
|
|
// the stack from being wrong.
|
|
//
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_INIT, "ClasspScanForSpecialByInquiry: "
|
|
"completed callback\n"));
|
|
return;
|
|
|
|
} // else the strings did not match
|
|
|
|
} // none of the devices matched.
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_INIT, "ClasspScanForSpecialByInquiry: no match found for %p\n",
|
|
FdoExtension->DeviceObject));
|
|
return;
|
|
|
|
} // end ClasspScanForSpecialByInquiry()
|
|
|
|
|
|
//
|
|
// In order to provide better performance without the need to reboot,
|
|
// we need to implement a self-adjusting method to set and clear the
|
|
// srb flags based upon current performance.
|
|
//
|
|
// whenever there is an error, immediately grab the spin lock. the
|
|
// MP perf hit here is acceptable, since we're in an error path. this
|
|
// is also neccessary because we are guaranteed to be modifying the
|
|
// SRB flags here, setting SuccessfulIO to zero, and incrementing the
|
|
// actual error count (which is always done within this spinlock).
|
|
//
|
|
// whenever there is no error, increment a counter. if there have been
|
|
// errors on the device, and we've enabled dynamic perf, *and* we've
|
|
// just crossed the perf threshhold, then grab the spin lock and
|
|
// double check that the threshhold has, indeed been hit(*). then
|
|
// decrement the error count, and if it's dropped sufficiently, undo
|
|
// some of the safety changes made in the SRB flags due to the errors.
|
|
//
|
|
// * this works in all cases. even if lots of ios occur after the
|
|
// previous guy went in and cleared the successfulio counter, that
|
|
// just means that we've hit the threshhold again, and so it's proper
|
|
// to run the inner loop again.
|
|
//
|
|
|
|
VOID
|
|
ClasspPerfIncrementErrorCount(
|
|
IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension
|
|
)
|
|
{
|
|
PCLASS_PRIVATE_FDO_DATA fdoData = FdoExtension->PrivateFdoData;
|
|
KIRQL oldIrql;
|
|
ULONG errors;
|
|
|
|
KeAcquireSpinLock(&fdoData->SpinLock, &oldIrql);
|
|
|
|
fdoData->Perf.SuccessfulIO = 0; // implicit interlock
|
|
errors = InterlockedIncrement((volatile LONG *)&FdoExtension->ErrorCount);
|
|
|
|
if (!fdoData->DisableThrottling) {
|
|
|
|
if (errors >= CLASS_ERROR_LEVEL_1) {
|
|
|
|
//
|
|
// If the error count has exceeded the error limit, then disable
|
|
// any tagged queuing, multiple requests per lu queueing
|
|
// and sychronous data transfers.
|
|
//
|
|
// Clearing the no queue freeze flag prevents the port driver
|
|
// from sending multiple requests per logical unit.
|
|
//
|
|
|
|
CLEAR_FLAG(FdoExtension->SrbFlags, SRB_FLAGS_NO_QUEUE_FREEZE);
|
|
CLEAR_FLAG(FdoExtension->SrbFlags, SRB_FLAGS_QUEUE_ACTION_ENABLE);
|
|
|
|
SET_FLAG(FdoExtension->SrbFlags, SRB_FLAGS_DISABLE_SYNCH_TRANSFER);
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "ClasspPerfIncrementErrorCount: "
|
|
"Too many errors; disabling tagged queuing and "
|
|
"synchronous data tranfers.\n"));
|
|
|
|
}
|
|
|
|
if (errors >= CLASS_ERROR_LEVEL_2) {
|
|
|
|
//
|
|
// If a second threshold is reached, disable disconnects.
|
|
//
|
|
|
|
SET_FLAG(FdoExtension->SrbFlags, SRB_FLAGS_DISABLE_DISCONNECT);
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "ClasspPerfIncrementErrorCount: "
|
|
"Too many errors; disabling disconnects.\n"));
|
|
}
|
|
}
|
|
|
|
KeReleaseSpinLock(&fdoData->SpinLock, oldIrql);
|
|
return;
|
|
}
|
|
|
|
VOID
|
|
ClasspPerfIncrementSuccessfulIo(
|
|
IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension
|
|
)
|
|
{
|
|
PCLASS_PRIVATE_FDO_DATA fdoData = FdoExtension->PrivateFdoData;
|
|
KIRQL oldIrql;
|
|
ULONG errors;
|
|
ULONG succeeded = 0;
|
|
|
|
//
|
|
// don't take a hit from the interlocked op unless we're in
|
|
// a degraded state and we've got a threshold to hit.
|
|
//
|
|
|
|
if (FdoExtension->ErrorCount == 0) {
|
|
return;
|
|
}
|
|
|
|
if (fdoData->Perf.ReEnableThreshhold == 0) {
|
|
return;
|
|
}
|
|
|
|
succeeded = InterlockedIncrement((volatile LONG *)&fdoData->Perf.SuccessfulIO);
|
|
if (succeeded < fdoData->Perf.ReEnableThreshhold) {
|
|
return;
|
|
}
|
|
|
|
//
|
|
// if we hit the threshold, grab the spinlock and verify we've
|
|
// actually done so. this allows us to ignore the spinlock 99%
|
|
// of the time.
|
|
//
|
|
|
|
KeAcquireSpinLock(&fdoData->SpinLock, &oldIrql);
|
|
|
|
//
|
|
// re-read the value, so we don't run this multiple times
|
|
// for a single threshhold being hit. this keeps errorcount
|
|
// somewhat useful.
|
|
//
|
|
|
|
succeeded = fdoData->Perf.SuccessfulIO;
|
|
|
|
if ((FdoExtension->ErrorCount != 0) &&
|
|
(fdoData->Perf.ReEnableThreshhold <= succeeded)
|
|
) {
|
|
|
|
fdoData->Perf.SuccessfulIO = 0; // implicit interlock
|
|
|
|
NT_ASSERT(FdoExtension->ErrorCount > 0);
|
|
errors = InterlockedDecrement((volatile LONG *)&FdoExtension->ErrorCount);
|
|
|
|
//
|
|
// note: do in reverse order of the sets "just in case"
|
|
//
|
|
|
|
if (errors < CLASS_ERROR_LEVEL_2) {
|
|
if (errors == CLASS_ERROR_LEVEL_2 - 1) {
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "ClasspPerfIncrementSuccessfulIo: "
|
|
"Error level 2 no longer required.\n"));
|
|
}
|
|
if (!TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_DISABLE_DISCONNECT)) {
|
|
CLEAR_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_DISABLE_DISCONNECT);
|
|
}
|
|
}
|
|
|
|
if (errors < CLASS_ERROR_LEVEL_1) {
|
|
if (errors == CLASS_ERROR_LEVEL_1 - 1) {
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "ClasspPerfIncrementSuccessfulIo: "
|
|
"Error level 1 no longer required.\n"));
|
|
}
|
|
if (!TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_DISABLE_SYNCH_TRANSFER)) {
|
|
CLEAR_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_DISABLE_SYNCH_TRANSFER);
|
|
}
|
|
if (TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_QUEUE_ACTION_ENABLE)) {
|
|
SET_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_QUEUE_ACTION_ENABLE);
|
|
}
|
|
if (TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_NO_QUEUE_FREEZE)) {
|
|
SET_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_NO_QUEUE_FREEZE);
|
|
}
|
|
}
|
|
} // end of threshhold definitely being hit for first time
|
|
|
|
KeReleaseSpinLock(&fdoData->SpinLock, oldIrql);
|
|
return;
|
|
}
|
|
|
|
|
|
PMDL ClasspBuildDeviceMdl(PVOID Buffer, ULONG BufferLen, BOOLEAN WriteToDevice)
|
|
{
|
|
PMDL mdl;
|
|
|
|
mdl = IoAllocateMdl(Buffer, BufferLen, FALSE, FALSE, NULL);
|
|
if (mdl){
|
|
_SEH2_TRY {
|
|
MmProbeAndLockPages(mdl, KernelMode, WriteToDevice ? IoReadAccess : IoWriteAccess);
|
|
#ifdef _MSC_VER
|
|
#pragma warning(suppress: 6320) // We want to handle any exception that MmProbeAndLockPages might throw
|
|
#endif
|
|
} _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) {
|
|
NTSTATUS status = _SEH2_GetExceptionCode();
|
|
|
|
TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_INIT, "ClasspBuildDeviceMdl: MmProbeAndLockPages failed with %xh.", status));
|
|
IoFreeMdl(mdl);
|
|
mdl = NULL;
|
|
} _SEH2_END;
|
|
}
|
|
else {
|
|
TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_INIT, "ClasspBuildDeviceMdl: IoAllocateMdl failed"));
|
|
}
|
|
|
|
return mdl;
|
|
}
|
|
|
|
|
|
PMDL BuildDeviceInputMdl(PVOID Buffer, ULONG BufferLen)
|
|
{
|
|
return ClasspBuildDeviceMdl(Buffer, BufferLen, FALSE);
|
|
}
|
|
|
|
|
|
VOID ClasspFreeDeviceMdl(PMDL Mdl)
|
|
{
|
|
MmUnlockPages(Mdl);
|
|
IoFreeMdl(Mdl);
|
|
}
|
|
|
|
|
|
VOID FreeDeviceInputMdl(PMDL Mdl)
|
|
{
|
|
ClasspFreeDeviceMdl(Mdl);
|
|
return;
|
|
}
|
|
|
|
|
|
#if 0
|
|
VOID
|
|
ClasspPerfResetCounters(
|
|
IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension
|
|
)
|
|
{
|
|
PCLASS_PRIVATE_FDO_DATA fdoData = FdoExtension->PrivateFdoData;
|
|
KIRQL oldIrql;
|
|
|
|
KeAcquireSpinLock(&fdoData->SpinLock, &oldIrql);
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "ClasspPerfResetCounters: "
|
|
"Resetting all perf counters.\n"));
|
|
fdoData->Perf.SuccessfulIO = 0;
|
|
FdoExtension->ErrorCount = 0;
|
|
|
|
if (!TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_DISABLE_DISCONNECT)) {
|
|
CLEAR_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_DISABLE_DISCONNECT);
|
|
}
|
|
if (!TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_DISABLE_SYNCH_TRANSFER)) {
|
|
CLEAR_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_DISABLE_SYNCH_TRANSFER);
|
|
}
|
|
if (TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_QUEUE_ACTION_ENABLE)) {
|
|
SET_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_QUEUE_ACTION_ENABLE);
|
|
}
|
|
if (TEST_FLAG(fdoData->Perf.OriginalSrbFlags,
|
|
SRB_FLAGS_NO_QUEUE_FREEZE)) {
|
|
SET_FLAG(FdoExtension->SrbFlags,
|
|
SRB_FLAGS_NO_QUEUE_FREEZE);
|
|
}
|
|
KeReleaseSpinLock(&fdoData->SpinLock, oldIrql);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
|
|
/*++
|
|
|
|
ClasspDuidGetDeviceIdProperty
|
|
|
|
Routine Description:
|
|
|
|
Add StorageDeviceIdProperty to the device unique ID structure.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the device object
|
|
Irp - a pointer to the I/O request packet
|
|
|
|
Return Value:
|
|
|
|
Status Code
|
|
|
|
--*/
|
|
NTSTATUS
|
|
ClasspDuidGetDeviceIdProperty(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp
|
|
)
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PSTORAGE_DEVICE_ID_DESCRIPTOR deviceIdDescriptor = NULL;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_DESCRIPTOR_HEADER descHeader;
|
|
PSTORAGE_DEVICE_UNIQUE_IDENTIFIER storageDuid;
|
|
PUCHAR dest;
|
|
|
|
STORAGE_PROPERTY_ID propertyId = StorageDeviceIdProperty;
|
|
|
|
NTSTATUS status;
|
|
|
|
ULONG queryLength;
|
|
ULONG offset;
|
|
|
|
//
|
|
// Get the VPD page 83h data.
|
|
//
|
|
|
|
status = ClassGetDescriptor(commonExtension->LowerDeviceObject,
|
|
&propertyId,
|
|
(PVOID *)&deviceIdDescriptor);
|
|
|
|
if (!NT_SUCCESS(status) || !deviceIdDescriptor) {
|
|
goto FnExit;
|
|
}
|
|
|
|
queryLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
descHeader = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
//
|
|
// Adjust required size and potential destination location.
|
|
//
|
|
|
|
offset = descHeader->Size;
|
|
dest = (PUCHAR)descHeader + offset;
|
|
|
|
descHeader->Size += deviceIdDescriptor->Size;
|
|
|
|
if (queryLength < descHeader->Size) {
|
|
|
|
//
|
|
// Output buffer is too small. Return error and make sure
|
|
// the caller gets info about required buffer size.
|
|
//
|
|
|
|
Irp->IoStatus.Information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto FnExit;
|
|
}
|
|
|
|
storageDuid = Irp->AssociatedIrp.SystemBuffer;
|
|
storageDuid->StorageDeviceIdOffset = offset;
|
|
|
|
RtlCopyMemory(dest,
|
|
deviceIdDescriptor,
|
|
deviceIdDescriptor->Size);
|
|
|
|
Irp->IoStatus.Information = storageDuid->Size;
|
|
status = STATUS_SUCCESS;
|
|
|
|
FnExit:
|
|
|
|
FREE_POOL(deviceIdDescriptor);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
/*++
|
|
|
|
ClasspDuidGetDeviceProperty
|
|
|
|
Routine Description:
|
|
|
|
Add StorageDeviceProperty to the device unique ID structure.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the device object
|
|
Irp - a pointer to the I/O request packet
|
|
|
|
Return Value:
|
|
|
|
Status Code
|
|
|
|
--*/
|
|
NTSTATUS
|
|
ClasspDuidGetDeviceProperty(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
PSTORAGE_DEVICE_DESCRIPTOR deviceDescriptor = fdoExtension->DeviceDescriptor;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_DESCRIPTOR_HEADER descHeader;
|
|
PSTORAGE_DEVICE_UNIQUE_IDENTIFIER storageDuid;
|
|
PUCHAR dest;
|
|
|
|
NTSTATUS status = STATUS_NOT_FOUND;
|
|
|
|
ULONG queryLength;
|
|
ULONG offset;
|
|
|
|
//
|
|
// Use the StorageDeviceProperty already cached in the device extension.
|
|
//
|
|
|
|
if (!deviceDescriptor) {
|
|
goto FnExit;
|
|
}
|
|
|
|
queryLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
descHeader = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
//
|
|
// Use this info only if serial number is available.
|
|
//
|
|
|
|
if (deviceDescriptor->SerialNumberOffset == 0) {
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// Adjust required size and potential destination location.
|
|
//
|
|
|
|
offset = descHeader->Size;
|
|
dest = (PUCHAR)descHeader + offset;
|
|
|
|
descHeader->Size += deviceDescriptor->Size;
|
|
|
|
if (queryLength < descHeader->Size) {
|
|
|
|
//
|
|
// Output buffer is too small. Return error and make sure
|
|
// the caller get info about required buffer size.
|
|
//
|
|
|
|
Irp->IoStatus.Information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto FnExit;
|
|
}
|
|
|
|
storageDuid = Irp->AssociatedIrp.SystemBuffer;
|
|
storageDuid->StorageDeviceOffset = offset;
|
|
|
|
RtlCopyMemory(dest,
|
|
deviceDescriptor,
|
|
deviceDescriptor->Size);
|
|
|
|
Irp->IoStatus.Information = storageDuid->Size;
|
|
status = STATUS_SUCCESS;
|
|
|
|
FnExit:
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
ClasspDuidGetDriveLayout
|
|
|
|
Routine Description:
|
|
|
|
Add drive layout signature to the device unique ID structure.
|
|
Layout signature is only added for disk-type devices.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the device object
|
|
Irp - a pointer to the I/O request packet
|
|
|
|
Return Value:
|
|
|
|
Status Code
|
|
|
|
--*/
|
|
NTSTATUS
|
|
ClasspDuidGetDriveLayout(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
PDRIVE_LAYOUT_INFORMATION_EX layoutEx = NULL;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_DESCRIPTOR_HEADER descHeader;
|
|
PSTORAGE_DEVICE_UNIQUE_IDENTIFIER storageDuid;
|
|
PSTORAGE_DEVICE_LAYOUT_SIGNATURE driveLayoutSignature;
|
|
|
|
NTSTATUS status = STATUS_NOT_FOUND;
|
|
|
|
ULONG queryLength;
|
|
ULONG offset;
|
|
|
|
//
|
|
// Only process disk-type devices.
|
|
//
|
|
|
|
if (DeviceObject->DeviceType != FILE_DEVICE_DISK) {
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// Get current partition table and process only if GPT
|
|
// or MBR layout.
|
|
//
|
|
|
|
status = IoReadPartitionTableEx(DeviceObject, &layoutEx);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
status = STATUS_NOT_FOUND;
|
|
goto FnExit;
|
|
}
|
|
|
|
if (layoutEx->PartitionStyle != PARTITION_STYLE_GPT &&
|
|
layoutEx->PartitionStyle != PARTITION_STYLE_MBR) {
|
|
status = STATUS_NOT_FOUND;
|
|
goto FnExit;
|
|
}
|
|
|
|
queryLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
descHeader = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
//
|
|
// Adjust required size and potential destination location.
|
|
//
|
|
|
|
offset = descHeader->Size;
|
|
driveLayoutSignature = (PSTORAGE_DEVICE_LAYOUT_SIGNATURE)((PUCHAR)descHeader + offset);
|
|
|
|
descHeader->Size += sizeof(STORAGE_DEVICE_LAYOUT_SIGNATURE);
|
|
|
|
if (queryLength < descHeader->Size) {
|
|
|
|
//
|
|
// Output buffer is too small. Return error and make sure
|
|
// the caller get info about required buffer size.
|
|
//
|
|
|
|
Irp->IoStatus.Information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto FnExit;
|
|
}
|
|
|
|
storageDuid = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
driveLayoutSignature->Size = sizeof(STORAGE_DEVICE_LAYOUT_SIGNATURE);
|
|
driveLayoutSignature->Version = DUID_VERSION_1;
|
|
|
|
if (layoutEx->PartitionStyle == PARTITION_STYLE_MBR) {
|
|
|
|
driveLayoutSignature->Mbr = TRUE;
|
|
|
|
RtlCopyMemory(&driveLayoutSignature->DeviceSpecific.MbrSignature,
|
|
&layoutEx->Mbr.Signature,
|
|
sizeof(layoutEx->Mbr.Signature));
|
|
|
|
} else {
|
|
|
|
driveLayoutSignature->Mbr = FALSE;
|
|
|
|
RtlCopyMemory(&driveLayoutSignature->DeviceSpecific.GptDiskId,
|
|
&layoutEx->Gpt.DiskId,
|
|
sizeof(layoutEx->Gpt.DiskId));
|
|
}
|
|
|
|
storageDuid->DriveLayoutSignatureOffset = offset;
|
|
|
|
Irp->IoStatus.Information = storageDuid->Size;
|
|
status = STATUS_SUCCESS;
|
|
|
|
|
|
FnExit:
|
|
|
|
FREE_POOL(layoutEx);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
ClasspDuidQueryProperty
|
|
|
|
Routine Description:
|
|
|
|
Handles IOCTL_STORAGE_QUERY_PROPERTY for device unique ID requests
|
|
(when PropertyId is StorageDeviceUniqueIdProperty).
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the device object
|
|
Irp - a pointer to the I/O request packet
|
|
|
|
Return Value:
|
|
|
|
Status Code
|
|
|
|
--*/
|
|
NTSTATUS
|
|
ClasspDuidQueryProperty(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
PSTORAGE_PROPERTY_QUERY query = Irp->AssociatedIrp.SystemBuffer;
|
|
PSTORAGE_DESCRIPTOR_HEADER descHeader;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
NTSTATUS status;
|
|
|
|
ULONG outLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
BOOLEAN includeOptionalIds;
|
|
BOOLEAN overflow = FALSE;
|
|
BOOLEAN infoFound = FALSE;
|
|
BOOLEAN useStatus = TRUE; // Use the status directly instead of relying on overflow and infoFound flags.
|
|
|
|
//
|
|
// Must run at less then dispatch.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
Irp->IoStatus.Information = 0;
|
|
status = STATUS_SUCCESS;
|
|
goto FnExit;
|
|
}
|
|
|
|
if (query->QueryType != PropertyStandardQuery) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// Check AdditionalParameters validity.
|
|
//
|
|
|
|
if (query->AdditionalParameters[0] == DUID_INCLUDE_SOFTWARE_IDS) {
|
|
includeOptionalIds = TRUE;
|
|
} else if (query->AdditionalParameters[0] == DUID_HARDWARE_IDS_ONLY) {
|
|
includeOptionalIds = FALSE;
|
|
} else {
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// Verify output parameters.
|
|
//
|
|
|
|
if (outLength < sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// From this point forward the status depends on the overflow
|
|
// and infoFound flags.
|
|
//
|
|
|
|
useStatus = FALSE;
|
|
|
|
descHeader = Irp->AssociatedIrp.SystemBuffer;
|
|
RtlZeroMemory(descHeader, outLength);
|
|
|
|
descHeader->Version = DUID_VERSION_1;
|
|
descHeader->Size = sizeof(STORAGE_DEVICE_UNIQUE_IDENTIFIER);
|
|
|
|
//
|
|
// Try to build device unique id from StorageDeviceIdProperty.
|
|
//
|
|
|
|
status = ClasspDuidGetDeviceIdProperty(DeviceObject,
|
|
Irp);
|
|
|
|
if (status == STATUS_BUFFER_OVERFLOW) {
|
|
overflow = TRUE;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
infoFound = TRUE;
|
|
}
|
|
|
|
//
|
|
// Try to build device unique id from StorageDeviceProperty.
|
|
//
|
|
|
|
status = ClasspDuidGetDeviceProperty(DeviceObject,
|
|
Irp);
|
|
|
|
if (status == STATUS_BUFFER_OVERFLOW) {
|
|
overflow = TRUE;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
infoFound = TRUE;
|
|
}
|
|
|
|
//
|
|
// The following portion is optional and only included if the
|
|
// caller requested software IDs.
|
|
//
|
|
|
|
if (!includeOptionalIds) {
|
|
goto FnExit;
|
|
}
|
|
|
|
//
|
|
// Try to build device unique id from drive layout signature (disk
|
|
// devices only).
|
|
//
|
|
|
|
status = ClasspDuidGetDriveLayout(DeviceObject,
|
|
Irp);
|
|
|
|
if (status == STATUS_BUFFER_OVERFLOW) {
|
|
overflow = TRUE;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
infoFound = TRUE;
|
|
}
|
|
|
|
FnExit:
|
|
|
|
if (!useStatus) {
|
|
|
|
//
|
|
// Return overflow, success, or a generic error.
|
|
//
|
|
|
|
if (overflow) {
|
|
|
|
//
|
|
// If output buffer is STORAGE_DESCRIPTOR_HEADER, then return
|
|
// success to the user. Otherwise, send an error so the user
|
|
// knows a larger buffer is required.
|
|
//
|
|
|
|
if (outLength == sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
status = STATUS_SUCCESS;
|
|
Irp->IoStatus.Information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
} else {
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
}
|
|
|
|
} else if (infoFound) {
|
|
status = STATUS_SUCCESS;
|
|
|
|
//
|
|
// Exercise the compare routine. This should always succeed.
|
|
//
|
|
|
|
NT_ASSERT(DuidExactMatch == CompareStorageDuids(Irp->AssociatedIrp.SystemBuffer,
|
|
Irp->AssociatedIrp.SystemBuffer));
|
|
|
|
} else {
|
|
status = STATUS_NOT_FOUND;
|
|
}
|
|
}
|
|
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
/*++////////////////////////////////////////////////////////////////////////////
|
|
|
|
ClasspWriteCacheProperty()
|
|
|
|
Routine Description:
|
|
|
|
This routine reads the caching mode page from the device to
|
|
build the Write Cache property page.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - The device object to handle this irp
|
|
|
|
Irp - The IRP for this request
|
|
|
|
Srb - SRB allocated by the dispatch routine
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
|
|
NTSTATUS ClasspWriteCacheProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
PSTORAGE_WRITE_CACHE_PROPERTY writeCache;
|
|
PSTORAGE_PROPERTY_QUERY query = Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PMODE_PARAMETER_HEADER modeData = NULL;
|
|
PMODE_CACHING_PAGE pageData = NULL;
|
|
ULONG length, information = 0;
|
|
NTSTATUS status;
|
|
PCDB cdb;
|
|
|
|
//
|
|
// Must run at less then dispatch.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto WriteCacheExit;
|
|
}
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
status = STATUS_SUCCESS;
|
|
goto WriteCacheExit;
|
|
}
|
|
|
|
if (query->QueryType != PropertyStandardQuery) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto WriteCacheExit;
|
|
}
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (length < sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
goto WriteCacheExit;
|
|
}
|
|
|
|
writeCache = (PSTORAGE_WRITE_CACHE_PROPERTY) Irp->AssociatedIrp.SystemBuffer;
|
|
RtlZeroMemory(writeCache, length);
|
|
|
|
//
|
|
// Set version and required size.
|
|
//
|
|
|
|
writeCache->Version = sizeof(STORAGE_WRITE_CACHE_PROPERTY);
|
|
writeCache->Size = sizeof(STORAGE_WRITE_CACHE_PROPERTY);
|
|
|
|
if (length < sizeof(STORAGE_WRITE_CACHE_PROPERTY)) {
|
|
information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
status = STATUS_SUCCESS;
|
|
goto WriteCacheExit;
|
|
}
|
|
|
|
//
|
|
// Set known values
|
|
//
|
|
|
|
writeCache->NVCacheEnabled = FALSE;
|
|
writeCache->UserDefinedPowerProtection = TEST_FLAG(fdoExtension->DeviceFlags, DEV_POWER_PROTECTED);
|
|
|
|
//
|
|
// Check for flush cache support by sending a sync cache command
|
|
// to the device.
|
|
//
|
|
|
|
//
|
|
// Set timeout value and mark the request as not being a tagged request.
|
|
//
|
|
SrbSetTimeOutValue(Srb, fdoExtension->TimeOutValue * 4);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, fdoExtension->SrbFlags);
|
|
|
|
SrbSetCdbLength(Srb, 10);
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->CDB10.OperationCode = SCSIOP_SYNCHRONIZE_CACHE;
|
|
|
|
status = ClassSendSrbSynchronous(DeviceObject,
|
|
Srb,
|
|
NULL,
|
|
0,
|
|
TRUE);
|
|
if (NT_SUCCESS(status)) {
|
|
writeCache->FlushCacheSupported = TRUE;
|
|
} else {
|
|
//
|
|
// Device does not support sync cache
|
|
//
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_IOCTL, "ClasspWriteCacheProperty: Synchronize cache failed with status 0x%X\n", status));
|
|
writeCache->FlushCacheSupported = FALSE;
|
|
//
|
|
// Reset the status if there was any failure
|
|
//
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
modeData = ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned,
|
|
MODE_PAGE_DATA_SIZE,
|
|
CLASS_TAG_MODE_DATA);
|
|
|
|
if (modeData == NULL) {
|
|
TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_IOCTL, "ClasspWriteCacheProperty: Unable to allocate mode data buffer\n"));
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto WriteCacheExit;
|
|
}
|
|
|
|
RtlZeroMemory(modeData, MODE_PAGE_DATA_SIZE);
|
|
|
|
length = ClassModeSense(DeviceObject,
|
|
(PCHAR) modeData,
|
|
MODE_PAGE_DATA_SIZE,
|
|
MODE_PAGE_CACHING);
|
|
|
|
if (length < sizeof(MODE_PARAMETER_HEADER)) {
|
|
|
|
//
|
|
// Retry the request in case of a check condition.
|
|
//
|
|
|
|
length = ClassModeSense(DeviceObject,
|
|
(PCHAR) modeData,
|
|
MODE_PAGE_DATA_SIZE,
|
|
MODE_PAGE_CACHING);
|
|
|
|
if (length < sizeof(MODE_PARAMETER_HEADER)) {
|
|
TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_IOCTL, "ClasspWriteCacheProperty: Mode Sense failed\n"));
|
|
status = STATUS_IO_DEVICE_ERROR;
|
|
goto WriteCacheExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If the length is greater than length indicated by the mode data reset
|
|
// the data to the mode data.
|
|
//
|
|
|
|
if (length > (ULONG) (modeData->ModeDataLength + 1)) {
|
|
length = modeData->ModeDataLength + 1;
|
|
}
|
|
|
|
//
|
|
// Look for caching page in the returned mode page data.
|
|
//
|
|
|
|
pageData = ClassFindModePage((PCHAR) modeData,
|
|
length,
|
|
MODE_PAGE_CACHING,
|
|
TRUE);
|
|
|
|
//
|
|
// Check if valid caching page exists.
|
|
//
|
|
|
|
if (pageData == NULL) {
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_IOCTL, "ClasspWriteCacheProperty: Unable to find caching mode page.\n"));
|
|
//
|
|
// Set write cache value as unknown.
|
|
//
|
|
writeCache->WriteCacheEnabled = WriteCacheEnableUnknown;
|
|
writeCache->WriteCacheType = WriteCacheTypeUnknown;
|
|
} else {
|
|
writeCache->WriteCacheEnabled = pageData->WriteCacheEnable ?
|
|
WriteCacheEnabled : WriteCacheDisabled;
|
|
|
|
writeCache->WriteCacheType = pageData->WriteCacheEnable ?
|
|
WriteCacheTypeWriteBack : WriteCacheTypeUnknown;
|
|
}
|
|
|
|
//
|
|
// Check write through support. If the device previously failed a write request
|
|
// with FUA bit is set, then CLASS_SPECIAL_FUA_NOT_SUPPORTED will be set,
|
|
// which means write through is not support by the device.
|
|
//
|
|
|
|
if ((modeData->DeviceSpecificParameter & MODE_DSP_FUA_SUPPORTED) &&
|
|
(!TEST_FLAG(fdoExtension->ScanForSpecialFlags, CLASS_SPECIAL_FUA_NOT_SUPPORTED))) {
|
|
writeCache->WriteThroughSupported = WriteThroughSupported;
|
|
} else {
|
|
writeCache->WriteThroughSupported = WriteThroughNotSupported;
|
|
}
|
|
|
|
//
|
|
// Get the changeable caching mode page and check write cache is changeable.
|
|
//
|
|
|
|
RtlZeroMemory(modeData, MODE_PAGE_DATA_SIZE);
|
|
|
|
length = ClasspModeSense(DeviceObject,
|
|
(PCHAR) modeData,
|
|
MODE_PAGE_DATA_SIZE,
|
|
MODE_PAGE_CACHING,
|
|
MODE_SENSE_CHANGEABLE_VALUES);
|
|
|
|
if (length < sizeof(MODE_PARAMETER_HEADER)) {
|
|
|
|
//
|
|
// Retry the request in case of a check condition.
|
|
//
|
|
|
|
length = ClasspModeSense(DeviceObject,
|
|
(PCHAR) modeData,
|
|
MODE_PAGE_DATA_SIZE,
|
|
MODE_PAGE_CACHING,
|
|
MODE_SENSE_CHANGEABLE_VALUES);
|
|
|
|
if (length < sizeof(MODE_PARAMETER_HEADER)) {
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_IOCTL, "ClasspWriteCacheProperty: Mode Sense failed\n"));
|
|
|
|
//
|
|
// If the device fails to return changeable pages, then
|
|
// set the write cache changeable value to unknown.
|
|
//
|
|
|
|
writeCache->WriteCacheChangeable = WriteCacheChangeUnknown;
|
|
information = sizeof(STORAGE_WRITE_CACHE_PROPERTY);
|
|
goto WriteCacheExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If the length is greater than length indicated by the mode data reset
|
|
// the data to the mode data.
|
|
//
|
|
|
|
if (length > (ULONG) (modeData->ModeDataLength + 1)) {
|
|
length = modeData->ModeDataLength + 1;
|
|
}
|
|
|
|
//
|
|
// Look for caching page in the returned mode page data.
|
|
//
|
|
|
|
pageData = ClassFindModePage((PCHAR) modeData,
|
|
length,
|
|
MODE_PAGE_CACHING,
|
|
TRUE);
|
|
//
|
|
// Check if valid caching page exists.
|
|
//
|
|
|
|
if (pageData == NULL) {
|
|
TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_IOCTL, "ClasspWriteCacheProperty: Unable to find caching mode page.\n"));
|
|
//
|
|
// Set write cache changeable value to unknown.
|
|
//
|
|
writeCache->WriteCacheChangeable = WriteCacheChangeUnknown;
|
|
} else {
|
|
writeCache->WriteCacheChangeable = pageData->WriteCacheEnable ?
|
|
WriteCacheChangeable : WriteCacheNotChangeable;
|
|
}
|
|
|
|
information = sizeof(STORAGE_WRITE_CACHE_PROPERTY);
|
|
|
|
WriteCacheExit:
|
|
|
|
FREE_POOL(modeData);
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
ULONG
|
|
ClasspCalculateLogicalSectorSize (
|
|
_In_ PDEVICE_OBJECT Fdo,
|
|
_In_ ULONG BytesPerBlockInBigEndian
|
|
)
|
|
/*++
|
|
Convert the big-endian value.
|
|
if it's 0, default to the standard 512 bytes.
|
|
if it's not a power of 2 value, round down to power of 2.
|
|
--*/
|
|
{
|
|
ULONG logicalSectorSize;
|
|
|
|
REVERSE_BYTES(&logicalSectorSize, &BytesPerBlockInBigEndian);
|
|
|
|
if (logicalSectorSize == 0) {
|
|
logicalSectorSize = 512;
|
|
} else {
|
|
//
|
|
// Clear all but the highest set bit.
|
|
// That will give us a bytesPerSector value that is a power of 2.
|
|
//
|
|
if (logicalSectorSize & (logicalSectorSize-1)) {
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_INIT, "FDO %ph has non-standard sector size 0x%x.", Fdo, logicalSectorSize));
|
|
do {
|
|
logicalSectorSize &= logicalSectorSize-1;
|
|
}
|
|
while (logicalSectorSize & (logicalSectorSize-1));
|
|
}
|
|
}
|
|
|
|
return logicalSectorSize;
|
|
}
|
|
|
|
NTSTATUS
|
|
InterpretReadCapacity16Data (
|
|
_Inout_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_ PREAD_CAPACITY16_DATA ReadCapacity16Data
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
USHORT lowestAlignedBlock;
|
|
USHORT logicalBlocksPerPhysicalBlock;
|
|
PCLASS_READ_CAPACITY16_DATA cachedData = &(FdoExtension->FunctionSupportInfo->ReadCapacity16Data);
|
|
|
|
// use Logical Sector Size from DiskGeometry to avoid duplicated calculation.
|
|
FdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesPerLogicalSector = ClasspCalculateLogicalSectorSize(FdoExtension->DeviceObject, ReadCapacity16Data->BytesPerBlock);
|
|
|
|
// FdoExtension->DiskGeometry.BytesPerSector might be 0 for class drivers that don't get READ CAPACITY info yet.
|
|
NT_ASSERT( (FdoExtension->DiskGeometry.BytesPerSector == 0) ||
|
|
(FdoExtension->DiskGeometry.BytesPerSector == FdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesPerLogicalSector) );
|
|
|
|
logicalBlocksPerPhysicalBlock = 1 << ReadCapacity16Data->LogicalPerPhysicalExponent;
|
|
lowestAlignedBlock = (ReadCapacity16Data->LowestAlignedBlock_MSB << 8) | ReadCapacity16Data->LowestAlignedBlock_LSB;
|
|
|
|
if (lowestAlignedBlock > logicalBlocksPerPhysicalBlock) {
|
|
// we get garbage data
|
|
status = STATUS_UNSUCCESSFUL;
|
|
} else {
|
|
// value of lowestAlignedBlock (from T10 spec) needs to be converted.
|
|
lowestAlignedBlock = (logicalBlocksPerPhysicalBlock - lowestAlignedBlock) % logicalBlocksPerPhysicalBlock;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
// fill output buffer
|
|
cachedData->BytesPerPhysicalSector = cachedData->BytesPerLogicalSector * logicalBlocksPerPhysicalBlock;
|
|
cachedData->BytesOffsetForSectorAlignment = cachedData->BytesPerLogicalSector * lowestAlignedBlock;
|
|
|
|
//
|
|
// Fill in the Logical Block Provisioning info. Note that we do not
|
|
// use these fields; we use the Provisioning Type and LBPRZ fields from
|
|
// the Logical Block Provisioning VPD page (0xB2).
|
|
//
|
|
cachedData->LBProvisioningEnabled = ReadCapacity16Data->LBPME;
|
|
cachedData->LBProvisioningReadZeros = ReadCapacity16Data->LBPRZ;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_INIT,
|
|
"InterpretReadCapacity16Data: Device\'s LBP enabled = %d\n",
|
|
cachedData->LBProvisioningEnabled));
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
ClassReadCapacity16 (
|
|
_Inout_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
This routine may send down a READ CAPACITY 16 command to retrieve info.
|
|
The info will be cached in FdoExtension->LowerLayerSupport->AccessAlignment.
|
|
|
|
After info retrieving finished, this function sets following field:
|
|
FdoExtension->LowerLayerSupport->AccessAlignment.LowerLayerSupported = Supported;
|
|
to indicate that info has been cached.
|
|
|
|
NOTE: some future processes may use this function to send the command anyway, it will be caller's decision
|
|
on checking 'AccessAlignment.LowerLayerSupported' in case the cached info is good enough.
|
|
*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
PREAD_CAPACITY16_DATA dataBuffer = NULL;
|
|
UCHAR bufferLength = sizeof(READ_CAPACITY16_DATA);
|
|
ULONG allocationBufferLength = bufferLength; //DMA buffer size for alignment
|
|
PCDB cdb;
|
|
ULONG dataTransferLength = 0;
|
|
|
|
//
|
|
// If the information retrieval has already been attempted, return the cached status.
|
|
//
|
|
if (FdoExtension->FunctionSupportInfo->ReadCapacity16Data.CommandStatus != -1) {
|
|
// get cached NTSTATUS from previous call.
|
|
return FdoExtension->FunctionSupportInfo->ReadCapacity16Data.CommandStatus;
|
|
}
|
|
|
|
if (ClasspIsObsoletePortDriver(FdoExtension)) {
|
|
FdoExtension->FunctionSupportInfo->ReadCapacity16Data.CommandStatus = STATUS_NOT_IMPLEMENTED;
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
allocationBufferLength = ALIGN_UP_BY(allocationBufferLength,KeGetRecommendedSharedDataAlignment());
|
|
dataBuffer = (PREAD_CAPACITY16_DATA)ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned, allocationBufferLength, '4CcS');
|
|
#else
|
|
dataBuffer = (PREAD_CAPACITY16_DATA)ExAllocatePoolWithTag(NonPagedPoolNx, bufferLength, '4CcS');
|
|
#endif
|
|
|
|
if (dataBuffer == NULL) {
|
|
// return without updating FdoExtension->FunctionSupportInfo->ReadCapacity16Data.CommandStatus
|
|
// the field will remain value as "-1", so that the command will be attempted next time this function is called.
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
RtlZeroMemory(dataBuffer, allocationBufferLength);
|
|
|
|
//
|
|
// Initialize the SRB.
|
|
//
|
|
if (FdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
|
|
status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)Srb,
|
|
STORAGE_ADDRESS_TYPE_BTL8,
|
|
CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
|
|
1,
|
|
SrbExDataTypeScsiCdb16);
|
|
if (NT_SUCCESS(status)) {
|
|
((PSTORAGE_REQUEST_BLOCK)Srb)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;
|
|
} else {
|
|
//
|
|
// Should not occur.
|
|
//
|
|
NT_ASSERT(FALSE);
|
|
}
|
|
} else {
|
|
RtlZeroMemory(Srb, sizeof(SCSI_REQUEST_BLOCK));
|
|
Srb->Length = sizeof(SCSI_REQUEST_BLOCK);
|
|
Srb->Function = SRB_FUNCTION_EXECUTE_SCSI;
|
|
}
|
|
|
|
//prepare the Srb
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
|
|
SrbSetTimeOutValue(Srb, FdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, FdoExtension->SrbFlags);
|
|
|
|
SrbSetCdbLength(Srb, 16);
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->READ_CAPACITY16.OperationCode = SCSIOP_READ_CAPACITY16;
|
|
cdb->READ_CAPACITY16.ServiceAction = SERVICE_ACTION_READ_CAPACITY16;
|
|
cdb->READ_CAPACITY16.AllocationLength[3] = bufferLength;
|
|
|
|
status = ClassSendSrbSynchronous(FdoExtension->DeviceObject,
|
|
Srb,
|
|
dataBuffer,
|
|
allocationBufferLength,
|
|
FALSE);
|
|
|
|
dataTransferLength = SrbGetDataTransferLength(Srb);
|
|
}
|
|
|
|
if (NT_SUCCESS(status) && (dataTransferLength < 16))
|
|
{
|
|
// the device should return at least 16 bytes of data for this command.
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
}
|
|
|
|
//
|
|
// Handle the case where we get back STATUS_DATA_OVERRUN b/c the input
|
|
// buffer was larger than necessary.
|
|
//
|
|
if (status == STATUS_DATA_OVERRUN && dataTransferLength < bufferLength)
|
|
{
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
// cache data into FdoExtension
|
|
status = InterpretReadCapacity16Data(FdoExtension, dataBuffer);
|
|
}
|
|
|
|
// cache the status indicates that this funciton has been called.
|
|
FdoExtension->FunctionSupportInfo->ReadCapacity16Data.CommandStatus = status;
|
|
|
|
ExFreePool(dataBuffer);
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClasspAccessAlignmentProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
At first time of receiving the request, this function will forward it to lower stack to determine if it's supportted.
|
|
If it's not supported, SCSIOP_READ_CAPACITY16 will be sent down to retrieve the information.
|
|
*/
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PSTORAGE_PROPERTY_QUERY query = (PSTORAGE_PROPERTY_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
ULONG length = 0;
|
|
ULONG information = 0;
|
|
|
|
PSTORAGE_ACCESS_ALIGNMENT_DESCRIPTOR accessAlignment;
|
|
|
|
//
|
|
// check registry setting and fail the IOCTL if it's required.
|
|
// this registry setting can be used to work around issues which upper layer doesn't support large physical sector size.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->RegAccessAlignmentQueryNotSupported) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
|
|
if ( (DeviceObject->DeviceType != FILE_DEVICE_DISK) ||
|
|
(TEST_FLAG(DeviceObject->Characteristics, FILE_FLOPPY_DISKETTE)) ||
|
|
(fdoExtension->FunctionSupportInfo->LowerLayerSupport.AccessAlignmentProperty == Supported) ) {
|
|
// if it's not disk, forward the request to lower layer,
|
|
// if the IOCTL is supported by lower stack, forward it down.
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
} else if (query->QueryType != PropertyStandardQuery) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Request validation.
|
|
// Note that InputBufferLength and IsFdo have been validated before entering this routine.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto Exit;
|
|
}
|
|
|
|
// do not touch this buffer because it can still be used as input buffer for lower layer in 'SupportUnknown' case.
|
|
accessAlignment = (PSTORAGE_ACCESS_ALIGNMENT_DESCRIPTOR)Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (length < sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR)) {
|
|
|
|
if (length >= sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
accessAlignment->Version = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
accessAlignment->Size = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
}
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto Exit;
|
|
}
|
|
|
|
// not support Cache Line,
|
|
// 'BytesPerCacheLine' and 'BytesOffsetForCacheAlignment' fields are zero-ed.
|
|
|
|
//
|
|
// note that 'Supported' case has been handled at the beginning of this function.
|
|
//
|
|
switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.AccessAlignmentProperty) {
|
|
case SupportUnknown: {
|
|
// send down request and wait for the request to complete.
|
|
status = ClassForwardIrpSynchronous(commonExtension, Irp);
|
|
|
|
if (ClasspLowerLayerNotSupport(status)) {
|
|
// case 1: the request is not supported by lower layer, sends down command
|
|
// some port drivers (or filter drivers) return STATUS_INVALID_DEVICE_REQUEST if a request is not supported.
|
|
|
|
// ClassReadCapacity16() will either return status from cached data or send command to retrieve the information.
|
|
if (ClasspIsObsoletePortDriver(fdoExtension) == FALSE) {
|
|
status = ClassReadCapacity16(fdoExtension, Srb);
|
|
} else {
|
|
fdoExtension->FunctionSupportInfo->ReadCapacity16Data.CommandStatus = status;
|
|
}
|
|
|
|
// data is ready in fdoExtension
|
|
// set the support status after the SCSI command is executed to avoid racing condition between multiple same type of requests.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.AccessAlignmentProperty = NotSupported;
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
// fill output buffer
|
|
RtlZeroMemory(accessAlignment, length);
|
|
accessAlignment->Version = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
accessAlignment->Size = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
accessAlignment->BytesPerLogicalSector = fdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesPerLogicalSector;
|
|
accessAlignment->BytesPerPhysicalSector = fdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesPerPhysicalSector;
|
|
accessAlignment->BytesOffsetForSectorAlignment = fdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesOffsetForSectorAlignment;
|
|
|
|
// set returned data length
|
|
information = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
} else {
|
|
information = 0;
|
|
}
|
|
|
|
} else {
|
|
// case 2: the request is supported and it completes successfully
|
|
// case 3: the request is supported by lower stack but other failure status is returned.
|
|
// from now on, the same request will be send down to lower stack directly.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.AccessAlignmentProperty = Supported;
|
|
information = (ULONG)Irp->IoStatus.Information;
|
|
|
|
|
|
}
|
|
|
|
|
|
goto Exit;
|
|
|
|
break;
|
|
}
|
|
|
|
case NotSupported: {
|
|
|
|
// ClassReadCapacity16() will either return status from cached data or send command to retrieve the information.
|
|
status = ClassReadCapacity16(fdoExtension, Srb);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
RtlZeroMemory(accessAlignment, length);
|
|
accessAlignment->Version = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
accessAlignment->Size = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
accessAlignment->BytesPerLogicalSector = fdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesPerLogicalSector;
|
|
accessAlignment->BytesPerPhysicalSector = fdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesPerPhysicalSector;
|
|
accessAlignment->BytesOffsetForSectorAlignment = fdoExtension->FunctionSupportInfo->ReadCapacity16Data.BytesOffsetForSectorAlignment;
|
|
|
|
information = sizeof(STORAGE_ACCESS_ALIGNMENT_DESCRIPTOR);
|
|
} else {
|
|
information = 0;
|
|
}
|
|
goto Exit;
|
|
|
|
break;
|
|
}
|
|
|
|
case Supported: {
|
|
NT_ASSERT(FALSE); // this case is handled at the beginning of the function.
|
|
status = STATUS_INTERNAL_ERROR;
|
|
break;
|
|
}
|
|
|
|
} // end of switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.AccessAlignmentProperty)
|
|
|
|
Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
static
|
|
NTSTATUS
|
|
IncursSeekPenalty (
|
|
_In_ USHORT MediumRotationRate,
|
|
_In_ PBOOLEAN IncursSeekPenalty
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
|
|
if (MediumRotationRate == 0x0001) {
|
|
// Non-rotating media (e.g., solid state device)
|
|
*IncursSeekPenalty = FALSE;
|
|
status = STATUS_SUCCESS;
|
|
} else if ( (MediumRotationRate >= 0x401) &&
|
|
(MediumRotationRate <= 0xFFFE) ) {
|
|
// Nominal media rotation rate in rotations per minute (rpm)
|
|
*IncursSeekPenalty = TRUE;
|
|
status = STATUS_SUCCESS;
|
|
} else {
|
|
// Unknown cases:
|
|
// 0 - Rate not reported
|
|
// 0002h-0400h - Reserved
|
|
// FFFFh - Reserved
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
ClasspDeviceMediaTypeProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine returns the medium product type reported by the device for the associated LU.
|
|
|
|
This function must be called at IRQL < DISPATCH_LEVEL.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the device object associated with this request
|
|
Irp - The IRP to be processed
|
|
Srb - The SRB associated with the request
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS code
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PSTORAGE_PROPERTY_QUERY query = (PSTORAGE_PROPERTY_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
PSTORAGE_MEDIUM_PRODUCT_TYPE_DESCRIPTOR pDesc = (PSTORAGE_MEDIUM_PRODUCT_TYPE_DESCRIPTOR)Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack;
|
|
ULONG length = 0;
|
|
ULONG information = 0;
|
|
|
|
irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceMediaTypeProperty (%p): Entering function.\n",
|
|
DeviceObject));
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
|
|
//
|
|
// In order to maintain consistency with the how the rest of the properties
|
|
// are handled, always return success for PropertyExistsQuery.
|
|
//
|
|
status = STATUS_SUCCESS;
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
|
|
} else if (query->QueryType != PropertyStandardQuery) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceMediaTypeProperty (%p): Unsupported query type %x for media type property.\n",
|
|
DeviceObject,
|
|
query->QueryType));
|
|
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
}
|
|
|
|
//
|
|
// Validate the request.
|
|
// InputBufferLength and IsFdo have already been validated.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceMediaTypeProperty (%p): Query property for media type at incorrect IRQL.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
}
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (length >= sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
information = sizeof(STORAGE_MEDIUM_PRODUCT_TYPE_DESCRIPTOR);
|
|
pDesc->Version = sizeof(STORAGE_MEDIUM_PRODUCT_TYPE_DESCRIPTOR);
|
|
pDesc->Size = sizeof(STORAGE_MEDIUM_PRODUCT_TYPE_DESCRIPTOR);
|
|
} else {
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
}
|
|
|
|
if (length < sizeof(STORAGE_MEDIUM_PRODUCT_TYPE_DESCRIPTOR)) {
|
|
|
|
status = STATUS_SUCCESS;
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
}
|
|
|
|
//
|
|
// Only query BlockDeviceCharacteristics VPD page if device support has been confirmed.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.BlockDeviceCharacteristics == TRUE) {
|
|
status = ClasspDeviceGetBlockDeviceCharacteristicsVPDPage(fdoExtension, Srb);
|
|
} else {
|
|
//
|
|
// Otherwise device was previously found lacking support for this VPD page. Fail the request.
|
|
//
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
}
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
|
|
status = fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.CommandStatus;
|
|
information = 0;
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetBlockDeviceCharacteristicsVPDPage (%p): VPD retrieval fails with %x.\n",
|
|
DeviceObject,
|
|
status));
|
|
|
|
goto __ClasspDeviceMediaTypeProperty_Exit;
|
|
}
|
|
|
|
//
|
|
// Fill in the output buffer. All data is copied from the FDO extension, cached
|
|
// from device response to earlier VPD_BLOCK_DEVICE_CHARACTERISTICS query.
|
|
//
|
|
pDesc->MediumProductType = fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.MediumProductType;
|
|
status = STATUS_SUCCESS;
|
|
|
|
__ClasspDeviceMediaTypeProperty_Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceMediaTypeProperty (%p): Exiting function with status %x.\n",
|
|
DeviceObject,
|
|
status));
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClasspDeviceGetBlockDeviceCharacteristicsVPDPage(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION fdoExtension,
|
|
_In_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function sends an INQUIRY command request for VPD_BLOCK_DEVICE_CHARACTERISTICS to
|
|
the device. Relevant data from the response is cached in the FDO extension.
|
|
|
|
Arguments:
|
|
FdoExtension: The FDO extension of the device to which the INQUIRY command will be sent.
|
|
Srb: Allocated by the caller.
|
|
SrbSize: The size of the Srb buffer in bytes.
|
|
|
|
Return Value:
|
|
|
|
STATUS_INVALID_PARAMETER: May be returned if the LogPage buffer is NULL or
|
|
not large enough.
|
|
STATUS_SUCCESS: The log page was obtained and placed in the LogPage buffer.
|
|
|
|
This function may return other NTSTATUS codes from internal function calls.
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
PCDB cdb;
|
|
UCHAR bufferLength = sizeof(VPD_BLOCK_DEVICE_CHARACTERISTICS_PAGE); // data is 64 bytes
|
|
ULONG allocationBufferLength = bufferLength;
|
|
PVPD_BLOCK_DEVICE_CHARACTERISTICS_PAGE dataBuffer = NULL;
|
|
|
|
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
allocationBufferLength = ALIGN_UP_BY(allocationBufferLength,KeGetRecommendedSharedDataAlignment());
|
|
dataBuffer = (PVPD_BLOCK_DEVICE_CHARACTERISTICS_PAGE)ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned,
|
|
allocationBufferLength,
|
|
'5CcS'
|
|
);
|
|
#else
|
|
|
|
dataBuffer = (PVPD_BLOCK_DEVICE_CHARACTERISTICS_PAGE)ExAllocatePoolWithTag(NonPagedPoolNx,
|
|
bufferLength,
|
|
'5CcS'
|
|
);
|
|
#endif
|
|
if (dataBuffer == NULL) {
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Exit;
|
|
}
|
|
|
|
RtlZeroMemory(dataBuffer, allocationBufferLength);
|
|
|
|
// prepare the Srb
|
|
SrbSetTimeOutValue(Srb, fdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, fdoExtension->SrbFlags);
|
|
|
|
SrbSetCdbLength(Srb, 6);
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->CDB6INQUIRY3.OperationCode = SCSIOP_INQUIRY;
|
|
cdb->CDB6INQUIRY3.EnableVitalProductData = 1; //EVPD bit
|
|
cdb->CDB6INQUIRY3.PageCode = VPD_BLOCK_DEVICE_CHARACTERISTICS;
|
|
cdb->CDB6INQUIRY3.AllocationLength = bufferLength; //AllocationLength field in CDB6INQUIRY3 is only one byte.
|
|
|
|
status = ClassSendSrbSynchronous(fdoExtension->CommonExtension.DeviceObject,
|
|
Srb,
|
|
dataBuffer,
|
|
allocationBufferLength,
|
|
FALSE);
|
|
if (NT_SUCCESS(status)) {
|
|
if (SrbGetDataTransferLength(Srb) < 0x8) {
|
|
// the device should return at least 8 bytes of data for use.
|
|
status = STATUS_UNSUCCESSFUL;
|
|
} else if ( (dataBuffer->PageLength != 0x3C) || (dataBuffer->PageCode != VPD_BLOCK_DEVICE_CHARACTERISTICS) ) {
|
|
// 'PageLength' shall be 0x3C; and 'PageCode' shall match.
|
|
status = STATUS_UNSUCCESSFUL;
|
|
} else {
|
|
// cache data into fdoExtension
|
|
fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.MediumRotationRate = (dataBuffer->MediumRotationRateMsb << 8) |
|
|
dataBuffer->MediumRotationRateLsb;
|
|
fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.MediumProductType = dataBuffer->MediumProductType;
|
|
fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.NominalFormFactor = dataBuffer->NominalFormFactor;
|
|
}
|
|
} else {
|
|
// the command failed, surface up the command error from 'status' variable. Nothing to do here.
|
|
}
|
|
|
|
Exit:
|
|
if (dataBuffer != NULL) {
|
|
ExFreePool(dataBuffer);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClasspDeviceSeekPenaltyProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
At first time of receiving the request, this function will forward it to lower stack to determine if it's supportted.
|
|
If it's not supported, INQUIRY (Block Device Characteristics VPD page) will be sent down to retrieve the information.
|
|
*/
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PSTORAGE_PROPERTY_QUERY query = (PSTORAGE_PROPERTY_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
ULONG length = 0;
|
|
ULONG information = 0;
|
|
BOOLEAN incursSeekPenalty = TRUE;
|
|
PDEVICE_SEEK_PENALTY_DESCRIPTOR seekPenalty;
|
|
|
|
if ( (DeviceObject->DeviceType != FILE_DEVICE_DISK) ||
|
|
(TEST_FLAG(DeviceObject->Characteristics, FILE_FLOPPY_DISKETTE)) ||
|
|
(fdoExtension->FunctionSupportInfo->LowerLayerSupport.SeekPenaltyProperty == Supported) ) {
|
|
// if it's not disk, forward the request to lower layer,
|
|
// if the IOCTL is supported by lower stack, forward it down.
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
} else if (query->QueryType != PropertyStandardQuery) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Request validation.
|
|
// Note that InputBufferLength and IsFdo have been validated beforing entering this routine.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto Exit;
|
|
}
|
|
|
|
// do not touch this buffer because it can still be used as input buffer for lower layer in 'SupportUnknown' case.
|
|
seekPenalty = (PDEVICE_SEEK_PENALTY_DESCRIPTOR)Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (length < sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR)) {
|
|
|
|
if (length >= sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
seekPenalty->Version = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
seekPenalty->Size = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
}
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// note that 'Supported' case has been handled at the beginning of this function.
|
|
//
|
|
switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.SeekPenaltyProperty) {
|
|
case SupportUnknown: {
|
|
// send down request and wait for the request to complete.
|
|
status = ClassForwardIrpSynchronous(commonExtension, Irp);
|
|
|
|
if (ClasspLowerLayerNotSupport(status)) {
|
|
// case 1: the request is not supported by lower layer, sends down command
|
|
// some port drivers (or filter drivers) return STATUS_INVALID_DEVICE_REQUEST if a request is not supported.
|
|
|
|
// send INQUIRY command if the VPD page is supported.
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.BlockDeviceCharacteristics == TRUE) {
|
|
status = ClasspDeviceGetBlockDeviceCharacteristicsVPDPage(fdoExtension, Srb);
|
|
} else {
|
|
// the INQUIRY - VPD page command to discover the info is not supported, fail the request.
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
status = IncursSeekPenalty(fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.MediumRotationRate, &incursSeekPenalty);
|
|
}
|
|
|
|
fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.CommandStatus = status;
|
|
|
|
// data is ready in fdoExtension
|
|
// set the support status after the SCSI command is executed to avoid racing condition between multiple same type of requests.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.SeekPenaltyProperty = NotSupported;
|
|
|
|
// fill output buffer
|
|
if (NT_SUCCESS(status)) {
|
|
RtlZeroMemory(seekPenalty, length);
|
|
seekPenalty->Version = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
seekPenalty->Size = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
seekPenalty->IncursSeekPenalty = incursSeekPenalty;
|
|
information = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
|
|
|
|
} else {
|
|
information = 0;
|
|
}
|
|
|
|
} else {
|
|
// case 2: the request is supported and it completes successfully
|
|
// case 3: the request is supported by lower stack but other failure status is returned.
|
|
// from now on, the same request will be send down to lower stack directly.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.SeekPenaltyProperty = Supported;
|
|
information = (ULONG)Irp->IoStatus.Information;
|
|
|
|
}
|
|
|
|
|
|
goto Exit;
|
|
|
|
break;
|
|
}
|
|
|
|
case NotSupported: {
|
|
status = fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.CommandStatus;
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
status = IncursSeekPenalty(fdoExtension->FunctionSupportInfo->DeviceCharacteristicsData.MediumRotationRate, &incursSeekPenalty);
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
RtlZeroMemory(seekPenalty, length);
|
|
seekPenalty->Version = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
seekPenalty->Size = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
seekPenalty->IncursSeekPenalty = incursSeekPenalty;
|
|
information = sizeof(DEVICE_SEEK_PENALTY_DESCRIPTOR);
|
|
|
|
} else {
|
|
information = 0;
|
|
}
|
|
|
|
goto Exit;
|
|
|
|
break;
|
|
}
|
|
|
|
case Supported: {
|
|
NT_ASSERT(FALSE); // this case is handled at the begining of the function.
|
|
break;
|
|
}
|
|
|
|
} // end of switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.SeekPenaltyProperty)
|
|
|
|
Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClasspDeviceGetLBProvisioningVPDPage(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_opt_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
USHORT pageLength = 0;
|
|
|
|
PVOID dataBuffer = NULL;
|
|
UCHAR bufferLength = VPD_MAX_BUFFER_SIZE; // use biggest buffer possible
|
|
ULONG allocationBufferLength = bufferLength; // Since the CDB size may differ from the actual buffer allocation
|
|
PCDB cdb;
|
|
ULONG dataTransferLength = 0;
|
|
PVPD_LOGICAL_BLOCK_PROVISIONING_PAGE lbProvisioning = NULL;
|
|
|
|
//
|
|
// if the informaiton has been attempted to retrieve, return the cached status.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus != -1) {
|
|
// get cached NTSTATUS from previous call.
|
|
return fdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus;
|
|
}
|
|
|
|
//
|
|
// Initialize LBProvisioningData fields to 'unsupported' defaults.
|
|
//
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.ProvisioningType = PROVISIONING_TYPE_UNKNOWN;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPRZ = FALSE;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPU = FALSE;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.ANC_SUP = FALSE;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.ThresholdExponent = 0;
|
|
|
|
//
|
|
// Try to get the Thin Provisioning VPD page (0xB2), if it is supported.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.LBProvisioning == TRUE &&
|
|
Srb != NULL)
|
|
{
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
//
|
|
allocationBufferLength = ALIGN_UP_BY(allocationBufferLength,KeGetRecommendedSharedDataAlignment());
|
|
dataBuffer = ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned, allocationBufferLength,'0CcS');
|
|
#else
|
|
dataBuffer = ExAllocatePoolWithTag(NonPagedPoolNx, bufferLength,'0CcS');
|
|
#endif
|
|
if (dataBuffer == NULL) {
|
|
// return without updating FdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus
|
|
// the field will remain value as "-1", so that the command will be attempted next time this function is called.
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Exit;
|
|
}
|
|
|
|
lbProvisioning = (PVPD_LOGICAL_BLOCK_PROVISIONING_PAGE)dataBuffer;
|
|
|
|
RtlZeroMemory(dataBuffer, allocationBufferLength);
|
|
|
|
if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
|
|
status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)Srb,
|
|
STORAGE_ADDRESS_TYPE_BTL8,
|
|
CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
|
|
1,
|
|
SrbExDataTypeScsiCdb16);
|
|
if (NT_SUCCESS(status)) {
|
|
((PSTORAGE_REQUEST_BLOCK)Srb)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;
|
|
} else {
|
|
//
|
|
// Should not occur.
|
|
//
|
|
NT_ASSERT(FALSE);
|
|
}
|
|
} else {
|
|
RtlZeroMemory(Srb, sizeof(SCSI_REQUEST_BLOCK));
|
|
Srb->Length = sizeof(SCSI_REQUEST_BLOCK);
|
|
Srb->Function = SRB_FUNCTION_EXECUTE_SCSI;
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
// prepare the Srb
|
|
SrbSetTimeOutValue(Srb, fdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, fdoExtension->SrbFlags);
|
|
|
|
SrbSetCdbLength(Srb, 6);
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->CDB6INQUIRY3.OperationCode = SCSIOP_INQUIRY;
|
|
cdb->CDB6INQUIRY3.EnableVitalProductData = 1; //EVPD bit
|
|
cdb->CDB6INQUIRY3.PageCode = VPD_LOGICAL_BLOCK_PROVISIONING;
|
|
cdb->CDB6INQUIRY3.AllocationLength = bufferLength; //AllocationLength field in CDB6INQUIRY3 is only one byte.
|
|
|
|
status = ClassSendSrbSynchronous(fdoExtension->DeviceObject,
|
|
Srb,
|
|
dataBuffer,
|
|
allocationBufferLength,
|
|
FALSE);
|
|
|
|
dataTransferLength = SrbGetDataTransferLength(Srb);
|
|
}
|
|
|
|
//
|
|
// Handle the case where we get back STATUS_DATA_OVERRUN b/c the input
|
|
// buffer was larger than necessary.
|
|
//
|
|
if (status == STATUS_DATA_OVERRUN && dataTransferLength < bufferLength)
|
|
{
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
REVERSE_BYTES_SHORT(&pageLength, &(lbProvisioning->PageLength));
|
|
}
|
|
|
|
if ( NT_SUCCESS(status) &&
|
|
((dataTransferLength < 0x08) ||
|
|
(pageLength < (FIELD_OFFSET(VPD_LOGICAL_BLOCK_PROVISIONING_PAGE, Reserved2) - FIELD_OFFSET(VPD_LOGICAL_BLOCK_PROVISIONING_PAGE,ThresholdExponent))) ||
|
|
(lbProvisioning->PageCode != VPD_LOGICAL_BLOCK_PROVISIONING)) ) {
|
|
// the device should return at least 8 bytes of data for use.
|
|
// 'PageCode' shall match and we need all the relevant data after the header.
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
}
|
|
|
|
//
|
|
// Fill in the FDO extension with either the data from the VPD page, or
|
|
// use defaults if there was an error.
|
|
//
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.ProvisioningType = lbProvisioning->ProvisioningType;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPRZ = lbProvisioning->LBPRZ;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPU = lbProvisioning->LBPU;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.ANC_SUP = lbProvisioning->ANC_SUP;
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.ThresholdExponent = lbProvisioning->ThresholdExponent;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_PNP,
|
|
"ClasspDeviceGetLBProvisioningVPDPage (%p): %s %s (rev %s) reported following parameters: \
|
|
\n\t\t\tProvisioningType: %u \
|
|
\n\t\t\tLBPRZ: %u \
|
|
\n\t\t\tLBPU: %u \
|
|
\n\t\t\tANC_SUP: %I64u \
|
|
\n\t\t\tThresholdExponent: %u\n",
|
|
DeviceObject,
|
|
(PCSZ)(((PUCHAR)fdoExtension->DeviceDescriptor) + fdoExtension->DeviceDescriptor->VendorIdOffset),
|
|
(PCSZ)(((PUCHAR)fdoExtension->DeviceDescriptor) + fdoExtension->DeviceDescriptor->ProductIdOffset),
|
|
(PCSZ)(((PUCHAR)fdoExtension->DeviceDescriptor) + fdoExtension->DeviceDescriptor->ProductRevisionOffset),
|
|
lbProvisioning->ProvisioningType,
|
|
lbProvisioning->LBPRZ,
|
|
lbProvisioning->LBPU,
|
|
lbProvisioning->ANC_SUP,
|
|
lbProvisioning->ThresholdExponent));
|
|
}
|
|
} else {
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
}
|
|
|
|
fdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus = status;
|
|
|
|
Exit:
|
|
FREE_POOL(dataBuffer);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
NTSTATUS ClasspDeviceGetBlockLimitsVPDPage(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_Inout_bytecount_(SrbSize) PSCSI_REQUEST_BLOCK Srb,
|
|
_In_ ULONG SrbSize,
|
|
_Out_ PCLASS_VPD_B0_DATA BlockLimitsData
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
PVOID dataBuffer = NULL;
|
|
UCHAR bufferLength = VPD_MAX_BUFFER_SIZE; // use biggest buffer possible
|
|
ULONG allocationBufferLength = bufferLength;
|
|
PCDB cdb;
|
|
PVPD_BLOCK_LIMITS_PAGE blockLimits = NULL;
|
|
ULONG dataTransferLength = 0;
|
|
|
|
//
|
|
// Set default values for UNMAP parameters based upon UNMAP support or lack
|
|
// thereof.
|
|
//
|
|
if (FdoExtension->FunctionSupportInfo->LBProvisioningData.LBPU) {
|
|
//
|
|
// If UNMAP is supported, we default to the maximum LBA count and
|
|
// block descriptor count. We also default the UNMAP granularity to
|
|
// a single block and specify no granularity alignment.
|
|
//
|
|
BlockLimitsData->MaxUnmapLbaCount = (ULONG)-1;
|
|
BlockLimitsData->MaxUnmapBlockDescrCount = (ULONG)-1;
|
|
BlockLimitsData->OptimalUnmapGranularity = 1;
|
|
BlockLimitsData->UnmapGranularityAlignment = 0;
|
|
BlockLimitsData->UGAVALID = FALSE;
|
|
} else {
|
|
BlockLimitsData->MaxUnmapLbaCount = 0;
|
|
BlockLimitsData->MaxUnmapBlockDescrCount = 0;
|
|
BlockLimitsData->OptimalUnmapGranularity = 0;
|
|
BlockLimitsData->UnmapGranularityAlignment = 0;
|
|
BlockLimitsData->UGAVALID = FALSE;
|
|
}
|
|
|
|
//
|
|
// Try to get the Block Limits VPD page (0xB0), if it is supported.
|
|
//
|
|
if (FdoExtension->FunctionSupportInfo->ValidInquiryPages.BlockLimits == TRUE)
|
|
{
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
allocationBufferLength = ALIGN_UP_BY(allocationBufferLength, KeGetRecommendedSharedDataAlignment());
|
|
dataBuffer = ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned, allocationBufferLength, '0CcS');
|
|
#else
|
|
dataBuffer = ExAllocatePoolWithTag(NonPagedPoolNx, bufferLength, '0CcS');
|
|
#endif
|
|
if (dataBuffer == NULL)
|
|
{
|
|
// return without updating FdoExtension->FunctionSupportInfo->BlockLimitsData.CommandStatus
|
|
// the field will remain value as "-1", so that the command will be attempted next time this function is called.
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Exit;
|
|
}
|
|
|
|
blockLimits = (PVPD_BLOCK_LIMITS_PAGE)dataBuffer;
|
|
|
|
RtlZeroMemory(dataBuffer, allocationBufferLength);
|
|
|
|
if (FdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(suppress:26015, "InitializeStorageRequestBlock ensures buffer access is bounded")
|
|
#endif
|
|
status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)Srb,
|
|
STORAGE_ADDRESS_TYPE_BTL8,
|
|
SrbSize,
|
|
1,
|
|
SrbExDataTypeScsiCdb16);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
((PSTORAGE_REQUEST_BLOCK)Srb)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;
|
|
} else {
|
|
//
|
|
// Should not occur.
|
|
//
|
|
NT_ASSERT(FALSE);
|
|
}
|
|
} else {
|
|
RtlZeroMemory(Srb, sizeof(SCSI_REQUEST_BLOCK));
|
|
Srb->Length = sizeof(SCSI_REQUEST_BLOCK);
|
|
Srb->Function = SRB_FUNCTION_EXECUTE_SCSI;
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
// prepare the Srb
|
|
SrbSetTimeOutValue(Srb, FdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, FdoExtension->SrbFlags);
|
|
|
|
SrbSetCdbLength(Srb, 6);
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->CDB6INQUIRY3.OperationCode = SCSIOP_INQUIRY;
|
|
cdb->CDB6INQUIRY3.EnableVitalProductData = 1; //EVPD bit
|
|
cdb->CDB6INQUIRY3.PageCode = VPD_BLOCK_LIMITS;
|
|
cdb->CDB6INQUIRY3.AllocationLength = bufferLength; //AllocationLength field in CDB6INQUIRY3 is only one byte.
|
|
|
|
status = ClassSendSrbSynchronous(FdoExtension->DeviceObject,
|
|
Srb,
|
|
dataBuffer,
|
|
allocationBufferLength,
|
|
FALSE);
|
|
dataTransferLength = SrbGetDataTransferLength(Srb);
|
|
}
|
|
|
|
//
|
|
// Handle the case where we get back STATUS_DATA_OVERRUN b/c the input
|
|
// buffer was larger than necessary.
|
|
//
|
|
|
|
if (status == STATUS_DATA_OVERRUN && dataTransferLength < bufferLength)
|
|
{
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
USHORT pageLength;
|
|
REVERSE_BYTES_SHORT(&pageLength, &(blockLimits->PageLength));
|
|
|
|
//
|
|
// Regardless of the device's support for unmap, cache away at least the basic block limits information
|
|
//
|
|
if (dataTransferLength >= 0x10 && blockLimits->PageCode == VPD_BLOCK_LIMITS) {
|
|
|
|
// (6:7) OPTIMAL TRANSFER LENGTH GRANULARITY
|
|
REVERSE_BYTES_SHORT(&BlockLimitsData->OptimalTransferLengthGranularity, &blockLimits->OptimalTransferLengthGranularity);
|
|
// (8:11) MAXIMUM TRANSFER LENGTH
|
|
REVERSE_BYTES(&BlockLimitsData->MaximumTransferLength, &blockLimits->MaximumTransferLength);
|
|
// (12:15) OPTIMAL TRANSFER LENGTH
|
|
REVERSE_BYTES(&BlockLimitsData->OptimalTransferLength, &blockLimits->OptimalTransferLength);
|
|
}
|
|
|
|
if ((dataTransferLength < 0x24) ||
|
|
(pageLength < (FIELD_OFFSET(VPD_BLOCK_LIMITS_PAGE,Reserved1) - FIELD_OFFSET(VPD_BLOCK_LIMITS_PAGE,Reserved0))) ||
|
|
(blockLimits->PageCode != VPD_BLOCK_LIMITS))
|
|
{
|
|
// the device should return at least 36 bytes of data for use.
|
|
// 'PageCode' shall match and we need all the relevant data after the header.
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
}
|
|
}
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
// cache data into FdoExtension
|
|
// (20:23) MAXIMUM UNMAP LBA COUNT
|
|
REVERSE_BYTES(&BlockLimitsData->MaxUnmapLbaCount, &blockLimits->MaximumUnmapLBACount);
|
|
// (24:27) MAXIMUM UNMAP BLOCK DESCRIPTOR COUNT
|
|
REVERSE_BYTES(&BlockLimitsData->MaxUnmapBlockDescrCount, &blockLimits->MaximumUnmapBlockDescriptorCount);
|
|
// (28:31) OPTIMAL UNMAP GRANULARITY
|
|
REVERSE_BYTES(&BlockLimitsData->OptimalUnmapGranularity, &blockLimits->OptimalUnmapGranularity);
|
|
|
|
// (32:35) UNMAP GRANULARITY ALIGNMENT; (32) bit7: UGAVALID
|
|
BlockLimitsData->UGAVALID = blockLimits->UGAValid;
|
|
if (BlockLimitsData->UGAVALID == TRUE) {
|
|
REVERSE_BYTES(&BlockLimitsData->UnmapGranularityAlignment, &blockLimits->UnmapGranularityAlignment);
|
|
BlockLimitsData->UnmapGranularityAlignment &= 0x7FFFFFFF; // remove value of UGAVALID bit.
|
|
} else {
|
|
BlockLimitsData->UnmapGranularityAlignment = 0;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_PNP,
|
|
"ClasspDeviceGetBlockLimitsVPDPage (%p): %s %s (rev %s) reported following parameters: \
|
|
\n\t\t\tOptimalTransferLengthGranularity: %u \
|
|
\n\t\t\tMaximumTransferLength: %u \
|
|
\n\t\t\tOptimalTransferLength: %u \
|
|
\n\t\t\tMaximumUnmapLBACount: %u \
|
|
\n\t\t\tMaximumUnmapBlockDescriptorCount: %u \
|
|
\n\t\t\tOptimalUnmapGranularity: %u \
|
|
\n\t\t\tUGAValid: %u \
|
|
\n\t\t\tUnmapGranularityAlignment: %u\n",
|
|
FdoExtension->DeviceObject,
|
|
(PCSZ)(((PUCHAR)FdoExtension->DeviceDescriptor) + FdoExtension->DeviceDescriptor->VendorIdOffset),
|
|
(PCSZ)(((PUCHAR)FdoExtension->DeviceDescriptor) + FdoExtension->DeviceDescriptor->ProductIdOffset),
|
|
(PCSZ)(((PUCHAR)FdoExtension->DeviceDescriptor) + FdoExtension->DeviceDescriptor->ProductRevisionOffset),
|
|
BlockLimitsData->OptimalTransferLengthGranularity,
|
|
BlockLimitsData->MaximumTransferLength,
|
|
BlockLimitsData->OptimalTransferLength,
|
|
BlockLimitsData->MaxUnmapLbaCount,
|
|
BlockLimitsData->MaxUnmapBlockDescrCount,
|
|
BlockLimitsData->OptimalUnmapGranularity,
|
|
BlockLimitsData->UGAVALID,
|
|
BlockLimitsData->UnmapGranularityAlignment));
|
|
|
|
}
|
|
} else {
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
}
|
|
|
|
BlockLimitsData->CommandStatus = status;
|
|
|
|
Exit:
|
|
FREE_POOL(dataBuffer);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
NTSTATUS ClasspDeviceTrimProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
At first time of receiving the request, this function will forward it to lower stack to determine if it's supportted.
|
|
If it's not supported, INQUIRY (Block Limits VPD page) will be sent down to retrieve the information.
|
|
*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PSTORAGE_PROPERTY_QUERY query = (PSTORAGE_PROPERTY_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
ULONG length = 0;
|
|
ULONG information = 0;
|
|
|
|
PDEVICE_TRIM_DESCRIPTOR trimDescr;
|
|
|
|
UNREFERENCED_PARAMETER(Srb);
|
|
|
|
if ( (DeviceObject->DeviceType != FILE_DEVICE_DISK) ||
|
|
(TEST_FLAG(DeviceObject->Characteristics, FILE_FLOPPY_DISKETTE)) ||
|
|
(fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProperty == Supported) ) {
|
|
// if it's not disk, forward the request to lower layer,
|
|
// if the IOCTL is supported by lower stack, forward it down.
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
} else if (query->QueryType != PropertyStandardQuery) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Request validation.
|
|
// Note that InputBufferLength and IsFdo have been validated beforing entering this routine.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto Exit;
|
|
}
|
|
|
|
// do not touch this buffer because it can still be used as input buffer for lower layer in 'SupportUnknown' case.
|
|
trimDescr = (PDEVICE_TRIM_DESCRIPTOR)Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (length < sizeof(DEVICE_TRIM_DESCRIPTOR)) {
|
|
|
|
if (length >= sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
trimDescr->Version = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
trimDescr->Size = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
}
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// note that 'Supported' case has been handled at the beginning of this function.
|
|
//
|
|
switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProperty) {
|
|
case SupportUnknown: {
|
|
// send down request and wait for the request to complete.
|
|
status = ClassForwardIrpSynchronous(commonExtension, Irp);
|
|
|
|
if ( (status == STATUS_NOT_SUPPORTED) ||
|
|
(status == STATUS_NOT_IMPLEMENTED) ||
|
|
(status == STATUS_INVALID_DEVICE_REQUEST) ||
|
|
(status == STATUS_INVALID_PARAMETER_1) ) {
|
|
// case 1: the request is not supported by lower layer, sends down command
|
|
// some port drivers (or filter drivers) return STATUS_INVALID_DEVICE_REQUEST if a request is not supported.
|
|
status = fdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus;
|
|
NT_ASSERT(status != -1);
|
|
|
|
// data is ready in fdoExtension
|
|
// set the support status after the SCSI command is executed to avoid racing condition between multiple same type of requests.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProperty = NotSupported;
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
// fill output buffer
|
|
RtlZeroMemory(trimDescr, length);
|
|
trimDescr->Version = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
trimDescr->Size = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
trimDescr->TrimEnabled = ClasspSupportsUnmap(fdoExtension->FunctionSupportInfo);
|
|
|
|
// set returned data length
|
|
information = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
} else {
|
|
// there was error retrieving TrimProperty. Surface the error up from 'status' variable.
|
|
information = 0;
|
|
}
|
|
goto Exit;
|
|
} else {
|
|
// case 2: the request is supported and it completes successfully
|
|
// case 3: the request is supported by lower stack but other failure status is returned.
|
|
// from now on, the same request will be send down to lower stack directly.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProperty = Supported;
|
|
information = (ULONG)Irp->IoStatus.Information;
|
|
goto Exit;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case NotSupported: {
|
|
status = fdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus;
|
|
NT_ASSERT(status != -1);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
RtlZeroMemory(trimDescr, length);
|
|
trimDescr->Version = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
trimDescr->Size = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
trimDescr->TrimEnabled = ClasspSupportsUnmap(fdoExtension->FunctionSupportInfo);
|
|
|
|
information = sizeof(DEVICE_TRIM_DESCRIPTOR);
|
|
} else {
|
|
information = 0;
|
|
}
|
|
goto Exit;
|
|
|
|
break;
|
|
}
|
|
|
|
case Supported: {
|
|
NT_ASSERT(FALSE); // this case is handled at the begining of the function.
|
|
break;
|
|
}
|
|
|
|
} // end of switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProperty)
|
|
|
|
Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClasspDeviceLBProvisioningProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
NTSTATUS blockLimitsStatus;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PSTORAGE_PROPERTY_QUERY query = (PSTORAGE_PROPERTY_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
ULONG length = 0;
|
|
ULONG information = 0;
|
|
CLASS_VPD_B0_DATA blockLimitsData;
|
|
ULONG generationCount;
|
|
|
|
PDEVICE_LB_PROVISIONING_DESCRIPTOR lbpDescr;
|
|
|
|
UNREFERENCED_PARAMETER(Srb);
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
} else if (query->QueryType != PropertyStandardQuery) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Request validation.
|
|
// Note that InputBufferLength and IsFdo have been validated beforing entering this routine.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto Exit;
|
|
}
|
|
|
|
lbpDescr = (PDEVICE_LB_PROVISIONING_DESCRIPTOR)Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
RtlZeroMemory(lbpDescr, length);
|
|
|
|
if (length < DEVICE_LB_PROVISIONING_DESCRIPTOR_V1_SIZE) {
|
|
|
|
if (length >= sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
lbpDescr->Version = sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR);
|
|
lbpDescr->Size = sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR);
|
|
status = STATUS_SUCCESS;
|
|
goto Exit;
|
|
}
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Set the structure version/size based upon the size of the given output
|
|
// buffer. We may be working with an older component that was built with
|
|
// the V1 structure definition.
|
|
//
|
|
if (length < sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR)) {
|
|
lbpDescr->Version = DEVICE_LB_PROVISIONING_DESCRIPTOR_V1_SIZE;
|
|
lbpDescr->Size = DEVICE_LB_PROVISIONING_DESCRIPTOR_V1_SIZE;
|
|
information = DEVICE_LB_PROVISIONING_DESCRIPTOR_V1_SIZE;
|
|
} else {
|
|
lbpDescr->Version = sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR);
|
|
lbpDescr->Size = sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR);
|
|
information = sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR);
|
|
}
|
|
|
|
//
|
|
// Take a snapshot of the block limits data since it can change.
|
|
// If we failed to get the block limits data, we'll just set the Optimal
|
|
// Unmap Granularity (and alignment) will default to 0. We don't want to
|
|
// fail the request outright since there is some non-block limits data that
|
|
// we can return.
|
|
//
|
|
blockLimitsStatus = ClasspBlockLimitsDataSnapshot(fdoExtension,
|
|
TRUE,
|
|
&blockLimitsData,
|
|
&generationCount);
|
|
|
|
//
|
|
// Fill in the output buffer. All data is copied from the FDO extension where we
|
|
// cached Logical Block Provisioning info when the device was first initialized.
|
|
//
|
|
|
|
lbpDescr->ThinProvisioningEnabled = ClasspIsThinProvisioned(fdoExtension->FunctionSupportInfo);
|
|
|
|
//
|
|
// Make sure we have a non-zero value for the number of bytes per block.
|
|
//
|
|
if (fdoExtension->DiskGeometry.BytesPerSector == 0)
|
|
{
|
|
status = ClassReadDriveCapacity(fdoExtension->DeviceObject);
|
|
if(!NT_SUCCESS(status) || fdoExtension->DiskGeometry.BytesPerSector == 0)
|
|
{
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
information = 0;
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
lbpDescr->ThinProvisioningReadZeros = fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPRZ;
|
|
lbpDescr->AnchorSupported = fdoExtension->FunctionSupportInfo->LBProvisioningData.ANC_SUP;
|
|
|
|
if (NT_SUCCESS(blockLimitsStatus)) {
|
|
lbpDescr->UnmapGranularityAlignmentValid = blockLimitsData.UGAVALID;
|
|
|
|
//
|
|
// Granularity and Alignment are given to us in units of blocks,
|
|
// but we convert and return them in bytes as it is more convenient
|
|
// to the caller.
|
|
//
|
|
lbpDescr->OptimalUnmapGranularity = (ULONGLONG)blockLimitsData.OptimalUnmapGranularity * fdoExtension->DiskGeometry.BytesPerSector;
|
|
lbpDescr->UnmapGranularityAlignment = (ULONGLONG)blockLimitsData.UnmapGranularityAlignment * fdoExtension->DiskGeometry.BytesPerSector;
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINBLUE)
|
|
//
|
|
// If the output buffer is large enough (i.e. not a V1 structure) copy
|
|
// over the max UNMAP LBA count and max UNMAP block descriptor count.
|
|
//
|
|
if (length >= sizeof(DEVICE_LB_PROVISIONING_DESCRIPTOR)) {
|
|
lbpDescr->MaxUnmapLbaCount = blockLimitsData.MaxUnmapLbaCount;
|
|
lbpDescr->MaxUnmapBlockDescriptorCount = blockLimitsData.MaxUnmapBlockDescrCount;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
VOID
|
|
ConvertDataSetRangeToUnmapBlockDescr(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_ PUNMAP_BLOCK_DESCRIPTOR BlockDescr,
|
|
_Inout_ PULONG CurrentBlockDescrIndex,
|
|
_In_ ULONG MaxBlockDescrIndex,
|
|
_Inout_ PULONGLONG CurrentLbaCount,
|
|
_In_ ULONGLONG MaxLbaCount,
|
|
_Inout_ PDEVICE_DATA_SET_RANGE DataSetRange
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Convert DEVICE_DATA_SET_RANGE entry to be UNMAP_BLOCK_DESCRIPTOR entries.
|
|
|
|
As LengthInBytes field in DEVICE_DATA_SET_RANGE structure is 64 bits (bytes)
|
|
and LbaCount field in UNMAP_BLOCK_DESCRIPTOR structure is 32 bits (sectors),
|
|
it's possible that one DEVICE_DATA_SET_RANGE entry needs multiple UNMAP_BLOCK_DESCRIPTOR entries.
|
|
We must also take the unmap granularity into consideration and split up the
|
|
the given ranges so that they are aligned with the specified granularity.
|
|
|
|
Arguments:
|
|
All arguments must be validated by the caller.
|
|
|
|
FdoExtension - The FDO extension of the device to which the unmap
|
|
command that will use the resulting unmap block descriptors will be
|
|
sent.
|
|
BlockDescr - Pointer to a buffer that will contain the unmap block
|
|
descriptors. This buffer should be allocated by the caller and the
|
|
caller should also ensure that it is large enough to contain all the
|
|
requested descriptors. Its size is implied by MaxBlockDescrIndex.
|
|
CurrentBlockDescrIndex - This contains the next block descriptor index to
|
|
be processed when this function returns. This function should be called
|
|
again with the same parameter to continue processing.
|
|
MaxBlockDescrIndex - This is the index of the last unmap block descriptor,
|
|
provided so that the function does not go off the end of BlockDescr.
|
|
CurrentLbaCount - This contains the number of LBAs left to be processed
|
|
when this function returns. This function should be called again with
|
|
the same parameter to continue processing.
|
|
MaxLbaCount - This is the max number of LBAs that can be sent in a single
|
|
unmap command.
|
|
DataSetRange - This range will be modified to reflect the un-converted part.
|
|
It must be valid (including being granularity-aligned) when it is first
|
|
passed to this function.
|
|
|
|
Return Value:
|
|
|
|
Count of UNMAP_BLOCK_DESCRIPTOR entries converted.
|
|
|
|
NOTE: if LengthInBytes does not reach to 0, the conversion for DEVICE_DATA_SET_RANGE entry
|
|
is not completed. Further conversion is needed by calling this function again.
|
|
|
|
--*/
|
|
{
|
|
|
|
ULONGLONG startingSector;
|
|
ULONGLONG sectorCount;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"ConvertDataSetRangeToUnmapBlockDescr (%p): Generating UNMAP Block Descriptors from DataSetRange: \
|
|
\n\t\tStartingOffset = %I64u bytes \
|
|
\n\t\tLength = %I64u bytes\n",
|
|
FdoExtension->DeviceObject,
|
|
DataSetRange->StartingOffset,
|
|
DataSetRange->LengthInBytes));
|
|
|
|
while ( (DataSetRange->LengthInBytes > 0) &&
|
|
(*CurrentBlockDescrIndex < MaxBlockDescrIndex) &&
|
|
(*CurrentLbaCount < MaxLbaCount) ) {
|
|
|
|
//
|
|
// Convert the starting offset and length from bytes to blocks.
|
|
//
|
|
startingSector = (ULONGLONG)(DataSetRange->StartingOffset / FdoExtension->DiskGeometry.BytesPerSector);
|
|
sectorCount = (DataSetRange->LengthInBytes / FdoExtension->DiskGeometry.BytesPerSector);
|
|
|
|
//
|
|
// Make sure the sector count isn't more than can be specified with a
|
|
// single descriptor.
|
|
//
|
|
if (sectorCount > MAXULONG) {
|
|
sectorCount = MAXULONG;
|
|
}
|
|
|
|
//
|
|
// The max LBA count is the max number of LBAs that can be unmapped with
|
|
// a single UNMAP command. Make sure we don't exceed this value.
|
|
//
|
|
if ((*CurrentLbaCount + sectorCount) > MaxLbaCount) {
|
|
sectorCount = MaxLbaCount - *CurrentLbaCount;
|
|
}
|
|
|
|
REVERSE_BYTES_QUAD(BlockDescr[*CurrentBlockDescrIndex].StartingLba, &startingSector);
|
|
REVERSE_BYTES(BlockDescr[*CurrentBlockDescrIndex].LbaCount, (PULONG)§orCount);
|
|
|
|
DataSetRange->StartingOffset += sectorCount * FdoExtension->DiskGeometry.BytesPerSector;
|
|
DataSetRange->LengthInBytes -= sectorCount * FdoExtension->DiskGeometry.BytesPerSector;
|
|
|
|
*CurrentBlockDescrIndex += 1;
|
|
*CurrentLbaCount += (ULONG)sectorCount;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"ConvertDataSetRangeToUnmapBlockDescr (%p): Generated UNMAP Block Descriptor: \
|
|
\n\t\t\tStartingLBA = %I64u \
|
|
\n\t\t\tLBACount = %I64u\n",
|
|
FdoExtension->DeviceObject,
|
|
startingSector,
|
|
sectorCount));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
DeviceProcessDsmTrimRequest(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_ PDEVICE_DATA_SET_RANGE DataSetRanges,
|
|
_In_ ULONG DataSetRangesCount,
|
|
_In_ ULONG UnmapGranularity,
|
|
_In_ ULONG SrbFlags,
|
|
_In_ PIRP Irp,
|
|
_In_ PGUID ActivityId,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Process TRIM request that received from upper layer.
|
|
|
|
Arguments:
|
|
|
|
FdoExtension
|
|
DataSetRanges - this parameter must be already validated in caller.
|
|
DataSetRangesCount - this parameter must be already validated in caller.
|
|
UnmapGranularity - The unmap granularity in blocks. This is used to split
|
|
up the unmap command into chunks that are granularity-aligned.
|
|
Srb - The SRB to use for the unmap command. The caller must allocate it,
|
|
but this function will take care of initialzing it.
|
|
|
|
Return Value:
|
|
|
|
status of the operation
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
PUNMAP_LIST_HEADER buffer = NULL;
|
|
PUNMAP_BLOCK_DESCRIPTOR blockDescrPointer;
|
|
ULONG bufferLength;
|
|
ULONG maxBlockDescrCount;
|
|
ULONG neededBlockDescrCount;
|
|
ULONG i;
|
|
|
|
BOOLEAN allDataSetRangeFullyConverted;
|
|
BOOLEAN needToSendCommand;
|
|
BOOLEAN tempDataSetRangeFullyConverted;
|
|
|
|
ULONG dataSetRangeIndex;
|
|
DEVICE_DATA_SET_RANGE tempDataSetRange;
|
|
|
|
ULONG blockDescrIndex;
|
|
ULONGLONG lbaCount;
|
|
ULONGLONG maxLbaCount;
|
|
ULONGLONG maxParameterListLength;
|
|
|
|
|
|
UNREFERENCED_PARAMETER(UnmapGranularity);
|
|
UNREFERENCED_PARAMETER(ActivityId);
|
|
UNREFERENCED_PARAMETER(Irp);
|
|
|
|
//
|
|
// The given LBA ranges are in DEVICE_DATA_SET_RANGE format and need to be converted into UNMAP Block Descriptors.
|
|
// The UNMAP command is able to carry 0xFFFF bytes (0xFFF8 in reality as there are 8 bytes of header plus n*16 bytes of Block Descriptors) of data.
|
|
// The actual size will also be constrained by the Maximum LBA Count and Maximum Transfer Length.
|
|
//
|
|
|
|
//
|
|
// 1.1 Calculate how many Block Descriptors are needed to complete this request.
|
|
//
|
|
neededBlockDescrCount = 0;
|
|
for (i = 0; i < DataSetRangesCount; i++) {
|
|
lbaCount = DataSetRanges[i].LengthInBytes / FdoExtension->DiskGeometry.BytesPerSector;
|
|
|
|
//
|
|
// 1.1.1 the UNMAP_BLOCK_DESCRIPTOR LbaCount is 32 bits, the max value is 0xFFFFFFFF
|
|
//
|
|
if (lbaCount > 0) {
|
|
neededBlockDescrCount += (ULONG)((lbaCount - 1) / MAXULONG + 1);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Honor Max Unmap Block Descriptor Count if it has been specified. Otherwise,
|
|
// use the maximum value that the Parameter List Length field will allow (0xFFFF).
|
|
// If the count is 0xFFFFFFFF, then no maximum is specified.
|
|
//
|
|
if (FdoExtension->FunctionSupportInfo->BlockLimitsData.MaxUnmapBlockDescrCount != 0 &&
|
|
FdoExtension->FunctionSupportInfo->BlockLimitsData.MaxUnmapBlockDescrCount != MAXULONG)
|
|
{
|
|
maxParameterListLength = (ULONGLONG)(FdoExtension->FunctionSupportInfo->BlockLimitsData.MaxUnmapBlockDescrCount * sizeof(UNMAP_BLOCK_DESCRIPTOR))
|
|
+ sizeof(UNMAP_LIST_HEADER);
|
|
|
|
//
|
|
// In the SBC-3, the Max Unmap Block Descriptor Count field in the 0xB0
|
|
// page is 4 bytes and the Parameter List Length in the UNMAP command is
|
|
// 2 bytes, therefore it is possible that the Max Unmap Block Descriptor
|
|
// Count could imply more bytes than can be specified in the Parameter
|
|
// List Length field. Adjust for that here.
|
|
//
|
|
maxParameterListLength = min(maxParameterListLength, MAXUSHORT);
|
|
}
|
|
else
|
|
{
|
|
maxParameterListLength = MAXUSHORT;
|
|
}
|
|
|
|
//
|
|
// 1.2 Calculate the buffer size needed, capped by the device's limitations.
|
|
//
|
|
bufferLength = min(FdoExtension->PrivateFdoData->HwMaxXferLen, (ULONG)maxParameterListLength);
|
|
bufferLength = min(bufferLength, (neededBlockDescrCount * sizeof(UNMAP_BLOCK_DESCRIPTOR) + sizeof(UNMAP_LIST_HEADER)));
|
|
|
|
maxBlockDescrCount = (bufferLength - sizeof(UNMAP_LIST_HEADER)) / sizeof(UNMAP_BLOCK_DESCRIPTOR);
|
|
|
|
if (maxBlockDescrCount == 0) {
|
|
//
|
|
// This shouldn't happen since we've already done validation.
|
|
//
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"DeviceProcessDsmTrimRequest (%p): Max Block Descriptor count is Zero\n",
|
|
FdoExtension->DeviceObject));
|
|
|
|
NT_ASSERT(maxBlockDescrCount != 0);
|
|
status = STATUS_DATA_ERROR;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// The Maximum LBA Count is set during device initialization.
|
|
//
|
|
maxLbaCount = (ULONGLONG)FdoExtension->FunctionSupportInfo->BlockLimitsData.MaxUnmapLbaCount;
|
|
if (maxLbaCount == 0) {
|
|
//
|
|
// This shouldn't happen since we've already done validation.
|
|
//
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"DeviceProcessDsmTrimRequest (%p): Max LBA count is Zero\n",
|
|
FdoExtension->DeviceObject));
|
|
|
|
NT_ASSERT(maxLbaCount != 0);
|
|
status = STATUS_DATA_ERROR;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Finally, allocate the buffer we'll use to send the UNMAP command.
|
|
//
|
|
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
bufferLength = ALIGN_UP_BY(bufferLength,KeGetRecommendedSharedDataAlignment());
|
|
buffer = (PUNMAP_LIST_HEADER)ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned, bufferLength, CLASS_TAG_LB_PROVISIONING);
|
|
#else
|
|
buffer = (PUNMAP_LIST_HEADER)ExAllocatePoolWithTag(NonPagedPoolNx, bufferLength, CLASS_TAG_LB_PROVISIONING);
|
|
#endif
|
|
|
|
if (buffer == NULL) {
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Exit;
|
|
}
|
|
|
|
RtlZeroMemory(buffer, bufferLength);
|
|
|
|
blockDescrPointer = &buffer->Descriptors[0];
|
|
|
|
allDataSetRangeFullyConverted = FALSE;
|
|
needToSendCommand = FALSE;
|
|
tempDataSetRangeFullyConverted = TRUE;
|
|
dataSetRangeIndex = 0;
|
|
RtlZeroMemory(&tempDataSetRange, sizeof(tempDataSetRange));
|
|
|
|
blockDescrIndex = 0;
|
|
lbaCount = 0;
|
|
|
|
|
|
while (!allDataSetRangeFullyConverted) {
|
|
|
|
//
|
|
// If the previous entry conversion completed, go on to the next one;
|
|
// otherwise, continue processing the current entry.
|
|
//
|
|
if (tempDataSetRangeFullyConverted) {
|
|
tempDataSetRange.StartingOffset = DataSetRanges[dataSetRangeIndex].StartingOffset;
|
|
tempDataSetRange.LengthInBytes = DataSetRanges[dataSetRangeIndex].LengthInBytes;
|
|
dataSetRangeIndex++;
|
|
}
|
|
|
|
ConvertDataSetRangeToUnmapBlockDescr(FdoExtension,
|
|
blockDescrPointer,
|
|
&blockDescrIndex,
|
|
maxBlockDescrCount,
|
|
&lbaCount,
|
|
maxLbaCount,
|
|
&tempDataSetRange
|
|
);
|
|
|
|
tempDataSetRangeFullyConverted = (tempDataSetRange.LengthInBytes == 0) ? TRUE : FALSE;
|
|
|
|
allDataSetRangeFullyConverted = tempDataSetRangeFullyConverted && (dataSetRangeIndex == DataSetRangesCount);
|
|
|
|
//
|
|
// Send the UNMAP command when the buffer is full or when all input entries are converted.
|
|
//
|
|
if ((blockDescrIndex == maxBlockDescrCount) || // Buffer full or block descriptor count reached
|
|
(lbaCount == maxLbaCount) || // Block LBA count reached
|
|
allDataSetRangeFullyConverted) { // All DataSetRanges have been converted
|
|
|
|
USHORT transferSize;
|
|
USHORT tempSize;
|
|
PCDB cdb;
|
|
|
|
//
|
|
// Get the transfer size, including the header.
|
|
//
|
|
transferSize = (USHORT)(blockDescrIndex * sizeof(UNMAP_BLOCK_DESCRIPTOR) + sizeof(UNMAP_LIST_HEADER));
|
|
if (transferSize > bufferLength)
|
|
{
|
|
//
|
|
// This should never happen.
|
|
//
|
|
NT_ASSERT(transferSize <= bufferLength);
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
break;
|
|
}
|
|
|
|
tempSize = transferSize - (USHORT)FIELD_OFFSET(UNMAP_LIST_HEADER, BlockDescrDataLength);
|
|
REVERSE_BYTES_SHORT(buffer->DataLength, &tempSize);
|
|
tempSize = transferSize - (USHORT)FIELD_OFFSET(UNMAP_LIST_HEADER, Descriptors[0]);
|
|
REVERSE_BYTES_SHORT(buffer->BlockDescrDataLength, &tempSize);
|
|
|
|
//
|
|
// Initialize the SRB.
|
|
//
|
|
if (FdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
|
|
status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)Srb,
|
|
STORAGE_ADDRESS_TYPE_BTL8,
|
|
CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
|
|
1,
|
|
SrbExDataTypeScsiCdb16);
|
|
if (NT_SUCCESS(status)) {
|
|
((PSTORAGE_REQUEST_BLOCK)Srb)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;
|
|
} else {
|
|
//
|
|
// Should not occur.
|
|
//
|
|
NT_ASSERT(FALSE);
|
|
break;
|
|
}
|
|
|
|
} else {
|
|
RtlZeroMemory(Srb, sizeof(SCSI_REQUEST_BLOCK));
|
|
Srb->Length = sizeof(SCSI_REQUEST_BLOCK);
|
|
Srb->Function = SRB_FUNCTION_EXECUTE_SCSI;
|
|
}
|
|
|
|
//
|
|
// Prepare the Srb
|
|
//
|
|
SrbSetTimeOutValue(Srb, FdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
|
|
//
|
|
// Set the SrbFlags to indicate that it's a data-out operation.
|
|
// Also set any passed-in SrbFlags.
|
|
//
|
|
SrbAssignSrbFlags(Srb, FdoExtension->SrbFlags);
|
|
SrbClearSrbFlags(Srb, SRB_FLAGS_DATA_IN);
|
|
SrbSetSrbFlags(Srb, SRB_FLAGS_DATA_OUT);
|
|
SrbSetSrbFlags(Srb, SrbFlags);
|
|
|
|
SrbSetCdbLength(Srb, 10);
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->UNMAP.OperationCode = SCSIOP_UNMAP;
|
|
cdb->UNMAP.Anchor = 0;
|
|
cdb->UNMAP.GroupNumber = 0;
|
|
cdb->UNMAP.AllocationLength[0] = (UCHAR)(transferSize >> 8);
|
|
cdb->UNMAP.AllocationLength[1] = (UCHAR)transferSize;
|
|
|
|
status = ClassSendSrbSynchronous(FdoExtension->DeviceObject,
|
|
Srb,
|
|
buffer,
|
|
transferSize,
|
|
TRUE);
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"DeviceProcessDsmTrimRequest (%p): UNMAP command issued. Returned NTSTATUS: %!STATUS!.\n",
|
|
FdoExtension->DeviceObject,
|
|
status
|
|
));
|
|
|
|
//
|
|
// Clear the buffer so we can re-use it.
|
|
//
|
|
blockDescrIndex = 0;
|
|
lbaCount = 0;
|
|
RtlZeroMemory(buffer, bufferLength);
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
|
|
FREE_POOL(buffer);
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClasspDeviceTrimProcess(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp,
|
|
_In_ PGUID ActivityId,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
This function is to process IOCTL_STORAGE_MANAGE_DATA_SET_ATTRIBUTES with DeviceDsmAction_Trim.
|
|
At first time of receiving the request, this function will forward it to lower stack to determine if it's supportted.
|
|
If it's not supported, UNMAP (with anchor attribute set) will be sent down to process the request.
|
|
*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
|
|
PDEVICE_MANAGE_DATA_SET_ATTRIBUTES dsmAttributes = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
PDEVICE_DATA_SET_RANGE dataSetRanges;
|
|
ULONG dataSetRangesCount;
|
|
DEVICE_DATA_SET_RANGE entireDataSetRange = {0};
|
|
ULONG i;
|
|
ULONGLONG granularityAlignmentInBytes;
|
|
ULONG granularityInBlocks;
|
|
ULONG srbFlags = 0;
|
|
|
|
CLASS_VPD_B0_DATA blockLimitsData;
|
|
ULONG generationCount;
|
|
|
|
|
|
if ( (DeviceObject->DeviceType != FILE_DEVICE_DISK) ||
|
|
(TEST_FLAG(DeviceObject->Characteristics, FILE_FLOPPY_DISKETTE)) ||
|
|
(fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProcess == Supported) ) {
|
|
// if it's not disk, forward the request to lower layer,
|
|
// if the IOCTL is supported by lower stack, forward it down.
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspDeviceTrimProcess (%p): Lower layer supports Trim DSM IOCTL, forwarding IOCTL.\n",
|
|
DeviceObject));
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// Request validation.
|
|
// Note that InputBufferLength and IsFdo have been validated beforing entering this routine.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
//
|
|
// If the caller has not set the "entire dataset range" flag then at least
|
|
// one dataset range should be specified. However, if the caller *has* set
|
|
// the flag, then there should not be any dataset ranges specified.
|
|
//
|
|
if ((!TEST_FLAG(dsmAttributes->Flags, DEVICE_DSM_FLAG_ENTIRE_DATA_SET_RANGE) &&
|
|
(dsmAttributes->DataSetRangesOffset == 0 ||
|
|
dsmAttributes->DataSetRangesLength == 0)) ||
|
|
(TEST_FLAG(dsmAttributes->Flags, DEVICE_DSM_FLAG_ENTIRE_DATA_SET_RANGE) &&
|
|
(dsmAttributes->DataSetRangesOffset != 0 ||
|
|
dsmAttributes->DataSetRangesLength != 0))) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// note that 'Supported' case has been handled at the beginning of this function.
|
|
//
|
|
switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProcess) {
|
|
case SupportUnknown: {
|
|
// send down request and wait for the request to complete.
|
|
status = ClassForwardIrpSynchronous(commonExtension, Irp);
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspDeviceTrimProcess (%p): Trim DSM IOCTL support unknown. Forwarded IOCTL and received NTSTATUS %!STATUS!.\n",
|
|
DeviceObject,
|
|
status));
|
|
|
|
if (ClasspLowerLayerNotSupport(status)) {
|
|
// case 1: the request is not supported by lower layer, sends down command
|
|
// some port drivers (or filter drivers) return STATUS_INVALID_DEVICE_REQUEST if a request is not supported.
|
|
// In this case we'll just fall through to the NotSupported case so that we can handle it ourselves.
|
|
|
|
//
|
|
// VPD pages 0xB2 and 0xB0 should have been cached in Start Device phase - ClassPnpStartDevice.
|
|
// 0xB2 page: fdoExtension->FunctionSupportInfo->LBProvisioningData;
|
|
// 0xB0 page: fdoExtension->FunctionSupportInfo->BlockLimitsData
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.LBProvisioning == TRUE) {
|
|
NT_ASSERT(fdoExtension->FunctionSupportInfo->LBProvisioningData.CommandStatus != -1);
|
|
}
|
|
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.BlockLimits == TRUE) {
|
|
NT_ASSERT(fdoExtension->FunctionSupportInfo->BlockLimitsData.CommandStatus != -1);
|
|
}
|
|
|
|
} else {
|
|
|
|
// case 2: the request is supported and it completes successfully
|
|
// case 3: the request is supported by lower stack but other failure status is returned.
|
|
// from now on, the same request will be send down to lower stack directly.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProcess = Supported;
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
case NotSupported: {
|
|
|
|
// send UNMAP command if it is supported. don't need to check 'status' value.
|
|
if (ClasspSupportsUnmap(fdoExtension->FunctionSupportInfo))
|
|
{
|
|
//
|
|
// Make sure that we know the bytes per sector (logical block) as it's
|
|
// necessary for calculations involving granularity and alignment.
|
|
//
|
|
if (fdoExtension->DiskGeometry.BytesPerSector == 0) {
|
|
status = ClassReadDriveCapacity(fdoExtension->DeviceObject);
|
|
if(!NT_SUCCESS(status) || fdoExtension->DiskGeometry.BytesPerSector == 0) {
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Take a snapshot of the block limits data since it can change.
|
|
// It's acceptable if the block limits data is outdated since
|
|
// there isn't a hard requirement on the unmap granularity.
|
|
//
|
|
ClasspBlockLimitsDataSnapshot(fdoExtension,
|
|
FALSE,
|
|
&blockLimitsData,
|
|
&generationCount);
|
|
|
|
//
|
|
// Check to see if the Optimal Unmap Granularity and Unmap Granularity
|
|
// Alignment have been specified. If not, default the granularity to
|
|
// one block and the alignment to zero.
|
|
//
|
|
if (blockLimitsData.OptimalUnmapGranularity != 0)
|
|
{
|
|
granularityInBlocks = blockLimitsData.OptimalUnmapGranularity;
|
|
}
|
|
else
|
|
{
|
|
granularityInBlocks = 1;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspDeviceTrimProcess (%p): Optimal Unmap Granularity not provided, defaulted to 1.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
if (blockLimitsData.UGAVALID == TRUE)
|
|
{
|
|
granularityAlignmentInBytes = (ULONGLONG)blockLimitsData.UnmapGranularityAlignment * fdoExtension->DiskGeometry.BytesPerSector;
|
|
}
|
|
else
|
|
{
|
|
granularityAlignmentInBytes = 0;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspDeviceTrimProcess (%p): Unmap Granularity Alignment not provided, defaulted to 0.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
if (TEST_FLAG(dsmAttributes->Flags, DEVICE_DSM_FLAG_ENTIRE_DATA_SET_RANGE))
|
|
{
|
|
//
|
|
// The caller wants to UNMAP the entire disk so we need to build a single
|
|
// dataset range that represents the entire disk.
|
|
//
|
|
entireDataSetRange.StartingOffset = granularityAlignmentInBytes;
|
|
entireDataSetRange.LengthInBytes = (ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart - (ULONGLONG)entireDataSetRange.StartingOffset;
|
|
|
|
dataSetRanges = &entireDataSetRange;
|
|
dataSetRangesCount = 1;
|
|
}
|
|
else
|
|
{
|
|
|
|
dataSetRanges = (PDEVICE_DATA_SET_RANGE)((PUCHAR)dsmAttributes + dsmAttributes->DataSetRangesOffset);
|
|
dataSetRangesCount = dsmAttributes->DataSetRangesLength / sizeof(DEVICE_DATA_SET_RANGE);
|
|
|
|
//
|
|
// Validate the data ranges. Make sure the range is block-aligned,
|
|
// falls in a valid portion of the disk, and is non-zero.
|
|
//
|
|
for (i = 0; i < dataSetRangesCount; i++)
|
|
{
|
|
if ((dataSetRanges[i].StartingOffset % fdoExtension->DiskGeometry.BytesPerSector != 0) ||
|
|
(dataSetRanges[i].LengthInBytes % fdoExtension->DiskGeometry.BytesPerSector != 0) ||
|
|
(dataSetRanges[i].StartingOffset < (LONGLONG)granularityAlignmentInBytes) ||
|
|
(dataSetRanges[i].LengthInBytes == 0) ||
|
|
((ULONGLONG)dataSetRanges[i].StartingOffset + dataSetRanges[i].LengthInBytes > (ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart))
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceTrimProcess (%p): Invalid dataset range. StartingOffset = %I64x, LengthInBytes = %I64x\n",
|
|
DeviceObject,
|
|
dataSetRanges[i].StartingOffset,
|
|
dataSetRanges[i].LengthInBytes));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (!TEST_FLAG(dsmAttributes->Flags, DEVICE_DSM_FLAG_TRIM_NOT_FS_ALLOCATED))
|
|
{
|
|
{
|
|
//
|
|
// For security reasons, file-level TRIM must be forwarded on only
|
|
// if reading the unmapped blocks' contents will return back zeros.
|
|
// This is because if LBPRZ bit is not set, it indicates that a read
|
|
// of unmapped blocks may return "any" data thus potentially leaking
|
|
// in data (into the read buffer) from other blocks.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.LBProvisioning &&
|
|
!fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPRZ) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceTrimProcess (%p): Device does not support file level TRIM.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_TRIM_READ_ZERO_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
// process DSM IOCTL
|
|
status = DeviceProcessDsmTrimRequest(fdoExtension,
|
|
dataSetRanges,
|
|
dataSetRangesCount,
|
|
granularityInBlocks,
|
|
srbFlags,
|
|
Irp,
|
|
ActivityId,
|
|
Srb);
|
|
} else {
|
|
// DSM IOCTL should be completed as not supported
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceTrimProcess (%p): Device does not support UNMAP.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
// set the support status after the SCSI command is executed to avoid racing condition between multiple same type of requests.
|
|
fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProcess = NotSupported;
|
|
|
|
break;
|
|
}
|
|
|
|
case Supported: {
|
|
NT_ASSERT(FALSE); // this case is handled at the begining of the function.
|
|
break;
|
|
}
|
|
|
|
} // end of switch (fdoExtension->FunctionSupportInfo->LowerLayerSupport.TrimProcess)
|
|
|
|
Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = 0;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
GetLBAStatus(
|
|
_In_ PFUNCTIONAL_DEVICE_EXTENSION FdoExtension,
|
|
_In_ PSCSI_REQUEST_BLOCK Srb,
|
|
_In_ ULONGLONG StartingLBA,
|
|
_Inout_ PLBA_STATUS_LIST_HEADER LBAStatusHeader,
|
|
_In_ ULONG LBAStatusSize,
|
|
_In_ BOOLEAN ConsolidateableBlocksOnly
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Send down a Get LBA Status command for the given range.
|
|
|
|
Arguments:
|
|
FdoExtension: The FDO extension of the device to which Get LBA Status will
|
|
be sent.
|
|
Srb: This should be allocated and initialized before it's passed in. It
|
|
will be used for the Get LBA Status command.
|
|
StartingLBA: The LBA that is at the beginning of the requested range.
|
|
LBAStatusHeader: Caller-allocated output buffer.
|
|
LBASTatusSize: Size of the caller-allocated output buffer.
|
|
|
|
Return Value:
|
|
|
|
Status of the operation.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
PCDB cdb;
|
|
|
|
if (LBAStatusHeader == NULL || LBAStatusSize == 0)
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Build and send down the Get LBA Status command.
|
|
//
|
|
SrbSetTimeOutValue(Srb, FdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, FdoExtension->SrbFlags);
|
|
SrbSetCdbLength(Srb, sizeof(cdb->GET_LBA_STATUS));
|
|
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->GET_LBA_STATUS.OperationCode = SCSIOP_GET_LBA_STATUS;
|
|
cdb->GET_LBA_STATUS.ServiceAction = SERVICE_ACTION_GET_LBA_STATUS;
|
|
REVERSE_BYTES_QUAD(&(cdb->GET_LBA_STATUS.StartingLBA), &StartingLBA);
|
|
REVERSE_BYTES(&(cdb->GET_LBA_STATUS.AllocationLength), &LBAStatusSize);
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"GetLBAStatus (%p): sending command with StartingLBA = 0x%I64x, AllocationLength = 0x%I64x, ConsolidateableBlocksOnly = %u\n",
|
|
FdoExtension->DeviceObject,
|
|
StartingLBA,
|
|
LBAStatusSize,
|
|
ConsolidateableBlocksOnly));
|
|
|
|
status = ClassSendSrbSynchronous(FdoExtension->DeviceObject,
|
|
Srb,
|
|
LBAStatusHeader,
|
|
LBAStatusSize,
|
|
FALSE);
|
|
|
|
//
|
|
// Handle the case where we get back STATUS_DATA_OVERRUN b/c the input
|
|
// buffer was larger than necessary.
|
|
//
|
|
if (status == STATUS_DATA_OVERRUN &&
|
|
SrbGetDataTransferLength(Srb) < LBAStatusSize)
|
|
{
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
// log command.
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"GetLBAStatus (%p): command returned NT Status: %!STATUS!\n",
|
|
FdoExtension->DeviceObject,
|
|
status
|
|
));
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
NTSTATUS ClasspDeviceGetLBAStatus(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function is to process IOCTL_STORAGE_MANAGE_DATA_SET_ATTRIBUTES with DeviceDsmAction_Allocation.
|
|
|
|
1. This function will only handle the first dataset range.
|
|
2. This function will not handle dataset ranges whose LengthInBytes is greater than:
|
|
((MAXULONG - sizeof(LBA_STATUS_LIST_HEADER)) / sizeof(LBA_STATUS_DESCRIPTOR)) * BytesPerSlab
|
|
|
|
The input buffer should consist of a DEVICE_MANAGE_DATA_SET_ATTRIBUTES followed
|
|
in memory by a single DEVICE_DATA_SET_RANGE that specifies the requested range
|
|
of slabs for which mapping status is desired.
|
|
|
|
The output buffer will consist of a DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT
|
|
followed in memory by a single DEVICE_DATA_SET_LB_PROVISIONING_STATE that
|
|
contains a bitmap that represents the mapped status of the slabs in the requested
|
|
range. Note that the number of slabs returned may be less than the number
|
|
requested.
|
|
|
|
Thus function will automatically re-align the given range offset if it was
|
|
not slab-aligned. The delta between the given range offset and the properly
|
|
aligned offset will be given in returned DEVICE_DATA_SET_LB_PROVISIONING_STATE.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO of the device to which Get LBA Status will be sent.
|
|
Irp: The IRP for the request. This function will read the input buffer and
|
|
write to the output buffer at the current IRP stack location.
|
|
Srb: This should be allocated and initialized before it's passed in. It
|
|
will be used for the Get LBA Status command.
|
|
|
|
Return Value:
|
|
|
|
STATUS_INVALID_PARAMETER: May be returned under the following conditions:
|
|
- If the requested range was too large. The caller should try again with a
|
|
smaller range. See above for how to calculate the maximum range.
|
|
- If the given starting offset was not within the valid range of the device.
|
|
STATUS_NOT_SUPPORTED: The storage did not report some information critical to
|
|
the execution of this function (e.g. Optimal Unmap Granularity).
|
|
STATUS_BUFFER_TOO_SMALL: The output buffer is not large enough to hold the max
|
|
data that could be returned from this function. If the output buffer is
|
|
at least the size of a ULONG, we will write the required output buffer size
|
|
to the first ULONG bytes of the output buffer.
|
|
STATUS_UNSUCCESSFUL: The Get LBA Status command succeeded but did not
|
|
return data as expected.
|
|
--*/
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PDEVICE_MANAGE_DATA_SET_ATTRIBUTES dsmAttributes = (PDEVICE_MANAGE_DATA_SET_ATTRIBUTES)Irp->AssociatedIrp.SystemBuffer;
|
|
PDEVICE_DATA_SET_RANGE dataSetRanges = NULL;
|
|
PDEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT dsmOutput = (PDEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT)Irp->AssociatedIrp.SystemBuffer;
|
|
ULONG dsmOutputLength;
|
|
NTSTATUS finalStatus;
|
|
NTSTATUS getLBAWorkerStatus;
|
|
ULONG retryCount;
|
|
ULONG retryCountMax;
|
|
CLASS_VPD_B0_DATA blockLimitsData;
|
|
ULONG generationCount1;
|
|
ULONG generationCount2;
|
|
BOOLEAN blockLimitsDataMayHaveChanged;
|
|
ULONG_PTR information = 0;
|
|
LONGLONG startingOffset;
|
|
ULONGLONG lengthInBytes;
|
|
BOOLEAN consolidateableBlocksOnly = FALSE;
|
|
ULONG outputVersion;
|
|
|
|
//
|
|
// Basic parameter validation.
|
|
// Note that InputBufferLength and IsFdo have been validated beforing entering this routine.
|
|
//
|
|
if (dsmOutput == NULL ||
|
|
dsmAttributes == NULL)
|
|
{
|
|
finalStatus = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
|
|
if (TEST_FLAG(dsmAttributes->Flags, DEVICE_DSM_FLAG_ENTIRE_DATA_SET_RANGE)) {
|
|
//
|
|
// The caller wants the mapping status of the entire disk.
|
|
//
|
|
ULONG unmapGranularityAlignment = 0;
|
|
if (fdoExtension->FunctionSupportInfo->BlockLimitsData.UGAVALID) {
|
|
unmapGranularityAlignment = fdoExtension->FunctionSupportInfo->BlockLimitsData.UnmapGranularityAlignment;
|
|
}
|
|
startingOffset = unmapGranularityAlignment;
|
|
lengthInBytes = (ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart - (ULONGLONG)startingOffset;
|
|
} else {
|
|
if (dsmAttributes->DataSetRangesOffset == 0 ||
|
|
dsmAttributes->DataSetRangesLength == 0) {
|
|
finalStatus = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// We only service the first dataset range specified.
|
|
//
|
|
dataSetRanges = (PDEVICE_DATA_SET_RANGE)((PUCHAR)dsmAttributes + dsmAttributes->DataSetRangesOffset);
|
|
startingOffset = dataSetRanges[0].StartingOffset;
|
|
lengthInBytes = dataSetRanges[0].LengthInBytes;
|
|
}
|
|
|
|
|
|
//
|
|
// See if the sender is requesting a specific version of the output data
|
|
// structure. Othwerwise, default to V1.
|
|
//
|
|
outputVersion = DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V1;
|
|
#if (NTDDI_VERSION >= NTDDI_WINBLUE)
|
|
if ((dsmAttributes->ParameterBlockOffset >= sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES)) &&
|
|
(dsmAttributes->ParameterBlockLength >= sizeof(DEVICE_DATA_SET_LBP_STATE_PARAMETERS))) {
|
|
PDEVICE_DATA_SET_LBP_STATE_PARAMETERS parameters = Add2Ptr(dsmAttributes, dsmAttributes->ParameterBlockOffset);
|
|
if ((parameters->Version == DEVICE_DATA_SET_LBP_STATE_PARAMETERS_VERSION_V1) &&
|
|
(parameters->Size >= sizeof(DEVICE_DATA_SET_LBP_STATE_PARAMETERS))) {
|
|
|
|
outputVersion = parameters->OutputVersion;
|
|
|
|
if ((outputVersion != DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V1) &&
|
|
(outputVersion != DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V2)) {
|
|
finalStatus = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// Take a snapshot of the block limits data for the worker function to use.
|
|
// We need to fail the request if we fail to get updated block limits data
|
|
// since we need an accurate Optimal Unmap Granularity value to properly
|
|
// convert the returned mapping descriptors into a bitmap.
|
|
//
|
|
finalStatus = ClasspBlockLimitsDataSnapshot(fdoExtension,
|
|
TRUE,
|
|
&blockLimitsData,
|
|
&generationCount1);
|
|
|
|
if (!NT_SUCCESS(finalStatus)) {
|
|
information = 0;
|
|
goto Exit;
|
|
}
|
|
|
|
if (dsmAttributes->Flags & DEVICE_DSM_FLAG_ALLOCATION_CONSOLIDATEABLE_ONLY) {
|
|
consolidateableBlocksOnly = TRUE;
|
|
}
|
|
|
|
//
|
|
// The retry logic is to handle the case when block limits data changes during rare occasions
|
|
// (e.g. diff-VHD fork or merge).
|
|
//
|
|
retryCountMax = GET_LBA_STATUS_RETRY_COUNT_MAX;
|
|
for (retryCount = 0; retryCount < retryCountMax; retryCount++) {
|
|
|
|
dsmOutputLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
getLBAWorkerStatus = ClasspDeviceGetLBAStatusWorker(DeviceObject,
|
|
&blockLimitsData,
|
|
startingOffset,
|
|
lengthInBytes,
|
|
dsmOutput,
|
|
&dsmOutputLength,
|
|
Srb,
|
|
consolidateableBlocksOnly,
|
|
outputVersion,
|
|
&blockLimitsDataMayHaveChanged);
|
|
|
|
if (!NT_SUCCESS(getLBAWorkerStatus) && !blockLimitsDataMayHaveChanged) {
|
|
information = 0;
|
|
finalStatus = getLBAWorkerStatus;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Again, we need to fail the request if we fail to get updated block
|
|
// limits data since we need an accurate Optimal Unmap Granularity value.
|
|
//
|
|
finalStatus = ClasspBlockLimitsDataSnapshot(fdoExtension,
|
|
TRUE,
|
|
&blockLimitsData,
|
|
&generationCount2);
|
|
if (!NT_SUCCESS(finalStatus)) {
|
|
information = 0;
|
|
goto Exit;
|
|
}
|
|
|
|
if (generationCount1 == generationCount2) {
|
|
//
|
|
// Block limits data stays the same during the call to ClasspDeviceGetLBAStatusWorker()
|
|
// The result from ClasspDeviceGetLBAStatusWorker() is valid.
|
|
//
|
|
finalStatus = getLBAWorkerStatus;
|
|
if (NT_SUCCESS(finalStatus)) {
|
|
information = dsmOutputLength;
|
|
}
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Try again with the latest block limits data
|
|
//
|
|
generationCount1 = generationCount2;
|
|
information = 0;
|
|
finalStatus = STATUS_DEVICE_DATA_ERROR;
|
|
}
|
|
|
|
Exit:
|
|
Irp->IoStatus.Information = information;
|
|
Irp->IoStatus.Status = finalStatus;
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
return finalStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
ClasspDeviceGetLBAStatusWorker(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PCLASS_VPD_B0_DATA BlockLimitsData,
|
|
_In_ ULONGLONG StartingOffset,
|
|
_In_ ULONGLONG LengthInBytes,
|
|
_Out_ PDEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT DsmOutput,
|
|
_Inout_ PULONG DsmOutputLength,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb,
|
|
_In_ BOOLEAN ConsolidateableBlocksOnly,
|
|
_In_ ULONG OutputVersion,
|
|
_Out_ PBOOLEAN BlockLimitsDataMayHaveChanged
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function is to process IOCTL_STORAGE_MANAGE_DATA_SET_ATTRIBUTES with DeviceDsmAction_Allocation.
|
|
|
|
1. This function will only handle the first dataset range.
|
|
2. This function will not handle dataset ranges whose LengthInBytes is greater than:
|
|
((MAXULONG - sizeof(LBA_STATUS_LIST_HEADER)) / sizeof(LBA_STATUS_DESCRIPTOR)) * BytesPerSlab
|
|
|
|
The input buffer should consist of a DEVICE_MANAGE_DATA_SET_ATTRIBUTES followed
|
|
in memory by a single DEVICE_DATA_SET_RANGE that specifies the requested range
|
|
of slabs for which mapping status is desired.
|
|
|
|
The output buffer will consist of a DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT
|
|
followed in memory by a single DEVICE_DATA_SET_LB_PROVISIONING_STATE that
|
|
contains a bitmap that represents the mapped status of the slabs in the requested
|
|
range. Note that the number of slabs returned may be less than the number
|
|
requested.
|
|
|
|
Thus function will automatically re-align the given range offset if it was
|
|
not slab-aligned. The delta between the given range offset and the properly
|
|
aligned offset will be given in returned DEVICE_DATA_SET_LB_PROVISIONING_STATE.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO of the device to which Get LBA Status will be sent.
|
|
BlockLimitsData: Block limits data of the device
|
|
StartingOffset: Starting byte offset of byte range to query LBA status (must be sector aligned)
|
|
LengthInBytes: Length of byte range to query LBA status (multiple of sector size)
|
|
DsmOutput: Output data buffer
|
|
DsmOutputLength: output data buffer size. It will be updated with actual bytes used.
|
|
Srb: This should be allocated and initialized before it's passed in. It
|
|
will be used for the Get LBA Status command.
|
|
ConsolidateableBlocksOnly: Only blocks that are eligible for consolidation
|
|
should be returned.
|
|
OutputVersion: The version of the DEVICE_DATA_SET_LB_PROVISIONING_STATE
|
|
structure to return. This should be one of:
|
|
- DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V1
|
|
- DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V2
|
|
BlockLimitsDataMayHaveChanged: if this function fails, this flag indicates
|
|
if the failure can be caused by changes in device's block limit data.
|
|
|
|
Return Value:
|
|
|
|
STATUS_INVALID_PARAMETER: May be returned under the following conditions:
|
|
- If the requested range was too large. The caller should try again with a
|
|
smaller range. See above for how to calculate the maximum range.
|
|
- If the given starting offset was not within the valid range of the device.
|
|
STATUS_NOT_SUPPORTED: The storage did not report some information critical to
|
|
the execution of this function (e.g. Optimal Unmap Granularity).
|
|
STATUS_BUFFER_TOO_SMALL: The output buffer is not large enough to hold the max
|
|
data that could be returned from this function. If the output buffer is
|
|
at least the size of a ULONG, we will write the required output buffer size
|
|
to the first ULONG bytes of the output buffer.
|
|
STATUS_DEVICE_DATA_ERROR: The Get LBA Status command succeeded but did not
|
|
return data as expected.
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
|
|
PDEVICE_DATA_SET_LB_PROVISIONING_STATE lbpState;
|
|
ULONG bitMapGranularityInBits = FIELD_SIZE(DEVICE_DATA_SET_LB_PROVISIONING_STATE,SlabAllocationBitMap[0]) * 8;
|
|
ULONG requiredOutputLength;
|
|
ULONG outputLength = *DsmOutputLength;
|
|
|
|
ULONG blocksPerSlab;
|
|
ULONGLONG bytesPerSlab;
|
|
ULONGLONG alignmentInBytes = 0;
|
|
ULONG alignmentInBlocks = 0;
|
|
ULONG maxBufferSize;
|
|
ULONG maxSlabs;
|
|
ULONGLONG requestedSlabs; // Total number of slabs requested by the caller.
|
|
ULONGLONG startingLBA;
|
|
ULONGLONG startingOffsetDelta;
|
|
ULONG totalProcessedSlabs = 0; // Total number of slabs we processed.
|
|
ULONGLONG slabsPerCommand; // Number of slabs we can ask for in one Get LBA Status command.
|
|
BOOLEAN doneProcessing = FALSE; // Indicates we should break out of the Get LBA Status loop.
|
|
|
|
ULONG lbaStatusSize;
|
|
PLBA_STATUS_LIST_HEADER lbaStatusListHeader = NULL;
|
|
|
|
//
|
|
// This function can fail if the block limits data on the device changes.
|
|
// This flag tells the caller if it should retry with a newer block limits data
|
|
//
|
|
*BlockLimitsDataMayHaveChanged = FALSE;
|
|
|
|
//
|
|
// Make sure we're running at PASSIVE_LEVEL
|
|
//
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL)
|
|
{
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Don't send down a Get LBA Status command if UNMAP isn't supported.
|
|
//
|
|
if (!fdoExtension->FunctionSupportInfo->LBProvisioningData.LBPU)
|
|
{
|
|
return STATUS_NOT_SUPPORTED;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Make sure we have a non-zero value for the number of bytes per block.
|
|
// Otherwise we will end up dividing by zero later on.
|
|
//
|
|
if (fdoExtension->DiskGeometry.BytesPerSector == 0)
|
|
{
|
|
status = ClassReadDriveCapacity(fdoExtension->DeviceObject);
|
|
if(!NT_SUCCESS(status) || fdoExtension->DiskGeometry.BytesPerSector == 0)
|
|
{
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// We only service the first dataset range specified.
|
|
//
|
|
if (BlockLimitsData->UGAVALID == TRUE) {
|
|
alignmentInBlocks = BlockLimitsData->UnmapGranularityAlignment;
|
|
alignmentInBytes = (ULONGLONG)alignmentInBlocks * (ULONGLONG)fdoExtension->DiskGeometry.BytesPerSector;
|
|
}
|
|
|
|
//
|
|
// Make sure the specified range is valid. The Unmap Granularity Alignment
|
|
// defines a region at the beginning of the disk that cannot be
|
|
// mapped/unmapped so the specified range should not include any part of that
|
|
// region.
|
|
//
|
|
if (LengthInBytes == 0 ||
|
|
StartingOffset < alignmentInBytes ||
|
|
StartingOffset + LengthInBytes > (ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart)
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Invalid range, length is %I64u bytes, starting offset is %I64u bytes, Unmap alignment is %I64u bytes, and disk size is %I64u bytes\n",
|
|
DeviceObject,
|
|
LengthInBytes,
|
|
StartingOffset,
|
|
alignmentInBytes,
|
|
(ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Calculate the number of bytes per slab so that we can convert (and
|
|
// possibly align) the given offset (given in bytes) to slabs.
|
|
//
|
|
blocksPerSlab = BlockLimitsData->OptimalUnmapGranularity;
|
|
bytesPerSlab = (ULONGLONG)blocksPerSlab * (ULONGLONG)fdoExtension->DiskGeometry.BytesPerSector;
|
|
|
|
//
|
|
// If the starting offset is not slab-aligned, we need to adjust it to
|
|
// be aligned with the next highest slab. We also need to save the delta
|
|
// to return to the user later.
|
|
//
|
|
if (((StartingOffset - alignmentInBytes) % bytesPerSlab) != 0)
|
|
{
|
|
startingLBA = (((StartingOffset - alignmentInBytes) / bytesPerSlab) + 1) * (ULONGLONG)blocksPerSlab + alignmentInBlocks;
|
|
startingOffsetDelta = (startingLBA * fdoExtension->DiskGeometry.BytesPerSector) - StartingOffset;
|
|
}
|
|
else
|
|
{
|
|
startingLBA = ((StartingOffset - alignmentInBytes) / bytesPerSlab) * (ULONGLONG)blocksPerSlab + alignmentInBlocks;
|
|
startingOffsetDelta = 0;
|
|
}
|
|
|
|
//
|
|
// Caclulate the number of slabs the caller requested.
|
|
//
|
|
if ((LengthInBytes % bytesPerSlab) == 0) {
|
|
requestedSlabs = (LengthInBytes / bytesPerSlab);
|
|
} else {
|
|
//
|
|
// Round up the number of requested slabs if the length indicates a
|
|
// partial slab. This should cover the case where the user specifies
|
|
// a dataset range for the whole disk, but the size of the disk is not
|
|
// a slab-multiple. Rounding up allows us to return the status of the
|
|
// partial slab
|
|
//
|
|
requestedSlabs = (LengthInBytes / bytesPerSlab) + 1;
|
|
}
|
|
|
|
//
|
|
// If the caller asked for no slabs then return STATUS_INVALID_PARAMETER.
|
|
//
|
|
if (requestedSlabs == 0)
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Invalid number (%I64u) of slabs requested\n",
|
|
DeviceObject,
|
|
requestedSlabs));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Cap requested slabs at MAXULONG since SlabAllocationBitMapBitCount
|
|
// is a 4-byte field. We may return less data than requested, but the
|
|
// caller can simply re-query for the omitted portion(s).
|
|
//
|
|
requestedSlabs = min(requestedSlabs, MAXULONG);
|
|
|
|
//
|
|
// Calculate the required size of the output buffer based upon the desired
|
|
// version of the output structure.
|
|
// In the worst case, Get LBA Status returns a descriptor for each slab
|
|
// requested, thus the required output buffer length is equal to:
|
|
// 1. The size of DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT; plus
|
|
// 2. The size of DEVICE_DATA_SET_LB_PROVISIONING_STATE(_V2); plus
|
|
// 3. The size of a ULONG array large enough to hold a bit for each slab requested.
|
|
// (The first element is already allocated in DEVICE_DATA_SET_LB_PROVISIONING_STATE(_V2).)
|
|
//
|
|
#if (NTDDI_VERSION >= NTDDI_WINBLUE)
|
|
if (OutputVersion == DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V2) {
|
|
|
|
requiredOutputLength = (ULONG)(sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT)
|
|
+ sizeof(DEVICE_DATA_SET_LB_PROVISIONING_STATE_V2)
|
|
+ (((requestedSlabs - 1) / bitMapGranularityInBits))
|
|
* FIELD_SIZE(DEVICE_DATA_SET_LB_PROVISIONING_STATE_V2, SlabAllocationBitMap[0]));
|
|
|
|
} else
|
|
#else
|
|
UNREFERENCED_PARAMETER(OutputVersion);
|
|
#endif
|
|
{
|
|
|
|
requiredOutputLength = (ULONG)(sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT)
|
|
+ sizeof(DEVICE_DATA_SET_LB_PROVISIONING_STATE)
|
|
+ (((requestedSlabs - 1) / bitMapGranularityInBits))
|
|
* FIELD_SIZE(DEVICE_DATA_SET_LB_PROVISIONING_STATE, SlabAllocationBitMap[0]));
|
|
}
|
|
|
|
//
|
|
// The output buffer is not big enough to hold the requested data.
|
|
// Inform the caller of the correct buffer size.
|
|
//
|
|
if (outputLength < requiredOutputLength)
|
|
{
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Given output buffer is %u bytes, needs to be %u bytes\n",
|
|
DeviceObject,
|
|
outputLength,
|
|
requiredOutputLength));
|
|
|
|
//
|
|
// If the output buffer is big enough, write the required buffer
|
|
// length to the first ULONG bytes of the output buffer.
|
|
//
|
|
if (outputLength >= sizeof(ULONG))
|
|
{
|
|
*((PULONG)DsmOutput) = requiredOutputLength;
|
|
}
|
|
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Calculate the maximum number of slabs that could be returned by a single
|
|
// Get LBA Status command. The max buffer size could either be capped by
|
|
// the Parameter Data Length field or the Max Transfer Length of the
|
|
// adapter.
|
|
// The number of slabs we actually ask for in a single command is the
|
|
// smaller of the number of slabs requested by the user or the max number
|
|
// of slabs we can theoretically ask for in a single command.
|
|
//
|
|
maxBufferSize = MIN(MAXULONG, fdoExtension->PrivateFdoData->HwMaxXferLen);
|
|
maxSlabs = (maxBufferSize - sizeof(LBA_STATUS_LIST_HEADER)) / sizeof(LBA_STATUS_DESCRIPTOR);
|
|
slabsPerCommand = min(requestedSlabs, maxSlabs);
|
|
|
|
//
|
|
// Allocate the buffer that will contain the returned LBA Status Descriptors.
|
|
// Assume that in the worst case every other slab has a different mapping
|
|
// status. That means that there may be a descriptor for every slab requested.
|
|
//
|
|
lbaStatusSize = (ULONG)(sizeof(LBA_STATUS_LIST_HEADER) + (slabsPerCommand * sizeof(LBA_STATUS_DESCRIPTOR)));
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
lbaStatusSize = ALIGN_UP_BY(lbaStatusSize,KeGetRecommendedSharedDataAlignment());
|
|
lbaStatusListHeader = (PLBA_STATUS_LIST_HEADER)ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned, lbaStatusSize, CLASS_TAG_LB_PROVISIONING);
|
|
#else
|
|
lbaStatusListHeader = (PLBA_STATUS_LIST_HEADER)ExAllocatePoolWithTag(NonPagedPoolNx, lbaStatusSize, CLASS_TAG_LB_PROVISIONING);
|
|
#endif
|
|
|
|
if (lbaStatusListHeader == NULL)
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Failed to allocate %u bytes for descriptors\n",
|
|
DeviceObject,
|
|
lbaStatusSize));
|
|
|
|
NT_ASSERT(lbaStatusListHeader != NULL);
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Exit;
|
|
}
|
|
|
|
//
|
|
// Set default values for the output buffer.
|
|
// If we process at least one slab from the device we will update the
|
|
// offset and lengths accordingly.
|
|
//
|
|
DsmOutput->Action = DeviceDsmAction_Allocation;
|
|
DsmOutput->Size = sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT);
|
|
DsmOutput->OutputBlockOffset = 0;
|
|
DsmOutput->OutputBlockLength = 0;
|
|
*DsmOutputLength = DsmOutput->Size;
|
|
|
|
//
|
|
// The returned DEVICE_DATA_SET_LB_PROVISIONING_STATE is at the end of the
|
|
// DSM output structure. Zero it out before we start to fill it in.
|
|
//
|
|
lbpState = Add2Ptr(DsmOutput, sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT));
|
|
RtlZeroMemory(lbpState, requiredOutputLength - sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT));
|
|
|
|
do {
|
|
//
|
|
// Send down GetLBAStatus for the current range.
|
|
//
|
|
status = GetLBAStatus(fdoExtension,
|
|
Srb,
|
|
startingLBA,
|
|
lbaStatusListHeader,
|
|
lbaStatusSize,
|
|
ConsolidateableBlocksOnly);
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
ULONG descrIndex = 0;
|
|
ULONG descrSize = 0;
|
|
ULONG descrSizeOverhead;
|
|
ULONG descrCount = 0;
|
|
ULONGLONG expectedStartingLBA;
|
|
BOOLEAN processCurrentDescriptor = TRUE;
|
|
ULONG commandProcessedSlabs = 0; // Number of slabs processed for this command.
|
|
|
|
descrSizeOverhead = FIELD_OFFSET(LBA_STATUS_LIST_HEADER, Descriptors[0]) -
|
|
RTL_SIZEOF_THROUGH_FIELD(LBA_STATUS_LIST_HEADER, ParameterLength);
|
|
REVERSE_BYTES(&descrSize, &(lbaStatusListHeader->ParameterLength));
|
|
|
|
//
|
|
// If the returned Parameter Data Length field describes more
|
|
// descriptors than we allocated space for then make sure we don't
|
|
// try to process more descriptors than are actually present.
|
|
//
|
|
if (descrSize > (lbaStatusSize - RTL_SIZEOF_THROUGH_FIELD(LBA_STATUS_LIST_HEADER, ParameterLength))) {
|
|
descrSize = (lbaStatusSize - RTL_SIZEOF_THROUGH_FIELD(LBA_STATUS_LIST_HEADER, ParameterLength));
|
|
}
|
|
|
|
if (descrSize >= descrSizeOverhead) {
|
|
descrSize -= descrSizeOverhead;
|
|
descrCount = descrSize / sizeof(LBA_STATUS_DESCRIPTOR);
|
|
|
|
//
|
|
// Make sure at least one descriptor was returned.
|
|
//
|
|
if (descrCount > 0) {
|
|
//
|
|
// We expect the first starting LBA returned by the device to be the
|
|
// same starting LBA we specified in the command.
|
|
//
|
|
expectedStartingLBA = startingLBA;
|
|
|
|
//
|
|
// Translate the returned LBA status descriptors into a bitmap where each bit represents
|
|
// a slab. The slab size is represented by the Optimal Unmap Granularity.
|
|
// 1 = The slab is mapped.
|
|
// 0 = The slab is unmapped (deallocated or anchored).
|
|
//
|
|
for (descrIndex = 0; descrIndex < descrCount && totalProcessedSlabs < requestedSlabs && !doneProcessing; descrIndex++)
|
|
{
|
|
PLBA_STATUS_DESCRIPTOR lbaStatusDescr = &(lbaStatusListHeader->Descriptors[descrIndex]);
|
|
ULONGLONG returnedStartingLBA;
|
|
ULONG mapped = (lbaStatusDescr->ProvisioningStatus != LBA_STATUS_MAPPED) ? 0x0 : 0x1;
|
|
ULONG lbaCount = 0;
|
|
|
|
REVERSE_BYTES_QUAD(&returnedStartingLBA, &(lbaStatusDescr->StartingLBA));
|
|
REVERSE_BYTES(&lbaCount, &(lbaStatusDescr->LogicalBlockCount));
|
|
|
|
if (returnedStartingLBA != expectedStartingLBA)
|
|
{
|
|
//
|
|
// We expect the descriptors will express a contiguous range of LBAs.
|
|
// If the starting LBA is not contiguous with the LBA range from the
|
|
// previous descriptor then we should not process any more descriptors,
|
|
// including the current one.
|
|
//
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Device returned starting LBA = %I64x when %I64x was expected.\n",
|
|
DeviceObject,
|
|
returnedStartingLBA,
|
|
startingLBA));
|
|
|
|
doneProcessing = TRUE;
|
|
processCurrentDescriptor = FALSE;
|
|
*BlockLimitsDataMayHaveChanged = TRUE;
|
|
}
|
|
else if (lbaCount > 0 && lbaCount % blocksPerSlab != 0)
|
|
{
|
|
//
|
|
// If the device returned an LBA count with a partial slab, round
|
|
// the LBA count up to the nearest slab and set a flag to stop
|
|
// processing further descriptors. This is mainly to handle the
|
|
// case where disk size may not be slab-aligned and thus the last
|
|
// "slab" is actually a partial slab.
|
|
//
|
|
TracePrint((TRACE_LEVEL_WARNING,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Device returned an LBA count (%u) that is not a multiple of the slab size (%u)\n",
|
|
DeviceObject,
|
|
lbaCount,
|
|
blocksPerSlab));
|
|
|
|
lbaCount = ((lbaCount / blocksPerSlab) + 1) * blocksPerSlab;
|
|
|
|
doneProcessing = TRUE;
|
|
processCurrentDescriptor = TRUE;
|
|
}
|
|
else if (lbaCount == 0)
|
|
{
|
|
//
|
|
// If the LBA count is 0, just skip this descriptor.
|
|
//
|
|
TracePrint((TRACE_LEVEL_WARNING,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Device returned a zero LBA count\n",
|
|
DeviceObject));
|
|
|
|
processCurrentDescriptor = FALSE;
|
|
}
|
|
|
|
//
|
|
// Generate bits for the slabs described in the current descriptor.
|
|
// It's possible the device may have returned more slabs than requested
|
|
// so we make sure to stop once we've processed all we need.
|
|
//
|
|
if (processCurrentDescriptor)
|
|
{
|
|
ULONG descrSlabs = lbaCount / blocksPerSlab; // Number of slabs in this descriptor.
|
|
|
|
for(; 0 < descrSlabs && totalProcessedSlabs < requestedSlabs; descrSlabs--, commandProcessedSlabs++, totalProcessedSlabs++)
|
|
{
|
|
ULONG bitMapIndex = totalProcessedSlabs / bitMapGranularityInBits;
|
|
ULONG bitPos = totalProcessedSlabs % bitMapGranularityInBits;
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINBLUE)
|
|
if (OutputVersion == DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V2) {
|
|
((PDEVICE_DATA_SET_LB_PROVISIONING_STATE_V2)lbpState)->SlabAllocationBitMap[bitMapIndex] |= (mapped << bitPos);
|
|
} else
|
|
#endif
|
|
{
|
|
lbpState->SlabAllocationBitMap[bitMapIndex] |= (mapped << bitPos);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Calculate the next expected starting LBA.
|
|
//
|
|
expectedStartingLBA = returnedStartingLBA + lbaCount;
|
|
}
|
|
|
|
if (commandProcessedSlabs > 0) {
|
|
|
|
//
|
|
// Calculate the starting LBA we'll use for the next command.
|
|
//
|
|
startingLBA += ((ULONGLONG)commandProcessedSlabs * (ULONGLONG)blocksPerSlab);
|
|
|
|
} else {
|
|
//
|
|
// This should never happen, but we should handle it gracefully anyway.
|
|
//
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): The slab allocation bitmap has zero length.\n",
|
|
DeviceObject));
|
|
|
|
NT_ASSERT(commandProcessedSlabs != 0);
|
|
doneProcessing = TRUE;
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
} else {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Device returned no LBA Status Descriptors.\n",
|
|
DeviceObject));
|
|
|
|
doneProcessing = TRUE;
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
} else {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): not enough bytes returned\n",
|
|
DeviceObject));
|
|
|
|
doneProcessing = TRUE;
|
|
status = STATUS_DEVICE_DATA_ERROR;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Loop until we encounter some error or we've processed all the requested slabs.
|
|
//
|
|
} while (NT_SUCCESS(status) &&
|
|
!doneProcessing &&
|
|
(totalProcessedSlabs < requestedSlabs));
|
|
|
|
//
|
|
// At least one slab was returned by the device and processed, which we
|
|
// consider success. It's up to the caller to detect truncation.
|
|
// Update the output buffer sizes, offsets, etc. accordingly.
|
|
//
|
|
if (totalProcessedSlabs > 0) {
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINBLUE)
|
|
if (OutputVersion == DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V2) {
|
|
PDEVICE_DATA_SET_LB_PROVISIONING_STATE_V2 lbpStateV2 = (PDEVICE_DATA_SET_LB_PROVISIONING_STATE_V2)lbpState;
|
|
|
|
lbpStateV2->SlabSizeInBytes = bytesPerSlab;
|
|
lbpStateV2->SlabOffsetDeltaInBytes = startingOffsetDelta;
|
|
lbpStateV2->SlabAllocationBitMapBitCount = totalProcessedSlabs;
|
|
lbpStateV2->SlabAllocationBitMapLength = ((totalProcessedSlabs - 1) / (ULONGLONG)bitMapGranularityInBits) + 1;
|
|
lbpStateV2->Version = DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V2;
|
|
|
|
//
|
|
// Note that there is already one element of the bitmap array allocated
|
|
// in the DEVICE_DATA_SET_LB_PROVISIONING_STATE_V2 structure itself, which
|
|
// is why we subtract 1 from SlabAllocationBitMapLength.
|
|
//
|
|
lbpStateV2->Size = sizeof(DEVICE_DATA_SET_LB_PROVISIONING_STATE_V2)
|
|
+ ((lbpStateV2->SlabAllocationBitMapLength - 1) * sizeof(lbpStateV2->SlabAllocationBitMap[0]));
|
|
|
|
} else
|
|
#endif
|
|
{
|
|
|
|
lbpState->SlabSizeInBytes = bytesPerSlab;
|
|
lbpState->SlabOffsetDeltaInBytes = (ULONG)startingOffsetDelta;
|
|
lbpState->SlabAllocationBitMapBitCount = totalProcessedSlabs;
|
|
lbpState->SlabAllocationBitMapLength = ((totalProcessedSlabs - 1) / bitMapGranularityInBits) + 1;
|
|
lbpState->Version = DEVICE_DATA_SET_LB_PROVISIONING_STATE_VERSION_V1;
|
|
|
|
//
|
|
// Note that there is already one element of the bitmap array allocated
|
|
// in the DEVICE_DATA_SET_LB_PROVISIONING_STATE structure itself, which
|
|
// is why we subtract 1 from SlabAllocationBitMapLength.
|
|
//
|
|
lbpState->Size = sizeof(DEVICE_DATA_SET_LB_PROVISIONING_STATE)
|
|
+ ((lbpState->SlabAllocationBitMapLength - 1) * sizeof(lbpState->SlabAllocationBitMap[0]));
|
|
}
|
|
|
|
DsmOutput->OutputBlockLength = lbpState->Size; // Size is at the same offset in all versions of the structure.
|
|
DsmOutput->OutputBlockOffset = sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES_OUTPUT);
|
|
*DsmOutputLength = DsmOutput->Size + DsmOutput->OutputBlockLength;
|
|
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceGetLBAStatusWorker (%p): Processed a total of %u slabs\n",
|
|
DeviceObject,
|
|
totalProcessedSlabs));
|
|
Exit:
|
|
|
|
FREE_POOL(lbaStatusListHeader);
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClassGetLBProvisioningLogPage(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PSCSI_REQUEST_BLOCK Srb,
|
|
_In_ ULONG LogPageSize,
|
|
_Inout_ PLOG_PAGE_LOGICAL_BLOCK_PROVISIONING LogPage
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function sends a LOG SENSE command to the given device and returns the
|
|
Logical Block Provisioning Log Page, if available.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO of the device to which the Log Sense command will be sent.
|
|
Srb: This should be allocated before it is passed in, but it does not have
|
|
to be initialized. This function will initialize it.
|
|
LogPageSize: The size of the LogPage buffer in bytes.
|
|
LogPage: A pointer to an already allocated output buffer that may contain
|
|
the LBP log page when this function returns.
|
|
|
|
Return Value:
|
|
|
|
STATUS_INVALID_PARAMETER: May be returned if the LogPage buffer is NULL or
|
|
not large enough.
|
|
STATUS_SUCCESS: The log page was obtained and placed in the LogPage buffer.
|
|
|
|
This function may return other NTSTATUS codes from internal function calls.
|
|
--*/
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
PCDB cdb = NULL;
|
|
|
|
//
|
|
// Make sure the caller passed in an adequate output buffer. The Allocation
|
|
// Length field in the Log Sense command is only 2 bytes so we need to also
|
|
// make sure that the given log page size isn't larger than MAXUSHORT.
|
|
//
|
|
if (LogPage == NULL ||
|
|
LogPageSize < sizeof(LOG_PAGE_LOGICAL_BLOCK_PROVISIONING) ||
|
|
LogPageSize > MAXUSHORT)
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassGetLBProvisioningLogPage: DO (%p), Invalid parameter, LogPage = %p, LogPageSize = %u.\n",
|
|
DeviceObject,
|
|
LogPage,
|
|
LogPageSize));
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Initialize the SRB.
|
|
//
|
|
if (fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK) {
|
|
status = InitializeStorageRequestBlock((PSTORAGE_REQUEST_BLOCK)Srb,
|
|
STORAGE_ADDRESS_TYPE_BTL8,
|
|
CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE,
|
|
1,
|
|
SrbExDataTypeScsiCdb16);
|
|
if (NT_SUCCESS(status)) {
|
|
((PSTORAGE_REQUEST_BLOCK)Srb)->SrbFunction = SRB_FUNCTION_EXECUTE_SCSI;
|
|
} else {
|
|
//
|
|
// Should not occur.
|
|
//
|
|
NT_ASSERT(FALSE);
|
|
}
|
|
} else {
|
|
RtlZeroMemory(Srb, sizeof(SCSI_REQUEST_BLOCK));
|
|
Srb->Length = sizeof(SCSI_REQUEST_BLOCK);
|
|
Srb->Function = SRB_FUNCTION_EXECUTE_SCSI;
|
|
}
|
|
|
|
//
|
|
// Build and send down the Log Sense command.
|
|
//
|
|
SrbSetTimeOutValue(Srb, fdoExtension->TimeOutValue);
|
|
SrbSetRequestTag(Srb, SP_UNTAGGED);
|
|
SrbSetRequestAttribute(Srb, SRB_SIMPLE_TAG_REQUEST);
|
|
SrbAssignSrbFlags(Srb, fdoExtension->SrbFlags);
|
|
SrbSetCdbLength(Srb, sizeof(cdb->LOGSENSE));
|
|
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->LOGSENSE.OperationCode = SCSIOP_LOG_SENSE;
|
|
cdb->LOGSENSE.PageCode = LOG_PAGE_CODE_LOGICAL_BLOCK_PROVISIONING;
|
|
cdb->LOGSENSE.PCBit = 0;
|
|
cdb->LOGSENSE.ParameterPointer[0] = 0;
|
|
cdb->LOGSENSE.ParameterPointer[1] = 0;
|
|
REVERSE_BYTES_SHORT(&(cdb->LOGSENSE.AllocationLength), &LogPageSize);
|
|
|
|
status = ClassSendSrbSynchronous(fdoExtension->DeviceObject,
|
|
Srb,
|
|
LogPage,
|
|
LogPageSize,
|
|
FALSE);
|
|
|
|
//
|
|
// Handle the case where we get back STATUS_DATA_OVERRUN b/c the input
|
|
// buffer was larger than necessary.
|
|
//
|
|
if (status == STATUS_DATA_OVERRUN &&
|
|
SrbGetDataTransferLength(Srb) < LogPageSize)
|
|
{
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
//
|
|
// Log the command.
|
|
//
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClassGetLBProvisioningLogPage: DO (%p), LogSense command issued for LBP log page. NT Status: %!STATUS!.\n",
|
|
DeviceObject,
|
|
status
|
|
));
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS ClassInterpretLBProvisioningLogPage(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ ULONG LogPageSize,
|
|
_In_ PLOG_PAGE_LOGICAL_BLOCK_PROVISIONING LogPage,
|
|
_In_ ULONG ResourcesSize,
|
|
_Out_ PSTORAGE_LB_PROVISIONING_MAP_RESOURCES Resources
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function takes a Logical Block Provisioning log page (returned by
|
|
ClassGetLBProvisioningLogPage(), for example), interprets its contents,
|
|
and returns the interpreted data in a STORAGE_LB_PROVISIONING_MAP_RESOURCES
|
|
structure.
|
|
|
|
None, some, or all of the data in the output buffer may be valid. The
|
|
caller must look at the individual "Valid" fields to see which fields have
|
|
valid data.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO of the device from which the log page was obtained.
|
|
LogPageSize: The size of the LogPage buffer in bytes.
|
|
LogPage: A pointer to a valid LBP log page structure.
|
|
ResourcesSize: The size of the Resources buffer in bytes.
|
|
Resources: A pointer to an already allocated output buffer that may contain
|
|
the interpreted log page data when this function returns.
|
|
|
|
Return Value:
|
|
|
|
STATUS_NOT_SUPPORTED: May be returned if the threshold exponent from the
|
|
0xB2 page is invalid.
|
|
STATUS_INVALID_PARAMETER: May be returned if either the LogPage or Resources
|
|
buffers are NULL or too small.
|
|
STATUS_SUCCESS: The log page data was interpreted and the Resources output
|
|
buffer has data in it.
|
|
|
|
This function may return other NTSTATUS codes from internal function calls.
|
|
--*/
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
USHORT pageLength;
|
|
PLOG_PARAMETER_HEADER parameter;
|
|
PVOID endOfPage;
|
|
USHORT parameterCode;
|
|
ULONG resourceCount;
|
|
UCHAR thresholdExponent = fdoExtension->FunctionSupportInfo->LBProvisioningData.ThresholdExponent;
|
|
ULONGLONG thresholdSetSize;
|
|
|
|
//
|
|
// SBC-3 states that the threshold exponent (from the 0xB2 VPD page), must
|
|
// be non-zero and less than or equal to 32.
|
|
//
|
|
if (thresholdExponent < 0 || thresholdExponent > 32)
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassInterpretLBProvisioningLogPage: DO (%p), Threshold Exponent (%u) is invalid.\n",
|
|
DeviceObject,
|
|
thresholdExponent));
|
|
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
if (Resources == NULL ||
|
|
ResourcesSize < sizeof(STORAGE_LB_PROVISIONING_MAP_RESOURCES) ||
|
|
LogPage == NULL ||
|
|
LogPageSize < sizeof(LOG_PAGE_LOGICAL_BLOCK_PROVISIONING))
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassInterpretLBProvisioningLogPage: DO (%p), Invalid parameter, Resources = %p, ResourcesSize = %u, LogPage = %p, LogPageSize = %u.\n",
|
|
DeviceObject,
|
|
Resources,
|
|
ResourcesSize,
|
|
LogPage,
|
|
LogPageSize));
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Calculate the threshold set size (in LBAs).
|
|
//
|
|
thresholdSetSize = 1ULL << thresholdExponent;
|
|
|
|
REVERSE_BYTES_SHORT(&pageLength, &(LogPage->PageLength));
|
|
|
|
//
|
|
// Initialize the output buffer.
|
|
//
|
|
RtlZeroMemory(Resources, sizeof(STORAGE_LB_PROVISIONING_MAP_RESOURCES));
|
|
Resources->Size = sizeof(STORAGE_LB_PROVISIONING_MAP_RESOURCES);
|
|
Resources->Version = sizeof(STORAGE_LB_PROVISIONING_MAP_RESOURCES);
|
|
|
|
//
|
|
// Make sure we don't walk off the end of the log page buffer
|
|
// if pageLength is somehow longer than the buffer itself.
|
|
//
|
|
pageLength = (USHORT)min(pageLength, (LogPageSize - FIELD_OFFSET(LOG_PAGE_LOGICAL_BLOCK_PROVISIONING, Parameters)));
|
|
|
|
parameter = (PLOG_PARAMETER_HEADER)((PUCHAR)LogPage + FIELD_OFFSET(LOG_PAGE_LOGICAL_BLOCK_PROVISIONING, Parameters));
|
|
endOfPage = (PVOID)((PUCHAR)parameter + pageLength);
|
|
|
|
//
|
|
// Walk the parameters.
|
|
//
|
|
while ((PVOID)parameter < endOfPage)
|
|
{
|
|
if (parameter->ParameterLength > 0)
|
|
{
|
|
REVERSE_BYTES_SHORT(¶meterCode, &(parameter->ParameterCode));
|
|
switch(parameterCode)
|
|
{
|
|
case LOG_PAGE_LBP_PARAMETER_CODE_AVAILABLE:
|
|
{
|
|
REVERSE_BYTES(&resourceCount, &(((PLOG_PARAMETER_THRESHOLD_RESOURCE_COUNT)parameter)->ResourceCount));
|
|
Resources->AvailableMappingResources = (ULONGLONG)resourceCount * thresholdSetSize * (ULONGLONG)fdoExtension->DiskGeometry.BytesPerSector;
|
|
Resources->AvailableMappingResourcesValid = TRUE;
|
|
|
|
//
|
|
// Devices that implement SBC-3 revisions older than r27 will not specify
|
|
// an LBP log page parameter that has fields beyond ResourceCount.
|
|
//
|
|
if (parameter->ParameterLength > FIELD_OFFSET(LOG_PARAMETER_THRESHOLD_RESOURCE_COUNT, ResourceCount[3])) {
|
|
Resources->AvailableMappingResourcesScope = ((PLOG_PARAMETER_THRESHOLD_RESOURCE_COUNT)parameter)->Scope;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case LOG_PAGE_LBP_PARAMETER_CODE_USED:
|
|
{
|
|
REVERSE_BYTES(&resourceCount, &(((PLOG_PARAMETER_THRESHOLD_RESOURCE_COUNT)parameter)->ResourceCount));
|
|
Resources->UsedMappingResources = (ULONGLONG)resourceCount * thresholdSetSize * (ULONGLONG)fdoExtension->DiskGeometry.BytesPerSector;
|
|
Resources->UsedMappingResourcesValid = TRUE;
|
|
|
|
//
|
|
// Devices that implement SBC-3 revisions older than r27 will not specify
|
|
// an LBP log page parameter that has fields beyond ResourceCount.
|
|
//
|
|
if (parameter->ParameterLength > FIELD_OFFSET(LOG_PARAMETER_THRESHOLD_RESOURCE_COUNT, ResourceCount[3])) {
|
|
Resources->UsedMappingResourcesScope = ((PLOG_PARAMETER_THRESHOLD_RESOURCE_COUNT)parameter)->Scope;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Move to the next parameter.
|
|
//
|
|
parameter = (PLOG_PARAMETER_HEADER)((PUCHAR)parameter + sizeof(LOG_PARAMETER_HEADER) + parameter->ParameterLength);
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS ClassGetLBProvisioningResources(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb,
|
|
_In_ ULONG ResourcesSize,
|
|
_Inout_ PSTORAGE_LB_PROVISIONING_MAP_RESOURCES Resources
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function obtains the Logical Block Provisioning log page, interprets
|
|
its contents, and returns the interpreted data in a
|
|
STORAGE_LB_PROVISIONING_MAP_RESOURCES structure.
|
|
|
|
None, some, or all of the data in the output buffer may be valid. The
|
|
caller must look at the individual "Valid" fields to see which fields have
|
|
valid data.
|
|
|
|
Arguments:
|
|
DeviceObject: The target FDO.
|
|
Srb: This should be allocated before it is passed in, but it does not have
|
|
to be initialized.
|
|
ResourcesSize: The size of the Resources buffer in bytes.
|
|
Resources: A pointer to an already allocated output buffer that may contain
|
|
the interpreted log page data when this function returns.
|
|
|
|
Return Value:
|
|
|
|
STATUS_NOT_SUPPORTED: May be returned if the device does not have LBP enabled.
|
|
STATUS_INVALID_PARAMETER: May be returned if either the Resources buffer is
|
|
NULL or too small.
|
|
STATUS_INSUFFICIENT_RESOURCES: May be returned if a log page buffer could not
|
|
be allocated.
|
|
STATUS_SUCCESS: The log page data was obtained and the Resources output
|
|
buffer has data in it.
|
|
|
|
This function may return other NTSTATUS codes from internal function calls.
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
ULONG logPageSize;
|
|
PLOG_PAGE_LOGICAL_BLOCK_PROVISIONING logPage = NULL;
|
|
|
|
//
|
|
// This functionality is only supported for devices that support logical
|
|
// block provisioning.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->ValidInquiryPages.LBProvisioning == FALSE)
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassGetLBProvisioningResources: DO (%p), Device does not support logical block provisioning.\n",
|
|
DeviceObject));
|
|
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
//
|
|
// Validate the output buffer.
|
|
//
|
|
if (Resources == NULL ||
|
|
ResourcesSize < sizeof(STORAGE_LB_PROVISIONING_MAP_RESOURCES))
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassGetLBProvisioningResources: DO (%p), Invalid parameter, Resources = %p, ResourcesSize = %u.\n",
|
|
DeviceObject,
|
|
Resources,
|
|
ResourcesSize));
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Allocate a buffer for the log page. Currently the log page contains:
|
|
// 1. Log page header
|
|
// 2. Log page parameter for used resources
|
|
// 3. Log page parameter for available resources
|
|
//
|
|
logPageSize = sizeof(LOG_PAGE_LOGICAL_BLOCK_PROVISIONING) + (2 * sizeof(LOG_PARAMETER_THRESHOLD_RESOURCE_COUNT));
|
|
|
|
#if defined(_ARM_) || defined(_ARM64_)
|
|
//
|
|
// ARM has specific alignment requirements, although this will not have a functional impact on x86 or amd64
|
|
// based platforms. We are taking the conservative approach here.
|
|
//
|
|
logPageSize = ALIGN_UP_BY(logPageSize, KeGetRecommendedSharedDataAlignment());
|
|
logPage = (PLOG_PAGE_LOGICAL_BLOCK_PROVISIONING)ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned, logPageSize, CLASS_TAG_LB_PROVISIONING);
|
|
#else
|
|
logPage = (PLOG_PAGE_LOGICAL_BLOCK_PROVISIONING)ExAllocatePoolWithTag(NonPagedPoolNx, logPageSize, CLASS_TAG_LB_PROVISIONING);
|
|
#endif
|
|
if (logPage != NULL)
|
|
{
|
|
//
|
|
// Get the LBP log page from the device.
|
|
//
|
|
status = ClassGetLBProvisioningLogPage(DeviceObject,
|
|
Srb,
|
|
logPageSize,
|
|
logPage);
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
//
|
|
// Interpret the log page and fill in the output buffer.
|
|
//
|
|
status = ClassInterpretLBProvisioningLogPage(DeviceObject,
|
|
logPageSize,
|
|
logPage,
|
|
ResourcesSize,
|
|
Resources);
|
|
}
|
|
|
|
ExFreePool(logPage);
|
|
}
|
|
else
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassGetLBProvisioningResources: DO (%p), Failed to allocate memory for LBP log page.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
ClassDeviceGetLBProvisioningResources(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function returns the LBP resource counts in a
|
|
STORAGE_LB_PROVISIONING_MAP_RESOURCES structure in the IRP.
|
|
|
|
None, some, or all of the data in the output buffer may be valid. The
|
|
caller must look at the individual "Valid" fields to see which fields have
|
|
valid data.
|
|
|
|
Arguments:
|
|
DeviceObject: The target FDO.
|
|
Irp: The IRP which will contain the output buffer upon completion.
|
|
Srb: This should be allocated before it is passed in, but it does not have
|
|
to be initialized.
|
|
|
|
Return Value:
|
|
|
|
Some NTSTATUS code.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_LB_PROVISIONING_MAP_RESOURCES mapResources = (PSTORAGE_LB_PROVISIONING_MAP_RESOURCES)Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
status = ClassGetLBProvisioningResources(DeviceObject,
|
|
Srb,
|
|
irpStack->Parameters.DeviceIoControl.OutputBufferLength,
|
|
mapResources);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
Irp->IoStatus.Information = mapResources->Size;
|
|
} else {
|
|
Irp->IoStatus.Information = 0;
|
|
}
|
|
|
|
Irp->IoStatus.Status = status;
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
_Function_class_(IO_WORKITEM_ROUTINE)
|
|
_IRQL_requires_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassLogThresholdEvent(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_opt_ PVOID Context
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function logs a logical block provisioning soft threshold event to the
|
|
system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the soft
|
|
threshold.
|
|
Context: A pointer to the IO_WORKITEM in which this function is running.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
PIO_WORKITEM workItem = (PIO_WORKITEM)Context;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PSCSI_REQUEST_BLOCK srb = NULL;
|
|
STORAGE_LB_PROVISIONING_MAP_RESOURCES resources = {0};
|
|
ULONG resourcesSize = sizeof(STORAGE_LB_PROVISIONING_MAP_RESOURCES);
|
|
PIO_ERROR_LOG_PACKET errorLogEntry = NULL;
|
|
ULONG logEntrySize = sizeof(IO_ERROR_LOG_PACKET);
|
|
PWCHAR stringIndex = NULL;
|
|
LONG stringSize = 0;
|
|
ULONG srbSize;
|
|
|
|
//
|
|
// Allocate an SRB for getting the LBP log page.
|
|
//
|
|
if ((fdoExtension->AdapterDescriptor != NULL) &&
|
|
(fdoExtension->AdapterDescriptor->SrbType == SRB_TYPE_STORAGE_REQUEST_BLOCK)) {
|
|
srbSize = CLASS_SRBEX_SCSI_CDB16_BUFFER_SIZE;
|
|
} else {
|
|
srbSize = sizeof(SCSI_REQUEST_BLOCK);
|
|
}
|
|
|
|
srb = ExAllocatePoolWithTag(NonPagedPoolNx,
|
|
srbSize,
|
|
'ACcS');
|
|
if (srb != NULL) {
|
|
|
|
//
|
|
// Try to get the LBP resources from the device so we can report them in
|
|
// the system event log.
|
|
//
|
|
ClassGetLBProvisioningResources(DeviceObject,
|
|
srb,
|
|
resourcesSize,
|
|
&resources);
|
|
|
|
//
|
|
// We need to allocate enough space for 3 insertion strings:
|
|
// The first is a ULONG representing the disk number in decimal, which means
|
|
// a max of 10 digits, plus one for the NULL character.
|
|
// The second and third are ULONGLONGs representing the used and available
|
|
// bytes, which means a max of 20 digits, plus one for the NULL character.
|
|
// Make sure we do not exceed the max error log size or the max size of a
|
|
// UCHAR since the size gets truncated to a UCHAR when we pass it to
|
|
// IoAllocateErrorLogEntry().
|
|
//
|
|
logEntrySize = sizeof(IO_ERROR_LOG_PACKET) + (11 * sizeof(WCHAR)) + (2 * (21 * sizeof(WCHAR)));
|
|
logEntrySize = min(logEntrySize, ERROR_LOG_MAXIMUM_SIZE);
|
|
logEntrySize = min(logEntrySize, MAXUCHAR);
|
|
|
|
errorLogEntry = (PIO_ERROR_LOG_PACKET)IoAllocateErrorLogEntry(DeviceObject, (UCHAR)logEntrySize);
|
|
if (errorLogEntry != NULL)
|
|
{
|
|
//
|
|
// There are two event IDs we can use here. Both use the disk number,
|
|
// but one reports the available and used bytes while the other does not.
|
|
// We fall back on the latter if we failed to obtain the available and
|
|
// used byte counts from the LBP log page.
|
|
//
|
|
// The event insertion strings need to be in this order:
|
|
// 1. The disk number. (Both event IDs use this.)
|
|
// 2. Bytes used.
|
|
// 3. Bytes available.
|
|
//
|
|
|
|
RtlZeroMemory(errorLogEntry, logEntrySize);
|
|
errorLogEntry->StringOffset = sizeof(IO_ERROR_LOG_PACKET);
|
|
|
|
stringIndex = (PWCHAR)((ULONG_PTR)errorLogEntry + sizeof(IO_ERROR_LOG_PACKET));
|
|
stringSize = logEntrySize - sizeof(IO_ERROR_LOG_PACKET);
|
|
|
|
//
|
|
// Add the disk number to the insertion strings.
|
|
//
|
|
status = RtlStringCbPrintfW(stringIndex, stringSize, L"%d", fdoExtension->DeviceNumber);
|
|
|
|
if (NT_SUCCESS(status) )
|
|
{
|
|
errorLogEntry->NumberOfStrings++;
|
|
|
|
if (resources.UsedMappingResourcesValid &&
|
|
resources.AvailableMappingResourcesValid)
|
|
{
|
|
//
|
|
// Add the used mapping resources to the insertion strings.
|
|
//
|
|
stringIndex += (wcslen(stringIndex) + 1);
|
|
stringSize -= (LONG)(wcslen(stringIndex) + 1) * sizeof(WCHAR);
|
|
|
|
status = RtlStringCbPrintfW(stringIndex, stringSize, L"%I64u", resources.UsedMappingResources);
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
errorLogEntry->NumberOfStrings++;
|
|
|
|
//
|
|
// Add the available mapping resources to the insertion strings.
|
|
//
|
|
stringIndex += (wcslen(stringIndex) + 1);
|
|
stringSize -= (LONG)(wcslen(stringIndex) + 1) * sizeof(WCHAR);
|
|
|
|
status = RtlStringCbPrintfW(stringIndex, stringSize, L"%I64u", resources.AvailableMappingResources);
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
errorLogEntry->NumberOfStrings++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
TracePrint((TRACE_LEVEL_WARNING,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogThresholdEvent: DO (%p), Used and available mapping resources were unavailable.\n",
|
|
DeviceObject));
|
|
}
|
|
}
|
|
|
|
//
|
|
// If we were able to successfully assemble all 3 insertion strings,
|
|
// then we can use one of the "extended" event IDs. Otherwise, use the basic
|
|
// event ID, which only requires the disk number.
|
|
//
|
|
if (errorLogEntry->NumberOfStrings == 3)
|
|
{
|
|
if (resources.UsedMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_DEDICATED_TO_LUN &&
|
|
resources.AvailableMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_DEDICATED_TO_LUN) {
|
|
|
|
errorLogEntry->ErrorCode = IO_WARNING_SOFT_THRESHOLD_REACHED_EX_LUN_LUN;
|
|
|
|
} else if (resources.UsedMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_DEDICATED_TO_LUN &&
|
|
resources.AvailableMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_NOT_DEDICATED_TO_LUN) {
|
|
|
|
errorLogEntry->ErrorCode = IO_WARNING_SOFT_THRESHOLD_REACHED_EX_LUN_POOL;
|
|
|
|
} else if (resources.UsedMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_NOT_DEDICATED_TO_LUN &&
|
|
resources.AvailableMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_DEDICATED_TO_LUN) {
|
|
|
|
errorLogEntry->ErrorCode = IO_WARNING_SOFT_THRESHOLD_REACHED_EX_POOL_LUN;
|
|
|
|
} else if (resources.UsedMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_NOT_DEDICATED_TO_LUN &&
|
|
resources.AvailableMappingResourcesScope == LOG_PAGE_LBP_RESOURCE_SCOPE_NOT_DEDICATED_TO_LUN) {
|
|
|
|
errorLogEntry->ErrorCode = IO_WARNING_SOFT_THRESHOLD_REACHED_EX_POOL_POOL;
|
|
|
|
} else {
|
|
|
|
errorLogEntry->ErrorCode = IO_WARNING_SOFT_THRESHOLD_REACHED_EX;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errorLogEntry->ErrorCode = IO_WARNING_SOFT_THRESHOLD_REACHED;
|
|
}
|
|
|
|
//
|
|
// Write the error log packet to the system error logging thread.
|
|
// It will be freed automatically.
|
|
//
|
|
IoWriteErrorLogEntry(errorLogEntry);
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogThresholdEvent: DO (%p), Soft threshold notification logged.\n",
|
|
DeviceObject));
|
|
}
|
|
else
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogThresholdEvent: DO (%p), Failed to allocate memory for error log entry.\n",
|
|
DeviceObject));
|
|
}
|
|
} else {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogThresholdEvent: DO (%p), Failed to allocate memory for SRB.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
|
|
//
|
|
// Clear the soft threshold event pending flag so that another can be queued.
|
|
//
|
|
InterlockedExchange((PLONG)&(fdoExtension->FunctionSupportInfo->LBProvisioningData.SoftThresholdEventPending), 0);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, (PIRP)workItem);
|
|
|
|
FREE_POOL(srb);
|
|
|
|
if (workItem != NULL) {
|
|
IoFreeWorkItem(workItem);
|
|
}
|
|
}
|
|
|
|
NTSTATUS
|
|
ClasspLogSystemEventWithDeviceNumber(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ NTSTATUS IoErrorCode
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function is a helper routine to log any system events that require
|
|
the DeviceNumber (e.g. disk number). It is basically a wrapper for the
|
|
IoWriteErrorLogEntry call.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device for which the event needs to be logged.
|
|
IoErrorCode: The IO error code for the event.
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS - if the event was logged
|
|
STATUS_INSUFFICIENT_RESOURCES - otherwise
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
PIO_ERROR_LOG_PACKET errorLogEntry = NULL;
|
|
ULONG logEntrySize = sizeof(IO_ERROR_LOG_PACKET);
|
|
PWCHAR stringIndex = NULL;
|
|
LONG stringSize = 0;
|
|
|
|
//
|
|
// We need to allocate enough space for one insertion string: a ULONG
|
|
// representing the disk number in decimal, which means a max of 10 digits,
|
|
// plus one for the NULL character.
|
|
// Make sure we do not exceed the max error log size or the max size of a
|
|
// UCHAR since the size gets truncated to a UCHAR when we pass it to
|
|
// IoAllocateErrorLogEntry().
|
|
//
|
|
logEntrySize = sizeof(IO_ERROR_LOG_PACKET) + (11 * sizeof(WCHAR));
|
|
logEntrySize = min(logEntrySize, ERROR_LOG_MAXIMUM_SIZE);
|
|
logEntrySize = min(logEntrySize, MAXUCHAR);
|
|
|
|
errorLogEntry = (PIO_ERROR_LOG_PACKET)IoAllocateErrorLogEntry(DeviceObject, (UCHAR)logEntrySize);
|
|
if (errorLogEntry) {
|
|
|
|
RtlZeroMemory(errorLogEntry, logEntrySize);
|
|
errorLogEntry->StringOffset = sizeof(IO_ERROR_LOG_PACKET);
|
|
errorLogEntry->ErrorCode = IoErrorCode;
|
|
|
|
stringIndex = (PWCHAR)((ULONG_PTR)errorLogEntry + sizeof(IO_ERROR_LOG_PACKET));
|
|
stringSize = logEntrySize - sizeof(IO_ERROR_LOG_PACKET);
|
|
|
|
//
|
|
// Add the disk number to the insertion strings.
|
|
//
|
|
status = RtlStringCbPrintfW(stringIndex, stringSize, L"%d", fdoExtension->DeviceNumber);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
errorLogEntry->NumberOfStrings++;
|
|
}
|
|
|
|
//
|
|
// Write the error log packet to the system error logging thread.
|
|
// It will be freed automatically.
|
|
//
|
|
IoWriteErrorLogEntry(errorLogEntry);
|
|
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
_Function_class_(IO_WORKITEM_ROUTINE)
|
|
_IRQL_requires_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassLogResourceExhaustionEvent(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_opt_ PVOID Context
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function logs a logical block provisioning permanent resource exhaustion
|
|
event to the system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the permanent
|
|
resource exhaustion.
|
|
Context: A pointer to the IO_WORKITEM in which this function is running.
|
|
|
|
--*/
|
|
{
|
|
PIO_WORKITEM workItem = (PIO_WORKITEM)Context;
|
|
|
|
if (NT_SUCCESS(ClasspLogSystemEventWithDeviceNumber(DeviceObject, IO_ERROR_DISK_RESOURCES_EXHAUSTED))) {
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogResourceExhaustionEvent: DO (%p), Permanent resource exhaustion logged.\n",
|
|
DeviceObject));
|
|
} else {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogResourceExhaustionEvent: DO (%p), Failed to allocate memory for error log entry.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, (PIRP)workItem);
|
|
|
|
if (workItem != NULL) {
|
|
IoFreeWorkItem(workItem);
|
|
}
|
|
}
|
|
|
|
|
|
VOID ClassQueueThresholdEventWorker(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function queues a delayed work item that will eventually log a
|
|
logical block provisioning soft threshold event to the system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the soft
|
|
threshold.
|
|
|
|
--*/
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)(DeviceObject->DeviceExtension);
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)(DeviceObject->DeviceExtension);
|
|
PIO_WORKITEM workItem = NULL;
|
|
|
|
if (commonExtension->IsFdo &&
|
|
InterlockedCompareExchange((PLONG)&(fdoExtension->FunctionSupportInfo->LBProvisioningData.SoftThresholdEventPending), 1, 0) == 0)
|
|
{
|
|
workItem = IoAllocateWorkItem(DeviceObject);
|
|
|
|
if (workItem)
|
|
{
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueThresholdEventWorker: DO (%p), Queueing soft threshold notification work item.\n",
|
|
DeviceObject));
|
|
|
|
|
|
ClassAcquireRemoveLock(DeviceObject, (PIRP)(workItem));
|
|
|
|
//
|
|
// Queue a work item to write the threshold notification to the
|
|
// system event log.
|
|
//
|
|
IoQueueWorkItem(workItem, ClassLogThresholdEvent, DelayedWorkQueue, workItem);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Clear the soft threshold event pending flag since this is normally
|
|
// done when the work item completes.
|
|
//
|
|
InterlockedExchange((PLONG)&(fdoExtension->FunctionSupportInfo->LBProvisioningData.SoftThresholdEventPending), 0);
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueThresholdEventWorker: DO (%p), Failed to allocate memory for the work item.\n",
|
|
DeviceObject));
|
|
}
|
|
}
|
|
}
|
|
|
|
VOID ClassQueueResourceExhaustionEventWorker(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function queues a delayed work item that will eventually log a
|
|
logical block provisioning permanent resource exhaustion event to the
|
|
system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the resource
|
|
exhaustion.
|
|
|
|
--*/
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)(DeviceObject->DeviceExtension);
|
|
PIO_WORKITEM workItem = NULL;
|
|
|
|
if (commonExtension->IsFdo)
|
|
{
|
|
workItem = IoAllocateWorkItem(DeviceObject);
|
|
|
|
if (workItem)
|
|
{
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueResourceExhaustionEventWorker: DO (%p), Queueing permanent resource exhaustion event work item.\n",
|
|
DeviceObject));
|
|
|
|
ClassAcquireRemoveLock(DeviceObject, (PIRP)(workItem));
|
|
|
|
//
|
|
// Queue a work item to write the threshold notification to the
|
|
// system event log.
|
|
//
|
|
IoQueueWorkItem(workItem, ClassLogResourceExhaustionEvent, DelayedWorkQueue, workItem);
|
|
}
|
|
else
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueResourceExhaustionEventWorker: DO (%p), Failed to allocate memory for the work item.\n",
|
|
DeviceObject));
|
|
}
|
|
}
|
|
}
|
|
|
|
_Function_class_(IO_WORKITEM_ROUTINE)
|
|
_IRQL_requires_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassLogCapacityChangedProcess(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_opt_ PVOID Context
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function logs a capacity changed event to the system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the capacity change.
|
|
Context: A pointer to the IO_WORKITEM in which this function is running.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PIO_WORKITEM workItem = (PIO_WORKITEM)Context;
|
|
|
|
status = ClasspLogSystemEventWithDeviceNumber(DeviceObject, IO_WARNING_DISK_CAPACITY_CHANGED);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogCapacityChangedEvent: DO (%p), Capacity changed logged.\n",
|
|
DeviceObject));
|
|
} else {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogCapacityChangedEvent: DO (%p), Failed to allocate memory for error log entry.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
//
|
|
// Get disk capacity and notify upper layer if capacity is changed.
|
|
//
|
|
status = ClassReadDriveCapacity(DeviceObject);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogCapacityChangedEvent: DO (%p), ClassReadDriveCapacity returned %!STATUS!.\n",
|
|
DeviceObject,
|
|
status));
|
|
}
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, (PIRP)workItem);
|
|
|
|
if (workItem != NULL) {
|
|
IoFreeWorkItem(workItem);
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
ClassQueueCapacityChangedEventWorker(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function queues a delayed work item that will eventually log a
|
|
disk capacity changed event to the system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the capacity change.
|
|
|
|
--*/
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)(DeviceObject->DeviceExtension);
|
|
PIO_WORKITEM workItem = NULL;
|
|
|
|
if (commonExtension->IsFdo)
|
|
{
|
|
workItem = IoAllocateWorkItem(DeviceObject);
|
|
|
|
if (workItem)
|
|
{
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueCapacityChangedEventWorker: DO (%p), Queueing capacity changed event work item.\n",
|
|
DeviceObject));
|
|
|
|
ClassAcquireRemoveLock(DeviceObject, (PIRP)(workItem));
|
|
|
|
//
|
|
// Queue a work item to write the threshold notification to the
|
|
// system event log.
|
|
//
|
|
IoQueueWorkItem(workItem, ClassLogCapacityChangedProcess, DelayedWorkQueue, workItem);
|
|
}
|
|
else
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueCapacityChangedEventWorker: DO (%p), Failed to allocate memory for the work item.\n",
|
|
DeviceObject));
|
|
}
|
|
}
|
|
}
|
|
|
|
_Function_class_(IO_WORKITEM_ROUTINE)
|
|
_IRQL_requires_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClassLogProvisioningTypeChangedEvent(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PVOID Context
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function logs a provisioning type changed event to the system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the provisioning type change.
|
|
Context: A pointer to the IO_WORKITEM in which this function is running.
|
|
|
|
--*/
|
|
{
|
|
PIO_WORKITEM workItem = (PIO_WORKITEM)Context;
|
|
|
|
if (NT_SUCCESS(ClasspLogSystemEventWithDeviceNumber(DeviceObject, IO_WARNING_DISK_PROVISIONING_TYPE_CHANGED))) {
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogProvisioningTypeChangedEvent: DO (%p), LB Provisioning Type changed logged.\n",
|
|
DeviceObject));
|
|
} else {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassLogProvisioningTypeChangedEvent: DO (%p), Failed to allocate memory for error log entry.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, (PIRP)workItem);
|
|
|
|
IoFreeWorkItem(workItem);
|
|
}
|
|
|
|
|
|
VOID
|
|
ClassQueueProvisioningTypeChangedEventWorker(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function queues a delayed work item that will eventually log a
|
|
provisioning type changed event to the system event log.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that reported the provisioning type change.
|
|
|
|
--*/
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)(DeviceObject->DeviceExtension);
|
|
PIO_WORKITEM workItem = NULL;
|
|
|
|
if (commonExtension->IsFdo)
|
|
{
|
|
workItem = IoAllocateWorkItem(DeviceObject);
|
|
|
|
if (workItem)
|
|
{
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueProvisioningTypeChangedEventWorker: DO (%p), Queueing LB provisioning type changed event work item.\n",
|
|
DeviceObject));
|
|
|
|
ClassAcquireRemoveLock(DeviceObject, (PIRP)(workItem));
|
|
|
|
//
|
|
// Queue a work item to write the threshold notification to the
|
|
// system event log.
|
|
//
|
|
IoQueueWorkItem(workItem, ClassLogProvisioningTypeChangedEvent, DelayedWorkQueue, workItem);
|
|
}
|
|
else
|
|
{
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClassQueueProvisioningTypeChangedEventWorker: DO (%p), Failed to allocate memory for the work item.\n",
|
|
DeviceObject));
|
|
}
|
|
}
|
|
}
|
|
|
|
_Function_class_(IO_WORKITEM_ROUTINE)
|
|
_IRQL_requires_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
ClasspLogIOEventWithContext(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_opt_ PVOID Context
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function logs an event to the system event log with dumpdata containing opcode and
|
|
sense data.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that retried the IO.
|
|
Context: A pointer to the OPCODE_SENSE_DATA_IO_LOG_MESSAGE_CONTEXT that has data to be logged as part of the message.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
POPCODE_SENSE_DATA_IO_LOG_MESSAGE_CONTEXT_HEADER ioLogMessageContextHeader = (POPCODE_SENSE_DATA_IO_LOG_MESSAGE_CONTEXT_HEADER)Context;
|
|
PIO_WORKITEM workItem;
|
|
PIO_ERROR_LOG_PACKET errorLogEntry = NULL;
|
|
ULONG logEntrySize;
|
|
PWCHAR stringIndex = NULL;
|
|
LONG stringSize = 0;
|
|
ULONG senseBufferSize;
|
|
ULONG stringsBufferLength = 0;
|
|
ULONG pdoNameLength = 0;
|
|
|
|
NT_ASSERT(ioLogMessageContextHeader != NULL);
|
|
_Analysis_assume_(ioLogMessageContextHeader != NULL);
|
|
|
|
switch (ioLogMessageContextHeader->ErrorCode) {
|
|
|
|
case IO_ERROR_IO_HARDWARE_ERROR:
|
|
case IO_WARNING_IO_OPERATION_RETRIED: {
|
|
|
|
//
|
|
// We need to allocate enough space for 3 insertion strings:
|
|
// 1. A ULONGLONG in Hex representing the LBA which means a max of 16 digits,
|
|
// plus two for "0x" plus one for the NULL character.
|
|
// 2. A ULONG representing the disk number in decimal, which means
|
|
// a max of 10 digits, plus one for the NULL character.
|
|
// 3. The PDO name, so that if the disk number is hidden from the
|
|
// user for some reason, there is still a way to associate the
|
|
// event with the correct device.
|
|
//
|
|
stringsBufferLength = (19 + 11) * sizeof(WCHAR);
|
|
|
|
//
|
|
// Query for the size of the PDO name.
|
|
//
|
|
status = IoGetDeviceProperty(fdoExtension->LowerPdo,
|
|
DevicePropertyPhysicalDeviceObjectName,
|
|
0,
|
|
NULL,
|
|
&pdoNameLength);
|
|
|
|
if (status == STATUS_BUFFER_TOO_SMALL && pdoNameLength > 0) {
|
|
stringsBufferLength += pdoNameLength;
|
|
} else {
|
|
pdoNameLength = 0;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
workItem = ioLogMessageContextHeader->WorkItem;
|
|
|
|
//
|
|
// DumpData[0] which is of ULONG size and will contain opcode|srbstatus|scsistatus.
|
|
// Then we will have sensebuffer, hence
|
|
// DumpDataSize = senseBufferSize + sizeof(ULONG)
|
|
// and DumpDataSize must be multiple of sizeof(ULONG)
|
|
// which means senseBufferSize needs to ULONG aligned
|
|
// Please note we will use original buffersize for padding later
|
|
//
|
|
senseBufferSize = ALIGN_UP_BY(ioLogMessageContextHeader->SenseDataSize, sizeof(ULONG));
|
|
|
|
logEntrySize = FIELD_OFFSET( IO_ERROR_LOG_PACKET, DumpData ) + sizeof(ULONG) + senseBufferSize;
|
|
|
|
//
|
|
// We need to make sure the string offset is WCHAR-aligned (the insertion strings
|
|
// come after the sense buffer in the dump data, if any).
|
|
// But we don't need to do anything special for it,
|
|
// since FIELD_OFFSET( IO_ERROR_LOG_PACKET, DumpData) is currently ULONG aligned
|
|
// and SenseBufferSize is also ULONG aligned. This means buffer that precedes the insertion string is ULONG aligned
|
|
// note stringoffset = FIELD_OFFSET( IO_ERROR_LOG_PACKET, DumpData ) + DumpDataSize
|
|
// This leads us to fact that stringoffset will always be ULONG aligned and effectively WCHAR aligned
|
|
//
|
|
|
|
//
|
|
// We need to allocate enough space for the insertion strings provided in the passed in Context
|
|
// as well as the opcode and the sense data, while making sure we cap at max error log size.
|
|
// The log packet is followed by the opcode, then the sense data, and then the
|
|
// insertion strings.
|
|
//
|
|
logEntrySize = logEntrySize + stringsBufferLength;
|
|
|
|
if (logEntrySize > ERROR_LOG_MAXIMUM_SIZE) {
|
|
if (senseBufferSize) {
|
|
if (logEntrySize - ERROR_LOG_MAXIMUM_SIZE < senseBufferSize) {
|
|
//
|
|
// In below steps, senseBufferSize will become same or less than as ioLogMessageContextHeader->SenseDataSize
|
|
// it can't be more than that.
|
|
//
|
|
senseBufferSize -= logEntrySize - ERROR_LOG_MAXIMUM_SIZE;
|
|
|
|
//
|
|
// Decrease the sensebuffersize further, if needed, to keep senseBufferSize ULONG aligned
|
|
//
|
|
senseBufferSize = ALIGN_DOWN_BY(senseBufferSize, sizeof(ULONG));
|
|
|
|
} else {
|
|
senseBufferSize = 0;
|
|
}
|
|
}
|
|
logEntrySize = ERROR_LOG_MAXIMUM_SIZE;
|
|
}
|
|
|
|
errorLogEntry = (PIO_ERROR_LOG_PACKET)IoAllocateErrorLogEntry(DeviceObject, (UCHAR)logEntrySize);
|
|
|
|
if (errorLogEntry) {
|
|
|
|
RtlZeroMemory(errorLogEntry, logEntrySize);
|
|
errorLogEntry->MajorFunctionCode = IRP_MJ_SCSI;
|
|
errorLogEntry->RetryCount = 1;
|
|
errorLogEntry->DumpDataSize = (USHORT)(sizeof(ULONG) + senseBufferSize);
|
|
errorLogEntry->StringOffset = (USHORT)(FIELD_OFFSET( IO_ERROR_LOG_PACKET, DumpData ) + errorLogEntry->DumpDataSize);
|
|
errorLogEntry->ErrorCode = ioLogMessageContextHeader->ErrorCode;
|
|
errorLogEntry->DumpData[0] = (((ULONG)(ioLogMessageContextHeader->OpCode)) << 24) |
|
|
(((ULONG)(ioLogMessageContextHeader->SrbStatus)) << 16) |
|
|
(((ULONG)(ioLogMessageContextHeader->ScsiStatus)) << 8);
|
|
|
|
//
|
|
// Copy sense data and do padding for sense data if needed, with '-'
|
|
//
|
|
if (senseBufferSize > ioLogMessageContextHeader->SenseDataSize) {
|
|
RtlCopyMemory(&errorLogEntry->DumpData[1], ioLogMessageContextHeader->SenseData, ioLogMessageContextHeader->SenseDataSize);
|
|
RtlFillMemory( (PCHAR)&errorLogEntry->DumpData[1] + ioLogMessageContextHeader->SenseDataSize , (senseBufferSize - ioLogMessageContextHeader->SenseDataSize) , '-' );
|
|
} else {
|
|
RtlCopyMemory(&errorLogEntry->DumpData[1], ioLogMessageContextHeader->SenseData, senseBufferSize);
|
|
}
|
|
|
|
stringIndex = (PWCHAR)((PCHAR)errorLogEntry->DumpData + errorLogEntry->DumpDataSize);
|
|
stringSize = logEntrySize - errorLogEntry->StringOffset;
|
|
|
|
//
|
|
// Add the strings
|
|
//
|
|
switch (ioLogMessageContextHeader->ErrorCode) {
|
|
case IO_ERROR_IO_HARDWARE_ERROR:
|
|
case IO_WARNING_IO_OPERATION_RETRIED: {
|
|
|
|
PIO_RETRIED_LOG_MESSAGE_CONTEXT ioLogMessageContext = (PIO_RETRIED_LOG_MESSAGE_CONTEXT)Context;
|
|
|
|
//
|
|
// The first is a "0x" plus ULONGLONG in hex representing the LBA plus the NULL character.
|
|
// The second is a ULONG representing the disk number plus the NULL character.
|
|
//
|
|
status = RtlStringCbPrintfW(stringIndex, stringSize, L"0x%I64x", ioLogMessageContext->Lba.QuadPart);
|
|
if (NT_SUCCESS(status)) {
|
|
errorLogEntry->NumberOfStrings++;
|
|
|
|
//
|
|
// Add the disk number to the insertion strings.
|
|
//
|
|
stringSize -= (ULONG)(wcslen(stringIndex) + 1) * sizeof(WCHAR);
|
|
stringIndex += (wcslen(stringIndex) + 1);
|
|
|
|
if (stringSize > 0) {
|
|
|
|
status = RtlStringCbPrintfW(stringIndex, stringSize, L"%d", ioLogMessageContext->DeviceNumber);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
errorLogEntry->NumberOfStrings++;
|
|
|
|
stringSize -= (ULONG)(wcslen(stringIndex) + 1) * sizeof(WCHAR);
|
|
stringIndex += (wcslen(stringIndex) + 1);
|
|
|
|
if (stringSize >= (LONG)pdoNameLength && pdoNameLength > 0) {
|
|
ULONG resultLength;
|
|
|
|
//
|
|
// Get the PDO name and place it in the insertion string buffer.
|
|
//
|
|
status = IoGetDeviceProperty(fdoExtension->LowerPdo,
|
|
DevicePropertyPhysicalDeviceObjectName,
|
|
pdoNameLength,
|
|
stringIndex,
|
|
&resultLength);
|
|
|
|
if (NT_SUCCESS(status) && resultLength > 0) {
|
|
errorLogEntry->NumberOfStrings++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// Write the error log packet to the system error logging thread.
|
|
// It will be freed automatically.
|
|
//
|
|
IoWriteErrorLogEntry(errorLogEntry);
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspLogIORetriedEvent: DO (%p), Soft threshold notification logged.\n",
|
|
DeviceObject));
|
|
}
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, (PIRP)workItem);
|
|
|
|
if (ioLogMessageContextHeader->SenseData) {
|
|
ExFreePool(ioLogMessageContextHeader->SenseData);
|
|
}
|
|
if (workItem) {
|
|
IoFreeWorkItem(workItem);
|
|
}
|
|
ExFreePool(ioLogMessageContextHeader);
|
|
}
|
|
|
|
|
|
VOID
|
|
ClasspQueueLogIOEventWithContextWorker(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ ULONG SenseBufferSize,
|
|
_In_ PVOID SenseData,
|
|
_In_ UCHAR SrbStatus,
|
|
_In_ UCHAR ScsiStatus,
|
|
_In_ ULONG ErrorCode,
|
|
_In_ ULONG CdbLength,
|
|
_In_opt_ PCDB Cdb,
|
|
_In_opt_ PTRANSFER_PACKET Pkt
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
Helper function that queues a delayed work item that will eventually
|
|
log an event to the system event log corresponding to passed in ErrorCode.
|
|
The dumpdata is fixed to include the opcode and the sense information.
|
|
But the number of insertion strings varies based on the passed in ErrorCode.
|
|
|
|
Arguments:
|
|
DeviceObject: The FDO that represents the device that was the target of the IO.
|
|
SesneBufferSize: Size of the SenseData buffer.
|
|
SenseData: Error information from the target (to be included in the dump data).
|
|
SrbStatus: Srb status returned by the miniport.
|
|
ScsiStatus: SCSI status associated with the request upon completion from lower layers.
|
|
ErrorCode: Numerical value of the error code.
|
|
CdbLength: Number of bytes of Cdb.
|
|
Cdb: Pointer to the CDB.
|
|
Pkt: The tranfer packet representing the IO of interest. This may be NULL.
|
|
|
|
--*/
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = (PCOMMON_DEVICE_EXTENSION)(DeviceObject->DeviceExtension);
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
POPCODE_SENSE_DATA_IO_LOG_MESSAGE_CONTEXT_HEADER ioLogMessageContextHeader = NULL;
|
|
PVOID senseData = NULL;
|
|
PIO_WORKITEM workItem = NULL;
|
|
ULONG senseBufferSize = 0;
|
|
LARGE_INTEGER lba = {0};
|
|
|
|
if (!commonExtension->IsFdo) {
|
|
return;
|
|
}
|
|
|
|
if (!Cdb) {
|
|
return;
|
|
}
|
|
|
|
workItem = IoAllocateWorkItem(DeviceObject);
|
|
if (!workItem) {
|
|
goto __ClasspQueueLogIOEventWithContextWorker_ExitWithMessage;
|
|
}
|
|
|
|
if (SenseBufferSize) {
|
|
senseData = ExAllocatePoolWithTag(NonPagedPoolNx, SenseBufferSize, CLASSPNP_POOL_TAG_LOG_MESSAGE);
|
|
if (senseData) {
|
|
senseBufferSize = SenseBufferSize;
|
|
}
|
|
}
|
|
|
|
if (CdbLength == 16) {
|
|
REVERSE_BYTES_QUAD(&lba, Cdb->CDB16.LogicalBlock);
|
|
} else {
|
|
((PFOUR_BYTE)&lba.LowPart)->Byte3 = Cdb->CDB10.LogicalBlockByte0;
|
|
((PFOUR_BYTE)&lba.LowPart)->Byte2 = Cdb->CDB10.LogicalBlockByte1;
|
|
((PFOUR_BYTE)&lba.LowPart)->Byte1 = Cdb->CDB10.LogicalBlockByte2;
|
|
((PFOUR_BYTE)&lba.LowPart)->Byte0 = Cdb->CDB10.LogicalBlockByte3;
|
|
}
|
|
|
|
//
|
|
// Calculate the amount of buffer required for the insertion strings.
|
|
//
|
|
switch (ErrorCode) {
|
|
case IO_ERROR_IO_HARDWARE_ERROR:
|
|
case IO_WARNING_IO_OPERATION_RETRIED: {
|
|
|
|
PIO_RETRIED_LOG_MESSAGE_CONTEXT ioLogMessageContext = NULL;
|
|
|
|
ioLogMessageContext = ExAllocatePoolWithTag(NonPagedPoolNx, sizeof(IO_RETRIED_LOG_MESSAGE_CONTEXT), CLASSPNP_POOL_TAG_LOG_MESSAGE);
|
|
if (!ioLogMessageContext) {
|
|
goto __ClasspQueueLogIOEventWithContextWorker_ExitWithMessage;
|
|
}
|
|
|
|
ioLogMessageContext->Lba.QuadPart = lba.QuadPart;
|
|
ioLogMessageContext->DeviceNumber = fdoExtension->DeviceNumber;
|
|
|
|
ioLogMessageContextHeader = (POPCODE_SENSE_DATA_IO_LOG_MESSAGE_CONTEXT_HEADER)ioLogMessageContext;
|
|
|
|
break;
|
|
}
|
|
|
|
default: goto __ClasspQueueLogIOEventWithContextWorker_Exit;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspQueueLogIOEventWithContextWorker: DO (%p), Pkt (%p), Queueing IO retried event log message work item.\n",
|
|
DeviceObject,
|
|
Pkt));
|
|
|
|
ioLogMessageContextHeader->WorkItem = workItem;
|
|
if (senseData) {
|
|
RtlCopyMemory(senseData, SenseData, SenseBufferSize);
|
|
}
|
|
ioLogMessageContextHeader->SenseData = senseData;
|
|
ioLogMessageContextHeader->SenseDataSize = senseBufferSize;
|
|
ioLogMessageContextHeader->SrbStatus = SrbStatus;
|
|
ioLogMessageContextHeader->ScsiStatus = ScsiStatus;
|
|
ioLogMessageContextHeader->OpCode = Cdb->CDB6GENERIC.OperationCode;
|
|
ioLogMessageContextHeader->Reserved = 0;
|
|
ioLogMessageContextHeader->ErrorCode = ErrorCode;
|
|
|
|
ClassAcquireRemoveLock(DeviceObject, (PIRP)(workItem));
|
|
|
|
//
|
|
// Queue a work item to write the system event log.
|
|
//
|
|
IoQueueWorkItem(workItem, ClasspLogIOEventWithContext, DelayedWorkQueue, ioLogMessageContextHeader);
|
|
|
|
return;
|
|
|
|
__ClasspQueueLogIOEventWithContextWorker_ExitWithMessage:
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_GENERAL,
|
|
"ClasspQueueLogIOEventWithContextWorker: DO (%p), Failed to allocate memory for the log message.\n",
|
|
DeviceObject));
|
|
|
|
__ClasspQueueLogIOEventWithContextWorker_Exit:
|
|
if (senseData) {
|
|
ExFreePool(senseData);
|
|
}
|
|
if (workItem) {
|
|
IoFreeWorkItem(workItem);
|
|
}
|
|
if (ioLogMessageContextHeader) {
|
|
ExFreePool(ioLogMessageContextHeader);
|
|
}
|
|
}
|
|
|
|
static
|
|
BOOLEAN
|
|
ValidPersistentReserveScope(
|
|
UCHAR Scope)
|
|
{
|
|
switch (Scope) {
|
|
case RESERVATION_SCOPE_LU:
|
|
case RESERVATION_SCOPE_ELEMENT:
|
|
|
|
return TRUE;
|
|
|
|
default:
|
|
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static
|
|
BOOLEAN
|
|
ValidPersistentReserveType(
|
|
UCHAR Type)
|
|
{
|
|
switch (Type) {
|
|
case RESERVATION_TYPE_WRITE_EXCLUSIVE:
|
|
case RESERVATION_TYPE_EXCLUSIVE:
|
|
case RESERVATION_TYPE_WRITE_EXCLUSIVE_REGISTRANTS:
|
|
case RESERVATION_TYPE_EXCLUSIVE_REGISTRANTS:
|
|
|
|
return TRUE;
|
|
|
|
default:
|
|
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
ClasspPersistentReserve
|
|
|
|
Routine Description:
|
|
|
|
Handles IOCTL_STORAGE_PERSISTENT_RESERVE_IN and IOCTL_STORAGE_PERSISTENT_RESERVE_OUT.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the device object
|
|
Irp - a pointer to the I/O request packet
|
|
Srb - pointer to preallocated SCSI_REQUEST_BLOCK.
|
|
|
|
Return Value:
|
|
|
|
Status Code
|
|
|
|
--*/
|
|
NTSTATUS
|
|
ClasspPersistentReserve(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
{
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PCDB cdb = NULL;
|
|
PPERSISTENT_RESERVE_COMMAND prCommand = Irp->AssociatedIrp.SystemBuffer;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
|
|
NTSTATUS status;
|
|
|
|
ULONG dataBufLen;
|
|
ULONG controlCode = irpStack->Parameters.DeviceIoControl.IoControlCode;
|
|
|
|
BOOLEAN writeToDevice;
|
|
|
|
//
|
|
// Check common input buffer parameters.
|
|
//
|
|
|
|
if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
|
|
sizeof(PERSISTENT_RESERVE_COMMAND) ||
|
|
prCommand->Size < sizeof(PERSISTENT_RESERVE_COMMAND)) {
|
|
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
FREE_POOL(Srb);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
goto ClasspPersistentReserve_Exit;
|
|
}
|
|
|
|
//
|
|
// Check buffer alignment. Only an issue if another kernel mode component
|
|
// (not the I/O manager) allocates the buffer.
|
|
//
|
|
|
|
if ((ULONG_PTR)prCommand & fdoExtension->AdapterDescriptor->AlignmentMask) {
|
|
|
|
status = STATUS_INVALID_USER_BUFFER;
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
FREE_POOL(Srb);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
goto ClasspPersistentReserve_Exit;
|
|
}
|
|
|
|
//
|
|
// Check additional parameters.
|
|
//
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
SrbSetCdbLength(Srb, 10);
|
|
cdb = SrbGetCdb(Srb);
|
|
|
|
if (controlCode == IOCTL_STORAGE_PERSISTENT_RESERVE_IN) {
|
|
|
|
//
|
|
// Check output buffer for PR In.
|
|
//
|
|
|
|
if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
|
|
prCommand->PR_IN.AllocationLength) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
switch (prCommand->PR_IN.ServiceAction) {
|
|
|
|
case RESERVATION_ACTION_READ_KEYS:
|
|
|
|
if (prCommand->PR_IN.AllocationLength < sizeof(PRI_REGISTRATION_LIST)) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
break;
|
|
|
|
case RESERVATION_ACTION_READ_RESERVATIONS:
|
|
|
|
if (prCommand->PR_IN.AllocationLength < sizeof(PRI_RESERVATION_LIST)) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
break;
|
|
}
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
FREE_POOL(Srb);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
goto ClasspPersistentReserve_Exit;
|
|
}
|
|
|
|
//
|
|
// Fill in the CDB.
|
|
//
|
|
|
|
cdb->PERSISTENT_RESERVE_IN.OperationCode = SCSIOP_PERSISTENT_RESERVE_IN;
|
|
cdb->PERSISTENT_RESERVE_IN.ServiceAction = prCommand->PR_IN.ServiceAction;
|
|
|
|
REVERSE_BYTES_SHORT(&(cdb->PERSISTENT_RESERVE_IN.AllocationLength),
|
|
&(prCommand->PR_IN.AllocationLength));
|
|
|
|
dataBufLen = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
writeToDevice = FALSE;
|
|
|
|
|
|
} else {
|
|
|
|
//
|
|
// Verify ServiceAction, Scope, and Type
|
|
//
|
|
|
|
switch (prCommand->PR_OUT.ServiceAction) {
|
|
|
|
case RESERVATION_ACTION_REGISTER:
|
|
case RESERVATION_ACTION_REGISTER_IGNORE_EXISTING:
|
|
case RESERVATION_ACTION_CLEAR:
|
|
|
|
// Scope and type ignored.
|
|
|
|
break;
|
|
|
|
case RESERVATION_ACTION_RESERVE:
|
|
case RESERVATION_ACTION_RELEASE:
|
|
case RESERVATION_ACTION_PREEMPT:
|
|
case RESERVATION_ACTION_PREEMPT_ABORT:
|
|
|
|
if (!ValidPersistentReserveScope(prCommand->PR_OUT.Scope) ||
|
|
!ValidPersistentReserveType(prCommand->PR_OUT.Type)) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Check input buffer for PR Out.
|
|
// Caller must include the PR parameter list.
|
|
//
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
|
|
if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
|
|
(sizeof(PERSISTENT_RESERVE_COMMAND) +
|
|
sizeof(PRO_PARAMETER_LIST)) ||
|
|
prCommand->Size <
|
|
irpStack->Parameters.DeviceIoControl.InputBufferLength) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
}
|
|
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
FREE_POOL(Srb);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
goto ClasspPersistentReserve_Exit;
|
|
}
|
|
|
|
//
|
|
// Fill in the CDB.
|
|
//
|
|
|
|
cdb->PERSISTENT_RESERVE_OUT.OperationCode = SCSIOP_PERSISTENT_RESERVE_OUT;
|
|
cdb->PERSISTENT_RESERVE_OUT.ServiceAction = prCommand->PR_OUT.ServiceAction;
|
|
cdb->PERSISTENT_RESERVE_OUT.Scope = prCommand->PR_OUT.Scope;
|
|
cdb->PERSISTENT_RESERVE_OUT.Type = prCommand->PR_OUT.Type;
|
|
|
|
cdb->PERSISTENT_RESERVE_OUT.ParameterListLength[1] = (UCHAR)sizeof(PRO_PARAMETER_LIST);
|
|
|
|
//
|
|
// Move the parameter list to the beginning of the data buffer (so it is aligned
|
|
// correctly and that the MDL describes it correctly).
|
|
//
|
|
|
|
RtlMoveMemory(prCommand,
|
|
prCommand->PR_OUT.ParameterList,
|
|
sizeof(PRO_PARAMETER_LIST));
|
|
|
|
dataBufLen = sizeof(PRO_PARAMETER_LIST);
|
|
writeToDevice = TRUE;
|
|
}
|
|
|
|
//
|
|
// Fill in the SRB
|
|
//
|
|
|
|
//
|
|
// Set timeout value.
|
|
//
|
|
|
|
SrbSetTimeOutValue(Srb, fdoExtension->TimeOutValue);
|
|
|
|
//
|
|
// Send as a tagged request.
|
|
//
|
|
|
|
SrbSetRequestAttribute(Srb, SRB_HEAD_OF_QUEUE_TAG_REQUEST);
|
|
SrbSetSrbFlags(Srb, SRB_FLAGS_NO_QUEUE_FREEZE | SRB_FLAGS_QUEUE_ACTION_ENABLE);
|
|
|
|
status = ClassSendSrbAsynchronous(DeviceObject,
|
|
Srb,
|
|
Irp,
|
|
prCommand,
|
|
dataBufLen,
|
|
writeToDevice);
|
|
|
|
ClasspPersistentReserve_Exit:
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
/*++
|
|
|
|
ClasspPriorityHint
|
|
|
|
Routine Description:
|
|
|
|
Handles IOCTL_STORAGE_CHECK_PRIORITY_HINT_SUPPORT.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the device object
|
|
Irp - a pointer to the I/O request packet
|
|
|
|
Return Value:
|
|
|
|
Status Code
|
|
|
|
--*/
|
|
NTSTATUS
|
|
ClasspPriorityHint(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PCLASS_PRIVATE_FDO_DATA fdoData = fdoExtension->PrivateFdoData;
|
|
PSTORAGE_PRIORITY_HINT_SUPPORT priSupport = Irp->AssociatedIrp.SystemBuffer;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
//
|
|
// Check whether this device supports idle priority.
|
|
//
|
|
if (!fdoData->IdlePrioritySupported) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto PriorityHintExit;
|
|
}
|
|
|
|
if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
|
|
sizeof(STORAGE_PRIORITY_HINT_SUPPORT)) {
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto PriorityHintExit;
|
|
}
|
|
|
|
RtlZeroMemory(priSupport, sizeof(STORAGE_PRIORITY_HINT_SUPPORT));
|
|
|
|
status = ClassForwardIrpSynchronous(commonExtension, Irp);
|
|
if (!NT_SUCCESS(status)) {
|
|
//
|
|
// If I/O priority is not supported by lower drivers, just set the
|
|
// priorities supported by class driver.
|
|
//
|
|
TracePrint((TRACE_LEVEL_FATAL, TRACE_FLAG_IOCTL, "ClasspPriorityHint: I/O priority not supported by port driver.\n"));
|
|
priSupport->SupportFlags = 0;
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_FATAL, TRACE_FLAG_IOCTL, "ClasspPriorityHint: I/O priorities supported by port driver: %X\n", priSupport->SupportFlags));
|
|
|
|
priSupport->SupportFlags |= (1 << IoPriorityVeryLow) |
|
|
(1 << IoPriorityLow) |
|
|
(1 << IoPriorityNormal) ;
|
|
|
|
TracePrint((TRACE_LEVEL_FATAL, TRACE_FLAG_IOCTL, "ClasspPriorityHint: I/O priorities supported: %X\n", priSupport->SupportFlags));
|
|
Irp->IoStatus.Information = sizeof(STORAGE_PRIORITY_HINT_SUPPORT);
|
|
|
|
PriorityHintExit:
|
|
|
|
Irp->IoStatus.Status = status;
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
return status;
|
|
}
|
|
|
|
/*++
|
|
|
|
ClasspConvertToScsiRequestBlock
|
|
|
|
Routine Description:
|
|
|
|
Convert an extended SRB to a SCSI_REQUEST_BLOCK. This function handles only
|
|
a single SRB and will not converted SRBs that are linked.
|
|
|
|
Arguments:
|
|
|
|
Srb - a pointer to a SCSI_REQUEST_BLOCK
|
|
SrbEx - a pointer to an extended SRB
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
VOID
|
|
ClasspConvertToScsiRequestBlock(
|
|
_Out_ PSCSI_REQUEST_BLOCK Srb,
|
|
_In_ PSTORAGE_REQUEST_BLOCK SrbEx
|
|
)
|
|
{
|
|
PSTOR_ADDR_BTL8 storAddrBtl8;
|
|
ULONG i;
|
|
BOOLEAN foundEntry = FALSE;
|
|
PSRBEX_DATA srbExData;
|
|
|
|
if ((Srb == NULL) || (SrbEx == NULL)) {
|
|
return;
|
|
}
|
|
|
|
RtlZeroMemory(Srb, sizeof(SCSI_REQUEST_BLOCK));
|
|
|
|
Srb->Length = sizeof(SCSI_REQUEST_BLOCK);
|
|
Srb->Function = (UCHAR)SrbEx->SrbFunction;
|
|
Srb->SrbStatus = SrbEx->SrbStatus;
|
|
Srb->QueueTag = (UCHAR)SrbEx->RequestTag;
|
|
Srb->QueueAction = (UCHAR)SrbEx->RequestAttribute;
|
|
Srb->SrbFlags = SrbEx->SrbFlags;
|
|
Srb->DataTransferLength = SrbEx->DataTransferLength;
|
|
Srb->TimeOutValue = SrbEx->TimeOutValue;
|
|
Srb->DataBuffer = SrbEx->DataBuffer;
|
|
Srb->OriginalRequest = SrbEx->OriginalRequest;
|
|
Srb->SrbExtension = SrbEx->MiniportContext;
|
|
Srb->InternalStatus = SrbEx->SystemStatus;
|
|
|
|
//
|
|
// Handle address fields
|
|
//
|
|
if (SrbEx->AddressOffset >= sizeof(STORAGE_REQUEST_BLOCK)) {
|
|
storAddrBtl8 = (PSTOR_ADDR_BTL8)((PCHAR)SrbEx + SrbEx->AddressOffset);
|
|
|
|
if (storAddrBtl8->Type == STOR_ADDRESS_TYPE_BTL8) {
|
|
Srb->PathId = storAddrBtl8->Path;
|
|
Srb->TargetId = storAddrBtl8->Target;
|
|
Srb->Lun = storAddrBtl8->Lun;
|
|
} else {
|
|
// Catch unsupported address types
|
|
NT_ASSERT(FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Handle SRB function specific fields
|
|
//
|
|
if (SrbEx->NumSrbExData > 0) {
|
|
|
|
for (i = 0; i < SrbEx->NumSrbExData; i++) {
|
|
|
|
if ((SrbEx->SrbExDataOffset[i] == 0) ||
|
|
(SrbEx->SrbExDataOffset[i] < sizeof(STORAGE_REQUEST_BLOCK))) {
|
|
// Catch invalid offsets
|
|
NT_ASSERT(FALSE);
|
|
continue;
|
|
}
|
|
|
|
srbExData = (PSRBEX_DATA)((PCHAR)SrbEx + SrbEx->SrbExDataOffset[i]);
|
|
|
|
switch (SrbEx->SrbFunction) {
|
|
|
|
case SRB_FUNCTION_EXECUTE_SCSI:
|
|
|
|
switch (srbExData->Type) {
|
|
|
|
case SrbExDataTypeScsiCdb16:
|
|
Srb->ScsiStatus = ((PSRBEX_DATA_SCSI_CDB16)srbExData)->ScsiStatus;
|
|
Srb->CdbLength = ((PSRBEX_DATA_SCSI_CDB16)srbExData)->CdbLength;
|
|
Srb->SenseInfoBufferLength = ((PSRBEX_DATA_SCSI_CDB16)srbExData)->SenseInfoBufferLength;
|
|
Srb->SenseInfoBuffer = ((PSRBEX_DATA_SCSI_CDB16)srbExData)->SenseInfoBuffer;
|
|
RtlCopyMemory(Srb->Cdb, ((PSRBEX_DATA_SCSI_CDB16)srbExData)->Cdb, sizeof(Srb->Cdb));
|
|
foundEntry = TRUE;
|
|
break;
|
|
|
|
case SrbExDataTypeScsiCdb32:
|
|
Srb->ScsiStatus = ((PSRBEX_DATA_SCSI_CDB32)srbExData)->ScsiStatus;
|
|
Srb->CdbLength = ((PSRBEX_DATA_SCSI_CDB32)srbExData)->CdbLength;
|
|
Srb->SenseInfoBufferLength = ((PSRBEX_DATA_SCSI_CDB32)srbExData)->SenseInfoBufferLength;
|
|
Srb->SenseInfoBuffer = ((PSRBEX_DATA_SCSI_CDB32)srbExData)->SenseInfoBuffer;
|
|
|
|
// Copy only the first 16 bytes
|
|
RtlCopyMemory(Srb->Cdb, ((PSRBEX_DATA_SCSI_CDB32)srbExData)->Cdb, sizeof(Srb->Cdb));
|
|
foundEntry = TRUE;
|
|
break;
|
|
|
|
case SrbExDataTypeScsiCdbVar:
|
|
Srb->ScsiStatus = ((PSRBEX_DATA_SCSI_CDB_VAR)srbExData)->ScsiStatus;
|
|
Srb->CdbLength = (UCHAR)((PSRBEX_DATA_SCSI_CDB_VAR)srbExData)->CdbLength;
|
|
Srb->SenseInfoBufferLength = ((PSRBEX_DATA_SCSI_CDB_VAR)srbExData)->SenseInfoBufferLength;
|
|
Srb->SenseInfoBuffer = ((PSRBEX_DATA_SCSI_CDB_VAR)srbExData)->SenseInfoBuffer;
|
|
|
|
// Copy only the first 16 bytes
|
|
RtlCopyMemory(Srb->Cdb, ((PSRBEX_DATA_SCSI_CDB_VAR)srbExData)->Cdb, sizeof(Srb->Cdb));
|
|
foundEntry = TRUE;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
|
|
}
|
|
break;
|
|
|
|
case SRB_FUNCTION_WMI:
|
|
|
|
if (srbExData->Type == SrbExDataTypeWmi) {
|
|
((PSCSI_WMI_REQUEST_BLOCK)Srb)->WMISubFunction = ((PSRBEX_DATA_WMI)srbExData)->WMISubFunction;
|
|
((PSCSI_WMI_REQUEST_BLOCK)Srb)->WMIFlags = ((PSRBEX_DATA_WMI)srbExData)->WMIFlags;
|
|
((PSCSI_WMI_REQUEST_BLOCK)Srb)->DataPath = ((PSRBEX_DATA_WMI)srbExData)->DataPath;
|
|
foundEntry = TRUE;
|
|
}
|
|
break;
|
|
|
|
case SRB_FUNCTION_PNP:
|
|
|
|
if (srbExData->Type == SrbExDataTypePnP) {
|
|
((PSCSI_PNP_REQUEST_BLOCK)Srb)->PnPAction = ((PSRBEX_DATA_PNP)srbExData)->PnPAction;
|
|
((PSCSI_PNP_REQUEST_BLOCK)Srb)->PnPSubFunction = ((PSRBEX_DATA_PNP)srbExData)->PnPSubFunction;
|
|
((PSCSI_PNP_REQUEST_BLOCK)Srb)->SrbPnPFlags = ((PSRBEX_DATA_PNP)srbExData)->SrbPnPFlags;
|
|
foundEntry = TRUE;
|
|
}
|
|
break;
|
|
|
|
case SRB_FUNCTION_POWER:
|
|
|
|
if (srbExData->Type == SrbExDataTypePower) {
|
|
((PSCSI_POWER_REQUEST_BLOCK)Srb)->DevicePowerState = ((PSRBEX_DATA_POWER)srbExData)->DevicePowerState;
|
|
((PSCSI_POWER_REQUEST_BLOCK)Srb)->PowerAction = ((PSRBEX_DATA_POWER)srbExData)->PowerAction;
|
|
((PSCSI_POWER_REQUEST_BLOCK)Srb)->SrbPowerFlags = ((PSRBEX_DATA_POWER)srbExData)->SrbPowerFlags;
|
|
foundEntry = TRUE;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
|
|
}
|
|
|
|
//
|
|
// Quit on first match
|
|
//
|
|
if (foundEntry) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
NTSTATUS
|
|
ClasspGetMaximumTokenListIdentifier(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_z_ PWSTR RegistryPath,
|
|
_Out_ PULONG MaximumListIdentifier
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine returns the maximum ListIdentifier (to be used when building TokenOperation
|
|
requests) by querying the value MaximumListIdentifier under the key 'RegistryPath'.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - The device handling the request.
|
|
RegistryPath - The absolute registry path under which MaximumListIdentifier resides.
|
|
MaximumListIdentifier - Returns the value being queried.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS or appropriate error status returned by Registry API.
|
|
|
|
--*/
|
|
|
|
{
|
|
RTL_QUERY_REGISTRY_TABLE queryTable[2];
|
|
ULONG value = 0;
|
|
NTSTATUS status;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_PNP,
|
|
"ClasspGetMaximumTokenListIdentifier (%p): Entering function.\n",
|
|
DeviceObject));
|
|
|
|
//
|
|
// Zero the table entries.
|
|
//
|
|
RtlZeroMemory(queryTable, sizeof(queryTable));
|
|
|
|
//
|
|
// The query table has two entries. One for the MaximumListIdentifier and
|
|
// the second which is the 'NULL' terminator.
|
|
//
|
|
// Indicate that there is NO call-back routine.
|
|
//
|
|
queryTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_TYPECHECK;
|
|
|
|
//
|
|
// The value to query.
|
|
//
|
|
queryTable[0].Name = REG_MAX_LIST_IDENTIFIER_VALUE;
|
|
|
|
//
|
|
// Where to put the value, the type of the value, default value and length.
|
|
//
|
|
queryTable[0].EntryContext = &value;
|
|
queryTable[0].DefaultType = (REG_DWORD << RTL_QUERY_REGISTRY_TYPECHECK_SHIFT) | REG_DWORD;
|
|
queryTable[0].DefaultData = &value;
|
|
queryTable[0].DefaultLength = sizeof(value);
|
|
|
|
//
|
|
// Try to get the maximum listIdentifier.
|
|
//
|
|
status = RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
|
|
RegistryPath,
|
|
queryTable,
|
|
NULL,
|
|
NULL);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
*MaximumListIdentifier = value;
|
|
} else {
|
|
*MaximumListIdentifier = 0;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_PNP,
|
|
"ClasspGetMaximumTokenListIdentifier (%p): Exiting function with status %x (maxListId %u).\n",
|
|
DeviceObject,
|
|
status,
|
|
*MaximumListIdentifier));
|
|
|
|
return status;
|
|
}
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
NTSTATUS
|
|
ClasspGetCopyOffloadMaxDuration(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_z_ PWSTR RegistryPath,
|
|
_Out_ PULONG MaxDuration
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine returns the maximum time (in seconds) that a Copy Offload
|
|
operation should take to complete by a target.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - The device handling the request.
|
|
RegistryPath - The absolute registry path under which MaxDuration resides.
|
|
MaxDuration - Returns the value being queried, in seconds.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS or appropriate error status returned by Registry API.
|
|
|
|
--*/
|
|
|
|
{
|
|
RTL_QUERY_REGISTRY_TABLE queryTable[2];
|
|
ULONG value = 0;
|
|
NTSTATUS status;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_PNP,
|
|
"ClasspGetCopyOffloadMaxDuration (%p): Entering function.\n",
|
|
DeviceObject));
|
|
|
|
//
|
|
// Zero the table entries.
|
|
//
|
|
RtlZeroMemory(queryTable, sizeof(queryTable));
|
|
|
|
//
|
|
// The query table has two entries. One for CopyOffloadMaxDuration and
|
|
// the second which is the 'NULL' terminator.
|
|
//
|
|
// Indicate that there is NO call-back routine.
|
|
//
|
|
queryTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_TYPECHECK;
|
|
|
|
//
|
|
// The value to query.
|
|
//
|
|
queryTable[0].Name = CLASSP_REG_COPY_OFFLOAD_MAX_TARGET_DURATION;
|
|
|
|
//
|
|
// Where to put the value, the type of the value, default value and length.
|
|
//
|
|
queryTable[0].EntryContext = &value;
|
|
queryTable[0].DefaultType = (REG_DWORD << RTL_QUERY_REGISTRY_TYPECHECK_SHIFT) | REG_NONE;
|
|
queryTable[0].DefaultData = &value;
|
|
queryTable[0].DefaultLength = sizeof(value);
|
|
|
|
//
|
|
// Try to get the max target duration.
|
|
//
|
|
status = RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
|
|
RegistryPath,
|
|
queryTable,
|
|
NULL,
|
|
NULL);
|
|
|
|
//
|
|
// Don't allow the user to set the value to lower than the default (4s) so
|
|
// they don't break ODX functionality if they accidentally set it too low.
|
|
//
|
|
if (NT_SUCCESS(status) &&
|
|
value > DEFAULT_MAX_TARGET_DURATION) {
|
|
*MaxDuration = value;
|
|
} else {
|
|
*MaxDuration = DEFAULT_MAX_TARGET_DURATION;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_PNP,
|
|
"ClasspGetCopyOffloadMaxDuration (%p): Exiting function with status %x (Max Duration %u seconds).\n",
|
|
DeviceObject,
|
|
status,
|
|
*MaxDuration));
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
_IRQL_requires_max_(APC_LEVEL)
|
|
_IRQL_requires_min_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
NTSTATUS
|
|
ClasspDeviceCopyOffloadProperty(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine returns the copy offload parameters associated with the device.
|
|
|
|
This function must be called at IRQL < DISPATCH_LEVEL.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the device object associated with this request
|
|
Irp - The IRP to be processed
|
|
Srb - The SRB associated with the request
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS code
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS status;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;
|
|
PSTORAGE_PROPERTY_QUERY query;
|
|
PIO_STACK_LOCATION irpStack;
|
|
ULONG length;
|
|
ULONG information;
|
|
PDEVICE_COPY_OFFLOAD_DESCRIPTOR copyOffloadDescr = (PDEVICE_COPY_OFFLOAD_DESCRIPTOR)Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
UNREFERENCED_PARAMETER(Srb);
|
|
|
|
PAGED_CODE();
|
|
|
|
fdoExtension = DeviceObject->DeviceExtension;
|
|
query = (PSTORAGE_PROPERTY_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
length = 0;
|
|
information = 0;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Entering function.\n",
|
|
DeviceObject));
|
|
|
|
//
|
|
// Check proper query type.
|
|
//
|
|
if (query->QueryType == PropertyExistsQuery) {
|
|
|
|
//
|
|
// In order to maintain consistency with the how the rest of the properties
|
|
// are handled, we shall always return success for PropertyExistsQuery.
|
|
//
|
|
status = STATUS_SUCCESS;
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
|
|
} else if (query->QueryType != PropertyStandardQuery) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Unsupported query type %x for Copy Offload property.\n",
|
|
DeviceObject,
|
|
query->QueryType));
|
|
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
}
|
|
|
|
//
|
|
// Request validation.
|
|
// Note that InputBufferLength and IsFdo have been validated beforing entering this routine.
|
|
//
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL) {
|
|
|
|
NT_ASSERT(KeGetCurrentIrql() < DISPATCH_LEVEL);
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Query property for Copy Offload called at incorrect IRQL.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_INVALID_LEVEL;
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
}
|
|
|
|
length = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (length < sizeof(DEVICE_COPY_OFFLOAD_DESCRIPTOR)) {
|
|
|
|
if (length >= sizeof(STORAGE_DESCRIPTOR_HEADER)) {
|
|
|
|
TracePrint((TRACE_LEVEL_WARNING,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Length %u specified for Copy Offload property enough only for header.\n",
|
|
DeviceObject,
|
|
length));
|
|
|
|
information = sizeof(STORAGE_DESCRIPTOR_HEADER);
|
|
copyOffloadDescr->Version = sizeof(DEVICE_COPY_OFFLOAD_DESCRIPTOR);
|
|
copyOffloadDescr->Size = sizeof(DEVICE_COPY_OFFLOAD_DESCRIPTOR);
|
|
|
|
status = STATUS_SUCCESS;
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Incorrect length %u specified for Copy Offload property.\n",
|
|
DeviceObject,
|
|
length));
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
}
|
|
|
|
if (!fdoExtension->FunctionSupportInfo->ValidInquiryPages.BlockDeviceRODLimits) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Command not supported on this device.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_DEVICE_FEATURE_NOT_SUPPORTED;
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
}
|
|
|
|
if (!NT_SUCCESS(fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.CommandStatus)) {
|
|
|
|
status = fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.CommandStatus;
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): VPD retrieval had failed with %x.\n",
|
|
DeviceObject,
|
|
status));
|
|
|
|
goto __ClasspDeviceCopyOffloadProperty_Exit;
|
|
}
|
|
|
|
//
|
|
// Fill in the output buffer. All data is copied from the FDO extension where we
|
|
// cached Block Limits and Block Device Token Limits info when the device was first initialized.
|
|
//
|
|
RtlZeroMemory(copyOffloadDescr, length);
|
|
copyOffloadDescr->Version = 1;
|
|
copyOffloadDescr->Size = sizeof(DEVICE_COPY_OFFLOAD_DESCRIPTOR);
|
|
copyOffloadDescr->MaximumTokenLifetime = fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumInactivityTimer;
|
|
copyOffloadDescr->DefaultTokenLifetime = fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.DefaultInactivityTimer;
|
|
copyOffloadDescr->MaximumTransferSize = fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumTokenTransferSize;
|
|
copyOffloadDescr->OptimalTransferCount = fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.OptimalTransferCount;
|
|
copyOffloadDescr->MaximumDataDescriptors = fdoExtension->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumRangeDescriptors;
|
|
|
|
if (NT_SUCCESS(fdoExtension->FunctionSupportInfo->BlockLimitsData.CommandStatus)) {
|
|
|
|
copyOffloadDescr->MaximumTransferLengthPerDescriptor = fdoExtension->FunctionSupportInfo->BlockLimitsData.MaximumTransferLength;
|
|
copyOffloadDescr->OptimalTransferLengthPerDescriptor = fdoExtension->FunctionSupportInfo->BlockLimitsData.OptimalTransferLength;
|
|
copyOffloadDescr->OptimalTransferLengthGranularity = fdoExtension->FunctionSupportInfo->BlockLimitsData.OptimalTransferLengthGranularity;
|
|
}
|
|
|
|
information = sizeof(DEVICE_COPY_OFFLOAD_DESCRIPTOR);
|
|
status = STATUS_SUCCESS;
|
|
|
|
__ClasspDeviceCopyOffloadProperty_Exit:
|
|
|
|
//
|
|
// Set the size and status in IRP
|
|
//
|
|
Irp->IoStatus.Information = information;
|
|
Irp->IoStatus.Status = status;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspDeviceCopyOffloadProperty (%p): Exiting function with status %x.\n",
|
|
DeviceObject,
|
|
status));
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
_IRQL_requires_max_(APC_LEVEL)
|
|
_IRQL_requires_min_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
NTSTATUS
|
|
ClasspValidateOffloadSupported(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine validates if this device supports offload requests.
|
|
|
|
This function must be called at IRQL < DISPATCH_LEVEL.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the device object associated with this request
|
|
Irp - The IRP to be processed
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS code
|
|
|
|
--*/
|
|
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExt;
|
|
NTSTATUS status;
|
|
|
|
PAGED_CODE();
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadSupported (%p): Entering function. Irp %p\n",
|
|
DeviceObject,
|
|
Irp));
|
|
|
|
fdoExt = DeviceObject->DeviceExtension;
|
|
status = STATUS_SUCCESS;
|
|
|
|
//
|
|
// For now this command is only supported by disk devices
|
|
//
|
|
if ((DeviceObject->DeviceType == FILE_DEVICE_DISK) &&
|
|
(!TEST_FLAG(DeviceObject->Characteristics, FILE_FLOPPY_DISKETTE))) {
|
|
|
|
if (!fdoExt->FunctionSupportInfo->ValidInquiryPages.BlockDeviceRODLimits) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadSupported (%p): Command not supported on this disk device.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_DEVICE_FEATURE_NOT_SUPPORTED;
|
|
goto __ClasspValidateOffloadSupported_Exit;
|
|
}
|
|
|
|
if (!NT_SUCCESS(fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.CommandStatus)) {
|
|
|
|
status = fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.CommandStatus;
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadSupported (%p): VPD retrieval failed with %x.\n",
|
|
DeviceObject,
|
|
status));
|
|
|
|
goto __ClasspValidateOffloadSupported_Exit;
|
|
}
|
|
} else {
|
|
|
|
TracePrint((TRACE_LEVEL_WARNING,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadSupported (%p): Suported only on Disk devices.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_DEVICE_FEATURE_NOT_SUPPORTED;
|
|
goto __ClasspValidateOffloadSupported_Exit;
|
|
}
|
|
|
|
__ClasspValidateOffloadSupported_Exit:
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadSupported (%p): Exiting function Irp %p with status %x.\n",
|
|
DeviceObject,
|
|
Irp,
|
|
status));
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
_IRQL_requires_max_(APC_LEVEL)
|
|
_IRQL_requires_min_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
NTSTATUS
|
|
ClasspValidateOffloadInputParameters(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine does some basic validation of the input parameters of the offload request.
|
|
|
|
This function must be called at IRQL < DISPATCH_LEVEL.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the device object associated with this request
|
|
Irp - The IRP to be processed
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS code
|
|
|
|
--*/
|
|
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;
|
|
PIO_STACK_LOCATION irpStack;
|
|
PDEVICE_MANAGE_DATA_SET_ATTRIBUTES dsmAttributes;
|
|
PDEVICE_DATA_SET_RANGE dataSetRanges;
|
|
ULONG dataSetRangesCount;
|
|
ULONG i;
|
|
NTSTATUS status;
|
|
|
|
PAGED_CODE();
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Entering function Irp %p.\n",
|
|
DeviceObject,
|
|
Irp));
|
|
|
|
fdoExtension = DeviceObject->DeviceExtension;
|
|
irpStack = IoGetCurrentIrpStackLocation (Irp);
|
|
dsmAttributes = Irp->AssociatedIrp.SystemBuffer;
|
|
status = STATUS_SUCCESS;
|
|
|
|
if (!dsmAttributes) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): NULL DsmAttributes passed in.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
|
|
if ((irpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES)) ||
|
|
(irpStack->Parameters.DeviceIoControl.InputBufferLength <
|
|
(sizeof(DEVICE_MANAGE_DATA_SET_ATTRIBUTES) + dsmAttributes->ParameterBlockLength + dsmAttributes->DataSetRangesLength))) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Input buffer size (%u) too small.\n",
|
|
DeviceObject,
|
|
irpStack->Parameters.DeviceIoControl.InputBufferLength));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
|
|
if ((dsmAttributes->DataSetRangesOffset == 0) ||
|
|
(dsmAttributes->DataSetRangesLength == 0)) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Incorrect DataSetRanges [offset %u, length %u].\n",
|
|
DeviceObject,
|
|
dsmAttributes->DataSetRangesOffset,
|
|
dsmAttributes->DataSetRangesLength));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
|
|
dataSetRanges = Add2Ptr(dsmAttributes, dsmAttributes->DataSetRangesOffset);
|
|
dataSetRangesCount = dsmAttributes->DataSetRangesLength / sizeof(DEVICE_DATA_SET_RANGE);
|
|
|
|
if (dataSetRangesCount == 0) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): DataSetRanges specifies no extents.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
|
|
//
|
|
// Some third party disk class drivers do not query the geometry at initialization time,
|
|
// so this information may not be available at this time. If that is the case, we'll
|
|
// first query that information before proceeding with the rest of our validations.
|
|
//
|
|
if (fdoExtension->DiskGeometry.BytesPerSector == 0) {
|
|
status = ClassReadDriveCapacity(fdoExtension->DeviceObject);
|
|
if ((!NT_SUCCESS(status)) || (fdoExtension->DiskGeometry.BytesPerSector == 0)) {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Couldn't retrieve disk geometry, status: %x, bytes/sector: %u.\n",
|
|
DeviceObject,
|
|
status,
|
|
fdoExtension->DiskGeometry.BytesPerSector));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Data must be aligned to sector boundary and
|
|
// LengthInBytes must be > 0 for it to be a valid LBA entry
|
|
//
|
|
for (i = 0; i < dataSetRangesCount; i++) {
|
|
if ((dataSetRanges[i].StartingOffset % fdoExtension->DiskGeometry.BytesPerSector != 0) ||
|
|
(dataSetRanges[i].LengthInBytes % fdoExtension->DiskGeometry.BytesPerSector != 0) ||
|
|
(dataSetRanges[i].LengthInBytes == 0) ) {
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Incorrect DataSetRanges entry %u [offset %I64x, length %I64x].\n",
|
|
DeviceObject,
|
|
i,
|
|
dataSetRanges[i].StartingOffset,
|
|
dataSetRanges[i].LengthInBytes));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
|
|
if ((ULONGLONG)dataSetRanges[i].StartingOffset + dataSetRanges[i].LengthInBytes > (ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Error! DataSetRange %u (starting LBA %I64x) specified length %I64x exceeds the medium's capacity (%I64x).\n",
|
|
DeviceObject,
|
|
i,
|
|
dataSetRanges[i].StartingOffset,
|
|
dataSetRanges[i].LengthInBytes,
|
|
fdoExtension->CommonExtension.PartitionLength.QuadPart));
|
|
|
|
status = STATUS_NONEXISTENT_SECTOR;
|
|
goto __ClasspValidateOffloadInputParameters_Exit;
|
|
}
|
|
}
|
|
|
|
__ClasspValidateOffloadInputParameters_Exit:
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspValidateOffloadInputParameters (%p): Exiting function Irp %p with status %x.\n",
|
|
DeviceObject,
|
|
Irp,
|
|
status));
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
_IRQL_requires_same_
|
|
NTSTATUS
|
|
ClasspGetTokenOperationCommandBufferLength(
|
|
_In_ PDEVICE_OBJECT Fdo,
|
|
_In_ ULONG ServiceAction,
|
|
_Inout_ PULONG CommandBufferLength,
|
|
_Out_opt_ PULONG TokenOperationBufferLength,
|
|
_Out_opt_ PULONG ReceiveTokenInformationBufferLength
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine description:
|
|
|
|
This routine calculates the buffer length required to service a TokenOperation and its
|
|
corresponding ReceiveTokenInformation command.
|
|
|
|
Arguments:
|
|
|
|
Fdo - The functional device object processing the PopulateToken/WriteUsingToken request
|
|
ServiceAction - Used to distinguish between a PopulateToken and a WriteUsingToken operation
|
|
CommandBufferLength - Returns the length of the buffer needed to service the token request (i.e. TokenOperation and its corresponding ReceiveTokenInformation command)
|
|
TokenOperationBufferLength - Optional parameter, which returns the length of the buffer needed to service just the TokenOperation command.
|
|
ReceiveTokenInformationBufferLength - Optional parameter, which returns the length of the buffer needed to service just the ReceiveTokenInformation command.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS
|
|
|
|
--*/
|
|
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExt = Fdo->DeviceExtension;
|
|
PCLASS_PRIVATE_FDO_DATA fdoData = fdoExt->PrivateFdoData;
|
|
ULONG tokenOperationBufferLength;
|
|
ULONG receiveTokenInformationBufferLength;
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = Fdo->DeviceExtension;
|
|
PSTORAGE_ADAPTER_DESCRIPTOR adapterDesc = commonExtension->PartitionZeroExtension->AdapterDescriptor;
|
|
ULONG hwMaxXferLen;
|
|
ULONG bufferLength = 0;
|
|
ULONG tokenOperationHeaderSize;
|
|
ULONG responseSize;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspGetTokenOperationCommandBufferLengths (%p): Entering function.\n",
|
|
Fdo));
|
|
|
|
NT_ASSERT(fdoExt->FunctionSupportInfo->ValidInquiryPages.BlockDeviceRODLimits &&
|
|
NT_SUCCESS(fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.CommandStatus));
|
|
|
|
if (ServiceAction == SERVICE_ACTION_POPULATE_TOKEN) {
|
|
tokenOperationHeaderSize = FIELD_OFFSET(POPULATE_TOKEN_HEADER, BlockDeviceRangeDescriptor);
|
|
responseSize = FIELD_OFFSET(RECEIVE_TOKEN_INFORMATION_RESPONSE_HEADER, TokenDescriptor) + sizeof(BLOCK_DEVICE_TOKEN_DESCRIPTOR);
|
|
} else {
|
|
tokenOperationHeaderSize = FIELD_OFFSET(WRITE_USING_TOKEN_HEADER, BlockDeviceRangeDescriptor);
|
|
responseSize = 0;
|
|
}
|
|
|
|
//
|
|
// The TokenOperation command can specify a parameter length of max 2^16.
|
|
// If the device has a max limit on the number of range descriptors that can be specified in
|
|
// the TokenOperation command, we are limited to the lesser of these two values.
|
|
//
|
|
if (fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumRangeDescriptors == 0) {
|
|
|
|
tokenOperationBufferLength = MAX_TOKEN_OPERATION_PARAMETER_DATA_LENGTH;
|
|
|
|
} else {
|
|
|
|
tokenOperationBufferLength = MIN(tokenOperationHeaderSize + fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumRangeDescriptors * sizeof(BLOCK_DEVICE_RANGE_DESCRIPTOR),
|
|
MAX_TOKEN_OPERATION_PARAMETER_DATA_LENGTH);
|
|
}
|
|
|
|
|
|
//
|
|
// The ReceiveTokenInformation command can specify a parameter length of max 2 ^ 32
|
|
// Also, since the sense data can be of variable size, we'll use MAX_SENSE_BUFFER_SIZE.
|
|
//
|
|
receiveTokenInformationBufferLength = MIN(FIELD_OFFSET(RECEIVE_TOKEN_INFORMATION_HEADER, SenseData) + MAX_SENSE_BUFFER_SIZE + responseSize,
|
|
MAX_RECEIVE_TOKEN_INFORMATION_PARAMETER_DATA_LENGTH);
|
|
|
|
//
|
|
// Since we're going to reuse the buffer for both the TokenOperation and the ReceiveTokenInformation
|
|
// commands, the buffer length needs to handle both operations.
|
|
//
|
|
bufferLength = MAX(tokenOperationBufferLength, receiveTokenInformationBufferLength);
|
|
|
|
//
|
|
// The buffer length needs to be further limited to the adapter's capability though.
|
|
//
|
|
hwMaxXferLen = MIN(fdoData->HwMaxXferLen, adapterDesc->MaximumTransferLength);
|
|
bufferLength = MIN(bufferLength, hwMaxXferLen);
|
|
|
|
*CommandBufferLength = bufferLength;
|
|
|
|
if (TokenOperationBufferLength) {
|
|
*TokenOperationBufferLength = tokenOperationBufferLength;
|
|
}
|
|
|
|
if (ReceiveTokenInformationBufferLength) {
|
|
*ReceiveTokenInformationBufferLength = receiveTokenInformationBufferLength;
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspGetTokenOperationCommandBufferLengths (%p): Exiting function with bufferLength %u (tokenOpBufLen %u, recTokenInfoBufLen %u).\n",
|
|
Fdo,
|
|
bufferLength,
|
|
tokenOperationBufferLength,
|
|
receiveTokenInformationBufferLength));
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
_IRQL_requires_same_
|
|
NTSTATUS
|
|
ClasspGetTokenOperationDescriptorLimits(
|
|
_In_ PDEVICE_OBJECT Fdo,
|
|
_In_ ULONG ServiceAction,
|
|
_In_ ULONG MaxParameterBufferLength,
|
|
_Out_ PULONG MaxBlockDescriptorsCount,
|
|
_Out_ PULONGLONG MaxBlockDescriptorsLength
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine description:
|
|
|
|
This routine calculates the maximum block descriptors and the maximum token transfer size
|
|
that can be accomodated in a single TokenOperation command.
|
|
|
|
Arguments:
|
|
|
|
Fdo - The functional device object processing the PopulateToken/WriteUsingToken request
|
|
ServiceAction - Used to distinguish between a PopulateToken and a WriteUsingToken operation
|
|
MaxParameterBufferLength - The length constraint of the entire buffer for the parameter list based on other limitations (e.g. adapter max transfer length)
|
|
MaxBlockDescriptorsCount - Returns the maximum number of the block range descriptors that can be passed in a single TokenOperation command.
|
|
MaxBlockDescriptorsLength - Returns the maximum cumulative number of blocks across all the descriptors that must not be exceeded in a single TokenOperation command.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS
|
|
|
|
--*/
|
|
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExt = Fdo->DeviceExtension;
|
|
ULONG tokenOperationHeaderSize = (ServiceAction == SERVICE_ACTION_POPULATE_TOKEN) ?
|
|
FIELD_OFFSET(POPULATE_TOKEN_HEADER, BlockDeviceRangeDescriptor) :
|
|
FIELD_OFFSET(WRITE_USING_TOKEN_HEADER, BlockDeviceRangeDescriptor);
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspGetTokenOperationDescriptorLimits (%p): Entering function.\n",
|
|
Fdo));
|
|
|
|
NT_ASSERT(fdoExt->FunctionSupportInfo->ValidInquiryPages.BlockDeviceRODLimits &&
|
|
NT_SUCCESS(fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.CommandStatus));
|
|
|
|
*MaxBlockDescriptorsCount = (MaxParameterBufferLength - tokenOperationHeaderSize) / sizeof(BLOCK_DEVICE_RANGE_DESCRIPTOR);
|
|
*MaxBlockDescriptorsLength = (fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumTokenTransferSize == 0) ?
|
|
MAX_TOKEN_TRANSFER_SIZE : fdoExt->FunctionSupportInfo->BlockDeviceRODLimitsData.MaximumTokenTransferSize;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspGetTokenOperationDescriptorLimits (%p): Exiting function with MaxDescr %u, MaxXferBlocks %I64u.\n",
|
|
Fdo,
|
|
*MaxBlockDescriptorsCount,
|
|
*MaxBlockDescriptorsLength));
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
_IRQL_requires_max_(APC_LEVEL)
|
|
_IRQL_requires_min_(PASSIVE_LEVEL)
|
|
_IRQL_requires_same_
|
|
VOID
|
|
ClasspConvertDataSetRangeToBlockDescr(
|
|
_In_ PDEVICE_OBJECT Fdo,
|
|
_In_ PVOID BlockDescr,
|
|
_Inout_ PULONG CurrentBlockDescrIndex,
|
|
_In_ ULONG MaxBlockDescrCount,
|
|
_Inout_ PULONG CurrentLbaCount,
|
|
_In_ ULONGLONG MaxLbaCount,
|
|
_Inout_ PDEVICE_DATA_SET_RANGE DataSetRange,
|
|
_Inout_ PULONGLONG TotalSectorsProcessed
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Convert DEVICE_DATA_SET_RANGE entry to be WINDOWS_BLOCK_DEVICE_RANGE_DESCRIPTOR entries.
|
|
|
|
As LengthInBytes field in DEVICE_DATA_SET_RANGE structure is 64 bits (bytes)
|
|
and LbaCount field in WINDOWS_BLOCK_DEVICE_RANGE_DESCRIPTOR structure is 32 bits (sectors),
|
|
it's possible that one DEVICE_DATA_SET_RANGE entry needs multiple
|
|
WINDOWS_BLOCK_DEVICE_RANGE_DESCRIPTOR entries. This routine handles the need for that
|
|
potential split.
|
|
|
|
Arguments:
|
|
|
|
Fdo - The functional device object
|
|
BlockDescr - Pointer to the start of the Token Operation command's block descriptor
|
|
CurrentBlockDescrIndex - Index into the block descriptor at which to update the DataSetRange info
|
|
It also gets updated to return the index to the next empty one.
|
|
MaxBlockDescrCount - Maximum number of block descriptors that the device can handle in a single TokenOperation command
|
|
CurrentLbaCount - Returns the LBA of the last successfully processed DataSetRange
|
|
MaxLbaCount - Maximum transfer size that the device is capable of handling in a single TokenOperation command
|
|
DataSetRange - Contains information about one range extent that needs to be converted into a block descriptor
|
|
TotalSectorsProcessed - Returns the number of sectors corresponding to the DataSetRange that were succesfully mapped into block descriptors
|
|
|
|
Return Value:
|
|
|
|
Nothing.
|
|
|
|
NOTE: if LengthInBytes does not reach to 0, the conversion for DEVICE_DATA_SET_RANGE entry
|
|
is not completed. Further conversion is needed by calling this function again.
|
|
|
|
--*/
|
|
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;
|
|
PBLOCK_DEVICE_RANGE_DESCRIPTOR blockDescr;
|
|
ULONGLONG startingSector;
|
|
ULONGLONG sectorCount;
|
|
ULONGLONG totalSectorCount;
|
|
ULONGLONG numberOfOptimalChunks;
|
|
USHORT optimalLbaPerDescrGranularity;
|
|
ULONG optimalLbaPerDescr;
|
|
ULONG maxLbaPerDescr;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspConvertDataSetRangeToBlockDescr (%p): Entering function. Starting offset %I64x.\n",
|
|
Fdo,
|
|
DataSetRange->StartingOffset));
|
|
|
|
fdoExtension = Fdo->DeviceExtension;
|
|
blockDescr = (PBLOCK_DEVICE_RANGE_DESCRIPTOR)BlockDescr;
|
|
totalSectorCount = 0;
|
|
|
|
|
|
//
|
|
// Since the OptimalTransferLength and the MaximumTransferLength are overloaded parameters for
|
|
// offloaded data transfers and regular read/write requests, it is not safe to use these values
|
|
// as they may report back what is used by regular read/write, which will cause a perf degradation
|
|
// in the offloaded case, since we may end up limiting the per block range descriptor length
|
|
// specified as opposed to what the target can actually handle in a single request.
|
|
// So until the SPC spec introduces these values specific to offloaded data transfers, we shall
|
|
// ignore them completely. The expectation we have from the target is as follows:
|
|
// 1. If the length specified in any of the block range descriptors is greater than the OTL that
|
|
// applies to ODX, the target will internally split into additional descriptors.
|
|
// 2. If the above causes it to run out of descriptors, or if the length specified in any of the
|
|
// descriptors is greater than the MTL that applies to ODX, the target will operate on as much
|
|
// data as possible and truncate the request to that point.
|
|
//
|
|
optimalLbaPerDescrGranularity = 0;
|
|
optimalLbaPerDescr = 0;
|
|
maxLbaPerDescr = 0;
|
|
|
|
if (optimalLbaPerDescr && maxLbaPerDescr) {
|
|
|
|
NT_ASSERT(optimalLbaPerDescr <= maxLbaPerDescr);
|
|
}
|
|
|
|
while ((DataSetRange->LengthInBytes > 0) &&
|
|
(*CurrentBlockDescrIndex < MaxBlockDescrCount) &&
|
|
(*CurrentLbaCount < MaxLbaCount)) {
|
|
|
|
startingSector = (ULONGLONG)(DataSetRange->StartingOffset / fdoExtension->DiskGeometry.BytesPerSector);
|
|
|
|
//
|
|
// Since the block descriptor has only 4 bytes for the number of logical blocks, we are
|
|
// constrained by that theoretical maximum.
|
|
//
|
|
sectorCount = MIN(DataSetRange->LengthInBytes / fdoExtension->DiskGeometry.BytesPerSector,
|
|
MAX_NUMBER_BLOCKS_PER_BLOCK_DEVICE_RANGE_DESCRIPTOR);
|
|
|
|
//
|
|
// We are constrained by MaxLbaCount.
|
|
//
|
|
if (((ULONGLONG)*CurrentLbaCount + sectorCount) >= MaxLbaCount) {
|
|
|
|
sectorCount = MaxLbaCount - *CurrentLbaCount;
|
|
}
|
|
|
|
//
|
|
// For each descriptor, the block count should be lesser than the MaximumTransferSize
|
|
//
|
|
if (maxLbaPerDescr > 0) {
|
|
|
|
//
|
|
// Each block device range descriptor can specify a max number of LBAs
|
|
//
|
|
sectorCount = MIN(sectorCount, maxLbaPerDescr);
|
|
}
|
|
|
|
//
|
|
// If the number of LBAs specified in the descriptor is greater than the OptimalTransferLength,
|
|
// processing of this descriptor by the target may incur a significant delay.
|
|
// So in order to allow the target to perform optimally, we'll further limit the number
|
|
// of blocks specified in any descriptor to be maximum OptimalTranferLength.
|
|
//
|
|
if (optimalLbaPerDescr > 0) {
|
|
|
|
sectorCount = MIN(sectorCount, optimalLbaPerDescr);
|
|
}
|
|
|
|
//
|
|
// In addition, it should either be an exact multiple of the OptimalTransferLengthGranularity,
|
|
// or be lesser than the OptimalTransferLengthGranularity (taken care of here).
|
|
//
|
|
if (optimalLbaPerDescrGranularity > 0) {
|
|
|
|
numberOfOptimalChunks = sectorCount / optimalLbaPerDescrGranularity;
|
|
|
|
if (numberOfOptimalChunks > 0) {
|
|
sectorCount = numberOfOptimalChunks * optimalLbaPerDescrGranularity;
|
|
}
|
|
}
|
|
|
|
NT_ASSERT(sectorCount <= MAX_NUMBER_BLOCKS_PER_BLOCK_DEVICE_RANGE_DESCRIPTOR);
|
|
|
|
REVERSE_BYTES_QUAD(blockDescr[*CurrentBlockDescrIndex].LogicalBlockAddress, &startingSector);
|
|
REVERSE_BYTES(blockDescr[*CurrentBlockDescrIndex].TransferLength, §orCount);
|
|
|
|
totalSectorCount += sectorCount;
|
|
|
|
DataSetRange->StartingOffset += sectorCount * fdoExtension->DiskGeometry.BytesPerSector;
|
|
DataSetRange->LengthInBytes -= sectorCount * fdoExtension->DiskGeometry.BytesPerSector;
|
|
|
|
*CurrentBlockDescrIndex += 1;
|
|
*CurrentLbaCount += (ULONG)sectorCount;
|
|
|
|
TracePrint((TRACE_LEVEL_INFORMATION,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspConvertDataSetRangeToBlockDescr (%p): Descriptor: %u, starting LBA: %I64x, length: %I64x bytes, media size: %I64x.\n",
|
|
Fdo,
|
|
*CurrentBlockDescrIndex - 1,
|
|
startingSector,
|
|
sectorCount * fdoExtension->DiskGeometry.BytesPerSector,
|
|
(ULONGLONG)fdoExtension->CommonExtension.PartitionLength.QuadPart));
|
|
}
|
|
|
|
*TotalSectorsProcessed = totalSectorCount;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspConvertDataSetRangeToBlockDescr (%p): Exiting function (starting offset %I64x). Total sectors processed %I64u.\n",
|
|
Fdo,
|
|
DataSetRange->StartingOffset,
|
|
totalSectorCount));
|
|
|
|
return;
|
|
}
|
|
|
|
_IRQL_requires_same_
|
|
PUCHAR
|
|
ClasspBinaryToAscii(
|
|
_In_reads_(Length) PUCHAR HexBuffer,
|
|
_In_ ULONG Length,
|
|
_Inout_ PULONG UpdateLength
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will convert HexBuffer into an ascii NULL-terminated string.
|
|
|
|
Note: This routine will allocate memory for storing the ascii string. It is
|
|
the responsibility of the caller to free this buffer.
|
|
|
|
Arguments:
|
|
|
|
HexBuffer - Pointer to the binary data.
|
|
Length - Length, in bytes, of HexBuffer.
|
|
UpdateLength - Storage to place the actual length of the returned string.
|
|
|
|
Return Value:
|
|
|
|
ASCII string equivalent of the hex buffer, or NULL if an error occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
static const UCHAR integerTable[] = {'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
|
|
ULONG i;
|
|
ULONG j;
|
|
ULONG actualLength;
|
|
PUCHAR buffer = NULL;
|
|
UCHAR highWord;
|
|
UCHAR lowWord;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspBinaryToAscii (HexBuff %p): Entering function.\n",
|
|
HexBuffer));
|
|
|
|
if (!HexBuffer || Length == 0) {
|
|
*UpdateLength = 0;
|
|
goto __ClasspBinaryToAscii_Exit;
|
|
}
|
|
|
|
//
|
|
// Each byte converts into 2 chars:
|
|
// e.g. 0x05 => '0' '5'
|
|
// 0x0C => '0' 'C'
|
|
// 0x12 => '1' '2'
|
|
// And we need a terminating NULL for the string.
|
|
//
|
|
actualLength = (Length * 2) + 1;
|
|
|
|
//
|
|
// Allocate the buffer.
|
|
//
|
|
buffer = ExAllocatePoolWithTag(NonPagedPoolNx, actualLength, CLASSPNP_POOL_TAG_TOKEN_OPERATION);
|
|
if (!buffer) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspBinaryToAscii (HexBuff %p): Failed to allocate buffer for ASCII equivalent.\n",
|
|
HexBuffer));
|
|
|
|
*UpdateLength = 0;
|
|
goto __ClasspBinaryToAscii_Exit;
|
|
}
|
|
|
|
RtlZeroMemory(buffer, actualLength);
|
|
|
|
for (i = 0, j = 0; i < Length; i++) {
|
|
|
|
//
|
|
// Split out each nibble from the binary byte.
|
|
//
|
|
highWord = HexBuffer[i] >> 4;
|
|
lowWord = HexBuffer[i] & 0x0F;
|
|
|
|
//
|
|
// Using the lookup table, convert and stuff into
|
|
// the ascii buffer.
|
|
//
|
|
buffer[j++] = integerTable[highWord];
|
|
#ifdef _MSC_VER
|
|
#pragma warning(suppress: 6386) // PREFast bug means it doesn't see that Length < actualLength
|
|
#endif
|
|
buffer[j++] = integerTable[lowWord];
|
|
}
|
|
|
|
//
|
|
// Update the caller's length field.
|
|
//
|
|
*UpdateLength = actualLength;
|
|
|
|
__ClasspBinaryToAscii_Exit:
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspBinaryToAscii (HexBuff %p): Exiting function with buffer %s.\n",
|
|
HexBuffer,
|
|
(buffer == NULL) ? "" : (const char*)buffer));
|
|
|
|
return buffer;
|
|
}
|
|
|
|
_IRQL_requires_same_
|
|
NTSTATUS
|
|
ClasspStorageEventNotification(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_ PIRP Irp
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine handles an asynchronous event notification (most likely from
|
|
port drivers). Currently, we only care about media status change events.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the device object associated with this request
|
|
Irp - The IRP to be processed
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS code
|
|
|
|
--*/
|
|
|
|
{
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension;
|
|
PIO_STACK_LOCATION irpStack;
|
|
PSTORAGE_EVENT_NOTIFICATION storageEvents;
|
|
NTSTATUS status;
|
|
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspStorageEventNotification (%p): Entering function Irp %p.\n",
|
|
DeviceObject,
|
|
Irp));
|
|
|
|
fdoExtension = DeviceObject->DeviceExtension;
|
|
irpStack = IoGetCurrentIrpStackLocation (Irp);
|
|
storageEvents = Irp->AssociatedIrp.SystemBuffer;
|
|
status = STATUS_SUCCESS;
|
|
|
|
if (!storageEvents) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspStorageEventNotification (%p): NULL storage events passed in.\n",
|
|
DeviceObject));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspStorageEventNotification_Exit;
|
|
}
|
|
|
|
if (irpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(STORAGE_EVENT_NOTIFICATION)) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspStorageEventNotification (%p): Input buffer size (%u) too small.\n",
|
|
DeviceObject,
|
|
irpStack->Parameters.DeviceIoControl.InputBufferLength));
|
|
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
goto __ClasspStorageEventNotification_Exit;
|
|
}
|
|
|
|
if ((storageEvents->Version != STORAGE_EVENT_NOTIFICATION_VERSION_V1) ||
|
|
(storageEvents->Size != sizeof(STORAGE_EVENT_NOTIFICATION))) {
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspStorageEventNotification (%p): Invalid version/size [version %u, size %u].\n",
|
|
DeviceObject,
|
|
storageEvents->Version,
|
|
storageEvents->Size));
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto __ClasspStorageEventNotification_Exit;
|
|
}
|
|
|
|
//
|
|
// Handle a media status event.
|
|
//
|
|
if (storageEvents->Events & STORAGE_EVENT_MEDIA_STATUS) {
|
|
|
|
//
|
|
// Only initiate operation if underlying port driver supports asynchronous notification
|
|
// and this is the FDO.
|
|
//
|
|
if ((fdoExtension->CommonExtension.IsFdo == TRUE) &&
|
|
(fdoExtension->FunctionSupportInfo->AsynchronousNotificationSupported)) {
|
|
ClassCheckMediaState(fdoExtension);
|
|
} else {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
}
|
|
|
|
__ClasspStorageEventNotification_Exit:
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_IOCTL,
|
|
"ClasspStorageEventNotification (%p): Exiting function Irp %p with status %x.\n",
|
|
DeviceObject,
|
|
Irp,
|
|
status));
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
Irp->IoStatus.Status = status;
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
VOID
|
|
ClasspZeroQERR(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will attempt to set the QERR bit of the mode Control page to
|
|
zero.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the device object associated with this request
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
PMODE_PARAMETER_HEADER modeData = NULL;
|
|
PMODE_CONTROL_PAGE pageData = NULL;
|
|
ULONG size = 0;
|
|
|
|
modeData = ExAllocatePoolWithTag(NonPagedPoolNxCacheAligned,
|
|
MODE_PAGE_DATA_SIZE,
|
|
CLASS_TAG_MODE_DATA);
|
|
|
|
if (modeData == NULL) {
|
|
TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_SCSI, "ClasspZeroQERR: Unable to allocate mode data buffer\n"));
|
|
goto ClasspZeroQERR_Exit;
|
|
}
|
|
|
|
RtlZeroMemory(modeData, MODE_PAGE_DATA_SIZE);
|
|
|
|
size = ClassModeSense(DeviceObject,
|
|
(PCHAR) modeData,
|
|
MODE_PAGE_DATA_SIZE,
|
|
MODE_PAGE_CONTROL);
|
|
|
|
if (size < sizeof(MODE_PARAMETER_HEADER)) {
|
|
|
|
//
|
|
// Retry the request in case of a check condition.
|
|
//
|
|
|
|
size = ClassModeSense(DeviceObject,
|
|
(PCHAR) modeData,
|
|
MODE_PAGE_DATA_SIZE,
|
|
MODE_PAGE_CONTROL);
|
|
|
|
if (size < sizeof(MODE_PARAMETER_HEADER)) {
|
|
TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_SCSI, "ClasspZeroQERR: Mode Sense failed\n"));
|
|
goto ClasspZeroQERR_Exit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If the size is greater than size indicated by the mode data reset
|
|
// the data to the mode data.
|
|
//
|
|
|
|
if (size > (ULONG) (modeData->ModeDataLength + 1)) {
|
|
size = modeData->ModeDataLength + 1;
|
|
}
|
|
|
|
//
|
|
// Look for control page in the returned mode page data.
|
|
//
|
|
|
|
pageData = ClassFindModePage((PCHAR) modeData,
|
|
size,
|
|
MODE_PAGE_CONTROL,
|
|
TRUE);
|
|
|
|
if (pageData) {
|
|
TracePrint((TRACE_LEVEL_VERBOSE,
|
|
TRACE_FLAG_SCSI,
|
|
"ClasspZeroQERR (%p): Current settings: QERR = %u, TST = %u, TAS = %u.\n",
|
|
DeviceObject,
|
|
pageData->QERR,
|
|
pageData->TST,
|
|
pageData->TAS));
|
|
|
|
if (pageData->QERR != 0) {
|
|
NTSTATUS status;
|
|
UCHAR pageSavable = 0;
|
|
|
|
//
|
|
// Set QERR to 0 with a Mode Select command. Re-use the modeData
|
|
// and pageData structures.
|
|
//
|
|
pageData->QERR = 0;
|
|
|
|
//
|
|
// We use the original Page Savable (PS) value for the Save Pages
|
|
// (SP) bit due to behavior described under the MODE SELECT(6)
|
|
// section of SPC-4.
|
|
//
|
|
pageSavable = pageData->PageSavable;
|
|
|
|
status = ClasspModeSelect(DeviceObject,
|
|
(PCHAR)modeData,
|
|
size,
|
|
pageSavable);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
TracePrint((TRACE_LEVEL_WARNING,
|
|
TRACE_FLAG_SCSI,
|
|
"ClasspZeroQERR (%p): Failed to set QERR = 0 with status %x\n",
|
|
DeviceObject,
|
|
status));
|
|
}
|
|
}
|
|
}
|
|
|
|
ClasspZeroQERR_Exit:
|
|
|
|
if (modeData != NULL) {
|
|
ExFreePool(modeData);
|
|
}
|
|
}
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
NTSTATUS
|
|
ClasspPowerActivateDevice(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine synchronously sends an IOCTL_STORAGE_POWER_ACTIVE to the port
|
|
PDO in order to take an active reference on the given device. The device
|
|
will remain powered up and active for as long as this active reference is
|
|
taken.
|
|
|
|
The caller should ensure idle power management is enabled for the device
|
|
before calling this function.
|
|
|
|
Call ClasspPowerIdleDevice to release the active reference.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the FDO associated with this request.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS if the active reference was successfully taken.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
PIRP irp;
|
|
KEVENT event;
|
|
IO_STATUS_BLOCK ioStatus;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
|
|
NT_ASSERT(fdoExtension->CommonExtension.IsFdo);
|
|
NT_ASSERT(fdoExtension->FunctionSupportInfo->IdlePower.IdlePowerEnabled);
|
|
|
|
KeInitializeEvent(&event, SynchronizationEvent, FALSE);
|
|
|
|
irp = IoBuildDeviceIoControlRequest(IOCTL_STORAGE_POWER_ACTIVE,
|
|
fdoExtension->LowerPdo,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
0,
|
|
FALSE,
|
|
&event,
|
|
&ioStatus);
|
|
|
|
if (irp != NULL) {
|
|
status = IoCallDriver(fdoExtension->LowerPdo, irp);
|
|
if (status == STATUS_PENDING) {
|
|
(VOID)KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL);
|
|
status = ioStatus.Status;
|
|
}
|
|
} else {
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
_IRQL_requires_max_(PASSIVE_LEVEL)
|
|
NTSTATUS
|
|
ClasspPowerIdleDevice(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine synchronously sends an IOCTL_STORAGE_POWER_IDLE to the port
|
|
PDO in order to release an active reference on the given device.
|
|
|
|
A call to ClasspPowerActivateDevice *must* have preceded a call to this
|
|
function.
|
|
|
|
The caller should ensure idle power management is enabled for the device
|
|
before calling this function.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Supplies the FDO associated with this request.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS if the active reference was successfully released.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
PIRP irp;
|
|
KEVENT event;
|
|
IO_STATUS_BLOCK ioStatus;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = (PFUNCTIONAL_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
|
|
|
NT_ASSERT(fdoExtension->CommonExtension.IsFdo);
|
|
NT_ASSERT(fdoExtension->FunctionSupportInfo->IdlePower.IdlePowerEnabled);
|
|
|
|
KeInitializeEvent(&event, SynchronizationEvent, FALSE);
|
|
|
|
irp = IoBuildDeviceIoControlRequest(IOCTL_STORAGE_POWER_IDLE,
|
|
fdoExtension->LowerPdo,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
0,
|
|
FALSE,
|
|
&event,
|
|
&ioStatus);
|
|
|
|
if (irp != NULL) {
|
|
status = IoCallDriver(fdoExtension->LowerPdo, irp);
|
|
if (status == STATUS_PENDING) {
|
|
(VOID)KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL);
|
|
status = ioStatus.Status;
|
|
}
|
|
} else {
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
NTSTATUS
|
|
ClasspGetHwFirmwareInfo(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
{
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
|
|
PSTORAGE_HW_FIRMWARE_INFO firmwareInfo = NULL;
|
|
PSTORAGE_HW_FIRMWARE_INFO_QUERY query = NULL;
|
|
|
|
IO_STATUS_BLOCK ioStatus = {0};
|
|
ULONG dataLength = sizeof(STORAGE_HW_FIRMWARE_INFO);
|
|
ULONG iteration = 1;
|
|
|
|
CLASS_FUNCTION_SUPPORT oldState;
|
|
KLOCK_QUEUE_HANDLE lockHandle;
|
|
|
|
//
|
|
// Try to get firmware information that contains only one slot.
|
|
// We will retry the query if the required buffer size is bigger than that.
|
|
//
|
|
retry:
|
|
|
|
firmwareInfo = ExAllocatePoolWithTag(NonPagedPoolNx, dataLength, CLASSPNP_POOL_TAG_FIRMWARE);
|
|
|
|
if (firmwareInfo == NULL) {
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_INIT, "ClasspGetHwFirmwareInfo: cannot allocate memory to hold data. \n"));
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
RtlZeroMemory(firmwareInfo, dataLength);
|
|
|
|
//
|
|
// Set up query data, making sure the "Flags" field indicating the request is for device itself.
|
|
//
|
|
query = (PSTORAGE_HW_FIRMWARE_INFO_QUERY)firmwareInfo;
|
|
|
|
query->Version = sizeof(STORAGE_HW_FIRMWARE_INFO_QUERY);
|
|
query->Size = sizeof(STORAGE_HW_FIRMWARE_INFO_QUERY);
|
|
query->Flags = 0;
|
|
|
|
//
|
|
// On the first pass we just want to get the first few
|
|
// bytes of the descriptor so we can read it's size
|
|
//
|
|
ClassSendDeviceIoControlSynchronous(IOCTL_STORAGE_FIRMWARE_GET_INFO,
|
|
commonExtension->LowerDeviceObject,
|
|
query,
|
|
sizeof(STORAGE_HW_FIRMWARE_INFO_QUERY),
|
|
dataLength,
|
|
FALSE,
|
|
&ioStatus
|
|
);
|
|
|
|
if (!NT_SUCCESS(ioStatus.Status) &&
|
|
(ioStatus.Status != STATUS_BUFFER_OVERFLOW)) {
|
|
if (ClasspLowerLayerNotSupport(ioStatus.Status)) {
|
|
oldState = InterlockedCompareExchange((PLONG)(&fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport), (LONG)NotSupported, (ULONG)SupportUnknown);
|
|
}
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_INIT, "ClasspGetHwFirmwareInfo: error %lx trying to "
|
|
"query hardware firmware information #%d \n", ioStatus.Status, iteration));
|
|
FREE_POOL(firmwareInfo);
|
|
return ioStatus.Status;
|
|
}
|
|
|
|
//
|
|
// Catch implementation issues from lower level driver.
|
|
//
|
|
if ((firmwareInfo->Version < sizeof(STORAGE_HW_FIRMWARE_INFO)) ||
|
|
(firmwareInfo->Size < sizeof(STORAGE_HW_FIRMWARE_INFO)) ||
|
|
(firmwareInfo->SlotCount == 0) ||
|
|
(firmwareInfo->ImagePayloadMaxSize > fdoExtension->AdapterDescriptor->MaximumTransferLength)) {
|
|
|
|
oldState = InterlockedCompareExchange((PLONG)(&fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport), (LONG)NotSupported, (ULONG)SupportUnknown);
|
|
|
|
TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_INIT, "ClasspGetHwFirmwareInfo: error in returned data! "
|
|
"Version: 0x%X, Size: 0x%X, SlotCount: 0x%X, ActiveSlot: 0x%X, PendingActiveSlot: 0x%X, ImagePayloadMaxSize: 0x%X \n",
|
|
firmwareInfo->Version,
|
|
firmwareInfo->Size,
|
|
firmwareInfo->SlotCount,
|
|
firmwareInfo->ActiveSlot,
|
|
firmwareInfo->PendingActivateSlot,
|
|
firmwareInfo->ImagePayloadMaxSize));
|
|
|
|
FREE_POOL(firmwareInfo);
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
//
|
|
// If the data size is bigger than sizeof(STORAGE_HW_FIRMWARE_INFO), e.g. device has more than one firmware slot,
|
|
// allocate a buffer to get all the data.
|
|
//
|
|
if ((firmwareInfo->Size > sizeof(STORAGE_HW_FIRMWARE_INFO)) &&
|
|
(iteration < 2)) {
|
|
|
|
dataLength = max(firmwareInfo->Size, sizeof(STORAGE_HW_FIRMWARE_INFO) + sizeof(STORAGE_HW_FIRMWARE_SLOT_INFO) * (firmwareInfo->SlotCount - 1));
|
|
|
|
//
|
|
// Retry the query with required buffer length.
|
|
//
|
|
FREE_POOL(firmwareInfo);
|
|
iteration++;
|
|
goto retry;
|
|
}
|
|
|
|
|
|
//
|
|
// Set the support status and use the memory we've allocated as caching buffer.
|
|
// In case of a competing thread already set the state, it will assign the caching buffer so release the current allocated one.
|
|
//
|
|
KeAcquireInStackQueuedSpinLock(&fdoExtension->FunctionSupportInfo->SyncLock, &lockHandle);
|
|
|
|
oldState = InterlockedCompareExchange((PLONG)(&fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport), (LONG)Supported, (ULONG)SupportUnknown);
|
|
|
|
if (oldState == SupportUnknown) {
|
|
fdoExtension->FunctionSupportInfo->HwFirmwareInfo = firmwareInfo;
|
|
} else if (oldState == Supported) {
|
|
//
|
|
// swap the buffers to keep the latest version.
|
|
//
|
|
PSTORAGE_HW_FIRMWARE_INFO cachedInfo = fdoExtension->FunctionSupportInfo->HwFirmwareInfo;
|
|
|
|
fdoExtension->FunctionSupportInfo->HwFirmwareInfo = firmwareInfo;
|
|
|
|
FREE_POOL(cachedInfo);
|
|
} else {
|
|
FREE_POOL(firmwareInfo);
|
|
}
|
|
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
|
|
return ioStatus.Status;
|
|
} // end ClasspGetHwFirmwareInfo()
|
|
|
|
#endif // #if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
#ifndef __REACTOS__ // the functions is not used
|
|
__inline
|
|
BOOLEAN
|
|
ClassDeviceHwFirmwareIsPortDriverSupported(
|
|
_In_ PDEVICE_OBJECT DeviceObject
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function informs the caller whether the port driver supports hardware firmware requests.
|
|
|
|
Arguments:
|
|
DeviceObject: The target object.
|
|
|
|
Return Value:
|
|
|
|
TRUE if the port driver is supported.
|
|
|
|
--*/
|
|
{
|
|
//
|
|
// If the request is for a FDO, process the request for Storport, SDstor and Spaceport only.
|
|
// Don't process it if we don't have a miniport descriptor.
|
|
//
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
|
|
BOOLEAN isSupported = FALSE;
|
|
if (commonExtension->IsFdo && (fdoExtension->MiniportDescriptor != NULL)) {
|
|
isSupported = ((fdoExtension->MiniportDescriptor->Portdriver == StoragePortCodeSetStorport) ||
|
|
(fdoExtension->MiniportDescriptor->Portdriver == StoragePortCodeSetSpaceport) ||
|
|
(fdoExtension->MiniportDescriptor->Portdriver == StoragePortCodeSetSDport ));
|
|
}
|
|
|
|
return isSupported;
|
|
}
|
|
#endif
|
|
|
|
NTSTATUS
|
|
ClassDeviceHwFirmwareGetInfoProcess(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp
|
|
)
|
|
/*
|
|
Routine Description:
|
|
|
|
This function processes the Storage Hardware Firmware Get Information request.
|
|
If the information is not cached yet, it gets from lower level driver.
|
|
|
|
Arguments:
|
|
DeviceObject: The target FDO.
|
|
Irp: The IRP which will contain the output buffer upon completion.
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS code.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_HW_FIRMWARE_INFO_QUERY query = (PSTORAGE_HW_FIRMWARE_INFO_QUERY)Irp->AssociatedIrp.SystemBuffer;
|
|
BOOLEAN passDown = FALSE;
|
|
BOOLEAN copyData = FALSE;
|
|
|
|
|
|
//
|
|
// Input buffer is not big enough to contain required input information.
|
|
//
|
|
if (irpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(STORAGE_HW_FIRMWARE_INFO_QUERY)) {
|
|
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
goto Exit_Firmware_Get_Info;
|
|
}
|
|
|
|
//
|
|
// Output buffer is too small to contain return data.
|
|
//
|
|
if (irpStack->Parameters.DeviceIoControl.OutputBufferLength < sizeof(STORAGE_HW_FIRMWARE_INFO)) {
|
|
|
|
status = STATUS_BUFFER_TOO_SMALL;
|
|
goto Exit_Firmware_Get_Info;
|
|
}
|
|
|
|
//
|
|
// Only process the request for a supported port driver.
|
|
//
|
|
if (!ClassDeviceHwFirmwareIsPortDriverSupported(DeviceObject)) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
goto Exit_Firmware_Get_Info;
|
|
}
|
|
|
|
//
|
|
// Buffer "FunctionSupportInfo" is allocated during start device process. Following check defends against the situation
|
|
// of receiving this IOCTL when the device is created but not started, or device start failed but did not get removed yet.
|
|
//
|
|
if (commonExtension->IsFdo && (fdoExtension->FunctionSupportInfo == NULL)) {
|
|
|
|
status = STATUS_UNSUCCESSFUL;
|
|
goto Exit_Firmware_Get_Info;
|
|
}
|
|
|
|
//
|
|
// Process the situation that request should be forwarded to lower level.
|
|
//
|
|
if (!commonExtension->IsFdo) {
|
|
passDown = TRUE;
|
|
}
|
|
|
|
if ((query->Flags & STORAGE_HW_FIRMWARE_REQUEST_FLAG_CONTROLLER) != 0) {
|
|
passDown = TRUE;
|
|
}
|
|
|
|
if (passDown) {
|
|
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// The request is for a FDO. Process the request.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport == NotSupported) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
goto Exit_Firmware_Get_Info;
|
|
} else {
|
|
//
|
|
// Retrieve information from lower layer for the request. The cached information is not used
|
|
// in case device firmware information changed.
|
|
//
|
|
status = ClasspGetHwFirmwareInfo(DeviceObject);
|
|
copyData = NT_SUCCESS(status);
|
|
}
|
|
|
|
Exit_Firmware_Get_Info:
|
|
|
|
if (copyData) {
|
|
//
|
|
// Firmware information is already cached in classpnp. Return a copy.
|
|
//
|
|
KLOCK_QUEUE_HANDLE lockHandle;
|
|
KeAcquireInStackQueuedSpinLock(&fdoExtension->FunctionSupportInfo->SyncLock, &lockHandle);
|
|
|
|
ULONG dataLength = min(irpStack->Parameters.DeviceIoControl.OutputBufferLength, fdoExtension->FunctionSupportInfo->HwFirmwareInfo->Size);
|
|
|
|
memcpy(Irp->AssociatedIrp.SystemBuffer, fdoExtension->FunctionSupportInfo->HwFirmwareInfo, dataLength);
|
|
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
|
|
Irp->IoStatus.Information = dataLength;
|
|
}
|
|
|
|
Irp->IoStatus.Status = status;
|
|
|
|
#else
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
Irp->IoStatus.Status = status;
|
|
#endif // #if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
_IRQL_requires_same_
|
|
_IRQL_requires_max_(DISPATCH_LEVEL)
|
|
NTSTATUS
|
|
ClassHwFirmwareDownloadComplete (
|
|
_In_ PDEVICE_OBJECT Fdo,
|
|
_In_ PIRP Irp,
|
|
_In_reads_opt_(_Inexpressible_("varies")) PVOID Context
|
|
)
|
|
{
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
PIRP originalIrp;
|
|
|
|
//
|
|
// Free the allocated buffer for firmware image.
|
|
//
|
|
if (Context != NULL) {
|
|
FREE_POOL(Context);
|
|
}
|
|
|
|
originalIrp = irpStack->Parameters.Others.Argument1;
|
|
|
|
NT_ASSERT(originalIrp != NULL);
|
|
|
|
originalIrp->IoStatus.Status = Irp->IoStatus.Status;
|
|
originalIrp->IoStatus.Information = Irp->IoStatus.Information;
|
|
|
|
ClassReleaseRemoveLock(Fdo, originalIrp);
|
|
ClassCompleteRequest(Fdo, originalIrp, IO_DISK_INCREMENT);
|
|
|
|
IoFreeIrp(Irp);
|
|
|
|
return STATUS_MORE_PROCESSING_REQUIRED;
|
|
|
|
} // end ClassHwFirmwareDownloadComplete()
|
|
#endif // #if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
|
|
NTSTATUS
|
|
ClassDeviceHwFirmwareDownloadProcess(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_HW_FIRMWARE_DOWNLOAD firmwareDownload = (PSTORAGE_HW_FIRMWARE_DOWNLOAD)Irp->AssociatedIrp.SystemBuffer;
|
|
BOOLEAN passDown = FALSE;
|
|
ULONG i;
|
|
ULONG bufferSize = 0;
|
|
PUCHAR firmwareImageBuffer = NULL;
|
|
PIRP irp2 = NULL;
|
|
PIO_STACK_LOCATION newStack = NULL;
|
|
PCDB cdb = NULL;
|
|
BOOLEAN lockHeld = FALSE;
|
|
KLOCK_QUEUE_HANDLE lockHandle;
|
|
|
|
|
|
//
|
|
// Input buffer is not big enough to contain required input information.
|
|
//
|
|
if (irpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(STORAGE_HW_FIRMWARE_DOWNLOAD)) {
|
|
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Input buffer basic validation.
|
|
//
|
|
if ((firmwareDownload->Version < sizeof(STORAGE_HW_FIRMWARE_DOWNLOAD)) ||
|
|
(firmwareDownload->Size > irpStack->Parameters.DeviceIoControl.InputBufferLength) ||
|
|
((firmwareDownload->BufferSize + FIELD_OFFSET(STORAGE_HW_FIRMWARE_DOWNLOAD, ImageBuffer)) > firmwareDownload->Size)) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Only process the request for a supported port driver.
|
|
//
|
|
if (!ClassDeviceHwFirmwareIsPortDriverSupported(DeviceObject)) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Buffer "FunctionSupportInfo" is allocated during start device process. Following check defends against the situation
|
|
// of receiving this IOCTL when the device is created but not started, or device start failed but did not get removed yet.
|
|
//
|
|
if (commonExtension->IsFdo && (fdoExtension->FunctionSupportInfo == NULL)) {
|
|
|
|
status = STATUS_UNSUCCESSFUL;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Process the situation that request should be forwarded to lower level.
|
|
//
|
|
if (!commonExtension->IsFdo) {
|
|
passDown = TRUE;
|
|
}
|
|
|
|
if ((firmwareDownload->Flags & STORAGE_HW_FIRMWARE_REQUEST_FLAG_CONTROLLER) != 0) {
|
|
passDown = TRUE;
|
|
}
|
|
|
|
if (passDown) {
|
|
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
FREE_POOL(Srb);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// If firmware information hasn't been cached in classpnp, retrieve it.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo == NULL) {
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport == NotSupported) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
goto Exit_Firmware_Download;
|
|
} else {
|
|
//
|
|
// If this is the first time of retrieving firmware information,
|
|
// send request to lower level to get it.
|
|
//
|
|
status = ClasspGetHwFirmwareInfo(DeviceObject);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Fail the request if the firmware information cannot be retrieved.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo == NULL) {
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport == NotSupported) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
} else {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Acquire the SyncLock to ensure the HwFirmwareInfo pointer doesn't change
|
|
// while we're dereferencing it.
|
|
//
|
|
lockHeld = TRUE;
|
|
KeAcquireInStackQueuedSpinLock(&fdoExtension->FunctionSupportInfo->SyncLock, &lockHandle);
|
|
|
|
//
|
|
// Validate the device support
|
|
//
|
|
if ((fdoExtension->FunctionSupportInfo->HwFirmwareInfo->SupportUpgrade == FALSE) ||
|
|
(fdoExtension->FunctionSupportInfo->HwFirmwareInfo->ImagePayloadAlignment == 0)) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Check if the slot can be used to hold firmware image.
|
|
//
|
|
for (i = 0; i < fdoExtension->FunctionSupportInfo->HwFirmwareInfo->SlotCount; i++) {
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo->Slot[i].SlotNumber == firmwareDownload->Slot) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((i >= fdoExtension->FunctionSupportInfo->HwFirmwareInfo->SlotCount) ||
|
|
(fdoExtension->FunctionSupportInfo->HwFirmwareInfo->Slot[i].ReadOnly == TRUE)) {
|
|
//
|
|
// Either the slot number is out of scope or the slot is read-only.
|
|
//
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Buffer size and alignment validation.
|
|
// Max Offset and Buffer Size can be represented by SCSI command is max value for 3 bytes.
|
|
//
|
|
if ((firmwareDownload->BufferSize == 0) ||
|
|
((firmwareDownload->BufferSize % fdoExtension->FunctionSupportInfo->HwFirmwareInfo->ImagePayloadAlignment) != 0) ||
|
|
(firmwareDownload->BufferSize > fdoExtension->FunctionSupportInfo->HwFirmwareInfo->ImagePayloadMaxSize) ||
|
|
(firmwareDownload->BufferSize > fdoExtension->AdapterDescriptor->MaximumTransferLength) ||
|
|
((firmwareDownload->Offset % fdoExtension->FunctionSupportInfo->HwFirmwareInfo->ImagePayloadAlignment) != 0) ||
|
|
(firmwareDownload->Offset > 0xFFFFFF) ||
|
|
(firmwareDownload->BufferSize > 0xFFFFFF)) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
|
|
//
|
|
// Process the request by translating it into WRITE BUFFER command.
|
|
//
|
|
if (((ULONG_PTR)firmwareDownload->ImageBuffer % fdoExtension->FunctionSupportInfo->HwFirmwareInfo->ImagePayloadAlignment) != 0) {
|
|
//
|
|
// Allocate buffer aligns to ImagePayloadAlignment to accommodate the alignment requirement.
|
|
//
|
|
bufferSize = ALIGN_UP_BY(firmwareDownload->BufferSize, fdoExtension->FunctionSupportInfo->HwFirmwareInfo->ImagePayloadAlignment);
|
|
|
|
//
|
|
// We're done accessing HwFirmwareInfo at this point so we can release
|
|
// the SyncLock.
|
|
//
|
|
NT_ASSERT(lockHeld);
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
lockHeld = FALSE;
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(suppress:6014, "The allocated memory that firmwareImageBuffer points to will be freed in ClassHwFirmwareDownloadComplete().")
|
|
#endif
|
|
firmwareImageBuffer = ExAllocatePoolWithTag(NonPagedPoolNx, bufferSize, CLASSPNP_POOL_TAG_FIRMWARE);
|
|
|
|
if (firmwareImageBuffer == NULL) {
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
RtlZeroMemory(firmwareImageBuffer, bufferSize);
|
|
|
|
RtlCopyMemory(firmwareImageBuffer, firmwareDownload->ImageBuffer, (ULONG)firmwareDownload->BufferSize);
|
|
|
|
} else {
|
|
NT_ASSERT(lockHeld);
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
lockHeld = FALSE;
|
|
|
|
firmwareImageBuffer = firmwareDownload->ImageBuffer;
|
|
bufferSize = (ULONG)firmwareDownload->BufferSize;
|
|
}
|
|
|
|
//
|
|
// Allocate a new irp to send the WRITE BUFFER command down.
|
|
// Similar process as IOCTL_STORAGE_CHECK_VERIFY.
|
|
//
|
|
irp2 = IoAllocateIrp((CCHAR)(DeviceObject->StackSize + 3), FALSE);
|
|
|
|
if (irp2 == NULL) {
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
if (firmwareImageBuffer != firmwareDownload->ImageBuffer) {
|
|
FREE_POOL(firmwareImageBuffer);
|
|
}
|
|
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
//
|
|
// Make sure to acquire the lock for the new irp.
|
|
//
|
|
ClassAcquireRemoveLock(DeviceObject, irp2);
|
|
|
|
irp2->Tail.Overlay.Thread = Irp->Tail.Overlay.Thread;
|
|
IoSetNextIrpStackLocation(irp2);
|
|
|
|
//
|
|
// Set the top stack location and shove the master Irp into the
|
|
// top location
|
|
//
|
|
newStack = IoGetCurrentIrpStackLocation(irp2);
|
|
newStack->Parameters.Others.Argument1 = Irp;
|
|
newStack->DeviceObject = DeviceObject;
|
|
|
|
//
|
|
// Stick the firmware download completion routine onto the stack
|
|
// and prepare the irp for the port driver
|
|
//
|
|
IoSetCompletionRoutine(irp2,
|
|
ClassHwFirmwareDownloadComplete,
|
|
(firmwareImageBuffer != firmwareDownload->ImageBuffer) ? firmwareImageBuffer : NULL,
|
|
TRUE,
|
|
TRUE,
|
|
TRUE);
|
|
|
|
IoSetNextIrpStackLocation(irp2);
|
|
newStack = IoGetCurrentIrpStackLocation(irp2);
|
|
newStack->DeviceObject = DeviceObject;
|
|
newStack->MajorFunction = irpStack->MajorFunction;
|
|
newStack->MinorFunction = irpStack->MinorFunction;
|
|
newStack->Flags = irpStack->Flags;
|
|
|
|
|
|
//
|
|
// Mark the master irp as pending - whether the lower level
|
|
// driver completes it immediately or not this should allow it
|
|
// to go all the way back up.
|
|
//
|
|
IoMarkIrpPending(Irp);
|
|
|
|
//
|
|
// Setup the CDB.
|
|
//
|
|
SrbSetCdbLength(Srb, CDB10GENERIC_LENGTH);
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->WRITE_BUFFER.OperationCode = SCSIOP_WRITE_DATA_BUFF;
|
|
cdb->WRITE_BUFFER.Mode = SCSI_WRITE_BUFFER_MODE_DOWNLOAD_MICROCODE_WITH_OFFSETS_SAVE_DEFER_ACTIVATE;
|
|
cdb->WRITE_BUFFER.ModeSpecific = 0; //Reserved for Mode 0x0E
|
|
cdb->WRITE_BUFFER.BufferID = firmwareDownload->Slot;
|
|
|
|
cdb->WRITE_BUFFER.BufferOffset[0] = *((PCHAR)&firmwareDownload->Offset + 2);
|
|
cdb->WRITE_BUFFER.BufferOffset[1] = *((PCHAR)&firmwareDownload->Offset + 1);
|
|
cdb->WRITE_BUFFER.BufferOffset[2] = *((PCHAR)&firmwareDownload->Offset);
|
|
|
|
cdb->WRITE_BUFFER.ParameterListLength[0] = *((PCHAR)&bufferSize + 2);
|
|
cdb->WRITE_BUFFER.ParameterListLength[1] = *((PCHAR)&bufferSize + 1);
|
|
cdb->WRITE_BUFFER.ParameterListLength[2] = *((PCHAR)&bufferSize);
|
|
|
|
//
|
|
// Send as a tagged command.
|
|
//
|
|
SrbSetRequestAttribute(Srb, SRB_HEAD_OF_QUEUE_TAG_REQUEST);
|
|
SrbSetSrbFlags(Srb, SRB_FLAGS_NO_QUEUE_FREEZE | SRB_FLAGS_QUEUE_ACTION_ENABLE);
|
|
|
|
//
|
|
// Set timeout value.
|
|
//
|
|
SrbSetTimeOutValue(Srb, fdoExtension->TimeOutValue);
|
|
|
|
//
|
|
// This routine uses a completion routine so we don't want to release
|
|
// the remove lock until then.
|
|
//
|
|
status = ClassSendSrbAsynchronous(DeviceObject,
|
|
Srb,
|
|
irp2,
|
|
firmwareImageBuffer,
|
|
bufferSize,
|
|
TRUE);
|
|
|
|
if (status != STATUS_PENDING) {
|
|
//
|
|
// If the new irp cannot be sent down, free allocated memory and bail out.
|
|
//
|
|
if (firmwareImageBuffer != firmwareDownload->ImageBuffer) {
|
|
FREE_POOL(firmwareImageBuffer);
|
|
}
|
|
|
|
//
|
|
// If the irp cannot be sent down, the Srb has been freed. NULL it to prevent from freeing it again.
|
|
//
|
|
Srb = NULL;
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, irp2);
|
|
|
|
IoFreeIrp(irp2);
|
|
|
|
goto Exit_Firmware_Download;
|
|
}
|
|
|
|
return status;
|
|
|
|
Exit_Firmware_Download:
|
|
|
|
//
|
|
// Release the SyncLock if it's still held.
|
|
// This should only happen in the failure path.
|
|
//
|
|
if (lockHeld) {
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
lockHeld = FALSE;
|
|
}
|
|
|
|
//
|
|
// Firmware Download request will be failed.
|
|
//
|
|
NT_ASSERT(!NT_SUCCESS(status));
|
|
|
|
Irp->IoStatus.Status = status;
|
|
|
|
#else
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
Irp->IoStatus.Status = status;
|
|
#endif // #if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
FREE_POOL(Srb);
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
ClassDeviceHwFirmwareActivateProcess(
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_Inout_ PIRP Irp,
|
|
_Inout_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
#if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
|
|
PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
|
|
|
|
PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PSTORAGE_HW_FIRMWARE_ACTIVATE firmwareActivate = (PSTORAGE_HW_FIRMWARE_ACTIVATE)Irp->AssociatedIrp.SystemBuffer;
|
|
BOOLEAN passDown = FALSE;
|
|
PCDB cdb = NULL;
|
|
ULONG i;
|
|
BOOLEAN lockHeld = FALSE;
|
|
KLOCK_QUEUE_HANDLE lockHandle;
|
|
|
|
|
|
//
|
|
// Input buffer is not big enough to contain required input information.
|
|
//
|
|
if (irpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(STORAGE_HW_FIRMWARE_ACTIVATE)) {
|
|
|
|
status = STATUS_INFO_LENGTH_MISMATCH;
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// Input buffer basic validation.
|
|
//
|
|
if ((firmwareActivate->Version < sizeof(STORAGE_HW_FIRMWARE_ACTIVATE)) ||
|
|
(firmwareActivate->Size > irpStack->Parameters.DeviceIoControl.InputBufferLength)) {
|
|
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// Only process the request for a supported port driver.
|
|
//
|
|
if (!ClassDeviceHwFirmwareIsPortDriverSupported(DeviceObject)) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// Buffer "FunctionSupportInfo" is allocated during start device process. Following check defends against the situation
|
|
// of receiving this IOCTL when the device is created but not started, or device start failed but did not get removed yet.
|
|
//
|
|
if (commonExtension->IsFdo && (fdoExtension->FunctionSupportInfo == NULL)) {
|
|
|
|
status = STATUS_UNSUCCESSFUL;
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// Process the situation that request should be forwarded to lower level.
|
|
//
|
|
if (!commonExtension->IsFdo) {
|
|
passDown = TRUE;
|
|
}
|
|
|
|
if ((firmwareActivate->Flags & STORAGE_HW_FIRMWARE_REQUEST_FLAG_CONTROLLER) != 0) {
|
|
passDown = TRUE;
|
|
}
|
|
|
|
if (passDown) {
|
|
|
|
IoCopyCurrentIrpStackLocationToNext(Irp);
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
status = IoCallDriver(commonExtension->LowerDeviceObject, Irp);
|
|
FREE_POOL(Srb);
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// If firmware information hasn't been cached in classpnp, retrieve it.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo == NULL) {
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport == NotSupported) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
goto Exit_Firmware_Activate;
|
|
} else {
|
|
//
|
|
// If this is the first time of retrieving firmware information,
|
|
// send request to lower level to get it.
|
|
//
|
|
status = ClasspGetHwFirmwareInfo(DeviceObject);
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Fail the request if the firmware information cannot be retrieved.
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo == NULL) {
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareGetInfoSupport == NotSupported) {
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
} else {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// Acquire the SyncLock to ensure the HwFirmwareInfo pointer doesn't change
|
|
// while we're dereferencing it.
|
|
//
|
|
lockHeld = TRUE;
|
|
KeAcquireInStackQueuedSpinLock(&fdoExtension->FunctionSupportInfo->SyncLock, &lockHandle);
|
|
|
|
//
|
|
// Validate the device support
|
|
//
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo->SupportUpgrade == FALSE) {
|
|
status = STATUS_NOT_SUPPORTED;
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// Check if the slot number is valid.
|
|
//
|
|
for (i = 0; i < fdoExtension->FunctionSupportInfo->HwFirmwareInfo->SlotCount; i++) {
|
|
if (fdoExtension->FunctionSupportInfo->HwFirmwareInfo->Slot[i].SlotNumber == firmwareActivate->Slot) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (i >= fdoExtension->FunctionSupportInfo->HwFirmwareInfo->SlotCount) {
|
|
//
|
|
// Either the slot number is out of scope or the slot is read-only.
|
|
//
|
|
status = STATUS_INVALID_PARAMETER;
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
//
|
|
// We're done accessing HwFirmwareInfo at this point so we can release
|
|
// the SyncLock.
|
|
//
|
|
NT_ASSERT(lockHeld);
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
lockHeld = FALSE;
|
|
|
|
//
|
|
// Process the request by translating it into WRITE BUFFER command.
|
|
//
|
|
//
|
|
// Setup the CDB. This should be an untagged request.
|
|
//
|
|
SrbSetCdbLength(Srb, CDB10GENERIC_LENGTH);
|
|
cdb = SrbGetCdb(Srb);
|
|
cdb->WRITE_BUFFER.OperationCode = SCSIOP_WRITE_DATA_BUFF;
|
|
cdb->WRITE_BUFFER.Mode = SCSI_WRITE_BUFFER_MODE_ACTIVATE_DEFERRED_MICROCODE;
|
|
cdb->WRITE_BUFFER.ModeSpecific = 0; //Reserved for Mode 0x0F
|
|
cdb->WRITE_BUFFER.BufferID = firmwareActivate->Slot; //NOTE: this field will be ignored by SCSI device.
|
|
|
|
//
|
|
// Set timeout value.
|
|
//
|
|
SrbSetTimeOutValue(Srb, FIRMWARE_ACTIVATE_TIMEOUT_VALUE);
|
|
|
|
//
|
|
// This routine uses a completion routine - ClassIoComplete() so we don't want to release
|
|
// the remove lock until then.
|
|
//
|
|
status = ClassSendSrbAsynchronous(DeviceObject,
|
|
Srb,
|
|
Irp,
|
|
NULL,
|
|
0,
|
|
FALSE);
|
|
|
|
if (status != STATUS_PENDING) {
|
|
//
|
|
// If the irp cannot be sent down, the Srb has been freed. NULL it to prevent from freeing it again.
|
|
//
|
|
Srb = NULL;
|
|
|
|
goto Exit_Firmware_Activate;
|
|
}
|
|
|
|
return status;
|
|
|
|
Exit_Firmware_Activate:
|
|
|
|
//
|
|
// Release the SyncLock if it's still held.
|
|
// This should only happen in the failure path.
|
|
//
|
|
if (lockHeld) {
|
|
KeReleaseInStackQueuedSpinLock(&lockHandle);
|
|
lockHeld = FALSE;
|
|
}
|
|
|
|
//
|
|
// Firmware Activate request will be failed.
|
|
//
|
|
NT_ASSERT(!NT_SUCCESS(status));
|
|
|
|
Irp->IoStatus.Status = status;
|
|
|
|
#else
|
|
status = STATUS_NOT_IMPLEMENTED;
|
|
Irp->IoStatus.Status = status;
|
|
#endif // #if (NTDDI_VERSION >= NTDDI_WINTHRESHOLD)
|
|
|
|
ClassReleaseRemoveLock(DeviceObject, Irp);
|
|
ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
|
|
|
|
FREE_POOL(Srb);
|
|
return status;
|
|
}
|
|
|
|
|
|
BOOLEAN
|
|
ClasspIsThinProvisioningError (
|
|
_In_ PSCSI_REQUEST_BLOCK Srb
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine checks whether the completed SRB Srb was completed with a thin provisioning
|
|
soft threshold error.
|
|
|
|
Arguments:
|
|
|
|
Srb - the SRB to be checked.
|
|
|
|
Return Value:
|
|
|
|
BOOLEAN
|
|
|
|
--*/
|
|
{
|
|
if (TEST_FLAG(Srb->SrbStatus, SRB_STATUS_AUTOSENSE_VALID)) {
|
|
PVOID senseBuffer = SrbGetSenseInfoBuffer(Srb);
|
|
if (senseBuffer) {
|
|
UCHAR senseKey = 0;
|
|
UCHAR addlSenseCode = 0;
|
|
UCHAR addlSenseCodeQual = 0;
|
|
BOOLEAN validSense = ScsiGetSenseKeyAndCodes(senseBuffer,
|
|
SrbGetSenseInfoBufferLength(Srb),
|
|
SCSI_SENSE_OPTIONS_NONE,
|
|
&senseKey,
|
|
&addlSenseCode,
|
|
&addlSenseCodeQual);
|
|
|
|
return (validSense
|
|
&& (senseKey == SCSI_SENSE_UNIT_ATTENTION)
|
|
&& (addlSenseCode == SCSI_ADSENSE_LB_PROVISIONING)
|
|
&& (addlSenseCodeQual == SCSI_SENSEQ_SOFT_THRESHOLD_REACHED));
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|