VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DevAHCI.cpp@ 65812

Last change on this file since 65812 was 65648, checked in by vboxsync, 8 years ago

gcc 7: Devices: fall thru

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 233.3 KB
Line 
1/* $Id: DevAHCI.cpp 65648 2017-02-07 11:43:22Z vboxsync $ */
2/** @file
3 * DevAHCI - AHCI controller device (disk and cdrom).
4 *
5 * Implements the AHCI standard 1.1
6 */
7
8/*
9 * Copyright (C) 2006-2016 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20/** @page pg_dev_ahci AHCI - Advanced Host Controller Interface Emulation.
21 *
22 * This component implements an AHCI serial ATA controller. The device is split
23 * into two parts. The first part implements the register interface for the
24 * guest and the second one does the data transfer.
25 *
26 * The guest can access the controller in two ways. The first one is the native
27 * way implementing the registers described in the AHCI specification and is
28 * the preferred one. The second implements the I/O ports used for booting from
29 * the hard disk and for guests which don't have an AHCI SATA driver.
30 *
31 * The data is transfered using the extended media interface, asynchronously if
32 * it is supported by the driver below otherwise it weill be done synchronous.
33 * Either way a thread is used to process new requests from the guest.
34 */
35
36
37/*********************************************************************************************************************************
38* Header Files *
39*********************************************************************************************************************************/
40#define LOG_GROUP LOG_GROUP_DEV_AHCI
41#include <VBox/vmm/pdmdev.h>
42#include <VBox/vmm/pdmstorageifs.h>
43#include <VBox/vmm/pdmqueue.h>
44#include <VBox/vmm/pdmthread.h>
45#include <VBox/vmm/pdmcritsect.h>
46#include <VBox/sup.h>
47#include <VBox/scsi.h>
48#include <VBox/ata.h>
49#include <iprt/assert.h>
50#include <iprt/asm.h>
51#include <iprt/string.h>
52#include <iprt/list.h>
53#ifdef IN_RING3
54# include <iprt/param.h>
55# include <iprt/thread.h>
56# include <iprt/semaphore.h>
57# include <iprt/alloc.h>
58# include <iprt/uuid.h>
59# include <iprt/time.h>
60#endif
61#include "VBoxDD.h"
62
63#if defined(VBOX_WITH_DTRACE) \
64 && defined(IN_RING3) \
65 && !defined(VBOX_DEVICE_STRUCT_TESTCASE)
66# include "dtrace/VBoxDD.h"
67#else
68# define VBOXDD_AHCI_REQ_SUBMIT(a,b,c,d) do { } while (0)
69# define VBOXDD_AHCI_REQ_COMPLETED(a,b,c,d) do { } while (0)
70#endif
71
72/** Maximum number of ports available.
73 * Spec defines 32 but we have one allocated for command completion coalescing
74 * and another for a reserved future feature.
75 */
76#define AHCI_MAX_NR_PORTS_IMPL 30
77/** Maximum number of command slots available. */
78#define AHCI_NR_COMMAND_SLOTS 32
79
80/** The current saved state version. */
81#define AHCI_SAVED_STATE_VERSION 9
82/** The saved state version before the ATAPI emulation was removed and the generic SCSI driver was used. */
83#define AHCI_SAVED_STATE_VERSION_PRE_ATAPI_REMOVE 8
84/** The saved state version before changing the port reset logic in an incompatible way. */
85#define AHCI_SAVED_STATE_VERSION_PRE_PORT_RESET_CHANGES 7
86/** Saved state version before the per port hotplug port was added. */
87#define AHCI_SAVED_STATE_VERSION_PRE_HOTPLUG_FLAG 6
88/** Saved state version before legacy ATA emulation was dropped. */
89#define AHCI_SAVED_STATE_VERSION_IDE_EMULATION 5
90/** Saved state version before ATAPI support was added. */
91#define AHCI_SAVED_STATE_VERSION_PRE_ATAPI 3
92/** The saved state version use in VirtualBox 3.0 and earlier.
93 * This was before the config was added and ahciIOTasks was dropped. */
94#define AHCI_SAVED_STATE_VERSION_VBOX_30 2
95/* for Older ATA state Read handling */
96#define ATA_CTL_SAVED_STATE_VERSION 3
97#define ATA_CTL_SAVED_STATE_VERSION_WITHOUT_FULL_SENSE 1
98#define ATA_CTL_SAVED_STATE_VERSION_WITHOUT_EVENT_STATUS 2
99
100/** The maximum number of release log entries per device. */
101#define MAX_LOG_REL_ERRORS 1024
102
103/**
104 * Maximum number of sectors to transfer in a READ/WRITE MULTIPLE request.
105 * Set to 1 to disable multi-sector read support. According to the ATA
106 * specification this must be a power of 2 and it must fit in an 8 bit
107 * value. Thus the only valid values are 1, 2, 4, 8, 16, 32, 64 and 128.
108 */
109#define ATA_MAX_MULT_SECTORS 128
110
111/**
112 * Fastest PIO mode supported by the drive.
113 */
114#define ATA_PIO_MODE_MAX 4
115/**
116 * Fastest MDMA mode supported by the drive.
117 */
118#define ATA_MDMA_MODE_MAX 2
119/**
120 * Fastest UDMA mode supported by the drive.
121 */
122#define ATA_UDMA_MODE_MAX 6
123
124/**
125 * Length of the configurable VPD data (without termination)
126 */
127#define AHCI_SERIAL_NUMBER_LENGTH 20
128#define AHCI_FIRMWARE_REVISION_LENGTH 8
129#define AHCI_MODEL_NUMBER_LENGTH 40
130
131/** ATAPI sense info size. */
132#define ATAPI_SENSE_SIZE 64
133
134/**
135 * Command Header.
136 */
137#pragma pack(1)
138typedef struct
139{
140 /** Description Information. */
141 uint32_t u32DescInf;
142 /** Command status. */
143 uint32_t u32PRDBC;
144 /** Command Table Base Address. */
145 uint32_t u32CmdTblAddr;
146 /** Command Table Base Address - upper 32-bits. */
147 uint32_t u32CmdTblAddrUp;
148 /** Reserved */
149 uint32_t u32Reserved[4];
150} CmdHdr;
151#pragma pack()
152AssertCompileSize(CmdHdr, 32);
153
154/* Defines for the command header. */
155#define AHCI_CMDHDR_PRDTL_MASK 0xffff0000
156#define AHCI_CMDHDR_PRDTL_ENTRIES(x) ((x & AHCI_CMDHDR_PRDTL_MASK) >> 16)
157#define AHCI_CMDHDR_C RT_BIT(10)
158#define AHCI_CMDHDR_B RT_BIT(9)
159#define AHCI_CMDHDR_R RT_BIT(8)
160#define AHCI_CMDHDR_P RT_BIT(7)
161#define AHCI_CMDHDR_W RT_BIT(6)
162#define AHCI_CMDHDR_A RT_BIT(5)
163#define AHCI_CMDHDR_CFL_MASK 0x1f
164
165#define AHCI_CMDHDR_PRDT_OFFSET 0x80
166#define AHCI_CMDHDR_ACMD_OFFSET 0x40
167
168/* Defines for the command FIS. */
169/* Defines that are used in the first double word. */
170#define AHCI_CMDFIS_TYPE 0 /* The first byte. */
171# define AHCI_CMDFIS_TYPE_H2D 0x27 /* Register - Host to Device FIS. */
172# define AHCI_CMDFIS_TYPE_H2D_SIZE 20 /* Five double words. */
173# define AHCI_CMDFIS_TYPE_D2H 0x34 /* Register - Device to Host FIS. */
174# define AHCI_CMDFIS_TYPE_D2H_SIZE 20 /* Five double words. */
175# define AHCI_CMDFIS_TYPE_SETDEVBITS 0xa1 /* Set Device Bits - Device to Host FIS. */
176# define AHCI_CMDFIS_TYPE_SETDEVBITS_SIZE 8 /* Two double words. */
177# define AHCI_CMDFIS_TYPE_DMAACTD2H 0x39 /* DMA Activate - Device to Host FIS. */
178# define AHCI_CMDFIS_TYPE_DMAACTD2H_SIZE 4 /* One double word. */
179# define AHCI_CMDFIS_TYPE_DMASETUP 0x41 /* DMA Setup - Bidirectional FIS. */
180# define AHCI_CMDFIS_TYPE_DMASETUP_SIZE 28 /* Seven double words. */
181# define AHCI_CMDFIS_TYPE_PIOSETUP 0x5f /* PIO Setup - Device to Host FIS. */
182# define AHCI_CMDFIS_TYPE_PIOSETUP_SIZE 20 /* Five double words. */
183# define AHCI_CMDFIS_TYPE_DATA 0x46 /* Data - Bidirectional FIS. */
184
185#define AHCI_CMDFIS_BITS 1 /* Interrupt and Update bit. */
186#define AHCI_CMDFIS_C RT_BIT(7) /* Host to device. */
187#define AHCI_CMDFIS_I RT_BIT(6) /* Device to Host. */
188#define AHCI_CMDFIS_D RT_BIT(5)
189
190#define AHCI_CMDFIS_CMD 2
191#define AHCI_CMDFIS_FET 3
192
193#define AHCI_CMDFIS_SECTN 4
194#define AHCI_CMDFIS_CYLL 5
195#define AHCI_CMDFIS_CYLH 6
196#define AHCI_CMDFIS_HEAD 7
197
198#define AHCI_CMDFIS_SECTNEXP 8
199#define AHCI_CMDFIS_CYLLEXP 9
200#define AHCI_CMDFIS_CYLHEXP 10
201#define AHCI_CMDFIS_FETEXP 11
202
203#define AHCI_CMDFIS_SECTC 12
204#define AHCI_CMDFIS_SECTCEXP 13
205#define AHCI_CMDFIS_CTL 15
206# define AHCI_CMDFIS_CTL_SRST RT_BIT(2) /* Reset device. */
207# define AHCI_CMDFIS_CTL_NIEN RT_BIT(1) /* Assert or clear interrupt. */
208
209/* For D2H FIS */
210#define AHCI_CMDFIS_STS 2
211#define AHCI_CMDFIS_ERR 3
212
213/** Pointer to a task state. */
214typedef struct AHCIREQ *PAHCIREQ;
215
216/** Task encountered a buffer overflow. */
217#define AHCI_REQ_OVERFLOW RT_BIT_32(0)
218/** Request is a PIO data command, if this flag is not set it either is
219 * a command which does not transfer data or a DMA command based on the transfer size. */
220#define AHCI_REQ_PIO_DATA RT_BIT_32(1)
221/** The request has the SACT register set. */
222#define AHCI_REQ_CLEAR_SACT RT_BIT_32(2)
223/** Flag whether the request is queued. */
224#define AHCI_REQ_IS_QUEUED RT_BIT_32(3)
225/** Flag whether the request is stored on the stack. */
226#define AHCI_REQ_IS_ON_STACK RT_BIT_32(4)
227/** Flag whether this request transfers data from the device to the HBA or
228 * the other way around .*/
229#define AHCI_REQ_XFER_2_HOST RT_BIT_32(5)
230
231/**
232 * A task state.
233 */
234typedef struct AHCIREQ
235{
236 /** The I/O request handle from the driver below associated with this request. */
237 PDMMEDIAEXIOREQ hIoReq;
238 /** Tag of the task. */
239 uint32_t uTag;
240 /** The command Fis for this task. */
241 uint8_t cmdFis[AHCI_CMDFIS_TYPE_H2D_SIZE];
242 /** The ATAPI command data. */
243 uint8_t aATAPICmd[ATAPI_PACKET_SIZE];
244 /** Size of one sector for the ATAPI transfer. */
245 uint32_t cbATAPISector;
246 /** Physical address of the command header. - GC */
247 RTGCPHYS GCPhysCmdHdrAddr;
248 /** Physical address of the PRDT */
249 RTGCPHYS GCPhysPrdtl;
250 /** Number of entries in the PRDTL. */
251 unsigned cPrdtlEntries;
252 /** Data direction. */
253 PDMMEDIAEXIOREQTYPE enmType;
254 /** Start offset. */
255 uint64_t uOffset;
256 /** Number of bytes to transfer. */
257 size_t cbTransfer;
258 /** Flags for this task. */
259 uint32_t fFlags;
260 /** SCSI status code. */
261 uint8_t u8ScsiSts;
262 /** Flag when the buffer is mapped. */
263 bool fMapped;
264 /** Page lock when the buffer is mapped. */
265 PGMPAGEMAPLOCK PgLck;
266} AHCIREQ;
267
268/**
269 * Notifier queue item.
270 */
271typedef struct DEVPORTNOTIFIERQUEUEITEM
272{
273 /** The core part owned by the queue manager. */
274 PDMQUEUEITEMCORE Core;
275 /** The port to process. */
276 uint8_t iPort;
277} DEVPORTNOTIFIERQUEUEITEM, *PDEVPORTNOTIFIERQUEUEITEM;
278
279
280/**
281 * @implements PDMIBASE
282 * @implements PDMIMEDIAPORT
283 * @implements PDMIMEDIAEXPORT
284 */
285typedef struct AHCIPort
286{
287 /** Pointer to the device instance - HC ptr */
288 PPDMDEVINSR3 pDevInsR3;
289 /** Pointer to the device instance - R0 ptr */
290 PPDMDEVINSR0 pDevInsR0;
291 /** Pointer to the device instance - RC ptr. */
292 PPDMDEVINSRC pDevInsRC;
293
294#if HC_ARCH_BITS == 64
295 uint32_t Alignment0;
296#endif
297
298 /** Pointer to the parent AHCI structure - R3 ptr. */
299 R3PTRTYPE(struct AHCI *) pAhciR3;
300 /** Pointer to the parent AHCI structure - R0 ptr. */
301 R0PTRTYPE(struct AHCI *) pAhciR0;
302 /** Pointer to the parent AHCI structure - RC ptr. */
303 RCPTRTYPE(struct AHCI *) pAhciRC;
304
305 /** Command List Base Address. */
306 uint32_t regCLB;
307 /** Command List Base Address upper bits. */
308 uint32_t regCLBU;
309 /** FIS Base Address. */
310 uint32_t regFB;
311 /** FIS Base Address upper bits. */
312 uint32_t regFBU;
313 /** Interrupt Status. */
314 volatile uint32_t regIS;
315 /** Interrupt Enable. */
316 uint32_t regIE;
317 /** Command. */
318 uint32_t regCMD;
319 /** Task File Data. */
320 uint32_t regTFD;
321 /** Signature */
322 uint32_t regSIG;
323 /** Serial ATA Status. */
324 uint32_t regSSTS;
325 /** Serial ATA Control. */
326 uint32_t regSCTL;
327 /** Serial ATA Error. */
328 uint32_t regSERR;
329 /** Serial ATA Active. */
330 volatile uint32_t regSACT;
331 /** Command Issue. */
332 uint32_t regCI;
333
334 /** Current number of active tasks. */
335 volatile uint32_t cTasksActive;
336 /** Command List Base Address */
337 volatile RTGCPHYS GCPhysAddrClb;
338 /** FIS Base Address */
339 volatile RTGCPHYS GCPhysAddrFb;
340
341 /** Device is powered on. */
342 bool fPoweredOn;
343 /** Device has spun up. */
344 bool fSpunUp;
345 /** First D2H FIS was send. */
346 bool fFirstD2HFisSend;
347 /** Attached device is a CD/DVD drive. */
348 bool fATAPI;
349 /** Flag whether this port is in a reset state. */
350 volatile bool fPortReset;
351 /** Flag whether TRIM is supported. */
352 bool fTrimEnabled;
353 /** Flag if we are in a device reset. */
354 bool fResetDevice;
355 /** Flag whether this port is hot plug capable. */
356 bool fHotpluggable;
357 /** Flag whether the port is in redo task mode. */
358 volatile bool fRedo;
359 /** Flag whether the worker thread is sleeping. */
360 volatile bool fWrkThreadSleeping;
361
362 bool afAlignment[4];
363
364 /** Number of total sectors. */
365 uint64_t cTotalSectors;
366 /** Size of one sector. */
367 uint32_t cbSector;
368 /** Currently configured number of sectors in a multi-sector transfer. */
369 uint32_t cMultSectors;
370 /** Currently active transfer mode (MDMA/UDMA) and speed. */
371 uint8_t uATATransferMode;
372 /** ATAPI sense data. */
373 uint8_t abATAPISense[ATAPI_SENSE_SIZE];
374 /** Exponent of logical sectors in a physical sector, number of logical sectors is 2^exp. */
375 uint8_t cLogSectorsPerPhysicalExp;
376 /** The LUN. */
377 RTUINT iLUN;
378
379 /** Bitmap for finished tasks (R3 -> Guest). */
380 volatile uint32_t u32TasksFinished;
381 /** Bitmap for finished queued tasks (R3 -> Guest). */
382 volatile uint32_t u32QueuedTasksFinished;
383 /** Bitmap for new queued tasks (Guest -> R3). */
384 volatile uint32_t u32TasksNew;
385 /** Bitmap of tasks which must be redone because of a non fatal error. */
386 volatile uint32_t u32TasksRedo;
387
388 /** Current command slot processed.
389 * Accessed by the guest by reading the CMD register.
390 * Holds the command slot of the command processed at the moment. */
391 volatile uint32_t u32CurrentCommandSlot;
392
393#if HC_ARCH_BITS == 64
394 uint32_t u32Alignment2;
395#endif
396
397 /** Device specific settings (R3 only stuff). */
398 /** Pointer to the attached driver's base interface. */
399 R3PTRTYPE(PPDMIBASE) pDrvBase;
400 /** Pointer to the attached driver's block interface. */
401 R3PTRTYPE(PPDMIMEDIA) pDrvMedia;
402 /** Pointer to the attached driver's extended interface. */
403 R3PTRTYPE(PPDMIMEDIAEX) pDrvMediaEx;
404 /** The base interface. */
405 PDMIBASE IBase;
406 /** The block port interface. */
407 PDMIMEDIAPORT IPort;
408 /** The extended media port interface. */
409 PDMIMEDIAEXPORT IMediaExPort;
410 /** Physical geometry of this image. */
411 PDMMEDIAGEOMETRY PCHSGeometry;
412 /** The status LED state for this drive. */
413 PDMLED Led;
414
415#if HC_ARCH_BITS == 64
416 uint32_t u32Alignment3;
417#endif
418
419 /** Async IO Thread. */
420 R3PTRTYPE(PPDMTHREAD) pAsyncIOThread;
421 /** First task throwing an error. */
422 R3PTRTYPE(volatile PAHCIREQ) pTaskErr;
423
424 /** The event semaphore the processing thread waits on. */
425 SUPSEMEVENT hEvtProcess;
426
427 /** The serial numnber to use for IDENTIFY DEVICE commands. */
428 char szSerialNumber[AHCI_SERIAL_NUMBER_LENGTH+1]; /** < one extra byte for termination */
429 /** The firmware revision to use for IDENTIFY DEVICE commands. */
430 char szFirmwareRevision[AHCI_FIRMWARE_REVISION_LENGTH+1]; /** < one extra byte for termination */
431 /** The model number to use for IDENTIFY DEVICE commands. */
432 char szModelNumber[AHCI_MODEL_NUMBER_LENGTH+1]; /** < one extra byte for termination */
433 /** Error counter */
434 uint32_t cErrors;
435
436 uint32_t u32Alignment5;
437
438} AHCIPort;
439/** Pointer to the state of an AHCI port. */
440typedef AHCIPort *PAHCIPort;
441
442AssertCompileSizeAlignment(AHCIPort, 8);
443
444/**
445 * Main AHCI device state.
446 *
447 * @implements PDMILEDPORTS
448 */
449typedef struct AHCI
450{
451 /** The PCI device structure. */
452 PDMPCIDEV dev;
453 /** Pointer to the device instance - R3 ptr */
454 PPDMDEVINSR3 pDevInsR3;
455 /** Pointer to the device instance - R0 ptr */
456 PPDMDEVINSR0 pDevInsR0;
457 /** Pointer to the device instance - RC ptr. */
458 PPDMDEVINSRC pDevInsRC;
459
460#if HC_ARCH_BITS == 64
461 uint32_t Alignment0;
462#endif
463
464 /** Status LUN: The base interface. */
465 PDMIBASE IBase;
466 /** Status LUN: Leds interface. */
467 PDMILEDPORTS ILeds;
468 /** Status LUN: Partner of ILeds. */
469 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
470 /** Status LUN: Media Notifys. */
471 R3PTRTYPE(PPDMIMEDIANOTIFY) pMediaNotify;
472
473#if HC_ARCH_BITS == 32
474 uint32_t Alignment1;
475#endif
476
477 /** Base address of the MMIO region. */
478 RTGCPHYS MMIOBase;
479 /** Base address of the I/O port region for Idx/Data. */
480 RTIOPORT IOPortBase;
481
482 /** Global Host Control register of the HBA */
483
484 /** HBA Capabilities - Readonly */
485 uint32_t regHbaCap;
486 /** HBA Control */
487 uint32_t regHbaCtrl;
488 /** Interrupt Status */
489 uint32_t regHbaIs;
490 /** Ports Implemented - Readonly */
491 uint32_t regHbaPi;
492 /** AHCI Version - Readonly */
493 uint32_t regHbaVs;
494 /** Command completion coalescing control */
495 uint32_t regHbaCccCtl;
496 /** Command completion coalescing ports */
497 uint32_t regHbaCccPorts;
498
499 /** Index register for BIOS access. */
500 uint32_t regIdx;
501
502#if HC_ARCH_BITS == 64
503 uint32_t Alignment3;
504#endif
505
506 /** Countdown timer for command completion coalescing - R3 ptr */
507 PTMTIMERR3 pHbaCccTimerR3;
508 /** Countdown timer for command completion coalescing - R0 ptr */
509 PTMTIMERR0 pHbaCccTimerR0;
510 /** Countdown timer for command completion coalescing - RC ptr */
511 PTMTIMERRC pHbaCccTimerRC;
512
513#if HC_ARCH_BITS == 64
514 uint32_t Alignment4;
515#endif
516
517 /** Queue to send tasks to R3. - HC ptr */
518 R3PTRTYPE(PPDMQUEUE) pNotifierQueueR3;
519 /** Queue to send tasks to R3. - HC ptr */
520 R0PTRTYPE(PPDMQUEUE) pNotifierQueueR0;
521 /** Queue to send tasks to R3. - RC ptr */
522 RCPTRTYPE(PPDMQUEUE) pNotifierQueueRC;
523
524#if HC_ARCH_BITS == 64
525 uint32_t Alignment5;
526#endif
527
528
529 /** Which port number is used to mark an CCC interrupt */
530 uint8_t uCccPortNr;
531
532#if HC_ARCH_BITS == 64
533 uint32_t Alignment6;
534#endif
535
536 /** Timeout value */
537 uint64_t uCccTimeout;
538 /** Number of completions used to assert an interrupt */
539 uint32_t uCccNr;
540 /** Current number of completed commands */
541 uint32_t uCccCurrentNr;
542
543 /** Register structure per port */
544 AHCIPort ahciPort[AHCI_MAX_NR_PORTS_IMPL];
545
546 /** The critical section. */
547 PDMCRITSECT lock;
548
549 /** Bitmask of ports which asserted an interrupt. */
550 volatile uint32_t u32PortsInterrupted;
551 /** Number of I/O threads currently active - used for async controller reset handling. */
552 volatile uint32_t cThreadsActive;
553 /** Device is in a reset state. */
554 bool fReset;
555 /** Supports 64bit addressing */
556 bool f64BitAddr;
557 /** GC enabled. */
558 bool fGCEnabled;
559 /** R0 enabled. */
560 bool fR0Enabled;
561 /** Indicates that PDMDevHlpAsyncNotificationCompleted should be called when
562 * a port is entering the idle state. */
563 bool volatile fSignalIdle;
564 /** Flag whether the controller has BIOS access enabled. */
565 bool fBootable;
566 /** Flag whether the legacy port reset method should be used to make it work with saved states. */
567 bool fLegacyPortResetMethod;
568
569 /** Number of usable ports on this controller. */
570 uint32_t cPortsImpl;
571 /** Number of usable command slots for each port. */
572 uint32_t cCmdSlotsAvail;
573
574 /** Flag whether we have written the first 4bytes in an 8byte MMIO write successfully. */
575 volatile bool f8ByteMMIO4BytesWrittenSuccessfully;
576
577#if HC_ARCH_BITS == 64
578 uint32_t Alignment7;
579#endif
580
581 /** The support driver session handle. */
582 R3R0PTRTYPE(PSUPDRVSESSION) pSupDrvSession;
583} AHCI;
584/** Pointer to the state of an AHCI device. */
585typedef AHCI *PAHCI;
586
587AssertCompileMemberAlignment(AHCI, ahciPort, 8);
588
589/**
590 * Scatter gather list entry.
591 */
592typedef struct
593{
594 /** Data Base Address. */
595 uint32_t u32DBA;
596 /** Data Base Address - Upper 32-bits. */
597 uint32_t u32DBAUp;
598 /** Reserved */
599 uint32_t u32Reserved;
600 /** Description information. */
601 uint32_t u32DescInf;
602} SGLEntry;
603AssertCompileSize(SGLEntry, 16);
604
605#ifdef IN_RING3
606/**
607 * Memory buffer callback.
608 *
609 * @returns nothing.
610 * @param pThis The NVME controller instance.
611 * @param GCPhys The guest physical address of the memory buffer.
612 * @param pSgBuf The pointer to the host R3 S/G buffer.
613 * @param cbCopy How many bytes to copy between the two buffers.
614 * @param pcbSkip Initially contains the amount of bytes to skip
615 * starting from the guest physical address before
616 * accessing the S/G buffer and start copying data.
617 * On return this contains the remaining amount if
618 * cbCopy < *pcbSkip or 0 otherwise.
619 */
620typedef DECLCALLBACK(void) AHCIR3MEMCOPYCALLBACK(PAHCI pThis, RTGCPHYS GCPhys, PRTSGBUF pSgBuf, size_t cbCopy,
621 size_t *pcbSkip);
622/** Pointer to a memory copy buffer callback. */
623typedef AHCIR3MEMCOPYCALLBACK *PAHCIR3MEMCOPYCALLBACK;
624#endif
625
626/** Defines for a scatter gather list entry. */
627#define SGLENTRY_DBA_READONLY ~(RT_BIT(0))
628#define SGLENTRY_DESCINF_I RT_BIT(31)
629#define SGLENTRY_DESCINF_DBC 0x3fffff
630#define SGLENTRY_DESCINF_READONLY 0x803fffff
631
632/* Defines for the global host control registers for the HBA. */
633
634#define AHCI_HBA_GLOBAL_SIZE 0x100
635
636/* Defines for the HBA Capabilities - Readonly */
637#define AHCI_HBA_CAP_S64A RT_BIT(31)
638#define AHCI_HBA_CAP_SNCQ RT_BIT(30)
639#define AHCI_HBA_CAP_SIS RT_BIT(28)
640#define AHCI_HBA_CAP_SSS RT_BIT(27)
641#define AHCI_HBA_CAP_SALP RT_BIT(26)
642#define AHCI_HBA_CAP_SAL RT_BIT(25)
643#define AHCI_HBA_CAP_SCLO RT_BIT(24)
644#define AHCI_HBA_CAP_ISS (RT_BIT(23) | RT_BIT(22) | RT_BIT(21) | RT_BIT(20))
645# define AHCI_HBA_CAP_ISS_SHIFT(x) (((x) << 20) & AHCI_HBA_CAP_ISS)
646# define AHCI_HBA_CAP_ISS_GEN1 RT_BIT(0)
647# define AHCI_HBA_CAP_ISS_GEN2 RT_BIT(1)
648#define AHCI_HBA_CAP_SNZO RT_BIT(19)
649#define AHCI_HBA_CAP_SAM RT_BIT(18)
650#define AHCI_HBA_CAP_SPM RT_BIT(17)
651#define AHCI_HBA_CAP_PMD RT_BIT(15)
652#define AHCI_HBA_CAP_SSC RT_BIT(14)
653#define AHCI_HBA_CAP_PSC RT_BIT(13)
654#define AHCI_HBA_CAP_NCS (RT_BIT(12) | RT_BIT(11) | RT_BIT(10) | RT_BIT(9) | RT_BIT(8))
655#define AHCI_HBA_CAP_NCS_SET(x) (((x-1) << 8) & AHCI_HBA_CAP_NCS) /* 0's based */
656#define AHCI_HBA_CAP_CCCS RT_BIT(7)
657#define AHCI_HBA_CAP_NP (RT_BIT(4) | RT_BIT(3) | RT_BIT(2) | RT_BIT(1) | RT_BIT(0))
658#define AHCI_HBA_CAP_NP_SET(x) ((x-1) & AHCI_HBA_CAP_NP) /* 0's based */
659
660/* Defines for the HBA Control register - Read/Write */
661#define AHCI_HBA_CTRL_AE RT_BIT(31)
662#define AHCI_HBA_CTRL_IE RT_BIT(1)
663#define AHCI_HBA_CTRL_HR RT_BIT(0)
664#define AHCI_HBA_CTRL_RW_MASK (RT_BIT(0) | RT_BIT(1)) /* Mask for the used bits */
665
666/* Defines for the HBA Version register - Readonly (We support AHCI 1.0) */
667#define AHCI_HBA_VS_MJR (1 << 16)
668#define AHCI_HBA_VS_MNR 0x100
669
670/* Defines for the command completion coalescing control register */
671#define AHCI_HBA_CCC_CTL_TV 0xffff0000
672#define AHCI_HBA_CCC_CTL_TV_SET(x) (x << 16)
673#define AHCI_HBA_CCC_CTL_TV_GET(x) ((x & AHCI_HBA_CCC_CTL_TV) >> 16)
674
675#define AHCI_HBA_CCC_CTL_CC 0xff00
676#define AHCI_HBA_CCC_CTL_CC_SET(x) (x << 8)
677#define AHCI_HBA_CCC_CTL_CC_GET(x) ((x & AHCI_HBA_CCC_CTL_CC) >> 8)
678
679#define AHCI_HBA_CCC_CTL_INT 0xf8
680#define AHCI_HBA_CCC_CTL_INT_SET(x) (x << 3)
681#define AHCI_HBA_CCC_CTL_INT_GET(x) ((x & AHCI_HBA_CCC_CTL_INT) >> 3)
682
683#define AHCI_HBA_CCC_CTL_EN RT_BIT(0)
684
685/* Defines for the port registers. */
686
687#define AHCI_PORT_REGISTER_SIZE 0x80
688
689#define AHCI_PORT_CLB_RESERVED 0xfffffc00 /* For masking out the reserved bits. */
690
691#define AHCI_PORT_FB_RESERVED 0xffffff00 /* For masking out the reserved bits. */
692
693#define AHCI_PORT_IS_CPDS RT_BIT(31)
694#define AHCI_PORT_IS_TFES RT_BIT(30)
695#define AHCI_PORT_IS_HBFS RT_BIT(29)
696#define AHCI_PORT_IS_HBDS RT_BIT(28)
697#define AHCI_PORT_IS_IFS RT_BIT(27)
698#define AHCI_PORT_IS_INFS RT_BIT(26)
699#define AHCI_PORT_IS_OFS RT_BIT(24)
700#define AHCI_PORT_IS_IPMS RT_BIT(23)
701#define AHCI_PORT_IS_PRCS RT_BIT(22)
702#define AHCI_PORT_IS_DIS RT_BIT(7)
703#define AHCI_PORT_IS_PCS RT_BIT(6)
704#define AHCI_PORT_IS_DPS RT_BIT(5)
705#define AHCI_PORT_IS_UFS RT_BIT(4)
706#define AHCI_PORT_IS_SDBS RT_BIT(3)
707#define AHCI_PORT_IS_DSS RT_BIT(2)
708#define AHCI_PORT_IS_PSS RT_BIT(1)
709#define AHCI_PORT_IS_DHRS RT_BIT(0)
710#define AHCI_PORT_IS_READONLY 0xfd8000af /* Readonly mask including reserved bits. */
711
712#define AHCI_PORT_IE_CPDE RT_BIT(31)
713#define AHCI_PORT_IE_TFEE RT_BIT(30)
714#define AHCI_PORT_IE_HBFE RT_BIT(29)
715#define AHCI_PORT_IE_HBDE RT_BIT(28)
716#define AHCI_PORT_IE_IFE RT_BIT(27)
717#define AHCI_PORT_IE_INFE RT_BIT(26)
718#define AHCI_PORT_IE_OFE RT_BIT(24)
719#define AHCI_PORT_IE_IPME RT_BIT(23)
720#define AHCI_PORT_IE_PRCE RT_BIT(22)
721#define AHCI_PORT_IE_DIE RT_BIT(7) /* Not supported for now, readonly. */
722#define AHCI_PORT_IE_PCE RT_BIT(6)
723#define AHCI_PORT_IE_DPE RT_BIT(5)
724#define AHCI_PORT_IE_UFE RT_BIT(4)
725#define AHCI_PORT_IE_SDBE RT_BIT(3)
726#define AHCI_PORT_IE_DSE RT_BIT(2)
727#define AHCI_PORT_IE_PSE RT_BIT(1)
728#define AHCI_PORT_IE_DHRE RT_BIT(0)
729#define AHCI_PORT_IE_READONLY (0xfdc000ff) /* Readonly mask including reserved bits. */
730
731#define AHCI_PORT_CMD_ICC (RT_BIT(28) | RT_BIT(29) | RT_BIT(30) | RT_BIT(31))
732#define AHCI_PORT_CMD_ICC_SHIFT(x) ((x) << 28)
733# define AHCI_PORT_CMD_ICC_IDLE 0x0
734# define AHCI_PORT_CMD_ICC_ACTIVE 0x1
735# define AHCI_PORT_CMD_ICC_PARTIAL 0x2
736# define AHCI_PORT_CMD_ICC_SLUMBER 0x6
737#define AHCI_PORT_CMD_ASP RT_BIT(27) /* Not supported - Readonly */
738#define AHCI_PORT_CMD_ALPE RT_BIT(26) /* Not supported - Readonly */
739#define AHCI_PORT_CMD_DLAE RT_BIT(25)
740#define AHCI_PORT_CMD_ATAPI RT_BIT(24)
741#define AHCI_PORT_CMD_CPD RT_BIT(20)
742#define AHCI_PORT_CMD_ISP RT_BIT(19) /* Readonly */
743#define AHCI_PORT_CMD_HPCP RT_BIT(18)
744#define AHCI_PORT_CMD_PMA RT_BIT(17) /* Not supported - Readonly */
745#define AHCI_PORT_CMD_CPS RT_BIT(16)
746#define AHCI_PORT_CMD_CR RT_BIT(15) /* Readonly */
747#define AHCI_PORT_CMD_FR RT_BIT(14) /* Readonly */
748#define AHCI_PORT_CMD_ISS RT_BIT(13) /* Readonly */
749#define AHCI_PORT_CMD_CCS (RT_BIT(8) | RT_BIT(9) | RT_BIT(10) | RT_BIT(11) | RT_BIT(12))
750#define AHCI_PORT_CMD_CCS_SHIFT(x) (x << 8) /* Readonly */
751#define AHCI_PORT_CMD_FRE RT_BIT(4)
752#define AHCI_PORT_CMD_CLO RT_BIT(3)
753#define AHCI_PORT_CMD_POD RT_BIT(2)
754#define AHCI_PORT_CMD_SUD RT_BIT(1)
755#define AHCI_PORT_CMD_ST RT_BIT(0)
756#define AHCI_PORT_CMD_READONLY (0xff02001f & ~(AHCI_PORT_CMD_ASP | AHCI_PORT_CMD_ALPE | AHCI_PORT_CMD_PMA))
757
758#define AHCI_PORT_SCTL_IPM (RT_BIT(11) | RT_BIT(10) | RT_BIT(9) | RT_BIT(8))
759#define AHCI_PORT_SCTL_IPM_GET(x) ((x & AHCI_PORT_SCTL_IPM) >> 8)
760#define AHCI_PORT_SCTL_SPD (RT_BIT(7) | RT_BIT(6) | RT_BIT(5) | RT_BIT(4))
761#define AHCI_PORT_SCTL_SPD_GET(x) ((x & AHCI_PORT_SCTL_SPD) >> 4)
762#define AHCI_PORT_SCTL_DET (RT_BIT(3) | RT_BIT(2) | RT_BIT(1) | RT_BIT(0))
763#define AHCI_PORT_SCTL_DET_GET(x) (x & AHCI_PORT_SCTL_DET)
764#define AHCI_PORT_SCTL_DET_NINIT 0
765#define AHCI_PORT_SCTL_DET_INIT 1
766#define AHCI_PORT_SCTL_DET_OFFLINE 4
767#define AHCI_PORT_SCTL_READONLY 0xfff
768
769#define AHCI_PORT_SSTS_IPM (RT_BIT(11) | RT_BIT(10) | RT_BIT(9) | RT_BIT(8))
770#define AHCI_PORT_SSTS_IPM_GET(x) ((x & AHCI_PORT_SCTL_IPM) >> 8)
771#define AHCI_PORT_SSTS_SPD (RT_BIT(7) | RT_BIT(6) | RT_BIT(5) | RT_BIT(4))
772#define AHCI_PORT_SSTS_SPD_GET(x) ((x & AHCI_PORT_SCTL_SPD) >> 4)
773#define AHCI_PORT_SSTS_DET (RT_BIT(3) | RT_BIT(2) | RT_BIT(1) | RT_BIT(0))
774#define AHCI_PORT_SSTS_DET_GET(x) (x & AHCI_PORT_SCTL_DET)
775
776#define AHCI_PORT_TFD_BSY RT_BIT(7)
777#define AHCI_PORT_TFD_DRQ RT_BIT(3)
778#define AHCI_PORT_TFD_ERR RT_BIT(0)
779
780#define AHCI_PORT_SERR_X RT_BIT(26)
781#define AHCI_PORT_SERR_W RT_BIT(18)
782#define AHCI_PORT_SERR_N RT_BIT(16)
783
784/* Signatures for attached storage devices. */
785#define AHCI_PORT_SIG_DISK 0x00000101
786#define AHCI_PORT_SIG_ATAPI 0xeb140101
787
788/*
789 * The AHCI spec defines an area of memory where the HBA posts received FIS's from the device.
790 * regFB points to the base of this area.
791 * Every FIS type has an offset where it is posted in this area.
792 */
793#define AHCI_RECFIS_DSFIS_OFFSET 0x00 /* DMA Setup FIS */
794#define AHCI_RECFIS_PSFIS_OFFSET 0x20 /* PIO Setup FIS */
795#define AHCI_RECFIS_RFIS_OFFSET 0x40 /* D2H Register FIS */
796#define AHCI_RECFIS_SDBFIS_OFFSET 0x58 /* Set Device Bits FIS */
797#define AHCI_RECFIS_UFIS_OFFSET 0x60 /* Unknown FIS type */
798
799/** Mask to get the LBA value from a LBA range. */
800#define AHCI_RANGE_LBA_MASK UINT64_C(0xffffffffffff)
801/** Mas to get the length value from a LBA range. */
802#define AHCI_RANGE_LENGTH_MASK UINT64_C(0xffff000000000000)
803/** Returns the length of the range in sectors. */
804#define AHCI_RANGE_LENGTH_GET(val) (((val) & AHCI_RANGE_LENGTH_MASK) >> 48)
805
806/**
807 * AHCI register operator.
808 */
809typedef struct ahci_opreg
810{
811 const char *pszName;
812 int (*pfnRead )(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value);
813 int (*pfnWrite)(PAHCI pAhci, uint32_t iReg, uint32_t u32Value);
814} AHCIOPREG;
815
816/**
817 * AHCI port register operator.
818 */
819typedef struct pAhciPort_opreg
820{
821 const char *pszName;
822 int (*pfnRead )(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value);
823 int (*pfnWrite)(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value);
824} AHCIPORTOPREG;
825
826#ifndef VBOX_DEVICE_STRUCT_TESTCASE
827RT_C_DECLS_BEGIN
828#ifdef IN_RING3
829static void ahciHBAReset(PAHCI pThis);
830static int ahciPostFisIntoMemory(PAHCIPort pAhciPort, unsigned uFisType, uint8_t *cmdFis);
831static void ahciPostFirstD2HFisIntoMemory(PAHCIPort pAhciPort);
832static size_t ahciR3CopyBufferToPrdtl(PAHCI pThis, PAHCIREQ pAhciReq, const void *pvSrc,
833 size_t cbSrc, size_t cbSkip);
834static bool ahciCancelActiveTasks(PAHCIPort pAhciPort);
835#endif
836RT_C_DECLS_END
837
838#define PCIDEV_2_PAHCI(pPciDev) ( (PAHCI)(pPciDev) )
839#define PDMIBASE_2_PAHCIPORT(pInterface) ( (PAHCIPort)((uintptr_t)(pInterface) - RT_OFFSETOF(AHCIPort, IBase)) )
840#define PDMIMEDIAPORT_2_PAHCIPORT(pInterface) ( (PAHCIPort)((uintptr_t)(pInterface) - RT_OFFSETOF(AHCIPort, IPort)) )
841#define PDMIBASE_2_PAHCI(pInterface) ( (PAHCI)((uintptr_t)(pInterface) - RT_OFFSETOF(AHCI, IBase)) )
842#define PDMILEDPORTS_2_PAHCI(pInterface) ( (PAHCI)((uintptr_t)(pInterface) - RT_OFFSETOF(AHCI, ILeds)) )
843
844#define AHCI_RTGCPHYS_FROM_U32(Hi, Lo) ( (RTGCPHYS)RT_MAKE_U64(Lo, Hi) )
845
846#ifdef IN_RING3
847
848# ifdef LOG_USE_C99
849# define ahciLog(a) \
850 Log(("R3 P%u: %M", pAhciPort->iLUN, _LogRelRemoveParentheseis a))
851# else
852# define ahciLog(a) \
853 do { Log(("R3 P%u: ", pAhciPort->iLUN)); Log(a); } while(0)
854# endif
855
856#elif defined(IN_RING0)
857
858# ifdef LOG_USE_C99
859# define ahciLog(a) \
860 Log(("R0 P%u: %M", pAhciPort->iLUN, _LogRelRemoveParentheseis a))
861# else
862# define ahciLog(a) \
863 do { Log(("R0 P%u: ", pAhciPort->iLUN)); Log(a); } while(0)
864# endif
865
866#elif defined(IN_RC)
867
868# ifdef LOG_USE_C99
869# define ahciLog(a) \
870 Log(("GC P%u: %M", pAhciPort->iLUN, _LogRelRemoveParentheseis a))
871# else
872# define ahciLog(a) \
873 do { Log(("GC P%u: ", pAhciPort->iLUN)); Log(a); } while(0)
874# endif
875
876#endif
877
878/**
879 * Update PCI IRQ levels
880 */
881static void ahciHbaClearInterrupt(PAHCI pAhci)
882{
883 Log(("%s: Clearing interrupt\n", __FUNCTION__));
884 PDMDevHlpPCISetIrq(pAhci->CTX_SUFF(pDevIns), 0, 0);
885}
886
887/**
888 * Updates the IRQ level and sets port bit in the global interrupt status register of the HBA.
889 */
890static int ahciHbaSetInterrupt(PAHCI pAhci, uint8_t iPort, int rcBusy)
891{
892 Log(("P%u: %s: Setting interrupt\n", iPort, __FUNCTION__));
893
894 int rc = PDMCritSectEnter(&pAhci->lock, rcBusy);
895 if (rc != VINF_SUCCESS)
896 return rc;
897
898 if (pAhci->regHbaCtrl & AHCI_HBA_CTRL_IE)
899 {
900 if ((pAhci->regHbaCccCtl & AHCI_HBA_CCC_CTL_EN) && (pAhci->regHbaCccPorts & (1 << iPort)))
901 {
902 pAhci->uCccCurrentNr++;
903 if (pAhci->uCccCurrentNr >= pAhci->uCccNr)
904 {
905 /* Reset command completion coalescing state. */
906 TMTimerSetMillies(pAhci->CTX_SUFF(pHbaCccTimer), pAhci->uCccTimeout);
907 pAhci->uCccCurrentNr = 0;
908
909 pAhci->u32PortsInterrupted |= (1 << pAhci->uCccPortNr);
910 if (!(pAhci->u32PortsInterrupted & ~(1 << pAhci->uCccPortNr)))
911 {
912 Log(("P%u: %s: Fire interrupt\n", iPort, __FUNCTION__));
913 PDMDevHlpPCISetIrq(pAhci->CTX_SUFF(pDevIns), 0, 1);
914 }
915 }
916 }
917 else
918 {
919 /* If only the bit of the actual port is set assert an interrupt
920 * because the interrupt status register was already read by the guest
921 * and we need to send a new notification.
922 * Otherwise an interrupt is still pending.
923 */
924 ASMAtomicOrU32((volatile uint32_t *)&pAhci->u32PortsInterrupted, (1 << iPort));
925 if (!(pAhci->u32PortsInterrupted & ~(1 << iPort)))
926 {
927 Log(("P%u: %s: Fire interrupt\n", iPort, __FUNCTION__));
928 PDMDevHlpPCISetIrq(pAhci->CTX_SUFF(pDevIns), 0, 1);
929 }
930 }
931 }
932
933 PDMCritSectLeave(&pAhci->lock);
934 return VINF_SUCCESS;
935}
936
937#ifdef IN_RING3
938
939/*
940 * Assert irq when an CCC timeout occurs
941 */
942static DECLCALLBACK(void) ahciCccTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
943{
944 RT_NOREF(pDevIns, pTimer);
945 PAHCI pAhci = (PAHCI)pvUser;
946
947 int rc = ahciHbaSetInterrupt(pAhci, pAhci->uCccPortNr, VERR_IGNORED);
948 AssertRC(rc);
949}
950
951/**
952 * Finishes the port reset of the given port.
953 *
954 * @returns nothing.
955 * @param pAhciPort The port to finish the reset on.
956 */
957static void ahciPortResetFinish(PAHCIPort pAhciPort)
958{
959 /* Cancel all tasks first. */
960 bool fAllTasksCanceled = ahciCancelActiveTasks(pAhciPort);
961 Assert(fAllTasksCanceled); NOREF(fAllTasksCanceled);
962
963 /* Signature for SATA device. */
964 if (pAhciPort->fATAPI)
965 pAhciPort->regSIG = AHCI_PORT_SIG_ATAPI;
966 else
967 pAhciPort->regSIG = AHCI_PORT_SIG_DISK;
968
969 /* We received a COMINIT from the device. Tell the guest. */
970 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_PCS);
971 pAhciPort->regSERR |= AHCI_PORT_SERR_X;
972 pAhciPort->regTFD |= ATA_STAT_BUSY;
973
974 if ((pAhciPort->regCMD & AHCI_PORT_CMD_FRE) && (!pAhciPort->fFirstD2HFisSend))
975 {
976 ahciPostFirstD2HFisIntoMemory(pAhciPort);
977 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_DHRS);
978
979 if (pAhciPort->regIE & AHCI_PORT_IE_DHRE)
980 {
981 int rc = ahciHbaSetInterrupt(pAhciPort->CTX_SUFF(pAhci), pAhciPort->iLUN, VERR_IGNORED);
982 AssertRC(rc);
983 }
984 }
985
986 pAhciPort->regSSTS = (0x01 << 8) | /* Interface is active. */
987 (0x03 << 0); /* Device detected and communication established. */
988
989 /*
990 * Use the maximum allowed speed.
991 * (Not that it changes anything really)
992 */
993 switch (AHCI_PORT_SCTL_SPD_GET(pAhciPort->regSCTL))
994 {
995 case 0x01:
996 pAhciPort->regSSTS |= (0x01 << 4); /* Generation 1 (1.5GBps) speed. */
997 break;
998 case 0x02:
999 case 0x00:
1000 default:
1001 pAhciPort->regSSTS |= (0x02 << 4); /* Generation 2 (3.0GBps) speed. */
1002 break;
1003 }
1004
1005 ASMAtomicXchgBool(&pAhciPort->fPortReset, false);
1006}
1007
1008#endif /* IN_RING3 */
1009
1010/**
1011 * Kicks the I/O thread from RC or R0.
1012 *
1013 * @returns nothing.
1014 * @param pAhci The AHCI controller instance.
1015 * @param pAhciPort The port to kick.
1016 */
1017static void ahciIoThreadKick(PAHCI pAhci, PAHCIPort pAhciPort)
1018{
1019#ifdef IN_RC
1020 PDEVPORTNOTIFIERQUEUEITEM pItem = (PDEVPORTNOTIFIERQUEUEITEM)PDMQueueAlloc(pAhci->CTX_SUFF(pNotifierQueue));
1021 AssertMsg(VALID_PTR(pItem), ("Allocating item for queue failed\n"));
1022
1023 if (pItem)
1024 {
1025 pItem->iPort = pAhciPort->iLUN;
1026 PDMQueueInsert(pAhci->CTX_SUFF(pNotifierQueue), (PPDMQUEUEITEMCORE)pItem);
1027 }
1028#else
1029 LogFlowFunc(("Signal event semaphore\n"));
1030 int rc = SUPSemEventSignal(pAhci->pSupDrvSession, pAhciPort->hEvtProcess);
1031 AssertRC(rc);
1032#endif
1033}
1034
1035static int PortCmdIssue_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1036{
1037 RT_NOREF1(iReg);
1038 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1039
1040 /* Update the CI register first. */
1041 uint32_t uCIValue = ASMAtomicXchgU32(&pAhciPort->u32TasksFinished, 0);
1042 pAhciPort->regCI &= ~uCIValue;
1043
1044 if ( (pAhciPort->regCMD & AHCI_PORT_CMD_CR)
1045 && u32Value > 0)
1046 {
1047 /*
1048 * Clear all tasks which are already marked as busy. The guest
1049 * shouldn't write already busy tasks actually.
1050 */
1051 u32Value &= ~pAhciPort->regCI;
1052
1053 ASMAtomicOrU32(&pAhciPort->u32TasksNew, u32Value);
1054
1055 /* Send a notification to R3 if u32TasksNew was 0 before our write. */
1056 if (ASMAtomicReadBool(&pAhciPort->fWrkThreadSleeping))
1057 ahciIoThreadKick(pAhci, pAhciPort);
1058 }
1059
1060 pAhciPort->regCI |= u32Value;
1061
1062 return VINF_SUCCESS;
1063}
1064
1065static int PortCmdIssue_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1066{
1067 RT_NOREF2(pAhci, iReg);
1068
1069 uint32_t uCIValue = ASMAtomicXchgU32(&pAhciPort->u32TasksFinished, 0);
1070 ahciLog(("%s: read regCI=%#010x uCIValue=%#010x\n", __FUNCTION__, pAhciPort->regCI, uCIValue));
1071
1072 pAhciPort->regCI &= ~uCIValue;
1073 *pu32Value = pAhciPort->regCI;
1074
1075 return VINF_SUCCESS;
1076}
1077
1078static int PortSActive_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1079{
1080 RT_NOREF2(pAhci, iReg);
1081 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1082
1083 pAhciPort->regSACT |= u32Value;
1084
1085 return VINF_SUCCESS;
1086}
1087
1088static int PortSActive_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1089{
1090 RT_NOREF2(pAhci, iReg);
1091
1092 uint32_t u32TasksFinished = ASMAtomicXchgU32(&pAhciPort->u32QueuedTasksFinished, 0);
1093 pAhciPort->regSACT &= ~u32TasksFinished;
1094
1095 ahciLog(("%s: read regSACT=%#010x regCI=%#010x u32TasksFinished=%#010x\n",
1096 __FUNCTION__, pAhciPort->regSACT, pAhciPort->regCI, u32TasksFinished));
1097
1098 *pu32Value = pAhciPort->regSACT;
1099
1100 return VINF_SUCCESS;
1101}
1102
1103static int PortSError_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1104{
1105 RT_NOREF2(pAhci, iReg);
1106 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1107
1108 if ( (u32Value & AHCI_PORT_SERR_X)
1109 && (pAhciPort->regSERR & AHCI_PORT_SERR_X))
1110 {
1111 ASMAtomicAndU32(&pAhciPort->regIS, ~AHCI_PORT_IS_PCS);
1112 pAhciPort->regTFD |= ATA_STAT_ERR;
1113 pAhciPort->regTFD &= ~(ATA_STAT_DRQ | ATA_STAT_BUSY);
1114 }
1115
1116 if ( (u32Value & AHCI_PORT_SERR_N)
1117 && (pAhciPort->regSERR & AHCI_PORT_SERR_N))
1118 ASMAtomicAndU32(&pAhciPort->regIS, ~AHCI_PORT_IS_PRCS);
1119
1120 pAhciPort->regSERR &= ~u32Value;
1121
1122 return VINF_SUCCESS;
1123}
1124
1125static int PortSError_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1126{
1127 RT_NOREF2(pAhci, iReg);
1128 ahciLog(("%s: read regSERR=%#010x\n", __FUNCTION__, pAhciPort->regSERR));
1129 *pu32Value = pAhciPort->regSERR;
1130 return VINF_SUCCESS;
1131}
1132
1133static int PortSControl_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1134{
1135 RT_NOREF2(pAhci, iReg);
1136 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1137 ahciLog(("%s: IPM=%d SPD=%d DET=%d\n", __FUNCTION__,
1138 AHCI_PORT_SCTL_IPM_GET(u32Value), AHCI_PORT_SCTL_SPD_GET(u32Value), AHCI_PORT_SCTL_DET_GET(u32Value)));
1139
1140#ifndef IN_RING3
1141 RT_NOREF2(pAhciPort, u32Value);
1142 return VINF_IOM_R3_MMIO_WRITE;
1143#else
1144 if ((u32Value & AHCI_PORT_SCTL_DET) == AHCI_PORT_SCTL_DET_INIT)
1145 {
1146 if (!ASMAtomicXchgBool(&pAhciPort->fPortReset, true))
1147 LogRel(("AHCI#%u: Port %d reset\n", pAhci->CTX_SUFF(pDevIns)->iInstance,
1148 pAhciPort->iLUN));
1149
1150 pAhciPort->regSSTS = 0;
1151 pAhciPort->regSIG = UINT32_MAX;
1152 pAhciPort->regTFD = 0x7f;
1153 pAhciPort->fFirstD2HFisSend = false;
1154 pAhciPort->regSCTL = u32Value;
1155 }
1156 else if ( (u32Value & AHCI_PORT_SCTL_DET) == AHCI_PORT_SCTL_DET_NINIT
1157 && (pAhciPort->regSCTL & AHCI_PORT_SCTL_DET) == AHCI_PORT_SCTL_DET_INIT
1158 && pAhciPort->pDrvBase)
1159 {
1160 /* Do the port reset here, so the guest sees the new status immediately. */
1161 if (pAhci->fLegacyPortResetMethod)
1162 {
1163 ahciPortResetFinish(pAhciPort);
1164 pAhciPort->regSCTL = u32Value; /* Update after finishing the reset, so the I/O thread doesn't get a chance to do the reset. */
1165 }
1166 else
1167 {
1168 pAhciPort->regSSTS = 0x1; /* Indicate device presence detected but communication not established. */
1169 pAhciPort->regSCTL = u32Value; /* Update before kicking the I/O thread. */
1170
1171 /* Kick the thread to finish the reset. */
1172 ahciIoThreadKick(pAhci, pAhciPort);
1173 }
1174 }
1175 else /* Just update the value if there is no device attached. */
1176 pAhciPort->regSCTL = u32Value;
1177
1178 return VINF_SUCCESS;
1179#endif
1180}
1181
1182static int PortSControl_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1183{
1184 RT_NOREF2(pAhci, iReg);
1185 ahciLog(("%s: read regSCTL=%#010x\n", __FUNCTION__, pAhciPort->regSCTL));
1186 ahciLog(("%s: IPM=%d SPD=%d DET=%d\n", __FUNCTION__,
1187 AHCI_PORT_SCTL_IPM_GET(pAhciPort->regSCTL), AHCI_PORT_SCTL_SPD_GET(pAhciPort->regSCTL),
1188 AHCI_PORT_SCTL_DET_GET(pAhciPort->regSCTL)));
1189
1190 *pu32Value = pAhciPort->regSCTL;
1191 return VINF_SUCCESS;
1192}
1193
1194static int PortSStatus_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1195{
1196 RT_NOREF2(pAhci, iReg);
1197 ahciLog(("%s: read regSSTS=%#010x\n", __FUNCTION__, pAhciPort->regSSTS));
1198 ahciLog(("%s: IPM=%d SPD=%d DET=%d\n", __FUNCTION__,
1199 AHCI_PORT_SSTS_IPM_GET(pAhciPort->regSSTS), AHCI_PORT_SSTS_SPD_GET(pAhciPort->regSSTS),
1200 AHCI_PORT_SSTS_DET_GET(pAhciPort->regSSTS)));
1201
1202 *pu32Value = pAhciPort->regSSTS;
1203 return VINF_SUCCESS;
1204}
1205
1206static int PortSignature_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1207{
1208 RT_NOREF2(pAhci, iReg);
1209 ahciLog(("%s: read regSIG=%#010x\n", __FUNCTION__, pAhciPort->regSIG));
1210 *pu32Value = pAhciPort->regSIG;
1211 return VINF_SUCCESS;
1212}
1213
1214static int PortTaskFileData_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1215{
1216 RT_NOREF2(pAhci, iReg);
1217 ahciLog(("%s: read regTFD=%#010x\n", __FUNCTION__, pAhciPort->regTFD));
1218 ahciLog(("%s: ERR=%x BSY=%d DRQ=%d ERR=%d\n", __FUNCTION__,
1219 (pAhciPort->regTFD >> 8), (pAhciPort->regTFD & AHCI_PORT_TFD_BSY) >> 7,
1220 (pAhciPort->regTFD & AHCI_PORT_TFD_DRQ) >> 3, (pAhciPort->regTFD & AHCI_PORT_TFD_ERR)));
1221 *pu32Value = pAhciPort->regTFD;
1222 return VINF_SUCCESS;
1223}
1224
1225/**
1226 * Read from the port command register.
1227 */
1228static int PortCmd_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1229{
1230 RT_NOREF2(pAhci, iReg);
1231 ahciLog(("%s: read regCMD=%#010x\n", __FUNCTION__, pAhciPort->regCMD | AHCI_PORT_CMD_CCS_SHIFT(pAhciPort->u32CurrentCommandSlot)));
1232 ahciLog(("%s: ICC=%d ASP=%d ALPE=%d DLAE=%d ATAPI=%d CPD=%d ISP=%d HPCP=%d PMA=%d CPS=%d CR=%d FR=%d ISS=%d CCS=%d FRE=%d CLO=%d POD=%d SUD=%d ST=%d\n",
1233 __FUNCTION__, (pAhciPort->regCMD & AHCI_PORT_CMD_ICC) >> 28, (pAhciPort->regCMD & AHCI_PORT_CMD_ASP) >> 27,
1234 (pAhciPort->regCMD & AHCI_PORT_CMD_ALPE) >> 26, (pAhciPort->regCMD & AHCI_PORT_CMD_DLAE) >> 25,
1235 (pAhciPort->regCMD & AHCI_PORT_CMD_ATAPI) >> 24, (pAhciPort->regCMD & AHCI_PORT_CMD_CPD) >> 20,
1236 (pAhciPort->regCMD & AHCI_PORT_CMD_ISP) >> 19, (pAhciPort->regCMD & AHCI_PORT_CMD_HPCP) >> 18,
1237 (pAhciPort->regCMD & AHCI_PORT_CMD_PMA) >> 17, (pAhciPort->regCMD & AHCI_PORT_CMD_CPS) >> 16,
1238 (pAhciPort->regCMD & AHCI_PORT_CMD_CR) >> 15, (pAhciPort->regCMD & AHCI_PORT_CMD_FR) >> 14,
1239 (pAhciPort->regCMD & AHCI_PORT_CMD_ISS) >> 13, pAhciPort->u32CurrentCommandSlot,
1240 (pAhciPort->regCMD & AHCI_PORT_CMD_FRE) >> 4, (pAhciPort->regCMD & AHCI_PORT_CMD_CLO) >> 3,
1241 (pAhciPort->regCMD & AHCI_PORT_CMD_POD) >> 2, (pAhciPort->regCMD & AHCI_PORT_CMD_SUD) >> 1,
1242 (pAhciPort->regCMD & AHCI_PORT_CMD_ST)));
1243 *pu32Value = pAhciPort->regCMD | AHCI_PORT_CMD_CCS_SHIFT(pAhciPort->u32CurrentCommandSlot);
1244 return VINF_SUCCESS;
1245}
1246
1247/**
1248 * Write to the port command register.
1249 * This is the register where all the data transfer is started
1250 */
1251static int PortCmd_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1252{
1253 RT_NOREF1(iReg);
1254 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1255 ahciLog(("%s: ICC=%d ASP=%d ALPE=%d DLAE=%d ATAPI=%d CPD=%d ISP=%d HPCP=%d PMA=%d CPS=%d CR=%d FR=%d ISS=%d CCS=%d FRE=%d CLO=%d POD=%d SUD=%d ST=%d\n",
1256 __FUNCTION__, (u32Value & AHCI_PORT_CMD_ICC) >> 28, (u32Value & AHCI_PORT_CMD_ASP) >> 27,
1257 (u32Value & AHCI_PORT_CMD_ALPE) >> 26, (u32Value & AHCI_PORT_CMD_DLAE) >> 25,
1258 (u32Value & AHCI_PORT_CMD_ATAPI) >> 24, (u32Value & AHCI_PORT_CMD_CPD) >> 20,
1259 (u32Value & AHCI_PORT_CMD_ISP) >> 19, (u32Value & AHCI_PORT_CMD_HPCP) >> 18,
1260 (u32Value & AHCI_PORT_CMD_PMA) >> 17, (u32Value & AHCI_PORT_CMD_CPS) >> 16,
1261 (u32Value & AHCI_PORT_CMD_CR) >> 15, (u32Value & AHCI_PORT_CMD_FR) >> 14,
1262 (u32Value & AHCI_PORT_CMD_ISS) >> 13, (u32Value & AHCI_PORT_CMD_CCS) >> 8,
1263 (u32Value & AHCI_PORT_CMD_FRE) >> 4, (u32Value & AHCI_PORT_CMD_CLO) >> 3,
1264 (u32Value & AHCI_PORT_CMD_POD) >> 2, (u32Value & AHCI_PORT_CMD_SUD) >> 1,
1265 (u32Value & AHCI_PORT_CMD_ST)));
1266
1267 /* The PxCMD.CCS bits are R/O and maintained separately. */
1268 u32Value &= ~AHCI_PORT_CMD_CCS;
1269
1270 if (pAhciPort->fPoweredOn && pAhciPort->fSpunUp)
1271 {
1272 if (u32Value & AHCI_PORT_CMD_CLO)
1273 {
1274 ahciLog(("%s: Command list override requested\n", __FUNCTION__));
1275 u32Value &= ~(AHCI_PORT_TFD_BSY | AHCI_PORT_TFD_DRQ);
1276 /* Clear the CLO bit. */
1277 u32Value &= ~(AHCI_PORT_CMD_CLO);
1278 }
1279
1280 if (u32Value & AHCI_PORT_CMD_ST)
1281 {
1282 /*
1283 * Set engine state to running if there is a device attached and
1284 * IS.PCS is clear.
1285 */
1286 if ( pAhciPort->pDrvBase
1287 && !(pAhciPort->regIS & AHCI_PORT_IS_PCS))
1288 {
1289 ahciLog(("%s: Engine starts\n", __FUNCTION__));
1290 u32Value |= AHCI_PORT_CMD_CR;
1291
1292 /* If there is something in CI, kick the I/O thread. */
1293 if ( pAhciPort->regCI > 0
1294 && ASMAtomicReadBool(&pAhciPort->fWrkThreadSleeping))
1295 {
1296 ASMAtomicOrU32(&pAhciPort->u32TasksNew, pAhciPort->regCI);
1297#ifdef IN_RC
1298 PDEVPORTNOTIFIERQUEUEITEM pItem = (PDEVPORTNOTIFIERQUEUEITEM)PDMQueueAlloc(pAhci->CTX_SUFF(pNotifierQueue));
1299 AssertMsg(VALID_PTR(pItem), ("Allocating item for queue failed\n"));
1300
1301 pItem->iPort = pAhciPort->iLUN;
1302 PDMQueueInsert(pAhci->CTX_SUFF(pNotifierQueue), (PPDMQUEUEITEMCORE)pItem);
1303#else
1304 LogFlowFunc(("Signal event semaphore\n"));
1305 int rc = SUPSemEventSignal(pAhci->pSupDrvSession, pAhciPort->hEvtProcess);
1306 AssertRC(rc);
1307#endif
1308 }
1309 }
1310 else
1311 u32Value &= ~AHCI_PORT_CMD_CR;
1312 }
1313 else
1314 {
1315 ahciLog(("%s: Engine stops\n", __FUNCTION__));
1316 /* Clear command issue register. */
1317 pAhciPort->regCI = 0;
1318 pAhciPort->regSACT = 0;
1319 /* Clear current command slot. */
1320 pAhciPort->u32CurrentCommandSlot = 0;
1321 u32Value &= ~AHCI_PORT_CMD_CR;
1322 }
1323 }
1324 else if (pAhciPort->pDrvBase)
1325 {
1326 if ((u32Value & AHCI_PORT_CMD_POD) && (pAhciPort->regCMD & AHCI_PORT_CMD_CPS) && !pAhciPort->fPoweredOn)
1327 {
1328 ahciLog(("%s: Power on the device\n", __FUNCTION__));
1329 pAhciPort->fPoweredOn = true;
1330
1331 /*
1332 * Set states in the Port Signature and SStatus registers.
1333 */
1334 if (pAhciPort->fATAPI)
1335 pAhciPort->regSIG = AHCI_PORT_SIG_ATAPI;
1336 else
1337 pAhciPort->regSIG = AHCI_PORT_SIG_DISK;
1338 pAhciPort->regSSTS = (0x01 << 8) | /* Interface is active. */
1339 (0x02 << 4) | /* Generation 2 (3.0GBps) speed. */
1340 (0x03 << 0); /* Device detected and communication established. */
1341
1342 if (pAhciPort->regCMD & AHCI_PORT_CMD_FRE)
1343 {
1344#ifndef IN_RING3
1345 return VINF_IOM_R3_MMIO_WRITE;
1346#else
1347 ahciPostFirstD2HFisIntoMemory(pAhciPort);
1348 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_DHRS);
1349
1350 if (pAhciPort->regIE & AHCI_PORT_IE_DHRE)
1351 {
1352 int rc = ahciHbaSetInterrupt(pAhciPort->CTX_SUFF(pAhci), pAhciPort->iLUN, VERR_IGNORED);
1353 AssertRC(rc);
1354 }
1355#endif
1356 }
1357 }
1358
1359 if ((u32Value & AHCI_PORT_CMD_SUD) && pAhciPort->fPoweredOn && !pAhciPort->fSpunUp)
1360 {
1361 ahciLog(("%s: Spin up the device\n", __FUNCTION__));
1362 pAhciPort->fSpunUp = true;
1363 }
1364 }
1365
1366 if (u32Value & AHCI_PORT_CMD_FRE)
1367 {
1368 ahciLog(("%s: FIS receive enabled\n", __FUNCTION__));
1369
1370 u32Value |= AHCI_PORT_CMD_FR;
1371
1372 /* Send the first D2H FIS only if it wasn't already send. */
1373 if ( !pAhciPort->fFirstD2HFisSend
1374 && pAhciPort->pDrvBase)
1375 {
1376#ifndef IN_RING3
1377 return VINF_IOM_R3_MMIO_WRITE;
1378#else
1379 ahciPostFirstD2HFisIntoMemory(pAhciPort);
1380 pAhciPort->fFirstD2HFisSend = true;
1381#endif
1382 }
1383 }
1384 else if (!(u32Value & AHCI_PORT_CMD_FRE))
1385 {
1386 ahciLog(("%s: FIS receive disabled\n", __FUNCTION__));
1387 u32Value &= ~AHCI_PORT_CMD_FR;
1388 }
1389
1390 pAhciPort->regCMD = u32Value;
1391
1392 return VINF_SUCCESS;
1393}
1394
1395/**
1396 * Read from the port interrupt enable register.
1397 */
1398static int PortIntrEnable_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1399{
1400 RT_NOREF2(pAhci, iReg);
1401 ahciLog(("%s: read regIE=%#010x\n", __FUNCTION__, pAhciPort->regIE));
1402 ahciLog(("%s: CPDE=%d TFEE=%d HBFE=%d HBDE=%d IFE=%d INFE=%d OFE=%d IPME=%d PRCE=%d DIE=%d PCE=%d DPE=%d UFE=%d SDBE=%d DSE=%d PSE=%d DHRE=%d\n",
1403 __FUNCTION__, (pAhciPort->regIE & AHCI_PORT_IE_CPDE) >> 31, (pAhciPort->regIE & AHCI_PORT_IE_TFEE) >> 30,
1404 (pAhciPort->regIE & AHCI_PORT_IE_HBFE) >> 29, (pAhciPort->regIE & AHCI_PORT_IE_HBDE) >> 28,
1405 (pAhciPort->regIE & AHCI_PORT_IE_IFE) >> 27, (pAhciPort->regIE & AHCI_PORT_IE_INFE) >> 26,
1406 (pAhciPort->regIE & AHCI_PORT_IE_OFE) >> 24, (pAhciPort->regIE & AHCI_PORT_IE_IPME) >> 23,
1407 (pAhciPort->regIE & AHCI_PORT_IE_PRCE) >> 22, (pAhciPort->regIE & AHCI_PORT_IE_DIE) >> 7,
1408 (pAhciPort->regIE & AHCI_PORT_IE_PCE) >> 6, (pAhciPort->regIE & AHCI_PORT_IE_DPE) >> 5,
1409 (pAhciPort->regIE & AHCI_PORT_IE_UFE) >> 4, (pAhciPort->regIE & AHCI_PORT_IE_SDBE) >> 3,
1410 (pAhciPort->regIE & AHCI_PORT_IE_DSE) >> 2, (pAhciPort->regIE & AHCI_PORT_IE_PSE) >> 1,
1411 (pAhciPort->regIE & AHCI_PORT_IE_DHRE)));
1412 *pu32Value = pAhciPort->regIE;
1413 return VINF_SUCCESS;
1414}
1415
1416/**
1417 * Write to the port interrupt enable register.
1418 */
1419static int PortIntrEnable_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1420{
1421 RT_NOREF1(iReg);
1422 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1423 ahciLog(("%s: CPDE=%d TFEE=%d HBFE=%d HBDE=%d IFE=%d INFE=%d OFE=%d IPME=%d PRCE=%d DIE=%d PCE=%d DPE=%d UFE=%d SDBE=%d DSE=%d PSE=%d DHRE=%d\n",
1424 __FUNCTION__, (u32Value & AHCI_PORT_IE_CPDE) >> 31, (u32Value & AHCI_PORT_IE_TFEE) >> 30,
1425 (u32Value & AHCI_PORT_IE_HBFE) >> 29, (u32Value & AHCI_PORT_IE_HBDE) >> 28,
1426 (u32Value & AHCI_PORT_IE_IFE) >> 27, (u32Value & AHCI_PORT_IE_INFE) >> 26,
1427 (u32Value & AHCI_PORT_IE_OFE) >> 24, (u32Value & AHCI_PORT_IE_IPME) >> 23,
1428 (u32Value & AHCI_PORT_IE_PRCE) >> 22, (u32Value & AHCI_PORT_IE_DIE) >> 7,
1429 (u32Value & AHCI_PORT_IE_PCE) >> 6, (u32Value & AHCI_PORT_IE_DPE) >> 5,
1430 (u32Value & AHCI_PORT_IE_UFE) >> 4, (u32Value & AHCI_PORT_IE_SDBE) >> 3,
1431 (u32Value & AHCI_PORT_IE_DSE) >> 2, (u32Value & AHCI_PORT_IE_PSE) >> 1,
1432 (u32Value & AHCI_PORT_IE_DHRE)));
1433
1434 u32Value &= AHCI_PORT_IE_READONLY;
1435
1436 /* Check if some a interrupt status bit changed*/
1437 uint32_t u32IntrStatus = ASMAtomicReadU32(&pAhciPort->regIS);
1438
1439 int rc = VINF_SUCCESS;
1440 if (u32Value & u32IntrStatus)
1441 rc = ahciHbaSetInterrupt(pAhci, pAhciPort->iLUN, VINF_IOM_R3_MMIO_WRITE);
1442
1443 if (rc == VINF_SUCCESS)
1444 pAhciPort->regIE = u32Value;
1445
1446 return rc;
1447}
1448
1449/**
1450 * Read from the port interrupt status register.
1451 */
1452static int PortIntrSts_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1453{
1454 RT_NOREF2(pAhci, iReg);
1455 ahciLog(("%s: read regIS=%#010x\n", __FUNCTION__, pAhciPort->regIS));
1456 ahciLog(("%s: CPDS=%d TFES=%d HBFS=%d HBDS=%d IFS=%d INFS=%d OFS=%d IPMS=%d PRCS=%d DIS=%d PCS=%d DPS=%d UFS=%d SDBS=%d DSS=%d PSS=%d DHRS=%d\n",
1457 __FUNCTION__, (pAhciPort->regIS & AHCI_PORT_IS_CPDS) >> 31, (pAhciPort->regIS & AHCI_PORT_IS_TFES) >> 30,
1458 (pAhciPort->regIS & AHCI_PORT_IS_HBFS) >> 29, (pAhciPort->regIS & AHCI_PORT_IS_HBDS) >> 28,
1459 (pAhciPort->regIS & AHCI_PORT_IS_IFS) >> 27, (pAhciPort->regIS & AHCI_PORT_IS_INFS) >> 26,
1460 (pAhciPort->regIS & AHCI_PORT_IS_OFS) >> 24, (pAhciPort->regIS & AHCI_PORT_IS_IPMS) >> 23,
1461 (pAhciPort->regIS & AHCI_PORT_IS_PRCS) >> 22, (pAhciPort->regIS & AHCI_PORT_IS_DIS) >> 7,
1462 (pAhciPort->regIS & AHCI_PORT_IS_PCS) >> 6, (pAhciPort->regIS & AHCI_PORT_IS_DPS) >> 5,
1463 (pAhciPort->regIS & AHCI_PORT_IS_UFS) >> 4, (pAhciPort->regIS & AHCI_PORT_IS_SDBS) >> 3,
1464 (pAhciPort->regIS & AHCI_PORT_IS_DSS) >> 2, (pAhciPort->regIS & AHCI_PORT_IS_PSS) >> 1,
1465 (pAhciPort->regIS & AHCI_PORT_IS_DHRS)));
1466 *pu32Value = pAhciPort->regIS;
1467 return VINF_SUCCESS;
1468}
1469
1470/**
1471 * Write to the port interrupt status register.
1472 */
1473static int PortIntrSts_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1474{
1475 RT_NOREF2(pAhci, iReg);
1476 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1477 ASMAtomicAndU32(&pAhciPort->regIS, ~(u32Value & AHCI_PORT_IS_READONLY));
1478
1479 return VINF_SUCCESS;
1480}
1481
1482/**
1483 * Read from the port FIS base address upper 32bit register.
1484 */
1485static int PortFisAddrUp_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1486{
1487 RT_NOREF2(pAhci, iReg);
1488 ahciLog(("%s: read regFBU=%#010x\n", __FUNCTION__, pAhciPort->regFBU));
1489 *pu32Value = pAhciPort->regFBU;
1490 return VINF_SUCCESS;
1491}
1492
1493/**
1494 * Write to the port FIS base address upper 32bit register.
1495 */
1496static int PortFisAddrUp_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1497{
1498 RT_NOREF2(pAhci, iReg);
1499 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1500
1501 pAhciPort->regFBU = u32Value;
1502 pAhciPort->GCPhysAddrFb = AHCI_RTGCPHYS_FROM_U32(pAhciPort->regFBU, pAhciPort->regFB);
1503
1504 return VINF_SUCCESS;
1505}
1506
1507/**
1508 * Read from the port FIS base address register.
1509 */
1510static int PortFisAddr_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1511{
1512 RT_NOREF2(pAhci, iReg);
1513 ahciLog(("%s: read regFB=%#010x\n", __FUNCTION__, pAhciPort->regFB));
1514 *pu32Value = pAhciPort->regFB;
1515 return VINF_SUCCESS;
1516}
1517
1518/**
1519 * Write to the port FIS base address register.
1520 */
1521static int PortFisAddr_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1522{
1523 RT_NOREF2(pAhci, iReg);
1524 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1525
1526 Assert(!(u32Value & ~AHCI_PORT_FB_RESERVED));
1527
1528 pAhciPort->regFB = (u32Value & AHCI_PORT_FB_RESERVED);
1529 pAhciPort->GCPhysAddrFb = AHCI_RTGCPHYS_FROM_U32(pAhciPort->regFBU, pAhciPort->regFB);
1530
1531 return VINF_SUCCESS;
1532}
1533
1534/**
1535 * Write to the port command list base address upper 32bit register.
1536 */
1537static int PortCmdLstAddrUp_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1538{
1539 RT_NOREF2(pAhci, iReg);
1540 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1541
1542 pAhciPort->regCLBU = u32Value;
1543 pAhciPort->GCPhysAddrClb = AHCI_RTGCPHYS_FROM_U32(pAhciPort->regCLBU, pAhciPort->regCLB);
1544
1545 return VINF_SUCCESS;
1546}
1547
1548/**
1549 * Read from the port command list base address upper 32bit register.
1550 */
1551static int PortCmdLstAddrUp_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1552{
1553 RT_NOREF2(pAhci, iReg);
1554 ahciLog(("%s: read regCLBU=%#010x\n", __FUNCTION__, pAhciPort->regCLBU));
1555 *pu32Value = pAhciPort->regCLBU;
1556 return VINF_SUCCESS;
1557}
1558
1559/**
1560 * Read from the port command list base address register.
1561 */
1562static int PortCmdLstAddr_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1563{
1564 RT_NOREF2(pAhci, iReg);
1565 ahciLog(("%s: read regCLB=%#010x\n", __FUNCTION__, pAhciPort->regCLB));
1566 *pu32Value = pAhciPort->regCLB;
1567 return VINF_SUCCESS;
1568}
1569
1570/**
1571 * Write to the port command list base address register.
1572 */
1573static int PortCmdLstAddr_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1574{
1575 RT_NOREF2(pAhci, iReg);
1576 ahciLog(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1577
1578 Assert(!(u32Value & ~AHCI_PORT_CLB_RESERVED));
1579
1580 pAhciPort->regCLB = (u32Value & AHCI_PORT_CLB_RESERVED);
1581 pAhciPort->GCPhysAddrClb = AHCI_RTGCPHYS_FROM_U32(pAhciPort->regCLBU, pAhciPort->regCLB);
1582
1583 return VINF_SUCCESS;
1584}
1585
1586/**
1587 * Read from the global Version register.
1588 */
1589static int HbaVersion_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1590{
1591 RT_NOREF1(iReg);
1592 Log(("%s: read regHbaVs=%#010x\n", __FUNCTION__, pAhci->regHbaVs));
1593 *pu32Value = pAhci->regHbaVs;
1594 return VINF_SUCCESS;
1595}
1596
1597/**
1598 * Read from the global Ports implemented register.
1599 */
1600static int HbaPortsImplemented_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1601{
1602 RT_NOREF1(iReg);
1603 Log(("%s: read regHbaPi=%#010x\n", __FUNCTION__, pAhci->regHbaPi));
1604 *pu32Value = pAhci->regHbaPi;
1605 return VINF_SUCCESS;
1606}
1607
1608/**
1609 * Write to the global interrupt status register.
1610 */
1611static int HbaInterruptStatus_w(PAHCI pAhci, uint32_t iReg, uint32_t u32Value)
1612{
1613 RT_NOREF1(iReg);
1614 Log(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1615
1616 int rc = PDMCritSectEnter(&pAhci->lock, VINF_IOM_R3_MMIO_WRITE);
1617 if (rc != VINF_SUCCESS)
1618 return rc;
1619
1620 pAhci->regHbaIs &= ~(u32Value);
1621
1622 /*
1623 * Update interrupt status register and check for ports who
1624 * set the interrupt inbetween.
1625 */
1626 bool fClear = true;
1627 pAhci->regHbaIs |= ASMAtomicXchgU32(&pAhci->u32PortsInterrupted, 0);
1628 if (!pAhci->regHbaIs)
1629 {
1630 unsigned i = 0;
1631
1632 /* Check if the cleared ports have a interrupt status bit set. */
1633 while ((u32Value > 0) && (i < AHCI_MAX_NR_PORTS_IMPL))
1634 {
1635 if (u32Value & 0x01)
1636 {
1637 PAHCIPort pAhciPort = &pAhci->ahciPort[i];
1638
1639 if (pAhciPort->regIE & pAhciPort->regIS)
1640 {
1641 Log(("%s: Interrupt status of port %u set -> Set interrupt again\n", __FUNCTION__, i));
1642 ASMAtomicOrU32(&pAhci->u32PortsInterrupted, 1 << i);
1643 fClear = false;
1644 break;
1645 }
1646 }
1647 u32Value >>= 1;
1648 i++;
1649 }
1650 }
1651 else
1652 fClear = false;
1653
1654 if (fClear)
1655 ahciHbaClearInterrupt(pAhci);
1656 else
1657 {
1658 Log(("%s: Not clearing interrupt: u32PortsInterrupted=%#010x\n", __FUNCTION__, pAhci->u32PortsInterrupted));
1659 /*
1660 * We need to set the interrupt again because the I/O APIC does not set it again even if the
1661 * line is still high.
1662 * We need to clear it first because the PCI bus only calls the interrupt controller if the state changes.
1663 */
1664 PDMDevHlpPCISetIrq(pAhci->CTX_SUFF(pDevIns), 0, 0);
1665 PDMDevHlpPCISetIrq(pAhci->CTX_SUFF(pDevIns), 0, 1);
1666 }
1667
1668 PDMCritSectLeave(&pAhci->lock);
1669 return VINF_SUCCESS;
1670}
1671
1672/**
1673 * Read from the global interrupt status register.
1674 */
1675static int HbaInterruptStatus_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1676{
1677 RT_NOREF1(iReg);
1678
1679 int rc = PDMCritSectEnter(&pAhci->lock, VINF_IOM_R3_MMIO_READ);
1680 if (rc != VINF_SUCCESS)
1681 return rc;
1682
1683 uint32_t u32PortsInterrupted = ASMAtomicXchgU32(&pAhci->u32PortsInterrupted, 0);
1684
1685 PDMCritSectLeave(&pAhci->lock);
1686 Log(("%s: read regHbaIs=%#010x u32PortsInterrupted=%#010x\n", __FUNCTION__, pAhci->regHbaIs, u32PortsInterrupted));
1687
1688 pAhci->regHbaIs |= u32PortsInterrupted;
1689
1690#ifdef LOG_ENABLED
1691 Log(("%s:", __FUNCTION__));
1692 unsigned i;
1693 for (i = 0; i < pAhci->cPortsImpl; i++)
1694 {
1695 if ((pAhci->regHbaIs >> i) & 0x01)
1696 Log((" P%d", i));
1697 }
1698 Log(("\n"));
1699#endif
1700
1701 *pu32Value = pAhci->regHbaIs;
1702
1703 return VINF_SUCCESS;
1704}
1705
1706/**
1707 * Write to the global control register.
1708 */
1709static int HbaControl_w(PAHCI pAhci, uint32_t iReg, uint32_t u32Value)
1710{
1711 RT_NOREF1(iReg);
1712 Log(("%s: write u32Value=%#010x\n"
1713 "%s: AE=%d IE=%d HR=%d\n",
1714 __FUNCTION__, u32Value,
1715 __FUNCTION__, (u32Value & AHCI_HBA_CTRL_AE) >> 31, (u32Value & AHCI_HBA_CTRL_IE) >> 1,
1716 (u32Value & AHCI_HBA_CTRL_HR)));
1717
1718#ifndef IN_RING3
1719 RT_NOREF2(pAhci, u32Value);
1720 return VINF_IOM_R3_MMIO_WRITE;
1721#else
1722 /*
1723 * Increase the active thread counter because we might set the host controller
1724 * reset bit.
1725 */
1726 ASMAtomicIncU32(&pAhci->cThreadsActive);
1727 ASMAtomicWriteU32(&pAhci->regHbaCtrl, (u32Value & AHCI_HBA_CTRL_RW_MASK) | AHCI_HBA_CTRL_AE);
1728
1729 /*
1730 * Do the HBA reset if requested and there is no other active thread at the moment,
1731 * the work is deferred to the last active thread otherwise.
1732 */
1733 uint32_t cThreadsActive = ASMAtomicDecU32(&pAhci->cThreadsActive);
1734 if ( (u32Value & AHCI_HBA_CTRL_HR)
1735 && !cThreadsActive)
1736 ahciHBAReset(pAhci);
1737
1738 return VINF_SUCCESS;
1739#endif
1740}
1741
1742/**
1743 * Read the global control register.
1744 */
1745static int HbaControl_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1746{
1747 RT_NOREF1(iReg);
1748 Log(("%s: read regHbaCtrl=%#010x\n"
1749 "%s: AE=%d IE=%d HR=%d\n",
1750 __FUNCTION__, pAhci->regHbaCtrl,
1751 __FUNCTION__, (pAhci->regHbaCtrl & AHCI_HBA_CTRL_AE) >> 31, (pAhci->regHbaCtrl & AHCI_HBA_CTRL_IE) >> 1,
1752 (pAhci->regHbaCtrl & AHCI_HBA_CTRL_HR)));
1753 *pu32Value = pAhci->regHbaCtrl;
1754 return VINF_SUCCESS;
1755}
1756
1757/**
1758 * Read the global capabilities register.
1759 */
1760static int HbaCapabilities_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1761{
1762 RT_NOREF1(iReg);
1763 Log(("%s: read regHbaCap=%#010x\n"
1764 "%s: S64A=%d SNCQ=%d SIS=%d SSS=%d SALP=%d SAL=%d SCLO=%d ISS=%d SNZO=%d SAM=%d SPM=%d PMD=%d SSC=%d PSC=%d NCS=%d NP=%d\n",
1765 __FUNCTION__, pAhci->regHbaCap,
1766 __FUNCTION__, (pAhci->regHbaCap & AHCI_HBA_CAP_S64A) >> 31, (pAhci->regHbaCap & AHCI_HBA_CAP_SNCQ) >> 30,
1767 (pAhci->regHbaCap & AHCI_HBA_CAP_SIS) >> 28, (pAhci->regHbaCap & AHCI_HBA_CAP_SSS) >> 27,
1768 (pAhci->regHbaCap & AHCI_HBA_CAP_SALP) >> 26, (pAhci->regHbaCap & AHCI_HBA_CAP_SAL) >> 25,
1769 (pAhci->regHbaCap & AHCI_HBA_CAP_SCLO) >> 24, (pAhci->regHbaCap & AHCI_HBA_CAP_ISS) >> 20,
1770 (pAhci->regHbaCap & AHCI_HBA_CAP_SNZO) >> 19, (pAhci->regHbaCap & AHCI_HBA_CAP_SAM) >> 18,
1771 (pAhci->regHbaCap & AHCI_HBA_CAP_SPM) >> 17, (pAhci->regHbaCap & AHCI_HBA_CAP_PMD) >> 15,
1772 (pAhci->regHbaCap & AHCI_HBA_CAP_SSC) >> 14, (pAhci->regHbaCap & AHCI_HBA_CAP_PSC) >> 13,
1773 (pAhci->regHbaCap & AHCI_HBA_CAP_NCS) >> 8, (pAhci->regHbaCap & AHCI_HBA_CAP_NP)));
1774 *pu32Value = pAhci->regHbaCap;
1775 return VINF_SUCCESS;
1776}
1777
1778/**
1779 * Write to the global command completion coalescing control register.
1780 */
1781static int HbaCccCtl_w(PAHCI pAhci, uint32_t iReg, uint32_t u32Value)
1782{
1783 RT_NOREF1(iReg);
1784 Log(("%s: write u32Value=%#010x\n"
1785 "%s: TV=%d CC=%d INT=%d EN=%d\n",
1786 __FUNCTION__, u32Value,
1787 __FUNCTION__, AHCI_HBA_CCC_CTL_TV_GET(u32Value), AHCI_HBA_CCC_CTL_CC_GET(u32Value),
1788 AHCI_HBA_CCC_CTL_INT_GET(u32Value), (u32Value & AHCI_HBA_CCC_CTL_EN)));
1789
1790 pAhci->regHbaCccCtl = u32Value;
1791 pAhci->uCccTimeout = AHCI_HBA_CCC_CTL_TV_GET(u32Value);
1792 pAhci->uCccPortNr = AHCI_HBA_CCC_CTL_INT_GET(u32Value);
1793 pAhci->uCccNr = AHCI_HBA_CCC_CTL_CC_GET(u32Value);
1794
1795 if (u32Value & AHCI_HBA_CCC_CTL_EN)
1796 TMTimerSetMillies(pAhci->CTX_SUFF(pHbaCccTimer), pAhci->uCccTimeout); /* Arm the timer */
1797 else
1798 TMTimerStop(pAhci->CTX_SUFF(pHbaCccTimer));
1799
1800 return VINF_SUCCESS;
1801}
1802
1803/**
1804 * Read the global command completion coalescing control register.
1805 */
1806static int HbaCccCtl_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1807{
1808 RT_NOREF1(iReg);
1809 Log(("%s: read regHbaCccCtl=%#010x\n"
1810 "%s: TV=%d CC=%d INT=%d EN=%d\n",
1811 __FUNCTION__, pAhci->regHbaCccCtl,
1812 __FUNCTION__, AHCI_HBA_CCC_CTL_TV_GET(pAhci->regHbaCccCtl), AHCI_HBA_CCC_CTL_CC_GET(pAhci->regHbaCccCtl),
1813 AHCI_HBA_CCC_CTL_INT_GET(pAhci->regHbaCccCtl), (pAhci->regHbaCccCtl & AHCI_HBA_CCC_CTL_EN)));
1814 *pu32Value = pAhci->regHbaCccCtl;
1815 return VINF_SUCCESS;
1816}
1817
1818/**
1819 * Write to the global command completion coalescing ports register.
1820 */
1821static int HbaCccPorts_w(PAHCI pAhci, uint32_t iReg, uint32_t u32Value)
1822{
1823 RT_NOREF1(iReg);
1824 Log(("%s: write u32Value=%#010x\n", __FUNCTION__, u32Value));
1825
1826 pAhci->regHbaCccPorts = u32Value;
1827
1828 return VINF_SUCCESS;
1829}
1830
1831/**
1832 * Read the global command completion coalescing ports register.
1833 */
1834static int HbaCccPorts_r(PAHCI pAhci, uint32_t iReg, uint32_t *pu32Value)
1835{
1836 RT_NOREF1(iReg);
1837 Log(("%s: read regHbaCccPorts=%#010x\n", __FUNCTION__, pAhci->regHbaCccPorts));
1838
1839#ifdef LOG_ENABLED
1840 Log(("%s:", __FUNCTION__));
1841 unsigned i;
1842 for (i = 0; i < pAhci->cPortsImpl; i++)
1843 {
1844 if ((pAhci->regHbaCccPorts >> i) & 0x01)
1845 Log((" P%d", i));
1846 }
1847 Log(("\n"));
1848#endif
1849
1850 *pu32Value = pAhci->regHbaCccPorts;
1851 return VINF_SUCCESS;
1852}
1853
1854/**
1855 * Invalid write to global register
1856 */
1857static int HbaInvalid_w(PAHCI pAhci, uint32_t iReg, uint32_t u32Value)
1858{
1859 RT_NOREF3(pAhci, iReg, u32Value);
1860 Log(("%s: Write denied!!! iReg=%u u32Value=%#010x\n", __FUNCTION__, iReg, u32Value));
1861 return VINF_SUCCESS;
1862}
1863
1864/**
1865 * Invalid Port write.
1866 */
1867static int PortInvalid_w(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t u32Value)
1868{
1869 RT_NOREF4(pAhci, pAhciPort, iReg, u32Value);
1870 ahciLog(("%s: Write denied!!! iReg=%u u32Value=%#010x\n", __FUNCTION__, iReg, u32Value));
1871 return VINF_SUCCESS;
1872}
1873
1874/**
1875 * Invalid Port read.
1876 */
1877static int PortInvalid_r(PAHCI pAhci, PAHCIPort pAhciPort, uint32_t iReg, uint32_t *pu32Value)
1878{
1879 RT_NOREF4(pAhci, pAhciPort, iReg, pu32Value);
1880 ahciLog(("%s: Read denied!!! iReg=%u\n", __FUNCTION__, iReg));
1881 return VINF_SUCCESS;
1882}
1883
1884/**
1885 * Register descriptor table for global HBA registers
1886 */
1887static const AHCIOPREG g_aOpRegs[] =
1888{
1889 {"HbaCapabilites", HbaCapabilities_r, HbaInvalid_w}, /* Readonly */
1890 {"HbaControl" , HbaControl_r, HbaControl_w},
1891 {"HbaInterruptStatus", HbaInterruptStatus_r, HbaInterruptStatus_w},
1892 {"HbaPortsImplemented", HbaPortsImplemented_r, HbaInvalid_w}, /* Readonly */
1893 {"HbaVersion", HbaVersion_r, HbaInvalid_w}, /* ReadOnly */
1894 {"HbaCccCtl", HbaCccCtl_r, HbaCccCtl_w},
1895 {"HbaCccPorts", HbaCccPorts_r, HbaCccPorts_w},
1896};
1897
1898/**
1899 * Register descriptor table for port registers
1900 */
1901static const AHCIPORTOPREG g_aPortOpRegs[] =
1902{
1903 {"PortCmdLstAddr", PortCmdLstAddr_r, PortCmdLstAddr_w},
1904 {"PortCmdLstAddrUp", PortCmdLstAddrUp_r, PortCmdLstAddrUp_w},
1905 {"PortFisAddr", PortFisAddr_r, PortFisAddr_w},
1906 {"PortFisAddrUp", PortFisAddrUp_r, PortFisAddrUp_w},
1907 {"PortIntrSts", PortIntrSts_r, PortIntrSts_w},
1908 {"PortIntrEnable", PortIntrEnable_r, PortIntrEnable_w},
1909 {"PortCmd", PortCmd_r, PortCmd_w},
1910 {"PortReserved1", PortInvalid_r, PortInvalid_w}, /* Not used. */
1911 {"PortTaskFileData", PortTaskFileData_r, PortInvalid_w}, /* Readonly */
1912 {"PortSignature", PortSignature_r, PortInvalid_w}, /* Readonly */
1913 {"PortSStatus", PortSStatus_r, PortInvalid_w}, /* Readonly */
1914 {"PortSControl", PortSControl_r, PortSControl_w},
1915 {"PortSError", PortSError_r, PortSError_w},
1916 {"PortSActive", PortSActive_r, PortSActive_w},
1917 {"PortCmdIssue", PortCmdIssue_r, PortCmdIssue_w},
1918 {"PortReserved2", PortInvalid_r, PortInvalid_w}, /* Not used. */
1919};
1920
1921#ifdef IN_RING3
1922/**
1923 * Reset initiated by system software for one port.
1924 *
1925 * @param pAhciPort The port to reset.
1926 */
1927static void ahciPortSwReset(PAHCIPort pAhciPort)
1928{
1929 bool fAllTasksCanceled;
1930
1931 /* Cancel all tasks first. */
1932 fAllTasksCanceled = ahciCancelActiveTasks(pAhciPort);
1933 Assert(fAllTasksCanceled);
1934
1935 Assert(pAhciPort->cTasksActive == 0);
1936
1937 pAhciPort->regIS = 0;
1938 pAhciPort->regIE = 0;
1939 pAhciPort->regCMD = AHCI_PORT_CMD_CPD | /* Cold presence detection */
1940 AHCI_PORT_CMD_SUD | /* Device has spun up. */
1941 AHCI_PORT_CMD_POD; /* Port is powered on. */
1942
1943 /* Hotplugging supported?. */
1944 if (pAhciPort->fHotpluggable)
1945 pAhciPort->regCMD |= AHCI_PORT_CMD_HPCP;
1946
1947 pAhciPort->regTFD = (1 << 8) | ATA_STAT_SEEK | ATA_STAT_WRERR;
1948 pAhciPort->regSIG = UINT32_MAX;
1949 pAhciPort->regSSTS = 0;
1950 pAhciPort->regSCTL = 0;
1951 pAhciPort->regSERR = 0;
1952 pAhciPort->regSACT = 0;
1953 pAhciPort->regCI = 0;
1954
1955 pAhciPort->fResetDevice = false;
1956 pAhciPort->fPoweredOn = true;
1957 pAhciPort->fSpunUp = true;
1958 pAhciPort->cMultSectors = ATA_MAX_MULT_SECTORS;
1959 pAhciPort->uATATransferMode = ATA_MODE_UDMA | 6;
1960
1961 pAhciPort->u32TasksNew = 0;
1962 pAhciPort->u32TasksRedo = 0;
1963 pAhciPort->u32TasksFinished = 0;
1964 pAhciPort->u32QueuedTasksFinished = 0;
1965 pAhciPort->u32CurrentCommandSlot = 0;
1966
1967 if (pAhciPort->pDrvBase)
1968 {
1969 pAhciPort->regCMD |= AHCI_PORT_CMD_CPS; /* Indicate that there is a device on that port */
1970
1971 if (pAhciPort->fPoweredOn)
1972 {
1973 /*
1974 * Set states in the Port Signature and SStatus registers.
1975 */
1976 if (pAhciPort->fATAPI)
1977 pAhciPort->regSIG = AHCI_PORT_SIG_ATAPI;
1978 else
1979 pAhciPort->regSIG = AHCI_PORT_SIG_DISK;
1980 pAhciPort->regSSTS = (0x01 << 8) | /* Interface is active. */
1981 (0x02 << 4) | /* Generation 2 (3.0GBps) speed. */
1982 (0x03 << 0); /* Device detected and communication established. */
1983 }
1984 }
1985}
1986
1987/**
1988 * Hardware reset used for machine power on and reset.
1989 *
1990 * @param pAhciPort The port to reset.
1991 */
1992static void ahciPortHwReset(PAHCIPort pAhciPort)
1993{
1994 /* Reset the address registers. */
1995 pAhciPort->regCLB = 0;
1996 pAhciPort->regCLBU = 0;
1997 pAhciPort->regFB = 0;
1998 pAhciPort->regFBU = 0;
1999
2000 /* Reset calculated addresses. */
2001 pAhciPort->GCPhysAddrClb = 0;
2002 pAhciPort->GCPhysAddrFb = 0;
2003}
2004
2005/**
2006 * Create implemented ports bitmap.
2007 *
2008 * @returns 32bit bitmask with a bit set for every implemented port.
2009 * @param cPorts Number of ports.
2010 */
2011static uint32_t ahciGetPortsImplemented(unsigned cPorts)
2012{
2013 uint32_t uPortsImplemented = 0;
2014
2015 for (unsigned i = 0; i < cPorts; i++)
2016 uPortsImplemented |= (1 << i);
2017
2018 return uPortsImplemented;
2019}
2020
2021/**
2022 * Reset the entire HBA.
2023 *
2024 * @param pThis The HBA state.
2025 */
2026static void ahciHBAReset(PAHCI pThis)
2027{
2028 unsigned i;
2029 int rc = VINF_SUCCESS;
2030
2031 LogRel(("AHCI#%u: Reset the HBA\n", pThis->CTX_SUFF(pDevIns)->iInstance));
2032
2033 /* Stop the CCC timer. */
2034 if (pThis->regHbaCccCtl & AHCI_HBA_CCC_CTL_EN)
2035 {
2036 rc = TMTimerStop(pThis->CTX_SUFF(pHbaCccTimer));
2037 if (RT_FAILURE(rc))
2038 AssertMsgFailed(("%s: Failed to stop timer!\n", __FUNCTION__));
2039 }
2040
2041 /* Reset every port */
2042 for (i = 0; i < pThis->cPortsImpl; i++)
2043 {
2044 PAHCIPort pAhciPort = &pThis->ahciPort[i];
2045
2046 pAhciPort->iLUN = i;
2047 ahciPortSwReset(pAhciPort);
2048 }
2049
2050 /* Init Global registers */
2051 pThis->regHbaCap = AHCI_HBA_CAP_ISS_SHIFT(AHCI_HBA_CAP_ISS_GEN2) |
2052 AHCI_HBA_CAP_S64A | /* 64bit addressing supported */
2053 AHCI_HBA_CAP_SAM | /* AHCI mode only */
2054 AHCI_HBA_CAP_SNCQ | /* Support native command queuing */
2055 AHCI_HBA_CAP_SSS | /* Staggered spin up */
2056 AHCI_HBA_CAP_CCCS | /* Support command completion coalescing */
2057 AHCI_HBA_CAP_NCS_SET(pThis->cCmdSlotsAvail) | /* Number of command slots we support */
2058 AHCI_HBA_CAP_NP_SET(pThis->cPortsImpl); /* Number of supported ports */
2059 pThis->regHbaCtrl = AHCI_HBA_CTRL_AE;
2060 pThis->regHbaPi = ahciGetPortsImplemented(pThis->cPortsImpl);
2061 pThis->regHbaVs = AHCI_HBA_VS_MJR | AHCI_HBA_VS_MNR;
2062 pThis->regHbaCccCtl = 0;
2063 pThis->regHbaCccPorts = 0;
2064 pThis->uCccTimeout = 0;
2065 pThis->uCccPortNr = 0;
2066 pThis->uCccNr = 0;
2067
2068 /* Clear pending interrupts. */
2069 pThis->regHbaIs = 0;
2070 pThis->u32PortsInterrupted = 0;
2071 ahciHbaClearInterrupt(pThis);
2072
2073 pThis->f64BitAddr = false;
2074 pThis->u32PortsInterrupted = 0;
2075 pThis->f8ByteMMIO4BytesWrittenSuccessfully = false;
2076 /* Clear the HBA Reset bit */
2077 pThis->regHbaCtrl &= ~AHCI_HBA_CTRL_HR;
2078}
2079#endif
2080
2081/**
2082 * Reads from a AHCI controller register.
2083 *
2084 * @returns VBox status code.
2085 *
2086 * @param pAhci The AHCI instance.
2087 * @param uReg The register to write.
2088 * @param pv Where to store the result.
2089 * @param cb Number of bytes read.
2090 */
2091static int ahciRegisterRead(PAHCI pAhci, uint32_t uReg, void *pv, unsigned cb)
2092{
2093 int rc = VINF_SUCCESS;
2094 uint32_t iReg;
2095
2096 /*
2097 * If the access offset is smaller than AHCI_HBA_GLOBAL_SIZE the guest accesses the global registers.
2098 * Otherwise it accesses the registers of a port.
2099 */
2100 if (uReg < AHCI_HBA_GLOBAL_SIZE)
2101 {
2102 iReg = uReg >> 2;
2103 Log3(("%s: Trying to read from global register %u\n", __FUNCTION__, iReg));
2104 if (iReg < RT_ELEMENTS(g_aOpRegs))
2105 {
2106 const AHCIOPREG *pReg = &g_aOpRegs[iReg];
2107 rc = pReg->pfnRead(pAhci, iReg, (uint32_t *)pv);
2108 }
2109 else
2110 {
2111 Log3(("%s: Trying to read global register %u/%u!!!\n", __FUNCTION__, iReg, RT_ELEMENTS(g_aOpRegs)));
2112 *(uint32_t *)pv = 0;
2113 }
2114 }
2115 else
2116 {
2117 uint32_t iRegOffset;
2118 uint32_t iPort;
2119
2120 /* Calculate accessed port. */
2121 uReg -= AHCI_HBA_GLOBAL_SIZE;
2122 iPort = uReg / AHCI_PORT_REGISTER_SIZE;
2123 iRegOffset = (uReg % AHCI_PORT_REGISTER_SIZE);
2124 iReg = iRegOffset >> 2;
2125
2126 Log3(("%s: Trying to read from port %u and register %u\n", __FUNCTION__, iPort, iReg));
2127
2128 if (RT_LIKELY( iPort < pAhci->cPortsImpl
2129 && iReg < RT_ELEMENTS(g_aPortOpRegs)))
2130 {
2131 const AHCIPORTOPREG *pPortReg = &g_aPortOpRegs[iReg];
2132 rc = pPortReg->pfnRead(pAhci, &pAhci->ahciPort[iPort], iReg, (uint32_t *)pv);
2133 }
2134 else
2135 {
2136 Log3(("%s: Trying to read port %u register %u/%u!!!\n", __FUNCTION__, iPort, iReg, RT_ELEMENTS(g_aPortOpRegs)));
2137 rc = VINF_IOM_MMIO_UNUSED_00;
2138 }
2139
2140 /*
2141 * Windows Vista tries to read one byte from some registers instead of four.
2142 * Correct the value according to the read size.
2143 */
2144 if (RT_SUCCESS(rc) && cb != sizeof(uint32_t))
2145 {
2146 switch (cb)
2147 {
2148 case 1:
2149 {
2150 uint8_t uNewValue;
2151 uint8_t *p = (uint8_t *)pv;
2152
2153 iRegOffset &= 3;
2154 Log3(("%s: iRegOffset=%u\n", __FUNCTION__, iRegOffset));
2155 uNewValue = p[iRegOffset];
2156 /* Clear old value */
2157 *(uint32_t *)pv = 0;
2158 *(uint8_t *)pv = uNewValue;
2159 break;
2160 }
2161 default:
2162 AssertMsgFailed(("%s: unsupported access width cb=%d iPort=%x iRegOffset=%x iReg=%x!!!\n",
2163 __FUNCTION__, cb, iPort, iRegOffset, iReg));
2164 }
2165 }
2166 }
2167
2168 return rc;
2169}
2170
2171/**
2172 * Writes a value to one of the AHCI controller registers.
2173 *
2174 * @returns VBox status code.
2175 *
2176 * @param pAhci The AHCI instance.
2177 * @param offReg The offset of the register to write to.
2178 * @param u32Value The value to write.
2179 */
2180static int ahciRegisterWrite(PAHCI pAhci, uint32_t offReg, uint32_t u32Value)
2181{
2182 int rc;
2183 uint32_t iReg;
2184
2185 /*
2186 * If the access offset is smaller than 100h the guest accesses the global registers.
2187 * Otherwise it accesses the registers of a port.
2188 */
2189 if (offReg < AHCI_HBA_GLOBAL_SIZE)
2190 {
2191 Log3(("Write global HBA register\n"));
2192 iReg = offReg >> 2;
2193 if (iReg < RT_ELEMENTS(g_aOpRegs))
2194 {
2195 const AHCIOPREG *pReg = &g_aOpRegs[iReg];
2196 rc = pReg->pfnWrite(pAhci, iReg, u32Value);
2197 }
2198 else
2199 {
2200 Log3(("%s: Trying to write global register %u/%u!!!\n", __FUNCTION__, iReg, RT_ELEMENTS(g_aOpRegs)));
2201 rc = VINF_SUCCESS;
2202 }
2203 }
2204 else
2205 {
2206 uint32_t iPort;
2207 Log3(("Write Port register\n"));
2208 /* Calculate accessed port. */
2209 offReg -= AHCI_HBA_GLOBAL_SIZE;
2210 iPort = offReg / AHCI_PORT_REGISTER_SIZE;
2211 iReg = (offReg % AHCI_PORT_REGISTER_SIZE) >> 2;
2212 Log3(("%s: Trying to write to port %u and register %u\n", __FUNCTION__, iPort, iReg));
2213 if (RT_LIKELY( iPort < pAhci->cPortsImpl
2214 && iReg < RT_ELEMENTS(g_aPortOpRegs)))
2215 {
2216 const AHCIPORTOPREG *pPortReg = &g_aPortOpRegs[iReg];
2217 rc = pPortReg->pfnWrite(pAhci, &pAhci->ahciPort[iPort], iReg, u32Value);
2218 }
2219 else
2220 {
2221 Log3(("%s: Trying to write port %u register %u/%u!!!\n", __FUNCTION__, iPort, iReg, RT_ELEMENTS(g_aPortOpRegs)));
2222 rc = VINF_SUCCESS;
2223 }
2224 }
2225
2226 return rc;
2227}
2228
2229/**
2230 * Memory mapped I/O Handler for read operations.
2231 *
2232 * @returns VBox status code.
2233 *
2234 * @param pDevIns The device instance.
2235 * @param pvUser User argument.
2236 * @param GCPhysAddr Physical address (in GC) where the read starts.
2237 * @param pv Where to store the result.
2238 * @param cb Number of bytes read.
2239 */
2240PDMBOTHCBDECL(int) ahciMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
2241{
2242 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
2243 Log2(("#%d ahciMMIORead: pvUser=%p:{%.*Rhxs} cb=%d GCPhysAddr=%RGp\n", pDevIns->iInstance, pv, cb, pv, cb, GCPhysAddr));
2244 RT_NOREF1(pvUser);
2245
2246 int rc = ahciRegisterRead(pAhci, GCPhysAddr - pAhci->MMIOBase, pv, cb);
2247
2248 Log2(("#%d ahciMMIORead: return pvUser=%p:{%.*Rhxs} cb=%d GCPhysAddr=%RGp rc=%Rrc\n",
2249 pDevIns->iInstance, pv, cb, pv, cb, GCPhysAddr, rc));
2250 return rc;
2251}
2252
2253
2254/**
2255 * Memory mapped I/O Handler for write operations.
2256 *
2257 * @returns VBox status code.
2258 *
2259 * @param pDevIns The device instance.
2260 * @param pvUser User argument.
2261 * @param GCPhysAddr Physical address (in GC) where the read starts.
2262 * @param pv Where to fetch the result.
2263 * @param cb Number of bytes to write.
2264 */
2265PDMBOTHCBDECL(int) ahciMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void const *pv, unsigned cb)
2266{
2267 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
2268 Assert(cb == 4 || cb == 8);
2269 Assert(!(GCPhysAddr & (cb - 1)));
2270
2271 /* Break up 64 bits writes into two dword writes. */
2272 /** @todo Eliminate this code once the IOM/EM starts taking care of these
2273 * situations. */
2274 if (cb == 8)
2275 {
2276 /*
2277 * Only write the first 4 bytes if they weren't already.
2278 * It is possible that the last write to the register caused a world
2279 * switch and we entered this function again.
2280 * Writing the first 4 bytes again could cause indeterminate behavior
2281 * which can cause errors in the guest.
2282 */
2283 int rc = VINF_SUCCESS;
2284 if (!pAhci->f8ByteMMIO4BytesWrittenSuccessfully)
2285 {
2286 rc = ahciMMIOWrite(pDevIns, pvUser, GCPhysAddr, pv, 4);
2287 if (rc != VINF_SUCCESS)
2288 return rc;
2289
2290 pAhci->f8ByteMMIO4BytesWrittenSuccessfully = true;
2291 }
2292
2293 rc = ahciMMIOWrite(pDevIns, pvUser, GCPhysAddr + 4, (uint8_t *)pv + 4, 4);
2294 /*
2295 * Reset flag again so that the first 4 bytes are written again on the next
2296 * 8byte MMIO access.
2297 */
2298 if (rc == VINF_SUCCESS)
2299 pAhci->f8ByteMMIO4BytesWrittenSuccessfully = false;
2300
2301 return rc;
2302 }
2303
2304 /* Do the access. */
2305 Log2(("#%d ahciMMIOWrite: pvUser=%p:{%.*Rhxs} cb=%d GCPhysAddr=%RGp\n", pDevIns->iInstance, pv, cb, pv, cb, GCPhysAddr));
2306 return ahciRegisterWrite(pAhci, GCPhysAddr - pAhci->MMIOBase, *(uint32_t const *)pv);
2307}
2308
2309PDMBOTHCBDECL(int) ahciLegacyFakeWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2310{
2311 RT_NOREF5(pDevIns, pvUser, Port, u32, cb);
2312 AssertMsgFailed(("Should not happen\n"));
2313 return VINF_SUCCESS;
2314}
2315
2316PDMBOTHCBDECL(int) ahciLegacyFakeRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
2317{
2318 RT_NOREF5(pDevIns, pvUser, Port, pu32, cb);
2319 AssertMsgFailed(("Should not happen\n"));
2320 return VINF_SUCCESS;
2321}
2322
2323/**
2324 * I/O port handler for writes to the index/data register pair.
2325 *
2326 * @returns VBox status code.
2327 *
2328 * @param pDevIns The device instance.
2329 * @param pvUser User argument.
2330 * @param Port Port address where the write starts.
2331 * @param u32 Where to fetch the result.
2332 * @param cb Number of bytes to write.
2333 */
2334PDMBOTHCBDECL(int) ahciIdxDataWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2335{
2336 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
2337 int rc = VINF_SUCCESS;
2338 RT_NOREF2(pvUser, cb);
2339
2340 if (Port - pAhci->IOPortBase >= 8)
2341 {
2342 unsigned iReg = (Port - pAhci->IOPortBase - 8) / 4;
2343
2344 Assert(cb == 4);
2345
2346 if (iReg == 0)
2347 {
2348 /* Write the index register. */
2349 pAhci->regIdx = u32;
2350 }
2351 else
2352 {
2353 /** @todo range check? */
2354 Assert(iReg == 1);
2355 rc = ahciRegisterWrite(pAhci, pAhci->regIdx, u32);
2356 if (rc == VINF_IOM_R3_MMIO_WRITE)
2357 rc = VINF_IOM_R3_IOPORT_WRITE;
2358 }
2359 }
2360 /* else: ignore */
2361
2362 Log2(("#%d ahciIdxDataWrite: pu32=%p:{%.*Rhxs} cb=%d Port=%#x rc=%Rrc\n",
2363 pDevIns->iInstance, &u32, cb, &u32, cb, Port, rc));
2364 return rc;
2365}
2366
2367/**
2368 * I/O port handler for reads from the index/data register pair.
2369 *
2370 * @returns VBox status code.
2371 *
2372 * @param pDevIns The device instance.
2373 * @param pvUser User argument.
2374 * @param Port Port address where the read starts.
2375 * @param pu32 Where to fetch the result.
2376 * @param cb Number of bytes to write.
2377 */
2378PDMBOTHCBDECL(int) ahciIdxDataRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
2379{
2380 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
2381 int rc = VINF_SUCCESS;
2382 RT_NOREF1(pvUser);
2383
2384 if (Port - pAhci->IOPortBase >= 8)
2385 {
2386 unsigned iReg = (Port - pAhci->IOPortBase - 8) / 4;
2387
2388 Assert(cb == 4);
2389
2390 if (iReg == 0)
2391 {
2392 /* Read the index register. */
2393 *pu32 = pAhci->regIdx;
2394 }
2395 else
2396 {
2397 Assert(iReg == 1);
2398 /** @todo range check? */
2399 rc = ahciRegisterRead(pAhci, pAhci->regIdx, pu32, cb);
2400 if (rc == VINF_IOM_R3_MMIO_READ)
2401 rc = VINF_IOM_R3_IOPORT_READ;
2402 else if (rc == VINF_IOM_MMIO_UNUSED_00)
2403 rc = VERR_IOM_IOPORT_UNUSED;
2404 }
2405 }
2406 else
2407 *pu32 = UINT32_C(0xffffffff);
2408
2409 Log2(("#%d ahciIdxDataRead: pu32=%p:{%.*Rhxs} cb=%d Port=%#x rc=%Rrc\n",
2410 pDevIns->iInstance, pu32, cb, pu32, cb, Port, rc));
2411 return rc;
2412}
2413
2414#ifdef IN_RING3
2415
2416/**
2417 * @callback_method_impl{FNPCIIOREGIONMAP}
2418 */
2419static DECLCALLBACK(int) ahciR3MMIOMap(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion,
2420 RTGCPHYS GCPhysAddress, RTGCPHYS cb, PCIADDRESSSPACE enmType)
2421{
2422 RT_NOREF(iRegion, enmType);
2423 PAHCI pThis = PCIDEV_2_PAHCI(pPciDev);
2424
2425 Log2(("%s: registering MMIO area at GCPhysAddr=%RGp cb=%RGp\n", __FUNCTION__, GCPhysAddress, cb));
2426
2427 Assert(enmType == PCI_ADDRESS_SPACE_MEM);
2428 Assert(cb >= 4352);
2429
2430 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
2431 /** @todo change this to IOMMMIO_FLAGS_WRITE_ONLY_DWORD once EM/IOM starts
2432 * handling 2nd DWORD failures on split accesses correctly. */
2433 int rc = PDMDevHlpMMIORegister(pDevIns, GCPhysAddress, cb, NULL /*pvUser*/,
2434 IOMMMIO_FLAGS_READ_DWORD | IOMMMIO_FLAGS_WRITE_ONLY_DWORD_QWORD,
2435 ahciMMIOWrite, ahciMMIORead, "AHCI");
2436 if (RT_FAILURE(rc))
2437 return rc;
2438
2439 if (pThis->fR0Enabled)
2440 {
2441 rc = PDMDevHlpMMIORegisterR0(pDevIns, GCPhysAddress, cb, NIL_RTR0PTR /*pvUser*/, "ahciMMIOWrite", "ahciMMIORead");
2442 if (RT_FAILURE(rc))
2443 return rc;
2444 }
2445
2446 if (pThis->fGCEnabled)
2447 {
2448 rc = PDMDevHlpMMIORegisterRC(pDevIns, GCPhysAddress, cb, NIL_RTRCPTR /*pvUser*/, "ahciMMIOWrite", "ahciMMIORead");
2449 if (RT_FAILURE(rc))
2450 return rc;
2451 }
2452
2453 pThis->MMIOBase = GCPhysAddress;
2454 return rc;
2455}
2456
2457
2458/**
2459 * @callback_method_impl{FNPCIIOREGIONMAP,
2460 * Map the legacy I/O port ranges to make Solaris work with the
2461 * controller.}
2462 */
2463static DECLCALLBACK(int) ahciR3LegacyFakeIORangeMap(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion,
2464 RTGCPHYS GCPhysAddress, RTGCPHYS cb, PCIADDRESSSPACE enmType)
2465{
2466 RT_NOREF(iRegion, enmType);
2467 PAHCI pThis = PCIDEV_2_PAHCI(pPciDev);
2468 int rc = VINF_SUCCESS;
2469
2470 Log2(("%s: registering fake I/O area at GCPhysAddr=%RGp cb=%RGp\n", __FUNCTION__, GCPhysAddress, cb));
2471
2472 Assert(enmType == PCI_ADDRESS_SPACE_IO);
2473
2474 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
2475 rc = PDMDevHlpIOPortRegister(pDevIns, (RTIOPORT)GCPhysAddress, cb, NULL,
2476 ahciLegacyFakeWrite, ahciLegacyFakeRead, NULL, NULL, "AHCI Fake");
2477 if (RT_FAILURE(rc))
2478 return rc;
2479
2480 if (pThis->fR0Enabled)
2481 {
2482 rc = PDMDevHlpIOPortRegisterR0(pDevIns, (RTIOPORT)GCPhysAddress, cb, 0,
2483 "ahciLegacyFakeWrite", "ahciLegacyFakeRead", NULL, NULL, "AHCI Fake");
2484 if (RT_FAILURE(rc))
2485 return rc;
2486 }
2487
2488 if (pThis->fGCEnabled)
2489 {
2490 rc = PDMDevHlpIOPortRegisterRC(pDevIns, (RTIOPORT)GCPhysAddress, cb, 0,
2491 "ahciLegacyFakeWrite", "ahciLegacyFakeRead", NULL, NULL, "AHCI Fake");
2492 if (RT_FAILURE(rc))
2493 return rc;
2494 }
2495
2496 return rc;
2497}
2498
2499/**
2500 * @callback_method_impl{FNPCIIOREGIONMAP,
2501 * Map the BMDMA I/O port range (used for the Index/Data pair register access)}
2502 */
2503static DECLCALLBACK(int) ahciR3IdxDataIORangeMap(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion,
2504 RTGCPHYS GCPhysAddress, RTGCPHYS cb, PCIADDRESSSPACE enmType)
2505{
2506 RT_NOREF(iRegion, enmType);
2507 PAHCI pThis = PCIDEV_2_PAHCI(pPciDev);
2508 int rc = VINF_SUCCESS;
2509
2510 Log2(("%s: registering fake I/O area at GCPhysAddr=%RGp cb=%RGp\n", __FUNCTION__, GCPhysAddress, cb));
2511
2512 Assert(enmType == PCI_ADDRESS_SPACE_IO);
2513
2514 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
2515 rc = PDMDevHlpIOPortRegister(pDevIns, (RTIOPORT)GCPhysAddress, cb, NULL,
2516 ahciIdxDataWrite, ahciIdxDataRead, NULL, NULL, "AHCI IDX/DATA");
2517 if (RT_FAILURE(rc))
2518 return rc;
2519
2520 if (pThis->fR0Enabled)
2521 {
2522 rc = PDMDevHlpIOPortRegisterR0(pDevIns, (RTIOPORT)GCPhysAddress, cb, 0,
2523 "ahciIdxDataWrite", "ahciIdxDataRead", NULL, NULL, "AHCI IDX/DATA");
2524 if (RT_FAILURE(rc))
2525 return rc;
2526 }
2527
2528 if (pThis->fGCEnabled)
2529 {
2530 rc = PDMDevHlpIOPortRegisterRC(pDevIns, (RTIOPORT)GCPhysAddress, cb, 0,
2531 "ahciIdxDataWrite", "ahciIdxDataRead", NULL, NULL, "AHCI IDX/DATA");
2532 if (RT_FAILURE(rc))
2533 return rc;
2534 }
2535
2536 pThis->IOPortBase = (RTIOPORT)GCPhysAddress;
2537 return rc;
2538}
2539
2540/* -=-=-=-=-=- PAHCI::ILeds -=-=-=-=-=- */
2541
2542/**
2543 * Gets the pointer to the status LED of a unit.
2544 *
2545 * @returns VBox status code.
2546 * @param pInterface Pointer to the interface structure containing the called function pointer.
2547 * @param iLUN The unit which status LED we desire.
2548 * @param ppLed Where to store the LED pointer.
2549 */
2550static DECLCALLBACK(int) ahciR3Status_QueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
2551{
2552 PAHCI pAhci = PDMILEDPORTS_2_PAHCI(pInterface);
2553 if (iLUN < AHCI_MAX_NR_PORTS_IMPL)
2554 {
2555 *ppLed = &pAhci->ahciPort[iLUN].Led;
2556 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
2557 return VINF_SUCCESS;
2558 }
2559 return VERR_PDM_LUN_NOT_FOUND;
2560}
2561
2562/**
2563 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
2564 */
2565static DECLCALLBACK(void *) ahciR3Status_QueryInterface(PPDMIBASE pInterface, const char *pszIID)
2566{
2567 PAHCI pThis = PDMIBASE_2_PAHCI(pInterface);
2568 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
2569 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->ILeds);
2570 return NULL;
2571}
2572
2573/**
2574 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
2575 */
2576static DECLCALLBACK(void *) ahciR3PortQueryInterface(PPDMIBASE pInterface, const char *pszIID)
2577{
2578 PAHCIPort pAhciPort = PDMIBASE_2_PAHCIPORT(pInterface);
2579 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pAhciPort->IBase);
2580 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAPORT, &pAhciPort->IPort);
2581 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAEXPORT, &pAhciPort->IMediaExPort);
2582 return NULL;
2583}
2584
2585/**
2586 * @interface_method_impl{PDMIMEDIAPORT,pfnQueryDeviceLocation}
2587 */
2588static DECLCALLBACK(int) ahciR3PortQueryDeviceLocation(PPDMIMEDIAPORT pInterface, const char **ppcszController,
2589 uint32_t *piInstance, uint32_t *piLUN)
2590{
2591 PAHCIPort pAhciPort = PDMIMEDIAPORT_2_PAHCIPORT(pInterface);
2592 PPDMDEVINS pDevIns = pAhciPort->CTX_SUFF(pDevIns);
2593
2594 AssertPtrReturn(ppcszController, VERR_INVALID_POINTER);
2595 AssertPtrReturn(piInstance, VERR_INVALID_POINTER);
2596 AssertPtrReturn(piLUN, VERR_INVALID_POINTER);
2597
2598 *ppcszController = pDevIns->pReg->szName;
2599 *piInstance = pDevIns->iInstance;
2600 *piLUN = pAhciPort->iLUN;
2601
2602 return VINF_SUCCESS;
2603}
2604
2605#ifdef LOG_ENABLED
2606
2607/**
2608 * Dump info about the FIS
2609 *
2610 * @returns nothing
2611 * @param pAhciPort The port the command FIS was read from.
2612 * @param cmdFis The FIS to print info from.
2613 */
2614static void ahciDumpFisInfo(PAHCIPort pAhciPort, uint8_t *cmdFis)
2615{
2616 ahciLog(("%s: *** Begin FIS info dump. ***\n", __FUNCTION__));
2617 /* Print FIS type. */
2618 switch (cmdFis[AHCI_CMDFIS_TYPE])
2619 {
2620 case AHCI_CMDFIS_TYPE_H2D:
2621 {
2622 ahciLog(("%s: Command Fis type: H2D\n", __FUNCTION__));
2623 ahciLog(("%s: Command Fis size: %d bytes\n", __FUNCTION__, AHCI_CMDFIS_TYPE_H2D_SIZE));
2624 if (cmdFis[AHCI_CMDFIS_BITS] & AHCI_CMDFIS_C)
2625 ahciLog(("%s: Command register update\n", __FUNCTION__));
2626 else
2627 ahciLog(("%s: Control register update\n", __FUNCTION__));
2628 ahciLog(("%s: CMD=%#04x \"%s\"\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_CMD], ATACmdText(cmdFis[AHCI_CMDFIS_CMD])));
2629 ahciLog(("%s: FEAT=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_FET]));
2630 ahciLog(("%s: SECTN=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_SECTN]));
2631 ahciLog(("%s: CYLL=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_CYLL]));
2632 ahciLog(("%s: CYLH=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_CYLH]));
2633 ahciLog(("%s: HEAD=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_HEAD]));
2634
2635 ahciLog(("%s: SECTNEXP=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_SECTNEXP]));
2636 ahciLog(("%s: CYLLEXP=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_CYLLEXP]));
2637 ahciLog(("%s: CYLHEXP=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_CYLHEXP]));
2638 ahciLog(("%s: FETEXP=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_FETEXP]));
2639
2640 ahciLog(("%s: SECTC=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_SECTC]));
2641 ahciLog(("%s: SECTCEXP=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_SECTCEXP]));
2642 ahciLog(("%s: CTL=%#04x\n", __FUNCTION__, cmdFis[AHCI_CMDFIS_CTL]));
2643 if (cmdFis[AHCI_CMDFIS_CTL] & AHCI_CMDFIS_CTL_SRST)
2644 ahciLog(("%s: Reset bit is set\n", __FUNCTION__));
2645 break;
2646 }
2647 case AHCI_CMDFIS_TYPE_D2H:
2648 {
2649 ahciLog(("%s: Command Fis type D2H\n", __FUNCTION__));
2650 ahciLog(("%s: Command Fis size: %d\n", __FUNCTION__, AHCI_CMDFIS_TYPE_D2H_SIZE));
2651 break;
2652 }
2653 case AHCI_CMDFIS_TYPE_SETDEVBITS:
2654 {
2655 ahciLog(("%s: Command Fis type Set Device Bits\n", __FUNCTION__));
2656 ahciLog(("%s: Command Fis size: %d\n", __FUNCTION__, AHCI_CMDFIS_TYPE_SETDEVBITS_SIZE));
2657 break;
2658 }
2659 case AHCI_CMDFIS_TYPE_DMAACTD2H:
2660 {
2661 ahciLog(("%s: Command Fis type DMA Activate H2D\n", __FUNCTION__));
2662 ahciLog(("%s: Command Fis size: %d\n", __FUNCTION__, AHCI_CMDFIS_TYPE_DMAACTD2H_SIZE));
2663 break;
2664 }
2665 case AHCI_CMDFIS_TYPE_DMASETUP:
2666 {
2667 ahciLog(("%s: Command Fis type DMA Setup\n", __FUNCTION__));
2668 ahciLog(("%s: Command Fis size: %d\n", __FUNCTION__, AHCI_CMDFIS_TYPE_DMASETUP_SIZE));
2669 break;
2670 }
2671 case AHCI_CMDFIS_TYPE_PIOSETUP:
2672 {
2673 ahciLog(("%s: Command Fis type PIO Setup\n", __FUNCTION__));
2674 ahciLog(("%s: Command Fis size: %d\n", __FUNCTION__, AHCI_CMDFIS_TYPE_PIOSETUP_SIZE));
2675 break;
2676 }
2677 case AHCI_CMDFIS_TYPE_DATA:
2678 {
2679 ahciLog(("%s: Command Fis type Data\n", __FUNCTION__));
2680 break;
2681 }
2682 default:
2683 ahciLog(("%s: ERROR Unknown command FIS type\n", __FUNCTION__));
2684 break;
2685 }
2686 ahciLog(("%s: *** End FIS info dump. ***\n", __FUNCTION__));
2687}
2688
2689/**
2690 * Dump info about the command header
2691 *
2692 * @returns nothing
2693 * @param pAhciPort Pointer to the port the command header was read from.
2694 * @param pCmdHdr The command header to print info from.
2695 */
2696static void ahciDumpCmdHdrInfo(PAHCIPort pAhciPort, CmdHdr *pCmdHdr)
2697{
2698 ahciLog(("%s: *** Begin command header info dump. ***\n", __FUNCTION__));
2699 ahciLog(("%s: Number of Scatter/Gatther List entries: %u\n", __FUNCTION__, AHCI_CMDHDR_PRDTL_ENTRIES(pCmdHdr->u32DescInf)));
2700 if (pCmdHdr->u32DescInf & AHCI_CMDHDR_C)
2701 ahciLog(("%s: Clear busy upon R_OK\n", __FUNCTION__));
2702 if (pCmdHdr->u32DescInf & AHCI_CMDHDR_B)
2703 ahciLog(("%s: BIST Fis\n", __FUNCTION__));
2704 if (pCmdHdr->u32DescInf & AHCI_CMDHDR_R)
2705 ahciLog(("%s: Device Reset Fis\n", __FUNCTION__));
2706 if (pCmdHdr->u32DescInf & AHCI_CMDHDR_P)
2707 ahciLog(("%s: Command prefetchable\n", __FUNCTION__));
2708 if (pCmdHdr->u32DescInf & AHCI_CMDHDR_W)
2709 ahciLog(("%s: Device write\n", __FUNCTION__));
2710 else
2711 ahciLog(("%s: Device read\n", __FUNCTION__));
2712 if (pCmdHdr->u32DescInf & AHCI_CMDHDR_A)
2713 ahciLog(("%s: ATAPI command\n", __FUNCTION__));
2714 else
2715 ahciLog(("%s: ATA command\n", __FUNCTION__));
2716
2717 ahciLog(("%s: Command FIS length %u DW\n", __FUNCTION__, (pCmdHdr->u32DescInf & AHCI_CMDHDR_CFL_MASK)));
2718 ahciLog(("%s: *** End command header info dump. ***\n", __FUNCTION__));
2719}
2720
2721#endif /* LOG_ENABLED */
2722
2723/**
2724 * Post the first D2H FIS from the device into guest memory.
2725 *
2726 * @returns nothing
2727 * @param pAhciPort Pointer to the port which "receives" the FIS.
2728 */
2729static void ahciPostFirstD2HFisIntoMemory(PAHCIPort pAhciPort)
2730{
2731 uint8_t d2hFis[AHCI_CMDFIS_TYPE_D2H_SIZE];
2732
2733 pAhciPort->fFirstD2HFisSend = true;
2734
2735 ahciLog(("%s: Sending First D2H FIS from FIFO\n", __FUNCTION__));
2736 memset(&d2hFis[0], 0, sizeof(d2hFis));
2737 d2hFis[AHCI_CMDFIS_TYPE] = AHCI_CMDFIS_TYPE_D2H;
2738 d2hFis[AHCI_CMDFIS_ERR] = 0x01;
2739
2740 d2hFis[AHCI_CMDFIS_STS] = 0x00;
2741
2742 /* Set the signature based on the device type. */
2743 if (pAhciPort->fATAPI)
2744 {
2745 d2hFis[AHCI_CMDFIS_CYLL] = 0x14;
2746 d2hFis[AHCI_CMDFIS_CYLH] = 0xeb;
2747 }
2748 else
2749 {
2750 d2hFis[AHCI_CMDFIS_CYLL] = 0x00;
2751 d2hFis[AHCI_CMDFIS_CYLH] = 0x00;
2752 }
2753
2754 d2hFis[AHCI_CMDFIS_HEAD] = 0x00;
2755 d2hFis[AHCI_CMDFIS_SECTN] = 0x01;
2756 d2hFis[AHCI_CMDFIS_SECTC] = 0x01;
2757
2758 pAhciPort->regTFD = (1 << 8) | ATA_STAT_SEEK | ATA_STAT_WRERR;
2759 if (!pAhciPort->fATAPI)
2760 pAhciPort->regTFD |= ATA_STAT_READY;
2761
2762 ahciPostFisIntoMemory(pAhciPort, AHCI_CMDFIS_TYPE_D2H, d2hFis);
2763}
2764
2765/**
2766 * Post the FIS in the memory area allocated by the guest and set interrupt if necessary.
2767 *
2768 * @returns VBox status code
2769 * @param pAhciPort The port which "receives" the FIS.
2770 * @param uFisType The type of the FIS.
2771 * @param pCmdFis Pointer to the FIS which is to be posted into memory.
2772 */
2773static int ahciPostFisIntoMemory(PAHCIPort pAhciPort, unsigned uFisType, uint8_t *pCmdFis)
2774{
2775 int rc = VINF_SUCCESS;
2776 RTGCPHYS GCPhysAddrRecFis = pAhciPort->GCPhysAddrFb;
2777 unsigned cbFis = 0;
2778
2779 ahciLog(("%s: pAhciPort=%p uFisType=%u pCmdFis=%p\n", __FUNCTION__, pAhciPort, uFisType, pCmdFis));
2780
2781 if (pAhciPort->regCMD & AHCI_PORT_CMD_FRE)
2782 {
2783 AssertMsg(GCPhysAddrRecFis, ("%s: GCPhysAddrRecFis is 0\n", __FUNCTION__));
2784
2785 /* Determine the offset and size of the FIS based on uFisType. */
2786 switch (uFisType)
2787 {
2788 case AHCI_CMDFIS_TYPE_D2H:
2789 {
2790 GCPhysAddrRecFis += AHCI_RECFIS_RFIS_OFFSET;
2791 cbFis = AHCI_CMDFIS_TYPE_D2H_SIZE;
2792 break;
2793 }
2794 case AHCI_CMDFIS_TYPE_SETDEVBITS:
2795 {
2796 GCPhysAddrRecFis += AHCI_RECFIS_SDBFIS_OFFSET;
2797 cbFis = AHCI_CMDFIS_TYPE_SETDEVBITS_SIZE;
2798 break;
2799 }
2800 case AHCI_CMDFIS_TYPE_DMASETUP:
2801 {
2802 GCPhysAddrRecFis += AHCI_RECFIS_DSFIS_OFFSET;
2803 cbFis = AHCI_CMDFIS_TYPE_DMASETUP_SIZE;
2804 break;
2805 }
2806 case AHCI_CMDFIS_TYPE_PIOSETUP:
2807 {
2808 GCPhysAddrRecFis += AHCI_RECFIS_PSFIS_OFFSET;
2809 cbFis = AHCI_CMDFIS_TYPE_PIOSETUP_SIZE;
2810 break;
2811 }
2812 default:
2813 /*
2814 * We should post the unknown FIS into memory too but this never happens because
2815 * we know which FIS types we generate. ;)
2816 */
2817 AssertMsgFailed(("%s: Unknown FIS type!\n", __FUNCTION__));
2818 }
2819
2820 /* Post the FIS into memory. */
2821 ahciLog(("%s: PDMDevHlpPCIPhysWrite GCPhysAddrRecFis=%RGp cbFis=%u\n", __FUNCTION__, GCPhysAddrRecFis, cbFis));
2822 PDMDevHlpPCIPhysWrite(pAhciPort->CTX_SUFF(pDevIns), GCPhysAddrRecFis, pCmdFis, cbFis);
2823 }
2824
2825 return rc;
2826}
2827
2828DECLINLINE(void) ahciReqSetStatus(PAHCIREQ pAhciReq, uint8_t u8Error, uint8_t u8Status)
2829{
2830 pAhciReq->cmdFis[AHCI_CMDFIS_ERR] = u8Error;
2831 pAhciReq->cmdFis[AHCI_CMDFIS_STS] = u8Status;
2832}
2833
2834static void ataPadString(uint8_t *pbDst, const char *pbSrc, uint32_t cbSize)
2835{
2836 for (uint32_t i = 0; i < cbSize; i++)
2837 {
2838 if (*pbSrc)
2839 pbDst[i ^ 1] = *pbSrc++;
2840 else
2841 pbDst[i ^ 1] = ' ';
2842 }
2843}
2844
2845static uint32_t ataChecksum(void* ptr, size_t count)
2846{
2847 uint8_t u8Sum = 0xa5, *p = (uint8_t*)ptr;
2848 size_t i;
2849
2850 for (i = 0; i < count; i++)
2851 {
2852 u8Sum += *p++;
2853 }
2854
2855 return (uint8_t)-(int32_t)u8Sum;
2856}
2857
2858static int ahciIdentifySS(PAHCIPort pAhciPort, void *pvBuf)
2859{
2860 uint16_t *p = (uint16_t *)pvBuf;
2861 memset(p, 0, 512);
2862 p[0] = RT_H2LE_U16(0x0040);
2863 p[1] = RT_H2LE_U16(RT_MIN(pAhciPort->PCHSGeometry.cCylinders, 16383));
2864 p[3] = RT_H2LE_U16(pAhciPort->PCHSGeometry.cHeads);
2865 /* Block size; obsolete, but required for the BIOS. */
2866 p[5] = RT_H2LE_U16(512);
2867 p[6] = RT_H2LE_U16(pAhciPort->PCHSGeometry.cSectors);
2868 ataPadString((uint8_t *)(p + 10), pAhciPort->szSerialNumber, AHCI_SERIAL_NUMBER_LENGTH); /* serial number */
2869 p[20] = RT_H2LE_U16(3); /* XXX: retired, cache type */
2870 p[21] = RT_H2LE_U16(512); /* XXX: retired, cache size in sectors */
2871 p[22] = RT_H2LE_U16(0); /* ECC bytes per sector */
2872 ataPadString((uint8_t *)(p + 23), pAhciPort->szFirmwareRevision, AHCI_FIRMWARE_REVISION_LENGTH); /* firmware version */
2873 ataPadString((uint8_t *)(p + 27), pAhciPort->szModelNumber, AHCI_MODEL_NUMBER_LENGTH); /* model */
2874#if ATA_MAX_MULT_SECTORS > 1
2875 p[47] = RT_H2LE_U16(0x8000 | ATA_MAX_MULT_SECTORS);
2876#endif
2877 p[48] = RT_H2LE_U16(1); /* dword I/O, used by the BIOS */
2878 p[49] = RT_H2LE_U16(1 << 11 | 1 << 9 | 1 << 8); /* DMA and LBA supported */
2879 p[50] = RT_H2LE_U16(1 << 14); /* No drive specific standby timer minimum */
2880 p[51] = RT_H2LE_U16(240); /* PIO transfer cycle */
2881 p[52] = RT_H2LE_U16(240); /* DMA transfer cycle */
2882 p[53] = RT_H2LE_U16(1 | 1 << 1 | 1 << 2); /* words 54-58,64-70,88 valid */
2883 p[54] = RT_H2LE_U16(RT_MIN(pAhciPort->PCHSGeometry.cCylinders, 16383));
2884 p[55] = RT_H2LE_U16(pAhciPort->PCHSGeometry.cHeads);
2885 p[56] = RT_H2LE_U16(pAhciPort->PCHSGeometry.cSectors);
2886 p[57] = RT_H2LE_U16(RT_MIN(pAhciPort->PCHSGeometry.cCylinders, 16383) * pAhciPort->PCHSGeometry.cHeads * pAhciPort->PCHSGeometry.cSectors);
2887 p[58] = RT_H2LE_U16(RT_MIN(pAhciPort->PCHSGeometry.cCylinders, 16383) * pAhciPort->PCHSGeometry.cHeads * pAhciPort->PCHSGeometry.cSectors >> 16);
2888 if (pAhciPort->cMultSectors)
2889 p[59] = RT_H2LE_U16(0x100 | pAhciPort->cMultSectors);
2890 if (pAhciPort->cTotalSectors <= (1 << 28) - 1)
2891 {
2892 p[60] = RT_H2LE_U16(pAhciPort->cTotalSectors);
2893 p[61] = RT_H2LE_U16(pAhciPort->cTotalSectors >> 16);
2894 }
2895 else
2896 {
2897 /* Report maximum number of sectors possible with LBA28 */
2898 p[60] = RT_H2LE_U16(((1 << 28) - 1) & 0xffff);
2899 p[61] = RT_H2LE_U16(((1 << 28) - 1) >> 16);
2900 }
2901 p[63] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_MDMA, ATA_MDMA_MODE_MAX, pAhciPort->uATATransferMode)); /* MDMA modes supported / mode enabled */
2902 p[64] = RT_H2LE_U16(ATA_PIO_MODE_MAX > 2 ? (1 << (ATA_PIO_MODE_MAX - 2)) - 1 : 0); /* PIO modes beyond PIO2 supported */
2903 p[65] = RT_H2LE_U16(120); /* minimum DMA multiword tx cycle time */
2904 p[66] = RT_H2LE_U16(120); /* recommended DMA multiword tx cycle time */
2905 p[67] = RT_H2LE_U16(120); /* minimum PIO cycle time without flow control */
2906 p[68] = RT_H2LE_U16(120); /* minimum PIO cycle time with IORDY flow control */
2907 if ( pAhciPort->fTrimEnabled
2908 || pAhciPort->cbSector != 512
2909 || pAhciPort->pDrvMedia->pfnIsNonRotational(pAhciPort->pDrvMedia))
2910 {
2911 p[80] = RT_H2LE_U16(0x1f0); /* support everything up to ATA/ATAPI-8 ACS */
2912 p[81] = RT_H2LE_U16(0x28); /* conforms to ATA/ATAPI-8 ACS */
2913 }
2914 else
2915 {
2916 p[80] = RT_H2LE_U16(0x7e); /* support everything up to ATA/ATAPI-6 */
2917 p[81] = RT_H2LE_U16(0x22); /* conforms to ATA/ATAPI-6 */
2918 }
2919 p[82] = RT_H2LE_U16(1 << 3 | 1 << 5 | 1 << 6); /* supports power management, write cache and look-ahead */
2920 p[83] = RT_H2LE_U16(1 << 14 | 1 << 10 | 1 << 12 | 1 << 13); /* supports LBA48, FLUSH CACHE and FLUSH CACHE EXT */
2921 p[84] = RT_H2LE_U16(1 << 14);
2922 p[85] = RT_H2LE_U16(1 << 3 | 1 << 5 | 1 << 6); /* enabled power management, write cache and look-ahead */
2923 p[86] = RT_H2LE_U16(1 << 10 | 1 << 12 | 1 << 13); /* enabled LBA48, FLUSH CACHE and FLUSH CACHE EXT */
2924 p[87] = RT_H2LE_U16(1 << 14);
2925 p[88] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_UDMA, ATA_UDMA_MODE_MAX, pAhciPort->uATATransferMode)); /* UDMA modes supported / mode enabled */
2926 p[93] = RT_H2LE_U16(0x00);
2927 p[100] = RT_H2LE_U16(pAhciPort->cTotalSectors);
2928 p[101] = RT_H2LE_U16(pAhciPort->cTotalSectors >> 16);
2929 p[102] = RT_H2LE_U16(pAhciPort->cTotalSectors >> 32);
2930 p[103] = RT_H2LE_U16(pAhciPort->cTotalSectors >> 48);
2931
2932 /* valid information, more than one logical sector per physical sector, 2^cLogSectorsPerPhysicalExp logical sectors per physical sector */
2933 if (pAhciPort->cLogSectorsPerPhysicalExp)
2934 p[106] = RT_H2LE_U16(RT_BIT(14) | RT_BIT(13) | pAhciPort->cLogSectorsPerPhysicalExp);
2935
2936 if (pAhciPort->cbSector != 512)
2937 {
2938 uint32_t cSectorSizeInWords = pAhciPort->cbSector / sizeof(uint16_t);
2939 /* Enable reporting of logical sector size. */
2940 p[106] |= RT_H2LE_U16(RT_BIT(12) | RT_BIT(14));
2941 p[117] = RT_H2LE_U16(cSectorSizeInWords);
2942 p[118] = RT_H2LE_U16(cSectorSizeInWords >> 16);
2943 }
2944
2945 if (pAhciPort->pDrvMedia->pfnIsNonRotational(pAhciPort->pDrvMedia))
2946 p[217] = RT_H2LE_U16(1); /* Non-rotational medium */
2947
2948 if (pAhciPort->fTrimEnabled) /** @todo Set bit 14 in word 69 too? (Deterministic read after TRIM). */
2949 p[169] = RT_H2LE_U16(1); /* DATA SET MANAGEMENT command supported. */
2950
2951 /* The following are SATA specific */
2952 p[75] = RT_H2LE_U16(pAhciPort->CTX_SUFF(pAhci)->cCmdSlotsAvail-1); /* Number of commands we support, 0's based */
2953 p[76] = RT_H2LE_U16((1 << 8) | (1 << 2)); /* Native command queuing and Serial ATA Gen2 (3.0 Gbps) speed supported */
2954
2955 uint32_t uCsum = ataChecksum(p, 510);
2956 p[255] = RT_H2LE_U16(0xa5 | (uCsum << 8)); /* Integrity word */
2957
2958 return VINF_SUCCESS;
2959}
2960
2961static int ahciR3AtapiIdentify(PAHCIREQ pAhciReq, PAHCIPort pAhciPort, size_t cbData, size_t *pcbData)
2962{
2963 uint16_t p[256];
2964
2965 memset(p, 0, 512);
2966 /* Removable CDROM, 50us response, 12 byte packets */
2967 p[0] = RT_H2LE_U16(2 << 14 | 5 << 8 | 1 << 7 | 2 << 5 | 0 << 0);
2968 ataPadString((uint8_t *)(p + 10), pAhciPort->szSerialNumber, AHCI_SERIAL_NUMBER_LENGTH); /* serial number */
2969 p[20] = RT_H2LE_U16(3); /* XXX: retired, cache type */
2970 p[21] = RT_H2LE_U16(512); /* XXX: retired, cache size in sectors */
2971 ataPadString((uint8_t *)(p + 23), pAhciPort->szFirmwareRevision, AHCI_FIRMWARE_REVISION_LENGTH); /* firmware version */
2972 ataPadString((uint8_t *)(p + 27), pAhciPort->szModelNumber, AHCI_MODEL_NUMBER_LENGTH); /* model */
2973 p[49] = RT_H2LE_U16(1 << 11 | 1 << 9 | 1 << 8); /* DMA and LBA supported */
2974 p[50] = RT_H2LE_U16(1 << 14); /* No drive specific standby timer minimum */
2975 p[51] = RT_H2LE_U16(240); /* PIO transfer cycle */
2976 p[52] = RT_H2LE_U16(240); /* DMA transfer cycle */
2977 p[53] = RT_H2LE_U16(1 << 1 | 1 << 2); /* words 64-70,88 are valid */
2978 p[63] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_MDMA, ATA_MDMA_MODE_MAX, pAhciPort->uATATransferMode)); /* MDMA modes supported / mode enabled */
2979 p[64] = RT_H2LE_U16(ATA_PIO_MODE_MAX > 2 ? (1 << (ATA_PIO_MODE_MAX - 2)) - 1 : 0); /* PIO modes beyond PIO2 supported */
2980 p[65] = RT_H2LE_U16(120); /* minimum DMA multiword tx cycle time */
2981 p[66] = RT_H2LE_U16(120); /* recommended DMA multiword tx cycle time */
2982 p[67] = RT_H2LE_U16(120); /* minimum PIO cycle time without flow control */
2983 p[68] = RT_H2LE_U16(120); /* minimum PIO cycle time with IORDY flow control */
2984 p[73] = RT_H2LE_U16(0x003e); /* ATAPI CDROM major */
2985 p[74] = RT_H2LE_U16(9); /* ATAPI CDROM minor */
2986 p[80] = RT_H2LE_U16(0x7e); /* support everything up to ATA/ATAPI-6 */
2987 p[81] = RT_H2LE_U16(0x22); /* conforms to ATA/ATAPI-6 */
2988 p[82] = RT_H2LE_U16(1 << 4 | 1 << 9); /* supports packet command set and DEVICE RESET */
2989 p[83] = RT_H2LE_U16(1 << 14);
2990 p[84] = RT_H2LE_U16(1 << 14);
2991 p[85] = RT_H2LE_U16(1 << 4 | 1 << 9); /* enabled packet command set and DEVICE RESET */
2992 p[86] = RT_H2LE_U16(0);
2993 p[87] = RT_H2LE_U16(1 << 14);
2994 p[88] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_UDMA, ATA_UDMA_MODE_MAX, pAhciPort->uATATransferMode)); /* UDMA modes supported / mode enabled */
2995 p[93] = RT_H2LE_U16((1 | 1 << 1) << ((pAhciPort->iLUN & 1) == 0 ? 0 : 8) | 1 << 13 | 1 << 14);
2996
2997 /* The following are SATA specific */
2998 p[75] = RT_H2LE_U16(31); /* We support 32 commands */
2999 p[76] = RT_H2LE_U16((1 << 8) | (1 << 2)); /* Native command queuing and Serial ATA Gen2 (3.0 Gbps) speed supported */
3000
3001 /* Copy the buffer in to the scatter gather list. */
3002 *pcbData = ahciR3CopyBufferToPrdtl(pAhciPort->CTX_SUFF(pAhci), pAhciReq, (void *)&p[0],
3003 RT_MIN(cbData, sizeof(p)), 0 /* cbSkip */);
3004 return VINF_SUCCESS;
3005}
3006
3007/**
3008 * Reset all values after a reset of the attached storage device.
3009 *
3010 * @returns nothing
3011 * @param pAhciPort The port the device is attached to.
3012 * @param pAhciReq The state to get the tag number from.
3013 */
3014static void ahciFinishStorageDeviceReset(PAHCIPort pAhciPort, PAHCIREQ pAhciReq)
3015{
3016 int rc;
3017
3018 /* Send a status good D2H FIS. */
3019 pAhciPort->fResetDevice = false;
3020 if (pAhciPort->regCMD & AHCI_PORT_CMD_FRE)
3021 ahciPostFirstD2HFisIntoMemory(pAhciPort);
3022
3023 /* As this is the first D2H FIS after the reset update the signature in the SIG register of the port. */
3024 if (pAhciPort->fATAPI)
3025 pAhciPort->regSIG = AHCI_PORT_SIG_ATAPI;
3026 else
3027 pAhciPort->regSIG = AHCI_PORT_SIG_DISK;
3028 ASMAtomicOrU32(&pAhciPort->u32TasksFinished, (1 << pAhciReq->uTag));
3029
3030 rc = ahciHbaSetInterrupt(pAhciPort->CTX_SUFF(pAhci), pAhciPort->iLUN, VERR_IGNORED);
3031 AssertRC(rc);
3032}
3033
3034/**
3035 * Initiates a device reset caused by ATA_DEVICE_RESET (ATAPI only).
3036 *
3037 * @returns nothing.
3038 * @param pAhciPort The device to reset.
3039 * @param pAhciReq The task state.
3040 */
3041static void ahciDeviceReset(PAHCIPort pAhciPort, PAHCIREQ pAhciReq)
3042{
3043 ASMAtomicWriteBool(&pAhciPort->fResetDevice, true);
3044
3045 /*
3046 * Because this ATAPI only and ATAPI can't have
3047 * more than one command active at a time the task counter should be 0
3048 * and it is possible to finish the reset now.
3049 */
3050 Assert(ASMAtomicReadU32(&pAhciPort->cTasksActive) == 0);
3051 ahciFinishStorageDeviceReset(pAhciPort, pAhciReq);
3052}
3053
3054/**
3055 * Create a PIO setup FIS and post it into the memory area of the guest.
3056 *
3057 * @returns nothing.
3058 * @param pAhciPort The port of the SATA controller.
3059 * @param cbTransfer Transfer size of the request.
3060 * @param pCmdFis Pointer to the command FIS from the guest.
3061 * @param fRead Flag whether this is a read request.
3062 * @param fInterrupt If an interrupt should be send to the guest.
3063 */
3064static void ahciSendPioSetupFis(PAHCIPort pAhciPort, size_t cbTransfer, uint8_t *pCmdFis,
3065 bool fRead, bool fInterrupt)
3066{
3067 uint8_t abPioSetupFis[20];
3068 bool fAssertIntr = false;
3069 PAHCI pAhci = pAhciPort->CTX_SUFF(pAhci);
3070
3071 ahciLog(("%s: building PIO setup Fis\n", __FUNCTION__));
3072
3073 AssertMsg( cbTransfer > 0
3074 && cbTransfer <= 65534,
3075 ("Can't send PIO setup FIS for requests with 0 bytes to transfer or greater than 65534\n"));
3076
3077 if (pAhciPort->regCMD & AHCI_PORT_CMD_FRE)
3078 {
3079 memset(&abPioSetupFis[0], 0, sizeof(abPioSetupFis));
3080 abPioSetupFis[AHCI_CMDFIS_TYPE] = AHCI_CMDFIS_TYPE_PIOSETUP;
3081 abPioSetupFis[AHCI_CMDFIS_BITS] = (fInterrupt ? AHCI_CMDFIS_I : 0);
3082 if (fRead)
3083 abPioSetupFis[AHCI_CMDFIS_BITS] |= AHCI_CMDFIS_D;
3084 abPioSetupFis[AHCI_CMDFIS_STS] = pCmdFis[AHCI_CMDFIS_STS];
3085 abPioSetupFis[AHCI_CMDFIS_ERR] = pCmdFis[AHCI_CMDFIS_ERR];
3086 abPioSetupFis[AHCI_CMDFIS_SECTN] = pCmdFis[AHCI_CMDFIS_SECTN];
3087 abPioSetupFis[AHCI_CMDFIS_CYLL] = pCmdFis[AHCI_CMDFIS_CYLL];
3088 abPioSetupFis[AHCI_CMDFIS_CYLH] = pCmdFis[AHCI_CMDFIS_CYLH];
3089 abPioSetupFis[AHCI_CMDFIS_HEAD] = pCmdFis[AHCI_CMDFIS_HEAD];
3090 abPioSetupFis[AHCI_CMDFIS_SECTNEXP] = pCmdFis[AHCI_CMDFIS_SECTNEXP];
3091 abPioSetupFis[AHCI_CMDFIS_CYLLEXP] = pCmdFis[AHCI_CMDFIS_CYLLEXP];
3092 abPioSetupFis[AHCI_CMDFIS_CYLHEXP] = pCmdFis[AHCI_CMDFIS_CYLHEXP];
3093 abPioSetupFis[AHCI_CMDFIS_SECTC] = pCmdFis[AHCI_CMDFIS_SECTC];
3094 abPioSetupFis[AHCI_CMDFIS_SECTCEXP] = pCmdFis[AHCI_CMDFIS_SECTCEXP];
3095
3096 /* Set transfer count. */
3097 abPioSetupFis[16] = (cbTransfer >> 8) & 0xff;
3098 abPioSetupFis[17] = cbTransfer & 0xff;
3099
3100 /* Update registers. */
3101 pAhciPort->regTFD = (pCmdFis[AHCI_CMDFIS_ERR] << 8) | pCmdFis[AHCI_CMDFIS_STS];
3102
3103 ahciPostFisIntoMemory(pAhciPort, AHCI_CMDFIS_TYPE_PIOSETUP, abPioSetupFis);
3104
3105 if (fInterrupt)
3106 {
3107 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_PSS);
3108 /* Check if we should assert an interrupt */
3109 if (pAhciPort->regIE & AHCI_PORT_IE_PSE)
3110 fAssertIntr = true;
3111 }
3112
3113 if (fAssertIntr)
3114 {
3115 int rc = ahciHbaSetInterrupt(pAhci, pAhciPort->iLUN, VERR_IGNORED);
3116 AssertRC(rc);
3117 }
3118 }
3119}
3120
3121/**
3122 * Build a D2H FIS and post into the memory area of the guest.
3123 *
3124 * @returns Nothing
3125 * @param pAhciPort The port of the SATA controller.
3126 * @param uTag The tag of the request.
3127 * @param pCmdFis Pointer to the command FIS from the guest.
3128 * @param fInterrupt If an interrupt should be send to the guest.
3129 */
3130static void ahciSendD2HFis(PAHCIPort pAhciPort, uint32_t uTag, uint8_t *pCmdFis, bool fInterrupt)
3131{
3132 uint8_t d2hFis[20];
3133 bool fAssertIntr = false;
3134 PAHCI pAhci = pAhciPort->CTX_SUFF(pAhci);
3135
3136 ahciLog(("%s: building D2H Fis\n", __FUNCTION__));
3137
3138 if (pAhciPort->regCMD & AHCI_PORT_CMD_FRE)
3139 {
3140 memset(&d2hFis[0], 0, sizeof(d2hFis));
3141 d2hFis[AHCI_CMDFIS_TYPE] = AHCI_CMDFIS_TYPE_D2H;
3142 d2hFis[AHCI_CMDFIS_BITS] = (fInterrupt ? AHCI_CMDFIS_I : 0);
3143 d2hFis[AHCI_CMDFIS_STS] = pCmdFis[AHCI_CMDFIS_STS];
3144 d2hFis[AHCI_CMDFIS_ERR] = pCmdFis[AHCI_CMDFIS_ERR];
3145 d2hFis[AHCI_CMDFIS_SECTN] = pCmdFis[AHCI_CMDFIS_SECTN];
3146 d2hFis[AHCI_CMDFIS_CYLL] = pCmdFis[AHCI_CMDFIS_CYLL];
3147 d2hFis[AHCI_CMDFIS_CYLH] = pCmdFis[AHCI_CMDFIS_CYLH];
3148 d2hFis[AHCI_CMDFIS_HEAD] = pCmdFis[AHCI_CMDFIS_HEAD];
3149 d2hFis[AHCI_CMDFIS_SECTNEXP] = pCmdFis[AHCI_CMDFIS_SECTNEXP];
3150 d2hFis[AHCI_CMDFIS_CYLLEXP] = pCmdFis[AHCI_CMDFIS_CYLLEXP];
3151 d2hFis[AHCI_CMDFIS_CYLHEXP] = pCmdFis[AHCI_CMDFIS_CYLHEXP];
3152 d2hFis[AHCI_CMDFIS_SECTC] = pCmdFis[AHCI_CMDFIS_SECTC];
3153 d2hFis[AHCI_CMDFIS_SECTCEXP] = pCmdFis[AHCI_CMDFIS_SECTCEXP];
3154
3155 /* Update registers. */
3156 pAhciPort->regTFD = (pCmdFis[AHCI_CMDFIS_ERR] << 8) | pCmdFis[AHCI_CMDFIS_STS];
3157
3158 ahciPostFisIntoMemory(pAhciPort, AHCI_CMDFIS_TYPE_D2H, d2hFis);
3159
3160 if (pCmdFis[AHCI_CMDFIS_STS] & ATA_STAT_ERR)
3161 {
3162 /* Error bit is set. */
3163 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_TFES);
3164 if (pAhciPort->regIE & AHCI_PORT_IE_TFEE)
3165 fAssertIntr = true;
3166 /*
3167 * Don't mark the command slot as completed because the guest
3168 * needs it to identify the failed command.
3169 */
3170 }
3171 else if (fInterrupt)
3172 {
3173 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_DHRS);
3174 /* Check if we should assert an interrupt */
3175 if (pAhciPort->regIE & AHCI_PORT_IE_DHRE)
3176 fAssertIntr = true;
3177
3178 /* Mark command as completed. */
3179 ASMAtomicOrU32(&pAhciPort->u32TasksFinished, RT_BIT_32(uTag));
3180 }
3181
3182 if (fAssertIntr)
3183 {
3184 int rc = ahciHbaSetInterrupt(pAhci, pAhciPort->iLUN, VERR_IGNORED);
3185 AssertRC(rc);
3186 }
3187 }
3188}
3189
3190/**
3191 * Build a SDB Fis and post it into the memory area of the guest.
3192 *
3193 * @returns Nothing
3194 * @param pAhciPort The port for which the SDB Fis is send.
3195 * @param uFinishedTasks Bitmask of finished tasks.
3196 * @param fInterrupt If an interrupt should be asserted.
3197 */
3198static void ahciSendSDBFis(PAHCIPort pAhciPort, uint32_t uFinishedTasks, bool fInterrupt)
3199{
3200 uint32_t sdbFis[2];
3201 bool fAssertIntr = false;
3202 PAHCI pAhci = pAhciPort->CTX_SUFF(pAhci);
3203 PAHCIREQ pTaskErr = ASMAtomicReadPtrT(&pAhciPort->pTaskErr, PAHCIREQ);
3204
3205 ahciLog(("%s: Building SDB FIS\n", __FUNCTION__));
3206
3207 if (pAhciPort->regCMD & AHCI_PORT_CMD_FRE)
3208 {
3209 memset(&sdbFis[0], 0, sizeof(sdbFis));
3210 sdbFis[0] = AHCI_CMDFIS_TYPE_SETDEVBITS;
3211 sdbFis[0] |= (fInterrupt ? (1 << 14) : 0);
3212 if (RT_UNLIKELY(pTaskErr))
3213 {
3214 sdbFis[0] = pTaskErr->cmdFis[AHCI_CMDFIS_ERR];
3215 sdbFis[0] |= (pTaskErr->cmdFis[AHCI_CMDFIS_STS] & 0x77) << 16; /* Some bits are marked as reserved and thus are masked out. */
3216
3217 /* Update registers. */
3218 pAhciPort->regTFD = (pTaskErr->cmdFis[AHCI_CMDFIS_ERR] << 8) | pTaskErr->cmdFis[AHCI_CMDFIS_STS];
3219 }
3220 else
3221 {
3222 sdbFis[0] = 0;
3223 sdbFis[0] |= (ATA_STAT_READY | ATA_STAT_SEEK) << 16;
3224 pAhciPort->regTFD = ATA_STAT_READY | ATA_STAT_SEEK;
3225 }
3226
3227 sdbFis[1] = pAhciPort->u32QueuedTasksFinished | uFinishedTasks;
3228
3229 ahciPostFisIntoMemory(pAhciPort, AHCI_CMDFIS_TYPE_SETDEVBITS, (uint8_t *)sdbFis);
3230
3231 if (RT_UNLIKELY(pTaskErr))
3232 {
3233 /* Error bit is set. */
3234 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_TFES);
3235 if (pAhciPort->regIE & AHCI_PORT_IE_TFEE)
3236 fAssertIntr = true;
3237 }
3238
3239 if (fInterrupt)
3240 {
3241 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_SDBS);
3242 /* Check if we should assert an interrupt */
3243 if (pAhciPort->regIE & AHCI_PORT_IE_SDBE)
3244 fAssertIntr = true;
3245 }
3246
3247 ASMAtomicOrU32(&pAhciPort->u32QueuedTasksFinished, uFinishedTasks);
3248
3249 if (fAssertIntr)
3250 {
3251 int rc = ahciHbaSetInterrupt(pAhci, pAhciPort->iLUN, VERR_IGNORED);
3252 AssertRC(rc);
3253 }
3254 }
3255}
3256
3257static uint32_t ahciGetNSectors(uint8_t *pCmdFis, bool fLBA48)
3258{
3259 /* 0 means either 256 (LBA28) or 65536 (LBA48) sectors. */
3260 if (fLBA48)
3261 {
3262 if (!pCmdFis[AHCI_CMDFIS_SECTC] && !pCmdFis[AHCI_CMDFIS_SECTCEXP])
3263 return 65536;
3264 else
3265 return pCmdFis[AHCI_CMDFIS_SECTCEXP] << 8 | pCmdFis[AHCI_CMDFIS_SECTC];
3266 }
3267 else
3268 {
3269 if (!pCmdFis[AHCI_CMDFIS_SECTC])
3270 return 256;
3271 else
3272 return pCmdFis[AHCI_CMDFIS_SECTC];
3273 }
3274}
3275
3276static uint64_t ahciGetSector(PAHCIPort pAhciPort, uint8_t *pCmdFis, bool fLBA48)
3277{
3278 uint64_t iLBA;
3279 if (pCmdFis[AHCI_CMDFIS_HEAD] & 0x40)
3280 {
3281 /* any LBA variant */
3282 if (fLBA48)
3283 {
3284 /* LBA48 */
3285 iLBA = ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLHEXP] << 40) |
3286 ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLLEXP] << 32) |
3287 ((uint64_t)pCmdFis[AHCI_CMDFIS_SECTNEXP] << 24) |
3288 ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLH] << 16) |
3289 ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLL] << 8) |
3290 pCmdFis[AHCI_CMDFIS_SECTN];
3291 }
3292 else
3293 {
3294 /* LBA */
3295 iLBA = ((pCmdFis[AHCI_CMDFIS_HEAD] & 0x0f) << 24) | (pCmdFis[AHCI_CMDFIS_CYLH] << 16) |
3296 (pCmdFis[AHCI_CMDFIS_CYLL] << 8) | pCmdFis[AHCI_CMDFIS_SECTN];
3297 }
3298 }
3299 else
3300 {
3301 /* CHS */
3302 iLBA = ((pCmdFis[AHCI_CMDFIS_CYLH] << 8) | pCmdFis[AHCI_CMDFIS_CYLL]) * pAhciPort->PCHSGeometry.cHeads * pAhciPort->PCHSGeometry.cSectors +
3303 (pCmdFis[AHCI_CMDFIS_HEAD] & 0x0f) * pAhciPort->PCHSGeometry.cSectors +
3304 (pCmdFis[AHCI_CMDFIS_SECTN] - 1);
3305 }
3306 return iLBA;
3307}
3308
3309static uint64_t ahciGetSectorQueued(uint8_t *pCmdFis)
3310{
3311 uint64_t uLBA;
3312
3313 uLBA = ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLHEXP] << 40) |
3314 ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLLEXP] << 32) |
3315 ((uint64_t)pCmdFis[AHCI_CMDFIS_SECTNEXP] << 24) |
3316 ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLH] << 16) |
3317 ((uint64_t)pCmdFis[AHCI_CMDFIS_CYLL] << 8) |
3318 pCmdFis[AHCI_CMDFIS_SECTN];
3319
3320 return uLBA;
3321}
3322
3323DECLINLINE(uint32_t) ahciGetNSectorsQueued(uint8_t *pCmdFis)
3324{
3325 if (!pCmdFis[AHCI_CMDFIS_FETEXP] && !pCmdFis[AHCI_CMDFIS_FET])
3326 return 65536;
3327 else
3328 return pCmdFis[AHCI_CMDFIS_FETEXP] << 8 | pCmdFis[AHCI_CMDFIS_FET];
3329}
3330
3331/**
3332 * Copy from guest to host memory worker.
3333 *
3334 * @copydoc AHCIR3MEMCOPYCALLBACK
3335 */
3336static DECLCALLBACK(void) ahciR3CopyBufferFromGuestWorker(PAHCI pThis, RTGCPHYS GCPhys, PRTSGBUF pSgBuf,
3337 size_t cbCopy, size_t *pcbSkip)
3338{
3339 size_t cbSkipped = RT_MIN(cbCopy, *pcbSkip);
3340 cbCopy -= cbSkipped;
3341 GCPhys += cbSkipped;
3342 *pcbSkip -= cbSkipped;
3343
3344 while (cbCopy)
3345 {
3346 size_t cbSeg = cbCopy;
3347 void *pvSeg = RTSgBufGetNextSegment(pSgBuf, &cbSeg);
3348
3349 AssertPtr(pvSeg);
3350 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhys, pvSeg, cbSeg);
3351 GCPhys += cbSeg;
3352 cbCopy -= cbSeg;
3353 }
3354}
3355
3356/**
3357 * Copy from host to guest memory worker.
3358 *
3359 * @copydoc AHCIR3MEMCOPYCALLBACK
3360 */
3361static DECLCALLBACK(void) ahciR3CopyBufferToGuestWorker(PAHCI pThis, RTGCPHYS GCPhys, PRTSGBUF pSgBuf,
3362 size_t cbCopy, size_t *pcbSkip)
3363{
3364 size_t cbSkipped = RT_MIN(cbCopy, *pcbSkip);
3365 cbCopy -= cbSkipped;
3366 GCPhys += cbSkipped;
3367 *pcbSkip -= cbSkipped;
3368
3369 while (cbCopy)
3370 {
3371 size_t cbSeg = cbCopy;
3372 void *pvSeg = RTSgBufGetNextSegment(pSgBuf, &cbSeg);
3373
3374 AssertPtr(pvSeg);
3375 PDMDevHlpPCIPhysWrite(pThis->CTX_SUFF(pDevIns), GCPhys, pvSeg, cbSeg);
3376 GCPhys += cbSeg;
3377 cbCopy -= cbSeg;
3378 }
3379}
3380
3381/**
3382 * Walks the PRDTL list copying data between the guest and host memory buffers.
3383 *
3384 * @returns Amount of bytes copied.
3385 * @param pThis The AHCI controller device instance.
3386 * @param pAhciReq AHCI request structure.
3387 * @param pfnCopyWorker The copy method to apply for each guest buffer.
3388 * @param pSgBuf The host S/G buffer.
3389 * @param cbSkip How many bytes to skip in advance before starting to copy.
3390 * @param cbCopy How many bytes to copy.
3391 */
3392static size_t ahciR3PrdtlWalk(PAHCI pThis, PAHCIREQ pAhciReq,
3393 PAHCIR3MEMCOPYCALLBACK pfnCopyWorker,
3394 PRTSGBUF pSgBuf, size_t cbSkip, size_t cbCopy)
3395{
3396 RTGCPHYS GCPhysPrdtl = pAhciReq->GCPhysPrdtl;
3397 unsigned cPrdtlEntries = pAhciReq->cPrdtlEntries;
3398 size_t cbCopied = 0;
3399
3400 /*
3401 * Add the amount to skip to the host buffer size to avoid a
3402 * few conditionals later on.
3403 */
3404 cbCopy += cbSkip;
3405
3406 AssertMsgReturn(cPrdtlEntries > 0, ("Copying 0 bytes is not possible\n"), 0);
3407
3408 do
3409 {
3410 SGLEntry aPrdtlEntries[32];
3411 uint32_t cPrdtlEntriesRead = cPrdtlEntries < RT_ELEMENTS(aPrdtlEntries)
3412 ? cPrdtlEntries
3413 : RT_ELEMENTS(aPrdtlEntries);
3414
3415 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysPrdtl, &aPrdtlEntries[0],
3416 cPrdtlEntriesRead * sizeof(SGLEntry));
3417
3418 for (uint32_t i = 0; (i < cPrdtlEntriesRead) && cbCopy; i++)
3419 {
3420 RTGCPHYS GCPhysAddrDataBase = AHCI_RTGCPHYS_FROM_U32(aPrdtlEntries[i].u32DBAUp, aPrdtlEntries[i].u32DBA);
3421 uint32_t cbThisCopy = (aPrdtlEntries[i].u32DescInf & SGLENTRY_DESCINF_DBC) + 1;
3422
3423 cbThisCopy = (uint32_t)RT_MIN(cbThisCopy, cbCopy);
3424
3425 /* Copy into SG entry. */
3426 pfnCopyWorker(pThis, GCPhysAddrDataBase, pSgBuf, cbThisCopy, &cbSkip);
3427
3428 cbCopy -= cbThisCopy;
3429 cbCopied += cbThisCopy;
3430 }
3431
3432 GCPhysPrdtl += cPrdtlEntriesRead * sizeof(SGLEntry);
3433 cPrdtlEntries -= cPrdtlEntriesRead;
3434 } while (cPrdtlEntries && cbCopy);
3435
3436 if (cbCopied < cbCopy)
3437 pAhciReq->fFlags |= AHCI_REQ_OVERFLOW;
3438
3439 return cbCopied;
3440}
3441
3442/**
3443 * Copies a data buffer into the S/G buffer set up by the guest.
3444 *
3445 * @returns Amount of bytes copied to the PRDTL.
3446 * @param pThis The AHCI controller device instance.
3447 * @param pAhciReq AHCI request structure.
3448 * @param pSgBuf The S/G buffer to copy from.
3449 * @param cbSkip How many bytes to skip in advance before starting to copy.
3450 * @param cbCopy How many bytes to copy.
3451 */
3452static size_t ahciR3CopySgBufToPrdtl(PAHCI pThis, PAHCIREQ pAhciReq, PRTSGBUF pSgBuf,
3453 size_t cbSkip, size_t cbCopy)
3454{
3455 return ahciR3PrdtlWalk(pThis, pAhciReq, ahciR3CopyBufferToGuestWorker,
3456 pSgBuf, cbSkip, cbCopy);
3457}
3458
3459/**
3460 * Copies the S/G buffer into a data buffer.
3461 *
3462 * @returns Amount of bytes copied from the PRDTL.
3463 * @param pThis The AHCI controller device instance.
3464 * @param pAhciReq AHCI request structure.
3465 * @param pSgBuf The S/G buffer to copy into.
3466 * @param cbSkip How many bytes to skip in advance before starting to copy.
3467 * @param cbCopy How many bytes to copy.
3468 */
3469static size_t ahciR3CopySgBufFromPrdtl(PAHCI pThis, PAHCIREQ pAhciReq, PRTSGBUF pSgBuf,
3470 size_t cbSkip, size_t cbCopy)
3471{
3472 return ahciR3PrdtlWalk(pThis, pAhciReq, ahciR3CopyBufferFromGuestWorker,
3473 pSgBuf, cbSkip, cbCopy);
3474}
3475
3476/**
3477 * Copy a simple memory buffer to the guest memory buffer.
3478 *
3479 * @returns Amount of bytes copied from the PRDTL.
3480 * @param pThis The AHCI controller device instance.
3481 * @param pAhciReq AHCI request structure.
3482 * @param pvSrc The buffer to copy from.
3483 * @param cbSrc How many bytes to copy.
3484 * @param cbSkip How many bytes to skip initially.
3485 */
3486static size_t ahciR3CopyBufferToPrdtl(PAHCI pThis, PAHCIREQ pAhciReq, const void *pvSrc,
3487 size_t cbSrc, size_t cbSkip)
3488{
3489 RTSGSEG Seg;
3490 RTSGBUF SgBuf;
3491 Seg.pvSeg = (void *)pvSrc;
3492 Seg.cbSeg = cbSrc;
3493 RTSgBufInit(&SgBuf, &Seg, 1);
3494 return ahciR3CopySgBufToPrdtl(pThis, pAhciReq, &SgBuf, cbSkip, cbSrc);
3495}
3496
3497/**
3498 * Calculates the size of the guest buffer described by the PRDT.
3499 *
3500 * @returns VBox status code.
3501 * @param pThis The AHCI controller device instance.
3502 * @param pAhciReq AHCI request structure.
3503 * @param pcbPrdt Where to store the size of the guest buffer.
3504 */
3505static int ahciR3PrdtQuerySize(PAHCI pThis, PAHCIREQ pAhciReq, size_t *pcbPrdt)
3506{
3507 RTGCPHYS GCPhysPrdtl = pAhciReq->GCPhysPrdtl;
3508 unsigned cPrdtlEntries = pAhciReq->cPrdtlEntries;
3509 size_t cbPrdt = 0;
3510
3511 do
3512 {
3513 SGLEntry aPrdtlEntries[32];
3514 uint32_t cPrdtlEntriesRead = cPrdtlEntries < RT_ELEMENTS(aPrdtlEntries)
3515 ? cPrdtlEntries
3516 : RT_ELEMENTS(aPrdtlEntries);
3517
3518 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysPrdtl, &aPrdtlEntries[0],
3519 cPrdtlEntriesRead * sizeof(SGLEntry));
3520
3521 for (uint32_t i = 0; i < cPrdtlEntriesRead; i++)
3522 cbPrdt += (aPrdtlEntries[i].u32DescInf & SGLENTRY_DESCINF_DBC) + 1;
3523
3524 GCPhysPrdtl += cPrdtlEntriesRead * sizeof(SGLEntry);
3525 cPrdtlEntries -= cPrdtlEntriesRead;
3526 } while (cPrdtlEntries);
3527
3528 *pcbPrdt = cbPrdt;
3529 return VINF_SUCCESS;
3530}
3531
3532/**
3533 * Cancels all active tasks on the port.
3534 *
3535 * @returns Whether all active tasks were canceled.
3536 * @param pAhciPort The AHCI port.
3537 */
3538static bool ahciCancelActiveTasks(PAHCIPort pAhciPort)
3539{
3540 if (pAhciPort->pDrvMediaEx)
3541 {
3542 int rc = pAhciPort->pDrvMediaEx->pfnIoReqCancelAll(pAhciPort->pDrvMediaEx);
3543 AssertRC(rc);
3544 }
3545 return true; /* always true for now because tasks don't use guest memory as the buffer which makes canceling a task impossible. */
3546}
3547
3548/**
3549 * Creates the array of ranges to trim.
3550 *
3551 * @returns VBox status code.
3552 * @param pAhciPort AHCI port state.
3553 * @param pAhciReq The request handling the TRIM request.
3554 * @param idxRangeStart Index of the first range to start copying.
3555 * @param paRanges Where to store the ranges.
3556 * @param cRanges Number of ranges fitting into the array.
3557 * @param pcRanges Where to store the amount of ranges actually copied on success.
3558 */
3559static int ahciTrimRangesCreate(PAHCIPort pAhciPort, PAHCIREQ pAhciReq, uint32_t idxRangeStart,
3560 PRTRANGE paRanges, uint32_t cRanges, uint32_t *pcRanges)
3561{
3562 SGLEntry aPrdtlEntries[32];
3563 uint64_t aRanges[64];
3564 uint32_t cPrdtlEntries = pAhciReq->cPrdtlEntries;
3565 RTGCPHYS GCPhysPrdtl = pAhciReq->GCPhysPrdtl;
3566 PPDMDEVINS pDevIns = pAhciPort->CTX_SUFF(pDevIns);
3567 int rc = VERR_PDM_MEDIAEX_IOBUF_OVERFLOW;
3568 uint32_t idxRange = 0;
3569
3570 LogFlowFunc(("pAhciPort=%#p pAhciReq=%#p\n", pAhciPort, pAhciReq));
3571
3572 AssertMsgReturn(pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_DISCARD, ("This is not a trim request\n"), VERR_INVALID_PARAMETER);
3573
3574 if (!cPrdtlEntries)
3575 pAhciReq->fFlags |= AHCI_REQ_OVERFLOW;
3576
3577 /* Convert the ranges from ATA to our format. */
3578 while ( cPrdtlEntries
3579 && idxRange < cRanges)
3580 {
3581 uint32_t cPrdtlEntriesRead = RT_MIN(cPrdtlEntries, RT_ELEMENTS(aPrdtlEntries));
3582
3583 rc = VINF_SUCCESS;
3584 PDMDevHlpPhysRead(pDevIns, GCPhysPrdtl, &aPrdtlEntries[0], cPrdtlEntriesRead * sizeof(SGLEntry));
3585
3586 for (uint32_t i = 0; i < cPrdtlEntriesRead && idxRange < cRanges; i++)
3587 {
3588 RTGCPHYS GCPhysAddrDataBase = AHCI_RTGCPHYS_FROM_U32(aPrdtlEntries[i].u32DBAUp, aPrdtlEntries[i].u32DBA);
3589 uint32_t cbThisCopy = (aPrdtlEntries[i].u32DescInf & SGLENTRY_DESCINF_DBC) + 1;
3590
3591 cbThisCopy = RT_MIN(cbThisCopy, sizeof(aRanges));
3592
3593 /* Copy into buffer. */
3594 PDMDevHlpPhysRead(pDevIns, GCPhysAddrDataBase, aRanges, cbThisCopy);
3595
3596 for (unsigned idxRangeSrc = 0; idxRangeSrc < RT_ELEMENTS(aRanges) && idxRange < cRanges; idxRangeSrc++)
3597 {
3598 /* Skip range if told to do so. */
3599 if (!idxRangeStart)
3600 {
3601 aRanges[idxRangeSrc] = RT_H2LE_U64(aRanges[idxRangeSrc]);
3602 if (AHCI_RANGE_LENGTH_GET(aRanges[idxRangeSrc]) != 0)
3603 {
3604 paRanges[idxRange].offStart = (aRanges[idxRangeSrc] & AHCI_RANGE_LBA_MASK) * pAhciPort->cbSector;
3605 paRanges[idxRange].cbRange = AHCI_RANGE_LENGTH_GET(aRanges[idxRangeSrc]) * pAhciPort->cbSector;
3606 idxRange++;
3607 }
3608 else
3609 break;
3610 }
3611 else
3612 idxRangeStart--;
3613 }
3614 }
3615
3616 GCPhysPrdtl += cPrdtlEntriesRead * sizeof(SGLEntry);
3617 cPrdtlEntries -= cPrdtlEntriesRead;
3618 }
3619
3620 *pcRanges = idxRange;
3621
3622 LogFlowFunc(("returns rc=%Rrc\n", rc));
3623 return rc;
3624}
3625
3626/**
3627 * Allocates a new AHCI request.
3628 *
3629 * @returns A new AHCI request structure or NULL if out of memory.
3630 * @param pAhciPort The AHCI port.
3631 * @param uTag The tag to assign.
3632 */
3633static PAHCIREQ ahciR3ReqAlloc(PAHCIPort pAhciPort, uint32_t uTag)
3634{
3635 PAHCIREQ pAhciReq = NULL;
3636 PDMMEDIAEXIOREQ hIoReq = NULL;
3637
3638 int rc = pAhciPort->pDrvMediaEx->pfnIoReqAlloc(pAhciPort->pDrvMediaEx, &hIoReq, (void **)&pAhciReq,
3639 uTag, PDMIMEDIAEX_F_SUSPEND_ON_RECOVERABLE_ERR);
3640 if (RT_SUCCESS(rc))
3641 {
3642 pAhciReq->hIoReq = hIoReq;
3643 pAhciReq->fMapped = false;
3644 }
3645 else
3646 pAhciReq = NULL;
3647 return pAhciReq;
3648}
3649
3650/**
3651 * Frees a given AHCI request structure.
3652 *
3653 * @returns nothing.
3654 * @param pAhciPort The AHCI port.
3655 * @param pAhciReq The request to free.
3656 */
3657static void ahciR3ReqFree(PAHCIPort pAhciPort, PAHCIREQ pAhciReq)
3658{
3659 if ( pAhciReq
3660 && !(pAhciReq->fFlags & AHCI_REQ_IS_ON_STACK))
3661 {
3662 int rc = pAhciPort->pDrvMediaEx->pfnIoReqFree(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq);
3663 AssertRC(rc);
3664 }
3665}
3666
3667/**
3668 * Complete a data transfer task by freeing all occupied resources
3669 * and notifying the guest.
3670 *
3671 * @returns Flag whether the given request was canceled inbetween;
3672 *
3673 * @param pAhciPort Pointer to the port where to request completed.
3674 * @param pAhciReq Pointer to the task which finished.
3675 * @param rcReq IPRT status code of the completed request.
3676 */
3677static bool ahciTransferComplete(PAHCIPort pAhciPort, PAHCIREQ pAhciReq, int rcReq)
3678{
3679 bool fCanceled = false;
3680
3681 LogFlowFunc(("pAhciPort=%p pAhciReq=%p rcReq=%d\n",
3682 pAhciPort, pAhciReq, rcReq));
3683
3684 VBOXDD_AHCI_REQ_COMPLETED(pAhciReq, rcReq, pAhciReq->uOffset, pAhciReq->cbTransfer);
3685
3686 if (pAhciReq->fMapped)
3687 PDMDevHlpPhysReleasePageMappingLock(pAhciPort->CTX_SUFF(pAhci)->CTX_SUFF(pDevIns),
3688 &pAhciReq->PgLck);
3689
3690 if (rcReq != VERR_PDM_MEDIAEX_IOREQ_CANCELED)
3691 {
3692 if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_READ)
3693 pAhciPort->Led.Actual.s.fReading = 0;
3694 else if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_WRITE)
3695 pAhciPort->Led.Actual.s.fWriting = 0;
3696 else if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_DISCARD)
3697 pAhciPort->Led.Actual.s.fWriting = 0;
3698 else if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_SCSI)
3699 {
3700 pAhciPort->Led.Actual.s.fWriting = 0;
3701 pAhciPort->Led.Actual.s.fReading = 0;
3702 }
3703
3704 if (RT_FAILURE(rcReq))
3705 {
3706 /* Log the error. */
3707 if (pAhciPort->cErrors++ < MAX_LOG_REL_ERRORS)
3708 {
3709 if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_FLUSH)
3710 LogRel(("AHCI#%uP%u: Flush returned rc=%Rrc\n",
3711 pAhciPort->CTX_SUFF(pDevIns)->iInstance, pAhciPort->iLUN, rcReq));
3712 else if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_DISCARD)
3713 LogRel(("AHCI#%uP%u: Trim returned rc=%Rrc\n",
3714 pAhciPort->CTX_SUFF(pDevIns)->iInstance, pAhciPort->iLUN, rcReq));
3715 else
3716 LogRel(("AHCI#%uP%u: %s at offset %llu (%zu bytes left) returned rc=%Rrc\n",
3717 pAhciPort->CTX_SUFF(pDevIns)->iInstance, pAhciPort->iLUN,
3718 pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_READ
3719 ? "Read"
3720 : "Write",
3721 pAhciReq->uOffset,
3722 pAhciReq->cbTransfer, rcReq));
3723 }
3724
3725 ahciReqSetStatus(pAhciReq, ID_ERR, ATA_STAT_READY | ATA_STAT_ERR);
3726 /*
3727 * We have to duplicate the request here as the underlying I/O
3728 * request will be freed later.
3729 */
3730 PAHCIREQ pReqDup = (PAHCIREQ)RTMemDup(pAhciReq, sizeof(AHCIREQ));
3731 if ( pReqDup
3732 && !ASMAtomicCmpXchgPtr(&pAhciPort->pTaskErr, pReqDup, NULL))
3733 RTMemFree(pReqDup);
3734 }
3735 else
3736 {
3737 /* Status will be set already for non I/O requests. */
3738 if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_SCSI)
3739 {
3740 if (pAhciReq->u8ScsiSts == SCSI_STATUS_OK)
3741 {
3742 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
3743 pAhciReq->cmdFis[AHCI_CMDFIS_SECTN] = (pAhciReq->cmdFis[AHCI_CMDFIS_SECTN] & ~7)
3744 | ((pAhciReq->fFlags & AHCI_REQ_XFER_2_HOST) ? ATAPI_INT_REASON_IO : 0)
3745 | (!pAhciReq->cbTransfer ? ATAPI_INT_REASON_CD : 0);
3746 }
3747 else
3748 {
3749 ahciReqSetStatus(pAhciReq, pAhciPort->abATAPISense[2] << 4, ATA_STAT_READY | ATA_STAT_ERR);
3750 pAhciReq->cmdFis[AHCI_CMDFIS_SECTN] = (pAhciReq->cmdFis[AHCI_CMDFIS_SECTN] & ~7) |
3751 ATAPI_INT_REASON_IO | ATAPI_INT_REASON_CD;
3752 pAhciReq->cbTransfer = 0;
3753 LogFlowFunc(("SCSI request completed with %u status\n", pAhciReq->u8ScsiSts));
3754 }
3755 }
3756 else if (pAhciReq->enmType != PDMMEDIAEXIOREQTYPE_INVALID)
3757 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
3758
3759 /* Write updated command header into memory of the guest. */
3760 uint32_t u32PRDBC = 0;
3761 if (pAhciReq->enmType != PDMMEDIAEXIOREQTYPE_INVALID)
3762 {
3763 size_t cbXfer = 0;
3764 int rc = pAhciPort->pDrvMediaEx->pfnIoReqQueryXferSize(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq, &cbXfer);
3765 AssertRC(rc);
3766 u32PRDBC = (uint32_t)RT_MIN(cbXfer, pAhciReq->cbTransfer);
3767 }
3768 else
3769 u32PRDBC = (uint32_t)pAhciReq->cbTransfer;
3770
3771 PDMDevHlpPCIPhysWrite(pAhciPort->CTX_SUFF(pDevIns), pAhciReq->GCPhysCmdHdrAddr + RT_OFFSETOF(CmdHdr, u32PRDBC),
3772 &u32PRDBC, sizeof(u32PRDBC));
3773
3774 if (pAhciReq->fFlags & AHCI_REQ_OVERFLOW)
3775 {
3776 /*
3777 * The guest tried to transfer more data than there is space in the buffer.
3778 * Terminate task and set the overflow bit.
3779 */
3780 /* Notify the guest. */
3781 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_OFS);
3782 if (pAhciPort->regIE & AHCI_PORT_IE_OFE)
3783 ahciHbaSetInterrupt(pAhciPort->CTX_SUFF(pAhci), pAhciPort->iLUN, VERR_IGNORED);
3784 }
3785 }
3786
3787 /*
3788 * Make a copy of the required data now and free the request. Otherwise the guest
3789 * might issue a new request with the same tag and we run into a conflict when allocating
3790 * a new request with the same tag later on.
3791 */
3792 uint32_t fFlags = pAhciReq->fFlags;
3793 uint32_t uTag = pAhciReq->uTag;
3794 size_t cbTransfer = pAhciReq->cbTransfer;
3795 bool fRead = pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_READ;
3796 uint8_t cmdFis[AHCI_CMDFIS_TYPE_H2D_SIZE];
3797 memcpy(&cmdFis[0], &pAhciReq->cmdFis[0], sizeof(cmdFis));
3798
3799 ahciR3ReqFree(pAhciPort, pAhciReq);
3800
3801 /* Post a PIO setup FIS first if this is a PIO command which transfers data. */
3802 if (fFlags & AHCI_REQ_PIO_DATA)
3803 ahciSendPioSetupFis(pAhciPort, cbTransfer, &cmdFis[0], fRead, false /* fInterrupt */);
3804
3805 if (fFlags & AHCI_REQ_CLEAR_SACT)
3806 {
3807 if (RT_SUCCESS(rcReq) && !ASMAtomicReadPtrT(&pAhciPort->pTaskErr, PAHCIREQ))
3808 ASMAtomicOrU32(&pAhciPort->u32QueuedTasksFinished, RT_BIT_32(uTag));
3809 }
3810
3811 if (fFlags & AHCI_REQ_IS_QUEUED)
3812 {
3813 /*
3814 * Always raise an interrupt after task completion; delaying
3815 * this (interrupt coalescing) increases latency and has a significant
3816 * impact on performance (see @bugref{5071})
3817 */
3818 ahciSendSDBFis(pAhciPort, 0, true);
3819 }
3820 else
3821 ahciSendD2HFis(pAhciPort, uTag, &cmdFis[0], true);
3822 }
3823 else
3824 {
3825 /*
3826 * Task was canceled, do the cleanup but DO NOT access the guest memory!
3827 * The guest might use it for other things now because it doesn't know about that task anymore.
3828 */
3829 fCanceled = true;
3830
3831 /* Leave a log message about the canceled request. */
3832 if (pAhciPort->cErrors++ < MAX_LOG_REL_ERRORS)
3833 {
3834 if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_FLUSH)
3835 LogRel(("AHCI#%uP%u: Canceled flush returned rc=%Rrc\n",
3836 pAhciPort->CTX_SUFF(pDevIns)->iInstance, pAhciPort->iLUN, rcReq));
3837 else if (pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_DISCARD)
3838 LogRel(("AHCI#%uP%u: Canceled trim returned rc=%Rrc\n",
3839 pAhciPort->CTX_SUFF(pDevIns)->iInstance,pAhciPort->iLUN, rcReq));
3840 else
3841 LogRel(("AHCI#%uP%u: Canceled %s at offset %llu (%zu bytes left) returned rc=%Rrc\n",
3842 pAhciPort->CTX_SUFF(pDevIns)->iInstance, pAhciPort->iLUN,
3843 pAhciReq->enmType == PDMMEDIAEXIOREQTYPE_READ
3844 ? "read"
3845 : "write",
3846 pAhciReq->uOffset,
3847 pAhciReq->cbTransfer, rcReq));
3848 }
3849
3850 ahciR3ReqFree(pAhciPort, pAhciReq);
3851 }
3852
3853 /*
3854 * Decrement the active task counter as the last step or we might run into a
3855 * hang during power off otherwise (see @bugref{7859}).
3856 * Before it could happen that we signal PDM that we are done while we still have to
3857 * copy the data to the guest but EMT might be busy destroying the driver chains
3858 * below us while we have to delegate copying data to EMT instead of doing it
3859 * on this thread.
3860 */
3861 ASMAtomicDecU32(&pAhciPort->cTasksActive);
3862
3863 if (pAhciPort->cTasksActive == 0 && pAhciPort->pAhciR3->fSignalIdle)
3864 PDMDevHlpAsyncNotificationCompleted(pAhciPort->pDevInsR3);
3865
3866 return fCanceled;
3867}
3868
3869/**
3870 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqCopyFromBuf}
3871 */
3872static DECLCALLBACK(int) ahciR3IoReqCopyFromBuf(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
3873 void *pvIoReqAlloc, uint32_t offDst, PRTSGBUF pSgBuf,
3874 size_t cbCopy)
3875{
3876 RT_NOREF1(hIoReq);
3877 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
3878 int rc = VINF_SUCCESS;
3879 PAHCIREQ pIoReq = (PAHCIREQ)pvIoReqAlloc;
3880
3881 ahciR3CopySgBufToPrdtl(pAhciPort->CTX_SUFF(pAhci), pIoReq, pSgBuf, offDst, cbCopy);
3882
3883 if (pIoReq->fFlags & AHCI_REQ_OVERFLOW)
3884 rc = VERR_PDM_MEDIAEX_IOBUF_OVERFLOW;
3885
3886 return rc;
3887}
3888
3889/**
3890 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqCopyToBuf}
3891 */
3892static DECLCALLBACK(int) ahciR3IoReqCopyToBuf(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
3893 void *pvIoReqAlloc, uint32_t offSrc, PRTSGBUF pSgBuf,
3894 size_t cbCopy)
3895{
3896 RT_NOREF1(hIoReq);
3897 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
3898 int rc = VINF_SUCCESS;
3899 PAHCIREQ pIoReq = (PAHCIREQ)pvIoReqAlloc;
3900
3901 ahciR3CopySgBufFromPrdtl(pAhciPort->CTX_SUFF(pAhci), pIoReq, pSgBuf, offSrc, cbCopy);
3902 if (pIoReq->fFlags & AHCI_REQ_OVERFLOW)
3903 rc = VERR_PDM_MEDIAEX_IOBUF_UNDERRUN;
3904
3905 return rc;
3906}
3907
3908/**
3909 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqQueryBuf}
3910 */
3911static DECLCALLBACK(int) ahciR3IoReqQueryBuf(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
3912 void *pvIoReqAlloc, void **ppvBuf, size_t *pcbBuf)
3913{
3914 RT_NOREF(hIoReq);
3915 int rc = VERR_NOT_SUPPORTED;
3916 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
3917 PAHCIREQ pIoReq = (PAHCIREQ)pvIoReqAlloc;
3918 PAHCI pThis = pAhciPort->CTX_SUFF(pAhci);
3919
3920 /* Only allow single 4KB page aligned buffers at the moment. */
3921 if ( pIoReq->cPrdtlEntries == 1
3922 && pIoReq->cbTransfer == _4K)
3923 {
3924 RTGCPHYS GCPhysPrdt = pIoReq->GCPhysPrdtl;
3925 SGLEntry PrdtEntry;
3926
3927 PDMDevHlpPhysRead(pThis->pDevInsR3, GCPhysPrdt, &PrdtEntry, sizeof(SGLEntry));
3928
3929 RTGCPHYS GCPhysAddrDataBase = AHCI_RTGCPHYS_FROM_U32(PrdtEntry.u32DBAUp, PrdtEntry.u32DBA);
3930 uint32_t cbData = (PrdtEntry.u32DescInf & SGLENTRY_DESCINF_DBC) + 1;
3931
3932 if ( cbData >= _4K
3933 && !(GCPhysAddrDataBase & (_4K - 1)))
3934 {
3935 rc = PDMDevHlpPhysGCPhys2CCPtr(pThis->pDevInsR3, GCPhysAddrDataBase,
3936 0, ppvBuf, &pIoReq->PgLck);
3937 if (RT_SUCCESS(rc))
3938 {
3939 pIoReq->fMapped = true;
3940 *pcbBuf = cbData;
3941 }
3942 else
3943 rc = VERR_NOT_SUPPORTED;
3944 }
3945 }
3946
3947 return rc;
3948}
3949
3950/**
3951 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqQueryDiscardRanges}
3952 */
3953static DECLCALLBACK(int) ahciR3IoReqQueryDiscardRanges(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
3954 void *pvIoReqAlloc, uint32_t idxRangeStart,
3955 uint32_t cRanges, PRTRANGE paRanges,
3956 uint32_t *pcRanges)
3957{
3958 RT_NOREF1(hIoReq);
3959 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
3960 PAHCIREQ pIoReq = (PAHCIREQ)pvIoReqAlloc;
3961
3962 return ahciTrimRangesCreate(pAhciPort, pIoReq, idxRangeStart, paRanges, cRanges, pcRanges);
3963}
3964
3965/**
3966 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqCompleteNotify}
3967 */
3968static DECLCALLBACK(int) ahciR3IoReqCompleteNotify(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
3969 void *pvIoReqAlloc, int rcReq)
3970{
3971 RT_NOREF(hIoReq);
3972 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
3973 ahciTransferComplete(pAhciPort, (PAHCIREQ)pvIoReqAlloc, rcReq);
3974 return VINF_SUCCESS;
3975}
3976
3977/**
3978 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqStateChanged}
3979 */
3980static DECLCALLBACK(void) ahciR3IoReqStateChanged(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
3981 void *pvIoReqAlloc, PDMMEDIAEXIOREQSTATE enmState)
3982{
3983 RT_NOREF2(hIoReq, pvIoReqAlloc);
3984 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
3985
3986 switch (enmState)
3987 {
3988 case PDMMEDIAEXIOREQSTATE_SUSPENDED:
3989 {
3990 /* Make sure the request is not accounted for so the VM can suspend successfully. */
3991 uint32_t cTasksActive = ASMAtomicDecU32(&pAhciPort->cTasksActive);
3992 if (!cTasksActive && pAhciPort->pAhciR3->fSignalIdle)
3993 PDMDevHlpAsyncNotificationCompleted(pAhciPort->pDevInsR3);
3994 break;
3995 }
3996 case PDMMEDIAEXIOREQSTATE_ACTIVE:
3997 /* Make sure the request is accounted for so the VM suspends only when the request is complete. */
3998 ASMAtomicIncU32(&pAhciPort->cTasksActive);
3999 break;
4000 default:
4001 AssertMsgFailed(("Invalid request state given %u\n", enmState));
4002 }
4003}
4004
4005/**
4006 * @interface_method_impl{PDMIMEDIAEXPORT,pfnMediumEjected}
4007 */
4008static DECLCALLBACK(void) ahciR3MediumEjected(PPDMIMEDIAEXPORT pInterface)
4009{
4010 PAHCIPort pAhciPort = RT_FROM_MEMBER(pInterface, AHCIPort, IMediaExPort);
4011 PAHCI pThis = pAhciPort->CTX_SUFF(pAhci);
4012
4013 if (pThis->pMediaNotify)
4014 {
4015 int rc = VMR3ReqCallNoWait(PDMDevHlpGetVM(pThis->CTX_SUFF(pDevIns)), VMCPUID_ANY,
4016 (PFNRT)pThis->pMediaNotify->pfnEjected, 2,
4017 pThis->pMediaNotify, pAhciPort->iLUN);
4018 AssertRC(rc);
4019 }
4020}
4021
4022/**
4023 * Process an non read/write ATA command.
4024 *
4025 * @returns The direction of the data transfer
4026 * @param pAhciPort The AHCI port of the request.
4027 * @param pAhciReq The AHCI request state.
4028 * @param pCmdFis Pointer to the command FIS.
4029 */
4030static PDMMEDIAEXIOREQTYPE ahciProcessCmd(PAHCIPort pAhciPort, PAHCIREQ pAhciReq, uint8_t *pCmdFis)
4031{
4032 PDMMEDIAEXIOREQTYPE enmType = PDMMEDIAEXIOREQTYPE_INVALID;
4033 bool fLBA48 = false;
4034
4035 AssertMsg(pCmdFis[AHCI_CMDFIS_TYPE] == AHCI_CMDFIS_TYPE_H2D, ("FIS is not a host to device Fis!!\n"));
4036
4037 pAhciReq->cbTransfer = 0;
4038
4039 switch (pCmdFis[AHCI_CMDFIS_CMD])
4040 {
4041 case ATA_IDENTIFY_DEVICE:
4042 {
4043 if (pAhciPort->pDrvMedia && !pAhciPort->fATAPI)
4044 {
4045 uint16_t u16Temp[256];
4046
4047 /* Fill the buffer. */
4048 ahciIdentifySS(pAhciPort, u16Temp);
4049
4050 /* Copy the buffer. */
4051 size_t cbCopied = ahciR3CopyBufferToPrdtl(pAhciPort->CTX_SUFF(pAhci), pAhciReq,
4052 &u16Temp[0], sizeof(u16Temp), 0 /* cbSkip */);
4053
4054 pAhciReq->fFlags |= AHCI_REQ_PIO_DATA;
4055 pAhciReq->cbTransfer = cbCopied;
4056 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
4057 }
4058 else
4059 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_SEEK | ATA_STAT_ERR);
4060 break;
4061 }
4062 case ATA_READ_NATIVE_MAX_ADDRESS_EXT:
4063 case ATA_READ_NATIVE_MAX_ADDRESS:
4064 break;
4065 case ATA_SET_FEATURES:
4066 {
4067 switch (pCmdFis[AHCI_CMDFIS_FET])
4068 {
4069 case 0x02: /* write cache enable */
4070 case 0xaa: /* read look-ahead enable */
4071 case 0x55: /* read look-ahead disable */
4072 case 0xcc: /* reverting to power-on defaults enable */
4073 case 0x66: /* reverting to power-on defaults disable */
4074 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
4075 break;
4076 case 0x82: /* write cache disable */
4077 enmType = PDMMEDIAEXIOREQTYPE_FLUSH;
4078 break;
4079 case 0x03:
4080 {
4081 /* set transfer mode */
4082 Log2(("%s: transfer mode %#04x\n", __FUNCTION__, pCmdFis[AHCI_CMDFIS_SECTC]));
4083 switch (pCmdFis[AHCI_CMDFIS_SECTC] & 0xf8)
4084 {
4085 case 0x00: /* PIO default */
4086 case 0x08: /* PIO mode */
4087 break;
4088 case ATA_MODE_MDMA: /* MDMA mode */
4089 pAhciPort->uATATransferMode = (pCmdFis[AHCI_CMDFIS_SECTC] & 0xf8) | RT_MIN(pCmdFis[AHCI_CMDFIS_SECTC] & 0x07, ATA_MDMA_MODE_MAX);
4090 break;
4091 case ATA_MODE_UDMA: /* UDMA mode */
4092 pAhciPort->uATATransferMode = (pCmdFis[AHCI_CMDFIS_SECTC] & 0xf8) | RT_MIN(pCmdFis[AHCI_CMDFIS_SECTC] & 0x07, ATA_UDMA_MODE_MAX);
4093 break;
4094 }
4095 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
4096 break;
4097 }
4098 default:
4099 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4100 }
4101 break;
4102 }
4103 case ATA_DEVICE_RESET:
4104 {
4105 if (!pAhciPort->fATAPI)
4106 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4107 else
4108 {
4109 /* Reset the device. */
4110 ahciDeviceReset(pAhciPort, pAhciReq);
4111 }
4112 break;
4113 }
4114 case ATA_FLUSH_CACHE_EXT:
4115 case ATA_FLUSH_CACHE:
4116 enmType = PDMMEDIAEXIOREQTYPE_FLUSH;
4117 break;
4118 case ATA_PACKET:
4119 if (!pAhciPort->fATAPI)
4120 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4121 else
4122 enmType = PDMMEDIAEXIOREQTYPE_SCSI;
4123 break;
4124 case ATA_IDENTIFY_PACKET_DEVICE:
4125 if (!pAhciPort->fATAPI)
4126 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4127 else
4128 {
4129 size_t cbData;
4130 ahciR3AtapiIdentify(pAhciReq, pAhciPort, 512, &cbData);
4131
4132 pAhciReq->fFlags |= AHCI_REQ_PIO_DATA;
4133 pAhciReq->cbTransfer = cbData;
4134 pAhciReq->cmdFis[AHCI_CMDFIS_SECTN] = (pAhciReq->cmdFis[AHCI_CMDFIS_SECTN] & ~7)
4135 | ((pAhciReq->fFlags & AHCI_REQ_XFER_2_HOST) ? ATAPI_INT_REASON_IO : 0)
4136 | (!pAhciReq->cbTransfer ? ATAPI_INT_REASON_CD : 0);
4137
4138 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
4139 }
4140 break;
4141 case ATA_SET_MULTIPLE_MODE:
4142 if ( pCmdFis[AHCI_CMDFIS_SECTC] != 0
4143 && ( pCmdFis[AHCI_CMDFIS_SECTC] > ATA_MAX_MULT_SECTORS
4144 || (pCmdFis[AHCI_CMDFIS_SECTC] & (pCmdFis[AHCI_CMDFIS_SECTC] - 1)) != 0))
4145 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4146 else
4147 {
4148 Log2(("%s: set multi sector count to %d\n", __FUNCTION__, pCmdFis[AHCI_CMDFIS_SECTC]));
4149 pAhciPort->cMultSectors = pCmdFis[AHCI_CMDFIS_SECTC];
4150 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
4151 }
4152 break;
4153 case ATA_STANDBY_IMMEDIATE:
4154 break; /* Do nothing. */
4155 case ATA_CHECK_POWER_MODE:
4156 pAhciReq->cmdFis[AHCI_CMDFIS_SECTC] = 0xff; /* drive active or idle */
4157 /* fall through */
4158 case ATA_INITIALIZE_DEVICE_PARAMETERS:
4159 case ATA_IDLE_IMMEDIATE:
4160 case ATA_RECALIBRATE:
4161 case ATA_NOP:
4162 case ATA_READ_VERIFY_SECTORS_EXT:
4163 case ATA_READ_VERIFY_SECTORS:
4164 case ATA_READ_VERIFY_SECTORS_WITHOUT_RETRIES:
4165 case ATA_SLEEP:
4166 ahciReqSetStatus(pAhciReq, 0, ATA_STAT_READY | ATA_STAT_SEEK);
4167 break;
4168 case ATA_READ_DMA_EXT:
4169 fLBA48 = true;
4170 /* fall through */
4171 case ATA_READ_DMA:
4172 {
4173 pAhciReq->cbTransfer = ahciGetNSectors(pCmdFis, fLBA48) * pAhciPort->cbSector;
4174 pAhciReq->uOffset = ahciGetSector(pAhciPort, pCmdFis, fLBA48) * pAhciPort->cbSector;
4175 enmType = PDMMEDIAEXIOREQTYPE_READ;
4176 break;
4177 }
4178 case ATA_WRITE_DMA_EXT:
4179 fLBA48 = true;
4180 /* fall through */
4181 case ATA_WRITE_DMA:
4182 {
4183 pAhciReq->cbTransfer = ahciGetNSectors(pCmdFis, fLBA48) * pAhciPort->cbSector;
4184 pAhciReq->uOffset = ahciGetSector(pAhciPort, pCmdFis, fLBA48) * pAhciPort->cbSector;
4185 enmType = PDMMEDIAEXIOREQTYPE_WRITE;
4186 break;
4187 }
4188 case ATA_READ_FPDMA_QUEUED:
4189 {
4190 pAhciReq->cbTransfer = ahciGetNSectorsQueued(pCmdFis) * pAhciPort->cbSector;
4191 pAhciReq->uOffset = ahciGetSectorQueued(pCmdFis) * pAhciPort->cbSector;
4192 pAhciReq->fFlags |= AHCI_REQ_IS_QUEUED;
4193 enmType = PDMMEDIAEXIOREQTYPE_READ;
4194 break;
4195 }
4196 case ATA_WRITE_FPDMA_QUEUED:
4197 {
4198 pAhciReq->cbTransfer = ahciGetNSectorsQueued(pCmdFis) * pAhciPort->cbSector;
4199 pAhciReq->uOffset = ahciGetSectorQueued(pCmdFis) * pAhciPort->cbSector;
4200 pAhciReq->fFlags |= AHCI_REQ_IS_QUEUED;
4201 enmType = PDMMEDIAEXIOREQTYPE_WRITE;
4202 break;
4203 }
4204 case ATA_READ_LOG_EXT:
4205 {
4206 size_t cbLogRead = ((pCmdFis[AHCI_CMDFIS_SECTCEXP] << 8) | pCmdFis[AHCI_CMDFIS_SECTC]) * 512;
4207 unsigned offLogRead = ((pCmdFis[AHCI_CMDFIS_CYLLEXP] << 8) | pCmdFis[AHCI_CMDFIS_CYLL]) * 512;
4208 unsigned iPage = pCmdFis[AHCI_CMDFIS_SECTN];
4209
4210 LogFlow(("Trying to read %zu bytes starting at offset %u from page %u\n", cbLogRead, offLogRead, iPage));
4211
4212 uint8_t aBuf[512];
4213
4214 memset(aBuf, 0, sizeof(aBuf));
4215
4216 if (offLogRead + cbLogRead <= sizeof(aBuf))
4217 {
4218 switch (iPage)
4219 {
4220 case 0x10:
4221 {
4222 LogFlow(("Reading error page\n"));
4223 PAHCIREQ pTaskErr = ASMAtomicXchgPtrT(&pAhciPort->pTaskErr, NULL, PAHCIREQ);
4224 if (pTaskErr)
4225 {
4226 aBuf[0] = (pTaskErr->fFlags & AHCI_REQ_IS_QUEUED) ? pTaskErr->uTag : (1 << 7);
4227 aBuf[2] = pTaskErr->cmdFis[AHCI_CMDFIS_STS];
4228 aBuf[3] = pTaskErr->cmdFis[AHCI_CMDFIS_ERR];
4229 aBuf[4] = pTaskErr->cmdFis[AHCI_CMDFIS_SECTN];
4230 aBuf[5] = pTaskErr->cmdFis[AHCI_CMDFIS_CYLL];
4231 aBuf[6] = pTaskErr->cmdFis[AHCI_CMDFIS_CYLH];
4232 aBuf[7] = pTaskErr->cmdFis[AHCI_CMDFIS_HEAD];
4233 aBuf[8] = pTaskErr->cmdFis[AHCI_CMDFIS_SECTNEXP];
4234 aBuf[9] = pTaskErr->cmdFis[AHCI_CMDFIS_CYLLEXP];
4235 aBuf[10] = pTaskErr->cmdFis[AHCI_CMDFIS_CYLHEXP];
4236 aBuf[12] = pTaskErr->cmdFis[AHCI_CMDFIS_SECTC];
4237 aBuf[13] = pTaskErr->cmdFis[AHCI_CMDFIS_SECTCEXP];
4238
4239 /* Calculate checksum */
4240 uint8_t uChkSum = 0;
4241 for (unsigned i = 0; i < RT_ELEMENTS(aBuf)-1; i++)
4242 uChkSum += aBuf[i];
4243
4244 aBuf[511] = (uint8_t)-(int8_t)uChkSum;
4245
4246 /* Finally free the error task state structure because it is completely unused now. */
4247 RTMemFree(pTaskErr);
4248 }
4249
4250 /*
4251 * Reading this log page results in an abort of all outstanding commands
4252 * and clearing the SActive register and TaskFile register.
4253 *
4254 * See SATA2 1.2 spec chapter 4.2.3.4
4255 */
4256 bool fAbortedAll = ahciCancelActiveTasks(pAhciPort);
4257 Assert(fAbortedAll); NOREF(fAbortedAll);
4258 ahciSendSDBFis(pAhciPort, 0xffffffff, true);
4259
4260 break;
4261 }
4262 }
4263
4264 /* Copy the buffer. */
4265 size_t cbCopied = ahciR3CopyBufferToPrdtl(pAhciPort->CTX_SUFF(pAhci), pAhciReq,
4266 &aBuf[offLogRead], cbLogRead, 0 /* cbSkip */);
4267
4268 pAhciReq->fFlags |= AHCI_REQ_PIO_DATA;
4269 pAhciReq->cbTransfer = cbCopied;
4270 }
4271
4272 break;
4273 }
4274 case ATA_DATA_SET_MANAGEMENT:
4275 {
4276 if (pAhciPort->fTrimEnabled)
4277 {
4278 /* Check that the trim bit is set and all other bits are 0. */
4279 if ( !(pAhciReq->cmdFis[AHCI_CMDFIS_FET] & UINT16_C(0x01))
4280 || (pAhciReq->cmdFis[AHCI_CMDFIS_FET] & ~UINT16_C(0x1)))
4281 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4282 else
4283 enmType = PDMMEDIAEXIOREQTYPE_DISCARD;
4284 break;
4285 }
4286 /* else: fall through and report error to the guest. */
4287 }
4288 /* fall thru */
4289 /* All not implemented commands go below. */
4290 case ATA_SECURITY_FREEZE_LOCK:
4291 case ATA_SMART:
4292 case ATA_NV_CACHE:
4293 case ATA_IDLE:
4294 case ATA_TRUSTED_RECEIVE_DMA: /* Windows 8+ */
4295 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4296 break;
4297 default: /* For debugging purposes. */
4298 AssertMsgFailed(("Unknown command issued (%#x)\n", pCmdFis[AHCI_CMDFIS_CMD]));
4299 ahciReqSetStatus(pAhciReq, ABRT_ERR, ATA_STAT_READY | ATA_STAT_ERR);
4300 }
4301
4302 return enmType;
4303}
4304
4305/**
4306 * Retrieve a command FIS from guest memory.
4307 *
4308 * @returns whether the H2D FIS was successfully read from the guest memory.
4309 * @param pAhciPort The AHCI port of the request.
4310 * @param pAhciReq The state of the actual task.
4311 */
4312static bool ahciPortTaskGetCommandFis(PAHCIPort pAhciPort, PAHCIREQ pAhciReq)
4313{
4314 AssertMsgReturn(pAhciPort->GCPhysAddrClb && pAhciPort->GCPhysAddrFb,
4315 ("%s: GCPhysAddrClb and/or GCPhysAddrFb are 0\n", __FUNCTION__),
4316 false);
4317
4318 /*
4319 * First we are reading the command header pointed to by regCLB.
4320 * From this we get the address of the command table which we are reading too.
4321 * We can process the Command FIS afterwards.
4322 */
4323 CmdHdr cmdHdr;
4324 pAhciReq->GCPhysCmdHdrAddr = pAhciPort->GCPhysAddrClb + pAhciReq->uTag * sizeof(CmdHdr);
4325 LogFlow(("%s: PDMDevHlpPhysRead GCPhysAddrCmdLst=%RGp cbCmdHdr=%u\n", __FUNCTION__,
4326 pAhciReq->GCPhysCmdHdrAddr, sizeof(CmdHdr)));
4327 PDMDevHlpPhysRead(pAhciPort->CTX_SUFF(pDevIns), pAhciReq->GCPhysCmdHdrAddr, &cmdHdr, sizeof(CmdHdr));
4328
4329#ifdef LOG_ENABLED
4330 /* Print some infos about the command header. */
4331 ahciDumpCmdHdrInfo(pAhciPort, &cmdHdr);
4332#endif
4333
4334 RTGCPHYS GCPhysAddrCmdTbl = AHCI_RTGCPHYS_FROM_U32(cmdHdr.u32CmdTblAddrUp, cmdHdr.u32CmdTblAddr);
4335
4336 AssertMsgReturn((cmdHdr.u32DescInf & AHCI_CMDHDR_CFL_MASK) * sizeof(uint32_t) == AHCI_CMDFIS_TYPE_H2D_SIZE,
4337 ("This is not a command FIS!!\n"),
4338 false);
4339
4340 /* Read the command Fis. */
4341 LogFlow(("%s: PDMDevHlpPhysRead GCPhysAddrCmdTbl=%RGp cbCmdFis=%u\n", __FUNCTION__, GCPhysAddrCmdTbl, AHCI_CMDFIS_TYPE_H2D_SIZE));
4342 PDMDevHlpPhysRead(pAhciPort->CTX_SUFF(pDevIns), GCPhysAddrCmdTbl, &pAhciReq->cmdFis[0], AHCI_CMDFIS_TYPE_H2D_SIZE);
4343
4344 AssertMsgReturn(pAhciReq->cmdFis[AHCI_CMDFIS_TYPE] == AHCI_CMDFIS_TYPE_H2D,
4345 ("This is not a command FIS\n"),
4346 false);
4347
4348 /* Set transfer direction. */
4349 pAhciReq->fFlags |= (cmdHdr.u32DescInf & AHCI_CMDHDR_W) ? 0 : AHCI_REQ_XFER_2_HOST;
4350
4351 /* If this is an ATAPI command read the atapi command. */
4352 if (cmdHdr.u32DescInf & AHCI_CMDHDR_A)
4353 {
4354 GCPhysAddrCmdTbl += AHCI_CMDHDR_ACMD_OFFSET;
4355 PDMDevHlpPhysRead(pAhciPort->CTX_SUFF(pDevIns), GCPhysAddrCmdTbl, &pAhciReq->aATAPICmd[0], ATAPI_PACKET_SIZE);
4356 }
4357
4358 /* We "received" the FIS. Clear the BSY bit in regTFD. */
4359 if ((cmdHdr.u32DescInf & AHCI_CMDHDR_C) && (pAhciReq->fFlags & AHCI_REQ_CLEAR_SACT))
4360 {
4361 /*
4362 * We need to send a FIS which clears the busy bit if this is a queued command so that the guest can queue other commands.
4363 * but this FIS does not assert an interrupt
4364 */
4365 ahciSendD2HFis(pAhciPort, pAhciReq->uTag, pAhciReq->cmdFis, false);
4366 pAhciPort->regTFD &= ~AHCI_PORT_TFD_BSY;
4367 }
4368
4369 pAhciReq->GCPhysPrdtl = AHCI_RTGCPHYS_FROM_U32(cmdHdr.u32CmdTblAddrUp, cmdHdr.u32CmdTblAddr) + AHCI_CMDHDR_PRDT_OFFSET;
4370 pAhciReq->cPrdtlEntries = AHCI_CMDHDR_PRDTL_ENTRIES(cmdHdr.u32DescInf);
4371
4372#ifdef LOG_ENABLED
4373 /* Print some infos about the FIS. */
4374 ahciDumpFisInfo(pAhciPort, &pAhciReq->cmdFis[0]);
4375
4376 /* Print the PRDT */
4377 ahciLog(("PRDT address %RGp number of entries %u\n", pAhciReq->GCPhysPrdtl, pAhciReq->cPrdtlEntries));
4378 RTGCPHYS GCPhysPrdtl = pAhciReq->GCPhysPrdtl;
4379
4380 for (unsigned i = 0; i < pAhciReq->cPrdtlEntries; i++)
4381 {
4382 SGLEntry SGEntry;
4383
4384 ahciLog(("Entry %u at address %RGp\n", i, GCPhysPrdtl));
4385 PDMDevHlpPhysRead(pAhciPort->CTX_SUFF(pDevIns), GCPhysPrdtl, &SGEntry, sizeof(SGLEntry));
4386
4387 RTGCPHYS GCPhysDataAddr = AHCI_RTGCPHYS_FROM_U32(SGEntry.u32DBAUp, SGEntry.u32DBA);
4388 ahciLog(("GCPhysAddr=%RGp Size=%u\n", GCPhysDataAddr, SGEntry.u32DescInf & SGLENTRY_DESCINF_DBC));
4389
4390 GCPhysPrdtl += sizeof(SGLEntry);
4391 }
4392#endif
4393
4394 return true;
4395}
4396
4397/**
4398 * Submits a given request for execution.
4399 *
4400 * @returns Flag whether the request was canceled inbetween.
4401 * @param pAhciPort The port the request is for.
4402 * @param pAhciReq The request to submit.
4403 * @param enmType The request type.
4404 */
4405static bool ahciR3ReqSubmit(PAHCIPort pAhciPort, PAHCIREQ pAhciReq, PDMMEDIAEXIOREQTYPE enmType)
4406{
4407 int rc = VINF_SUCCESS;
4408 bool fReqCanceled = false;
4409
4410 VBOXDD_AHCI_REQ_SUBMIT(pAhciReq, pAhciReq->enmType, pAhciReq->uOffset, pAhciReq->cbTransfer);
4411
4412 if (enmType == PDMMEDIAEXIOREQTYPE_FLUSH)
4413 rc = pAhciPort->pDrvMediaEx->pfnIoReqFlush(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq);
4414 else if (enmType == PDMMEDIAEXIOREQTYPE_DISCARD)
4415 {
4416 uint32_t cRangesMax;
4417
4418 /* The data buffer contains LBA range entries. Each range is 8 bytes big. */
4419 if (!pAhciReq->cmdFis[AHCI_CMDFIS_SECTC] && !pAhciReq->cmdFis[AHCI_CMDFIS_SECTCEXP])
4420 cRangesMax = 65536 * 512 / 8;
4421 else
4422 cRangesMax = pAhciReq->cmdFis[AHCI_CMDFIS_SECTC] * 512 / 8;
4423
4424 pAhciPort->Led.Asserted.s.fWriting = pAhciPort->Led.Actual.s.fWriting = 1;
4425 rc = pAhciPort->pDrvMediaEx->pfnIoReqDiscard(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq,
4426 cRangesMax);
4427 }
4428 else if (enmType == PDMMEDIAEXIOREQTYPE_READ)
4429 {
4430 pAhciPort->Led.Asserted.s.fReading = pAhciPort->Led.Actual.s.fReading = 1;
4431 rc = pAhciPort->pDrvMediaEx->pfnIoReqRead(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq,
4432 pAhciReq->uOffset, pAhciReq->cbTransfer);
4433 }
4434 else if (enmType == PDMMEDIAEXIOREQTYPE_WRITE)
4435 {
4436 pAhciPort->Led.Asserted.s.fWriting = pAhciPort->Led.Actual.s.fWriting = 1;
4437 rc = pAhciPort->pDrvMediaEx->pfnIoReqWrite(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq,
4438 pAhciReq->uOffset, pAhciReq->cbTransfer);
4439 }
4440 else if (enmType == PDMMEDIAEXIOREQTYPE_SCSI)
4441 {
4442 size_t cbBuf = 0;
4443
4444 if (pAhciReq->cPrdtlEntries)
4445 rc = ahciR3PrdtQuerySize(pAhciPort->CTX_SUFF(pAhci), pAhciReq, &cbBuf);
4446 pAhciReq->cbTransfer = cbBuf;
4447 if (RT_SUCCESS(rc))
4448 {
4449 if (cbBuf && (pAhciReq->fFlags & AHCI_REQ_XFER_2_HOST))
4450 pAhciPort->Led.Asserted.s.fReading = pAhciPort->Led.Actual.s.fReading = 1;
4451 else if (cbBuf)
4452 pAhciPort->Led.Asserted.s.fWriting = pAhciPort->Led.Actual.s.fWriting = 1;
4453 rc = pAhciPort->pDrvMediaEx->pfnIoReqSendScsiCmd(pAhciPort->pDrvMediaEx, pAhciReq->hIoReq,
4454 0, &pAhciReq->aATAPICmd[0], ATAPI_PACKET_SIZE,
4455 PDMMEDIAEXIOREQSCSITXDIR_UNKNOWN, cbBuf,
4456 &pAhciPort->abATAPISense[0], sizeof(pAhciPort->abATAPISense),
4457 &pAhciReq->u8ScsiSts, 30 * RT_MS_1SEC);
4458 }
4459 }
4460
4461 if (rc == VINF_SUCCESS)
4462 fReqCanceled = ahciTransferComplete(pAhciPort, pAhciReq, VINF_SUCCESS);
4463 else if (rc != VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS)
4464 fReqCanceled = ahciTransferComplete(pAhciPort, pAhciReq, rc);
4465
4466 return fReqCanceled;
4467}
4468
4469/**
4470 * Prepares the command for execution coping it from guest memory and doing a few
4471 * validation checks on it.
4472 *
4473 * @returns Whether the command was successfully fetched from guest memory and
4474 * can be continued.
4475 * @param pAhciPort The AHCI port the request is for.
4476 * @param pAhciReq Request structure to copy the command to.
4477 */
4478static bool ahciR3CmdPrepare(PAHCIPort pAhciPort, PAHCIREQ pAhciReq)
4479{
4480 /* Set current command slot */
4481 ASMAtomicWriteU32(&pAhciPort->u32CurrentCommandSlot, pAhciReq->uTag);
4482
4483 bool fContinue = ahciPortTaskGetCommandFis(pAhciPort, pAhciReq);
4484 if (fContinue)
4485 {
4486 /* Mark the task as processed by the HBA if this is a queued task so that it doesn't occur in the CI register anymore. */
4487 if (pAhciPort->regSACT & RT_BIT_32(pAhciReq->uTag))
4488 {
4489 pAhciReq->fFlags |= AHCI_REQ_CLEAR_SACT;
4490 ASMAtomicOrU32(&pAhciPort->u32TasksFinished, RT_BIT_32(pAhciReq->uTag));
4491 }
4492
4493 if (pAhciReq->cmdFis[AHCI_CMDFIS_BITS] & AHCI_CMDFIS_C)
4494 {
4495 /*
4496 * It is possible that the request counter can get one higher than the maximum because
4497 * the request counter is decremented after the guest was notified about the completed
4498 * request (see @bugref{7859}). If the completing thread is preempted in between the
4499 * guest might already issue another request before the request counter is decremented
4500 * which would trigger the following assertion incorrectly in the past.
4501 */
4502 AssertLogRelMsg(ASMAtomicReadU32(&pAhciPort->cTasksActive) <= AHCI_NR_COMMAND_SLOTS,
4503 ("AHCI#%uP%u: There are more than %u (+1) requests active",
4504 pAhciPort->CTX_SUFF(pDevIns)->iInstance, pAhciPort->iLUN,
4505 AHCI_NR_COMMAND_SLOTS));
4506 ASMAtomicIncU32(&pAhciPort->cTasksActive);
4507 }
4508 else
4509 {
4510 /* If the reset bit is set put the device into reset state. */
4511 if (pAhciReq->cmdFis[AHCI_CMDFIS_CTL] & AHCI_CMDFIS_CTL_SRST)
4512 {
4513 ahciLog(("%s: Setting device into reset state\n", __FUNCTION__));
4514 pAhciPort->fResetDevice = true;
4515 ahciSendD2HFis(pAhciPort, pAhciReq->uTag, pAhciReq->cmdFis, true);
4516 }
4517 else if (pAhciPort->fResetDevice) /* The bit is not set and we are in a reset state. */
4518 ahciFinishStorageDeviceReset(pAhciPort, pAhciReq);
4519 else /* We are not in a reset state update the control registers. */
4520 AssertMsgFailed(("%s: Update the control register\n", __FUNCTION__));
4521
4522 fContinue = false;
4523 }
4524 }
4525 else
4526 {
4527 /*
4528 * Couldn't find anything in either the AHCI or SATA spec which
4529 * indicates what should be done if the FIS is not read successfully.
4530 * The closest thing is in the state machine, stating that the device
4531 * should go into idle state again (SATA spec 1.0 chapter 8.7.1).
4532 * Do the same here and ignore any corrupt FIS types, after all
4533 * the guest messed up everything and this behavior is undefined.
4534 */
4535 fContinue = false;
4536 }
4537
4538 return fContinue;
4539}
4540
4541/**
4542 * Transmit queue consumer
4543 * Queue a new async task.
4544 *
4545 * @returns Success indicator.
4546 * If false the item will not be removed and the flushing will stop.
4547 * @param pDevIns The device instance.
4548 * @param pItem The item to consume. Upon return this item will be freed.
4549 */
4550static DECLCALLBACK(bool) ahciNotifyQueueConsumer(PPDMDEVINS pDevIns, PPDMQUEUEITEMCORE pItem)
4551{
4552 PDEVPORTNOTIFIERQUEUEITEM pNotifierItem = (PDEVPORTNOTIFIERQUEUEITEM)pItem;
4553 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
4554 PAHCIPort pAhciPort = &pThis->ahciPort[pNotifierItem->iPort];
4555 int rc = VINF_SUCCESS;
4556
4557 ahciLog(("%s: Got notification from GC\n", __FUNCTION__));
4558 /* Notify the async IO thread. */
4559 rc = SUPSemEventSignal(pThis->pSupDrvSession, pAhciPort->hEvtProcess);
4560 AssertRC(rc);
4561
4562 return true;
4563}
4564
4565/* The async IO thread for one port. */
4566static DECLCALLBACK(int) ahciAsyncIOLoop(PPDMDEVINS pDevIns, PPDMTHREAD pThread)
4567{
4568 RT_NOREF(pDevIns);
4569 PAHCIPort pAhciPort = (PAHCIPort)pThread->pvUser;
4570 PAHCI pAhci = pAhciPort->CTX_SUFF(pAhci);
4571 int rc = VINF_SUCCESS;
4572
4573 ahciLog(("%s: Port %d entering async IO loop.\n", __FUNCTION__, pAhciPort->iLUN));
4574
4575 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
4576 return VINF_SUCCESS;
4577
4578 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
4579 {
4580 unsigned idx = 0;
4581 uint32_t u32Tasks = 0;
4582 uint32_t u32RegHbaCtrl = 0;
4583
4584 ASMAtomicWriteBool(&pAhciPort->fWrkThreadSleeping, true);
4585 u32Tasks = ASMAtomicXchgU32(&pAhciPort->u32TasksNew, 0);
4586 if (!u32Tasks)
4587 {
4588 Assert(ASMAtomicReadBool(&pAhciPort->fWrkThreadSleeping));
4589 rc = SUPSemEventWaitNoResume(pAhci->pSupDrvSession, pAhciPort->hEvtProcess, RT_INDEFINITE_WAIT);
4590 AssertLogRelMsgReturn(RT_SUCCESS(rc) || rc == VERR_INTERRUPTED, ("%Rrc\n", rc), rc);
4591 if (RT_UNLIKELY(pThread->enmState != PDMTHREADSTATE_RUNNING))
4592 break;
4593 LogFlowFunc(("Woken up with rc=%Rrc\n", rc));
4594 u32Tasks = ASMAtomicXchgU32(&pAhciPort->u32TasksNew, 0);
4595 }
4596
4597 ASMAtomicWriteBool(&pAhciPort->fWrkThreadSleeping, false);
4598 ASMAtomicIncU32(&pAhci->cThreadsActive);
4599
4600 /* Check whether the thread should be suspended. */
4601 if (pAhci->fSignalIdle)
4602 {
4603 if (!ASMAtomicDecU32(&pAhci->cThreadsActive))
4604 PDMDevHlpAsyncNotificationCompleted(pAhciPort->pDevInsR3);
4605 continue;
4606 }
4607
4608 /*
4609 * Check whether the global host controller bit is set and go to sleep immediately again
4610 * if it is set.
4611 */
4612 u32RegHbaCtrl = ASMAtomicReadU32(&pAhci->regHbaCtrl);
4613 if ( u32RegHbaCtrl & AHCI_HBA_CTRL_HR
4614 && !ASMAtomicDecU32(&pAhci->cThreadsActive))
4615 {
4616 ahciHBAReset(pAhci);
4617 if (pAhci->fSignalIdle)
4618 PDMDevHlpAsyncNotificationCompleted(pAhciPort->pDevInsR3);
4619 continue;
4620 }
4621
4622 idx = ASMBitFirstSetU32(u32Tasks);
4623 while ( idx
4624 && !pAhciPort->fPortReset)
4625 {
4626 bool fReqCanceled = false;
4627
4628 /* Decrement to get the slot number. */
4629 idx--;
4630 ahciLog(("%s: Processing command at slot %d\n", __FUNCTION__, idx));
4631
4632 PAHCIREQ pAhciReq = ahciR3ReqAlloc(pAhciPort, idx);
4633 if (RT_LIKELY(pAhciReq))
4634 {
4635 pAhciReq->uTag = idx;
4636 pAhciReq->fFlags = 0;
4637
4638 bool fContinue = ahciR3CmdPrepare(pAhciPort, pAhciReq);
4639 if (fContinue)
4640 {
4641 PDMMEDIAEXIOREQTYPE enmType = ahciProcessCmd(pAhciPort, pAhciReq, pAhciReq->cmdFis);
4642 pAhciReq->enmType = enmType;
4643
4644 if (enmType != PDMMEDIAEXIOREQTYPE_INVALID)
4645 fReqCanceled = ahciR3ReqSubmit(pAhciPort, pAhciReq, enmType);
4646 else
4647 fReqCanceled = ahciTransferComplete(pAhciPort, pAhciReq, VINF_SUCCESS);
4648 } /* Command */
4649 else
4650 ahciR3ReqFree(pAhciPort, pAhciReq);
4651 }
4652 else /* !Request allocated, use on stack variant to signal the error. */
4653 {
4654 AHCIREQ Req;
4655 Req.uTag = idx;
4656 Req.fFlags = AHCI_REQ_IS_ON_STACK;
4657 Req.fMapped = false;
4658 Req.cbTransfer = 0;
4659 Req.uOffset = 0;
4660 Req.enmType = PDMMEDIAEXIOREQTYPE_INVALID;
4661
4662 bool fContinue = ahciR3CmdPrepare(pAhciPort, &Req);
4663 if (fContinue)
4664 fReqCanceled = ahciTransferComplete(pAhciPort, &Req, VERR_NO_MEMORY);
4665 }
4666
4667 /*
4668 * Don't process other requests if the last one was canceled,
4669 * the others are not valid anymore.
4670 */
4671 if (fReqCanceled)
4672 break;
4673
4674 u32Tasks &= ~RT_BIT_32(idx); /* Clear task bit. */
4675 idx = ASMBitFirstSetU32(u32Tasks);
4676 } /* while tasks available */
4677
4678 /* Check whether a port reset was active. */
4679 if ( ASMAtomicReadBool(&pAhciPort->fPortReset)
4680 && (pAhciPort->regSCTL & AHCI_PORT_SCTL_DET) == AHCI_PORT_SCTL_DET_NINIT)
4681 ahciPortResetFinish(pAhciPort);
4682
4683 /*
4684 * Check whether a host controller reset is pending and execute the reset
4685 * if this is the last active thread.
4686 */
4687 u32RegHbaCtrl = ASMAtomicReadU32(&pAhci->regHbaCtrl);
4688 uint32_t cThreadsActive = ASMAtomicDecU32(&pAhci->cThreadsActive);
4689 if ( (u32RegHbaCtrl & AHCI_HBA_CTRL_HR)
4690 && !cThreadsActive)
4691 ahciHBAReset(pAhci);
4692
4693 if (!cThreadsActive && pAhci->fSignalIdle)
4694 PDMDevHlpAsyncNotificationCompleted(pAhciPort->pDevInsR3);
4695 } /* While running */
4696
4697 ahciLog(("%s: Port %d async IO thread exiting\n", __FUNCTION__, pAhciPort->iLUN));
4698 return VINF_SUCCESS;
4699}
4700
4701/**
4702 * Unblock the async I/O thread so it can respond to a state change.
4703 *
4704 * @returns VBox status code.
4705 * @param pDevIns The device instance.
4706 * @param pThread The send thread.
4707 */
4708static DECLCALLBACK(int) ahciAsyncIOLoopWakeUp(PPDMDEVINS pDevIns, PPDMTHREAD pThread)
4709{
4710 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
4711 PAHCIPort pAhciPort = (PAHCIPort)pThread->pvUser;
4712 return SUPSemEventSignal(pThis->pSupDrvSession, pAhciPort->hEvtProcess);
4713}
4714
4715/* -=-=-=-=- DBGF -=-=-=-=- */
4716
4717/**
4718 * AHCI status info callback.
4719 *
4720 * @param pDevIns The device instance.
4721 * @param pHlp The output helpers.
4722 * @param pszArgs The arguments.
4723 */
4724static DECLCALLBACK(void) ahciR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
4725{
4726 RT_NOREF(pszArgs);
4727 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
4728
4729 /*
4730 * Show info.
4731 */
4732 pHlp->pfnPrintf(pHlp,
4733 "%s#%d: mmio=%RGp ports=%u GC=%RTbool R0=%RTbool\n",
4734 pDevIns->pReg->szName,
4735 pDevIns->iInstance,
4736 pThis->MMIOBase,
4737 pThis->cPortsImpl,
4738 pThis->fGCEnabled ? true : false,
4739 pThis->fR0Enabled ? true : false);
4740
4741 /*
4742 * Show global registers.
4743 */
4744 pHlp->pfnPrintf(pHlp, "HbaCap=%#x\n", pThis->regHbaCap);
4745 pHlp->pfnPrintf(pHlp, "HbaCtrl=%#x\n", pThis->regHbaCtrl);
4746 pHlp->pfnPrintf(pHlp, "HbaIs=%#x\n", pThis->regHbaIs);
4747 pHlp->pfnPrintf(pHlp, "HbaPi=%#x", pThis->regHbaPi);
4748 pHlp->pfnPrintf(pHlp, "HbaVs=%#x\n", pThis->regHbaVs);
4749 pHlp->pfnPrintf(pHlp, "HbaCccCtl=%#x\n", pThis->regHbaCccCtl);
4750 pHlp->pfnPrintf(pHlp, "HbaCccPorts=%#x\n", pThis->regHbaCccPorts);
4751 pHlp->pfnPrintf(pHlp, "PortsInterrupted=%#x\n", pThis->u32PortsInterrupted);
4752
4753 /*
4754 * Per port data.
4755 */
4756 for (unsigned i = 0; i < pThis->cPortsImpl; i++)
4757 {
4758 PAHCIPort pThisPort = &pThis->ahciPort[i];
4759
4760 pHlp->pfnPrintf(pHlp, "Port %d: device-attached=%RTbool\n",
4761 pThisPort->iLUN, pThisPort->pDrvBase != NULL);
4762 pHlp->pfnPrintf(pHlp, "PortClb=%#x\n", pThisPort->regCLB);
4763 pHlp->pfnPrintf(pHlp, "PortClbU=%#x\n", pThisPort->regCLBU);
4764 pHlp->pfnPrintf(pHlp, "PortFb=%#x\n", pThisPort->regFB);
4765 pHlp->pfnPrintf(pHlp, "PortFbU=%#x\n", pThisPort->regFBU);
4766 pHlp->pfnPrintf(pHlp, "PortIs=%#x\n", pThisPort->regIS);
4767 pHlp->pfnPrintf(pHlp, "PortIe=%#x\n", pThisPort->regIE);
4768 pHlp->pfnPrintf(pHlp, "PortCmd=%#x\n", pThisPort->regCMD);
4769 pHlp->pfnPrintf(pHlp, "PortTfd=%#x\n", pThisPort->regTFD);
4770 pHlp->pfnPrintf(pHlp, "PortSig=%#x\n", pThisPort->regSIG);
4771 pHlp->pfnPrintf(pHlp, "PortSSts=%#x\n", pThisPort->regSSTS);
4772 pHlp->pfnPrintf(pHlp, "PortSCtl=%#x\n", pThisPort->regSCTL);
4773 pHlp->pfnPrintf(pHlp, "PortSErr=%#x\n", pThisPort->regSERR);
4774 pHlp->pfnPrintf(pHlp, "PortSAct=%#x\n", pThisPort->regSACT);
4775 pHlp->pfnPrintf(pHlp, "PortCi=%#x\n", pThisPort->regCI);
4776 pHlp->pfnPrintf(pHlp, "PortPhysClb=%RGp\n", pThisPort->GCPhysAddrClb);
4777 pHlp->pfnPrintf(pHlp, "PortPhysFb=%RGp\n", pThisPort->GCPhysAddrFb);
4778 pHlp->pfnPrintf(pHlp, "PortActTasksActive=%u\n", pThisPort->cTasksActive);
4779 pHlp->pfnPrintf(pHlp, "PortPoweredOn=%RTbool\n", pThisPort->fPoweredOn);
4780 pHlp->pfnPrintf(pHlp, "PortSpunUp=%RTbool\n", pThisPort->fSpunUp);
4781 pHlp->pfnPrintf(pHlp, "PortFirstD2HFisSend=%RTbool\n", pThisPort->fFirstD2HFisSend);
4782 pHlp->pfnPrintf(pHlp, "PortATAPI=%RTbool\n", pThisPort->fATAPI);
4783 pHlp->pfnPrintf(pHlp, "PortTasksFinished=%#x\n", pThisPort->u32TasksFinished);
4784 pHlp->pfnPrintf(pHlp, "PortQueuedTasksFinished=%#x\n", pThisPort->u32QueuedTasksFinished);
4785 pHlp->pfnPrintf(pHlp, "PortTasksNew=%#x\n", pThisPort->u32TasksNew);
4786 pHlp->pfnPrintf(pHlp, "\n");
4787 }
4788}
4789
4790/* -=-=-=-=- Helper -=-=-=-=- */
4791
4792/**
4793 * Checks if all asynchronous I/O is finished, both AHCI and IDE.
4794 *
4795 * Used by ahciR3Reset, ahciR3Suspend and ahciR3PowerOff. ahciR3SavePrep makes
4796 * use of it in strict builds (which is why it's up here).
4797 *
4798 * @returns true if quiesced, false if busy.
4799 * @param pDevIns The device instance.
4800 */
4801static bool ahciR3AllAsyncIOIsFinished(PPDMDEVINS pDevIns)
4802{
4803 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
4804
4805 if (pThis->cThreadsActive)
4806 return false;
4807
4808 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->ahciPort); i++)
4809 {
4810 PAHCIPort pThisPort = &pThis->ahciPort[i];
4811 if (pThisPort->pDrvBase)
4812 {
4813 if ( (pThisPort->cTasksActive != 0)
4814 || (pThisPort->u32TasksNew != 0))
4815 return false;
4816 }
4817 }
4818 return true;
4819}
4820
4821/* -=-=-=-=- Saved State -=-=-=-=- */
4822
4823/**
4824 * @callback_method_impl{FNSSMDEVSAVEPREP}
4825 */
4826static DECLCALLBACK(int) ahciR3SavePrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4827{
4828 RT_NOREF(pDevIns, pSSM);
4829 Assert(ahciR3AllAsyncIOIsFinished(pDevIns));
4830 return VINF_SUCCESS;
4831}
4832
4833/**
4834 * @callback_method_impl{FNSSMDEVLOADPREP}
4835 */
4836static DECLCALLBACK(int) ahciR3LoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4837{
4838 RT_NOREF(pDevIns, pSSM);
4839 Assert(ahciR3AllAsyncIOIsFinished(pDevIns));
4840 return VINF_SUCCESS;
4841}
4842
4843/**
4844 * @callback_method_impl{FNSSMDEVLIVEEXEC}
4845 */
4846static DECLCALLBACK(int) ahciR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
4847{
4848 RT_NOREF(uPass);
4849 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
4850
4851 /* config. */
4852 SSMR3PutU32(pSSM, pThis->cPortsImpl);
4853 for (uint32_t i = 0; i < AHCI_MAX_NR_PORTS_IMPL; i++)
4854 {
4855 SSMR3PutBool(pSSM, pThis->ahciPort[i].pDrvBase != NULL);
4856 SSMR3PutBool(pSSM, pThis->ahciPort[i].fHotpluggable);
4857 SSMR3PutStrZ(pSSM, pThis->ahciPort[i].szSerialNumber);
4858 SSMR3PutStrZ(pSSM, pThis->ahciPort[i].szFirmwareRevision);
4859 SSMR3PutStrZ(pSSM, pThis->ahciPort[i].szModelNumber);
4860 }
4861
4862 static const char *s_apszIdeEmuPortNames[4] = { "PrimaryMaster", "PrimarySlave", "SecondaryMaster", "SecondarySlave" };
4863 for (uint32_t i = 0; i < RT_ELEMENTS(s_apszIdeEmuPortNames); i++)
4864 {
4865 uint32_t iPort;
4866 int rc = CFGMR3QueryU32Def(pDevIns->pCfg, s_apszIdeEmuPortNames[i], &iPort, i);
4867 AssertRCReturn(rc, rc);
4868 SSMR3PutU32(pSSM, iPort);
4869 }
4870
4871 return VINF_SSM_DONT_CALL_AGAIN;
4872}
4873
4874/**
4875 * @callback_method_impl{FNSSMDEVSAVEEXEC}
4876 */
4877static DECLCALLBACK(int) ahciR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4878{
4879 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
4880 uint32_t i;
4881 int rc;
4882
4883 Assert(!pThis->f8ByteMMIO4BytesWrittenSuccessfully);
4884
4885 /* The config */
4886 rc = ahciR3LiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
4887 AssertRCReturn(rc, rc);
4888
4889 /* The main device structure. */
4890 SSMR3PutU32(pSSM, pThis->regHbaCap);
4891 SSMR3PutU32(pSSM, pThis->regHbaCtrl);
4892 SSMR3PutU32(pSSM, pThis->regHbaIs);
4893 SSMR3PutU32(pSSM, pThis->regHbaPi);
4894 SSMR3PutU32(pSSM, pThis->regHbaVs);
4895 SSMR3PutU32(pSSM, pThis->regHbaCccCtl);
4896 SSMR3PutU32(pSSM, pThis->regHbaCccPorts);
4897 SSMR3PutU8(pSSM, pThis->uCccPortNr);
4898 SSMR3PutU64(pSSM, pThis->uCccTimeout);
4899 SSMR3PutU32(pSSM, pThis->uCccNr);
4900 SSMR3PutU32(pSSM, pThis->uCccCurrentNr);
4901 SSMR3PutU32(pSSM, pThis->u32PortsInterrupted);
4902 SSMR3PutBool(pSSM, pThis->fReset);
4903 SSMR3PutBool(pSSM, pThis->f64BitAddr);
4904 SSMR3PutBool(pSSM, pThis->fR0Enabled);
4905 SSMR3PutBool(pSSM, pThis->fGCEnabled);
4906 SSMR3PutBool(pSSM, pThis->fLegacyPortResetMethod);
4907
4908 /* Now every port. */
4909 for (i = 0; i < AHCI_MAX_NR_PORTS_IMPL; i++)
4910 {
4911 Assert(pThis->ahciPort[i].cTasksActive == 0);
4912 SSMR3PutU32(pSSM, pThis->ahciPort[i].regCLB);
4913 SSMR3PutU32(pSSM, pThis->ahciPort[i].regCLBU);
4914 SSMR3PutU32(pSSM, pThis->ahciPort[i].regFB);
4915 SSMR3PutU32(pSSM, pThis->ahciPort[i].regFBU);
4916 SSMR3PutGCPhys(pSSM, pThis->ahciPort[i].GCPhysAddrClb);
4917 SSMR3PutGCPhys(pSSM, pThis->ahciPort[i].GCPhysAddrFb);
4918 SSMR3PutU32(pSSM, pThis->ahciPort[i].regIS);
4919 SSMR3PutU32(pSSM, pThis->ahciPort[i].regIE);
4920 SSMR3PutU32(pSSM, pThis->ahciPort[i].regCMD);
4921 SSMR3PutU32(pSSM, pThis->ahciPort[i].regTFD);
4922 SSMR3PutU32(pSSM, pThis->ahciPort[i].regSIG);
4923 SSMR3PutU32(pSSM, pThis->ahciPort[i].regSSTS);
4924 SSMR3PutU32(pSSM, pThis->ahciPort[i].regSCTL);
4925 SSMR3PutU32(pSSM, pThis->ahciPort[i].regSERR);
4926 SSMR3PutU32(pSSM, pThis->ahciPort[i].regSACT);
4927 SSMR3PutU32(pSSM, pThis->ahciPort[i].regCI);
4928 SSMR3PutU32(pSSM, pThis->ahciPort[i].PCHSGeometry.cCylinders);
4929 SSMR3PutU32(pSSM, pThis->ahciPort[i].PCHSGeometry.cHeads);
4930 SSMR3PutU32(pSSM, pThis->ahciPort[i].PCHSGeometry.cSectors);
4931 SSMR3PutU64(pSSM, pThis->ahciPort[i].cTotalSectors);
4932 SSMR3PutU32(pSSM, pThis->ahciPort[i].cMultSectors);
4933 SSMR3PutU8(pSSM, pThis->ahciPort[i].uATATransferMode);
4934 SSMR3PutBool(pSSM, pThis->ahciPort[i].fResetDevice);
4935 SSMR3PutBool(pSSM, pThis->ahciPort[i].fPoweredOn);
4936 SSMR3PutBool(pSSM, pThis->ahciPort[i].fSpunUp);
4937 SSMR3PutU32(pSSM, pThis->ahciPort[i].u32TasksFinished);
4938 SSMR3PutU32(pSSM, pThis->ahciPort[i].u32QueuedTasksFinished);
4939 SSMR3PutU32(pSSM, pThis->ahciPort[i].u32CurrentCommandSlot);
4940
4941 /* ATAPI saved state. */
4942 SSMR3PutBool(pSSM, pThis->ahciPort[i].fATAPI);
4943 SSMR3PutMem(pSSM, &pThis->ahciPort[i].abATAPISense[0], sizeof(pThis->ahciPort[i].abATAPISense));
4944 }
4945
4946 return SSMR3PutU32(pSSM, UINT32_MAX); /* sanity/terminator */
4947}
4948
4949/**
4950 * Loads a saved legacy ATA emulated device state.
4951 *
4952 * @returns VBox status code.
4953 * @param pSSM The handle to the saved state.
4954 */
4955static int ahciR3LoadLegacyEmulationState(PSSMHANDLE pSSM)
4956{
4957 int rc;
4958 uint32_t u32Version;
4959 uint32_t u32;
4960 uint32_t u32IOBuffer;
4961
4962 /* Test for correct version. */
4963 rc = SSMR3GetU32(pSSM, &u32Version);
4964 AssertRCReturn(rc, rc);
4965 LogFlow(("LoadOldSavedStates u32Version = %d\n", u32Version));
4966
4967 if ( u32Version != ATA_CTL_SAVED_STATE_VERSION
4968 && u32Version != ATA_CTL_SAVED_STATE_VERSION_WITHOUT_FULL_SENSE
4969 && u32Version != ATA_CTL_SAVED_STATE_VERSION_WITHOUT_EVENT_STATUS)
4970 {
4971 AssertMsgFailed(("u32Version=%d\n", u32Version));
4972 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
4973 }
4974
4975 SSMR3Skip(pSSM, 19 + 5 * sizeof(bool) + 8 /* sizeof(BMDMAState) */);
4976
4977 for (uint32_t j = 0; j < 2; j++)
4978 {
4979 SSMR3Skip(pSSM, 88 + 5 * sizeof(bool) );
4980
4981 if (u32Version > ATA_CTL_SAVED_STATE_VERSION_WITHOUT_FULL_SENSE)
4982 SSMR3Skip(pSSM, 64);
4983 else
4984 SSMR3Skip(pSSM, 2);
4985 /** @todo triple-check this hack after passthrough is working */
4986 SSMR3Skip(pSSM, 1);
4987
4988 if (u32Version > ATA_CTL_SAVED_STATE_VERSION_WITHOUT_EVENT_STATUS)
4989 SSMR3Skip(pSSM, 4);
4990
4991 SSMR3Skip(pSSM, sizeof(PDMLED));
4992 SSMR3GetU32(pSSM, &u32IOBuffer);
4993 if (u32IOBuffer)
4994 SSMR3Skip(pSSM, u32IOBuffer);
4995 }
4996
4997 rc = SSMR3GetU32(pSSM, &u32);
4998 if (RT_FAILURE(rc))
4999 return rc;
5000 if (u32 != ~0U)
5001 {
5002 AssertMsgFailed(("u32=%#x expected ~0\n", u32));
5003 rc = VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
5004 return rc;
5005 }
5006
5007 return VINF_SUCCESS;
5008}
5009
5010/**
5011 * @callback_method_impl{FNSSMDEVLOADEXEC}
5012 */
5013static DECLCALLBACK(int) ahciR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
5014{
5015 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5016 uint32_t u32;
5017 int rc;
5018
5019 if ( uVersion > AHCI_SAVED_STATE_VERSION
5020 || uVersion < AHCI_SAVED_STATE_VERSION_VBOX_30)
5021 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
5022
5023 /* Deal with the priod after removing the saved IDE bits where the saved
5024 state version remained unchanged. */
5025 if ( uVersion == AHCI_SAVED_STATE_VERSION_IDE_EMULATION
5026 && SSMR3HandleRevision(pSSM) >= 79045
5027 && SSMR3HandleRevision(pSSM) < 79201)
5028 uVersion++;
5029
5030 /*
5031 * Check whether we have to resort to the legacy port reset method to
5032 * prevent older BIOS versions from failing after a reset.
5033 */
5034 if (uVersion <= AHCI_SAVED_STATE_VERSION_PRE_PORT_RESET_CHANGES)
5035 pThis->fLegacyPortResetMethod = true;
5036
5037 /* Verify config. */
5038 if (uVersion > AHCI_SAVED_STATE_VERSION_VBOX_30)
5039 {
5040 rc = SSMR3GetU32(pSSM, &u32);
5041 AssertRCReturn(rc, rc);
5042 if (u32 != pThis->cPortsImpl)
5043 {
5044 LogRel(("AHCI: Config mismatch: cPortsImpl - saved=%u config=%u\n", u32, pThis->cPortsImpl));
5045 if ( u32 < pThis->cPortsImpl
5046 || u32 > AHCI_MAX_NR_PORTS_IMPL)
5047 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch: cPortsImpl - saved=%u config=%u"),
5048 u32, pThis->cPortsImpl);
5049 }
5050
5051 for (uint32_t i = 0; i < AHCI_MAX_NR_PORTS_IMPL; i++)
5052 {
5053 bool fInUse;
5054 rc = SSMR3GetBool(pSSM, &fInUse);
5055 AssertRCReturn(rc, rc);
5056 if (fInUse != (pThis->ahciPort[i].pDrvBase != NULL))
5057 return SSMR3SetCfgError(pSSM, RT_SRC_POS,
5058 N_("The %s VM is missing a device on port %u. Please make sure the source and target VMs have compatible storage configurations"),
5059 fInUse ? "target" : "source", i );
5060
5061 if (uVersion > AHCI_SAVED_STATE_VERSION_PRE_HOTPLUG_FLAG)
5062 {
5063 bool fHotpluggable;
5064 rc = SSMR3GetBool(pSSM, &fHotpluggable);
5065 AssertRCReturn(rc, rc);
5066 if (fHotpluggable != pThis->ahciPort[i].fHotpluggable)
5067 return SSMR3SetCfgError(pSSM, RT_SRC_POS,
5068 N_("AHCI: Port %u config mismatch: Hotplug flag - saved=%RTbool config=%RTbool\n"),
5069 i, fHotpluggable, pThis->ahciPort[i].fHotpluggable);
5070 }
5071 else
5072 Assert(pThis->ahciPort[i].fHotpluggable);
5073
5074 char szSerialNumber[AHCI_SERIAL_NUMBER_LENGTH+1];
5075 rc = SSMR3GetStrZ(pSSM, szSerialNumber, sizeof(szSerialNumber));
5076 AssertRCReturn(rc, rc);
5077 if (strcmp(szSerialNumber, pThis->ahciPort[i].szSerialNumber))
5078 LogRel(("AHCI: Port %u config mismatch: Serial number - saved='%s' config='%s'\n",
5079 i, szSerialNumber, pThis->ahciPort[i].szSerialNumber));
5080
5081 char szFirmwareRevision[AHCI_FIRMWARE_REVISION_LENGTH+1];
5082 rc = SSMR3GetStrZ(pSSM, szFirmwareRevision, sizeof(szFirmwareRevision));
5083 AssertRCReturn(rc, rc);
5084 if (strcmp(szFirmwareRevision, pThis->ahciPort[i].szFirmwareRevision))
5085 LogRel(("AHCI: Port %u config mismatch: Firmware revision - saved='%s' config='%s'\n",
5086 i, szFirmwareRevision, pThis->ahciPort[i].szFirmwareRevision));
5087
5088 char szModelNumber[AHCI_MODEL_NUMBER_LENGTH+1];
5089 rc = SSMR3GetStrZ(pSSM, szModelNumber, sizeof(szModelNumber));
5090 AssertRCReturn(rc, rc);
5091 if (strcmp(szModelNumber, pThis->ahciPort[i].szModelNumber))
5092 LogRel(("AHCI: Port %u config mismatch: Model number - saved='%s' config='%s'\n",
5093 i, szModelNumber, pThis->ahciPort[i].szModelNumber));
5094 }
5095
5096 static const char *s_apszIdeEmuPortNames[4] = { "PrimaryMaster", "PrimarySlave", "SecondaryMaster", "SecondarySlave" };
5097 for (uint32_t i = 0; i < RT_ELEMENTS(s_apszIdeEmuPortNames); i++)
5098 {
5099 uint32_t iPort;
5100 rc = CFGMR3QueryU32Def(pDevIns->pCfg, s_apszIdeEmuPortNames[i], &iPort, i);
5101 AssertRCReturn(rc, rc);
5102
5103 uint32_t iPortSaved;
5104 rc = SSMR3GetU32(pSSM, &iPortSaved);
5105 AssertRCReturn(rc, rc);
5106
5107 if (iPortSaved != iPort)
5108 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("IDE %s config mismatch: saved=%u config=%u"),
5109 s_apszIdeEmuPortNames[i], iPortSaved, iPort);
5110 }
5111 }
5112
5113 if (uPass == SSM_PASS_FINAL)
5114 {
5115 /* Restore data. */
5116
5117 /* The main device structure. */
5118 SSMR3GetU32(pSSM, &pThis->regHbaCap);
5119 SSMR3GetU32(pSSM, &pThis->regHbaCtrl);
5120 SSMR3GetU32(pSSM, &pThis->regHbaIs);
5121 SSMR3GetU32(pSSM, &pThis->regHbaPi);
5122 SSMR3GetU32(pSSM, &pThis->regHbaVs);
5123 SSMR3GetU32(pSSM, &pThis->regHbaCccCtl);
5124 SSMR3GetU32(pSSM, &pThis->regHbaCccPorts);
5125 SSMR3GetU8(pSSM, &pThis->uCccPortNr);
5126 SSMR3GetU64(pSSM, &pThis->uCccTimeout);
5127 SSMR3GetU32(pSSM, &pThis->uCccNr);
5128 SSMR3GetU32(pSSM, &pThis->uCccCurrentNr);
5129
5130 SSMR3GetU32(pSSM, (uint32_t *)&pThis->u32PortsInterrupted);
5131 SSMR3GetBool(pSSM, &pThis->fReset);
5132 SSMR3GetBool(pSSM, &pThis->f64BitAddr);
5133 SSMR3GetBool(pSSM, &pThis->fR0Enabled);
5134 SSMR3GetBool(pSSM, &pThis->fGCEnabled);
5135 if (uVersion > AHCI_SAVED_STATE_VERSION_PRE_PORT_RESET_CHANGES)
5136 SSMR3GetBool(pSSM, &pThis->fLegacyPortResetMethod);
5137
5138 /* Now every port. */
5139 for (uint32_t i = 0; i < AHCI_MAX_NR_PORTS_IMPL; i++)
5140 {
5141 PAHCIPort pAhciPort = &pThis->ahciPort[i];
5142
5143 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regCLB);
5144 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regCLBU);
5145 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regFB);
5146 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regFBU);
5147 SSMR3GetGCPhys(pSSM, (RTGCPHYS *)&pThis->ahciPort[i].GCPhysAddrClb);
5148 SSMR3GetGCPhys(pSSM, (RTGCPHYS *)&pThis->ahciPort[i].GCPhysAddrFb);
5149 SSMR3GetU32(pSSM, (uint32_t *)&pThis->ahciPort[i].regIS);
5150 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regIE);
5151 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regCMD);
5152 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regTFD);
5153 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regSIG);
5154 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regSSTS);
5155 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regSCTL);
5156 SSMR3GetU32(pSSM, &pThis->ahciPort[i].regSERR);
5157 SSMR3GetU32(pSSM, (uint32_t *)&pThis->ahciPort[i].regSACT);
5158 SSMR3GetU32(pSSM, (uint32_t *)&pThis->ahciPort[i].regCI);
5159 SSMR3GetU32(pSSM, &pThis->ahciPort[i].PCHSGeometry.cCylinders);
5160 SSMR3GetU32(pSSM, &pThis->ahciPort[i].PCHSGeometry.cHeads);
5161 SSMR3GetU32(pSSM, &pThis->ahciPort[i].PCHSGeometry.cSectors);
5162 SSMR3GetU64(pSSM, &pThis->ahciPort[i].cTotalSectors);
5163 SSMR3GetU32(pSSM, &pThis->ahciPort[i].cMultSectors);
5164 SSMR3GetU8(pSSM, &pThis->ahciPort[i].uATATransferMode);
5165 SSMR3GetBool(pSSM, &pThis->ahciPort[i].fResetDevice);
5166
5167 if (uVersion <= AHCI_SAVED_STATE_VERSION_VBOX_30)
5168 SSMR3Skip(pSSM, AHCI_NR_COMMAND_SLOTS * sizeof(uint8_t)); /* no active data here */
5169
5170 if (uVersion < AHCI_SAVED_STATE_VERSION_IDE_EMULATION)
5171 {
5172 /* The old positions in the FIFO, not required. */
5173 SSMR3Skip(pSSM, 2*sizeof(uint8_t));
5174 }
5175 SSMR3GetBool(pSSM, &pThis->ahciPort[i].fPoweredOn);
5176 SSMR3GetBool(pSSM, &pThis->ahciPort[i].fSpunUp);
5177 SSMR3GetU32(pSSM, (uint32_t *)&pThis->ahciPort[i].u32TasksFinished);
5178 SSMR3GetU32(pSSM, (uint32_t *)&pThis->ahciPort[i].u32QueuedTasksFinished);
5179
5180 if (uVersion >= AHCI_SAVED_STATE_VERSION_IDE_EMULATION)
5181 SSMR3GetU32(pSSM, (uint32_t *)&pThis->ahciPort[i].u32CurrentCommandSlot);
5182
5183 if (uVersion > AHCI_SAVED_STATE_VERSION_PRE_ATAPI)
5184 {
5185 SSMR3GetBool(pSSM, &pThis->ahciPort[i].fATAPI);
5186 SSMR3GetMem(pSSM, pThis->ahciPort[i].abATAPISense, sizeof(pThis->ahciPort[i].abATAPISense));
5187 if (uVersion <= AHCI_SAVED_STATE_VERSION_PRE_ATAPI_REMOVE)
5188 {
5189 SSMR3Skip(pSSM, 1); /* cNotifiedMediaChange. */
5190 SSMR3Skip(pSSM, 4); /* MediaEventStatus */
5191 }
5192 }
5193 else if (pThis->ahciPort[i].fATAPI)
5194 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch: atapi - saved=false config=true"));
5195
5196 /* Check if we have tasks pending. */
5197 uint32_t fTasksOutstanding = pAhciPort->regCI & ~pAhciPort->u32TasksFinished;
5198 uint32_t fQueuedTasksOutstanding = pAhciPort->regSACT & ~pAhciPort->u32QueuedTasksFinished;
5199
5200 pAhciPort->u32TasksNew = fTasksOutstanding | fQueuedTasksOutstanding;
5201
5202 if (pAhciPort->u32TasksNew)
5203 {
5204 /*
5205 * There are tasks pending. The VM was saved after a task failed
5206 * because of non-fatal error. Set the redo flag.
5207 */
5208 pAhciPort->fRedo = true;
5209 }
5210 }
5211
5212 if (uVersion <= AHCI_SAVED_STATE_VERSION_IDE_EMULATION)
5213 {
5214 for (uint32_t i = 0; i < 2; i++)
5215 {
5216 rc = ahciR3LoadLegacyEmulationState(pSSM);
5217 if(RT_FAILURE(rc))
5218 return rc;
5219 }
5220 }
5221
5222 rc = SSMR3GetU32(pSSM, &u32);
5223 if (RT_FAILURE(rc))
5224 return rc;
5225 AssertMsgReturn(u32 == UINT32_MAX, ("%#x\n", u32), VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
5226 }
5227
5228 return VINF_SUCCESS;
5229}
5230
5231/* -=-=-=-=- device PDM interface -=-=-=-=- */
5232
5233static DECLCALLBACK(void) ahciR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
5234{
5235 uint32_t i;
5236 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
5237
5238 pAhci->pDevInsRC += offDelta;
5239 pAhci->pHbaCccTimerRC = TMTimerRCPtr(pAhci->pHbaCccTimerR3);
5240 pAhci->pNotifierQueueRC = PDMQueueRCPtr(pAhci->pNotifierQueueR3);
5241
5242 /* Relocate every port. */
5243 for (i = 0; i < RT_ELEMENTS(pAhci->ahciPort); i++)
5244 {
5245 PAHCIPort pAhciPort = &pAhci->ahciPort[i];
5246 pAhciPort->pAhciRC += offDelta;
5247 pAhciPort->pDevInsRC += offDelta;
5248 }
5249}
5250
5251/**
5252 * Configure the attached device for a port.
5253 *
5254 * Used by ahciR3Construct and ahciR3Attach.
5255 *
5256 * @returns VBox status code
5257 * @param pDevIns The device instance data.
5258 * @param pAhciPort The port for which the device is to be configured.
5259 */
5260static int ahciR3ConfigureLUN(PPDMDEVINS pDevIns, PAHCIPort pAhciPort)
5261{
5262 /* Query the media interface. */
5263 pAhciPort->pDrvMedia = PDMIBASE_QUERY_INTERFACE(pAhciPort->pDrvBase, PDMIMEDIA);
5264 AssertMsgReturn(VALID_PTR(pAhciPort->pDrvMedia),
5265 ("AHCI configuration error: LUN#%d misses the basic media interface!\n", pAhciPort->iLUN),
5266 VERR_PDM_MISSING_INTERFACE);
5267
5268 /* Get the extended media interface. */
5269 pAhciPort->pDrvMediaEx = PDMIBASE_QUERY_INTERFACE(pAhciPort->pDrvBase, PDMIMEDIAEX);
5270 AssertMsgReturn(VALID_PTR(pAhciPort->pDrvMediaEx),
5271 ("AHCI configuration error: LUN#%d misses the extended media interface!\n", pAhciPort->iLUN),
5272 VERR_PDM_MISSING_INTERFACE);
5273
5274 /*
5275 * Validate type.
5276 */
5277 PDMMEDIATYPE enmType = pAhciPort->pDrvMedia->pfnGetType(pAhciPort->pDrvMedia);
5278 AssertMsgReturn(enmType == PDMMEDIATYPE_HARD_DISK || enmType == PDMMEDIATYPE_CDROM || enmType == PDMMEDIATYPE_DVD,
5279 ("AHCI configuration error: LUN#%d isn't a disk or cd/dvd. enmType=%u\n", pAhciPort->iLUN, enmType),
5280 VERR_PDM_UNSUPPORTED_BLOCK_TYPE);
5281
5282 int rc = pAhciPort->pDrvMediaEx->pfnIoReqAllocSizeSet(pAhciPort->pDrvMediaEx, sizeof(AHCIREQ));
5283 if (RT_FAILURE(rc))
5284 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5285 N_("AHCI configuration error: LUN#%u: Failed to set I/O request size!"),
5286 pAhciPort->iLUN);
5287
5288 uint32_t fFeatures = 0;
5289 rc = pAhciPort->pDrvMediaEx->pfnQueryFeatures(pAhciPort->pDrvMediaEx, &fFeatures);
5290 if (RT_FAILURE(rc))
5291 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5292 N_("AHCI configuration error: LUN#%u: Failed to query features of device"),
5293 pAhciPort->iLUN);
5294
5295 if (fFeatures & PDMIMEDIAEX_FEATURE_F_DISCARD)
5296 pAhciPort->fTrimEnabled = true;
5297
5298 pAhciPort->fATAPI = (enmType == PDMMEDIATYPE_CDROM || enmType == PDMMEDIATYPE_DVD)
5299 && RT_BOOL(fFeatures & PDMIMEDIAEX_FEATURE_F_RAWSCSICMD);
5300 if (pAhciPort->fATAPI)
5301 {
5302 pAhciPort->PCHSGeometry.cCylinders = 0;
5303 pAhciPort->PCHSGeometry.cHeads = 0;
5304 pAhciPort->PCHSGeometry.cSectors = 0;
5305 LogRel(("AHCI: LUN#%d: CD/DVD\n", pAhciPort->iLUN));
5306 }
5307 else
5308 {
5309 pAhciPort->cbSector = pAhciPort->pDrvMedia->pfnGetSectorSize(pAhciPort->pDrvMedia);
5310 pAhciPort->cTotalSectors = pAhciPort->pDrvMedia->pfnGetSize(pAhciPort->pDrvMedia) / pAhciPort->cbSector;
5311 rc = pAhciPort->pDrvMedia->pfnBiosGetPCHSGeometry(pAhciPort->pDrvMedia, &pAhciPort->PCHSGeometry);
5312 if (rc == VERR_PDM_MEDIA_NOT_MOUNTED)
5313 {
5314 pAhciPort->PCHSGeometry.cCylinders = 0;
5315 pAhciPort->PCHSGeometry.cHeads = 16; /*??*/
5316 pAhciPort->PCHSGeometry.cSectors = 63; /*??*/
5317 }
5318 else if (rc == VERR_PDM_GEOMETRY_NOT_SET)
5319 {
5320 pAhciPort->PCHSGeometry.cCylinders = 0; /* autodetect marker */
5321 rc = VINF_SUCCESS;
5322 }
5323 AssertRC(rc);
5324
5325 if ( pAhciPort->PCHSGeometry.cCylinders == 0
5326 || pAhciPort->PCHSGeometry.cHeads == 0
5327 || pAhciPort->PCHSGeometry.cSectors == 0)
5328 {
5329 uint64_t cCylinders = pAhciPort->cTotalSectors / (16 * 63);
5330 pAhciPort->PCHSGeometry.cCylinders = RT_MAX(RT_MIN(cCylinders, 16383), 1);
5331 pAhciPort->PCHSGeometry.cHeads = 16;
5332 pAhciPort->PCHSGeometry.cSectors = 63;
5333 /* Set the disk geometry information. Ignore errors. */
5334 pAhciPort->pDrvMedia->pfnBiosSetPCHSGeometry(pAhciPort->pDrvMedia, &pAhciPort->PCHSGeometry);
5335 rc = VINF_SUCCESS;
5336 }
5337 LogRel(("AHCI: LUN#%d: disk, PCHS=%u/%u/%u, total number of sectors %Ld\n",
5338 pAhciPort->iLUN, pAhciPort->PCHSGeometry.cCylinders,
5339 pAhciPort->PCHSGeometry.cHeads, pAhciPort->PCHSGeometry.cSectors,
5340 pAhciPort->cTotalSectors));
5341 if (pAhciPort->fTrimEnabled)
5342 LogRel(("AHCI: LUN#%d: Enabled TRIM support\n", pAhciPort->iLUN));
5343 }
5344 return rc;
5345}
5346
5347/**
5348 * Callback employed by ahciR3Suspend and ahciR3PowerOff.
5349 *
5350 * @returns true if we've quiesced, false if we're still working.
5351 * @param pDevIns The device instance.
5352 */
5353static DECLCALLBACK(bool) ahciR3IsAsyncSuspendOrPowerOffDone(PPDMDEVINS pDevIns)
5354{
5355 if (!ahciR3AllAsyncIOIsFinished(pDevIns))
5356 return false;
5357
5358 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5359 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5360 return true;
5361}
5362
5363/**
5364 * Common worker for ahciR3Suspend and ahciR3PowerOff.
5365 */
5366static void ahciR3SuspendOrPowerOff(PPDMDEVINS pDevIns)
5367{
5368 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5369
5370 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
5371 if (!ahciR3AllAsyncIOIsFinished(pDevIns))
5372 PDMDevHlpSetAsyncNotification(pDevIns, ahciR3IsAsyncSuspendOrPowerOffDone);
5373 else
5374 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5375}
5376
5377/**
5378 * Suspend notification.
5379 *
5380 * @param pDevIns The device instance data.
5381 */
5382static DECLCALLBACK(void) ahciR3Suspend(PPDMDEVINS pDevIns)
5383{
5384 Log(("ahciR3Suspend\n"));
5385 ahciR3SuspendOrPowerOff(pDevIns);
5386}
5387
5388/**
5389 * Resume notification.
5390 *
5391 * @param pDevIns The device instance data.
5392 */
5393static DECLCALLBACK(void) ahciR3Resume(PPDMDEVINS pDevIns)
5394{
5395 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
5396
5397 /*
5398 * Check if one of the ports has pending tasks.
5399 * Queue a notification item again in this case.
5400 */
5401 for (unsigned i = 0; i < RT_ELEMENTS(pAhci->ahciPort); i++)
5402 {
5403 PAHCIPort pAhciPort = &pAhci->ahciPort[i];
5404
5405 if (pAhciPort->u32TasksRedo)
5406 {
5407 PDEVPORTNOTIFIERQUEUEITEM pItem = (PDEVPORTNOTIFIERQUEUEITEM)PDMQueueAlloc(pAhci->CTX_SUFF(pNotifierQueue));
5408 AssertMsg(pItem, ("Allocating item for queue failed\n"));
5409
5410 pAhciPort->u32TasksNew |= pAhciPort->u32TasksRedo;
5411 pAhciPort->u32TasksRedo = 0;
5412
5413 Assert(pAhciPort->fRedo);
5414 pAhciPort->fRedo = false;
5415
5416 pItem->iPort = pAhci->ahciPort[i].iLUN;
5417 PDMQueueInsert(pAhci->CTX_SUFF(pNotifierQueue), (PPDMQUEUEITEMCORE)pItem);
5418 }
5419 }
5420
5421 Log(("%s:\n", __FUNCTION__));
5422}
5423
5424/**
5425 * Initializes the VPD data of a attached device.
5426 *
5427 * @returns VBox status code.
5428 * @param pDevIns The device instance.
5429 * @param pAhciPort The attached device.
5430 * @param pszName Name of the port to get the CFGM node.
5431 */
5432static int ahciR3VpdInit(PPDMDEVINS pDevIns, PAHCIPort pAhciPort, const char *pszName)
5433{
5434
5435 /* Generate a default serial number. */
5436 char szSerial[AHCI_SERIAL_NUMBER_LENGTH+1];
5437 RTUUID Uuid;
5438
5439 int rc = VINF_SUCCESS;
5440 if (pAhciPort->pDrvMedia)
5441 rc = pAhciPort->pDrvMedia->pfnGetUuid(pAhciPort->pDrvMedia, &Uuid);
5442 else
5443 RTUuidClear(&Uuid);
5444
5445 if (RT_FAILURE(rc) || RTUuidIsNull(&Uuid))
5446 {
5447 /* Generate a predictable serial for drives which don't have a UUID. */
5448 RTStrPrintf(szSerial, sizeof(szSerial), "VB%x-1a2b3c4d",
5449 pAhciPort->iLUN);
5450 }
5451 else
5452 RTStrPrintf(szSerial, sizeof(szSerial), "VB%08x-%08x", Uuid.au32[0], Uuid.au32[3]);
5453
5454 /* Get user config if present using defaults otherwise. */
5455 PCFGMNODE pCfgNode = CFGMR3GetChild(pDevIns->pCfg, pszName);
5456 rc = CFGMR3QueryStringDef(pCfgNode, "SerialNumber", pAhciPort->szSerialNumber, sizeof(pAhciPort->szSerialNumber),
5457 szSerial);
5458 if (RT_FAILURE(rc))
5459 {
5460 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
5461 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
5462 N_("AHCI configuration error: \"SerialNumber\" is longer than 20 bytes"));
5463 return PDMDEV_SET_ERROR(pDevIns, rc,
5464 N_("AHCI configuration error: failed to read \"SerialNumber\" as string"));
5465 }
5466
5467 rc = CFGMR3QueryStringDef(pCfgNode, "FirmwareRevision", pAhciPort->szFirmwareRevision, sizeof(pAhciPort->szFirmwareRevision),
5468 "1.0");
5469 if (RT_FAILURE(rc))
5470 {
5471 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
5472 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
5473 N_("AHCI configuration error: \"FirmwareRevision\" is longer than 8 bytes"));
5474 return PDMDEV_SET_ERROR(pDevIns, rc,
5475 N_("AHCI configuration error: failed to read \"FirmwareRevision\" as string"));
5476 }
5477
5478 rc = CFGMR3QueryStringDef(pCfgNode, "ModelNumber", pAhciPort->szModelNumber, sizeof(pAhciPort->szModelNumber),
5479 pAhciPort->fATAPI ? "VBOX CD-ROM" : "VBOX HARDDISK");
5480 if (RT_FAILURE(rc))
5481 {
5482 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
5483 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
5484 N_("AHCI configuration error: \"ModelNumber\" is longer than 40 bytes"));
5485 return PDMDEV_SET_ERROR(pDevIns, rc,
5486 N_("AHCI configuration error: failed to read \"ModelNumber\" as string"));
5487 }
5488
5489 rc = CFGMR3QueryU8Def(pCfgNode, "LogicalSectorsPerPhysical", &pAhciPort->cLogSectorsPerPhysicalExp, 0);
5490 if (RT_FAILURE(rc))
5491 return PDMDEV_SET_ERROR(pDevIns, rc,
5492 N_("AHCI configuration error: failed to read \"LogicalSectorsPerPhysical\" as integer"));
5493 if (pAhciPort->cLogSectorsPerPhysicalExp >= 16)
5494 return PDMDEV_SET_ERROR(pDevIns, rc,
5495 N_("AHCI configuration error: \"LogicalSectorsPerPhysical\" must be between 0 and 15"));
5496
5497 return rc;
5498}
5499
5500
5501/**
5502 * Detach notification.
5503 *
5504 * One harddisk at one port has been unplugged.
5505 * The VM is suspended at this point.
5506 *
5507 * @param pDevIns The device instance.
5508 * @param iLUN The logical unit which is being detached.
5509 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
5510 */
5511static DECLCALLBACK(void) ahciR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
5512{
5513 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
5514 PAHCIPort pAhciPort = &pAhci->ahciPort[iLUN];
5515 int rc = VINF_SUCCESS;
5516
5517 Log(("%s:\n", __FUNCTION__));
5518
5519 AssertMsg(iLUN < pAhci->cPortsImpl, ("iLUN=%u", iLUN));
5520 AssertMsgReturnVoid( pAhciPort->fHotpluggable
5521 || (fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG),
5522 ("AHCI: Port %d is not marked hotpluggable\n", pAhciPort->iLUN));
5523
5524
5525 if (pAhciPort->pAsyncIOThread)
5526 {
5527 int rcThread;
5528 /* Destroy the thread. */
5529 rc = PDMR3ThreadDestroy(pAhciPort->pAsyncIOThread, &rcThread);
5530 if (RT_FAILURE(rc) || RT_FAILURE(rcThread))
5531 AssertMsgFailed(("%s Failed to destroy async IO thread rc=%Rrc rcThread=%Rrc\n", __FUNCTION__, rc, rcThread));
5532
5533 pAhciPort->pAsyncIOThread = NULL;
5534 pAhciPort->fWrkThreadSleeping = true;
5535 }
5536
5537 if (!(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG))
5538 {
5539 /*
5540 * Inform the guest about the removed device.
5541 */
5542 pAhciPort->regSSTS = 0;
5543 pAhciPort->regSIG = 0;
5544 /*
5545 * Clear CR bit too to prevent submission of new commands when CI is written
5546 * (AHCI Spec 1.2: 7.4 Interaction of the Command List and Port Change Status).
5547 */
5548 ASMAtomicAndU32(&pAhciPort->regCMD, ~(AHCI_PORT_CMD_CPS | AHCI_PORT_CMD_CR));
5549 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_CPDS | AHCI_PORT_IS_PRCS | AHCI_PORT_IS_PCS);
5550 ASMAtomicOrU32(&pAhciPort->regSERR, AHCI_PORT_SERR_X | AHCI_PORT_SERR_N);
5551 if ( (pAhciPort->regIE & AHCI_PORT_IE_CPDE)
5552 || (pAhciPort->regIE & AHCI_PORT_IE_PCE)
5553 || (pAhciPort->regIE & AHCI_PORT_IE_PRCE))
5554 ahciHbaSetInterrupt(pAhciPort->CTX_SUFF(pAhci), pAhciPort->iLUN, VERR_IGNORED);
5555 }
5556
5557 /*
5558 * Zero some important members.
5559 */
5560 pAhciPort->pDrvBase = NULL;
5561 pAhciPort->pDrvMedia = NULL;
5562 pAhciPort->pDrvMediaEx = NULL;
5563}
5564
5565/**
5566 * Attach command.
5567 *
5568 * This is called when we change block driver for one port.
5569 * The VM is suspended at this point.
5570 *
5571 * @returns VBox status code.
5572 * @param pDevIns The device instance.
5573 * @param iLUN The logical unit which is being detached.
5574 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
5575 */
5576static DECLCALLBACK(int) ahciR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
5577{
5578 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5579 PAHCIPort pAhciPort = &pThis->ahciPort[iLUN];
5580 int rc;
5581
5582 Log(("%s:\n", __FUNCTION__));
5583
5584 /* the usual paranoia */
5585 AssertMsg(iLUN < pThis->cPortsImpl, ("iLUN=%u", iLUN));
5586 AssertRelease(!pAhciPort->pDrvBase);
5587 AssertRelease(!pAhciPort->pDrvMedia);
5588 AssertRelease(!pAhciPort->pDrvMediaEx);
5589 Assert(pAhciPort->iLUN == iLUN);
5590
5591 AssertMsgReturn( pAhciPort->fHotpluggable
5592 || (fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG),
5593 ("AHCI: Port %d is not marked hotpluggable\n", pAhciPort->iLUN),
5594 VERR_INVALID_PARAMETER);
5595
5596 /*
5597 * Try attach the block device and get the interfaces,
5598 * required as well as optional.
5599 */
5600 rc = PDMDevHlpDriverAttach(pDevIns, pAhciPort->iLUN, &pAhciPort->IBase, &pAhciPort->pDrvBase, NULL);
5601 if (RT_SUCCESS(rc))
5602 rc = ahciR3ConfigureLUN(pDevIns, pAhciPort);
5603 else
5604 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", pAhciPort->iLUN, rc));
5605
5606 if (RT_FAILURE(rc))
5607 {
5608 pAhciPort->pDrvBase = NULL;
5609 pAhciPort->pDrvMedia = NULL;
5610 }
5611 else
5612 {
5613 char szName[24];
5614 RTStrPrintf(szName, sizeof(szName), "Port%d", iLUN);
5615
5616 rc = SUPSemEventCreate(pThis->pSupDrvSession, &pAhciPort->hEvtProcess);
5617 if (RT_FAILURE(rc))
5618 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5619 N_("AHCI: Failed to create SUP event semaphore"));
5620
5621 /* Create the async IO thread. */
5622 rc = PDMDevHlpThreadCreate(pDevIns, &pAhciPort->pAsyncIOThread, pAhciPort, ahciAsyncIOLoop, ahciAsyncIOLoopWakeUp, 0,
5623 RTTHREADTYPE_IO, szName);
5624 if (RT_FAILURE(rc))
5625 return rc;
5626
5627 /*
5628 * Init vendor product data.
5629 */
5630 if (RT_SUCCESS(rc))
5631 rc = ahciR3VpdInit(pDevIns, pAhciPort, szName);
5632
5633 /* Inform the guest about the added device in case of hotplugging. */
5634 if ( RT_SUCCESS(rc)
5635 && !(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG))
5636 {
5637 AssertMsgReturn(pAhciPort->fHotpluggable,
5638 ("AHCI: Port %d is not marked hotpluggable\n", pAhciPort->iLUN),
5639 VERR_NOT_SUPPORTED);
5640
5641 /*
5642 * Initialize registers
5643 */
5644 ASMAtomicOrU32(&pAhciPort->regCMD, AHCI_PORT_CMD_CPS);
5645 ASMAtomicOrU32(&pAhciPort->regIS, AHCI_PORT_IS_CPDS | AHCI_PORT_IS_PRCS | AHCI_PORT_IS_PCS);
5646 ASMAtomicOrU32(&pAhciPort->regSERR, AHCI_PORT_SERR_X | AHCI_PORT_SERR_N);
5647
5648 if (pAhciPort->fATAPI)
5649 pAhciPort->regSIG = AHCI_PORT_SIG_ATAPI;
5650 else
5651 pAhciPort->regSIG = AHCI_PORT_SIG_DISK;
5652 pAhciPort->regSSTS = (0x01 << 8) | /* Interface is active. */
5653 (0x02 << 4) | /* Generation 2 (3.0GBps) speed. */
5654 (0x03 << 0); /* Device detected and communication established. */
5655
5656 if ( (pAhciPort->regIE & AHCI_PORT_IE_CPDE)
5657 || (pAhciPort->regIE & AHCI_PORT_IE_PCE)
5658 || (pAhciPort->regIE & AHCI_PORT_IE_PRCE))
5659 ahciHbaSetInterrupt(pAhciPort->CTX_SUFF(pAhci), pAhciPort->iLUN, VERR_IGNORED);
5660 }
5661
5662 }
5663
5664 return rc;
5665}
5666
5667/**
5668 * Common reset worker.
5669 *
5670 * @param pDevIns The device instance data.
5671 */
5672static int ahciR3ResetCommon(PPDMDEVINS pDevIns)
5673{
5674 PAHCI pAhci = PDMINS_2_DATA(pDevIns, PAHCI);
5675
5676 ahciHBAReset(pAhci);
5677
5678 /* Hardware reset for the ports. */
5679 for (uint32_t i = 0; i < RT_ELEMENTS(pAhci->ahciPort); i++)
5680 ahciPortHwReset(&pAhci->ahciPort[i]);
5681 return VINF_SUCCESS;
5682}
5683
5684/**
5685 * Callback employed by ahciR3Reset.
5686 *
5687 * @returns true if we've quiesced, false if we're still working.
5688 * @param pDevIns The device instance.
5689 */
5690static DECLCALLBACK(bool) ahciR3IsAsyncResetDone(PPDMDEVINS pDevIns)
5691{
5692 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5693
5694 if (!ahciR3AllAsyncIOIsFinished(pDevIns))
5695 return false;
5696 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5697
5698 ahciR3ResetCommon(pDevIns);
5699 return true;
5700}
5701
5702/**
5703 * Reset notification.
5704 *
5705 * @param pDevIns The device instance data.
5706 */
5707static DECLCALLBACK(void) ahciR3Reset(PPDMDEVINS pDevIns)
5708{
5709 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5710
5711 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
5712 if (!ahciR3AllAsyncIOIsFinished(pDevIns))
5713 PDMDevHlpSetAsyncNotification(pDevIns, ahciR3IsAsyncResetDone);
5714 else
5715 {
5716 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5717 ahciR3ResetCommon(pDevIns);
5718 }
5719}
5720
5721/**
5722 * Poweroff notification.
5723 *
5724 * @param pDevIns Pointer to the device instance
5725 */
5726static DECLCALLBACK(void) ahciR3PowerOff(PPDMDEVINS pDevIns)
5727{
5728 Log(("achiR3PowerOff\n"));
5729 ahciR3SuspendOrPowerOff(pDevIns);
5730}
5731
5732/**
5733 * Destroy a driver instance.
5734 *
5735 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
5736 * resources can be freed correctly.
5737 *
5738 * @param pDevIns The device instance data.
5739 */
5740static DECLCALLBACK(int) ahciR3Destruct(PPDMDEVINS pDevIns)
5741{
5742 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5743 int rc = VINF_SUCCESS;
5744 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
5745
5746 /*
5747 * At this point the async I/O thread is suspended and will not enter
5748 * this module again. So, no coordination is needed here and PDM
5749 * will take care of terminating and cleaning up the thread.
5750 */
5751 if (PDMCritSectIsInitialized(&pThis->lock))
5752 {
5753 TMR3TimerDestroy(pThis->CTX_SUFF(pHbaCccTimer));
5754 pThis->CTX_SUFF(pHbaCccTimer) = NULL;
5755
5756 Log(("%s: Destruct every port\n", __FUNCTION__));
5757 for (unsigned iActPort = 0; iActPort < pThis->cPortsImpl; iActPort++)
5758 {
5759 PAHCIPort pAhciPort = &pThis->ahciPort[iActPort];
5760
5761 if (pAhciPort->hEvtProcess != NIL_SUPSEMEVENT)
5762 {
5763 SUPSemEventClose(pThis->pSupDrvSession, pAhciPort->hEvtProcess);
5764 pAhciPort->hEvtProcess = NIL_SUPSEMEVENT;
5765 }
5766 }
5767
5768 PDMR3CritSectDelete(&pThis->lock);
5769 }
5770
5771 return rc;
5772}
5773
5774/**
5775 * @interface_method_impl{PDMDEVREG,pfnConstruct}
5776 */
5777static DECLCALLBACK(int) ahciR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
5778{
5779 PAHCI pThis = PDMINS_2_DATA(pDevIns, PAHCI);
5780 PPDMIBASE pBase;
5781 int rc = VINF_SUCCESS;
5782 unsigned i = 0;
5783 bool fGCEnabled = false;
5784 bool fR0Enabled = false;
5785 uint32_t cbTotalBufferSize = 0;
5786 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
5787
5788 LogFlowFunc(("pThis=%#p\n", pThis));
5789
5790 /*
5791 * Validate and read configuration.
5792 */
5793 if (!CFGMR3AreValuesValid(pCfg, "GCEnabled\0"
5794 "R0Enabled\0"
5795 "PrimaryMaster\0"
5796 "PrimarySlave\0"
5797 "SecondaryMaster\0"
5798 "SecondarySlave\0"
5799 "PortCount\0"
5800 "Bootable\0"
5801 "CmdSlotsAvail\0"))
5802 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
5803 N_("AHCI configuration error: unknown option specified"));
5804
5805 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &fGCEnabled, true);
5806 if (RT_FAILURE(rc))
5807 return PDMDEV_SET_ERROR(pDevIns, rc,
5808 N_("AHCI configuration error: failed to read GCEnabled as boolean"));
5809 Log(("%s: fGCEnabled=%d\n", __FUNCTION__, fGCEnabled));
5810
5811 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &fR0Enabled, true);
5812 if (RT_FAILURE(rc))
5813 return PDMDEV_SET_ERROR(pDevIns, rc,
5814 N_("AHCI configuration error: failed to read R0Enabled as boolean"));
5815 Log(("%s: fR0Enabled=%d\n", __FUNCTION__, fR0Enabled));
5816
5817 rc = CFGMR3QueryU32Def(pCfg, "PortCount", &pThis->cPortsImpl, AHCI_MAX_NR_PORTS_IMPL);
5818 if (RT_FAILURE(rc))
5819 return PDMDEV_SET_ERROR(pDevIns, rc,
5820 N_("AHCI configuration error: failed to read PortCount as integer"));
5821 Log(("%s: cPortsImpl=%u\n", __FUNCTION__, pThis->cPortsImpl));
5822 if (pThis->cPortsImpl > AHCI_MAX_NR_PORTS_IMPL)
5823 return PDMDevHlpVMSetError(pDevIns, VERR_INVALID_PARAMETER, RT_SRC_POS,
5824 N_("AHCI configuration error: PortCount=%u should not exceed %u"),
5825 pThis->cPortsImpl, AHCI_MAX_NR_PORTS_IMPL);
5826 if (pThis->cPortsImpl < 1)
5827 return PDMDevHlpVMSetError(pDevIns, VERR_INVALID_PARAMETER, RT_SRC_POS,
5828 N_("AHCI configuration error: PortCount=%u should be at least 1"),
5829 pThis->cPortsImpl);
5830
5831 rc = CFGMR3QueryBoolDef(pCfg, "Bootable", &pThis->fBootable, true);
5832 if (RT_FAILURE(rc))
5833 return PDMDEV_SET_ERROR(pDevIns, rc,
5834 N_("AHCI configuration error: failed to read Bootable as boolean"));
5835
5836 rc = CFGMR3QueryU32Def(pCfg, "CmdSlotsAvail", &pThis->cCmdSlotsAvail, AHCI_NR_COMMAND_SLOTS);
5837 if (RT_FAILURE(rc))
5838 return PDMDEV_SET_ERROR(pDevIns, rc,
5839 N_("AHCI configuration error: failed to read CmdSlotsAvail as integer"));
5840 Log(("%s: cCmdSlotsAvail=%u\n", __FUNCTION__, pThis->cCmdSlotsAvail));
5841 if (pThis->cCmdSlotsAvail > AHCI_NR_COMMAND_SLOTS)
5842 return PDMDevHlpVMSetError(pDevIns, VERR_INVALID_PARAMETER, RT_SRC_POS,
5843 N_("AHCI configuration error: CmdSlotsAvail=%u should not exceed %u"),
5844 pThis->cPortsImpl, AHCI_NR_COMMAND_SLOTS);
5845 if (pThis->cCmdSlotsAvail < 1)
5846 return PDMDevHlpVMSetError(pDevIns, VERR_INVALID_PARAMETER, RT_SRC_POS,
5847 N_("AHCI configuration error: CmdSlotsAvail=%u should be at least 1"),
5848 pThis->cCmdSlotsAvail);
5849
5850 /*
5851 * Initialize the instance data (everything touched by the destructor need
5852 * to be initialized here!).
5853 */
5854 pThis->fR0Enabled = fR0Enabled;
5855 pThis->fGCEnabled = fGCEnabled;
5856 pThis->pDevInsR3 = pDevIns;
5857 pThis->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
5858 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5859 pThis->pSupDrvSession = PDMDevHlpGetSupDrvSession(pDevIns);
5860
5861 PCIDevSetVendorId (&pThis->dev, 0x8086); /* Intel */
5862 PCIDevSetDeviceId (&pThis->dev, 0x2829); /* ICH-8M */
5863 PCIDevSetCommand (&pThis->dev, 0x0000);
5864#ifdef VBOX_WITH_MSI_DEVICES
5865 PCIDevSetStatus (&pThis->dev, VBOX_PCI_STATUS_CAP_LIST);
5866 PCIDevSetCapabilityList(&pThis->dev, 0x80);
5867#else
5868 PCIDevSetCapabilityList(&pThis->dev, 0x70);
5869#endif
5870 PCIDevSetRevisionId (&pThis->dev, 0x02);
5871 PCIDevSetClassProg (&pThis->dev, 0x01);
5872 PCIDevSetClassSub (&pThis->dev, 0x06);
5873 PCIDevSetClassBase (&pThis->dev, 0x01);
5874 PCIDevSetBaseAddress (&pThis->dev, 5, false, false, false, 0x00000000);
5875
5876 PCIDevSetInterruptLine(&pThis->dev, 0x00);
5877 PCIDevSetInterruptPin (&pThis->dev, 0x01);
5878
5879 pThis->dev.abConfig[0x70] = VBOX_PCI_CAP_ID_PM; /* Capability ID: PCI Power Management Interface */
5880 pThis->dev.abConfig[0x71] = 0xa8; /* next */
5881 pThis->dev.abConfig[0x72] = 0x03; /* version ? */
5882
5883 pThis->dev.abConfig[0x90] = 0x40; /* AHCI mode. */
5884 pThis->dev.abConfig[0x92] = 0x3f;
5885 pThis->dev.abConfig[0x94] = 0x80;
5886 pThis->dev.abConfig[0x95] = 0x01;
5887 pThis->dev.abConfig[0x97] = 0x78;
5888
5889 pThis->dev.abConfig[0xa8] = 0x12; /* SATACR capability */
5890 pThis->dev.abConfig[0xa9] = 0x00; /* next */
5891 PCIDevSetWord(&pThis->dev, 0xaa, 0x0010); /* Revision */
5892 PCIDevSetDWord(&pThis->dev, 0xac, 0x00000028); /* SATA Capability Register 1 */
5893
5894 pThis->cThreadsActive = 0;
5895
5896 /* Initialize port members. */
5897 for (i = 0; i < AHCI_MAX_NR_PORTS_IMPL; i++)
5898 {
5899 PAHCIPort pAhciPort = &pThis->ahciPort[i];
5900 pAhciPort->pDevInsR3 = pDevIns;
5901 pAhciPort->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
5902 pAhciPort->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5903 pAhciPort->iLUN = i;
5904 pAhciPort->pAhciR3 = pThis;
5905 pAhciPort->pAhciR0 = PDMINS_2_DATA_R0PTR(pDevIns);
5906 pAhciPort->pAhciRC = PDMINS_2_DATA_RCPTR(pDevIns);
5907 pAhciPort->Led.u32Magic = PDMLED_MAGIC;
5908 pAhciPort->pDrvBase = NULL;
5909 pAhciPort->pAsyncIOThread = NULL;
5910 pAhciPort->hEvtProcess = NIL_SUPSEMEVENT;
5911 pAhciPort->fHotpluggable = true;
5912 }
5913
5914 /*
5915 * Init locks, using explicit locking where necessary.
5916 */
5917 rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
5918 if (RT_FAILURE(rc))
5919 return rc;
5920
5921 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->lock, RT_SRC_POS, "AHCI#%u", iInstance);
5922 if (RT_FAILURE(rc))
5923 {
5924 Log(("%s: Failed to create critical section.\n", __FUNCTION__));
5925 return rc;
5926 }
5927
5928 /*
5929 * Register the PCI device, it's I/O regions.
5930 */
5931 rc = PDMDevHlpPCIRegister (pDevIns, &pThis->dev);
5932 if (RT_FAILURE(rc))
5933 return rc;
5934
5935#ifdef VBOX_WITH_MSI_DEVICES
5936 PDMMSIREG MsiReg;
5937 RT_ZERO(MsiReg);
5938 MsiReg.cMsiVectors = 1;
5939 MsiReg.iMsiCapOffset = 0x80;
5940 MsiReg.iMsiNextOffset = 0x70;
5941 rc = PDMDevHlpPCIRegisterMsi(pDevIns, &MsiReg);
5942 if (RT_FAILURE(rc))
5943 {
5944 PCIDevSetCapabilityList(&pThis->dev, 0x70);
5945 /* That's OK, we can work without MSI */
5946 }
5947#endif
5948
5949 /*
5950 * Solaris 10 U5 fails to map the AHCI register space when the sets (0..5) for the legacy
5951 * IDE registers are not available.
5952 * We set up "fake" entries in the PCI configuration register.
5953 * That means they are available but read and writes from/to them have no effect.
5954 * No guest should access them anyway because the controller is marked as AHCI in the Programming interface
5955 * and we don't have an option to change to IDE emulation (real hardware provides an option in the BIOS
5956 * to switch to it which also changes device Id and other things in the PCI configuration space).
5957 */
5958 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, 8, PCI_ADDRESS_SPACE_IO, ahciR3LegacyFakeIORangeMap);
5959 if (RT_FAILURE(rc))
5960 return PDMDEV_SET_ERROR(pDevIns, rc,
5961 N_("AHCI cannot register PCI I/O region"));
5962
5963 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, 1, PCI_ADDRESS_SPACE_IO, ahciR3LegacyFakeIORangeMap);
5964 if (RT_FAILURE(rc))
5965 return PDMDEV_SET_ERROR(pDevIns, rc,
5966 N_("AHCI cannot register PCI I/O region"));
5967
5968 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 2, 8, PCI_ADDRESS_SPACE_IO, ahciR3LegacyFakeIORangeMap);
5969 if (RT_FAILURE(rc))
5970 return PDMDEV_SET_ERROR(pDevIns, rc,
5971 N_("AHCI cannot register PCI I/O region"));
5972
5973 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 3, 1, PCI_ADDRESS_SPACE_IO, ahciR3LegacyFakeIORangeMap);
5974 if (RT_FAILURE(rc))
5975 return PDMDEV_SET_ERROR(pDevIns, rc,
5976 N_("AHCI cannot register PCI I/O region"));
5977
5978 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 4, 0x10, PCI_ADDRESS_SPACE_IO, ahciR3IdxDataIORangeMap);
5979 if (RT_FAILURE(rc))
5980 return PDMDEV_SET_ERROR(pDevIns, rc,
5981 N_("AHCI cannot register PCI I/O region for BMDMA"));
5982
5983 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 5, 4352, PCI_ADDRESS_SPACE_MEM, ahciR3MMIOMap);
5984 if (RT_FAILURE(rc))
5985 return PDMDEV_SET_ERROR(pDevIns, rc,
5986 N_("AHCI cannot register PCI memory region for registers"));
5987
5988 /* Create the timer for command completion coalescing feature. */
5989 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, ahciCccTimer, pThis,
5990 TMTIMER_FLAGS_NO_CRIT_SECT, "AHCI CCC Timer", &pThis->pHbaCccTimerR3);
5991 if (RT_FAILURE(rc))
5992 {
5993 AssertMsgFailed(("pfnTMTimerCreate -> %Rrc\n", rc));
5994 return rc;
5995 }
5996 pThis->pHbaCccTimerR0 = TMTimerR0Ptr(pThis->pHbaCccTimerR3);
5997 pThis->pHbaCccTimerRC = TMTimerRCPtr(pThis->pHbaCccTimerR3);
5998
5999 /* Status LUN. */
6000 pThis->IBase.pfnQueryInterface = ahciR3Status_QueryInterface;
6001 pThis->ILeds.pfnQueryStatusLed = ahciR3Status_QueryStatusLed;
6002
6003 /*
6004 * Create the notification queue.
6005 *
6006 * We need 2 items for every port because of SMP races.
6007 */
6008 rc = PDMDevHlpQueueCreate(pDevIns, sizeof(DEVPORTNOTIFIERQUEUEITEM), AHCI_MAX_NR_PORTS_IMPL * 2, 0,
6009 ahciNotifyQueueConsumer, true, "AHCI-Xmit", &pThis->pNotifierQueueR3);
6010 if (RT_FAILURE(rc))
6011 return rc;
6012 pThis->pNotifierQueueR0 = PDMQueueR0Ptr(pThis->pNotifierQueueR3);
6013 pThis->pNotifierQueueRC = PDMQueueRCPtr(pThis->pNotifierQueueR3);
6014
6015 /* Initialize static members on every port. */
6016 for (i = 0; i < AHCI_MAX_NR_PORTS_IMPL; i++)
6017 ahciPortHwReset(&pThis->ahciPort[i]);
6018
6019 /* Attach drivers to every available port. */
6020 for (i = 0; i < pThis->cPortsImpl; i++)
6021 {
6022 char szName[24];
6023 RTStrPrintf(szName, sizeof(szName), "Port%u", i);
6024
6025 PAHCIPort pAhciPort = &pThis->ahciPort[i];
6026 /*
6027 * Init interfaces.
6028 */
6029 pAhciPort->IBase.pfnQueryInterface = ahciR3PortQueryInterface;
6030 pAhciPort->IMediaExPort.pfnIoReqCompleteNotify = ahciR3IoReqCompleteNotify;
6031 pAhciPort->IMediaExPort.pfnIoReqCopyFromBuf = ahciR3IoReqCopyFromBuf;
6032 pAhciPort->IMediaExPort.pfnIoReqCopyToBuf = ahciR3IoReqCopyToBuf;
6033 pAhciPort->IMediaExPort.pfnIoReqQueryBuf = ahciR3IoReqQueryBuf;
6034 pAhciPort->IMediaExPort.pfnIoReqQueryDiscardRanges = ahciR3IoReqQueryDiscardRanges;
6035 pAhciPort->IMediaExPort.pfnIoReqStateChanged = ahciR3IoReqStateChanged;
6036 pAhciPort->IMediaExPort.pfnMediumEjected = ahciR3MediumEjected;
6037 pAhciPort->IPort.pfnQueryDeviceLocation = ahciR3PortQueryDeviceLocation;
6038 pAhciPort->fWrkThreadSleeping = true;
6039
6040 /* Query per port configuration options if available. */
6041 PCFGMNODE pCfgPort = CFGMR3GetChild(pDevIns->pCfg, szName);
6042 if (pCfgPort)
6043 {
6044 rc = CFGMR3QueryBoolDef(pCfgPort, "Hotpluggable", &pAhciPort->fHotpluggable, true);
6045 if (RT_FAILURE(rc))
6046 return PDMDEV_SET_ERROR(pDevIns, rc,
6047 N_("AHCI configuration error: failed to read Hotpluggable as boolean"));
6048 }
6049
6050 /*
6051 * Attach the block driver
6052 */
6053 rc = PDMDevHlpDriverAttach(pDevIns, pAhciPort->iLUN, &pAhciPort->IBase, &pAhciPort->pDrvBase, szName);
6054 if (RT_SUCCESS(rc))
6055 {
6056 rc = ahciR3ConfigureLUN(pDevIns, pAhciPort);
6057 if (RT_FAILURE(rc))
6058 {
6059 Log(("%s: Failed to configure the %s.\n", __FUNCTION__, szName));
6060 return rc;
6061 }
6062
6063 /* Mark that a device is present on that port */
6064 if (i < 6)
6065 pThis->dev.abConfig[0x93] |= (1 << i);
6066
6067 /*
6068 * Init vendor product data.
6069 */
6070 rc = ahciR3VpdInit(pDevIns, pAhciPort, szName);
6071 if (RT_FAILURE(rc))
6072 return rc;
6073
6074 rc = SUPSemEventCreate(pThis->pSupDrvSession, &pAhciPort->hEvtProcess);
6075 if (RT_FAILURE(rc))
6076 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
6077 N_("AHCI: Failed to create SUP event semaphore"));
6078
6079 rc = PDMDevHlpThreadCreate(pDevIns, &pAhciPort->pAsyncIOThread, pAhciPort, ahciAsyncIOLoop,
6080 ahciAsyncIOLoopWakeUp, 0, RTTHREADTYPE_IO, szName);
6081 if (RT_FAILURE(rc))
6082 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
6083 N_("AHCI: Failed to create worker thread %s"), szName);
6084 }
6085 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
6086 {
6087 pAhciPort->pDrvBase = NULL;
6088 rc = VINF_SUCCESS;
6089 LogRel(("AHCI: %s: No driver attached\n", szName));
6090 }
6091 else
6092 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
6093 N_("AHCI: Failed to attach drive to %s"), szName);
6094 }
6095
6096 /*
6097 * Attach status driver (optional).
6098 */
6099 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
6100 if (RT_SUCCESS(rc))
6101 {
6102 pThis->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
6103 pThis->pMediaNotify = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIANOTIFY);
6104 }
6105 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
6106 {
6107 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
6108 return PDMDEV_SET_ERROR(pDevIns, rc, N_("AHCI cannot attach to status driver"));
6109 }
6110 rc = PDMDevHlpSSMRegisterEx(pDevIns, AHCI_SAVED_STATE_VERSION, sizeof(*pThis) + cbTotalBufferSize, NULL,
6111 NULL, ahciR3LiveExec, NULL,
6112 ahciR3SavePrep, ahciR3SaveExec, NULL,
6113 ahciR3LoadPrep, ahciR3LoadExec, NULL);
6114 if (RT_FAILURE(rc))
6115 return rc;
6116
6117 /*
6118 * Register the info item.
6119 */
6120 char szTmp[128];
6121 RTStrPrintf(szTmp, sizeof(szTmp), "%s%d", pDevIns->pReg->szName, pDevIns->iInstance);
6122 PDMDevHlpDBGFInfoRegister(pDevIns, szTmp, "AHCI info", ahciR3Info);
6123
6124 return ahciR3ResetCommon(pDevIns);
6125}
6126
6127/**
6128 * The device registration structure.
6129 */
6130const PDMDEVREG g_DeviceAHCI =
6131{
6132 /* u32Version */
6133 PDM_DEVREG_VERSION,
6134 /* szName */
6135 "ahci",
6136 /* szRCMod */
6137 "VBoxDDRC.rc",
6138 /* szR0Mod */
6139 "VBoxDDR0.r0",
6140 /* pszDescription */
6141 "Intel AHCI controller.\n",
6142 /* fFlags */
6143 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0 |
6144 PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION | PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION |
6145 PDM_DEVREG_FLAGS_FIRST_RESET_NOTIFICATION,
6146 /* fClass */
6147 PDM_DEVREG_CLASS_STORAGE,
6148 /* cMaxInstances */
6149 ~0U,
6150 /* cbInstance */
6151 sizeof(AHCI),
6152 /* pfnConstruct */
6153 ahciR3Construct,
6154 /* pfnDestruct */
6155 ahciR3Destruct,
6156 /* pfnRelocate */
6157 ahciR3Relocate,
6158 /* pfnMemSetup */
6159 NULL,
6160 /* pfnPowerOn */
6161 NULL,
6162 /* pfnReset */
6163 ahciR3Reset,
6164 /* pfnSuspend */
6165 ahciR3Suspend,
6166 /* pfnResume */
6167 ahciR3Resume,
6168 /* pfnAttach */
6169 ahciR3Attach,
6170 /* pfnDetach */
6171 ahciR3Detach,
6172 /* pfnQueryInterface. */
6173 NULL,
6174 /* pfnInitComplete */
6175 NULL,
6176 /* pfnPowerOff */
6177 ahciR3PowerOff,
6178 /* pfnSoftReset */
6179 NULL,
6180 /* u32VersionEnd */
6181 PDM_DEVREG_VERSION
6182};
6183
6184#endif /* IN_RING3 */
6185#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette