1 /* 2 * PROJECT: ReactOS Storport Driver 3 * LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+) 4 * PURPOSE: Miniport interface code 5 * COPYRIGHT: Copyright 2017 Eric Kohl (eric.kohl@reactos.org) 6 */ 7 8 /* INCLUDES *******************************************************************/ 9 10 #include "precomp.h" 11 12 #define NDEBUG 13 #include <debug.h> 14 15 16 /* FUNCTIONS ******************************************************************/ 17 18 static 19 NTSTATUS 20 InitializeConfiguration( 21 _In_ PPORT_CONFIGURATION_INFORMATION PortConfig, 22 _In_ PHW_INITIALIZATION_DATA InitData, 23 _In_ ULONG BusNumber, 24 _In_ ULONG SlotNumber) 25 { 26 PCONFIGURATION_INFORMATION ConfigInfo; 27 ULONG i; 28 29 DPRINT1("InitializeConfiguration(%p %p %lu %lu)\n", 30 PortConfig, InitData, BusNumber, SlotNumber); 31 32 /* Get the configurration information */ 33 ConfigInfo = IoGetConfigurationInformation(); 34 35 /* Initialize the port configuration */ 36 RtlZeroMemory(PortConfig, 37 sizeof(PORT_CONFIGURATION_INFORMATION)); 38 39 PortConfig->Length = sizeof(PORT_CONFIGURATION_INFORMATION); 40 PortConfig->SystemIoBusNumber = BusNumber; 41 PortConfig->SlotNumber = SlotNumber; 42 PortConfig->AdapterInterfaceType = InitData->AdapterInterfaceType; 43 44 PortConfig->MaximumTransferLength = -1; //SP_UNINITIALIZED_VALUE; 45 PortConfig->DmaChannel = -1; //SP_UNINITIALIZED_VALUE; 46 PortConfig->DmaPort = -1; //SP_UNINITIALIZED_VALUE; 47 48 PortConfig->InterruptMode = LevelSensitive; 49 50 PortConfig->Master = TRUE; 51 PortConfig->AtdiskPrimaryClaimed = ConfigInfo->AtDiskPrimaryAddressClaimed; 52 PortConfig->AtdiskSecondaryClaimed = ConfigInfo->AtDiskSecondaryAddressClaimed; 53 PortConfig->Dma32BitAddresses = TRUE; 54 PortConfig->DemandMode = FALSE; 55 PortConfig->MapBuffers = InitData->MapBuffers; 56 57 PortConfig->NeedPhysicalAddresses = TRUE; 58 PortConfig->TaggedQueuing = TRUE; 59 PortConfig->AutoRequestSense = TRUE; 60 PortConfig->MultipleRequestPerLu = TRUE; 61 PortConfig->ReceiveEvent = InitData->ReceiveEvent; 62 PortConfig->RealModeInitialized = FALSE; 63 PortConfig->BufferAccessScsiPortControlled = TRUE; 64 PortConfig->MaximumNumberOfTargets = 128; 65 66 PortConfig->SpecificLuExtensionSize = InitData->SpecificLuExtensionSize; 67 PortConfig->SrbExtensionSize = InitData->SrbExtensionSize; 68 PortConfig->MaximumNumberOfLogicalUnits = 1; 69 PortConfig->WmiDataProvider = TRUE; 70 71 PortConfig->NumberOfAccessRanges = InitData->NumberOfAccessRanges; 72 DPRINT1("NumberOfAccessRanges: %lu\n", PortConfig->NumberOfAccessRanges); 73 if (PortConfig->NumberOfAccessRanges != 0) 74 { 75 PortConfig->AccessRanges = ExAllocatePoolWithTag(NonPagedPool, 76 PortConfig->NumberOfAccessRanges * sizeof(ACCESS_RANGE), 77 TAG_ACCRESS_RANGE); 78 if (PortConfig->AccessRanges == NULL) 79 return STATUS_NO_MEMORY; 80 81 RtlZeroMemory(PortConfig->AccessRanges, 82 PortConfig->NumberOfAccessRanges * sizeof(ACCESS_RANGE)); 83 } 84 85 for (i = 0; i < 7; i++) 86 PortConfig->InitiatorBusId[i] = 0xff; 87 88 return STATUS_SUCCESS; 89 } 90 91 92 static 93 VOID 94 AssignResourcesToConfiguration( 95 _In_ PPORT_CONFIGURATION_INFORMATION PortConfiguration, 96 _In_ PCM_RESOURCE_LIST ResourceList, 97 _In_ ULONG NumberOfAccessRanges) 98 { 99 PCM_FULL_RESOURCE_DESCRIPTOR FullDescriptor; 100 PCM_PARTIAL_RESOURCE_DESCRIPTOR PartialDescriptor; 101 PCM_PARTIAL_RESOURCE_LIST PartialResourceList; 102 PACCESS_RANGE AccessRange; 103 INT i, j; 104 ULONG RangeNumber = 0, Interrupt = 0, Dma = 0; 105 106 DPRINT1("AssignResourceToConfiguration(%p %p %lu)\n", 107 PortConfiguration, ResourceList, NumberOfAccessRanges); 108 109 FullDescriptor = &ResourceList->List[0]; 110 for (i = 0; i < ResourceList->Count; i++) 111 { 112 PartialResourceList = &FullDescriptor->PartialResourceList; 113 114 for (j = 0; j < PartialResourceList->Count; j++) 115 { 116 PartialDescriptor = &PartialResourceList->PartialDescriptors[j]; 117 118 switch (PartialDescriptor->Type) 119 { 120 case CmResourceTypePort: 121 DPRINT1("Port: 0x%I64x (0x%lx)\n", 122 PartialDescriptor->u.Port.Start.QuadPart, 123 PartialDescriptor->u.Port.Length); 124 if (RangeNumber < NumberOfAccessRanges) 125 { 126 AccessRange = &((*(PortConfiguration->AccessRanges))[RangeNumber]); 127 AccessRange->RangeStart = PartialDescriptor->u.Port.Start; 128 AccessRange->RangeLength = PartialDescriptor->u.Port.Length; 129 AccessRange->RangeInMemory = FALSE; 130 RangeNumber++; 131 } 132 break; 133 134 case CmResourceTypeMemory: 135 DPRINT1("Memory: 0x%I64x (0x%lx)\n", 136 PartialDescriptor->u.Memory.Start.QuadPart, 137 PartialDescriptor->u.Memory.Length); 138 if (RangeNumber < NumberOfAccessRanges) 139 { 140 AccessRange = &((*(PortConfiguration->AccessRanges))[RangeNumber]); 141 AccessRange->RangeStart = PartialDescriptor->u.Memory.Start; 142 AccessRange->RangeLength = PartialDescriptor->u.Memory.Length; 143 AccessRange->RangeInMemory = TRUE; 144 RangeNumber++; 145 } 146 break; 147 148 case CmResourceTypeInterrupt: 149 DPRINT1("Interrupt: Level %lu Vector %lu\n", 150 PartialDescriptor->u.Interrupt.Level, 151 PartialDescriptor->u.Interrupt.Vector); 152 if (Interrupt == 0) 153 { 154 /* Copy interrupt data */ 155 PortConfiguration->BusInterruptLevel = PartialDescriptor->u.Interrupt.Level; 156 PortConfiguration->BusInterruptVector = PartialDescriptor->u.Interrupt.Vector; 157 158 /* Set interrupt mode accordingly to the resource */ 159 if (PartialDescriptor->Flags == CM_RESOURCE_INTERRUPT_LATCHED) 160 { 161 PortConfiguration->InterruptMode = Latched; 162 } 163 else if (PartialDescriptor->Flags == CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE) 164 { 165 PortConfiguration->InterruptMode = LevelSensitive; 166 } 167 } 168 else if (Interrupt == 1) 169 { 170 /* Copy interrupt data */ 171 PortConfiguration->BusInterruptLevel2 = PartialDescriptor->u.Interrupt.Level; 172 PortConfiguration->BusInterruptVector2 = PartialDescriptor->u.Interrupt.Vector; 173 174 /* Set interrupt mode accordingly to the resource */ 175 if (PartialDescriptor->Flags == CM_RESOURCE_INTERRUPT_LATCHED) 176 { 177 PortConfiguration->InterruptMode2 = Latched; 178 } 179 else if (PartialDescriptor->Flags == CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE) 180 { 181 PortConfiguration->InterruptMode2 = LevelSensitive; 182 } 183 } 184 Interrupt++; 185 break; 186 187 case CmResourceTypeDma: 188 DPRINT1("Dma: Channel: %lu Port: %lu\n", 189 PartialDescriptor->u.Dma.Channel, 190 PartialDescriptor->u.Dma.Port); 191 if (Dma == 0) 192 { 193 PortConfiguration->DmaChannel = PartialDescriptor->u.Dma.Channel; 194 PortConfiguration->DmaPort = PartialDescriptor->u.Dma.Port; 195 196 if (PartialDescriptor->Flags & CM_RESOURCE_DMA_8) 197 PortConfiguration->DmaWidth = Width8Bits; 198 else if ((PartialDescriptor->Flags & CM_RESOURCE_DMA_16) || 199 (PartialDescriptor->Flags & CM_RESOURCE_DMA_8_AND_16)) 200 PortConfiguration->DmaWidth = Width16Bits; 201 else if (PartialDescriptor->Flags & CM_RESOURCE_DMA_32) 202 PortConfiguration->DmaWidth = Width32Bits; 203 } 204 else if (Dma == 1) 205 { 206 PortConfiguration->DmaChannel2 = PartialDescriptor->u.Dma.Channel; 207 PortConfiguration->DmaPort2 = PartialDescriptor->u.Dma.Port; 208 209 if (PartialDescriptor->Flags & CM_RESOURCE_DMA_8) 210 PortConfiguration->DmaWidth2 = Width8Bits; 211 else if ((PartialDescriptor->Flags & CM_RESOURCE_DMA_16) || 212 (PartialDescriptor->Flags & CM_RESOURCE_DMA_8_AND_16)) 213 PortConfiguration->DmaWidth2 = Width16Bits; 214 else if (PartialDescriptor->Flags & CM_RESOURCE_DMA_32) 215 PortConfiguration->DmaWidth2 = Width32Bits; 216 } 217 Dma++; 218 break; 219 220 default: 221 DPRINT1("Other: %u\n", PartialDescriptor->Type); 222 break; 223 } 224 } 225 226 /* Advance to next CM_FULL_RESOURCE_DESCRIPTOR block in memory. */ 227 FullDescriptor = (PCM_FULL_RESOURCE_DESCRIPTOR)(FullDescriptor->PartialResourceList.PartialDescriptors + 228 FullDescriptor->PartialResourceList.Count); 229 } 230 } 231 232 233 NTSTATUS 234 MiniportInitialize( 235 _In_ PMINIPORT Miniport, 236 _In_ PFDO_DEVICE_EXTENSION DeviceExtension, 237 _In_ PHW_INITIALIZATION_DATA InitData) 238 { 239 PMINIPORT_DEVICE_EXTENSION MiniportExtension; 240 ULONG Size; 241 NTSTATUS Status; 242 243 DPRINT1("MiniportInitialize(%p %p %p)\n", 244 Miniport, DeviceExtension, InitData); 245 246 Miniport->DeviceExtension = DeviceExtension; 247 Miniport->InitData = InitData; 248 249 /* Calculate the miniport device extension size */ 250 Size = sizeof(MINIPORT_DEVICE_EXTENSION) + 251 Miniport->InitData->DeviceExtensionSize; 252 253 /* Allocate and initialize the miniport device extension */ 254 MiniportExtension = ExAllocatePoolWithTag(NonPagedPool, 255 Size, 256 TAG_MINIPORT_DATA); 257 if (MiniportExtension == NULL) 258 return STATUS_NO_MEMORY; 259 260 RtlZeroMemory(MiniportExtension, Size); 261 262 MiniportExtension->Miniport = Miniport; 263 Miniport->MiniportExtension = MiniportExtension; 264 265 /* Initialize the port configuration */ 266 Status = InitializeConfiguration(&Miniport->PortConfig, 267 InitData, 268 DeviceExtension->BusNumber, 269 DeviceExtension->SlotNumber); 270 if (!NT_SUCCESS(Status)) 271 return Status; 272 273 /* Assign the resources to the port configuration */ 274 AssignResourcesToConfiguration(&Miniport->PortConfig, 275 DeviceExtension->AllocatedResources, 276 InitData->NumberOfAccessRanges); 277 278 return STATUS_SUCCESS; 279 } 280 281 282 NTSTATUS 283 MiniportFindAdapter( 284 _In_ PMINIPORT Miniport) 285 { 286 BOOLEAN Reserved = FALSE; 287 ULONG Result; 288 NTSTATUS Status; 289 290 DPRINT1("MiniportFindAdapter(%p)\n", Miniport); 291 292 /* Call the miniport HwFindAdapter routine */ 293 Result = Miniport->InitData->HwFindAdapter(&Miniport->MiniportExtension->HwDeviceExtension, 294 NULL, 295 NULL, 296 NULL, 297 &Miniport->PortConfig, 298 &Reserved); 299 DPRINT1("HwFindAdapter() returned %lu\n", Result); 300 301 /* Convert the result to a status code */ 302 switch (Result) 303 { 304 case SP_RETURN_NOT_FOUND: 305 DPRINT1("SP_RETURN_NOT_FOUND\n"); 306 Status = STATUS_NOT_FOUND; 307 break; 308 309 case SP_RETURN_FOUND: 310 DPRINT1("SP_RETURN_FOUND\n"); 311 Status = STATUS_SUCCESS; 312 break; 313 314 case SP_RETURN_ERROR: 315 DPRINT1("SP_RETURN_ERROR\n"); 316 Status = STATUS_ADAPTER_HARDWARE_ERROR; 317 break; 318 319 case SP_RETURN_BAD_CONFIG: 320 DPRINT1("SP_RETURN_BAD_CONFIG\n"); 321 Status = STATUS_DEVICE_CONFIGURATION_ERROR; 322 break; 323 324 default: 325 DPRINT1("Unknown result: %lu\n", Result); 326 Status = STATUS_INTERNAL_ERROR; 327 break; 328 } 329 330 return Status; 331 } 332 333 334 NTSTATUS 335 MiniportHwInitialize( 336 _In_ PMINIPORT Miniport) 337 { 338 BOOLEAN Result; 339 340 DPRINT1("MiniportHwInitialize(%p)\n", Miniport); 341 342 /* Call the miniport HwInitialize routine */ 343 Result = Miniport->InitData->HwInitialize(&Miniport->MiniportExtension->HwDeviceExtension); 344 DPRINT1("HwInitialize() returned %u\n", Result); 345 346 return Result ? STATUS_SUCCESS : STATUS_UNSUCCESSFUL; 347 } 348 349 350 BOOLEAN 351 MiniportHwInterrupt( 352 _In_ PMINIPORT Miniport) 353 { 354 BOOLEAN Result; 355 356 DPRINT1("MiniportHwInterrupt(%p)\n", 357 Miniport); 358 359 Result = Miniport->InitData->HwInterrupt(&Miniport->MiniportExtension->HwDeviceExtension); 360 DPRINT1("HwInterrupt() returned %u\n", Result); 361 362 return Result; 363 } 364 365 366 BOOLEAN 367 MiniportStartIo( 368 _In_ PMINIPORT Miniport, 369 _In_ PSCSI_REQUEST_BLOCK Srb) 370 { 371 BOOLEAN Result; 372 373 DPRINT1("MiniportHwStartIo(%p %p)\n", 374 Miniport, Srb); 375 376 Result = Miniport->InitData->HwStartIo(&Miniport->MiniportExtension->HwDeviceExtension, Srb); 377 DPRINT1("HwStartIo() returned %u\n", Result); 378 379 return Result; 380 } 381 382 /* EOF */ 383