1 /*
2  * PROJECT:        ReactOS Kernel
3  * LICENSE:        GNU GPLv2 only as published by the Free Software Foundation
4  * PURPOSE:        To Implement AHCI Miniport driver targeting storport NT 5.2
5  * PROGRAMMERS:    Aman Priyadarshi (aman.eureka@gmail.com)
6  */
7 
8 #include <ntddk.h>
9 #include <ata.h>
10 #include <storport.h>
11 
12 #define NDEBUG
13 #include <debug.h>
14 
15 #define DEBUG 1
16 #if defined(_MSC_VER)
17 #pragma warning(disable:4214) // bit field types other than int
18 #pragma warning(disable:4201) // nameless struct/union
19 #endif
20 
21 #define MAXIMUM_AHCI_PORT_COUNT             32
22 #define MAXIMUM_AHCI_PRDT_ENTRIES           32
23 #define MAXIMUM_AHCI_PORT_NCS               30
24 #define MAXIMUM_QUEUE_BUFFER_SIZE           255
25 #define MAXIMUM_TRANSFER_LENGTH             (128*1024) // 128 KB
26 
27 #define DEVICE_ATA_BLOCK_SIZE               512
28 
29 // device type (DeviceParams)
30 #define AHCI_DEVICE_TYPE_ATA                1
31 #define AHCI_DEVICE_TYPE_ATAPI              2
32 #define AHCI_DEVICE_TYPE_NODEVICE           3
33 
34 // section 3.1.2
35 #define AHCI_Global_HBA_CAP_S64A            (1 << 31)
36 
37 // FIS Types : http://wiki.osdev.org/AHCI
38 #define FIS_TYPE_REG_H2D        0x27 // Register FIS - host to device
39 #define FIS_TYPE_REG_D2H        0x34 // Register FIS - device to host
40 #define FIS_TYPE_DMA_ACT        0x39 // DMA activate FIS - device to host
41 #define FIS_TYPE_DMA_SETUP      0x41 // DMA setup FIS - bidirectional
42 #define FIS_TYPE_BIST           0x58 // BIST activate FIS - bidirectional
43 #define FIS_TYPE_PIO_SETUP      0x5F // PIO setup FIS - device to host
44 #define FIS_TYPE_DEV_BITS       0xA1 // Set device bits FIS - device to host
45 
46 #define AHCI_ATA_CFIS_FisType               0
47 #define AHCI_ATA_CFIS_PMPort_C              1
48 #define AHCI_ATA_CFIS_CommandReg            2
49 #define AHCI_ATA_CFIS_FeaturesLow           3
50 #define AHCI_ATA_CFIS_LBA0                  4
51 #define AHCI_ATA_CFIS_LBA1                  5
52 #define AHCI_ATA_CFIS_LBA2                  6
53 #define AHCI_ATA_CFIS_Device                7
54 #define AHCI_ATA_CFIS_LBA3                  8
55 #define AHCI_ATA_CFIS_LBA4                  9
56 #define AHCI_ATA_CFIS_LBA5                  10
57 #define AHCI_ATA_CFIS_FeaturesHigh          11
58 #define AHCI_ATA_CFIS_SectorCountLow        12
59 #define AHCI_ATA_CFIS_SectorCountHigh       13
60 
61 // ATA Functions
62 #define ATA_FUNCTION_ATA_COMMAND            0x100
63 #define ATA_FUNCTION_ATA_IDENTIFY           0x101
64 #define ATA_FUNCTION_ATA_READ               0x102
65 
66 // ATAPI Functions
67 #define ATA_FUNCTION_ATAPI_COMMAND          0x200
68 
69 // ATA Flags
70 #define ATA_FLAGS_DATA_IN                   (1 << 1)
71 #define ATA_FLAGS_DATA_OUT                  (1 << 2)
72 #define ATA_FLAGS_48BIT_COMMAND             (1 << 3)
73 #define ATA_FLAGS_USE_DMA                   (1 << 4)
74 
75 #define IsAtaCommand(AtaFunction)           (AtaFunction & ATA_FUNCTION_ATA_COMMAND)
76 #define IsAtapiCommand(AtaFunction)         (AtaFunction & ATA_FUNCTION_ATAPI_COMMAND)
77 #define IsDataTransferNeeded(SrbExtension)  (SrbExtension->Flags & (ATA_FLAGS_DATA_IN | ATA_FLAGS_DATA_OUT))
78 #define IsAdapterCAPS64(CAP)                (CAP & AHCI_Global_HBA_CAP_S64A)
79 
80 // 3.1.1 NCS = CAP[12:08] -> Align
81 #define AHCI_Global_Port_CAP_NCS(x)         (((x) & 0xF00) >> 8)
82 
83 #define ROUND_UP(N, S) ((((N) + (S) - 1) / (S)) * (S))
84 //#define AhciDebugPrint(format, ...) StorPortDebugPrint(0, format, __VA_ARGS__)
85 #define AhciDebugPrint(format, ...) DbgPrint("(%s:%d) " format, __RELFILE__, __LINE__, ##__VA_ARGS__)
86 
87 typedef
88 VOID
89 (*PAHCI_COMPLETION_ROUTINE) (
90     __in PVOID PortExtension,
91     __in PVOID Srb
92     );
93 
94 //////////////////////////////////////////////////////////////
95 //              ---- Support Structures ---                 //
96 //////////////////////////////////////////////////////////////
97 
98 // section 3.3.5
99 typedef union _AHCI_INTERRUPT_STATUS
100 {
101     struct
102     {
103         ULONG DHRS:1;       //Device to Host Register FIS Interrupt
104         ULONG PSS :1;       //PIO Setup FIS Interrupt
105         ULONG DSS :1;       //DMA Setup FIS Interrupt
106         ULONG SDBS :1;      //Set Device Bits Interrupt
107         ULONG UFS :1;       //Unknown FIS Interrupt
108         ULONG DPS :1;       //Descriptor Processed
109         ULONG PCS :1;       //Port Connect Change Status
110         ULONG DMPS :1;      //Device Mechanical Presence Status (DMPS)
111         ULONG Reserved :14;
112         ULONG PRCS :1;      //PhyRdy Change Status
113         ULONG IPMS :1;      //Incorrect Port Multiplier Status
114         ULONG OFS :1;       //Overflow Status
115         ULONG Reserved2 :1;
116         ULONG INFS :1;      //Interface Non-fatal Error Status
117         ULONG IFS :1;       //Interface Fatal Error Status
118         ULONG HBDS :1;      //Host Bus Data Error Status
119         ULONG HBFS :1;      //Host Bus Fatal Error Status
120         ULONG TFES :1;      //Task File Error Status
121         ULONG CPDS :1;      //Cold Port Detect Status
122     };
123 
124     ULONG Status;
125 } AHCI_INTERRUPT_STATUS;
126 
127 typedef struct _AHCI_FIS_DMA_SETUP
128 {
129     ULONG ULONG0_1;         // FIS_TYPE_DMA_SETUP
130                             // Port multiplier
131                             // Reserved
132                             // Data transfer direction, 1 - device to host
133                             // Interrupt bit
134                             // Auto-activate. Specifies if DMA Activate FIS is needed
135     UCHAR Reserved[2];      // Reserved
136     ULONG DmaBufferLow;     // DMA Buffer Identifier. Used to Identify DMA buffer in host memory. SATA Spec says host specific and not in Spec. Trying AHCI spec might work.
137     ULONG DmaBufferHigh;
138     ULONG Reserved2;        // More reserved
139     ULONG DmaBufferOffset;  // Byte offset into buffer. First 2 bits must be 0
140     ULONG TranferCount;     // Number of bytes to transfer. Bit 0 must be 0
141     ULONG Reserved3;        // Reserved
142 } AHCI_FIS_DMA_SETUP;
143 
144 typedef struct _AHCI_PIO_SETUP_FIS
145 {
146     UCHAR FisType;
147     UCHAR Reserved1 :5;
148     UCHAR D :1;
149     UCHAR I :1;
150     UCHAR Reserved2 :1;
151     UCHAR Status;
152     UCHAR Error;
153 
154     UCHAR SectorNumber;
155     UCHAR CylLow;
156     UCHAR CylHigh;
157     UCHAR Dev_Head;
158 
159     UCHAR SectorNumb_Exp;
160     UCHAR CylLow_Exp;
161     UCHAR CylHigh_Exp;
162     UCHAR Reserved3;
163 
164     UCHAR SectorCount;
165     UCHAR SectorCount_Exp;
166     UCHAR Reserved4;
167     UCHAR E_Status;
168 
169     USHORT TransferCount;
170     UCHAR Reserved5[2];
171 } AHCI_PIO_SETUP_FIS;
172 
173 typedef struct _AHCI_D2H_REGISTER_FIS
174 {
175     UCHAR FisType;
176     UCHAR Reserved1 :6;
177     UCHAR I:1;
178     UCHAR Reserved2 :1;
179     UCHAR Status;
180     UCHAR Error;
181 
182     UCHAR SectorNumber;
183     UCHAR CylLow;
184     UCHAR CylHigh;
185     UCHAR Dev_Head;
186 
187     UCHAR SectorNum_Exp;
188     UCHAR CylLow_Exp;
189     UCHAR CylHigh_Exp;
190     UCHAR Reserved;
191 
192     UCHAR SectorCount;
193     UCHAR SectorCount_Exp;
194     UCHAR Reserved3[2];
195 
196     UCHAR Reserved4[4];
197 } AHCI_D2H_REGISTER_FIS;
198 
199 typedef struct _AHCI_SET_DEVICE_BITS_FIS
200 {
201     UCHAR FisType;
202 
203     UCHAR PMPort: 4;
204     UCHAR Reserved1 :2;
205     UCHAR I :1;
206     UCHAR N :1;
207 
208     UCHAR Status_Lo :3;
209     UCHAR Reserved2 :1;
210     UCHAR Status_Hi :3;
211     UCHAR Reserved3 :1;
212 
213     UCHAR Error;
214 
215     UCHAR Reserved5[4];
216 } AHCI_SET_DEVICE_BITS_FIS;
217 
218 typedef struct _AHCI_QUEUE
219 {
220     PVOID Buffer[MAXIMUM_QUEUE_BUFFER_SIZE];  // because Storahci hold Srb queue of 255 size
221     ULONG Head;
222     ULONG Tail;
223 } AHCI_QUEUE, *PAHCI_QUEUE;
224 
225 //////////////////////////////////////////////////////////////
226 //              ---------------------------                 //
227 //////////////////////////////////////////////////////////////
228 
229 typedef union _AHCI_COMMAND_HEADER_DESCRIPTION
230 {
231     struct
232     {
233         ULONG CFL : 5;       // Command FIS Length
234         ULONG A : 1;         // IsATAPI
235         ULONG W : 1;         // Write
236         ULONG P : 1;         // Prefetchable
237 
238         ULONG R : 1;         // Reset
239         ULONG B : 1;         // BIST
240         ULONG C : 1;         //Clear Busy upon R_OK
241         ULONG RSV : 1;
242         ULONG PMP : 4;       //Port Multiplier Port
243 
244         ULONG PRDTL : 16;    //Physical Region Descriptor Table Length
245     };
246 
247     ULONG Status;
248 } AHCI_COMMAND_HEADER_DESCRIPTION;
249 
250 typedef union _AHCI_GHC
251 {
252     struct
253     {
254         ULONG HR : 1;
255         ULONG IE : 1;
256         ULONG MRSM : 1;
257         ULONG RSV0 : 28;
258         ULONG AE : 1;
259     };
260 
261     ULONG Status;
262 } AHCI_GHC;
263 
264 // section 3.3.7
265 typedef union _AHCI_PORT_CMD
266 {
267     struct
268     {
269         ULONG ST : 1;
270         ULONG SUD : 1;
271         ULONG POD : 1;
272         ULONG CLO : 1;
273         ULONG FRE : 1;
274         ULONG RSV0 : 3;
275         ULONG CCS : 5;
276         ULONG MPSS : 1;
277         ULONG FR : 1;
278         ULONG CR : 1;
279         ULONG CPS : 1;
280         ULONG PMA : 1;
281         ULONG HPCP : 1;
282         ULONG MPSP : 1;
283         ULONG CPD : 1;
284         ULONG ESP : 1;
285         ULONG FBSCP : 1;
286         ULONG APSTE : 1;
287         ULONG ATAPI : 1;
288         ULONG DLAE : 1;
289         ULONG ALPE : 1;
290         ULONG ASP : 1;
291         ULONG ICC : 4;
292     };
293 
294     ULONG Status;
295 } AHCI_PORT_CMD;
296 
297 typedef union _AHCI_SERIAL_ATA_CONTROL
298 {
299     struct
300     {
301         ULONG DET :4;
302         ULONG SPD :4;
303         ULONG IPM :4;
304         ULONG SPM :4;
305         ULONG PMP :4;
306         ULONG DW11_Reserved :12;
307     };
308 
309     ULONG Status;
310 }  AHCI_SERIAL_ATA_CONTROL;
311 
312 typedef union _AHCI_SERIAL_ATA_STATUS
313 {
314     struct
315     {
316         ULONG DET :4;
317         ULONG SPD :4;
318         ULONG IPM :4;
319         ULONG RSV0 :20;
320     };
321 
322     ULONG Status;
323 }  AHCI_SERIAL_ATA_STATUS;
324 
325 typedef union _AHCI_TASK_FILE_DATA
326 {
327     struct
328     {
329         struct _STS
330         {
331             UCHAR ERR : 1;
332             UCHAR CS1 : 2;
333             UCHAR DRQ : 1;
334             UCHAR CS2 : 3;
335             UCHAR BSY : 1;
336         } STS;
337         UCHAR ERR;
338         USHORT RSV;
339     };
340 
341     ULONG Status;
342 } AHCI_TASK_FILE_DATA;
343 
344 typedef struct _AHCI_PRDT
345 {
346     ULONG DBA;
347     ULONG DBAU;
348     ULONG RSV0;
349 
350     ULONG DBC : 22;
351     ULONG RSV1 : 9;
352     ULONG I : 1;
353 } AHCI_PRDT, *PAHCI_PRDT;
354 
355 // 4.2.3 Command Table
356 typedef struct _AHCI_COMMAND_TABLE
357 {
358     // (16 * 32) + 64 + 16 + 48 = 648
359     // 128 byte aligned :D
360     UCHAR CFIS[64];
361     UCHAR ACMD[16];
362     UCHAR RSV0[48];
363     AHCI_PRDT PRDT[MAXIMUM_AHCI_PRDT_ENTRIES];
364 } AHCI_COMMAND_TABLE, *PAHCI_COMMAND_TABLE;
365 
366 // 4.2.2 Command Header
367 typedef struct _AHCI_COMMAND_HEADER
368 {
369     AHCI_COMMAND_HEADER_DESCRIPTION DI;   // DW 0
370     ULONG PRDBC;                          // DW 1
371     ULONG CTBA;                           // DW 2
372     ULONG CTBA_U;                         // DW 3
373     ULONG Reserved[4];                    // DW 4-7
374 } AHCI_COMMAND_HEADER, *PAHCI_COMMAND_HEADER;
375 
376 // Received FIS
377 typedef struct _AHCI_RECEIVED_FIS
378 {
379     struct _AHCI_FIS_DMA_SETUP          DmaSetupFIS;      // 0x00 -- DMA Setup FIS
380     ULONG                               pad0;             // 4 BYTE padding
381     struct _AHCI_PIO_SETUP_FIS          PioSetupFIS;      // 0x20 -- PIO Setup FIS
382     ULONG                               pad1[3];          // 12 BYTE padding
383     struct _AHCI_D2H_REGISTER_FIS       RegisterFIS;      // 0x40 -- Register – Device to Host FIS
384     ULONG                               pad2;             // 4 BYTE padding
385     struct _AHCI_SET_DEVICE_BITS_FIS    SetDeviceFIS;     // 0x58 -- Set Device Bit FIS
386     ULONG                               UnknowFIS[16];    // 0x60 -- Unknown FIS
387     ULONG                               Reserved[24];     // 0xA0 -- Reserved
388 } AHCI_RECEIVED_FIS, *PAHCI_RECEIVED_FIS;
389 
390 // Holds Port Information
391 typedef struct _AHCI_PORT
392 {
393     ULONG   CLB;                                // 0x00, command list base address, 1K-byte aligned
394     ULONG   CLBU;                               // 0x04, command list base address upper 32 bits
395     ULONG   FB;                                 // 0x08, FIS base address, 256-byte aligned
396     ULONG   FBU;                                // 0x0C, FIS base address upper 32 bits
397     ULONG   IS;                                 // 0x10, interrupt status
398     ULONG   IE;                                 // 0x14, interrupt enable
399     ULONG   CMD;                                // 0x18, command and status
400     ULONG   RSV0;                               // 0x1C, Reserved
401     ULONG   TFD;                                // 0x20, task file data
402     ULONG   SIG;                                // 0x24, signature
403     ULONG   SSTS;                               // 0x28, SATA status (SCR0:SStatus)
404     ULONG   SCTL;                               // 0x2C, SATA control (SCR2:SControl)
405     ULONG   SERR;                               // 0x30, SATA error (SCR1:SError)
406     ULONG   SACT;                               // 0x34, SATA active (SCR3:SActive)
407     ULONG   CI;                                 // 0x38, command issue
408     ULONG   SNTF;                               // 0x3C, SATA notification (SCR4:SNotification)
409     ULONG   FBS;                                // 0x40, FIS-based switch control
410     ULONG   RSV1[11];                           // 0x44 ~ 0x6F, Reserved
411     ULONG   Vendor[4];                          // 0x70 ~ 0x7F, vendor specific
412 } AHCI_PORT, *PAHCI_PORT;
413 
414 typedef union _AHCI_INTERRUPT_ENABLE
415 {
416     struct
417     {
418         ULONG DHRE :1;
419         ULONG PSE :1;
420         ULONG DSE :1;
421         ULONG SDBE :1;
422         ULONG UFE :1;
423         ULONG DPE :1;
424         ULONG PCE :1;
425         ULONG DMPE :1;
426         ULONG DW5_Reserved :14;
427         ULONG PRCE :1;
428         ULONG IPME :1;
429         ULONG OFE :1;
430         ULONG DW5_Reserved2 :1;
431         ULONG INFE :1;
432         ULONG IFE :1;
433         ULONG HBDE :1;
434         ULONG HBFE :1;
435         ULONG TFEE :1;
436         ULONG CPDE :1;
437     };
438 
439     ULONG Status;
440 } AHCI_INTERRUPT_ENABLE;
441 
442 typedef struct _AHCI_MEMORY_REGISTERS
443 {
444     // 0x00 - 0x2B, Generic Host Control
445     ULONG CAP;                                  // 0x00, Host capability
446     ULONG GHC;                                  // 0x04, Global host control
447     ULONG IS;                                   // 0x08, Interrupt status
448     ULONG PI;                                   // 0x0C, Port implemented
449     ULONG VS;                                   // 0x10, Version
450     ULONG CCC_CTL;                              // 0x14, Command completion coalescing control
451     ULONG CCC_PTS;                              // 0x18, Command completion coalescing ports
452     ULONG EM_LOC;                               // 0x1C, Enclosure management location
453     ULONG EM_CTL;                               // 0x20, Enclosure management control
454     ULONG CAP2;                                 // 0x24, Host capabilities extended
455     ULONG BOHC;                                 // 0x28, BIOS/OS handoff control and status
456     ULONG Reserved[0x1d];                       // 0x2C - 0x9F, Reserved
457     ULONG VendorSpecific[0x18];                 // 0xA0 - 0xFF, Vendor specific registers
458     AHCI_PORT PortList[MAXIMUM_AHCI_PORT_COUNT];
459 } AHCI_MEMORY_REGISTERS, *PAHCI_MEMORY_REGISTERS;
460 
461 // Holds information for each attached attached port to a given adapter.
462 typedef struct _AHCI_PORT_EXTENSION
463 {
464     ULONG PortNumber;
465     ULONG QueueSlots;                                   // slots which we have already assigned task (Slot)
466     ULONG CommandIssuedSlots;                           // slots which has been programmed
467     ULONG MaxPortQueueDepth;
468 
469     struct
470     {
471         UCHAR RemovableDevice;
472         UCHAR Lba48BitMode;
473         UCHAR AccessType;
474         UCHAR DeviceType;
475         UCHAR IsActive;
476         LARGE_INTEGER MaxLba;
477         ULONG BytesPerLogicalSector;
478         ULONG BytesPerPhysicalSector;
479         UCHAR VendorId[41];
480         UCHAR RevisionID[9];
481         UCHAR SerialNumber[21];
482     } DeviceParams;
483 
484     STOR_DPC CommandCompletion;
485     PAHCI_PORT Port;                                    // AHCI Port Infomation
486     AHCI_QUEUE SrbQueue;                                // pending Srbs
487     AHCI_QUEUE CompletionQueue;
488     PSCSI_REQUEST_BLOCK Slot[MAXIMUM_AHCI_PORT_NCS];    // Srbs which has been alloted a port
489     PAHCI_RECEIVED_FIS ReceivedFIS;
490     PAHCI_COMMAND_HEADER CommandList;
491     STOR_DEVICE_POWER_STATE DevicePowerState;           // Device Power State
492     PIDENTIFY_DEVICE_DATA IdentifyDeviceData;
493     STOR_PHYSICAL_ADDRESS IdentifyDeviceDataPhysicalAddress;
494     struct _AHCI_ADAPTER_EXTENSION* AdapterExtension;   // Port's Adapter Information
495 } AHCI_PORT_EXTENSION, *PAHCI_PORT_EXTENSION;
496 
497 // Holds Adapter Information
498 typedef struct _AHCI_ADAPTER_EXTENSION
499 {
500     ULONG   SystemIoBusNumber;
501     ULONG   SlotNumber;
502     ULONG   AhciBaseAddress;
503     PULONG  IS;// Interrupt Status, In case of MSIM == `1`
504     ULONG   PortImplemented;// bit-mapping of ports which are implemented
505     ULONG   PortCount;
506 
507     USHORT  VendorID;
508     USHORT  DeviceID;
509     USHORT  RevisionID;
510 
511     ULONG   Version;
512     ULONG   CAP;
513     ULONG   CAP2;
514     ULONG   LastInterruptPort;
515     ULONG   CurrentCommandSlot;
516 
517     PVOID NonCachedExtension; // holds virtual address to noncached buffer allocated for Port Extension
518 
519     struct
520     {
521         // Message per port or shared port?
522         ULONG MessagePerPort : 1;
523         ULONG Removed : 1;
524         ULONG Reserved : 30; // not in use -- maintain 4 byte alignment
525     } StateFlags;
526 
527     PAHCI_MEMORY_REGISTERS ABAR_Address;
528     AHCI_PORT_EXTENSION PortExtension[MAXIMUM_AHCI_PORT_COUNT];
529 } AHCI_ADAPTER_EXTENSION, *PAHCI_ADAPTER_EXTENSION;
530 
531 typedef struct _LOCAL_SCATTER_GATHER_LIST
532 {
533     ULONG                       NumberOfElements;
534     ULONG_PTR                   Reserved;
535     STOR_SCATTER_GATHER_ELEMENT List[MAXIMUM_AHCI_PRDT_ENTRIES];
536 } LOCAL_SCATTER_GATHER_LIST, *PLOCAL_SCATTER_GATHER_LIST;
537 
538 typedef struct _AHCI_SRB_EXTENSION
539 {
540     AHCI_COMMAND_TABLE CommandTable;
541     ULONG AtaFunction;
542     ULONG Flags;
543 
544     UCHAR CommandReg;
545     UCHAR FeaturesLow;
546     UCHAR LBA0;
547     UCHAR LBA1;
548     UCHAR LBA2;
549     UCHAR Device;
550     UCHAR LBA3;
551     UCHAR LBA4;
552     UCHAR LBA5;
553     UCHAR FeaturesHigh;
554 
555     UCHAR SectorCountLow;
556     UCHAR SectorCountHigh;
557 
558     ULONG SlotIndex;
559     LOCAL_SCATTER_GATHER_LIST Sgl;
560     PLOCAL_SCATTER_GATHER_LIST pSgl;
561     PAHCI_COMPLETION_ROUTINE CompletionRoutine;
562 
563     // for alignment purpose -- 128 byte alignment
564     // do not try to access (R/W) this field
565     UCHAR Reserved[128];
566 } AHCI_SRB_EXTENSION, *PAHCI_SRB_EXTENSION;
567 
568 //////////////////////////////////////////////////////////////
569 //                       Declarations                       //
570 //////////////////////////////////////////////////////////////
571 
572 VOID
573 AhciProcessIO (
574     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
575     __in UCHAR PathId,
576     __in PSCSI_REQUEST_BLOCK Srb
577     );
578 
579 BOOLEAN
580 AhciAdapterReset (
581     __in PAHCI_ADAPTER_EXTENSION AdapterExtension
582     );
583 
584 FORCEINLINE
585 VOID
586 AhciZeroMemory (
587     __out PCHAR Buffer,
588     __in ULONG BufferSize
589     );
590 
591 FORCEINLINE
592 BOOLEAN
593 IsPortValid (
594     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
595     __in ULONG pathId
596     );
597 
598 UCHAR DeviceRequestSense (
599     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
600     __in PSCSI_REQUEST_BLOCK Srb,
601     __in PCDB Cdb
602     );
603 
604 UCHAR DeviceRequestReadWrite (
605     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
606     __in PSCSI_REQUEST_BLOCK Srb,
607     __in PCDB Cdb
608     );
609 
610 UCHAR DeviceRequestCapacity (
611     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
612     __in PSCSI_REQUEST_BLOCK Srb,
613     __in PCDB Cdb
614     );
615 
616 UCHAR
617 DeviceInquiryRequest (
618     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
619     __in PSCSI_REQUEST_BLOCK Srb,
620     __in PCDB Cdb
621     );
622 
623 UCHAR DeviceRequestComplete (
624     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
625     __in PSCSI_REQUEST_BLOCK Srb,
626     __in PCDB Cdb
627     );
628 
629 UCHAR DeviceReportLuns (
630     __in PAHCI_ADAPTER_EXTENSION AdapterExtension,
631     __in PSCSI_REQUEST_BLOCK Srb,
632     __in PCDB Cdb
633     );
634 
635 FORCEINLINE
636 BOOLEAN
637 AddQueue (
638     __inout PAHCI_QUEUE Queue,
639     __in PVOID Srb
640     );
641 
642 FORCEINLINE
643 PVOID
644 RemoveQueue (
645     __inout PAHCI_QUEUE Queue
646     );
647 
648 __inline
649 PAHCI_SRB_EXTENSION
650 GetSrbExtension(
651     __in PSCSI_REQUEST_BLOCK Srb
652     );
653 
654 FORCEINLINE
655 ULONG64
656 AhciGetLba (
657     __in PCDB Cdb,
658     __in ULONG CdbLength
659     );
660 
661 //////////////////////////////////////////////////////////////
662 //                       Assertions                         //
663 //////////////////////////////////////////////////////////////
664 
665 // I assert every silly mistake I can do while coding
666 // because god never help me debugging the code
667 // but these asserts do :')
668 
669 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, CAP)            == 0x00);
670 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, GHC)            == 0x04);
671 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, IS)             == 0x08);
672 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, PI)             == 0x0C);
673 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, VS)             == 0x10);
674 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, CCC_CTL)        == 0x14);
675 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, CCC_PTS)        == 0x18);
676 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, EM_LOC)         == 0x1C);
677 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, EM_CTL)         == 0x20);
678 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, CAP2)           == 0x24);
679 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, BOHC)           == 0x28);
680 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, Reserved)       == 0x2C);
681 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, VendorSpecific) == 0xA0);
682 C_ASSERT(FIELD_OFFSET(AHCI_MEMORY_REGISTERS, PortList)       == 0x100);
683 
684 C_ASSERT(FIELD_OFFSET(AHCI_PORT, CLB)    == 0x00);
685 C_ASSERT(FIELD_OFFSET(AHCI_PORT, CLBU)   == 0x04);
686 C_ASSERT(FIELD_OFFSET(AHCI_PORT, FB)     == 0x08);
687 C_ASSERT(FIELD_OFFSET(AHCI_PORT, FBU)    == 0x0C);
688 C_ASSERT(FIELD_OFFSET(AHCI_PORT, IS)     == 0x10);
689 C_ASSERT(FIELD_OFFSET(AHCI_PORT, IE)     == 0x14);
690 C_ASSERT(FIELD_OFFSET(AHCI_PORT, CMD)    == 0x18);
691 C_ASSERT(FIELD_OFFSET(AHCI_PORT, RSV0)   == 0x1C);
692 C_ASSERT(FIELD_OFFSET(AHCI_PORT, TFD)    == 0x20);
693 C_ASSERT(FIELD_OFFSET(AHCI_PORT, SIG)    == 0x24);
694 C_ASSERT(FIELD_OFFSET(AHCI_PORT, SSTS)   == 0x28);
695 C_ASSERT(FIELD_OFFSET(AHCI_PORT, SCTL)   == 0x2C);
696 C_ASSERT(FIELD_OFFSET(AHCI_PORT, SERR)   == 0x30);
697 C_ASSERT(FIELD_OFFSET(AHCI_PORT, SACT)   == 0x34);
698 C_ASSERT(FIELD_OFFSET(AHCI_PORT, CI)     == 0x38);
699 C_ASSERT(FIELD_OFFSET(AHCI_PORT, SNTF)   == 0x3C);
700 C_ASSERT(FIELD_OFFSET(AHCI_PORT, FBS)    == 0x40);
701 C_ASSERT(FIELD_OFFSET(AHCI_PORT, RSV1)   == 0x44);
702 C_ASSERT(FIELD_OFFSET(AHCI_PORT, Vendor) == 0x70);
703 
704 C_ASSERT((sizeof(AHCI_COMMAND_TABLE) % 128) == 0);
705 
706 C_ASSERT(sizeof(AHCI_GHC)                        == sizeof(ULONG));
707 C_ASSERT(sizeof(AHCI_PORT_CMD)                   == sizeof(ULONG));
708 C_ASSERT(sizeof(AHCI_TASK_FILE_DATA)             == sizeof(ULONG));
709 C_ASSERT(sizeof(AHCI_INTERRUPT_ENABLE)           == sizeof(ULONG));
710 C_ASSERT(sizeof(AHCI_SERIAL_ATA_STATUS)          == sizeof(ULONG));
711 C_ASSERT(sizeof(AHCI_SERIAL_ATA_CONTROL)         == sizeof(ULONG));
712 C_ASSERT(sizeof(AHCI_COMMAND_HEADER_DESCRIPTION) == sizeof(ULONG));
713 
714 C_ASSERT(FIELD_OFFSET(AHCI_COMMAND_TABLE, CFIS) == 0x00);
715 C_ASSERT(FIELD_OFFSET(AHCI_COMMAND_TABLE, ACMD) == 0x40);
716 C_ASSERT(FIELD_OFFSET(AHCI_COMMAND_TABLE, RSV0) == 0x50);
717 C_ASSERT(FIELD_OFFSET(AHCI_COMMAND_TABLE, PRDT) == 0x80);
718