VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DevATA.cpp@ 57583

Last change on this file since 57583 was 57372, checked in by vboxsync, 9 years ago

scm: fixes in previous cleanup run.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 299.6 KB
Line 
1/* $Id: DevATA.cpp 57372 2015-08-14 22:01:25Z vboxsync $ */
2/** @file
3 * VBox storage devices: ATA/ATAPI controller device (disk and cdrom).
4 */
5
6/*
7 * Copyright (C) 2006-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Defined Constants And Macros *
21*********************************************************************************************************************************/
22/** Temporary instrumentation for tracking down potential virtual disk
23 * write performance issues. */
24#undef VBOX_INSTRUMENT_DMA_WRITES
25
26/** @name The SSM saved state versions.
27 * @{
28 */
29/** The current saved state version. */
30#define ATA_SAVED_STATE_VERSION 20
31/** The saved state version used by VirtualBox 3.0.
32 * This lacks the config part and has the type at the and. */
33#define ATA_SAVED_STATE_VERSION_VBOX_30 19
34#define ATA_SAVED_STATE_VERSION_WITH_BOOL_TYPE 18
35#define ATA_SAVED_STATE_VERSION_WITHOUT_FULL_SENSE 16
36#define ATA_SAVED_STATE_VERSION_WITHOUT_EVENT_STATUS 17
37/** @} */
38
39
40/*********************************************************************************************************************************
41* Header Files *
42*********************************************************************************************************************************/
43#define LOG_GROUP LOG_GROUP_DEV_IDE
44#include <VBox/vmm/pdmdev.h>
45#include <iprt/assert.h>
46#include <iprt/string.h>
47#ifdef IN_RING3
48# include <iprt/uuid.h>
49# include <iprt/semaphore.h>
50# include <iprt/thread.h>
51# include <iprt/time.h>
52# include <iprt/alloc.h>
53#endif /* IN_RING3 */
54#include <iprt/critsect.h>
55#include <iprt/asm.h>
56#include <VBox/vmm/stam.h>
57#include <VBox/vmm/mm.h>
58#include <VBox/vmm/pgm.h>
59
60#include <VBox/sup.h>
61#include <VBox/scsi.h>
62
63#include "PIIX3ATABmDma.h"
64#include "ide.h"
65#include "ATAPIPassthrough.h"
66#include "VBoxDD.h"
67
68
69/*********************************************************************************************************************************
70* Defined Constants And Macros *
71*********************************************************************************************************************************/
72/**
73 * Maximum number of sectors to transfer in a READ/WRITE MULTIPLE request.
74 * Set to 1 to disable multi-sector read support. According to the ATA
75 * specification this must be a power of 2 and it must fit in an 8 bit
76 * value. Thus the only valid values are 1, 2, 4, 8, 16, 32, 64 and 128.
77 */
78#define ATA_MAX_MULT_SECTORS 128
79
80/**
81 * Fastest PIO mode supported by the drive.
82 */
83#define ATA_PIO_MODE_MAX 4
84/**
85 * Fastest MDMA mode supported by the drive.
86 */
87#define ATA_MDMA_MODE_MAX 2
88/**
89 * Fastest UDMA mode supported by the drive.
90 */
91#define ATA_UDMA_MODE_MAX 6
92
93/** ATAPI sense info size. */
94#define ATAPI_SENSE_SIZE 64
95
96/** The maximum number of release log entries per device. */
97#define MAX_LOG_REL_ERRORS 1024
98
99/* MediaEventStatus */
100#define ATA_EVENT_STATUS_UNCHANGED 0 /**< medium event status not changed */
101#define ATA_EVENT_STATUS_MEDIA_NEW 1 /**< new medium inserted */
102#define ATA_EVENT_STATUS_MEDIA_REMOVED 2 /**< medium removed */
103#define ATA_EVENT_STATUS_MEDIA_CHANGED 3 /**< medium was removed + new medium was inserted */
104#define ATA_EVENT_STATUS_MEDIA_EJECT_REQUESTED 4 /**< medium eject requested (eject button pressed) */
105
106/* Media track type */
107#define ATA_MEDIA_TYPE_UNKNOWN 0 /**< unknown CD type */
108#define ATA_MEDIA_NO_DISC 0x70 /**< Door closed, no medium */
109
110
111/*********************************************************************************************************************************
112* Structures and Typedefs *
113*********************************************************************************************************************************/
114/**
115 * The state of an ATA device.
116 *
117 * @implements PDMIBASE
118 * @implements PDMIBLOCKPORT
119 * @implements PDMIMOUNTNOTIFY
120 */
121typedef struct ATADevState
122{
123 /** Flag indicating whether the current command uses LBA48 mode. */
124 bool fLBA48;
125 /** Flag indicating whether this drive implements the ATAPI command set. */
126 bool fATAPI;
127 /** Set if this interface has asserted the IRQ. */
128 bool fIrqPending;
129 /** Currently configured number of sectors in a multi-sector transfer. */
130 uint8_t cMultSectors;
131 /** PCHS disk geometry. */
132 PDMMEDIAGEOMETRY PCHSGeometry;
133 /** Total number of sectors on this disk. */
134 uint64_t cTotalSectors;
135 /** Sector size of the medium. */
136 uint32_t cbSector;
137 /** Number of sectors to transfer per IRQ. */
138 uint32_t cSectorsPerIRQ;
139
140 /** ATA/ATAPI register 1: feature (write-only). */
141 uint8_t uATARegFeature;
142 /** ATA/ATAPI register 1: feature, high order byte. */
143 uint8_t uATARegFeatureHOB;
144 /** ATA/ATAPI register 1: error (read-only). */
145 uint8_t uATARegError;
146 /** ATA/ATAPI register 2: sector count (read/write). */
147 uint8_t uATARegNSector;
148 /** ATA/ATAPI register 2: sector count, high order byte. */
149 uint8_t uATARegNSectorHOB;
150 /** ATA/ATAPI register 3: sector (read/write). */
151 uint8_t uATARegSector;
152 /** ATA/ATAPI register 3: sector, high order byte. */
153 uint8_t uATARegSectorHOB;
154 /** ATA/ATAPI register 4: cylinder low (read/write). */
155 uint8_t uATARegLCyl;
156 /** ATA/ATAPI register 4: cylinder low, high order byte. */
157 uint8_t uATARegLCylHOB;
158 /** ATA/ATAPI register 5: cylinder high (read/write). */
159 uint8_t uATARegHCyl;
160 /** ATA/ATAPI register 5: cylinder high, high order byte. */
161 uint8_t uATARegHCylHOB;
162 /** ATA/ATAPI register 6: select drive/head (read/write). */
163 uint8_t uATARegSelect;
164 /** ATA/ATAPI register 7: status (read-only). */
165 uint8_t uATARegStatus;
166 /** ATA/ATAPI register 7: command (write-only). */
167 uint8_t uATARegCommand;
168 /** ATA/ATAPI drive control register (write-only). */
169 uint8_t uATARegDevCtl;
170
171 /** Currently active transfer mode (MDMA/UDMA) and speed. */
172 uint8_t uATATransferMode;
173 /** Current transfer direction. */
174 uint8_t uTxDir;
175 /** Index of callback for begin transfer. */
176 uint8_t iBeginTransfer;
177 /** Index of callback for source/sink of data. */
178 uint8_t iSourceSink;
179 /** Flag indicating whether the current command transfers data in DMA mode. */
180 bool fDMA;
181 /** Set to indicate that ATAPI transfer semantics must be used. */
182 bool fATAPITransfer;
183
184 /** Total ATA/ATAPI transfer size, shared PIO/DMA. */
185 uint32_t cbTotalTransfer;
186 /** Elementary ATA/ATAPI transfer size, shared PIO/DMA. */
187 uint32_t cbElementaryTransfer;
188 /** Maximum ATAPI elementary transfer size, PIO only. */
189 uint32_t cbPIOTransferLimit;
190 /** ATAPI passthrough transfer size, shared PIO/DMA */
191 uint32_t cbAtapiPassthroughTransfer;
192 /** Current read/write buffer position, shared PIO/DMA. */
193 uint32_t iIOBufferCur;
194 /** First element beyond end of valid buffer content, shared PIO/DMA. */
195 uint32_t iIOBufferEnd;
196 /** Align the following fields correctly. */
197 uint32_t Alignment0;
198
199 /** ATA/ATAPI current PIO read/write transfer position. Not shared with DMA for safety reasons. */
200 uint32_t iIOBufferPIODataStart;
201 /** ATA/ATAPI current PIO read/write transfer end. Not shared with DMA for safety reasons. */
202 uint32_t iIOBufferPIODataEnd;
203
204 /** ATAPI current LBA position. */
205 uint32_t iATAPILBA;
206 /** ATAPI current sector size. */
207 uint32_t cbATAPISector;
208 /** ATAPI current command. */
209 uint8_t aATAPICmd[ATAPI_PACKET_SIZE];
210 /** ATAPI sense data. */
211 uint8_t abATAPISense[ATAPI_SENSE_SIZE];
212 /** HACK: Countdown till we report a newly unmounted drive as mounted. */
213 uint8_t cNotifiedMediaChange;
214 /** The same for GET_EVENT_STATUS for mechanism */
215 volatile uint32_t MediaEventStatus;
216
217 /** Media type if known. */
218 volatile uint32_t MediaTrackType;
219
220 /** The status LED state for this drive. */
221 PDMLED Led;
222
223 /** Size of I/O buffer. */
224 uint32_t cbIOBuffer;
225 /** Pointer to the I/O buffer. */
226 R3PTRTYPE(uint8_t *) pbIOBufferR3;
227 /** Pointer to the I/O buffer. */
228 R0PTRTYPE(uint8_t *) pbIOBufferR0;
229 /** Pointer to the I/O buffer. */
230 RCPTRTYPE(uint8_t *) pbIOBufferRC;
231
232 RTRCPTR Aligmnent1; /**< Align the statistics at an 8-byte boundary. */
233
234 /*
235 * No data that is part of the saved state after this point!!!!!
236 */
237
238 /* Release statistics: number of ATA DMA commands. */
239 STAMCOUNTER StatATADMA;
240 /* Release statistics: number of ATA PIO commands. */
241 STAMCOUNTER StatATAPIO;
242 /* Release statistics: number of ATAPI PIO commands. */
243 STAMCOUNTER StatATAPIDMA;
244 /* Release statistics: number of ATAPI PIO commands. */
245 STAMCOUNTER StatATAPIPIO;
246#ifdef VBOX_INSTRUMENT_DMA_WRITES
247 /* Release statistics: number of DMA sector writes and the time spent. */
248 STAMPROFILEADV StatInstrVDWrites;
249#endif
250
251 /** Statistics: number of read operations and the time spent reading. */
252 STAMPROFILEADV StatReads;
253 /** Statistics: number of bytes read. */
254 STAMCOUNTER StatBytesRead;
255 /** Statistics: number of write operations and the time spent writing. */
256 STAMPROFILEADV StatWrites;
257 /** Statistics: number of bytes written. */
258 STAMCOUNTER StatBytesWritten;
259 /** Statistics: number of flush operations and the time spend flushing. */
260 STAMPROFILE StatFlushes;
261
262 /** Mark the drive as having a non-rotational medium (i.e. as a SSD). */
263 bool fNonRotational;
264 /** Enable passing through commands directly to the ATAPI drive. */
265 bool fATAPIPassthrough;
266 /** Flag whether to overwrite inquiry data in passthrough mode. */
267 bool fOverwriteInquiry;
268 /** Number of errors we've reported to the release log.
269 * This is to prevent flooding caused by something going horribly wrong.
270 * this value against MAX_LOG_REL_ERRORS in places likely to cause floods
271 * like the ones we currently seeing on the linux smoke tests (2006-11-10). */
272 uint32_t cErrors;
273 /** Timestamp of last started command. 0 if no command pending. */
274 uint64_t u64CmdTS;
275
276 /** Pointer to the attached driver's base interface. */
277 R3PTRTYPE(PPDMIBASE) pDrvBase;
278 /** Pointer to the attached driver's block interface. */
279 R3PTRTYPE(PPDMIBLOCK) pDrvBlock;
280 /** Pointer to the attached driver's block bios interface. */
281 R3PTRTYPE(PPDMIBLOCKBIOS) pDrvBlockBios;
282 /** Pointer to the attached driver's mount interface.
283 * This is NULL if the driver isn't a removable unit. */
284 R3PTRTYPE(PPDMIMOUNT) pDrvMount;
285 /** The base interface. */
286 PDMIBASE IBase;
287 /** The block port interface. */
288 PDMIBLOCKPORT IPort;
289 /** The mount notify interface. */
290 PDMIMOUNTNOTIFY IMountNotify;
291 /** The LUN #. */
292 RTUINT iLUN;
293 RTUINT Alignment2; /**< Align pDevInsR3 correctly. */
294 /** Pointer to device instance. */
295 PPDMDEVINSR3 pDevInsR3;
296 /** Pointer to controller instance. */
297 R3PTRTYPE(struct ATACONTROLLER *) pControllerR3;
298 /** Pointer to device instance. */
299 PPDMDEVINSR0 pDevInsR0;
300 /** Pointer to controller instance. */
301 R0PTRTYPE(struct ATACONTROLLER *) pControllerR0;
302 /** Pointer to device instance. */
303 PPDMDEVINSRC pDevInsRC;
304 /** Pointer to controller instance. */
305 RCPTRTYPE(struct ATACONTROLLER *) pControllerRC;
306
307 /** The serial number to use for IDENTIFY DEVICE commands. */
308 char szSerialNumber[ATA_SERIAL_NUMBER_LENGTH+1];
309 /** The firmware revision to use for IDENTIFY DEVICE commands. */
310 char szFirmwareRevision[ATA_FIRMWARE_REVISION_LENGTH+1];
311 /** The model number to use for IDENTIFY DEVICE commands. */
312 char szModelNumber[ATA_MODEL_NUMBER_LENGTH+1];
313 /** The vendor identification string for SCSI INQUIRY commands. */
314 char szInquiryVendorId[ATAPI_INQUIRY_VENDOR_ID_LENGTH+1];
315 /** The product identification string for SCSI INQUIRY commands. */
316 char szInquiryProductId[ATAPI_INQUIRY_PRODUCT_ID_LENGTH+1];
317 /** The revision string for SCSI INQUIRY commands. */
318 char szInquiryRevision[ATAPI_INQUIRY_REVISION_LENGTH+1];
319 /** The current tracklist of the loaded medium if passthrough is used. */
320 R3PTRTYPE(PTRACKLIST) pTrackList;
321
322 uint8_t abAlignment4[HC_ARCH_BITS == 64 ? 7 : 3];
323} ATADevState;
324AssertCompileMemberAlignment(ATADevState, cTotalSectors, 8);
325AssertCompileMemberAlignment(ATADevState, StatATADMA, 8);
326AssertCompileMemberAlignment(ATADevState, u64CmdTS, 8);
327AssertCompileMemberAlignment(ATADevState, pDevInsR3, 8);
328AssertCompileMemberAlignment(ATADevState, szSerialNumber, 8);
329AssertCompileSizeAlignment(ATADevState, 8);
330
331
332typedef struct ATATransferRequest
333{
334 uint8_t iIf;
335 uint8_t iBeginTransfer;
336 uint8_t iSourceSink;
337 uint32_t cbTotalTransfer;
338 uint8_t uTxDir;
339} ATATransferRequest;
340
341
342typedef struct ATAAbortRequest
343{
344 uint8_t iIf;
345 bool fResetDrive;
346} ATAAbortRequest;
347
348
349typedef enum
350{
351 /** Begin a new transfer. */
352 ATA_AIO_NEW = 0,
353 /** Continue a DMA transfer. */
354 ATA_AIO_DMA,
355 /** Continue a PIO transfer. */
356 ATA_AIO_PIO,
357 /** Reset the drives on current controller, stop all transfer activity. */
358 ATA_AIO_RESET_ASSERTED,
359 /** Reset the drives on current controller, resume operation. */
360 ATA_AIO_RESET_CLEARED,
361 /** Abort the current transfer of a particular drive. */
362 ATA_AIO_ABORT
363} ATAAIO;
364
365
366typedef struct ATARequest
367{
368 ATAAIO ReqType;
369 union
370 {
371 ATATransferRequest t;
372 ATAAbortRequest a;
373 } u;
374} ATARequest;
375
376
377typedef struct ATACONTROLLER
378{
379 /** The base of the first I/O Port range. */
380 RTIOPORT IOPortBase1;
381 /** The base of the second I/O Port range. (0 if none) */
382 RTIOPORT IOPortBase2;
383 /** The assigned IRQ. */
384 RTUINT irq;
385 /** Access critical section */
386 PDMCRITSECT lock;
387
388 /** Selected drive. */
389 uint8_t iSelectedIf;
390 /** The interface on which to handle async I/O. */
391 uint8_t iAIOIf;
392 /** The state of the async I/O thread. */
393 uint8_t uAsyncIOState;
394 /** Flag indicating whether the next transfer is part of the current command. */
395 bool fChainedTransfer;
396 /** Set when the reset processing is currently active on this controller. */
397 bool fReset;
398 /** Flag whether the current transfer needs to be redone. */
399 bool fRedo;
400 /** Flag whether the redo suspend has been finished. */
401 bool fRedoIdle;
402 /** Flag whether the DMA operation to be redone is the final transfer. */
403 bool fRedoDMALastDesc;
404 /** The BusMaster DMA state. */
405 BMDMAState BmDma;
406 /** Pointer to first DMA descriptor. */
407 RTGCPHYS32 pFirstDMADesc;
408 /** Pointer to last DMA descriptor. */
409 RTGCPHYS32 pLastDMADesc;
410 /** Pointer to current DMA buffer (for redo operations). */
411 RTGCPHYS32 pRedoDMABuffer;
412 /** Size of current DMA buffer (for redo operations). */
413 uint32_t cbRedoDMABuffer;
414
415 /** The ATA/ATAPI interfaces of this controller. */
416 ATADevState aIfs[2];
417
418 /** Pointer to device instance. */
419 PPDMDEVINSR3 pDevInsR3;
420 /** Pointer to device instance. */
421 PPDMDEVINSR0 pDevInsR0;
422 /** Pointer to device instance. */
423 PPDMDEVINSRC pDevInsRC;
424
425 /** Set when the destroying the device instance and the thread must exit. */
426 uint32_t volatile fShutdown;
427 /** The async I/O thread handle. NIL_RTTHREAD if no thread. */
428 RTTHREAD AsyncIOThread;
429 /** The event semaphore the thread is waiting on for requests. */
430 SUPSEMEVENT hAsyncIOSem;
431 /** The support driver session handle. */
432 PSUPDRVSESSION pSupDrvSession;
433 /** The request queue for the AIO thread. One element is always unused. */
434 ATARequest aAsyncIORequests[4];
435 /** The position at which to insert a new request for the AIO thread. */
436 volatile uint8_t AsyncIOReqHead;
437 /** The position at which to get a new request for the AIO thread. */
438 volatile uint8_t AsyncIOReqTail;
439 /** Whether to call PDMDevHlpAsyncNotificationCompleted when idle. */
440 bool volatile fSignalIdle;
441 uint8_t Alignment3[1]; /**< Explicit padding of the 1 byte gap. */
442 /** Magic delay before triggering interrupts in DMA mode. */
443 uint32_t DelayIRQMillies;
444 /** The event semaphore the thread is waiting on during suspended I/O. */
445 RTSEMEVENT SuspendIOSem;
446 /** The lock protecting the request queue. */
447 PDMCRITSECT AsyncIORequestLock;
448#if 0 /*HC_ARCH_BITS == 32*/
449 uint32_t Alignment0;
450#endif
451
452 /** Timestamp we started the reset. */
453 uint64_t u64ResetTime;
454
455 /* Statistics */
456 STAMCOUNTER StatAsyncOps;
457 uint64_t StatAsyncMinWait;
458 uint64_t StatAsyncMaxWait;
459 STAMCOUNTER StatAsyncTimeUS;
460 STAMPROFILEADV StatAsyncTime;
461 STAMPROFILE StatLockWait;
462} ATACONTROLLER, *PATACONTROLLER;
463AssertCompileMemberAlignment(ATACONTROLLER, lock, 8);
464AssertCompileMemberAlignment(ATACONTROLLER, aIfs, 8);
465AssertCompileMemberAlignment(ATACONTROLLER, u64ResetTime, 8);
466AssertCompileMemberAlignment(ATACONTROLLER, StatAsyncOps, 8);
467AssertCompileMemberAlignment(ATACONTROLLER, AsyncIORequestLock, 8);
468AssertCompileSizeAlignment(ATACONTROLLER, 8);
469
470typedef enum CHIPSET
471{
472 /** PIIX3 chipset, must be 0 for saved state compatibility */
473 CHIPSET_PIIX3 = 0,
474 /** PIIX4 chipset, must be 1 for saved state compatibility */
475 CHIPSET_PIIX4 = 1,
476 /** ICH6 chipset */
477 CHIPSET_ICH6 = 2
478} CHIPSET;
479
480/**
481 * The state of the ATA PCI device.
482 *
483 * @extends PCIDEVICE
484 * @implements PDMILEDPORTS
485 */
486typedef struct PCIATAState
487{
488 PCIDEVICE dev;
489 /** The controllers. */
490 ATACONTROLLER aCts[2];
491 /** Pointer to device instance. */
492 PPDMDEVINSR3 pDevIns;
493 /** Status LUN: Base interface. */
494 PDMIBASE IBase;
495 /** Status LUN: Leds interface. */
496 PDMILEDPORTS ILeds;
497 /** Status LUN: Partner of ILeds. */
498 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
499 /** Status LUN: Media Notify. */
500 R3PTRTYPE(PPDMIMEDIANOTIFY) pMediaNotify;
501 /** Flag whether RC is enabled. */
502 bool fRCEnabled;
503 /** Flag whether R0 is enabled. */
504 bool fR0Enabled;
505 /** Flag indicating chipset being emulated. */
506 uint8_t u8Type;
507 bool Alignment0[HC_ARCH_BITS == 64 ? 5 : 1 ]; /**< Align the struct size. */
508} PCIATAState;
509
510#define PDMIBASE_2_PCIATASTATE(pInterface) ( (PCIATAState *)((uintptr_t)(pInterface) - RT_OFFSETOF(PCIATAState, IBase)) )
511#define PDMILEDPORTS_2_PCIATASTATE(pInterface) ( (PCIATAState *)((uintptr_t)(pInterface) - RT_OFFSETOF(PCIATAState, ILeds)) )
512#define PDMIBLOCKPORT_2_ATASTATE(pInterface) ( (ATADevState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ATADevState, IPort)) )
513#define PDMIMOUNT_2_ATASTATE(pInterface) ( (ATADevState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ATADevState, IMount)) )
514#define PDMIMOUNTNOTIFY_2_ATASTATE(pInterface) ( (ATADevState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ATADevState, IMountNotify)) )
515#define PCIDEV_2_PCIATASTATE(pPciDev) ( (PCIATAState *)(pPciDev) )
516
517#define ATACONTROLLER_IDX(pController) ( (pController) - PDMINS_2_DATA(CONTROLLER_2_DEVINS(pController), PCIATAState *)->aCts )
518
519#define ATADEVSTATE_2_CONTROLLER(pIf) ( (pIf)->CTX_SUFF(pController) )
520#define ATADEVSTATE_2_DEVINS(pIf) ( (pIf)->CTX_SUFF(pDevIns) )
521#define CONTROLLER_2_DEVINS(pController) ( (pController)->CTX_SUFF(pDevIns) )
522#define PDMIBASE_2_ATASTATE(pInterface) ( (ATADevState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ATADevState, IBase)) )
523#define PDMIBLOCKPORT_2_ATASTATE(pInterface) ( (ATADevState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ATADevState, IPort)) )
524
525#ifndef VBOX_DEVICE_STRUCT_TESTCASE
526
527
528/*********************************************************************************************************************************
529* Internal Functions *
530*********************************************************************************************************************************/
531RT_C_DECLS_BEGIN
532
533PDMBOTHCBDECL(int) ataIOPortWrite1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
534PDMBOTHCBDECL(int) ataIOPortRead1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *u32, unsigned cb);
535PDMBOTHCBDECL(int) ataIOPortWriteStr1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint8_t const *pbSrc,
536 uint32_t *pcTransfers, unsigned cb);
537PDMBOTHCBDECL(int) ataIOPortReadStr1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint8_t *pbDst,
538 uint32_t *pcTransfers, unsigned cb);
539PDMBOTHCBDECL(int) ataIOPortWrite1Other(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
540PDMBOTHCBDECL(int) ataIOPortRead1Other(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *u32, unsigned cb);
541PDMBOTHCBDECL(int) ataIOPortWrite2(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
542PDMBOTHCBDECL(int) ataIOPortRead2(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *u32, unsigned cb);
543PDMBOTHCBDECL(int) ataBMDMAIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
544PDMBOTHCBDECL(int) ataBMDMAIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
545RT_C_DECLS_END
546
547
548
549DECLINLINE(void) ataSetStatusValue(ATADevState *s, uint8_t stat)
550{
551 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
552
553 /* Freeze status register contents while processing RESET. */
554 if (!pCtl->fReset)
555 {
556 s->uATARegStatus = stat;
557 Log2(("%s: LUN#%d status %#04x\n", __FUNCTION__, s->iLUN, s->uATARegStatus));
558 }
559}
560
561
562DECLINLINE(void) ataSetStatus(ATADevState *s, uint8_t stat)
563{
564 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
565
566 /* Freeze status register contents while processing RESET. */
567 if (!pCtl->fReset)
568 {
569 s->uATARegStatus |= stat;
570 Log2(("%s: LUN#%d status %#04x\n", __FUNCTION__, s->iLUN, s->uATARegStatus));
571 }
572}
573
574
575DECLINLINE(void) ataUnsetStatus(ATADevState *s, uint8_t stat)
576{
577 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
578
579 /* Freeze status register contents while processing RESET. */
580 if (!pCtl->fReset)
581 {
582 s->uATARegStatus &= ~stat;
583 Log2(("%s: LUN#%d status %#04x\n", __FUNCTION__, s->iLUN, s->uATARegStatus));
584 }
585}
586
587#if defined(IN_RING3) || defined(IN_RING0)
588
589# ifdef IN_RING3
590typedef void (*PBeginTransferFunc)(ATADevState *);
591typedef bool (*PSourceSinkFunc)(ATADevState *);
592
593static void ataR3ReadWriteSectorsBT(ATADevState *);
594static void ataR3PacketBT(ATADevState *);
595static void atapiR3CmdBT(ATADevState *);
596static void atapiR3PassthroughCmdBT(ATADevState *);
597
598static bool ataR3IdentifySS(ATADevState *);
599static bool ataR3FlushSS(ATADevState *);
600static bool ataR3ReadSectorsSS(ATADevState *);
601static bool ataR3WriteSectorsSS(ATADevState *);
602static bool ataR3ExecuteDeviceDiagnosticSS(ATADevState *);
603static bool ataR3TrimSS(ATADevState *);
604static bool ataR3PacketSS(ATADevState *);
605static bool atapiR3GetConfigurationSS(ATADevState *);
606static bool atapiR3GetEventStatusNotificationSS(ATADevState *);
607static bool atapiR3IdentifySS(ATADevState *);
608static bool atapiR3InquirySS(ATADevState *);
609static bool atapiR3MechanismStatusSS(ATADevState *);
610static bool atapiR3ModeSenseErrorRecoverySS(ATADevState *);
611static bool atapiR3ModeSenseCDStatusSS(ATADevState *);
612static bool atapiR3ReadSS(ATADevState *);
613static bool atapiR3ReadCapacitySS(ATADevState *);
614static bool atapiR3ReadDiscInformationSS(ATADevState *);
615static bool atapiR3ReadTOCNormalSS(ATADevState *);
616static bool atapiR3ReadTOCMultiSS(ATADevState *);
617static bool atapiR3ReadTOCRawSS(ATADevState *);
618static bool atapiR3ReadTrackInformationSS(ATADevState *);
619static bool atapiR3RequestSenseSS(ATADevState *);
620static bool atapiR3PassthroughSS(ATADevState *);
621static bool atapiR3ReadDVDStructureSS(ATADevState *);
622# endif /* IN_RING3 */
623
624/**
625 * Begin of transfer function indexes for g_apfnBeginTransFuncs.
626 */
627typedef enum ATAFNBT
628{
629 ATAFN_BT_NULL = 0,
630 ATAFN_BT_READ_WRITE_SECTORS,
631 ATAFN_BT_PACKET,
632 ATAFN_BT_ATAPI_CMD,
633 ATAFN_BT_ATAPI_PASSTHROUGH_CMD,
634 ATAFN_BT_MAX
635} ATAFNBT;
636
637# ifdef IN_RING3
638/**
639 * Array of end transfer functions, the index is ATAFNET.
640 * Make sure ATAFNET and this array match!
641 */
642static const PBeginTransferFunc g_apfnBeginTransFuncs[ATAFN_BT_MAX] =
643{
644 NULL,
645 ataR3ReadWriteSectorsBT,
646 ataR3PacketBT,
647 atapiR3CmdBT,
648 atapiR3PassthroughCmdBT,
649};
650# endif /* IN_RING3 */
651
652/**
653 * Source/sink function indexes for g_apfnSourceSinkFuncs.
654 */
655typedef enum ATAFNSS
656{
657 ATAFN_SS_NULL = 0,
658 ATAFN_SS_IDENTIFY,
659 ATAFN_SS_FLUSH,
660 ATAFN_SS_READ_SECTORS,
661 ATAFN_SS_WRITE_SECTORS,
662 ATAFN_SS_EXECUTE_DEVICE_DIAGNOSTIC,
663 ATAFN_SS_TRIM,
664 ATAFN_SS_PACKET,
665 ATAFN_SS_ATAPI_GET_CONFIGURATION,
666 ATAFN_SS_ATAPI_GET_EVENT_STATUS_NOTIFICATION,
667 ATAFN_SS_ATAPI_IDENTIFY,
668 ATAFN_SS_ATAPI_INQUIRY,
669 ATAFN_SS_ATAPI_MECHANISM_STATUS,
670 ATAFN_SS_ATAPI_MODE_SENSE_ERROR_RECOVERY,
671 ATAFN_SS_ATAPI_MODE_SENSE_CD_STATUS,
672 ATAFN_SS_ATAPI_READ,
673 ATAFN_SS_ATAPI_READ_CAPACITY,
674 ATAFN_SS_ATAPI_READ_DISC_INFORMATION,
675 ATAFN_SS_ATAPI_READ_TOC_NORMAL,
676 ATAFN_SS_ATAPI_READ_TOC_MULTI,
677 ATAFN_SS_ATAPI_READ_TOC_RAW,
678 ATAFN_SS_ATAPI_READ_TRACK_INFORMATION,
679 ATAFN_SS_ATAPI_REQUEST_SENSE,
680 ATAFN_SS_ATAPI_PASSTHROUGH,
681 ATAFN_SS_ATAPI_READ_DVD_STRUCTURE,
682 ATAFN_SS_MAX
683} ATAFNSS;
684
685# ifdef IN_RING3
686/**
687 * Array of source/sink functions, the index is ATAFNSS.
688 * Make sure ATAFNSS and this array match!
689 */
690static const PSourceSinkFunc g_apfnSourceSinkFuncs[ATAFN_SS_MAX] =
691{
692 NULL,
693 ataR3IdentifySS,
694 ataR3FlushSS,
695 ataR3ReadSectorsSS,
696 ataR3WriteSectorsSS,
697 ataR3ExecuteDeviceDiagnosticSS,
698 ataR3TrimSS,
699 ataR3PacketSS,
700 atapiR3GetConfigurationSS,
701 atapiR3GetEventStatusNotificationSS,
702 atapiR3IdentifySS,
703 atapiR3InquirySS,
704 atapiR3MechanismStatusSS,
705 atapiR3ModeSenseErrorRecoverySS,
706 atapiR3ModeSenseCDStatusSS,
707 atapiR3ReadSS,
708 atapiR3ReadCapacitySS,
709 atapiR3ReadDiscInformationSS,
710 atapiR3ReadTOCNormalSS,
711 atapiR3ReadTOCMultiSS,
712 atapiR3ReadTOCRawSS,
713 atapiR3ReadTrackInformationSS,
714 atapiR3RequestSenseSS,
715 atapiR3PassthroughSS,
716 atapiR3ReadDVDStructureSS
717};
718# endif /* IN_RING3 */
719
720
721static const ATARequest g_ataDMARequest = { ATA_AIO_DMA, { { 0, 0, 0, 0, 0 } } };
722static const ATARequest g_ataPIORequest = { ATA_AIO_PIO, { { 0, 0, 0, 0, 0 } } };
723static const ATARequest g_ataResetARequest = { ATA_AIO_RESET_ASSERTED, { { 0, 0, 0, 0, 0 } } };
724static const ATARequest g_ataResetCRequest = { ATA_AIO_RESET_CLEARED, { { 0, 0, 0, 0, 0 } } };
725
726# ifdef IN_RING3
727static void ataR3AsyncIOClearRequests(PATACONTROLLER pCtl)
728{
729 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
730 AssertRC(rc);
731
732 pCtl->AsyncIOReqHead = 0;
733 pCtl->AsyncIOReqTail = 0;
734
735 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
736 AssertRC(rc);
737}
738# endif /* IN_RING3 */
739
740static void ataHCAsyncIOPutRequest(PATACONTROLLER pCtl, const ATARequest *pReq)
741{
742 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
743 AssertRC(rc);
744
745 Assert((pCtl->AsyncIOReqHead + 1) % RT_ELEMENTS(pCtl->aAsyncIORequests) != pCtl->AsyncIOReqTail);
746 memcpy(&pCtl->aAsyncIORequests[pCtl->AsyncIOReqHead], pReq, sizeof(*pReq));
747 pCtl->AsyncIOReqHead++;
748 pCtl->AsyncIOReqHead %= RT_ELEMENTS(pCtl->aAsyncIORequests);
749
750 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
751 AssertRC(rc);
752
753 rc = PDMHCCritSectScheduleExitEvent(&pCtl->lock, pCtl->hAsyncIOSem);
754 if (RT_FAILURE(rc))
755 {
756 rc = SUPSemEventSignal(pCtl->pSupDrvSession, pCtl->hAsyncIOSem);
757 AssertRC(rc);
758 }
759}
760
761# ifdef IN_RING3
762
763static const ATARequest *ataR3AsyncIOGetCurrentRequest(PATACONTROLLER pCtl)
764{
765 const ATARequest *pReq;
766
767 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
768 AssertRC(rc);
769
770 if (pCtl->AsyncIOReqHead != pCtl->AsyncIOReqTail)
771 pReq = &pCtl->aAsyncIORequests[pCtl->AsyncIOReqTail];
772 else
773 pReq = NULL;
774
775 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
776 AssertRC(rc);
777 return pReq;
778}
779
780
781/**
782 * Remove the request with the given type, as it's finished. The request
783 * is not removed blindly, as this could mean a RESET request that is not
784 * yet processed (but has cleared the request queue) is lost.
785 *
786 * @param pCtl Controller for which to remove the request.
787 * @param ReqType Type of the request to remove.
788 */
789static void ataR3AsyncIORemoveCurrentRequest(PATACONTROLLER pCtl, ATAAIO ReqType)
790{
791 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
792 AssertRC(rc);
793
794 if (pCtl->AsyncIOReqHead != pCtl->AsyncIOReqTail && pCtl->aAsyncIORequests[pCtl->AsyncIOReqTail].ReqType == ReqType)
795 {
796 pCtl->AsyncIOReqTail++;
797 pCtl->AsyncIOReqTail %= RT_ELEMENTS(pCtl->aAsyncIORequests);
798 }
799
800 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
801 AssertRC(rc);
802}
803
804
805/**
806 * Dump the request queue for a particular controller. First dump the queue
807 * contents, then the already processed entries, as long as they haven't been
808 * overwritten.
809 *
810 * @param pCtl Controller for which to dump the queue.
811 */
812static void ataR3AsyncIODumpRequests(PATACONTROLLER pCtl)
813{
814 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
815 AssertRC(rc);
816
817 LogRel(("PIIX3 ATA: Ctl#%d: request queue dump (topmost is current):\n", ATACONTROLLER_IDX(pCtl)));
818 uint8_t curr = pCtl->AsyncIOReqTail;
819 do
820 {
821 if (curr == pCtl->AsyncIOReqHead)
822 LogRel(("PIIX3 ATA: Ctl#%d: processed requests (topmost is oldest):\n", ATACONTROLLER_IDX(pCtl)));
823 switch (pCtl->aAsyncIORequests[curr].ReqType)
824 {
825 case ATA_AIO_NEW:
826 LogRel(("new transfer request, iIf=%d iBeginTransfer=%d iSourceSink=%d cbTotalTransfer=%d uTxDir=%d\n", pCtl->aAsyncIORequests[curr].u.t.iIf, pCtl->aAsyncIORequests[curr].u.t.iBeginTransfer, pCtl->aAsyncIORequests[curr].u.t.iSourceSink, pCtl->aAsyncIORequests[curr].u.t.cbTotalTransfer, pCtl->aAsyncIORequests[curr].u.t.uTxDir));
827 break;
828 case ATA_AIO_DMA:
829 LogRel(("dma transfer continuation\n"));
830 break;
831 case ATA_AIO_PIO:
832 LogRel(("pio transfer continuation\n"));
833 break;
834 case ATA_AIO_RESET_ASSERTED:
835 LogRel(("reset asserted request\n"));
836 break;
837 case ATA_AIO_RESET_CLEARED:
838 LogRel(("reset cleared request\n"));
839 break;
840 case ATA_AIO_ABORT:
841 LogRel(("abort request, iIf=%d fResetDrive=%d\n", pCtl->aAsyncIORequests[curr].u.a.iIf, pCtl->aAsyncIORequests[curr].u.a.fResetDrive));
842 break;
843 default:
844 LogRel(("unknown request %d\n", pCtl->aAsyncIORequests[curr].ReqType));
845 }
846 curr = (curr + 1) % RT_ELEMENTS(pCtl->aAsyncIORequests);
847 } while (curr != pCtl->AsyncIOReqTail);
848
849 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
850 AssertRC(rc);
851}
852
853
854/**
855 * Checks whether the request queue for a particular controller is empty
856 * or whether a particular controller is idle.
857 *
858 * @param pCtl Controller for which to check the queue.
859 * @param fStrict If set then the controller is checked to be idle.
860 */
861static bool ataR3AsyncIOIsIdle(PATACONTROLLER pCtl, bool fStrict)
862{
863 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
864 AssertRC(rc);
865
866 bool fIdle = pCtl->fRedoIdle;
867 if (!fIdle)
868 fIdle = (pCtl->AsyncIOReqHead == pCtl->AsyncIOReqTail);
869 if (fStrict)
870 fIdle &= (pCtl->uAsyncIOState == ATA_AIO_NEW);
871
872 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
873 AssertRC(rc);
874 return fIdle;
875}
876
877
878/**
879 * Send a transfer request to the async I/O thread.
880 *
881 * @param s Pointer to the ATA device state data.
882 * @param cbTotalTransfer Data transfer size.
883 * @param uTxDir Data transfer direction.
884 * @param iBeginTransfer Index of BeginTransfer callback.
885 * @param iSourceSink Index of SourceSink callback.
886 * @param fChainedTransfer Whether this is a transfer that is part of the previous command/transfer.
887 */
888static void ataR3StartTransfer(ATADevState *s, uint32_t cbTotalTransfer, uint8_t uTxDir, ATAFNBT iBeginTransfer,
889 ATAFNSS iSourceSink, bool fChainedTransfer)
890{
891 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
892 ATARequest Req;
893
894 Assert(PDMCritSectIsOwner(&pCtl->lock));
895
896 /* Do not issue new requests while the RESET line is asserted. */
897 if (pCtl->fReset)
898 {
899 Log2(("%s: Ctl#%d: suppressed new request as RESET is active\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
900 return;
901 }
902
903 /* If the controller is already doing something else right now, ignore
904 * the command that is being submitted. Some broken guests issue commands
905 * twice (e.g. the Linux kernel that comes with Acronis True Image 8). */
906 if (!fChainedTransfer && !ataR3AsyncIOIsIdle(pCtl, true /*fStrict*/))
907 {
908 Log(("%s: Ctl#%d: ignored command %#04x, controller state %d\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl), s->uATARegCommand, pCtl->uAsyncIOState));
909 LogRel(("PIIX3 IDE: guest issued command %#04x while controller busy\n", s->uATARegCommand));
910 return;
911 }
912
913 Req.ReqType = ATA_AIO_NEW;
914 if (fChainedTransfer)
915 Req.u.t.iIf = pCtl->iAIOIf;
916 else
917 Req.u.t.iIf = pCtl->iSelectedIf;
918 Req.u.t.cbTotalTransfer = cbTotalTransfer;
919 Req.u.t.uTxDir = uTxDir;
920 Req.u.t.iBeginTransfer = iBeginTransfer;
921 Req.u.t.iSourceSink = iSourceSink;
922 ataSetStatusValue(s, ATA_STAT_BUSY);
923 pCtl->fChainedTransfer = fChainedTransfer;
924
925 /*
926 * Kick the worker thread into action.
927 */
928 Log2(("%s: Ctl#%d: message to async I/O thread, new request\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
929 ataHCAsyncIOPutRequest(pCtl, &Req);
930}
931
932
933/**
934 * Send an abort command request to the async I/O thread.
935 *
936 * @param s Pointer to the ATA device state data.
937 * @param fResetDrive Whether to reset the drive or just abort a command.
938 */
939static void ataR3AbortCurrentCommand(ATADevState *s, bool fResetDrive)
940{
941 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
942 ATARequest Req;
943
944 Assert(PDMCritSectIsOwner(&pCtl->lock));
945
946 /* Do not issue new requests while the RESET line is asserted. */
947 if (pCtl->fReset)
948 {
949 Log2(("%s: Ctl#%d: suppressed aborting command as RESET is active\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
950 return;
951 }
952
953 Req.ReqType = ATA_AIO_ABORT;
954 Req.u.a.iIf = pCtl->iSelectedIf;
955 Req.u.a.fResetDrive = fResetDrive;
956 ataSetStatus(s, ATA_STAT_BUSY);
957 Log2(("%s: Ctl#%d: message to async I/O thread, abort command on LUN#%d\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl), s->iLUN));
958 ataHCAsyncIOPutRequest(pCtl, &Req);
959}
960# endif /* IN_RING3 */
961
962static void ataHCSetIRQ(ATADevState *s)
963{
964 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
965 PPDMDEVINS pDevIns = ATADEVSTATE_2_DEVINS(s);
966
967 if (!(s->uATARegDevCtl & ATA_DEVCTL_DISABLE_IRQ))
968 {
969 Log2(("%s: LUN#%d asserting IRQ\n", __FUNCTION__, s->iLUN));
970 /* The BMDMA unit unconditionally sets BM_STATUS_INT if the interrupt
971 * line is asserted. It monitors the line for a rising edge. */
972 if (!s->fIrqPending)
973 pCtl->BmDma.u8Status |= BM_STATUS_INT;
974 /* Only actually set the IRQ line if updating the currently selected drive. */
975 if (s == &pCtl->aIfs[pCtl->iSelectedIf])
976 {
977 /** @todo experiment with adaptive IRQ delivery: for reads it is
978 * better to wait for IRQ delivery, as it reduces latency. */
979 if (pCtl->irq == 16)
980 PDMDevHlpPCISetIrq(pDevIns, 0, 1);
981 else
982 PDMDevHlpISASetIrq(pDevIns, pCtl->irq, 1);
983 }
984 }
985 s->fIrqPending = true;
986}
987
988#endif /* IN_RING0 || IN_RING3 */
989
990static void ataUnsetIRQ(ATADevState *s)
991{
992 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
993 PPDMDEVINS pDevIns = ATADEVSTATE_2_DEVINS(s);
994
995 if (!(s->uATARegDevCtl & ATA_DEVCTL_DISABLE_IRQ))
996 {
997 Log2(("%s: LUN#%d deasserting IRQ\n", __FUNCTION__, s->iLUN));
998 /* Only actually unset the IRQ line if updating the currently selected drive. */
999 if (s == &pCtl->aIfs[pCtl->iSelectedIf])
1000 {
1001 if (pCtl->irq == 16)
1002 PDMDevHlpPCISetIrq(pDevIns, 0, 0);
1003 else
1004 PDMDevHlpISASetIrq(pDevIns, pCtl->irq, 0);
1005 }
1006 }
1007 s->fIrqPending = false;
1008}
1009
1010#if defined(IN_RING0) || defined(IN_RING3)
1011
1012static void ataHCPIOTransferStart(ATADevState *s, uint32_t start, uint32_t size)
1013{
1014 Log2(("%s: LUN#%d start %d size %d\n", __FUNCTION__, s->iLUN, start, size));
1015 s->iIOBufferPIODataStart = start;
1016 s->iIOBufferPIODataEnd = start + size;
1017 ataSetStatus(s, ATA_STAT_DRQ | ATA_STAT_SEEK);
1018 ataUnsetStatus(s, ATA_STAT_BUSY);
1019}
1020
1021
1022static void ataHCPIOTransferStop(ATADevState *s)
1023{
1024 Log2(("%s: LUN#%d\n", __FUNCTION__, s->iLUN));
1025 if (s->fATAPITransfer)
1026 {
1027 s->uATARegNSector = (s->uATARegNSector & ~7) | ATAPI_INT_REASON_IO | ATAPI_INT_REASON_CD;
1028 Log2(("%s: interrupt reason %#04x\n", __FUNCTION__, s->uATARegNSector));
1029 ataHCSetIRQ(s);
1030 s->fATAPITransfer = false;
1031 }
1032 s->cbTotalTransfer = 0;
1033 s->cbElementaryTransfer = 0;
1034 s->iIOBufferPIODataStart = 0;
1035 s->iIOBufferPIODataEnd = 0;
1036 s->iBeginTransfer = ATAFN_BT_NULL;
1037 s->iSourceSink = ATAFN_SS_NULL;
1038}
1039
1040
1041static void ataHCPIOTransferLimitATAPI(ATADevState *s)
1042{
1043 uint32_t cbLimit, cbTransfer;
1044
1045 cbLimit = s->cbPIOTransferLimit;
1046 /* Use maximum transfer size if the guest requested 0. Avoids a hang. */
1047 if (cbLimit == 0)
1048 cbLimit = 0xfffe;
1049 Log2(("%s: byte count limit=%d\n", __FUNCTION__, cbLimit));
1050 if (cbLimit == 0xffff)
1051 cbLimit--;
1052 cbTransfer = RT_MIN(s->cbTotalTransfer, s->iIOBufferEnd - s->iIOBufferCur);
1053 if (cbTransfer > cbLimit)
1054 {
1055 /* Byte count limit for clipping must be even in this case */
1056 if (cbLimit & 1)
1057 cbLimit--;
1058 cbTransfer = cbLimit;
1059 }
1060 s->uATARegLCyl = cbTransfer;
1061 s->uATARegHCyl = cbTransfer >> 8;
1062 s->cbElementaryTransfer = cbTransfer;
1063}
1064
1065# ifdef IN_RING3
1066
1067static uint32_t ataR3GetNSectors(ATADevState *s)
1068{
1069 /* 0 means either 256 (LBA28) or 65536 (LBA48) sectors. */
1070 if (s->fLBA48)
1071 {
1072 if (!s->uATARegNSector && !s->uATARegNSectorHOB)
1073 return 65536;
1074 else
1075 return s->uATARegNSectorHOB << 8 | s->uATARegNSector;
1076 }
1077 else
1078 {
1079 if (!s->uATARegNSector)
1080 return 256;
1081 else
1082 return s->uATARegNSector;
1083 }
1084}
1085
1086
1087static void ataR3PadString(uint8_t *pbDst, const char *pbSrc, uint32_t cbSize)
1088{
1089 for (uint32_t i = 0; i < cbSize; i++)
1090 {
1091 if (*pbSrc)
1092 pbDst[i ^ 1] = *pbSrc++;
1093 else
1094 pbDst[i ^ 1] = ' ';
1095 }
1096}
1097
1098
1099static void ataR3SCSIPadStr(uint8_t *pbDst, const char *pbSrc, uint32_t cbSize)
1100{
1101 for (uint32_t i = 0; i < cbSize; i++)
1102 {
1103 if (*pbSrc)
1104 pbDst[i] = *pbSrc++;
1105 else
1106 pbDst[i] = ' ';
1107 }
1108}
1109
1110
1111DECLINLINE(void) ataH2BE_U16(uint8_t *pbBuf, uint16_t val)
1112{
1113 pbBuf[0] = val >> 8;
1114 pbBuf[1] = val;
1115}
1116
1117
1118DECLINLINE(void) ataH2BE_U24(uint8_t *pbBuf, uint32_t val)
1119{
1120 pbBuf[0] = val >> 16;
1121 pbBuf[1] = val >> 8;
1122 pbBuf[2] = val;
1123}
1124
1125
1126DECLINLINE(void) ataH2BE_U32(uint8_t *pbBuf, uint32_t val)
1127{
1128 pbBuf[0] = val >> 24;
1129 pbBuf[1] = val >> 16;
1130 pbBuf[2] = val >> 8;
1131 pbBuf[3] = val;
1132}
1133
1134
1135DECLINLINE(uint16_t) ataBE2H_U16(const uint8_t *pbBuf)
1136{
1137 return (pbBuf[0] << 8) | pbBuf[1];
1138}
1139
1140
1141DECLINLINE(uint32_t) ataBE2H_U24(const uint8_t *pbBuf)
1142{
1143 return (pbBuf[0] << 16) | (pbBuf[1] << 8) | pbBuf[2];
1144}
1145
1146
1147DECLINLINE(uint32_t) ataBE2H_U32(const uint8_t *pbBuf)
1148{
1149 return (pbBuf[0] << 24) | (pbBuf[1] << 16) | (pbBuf[2] << 8) | pbBuf[3];
1150}
1151
1152
1153DECLINLINE(void) ataLBA2MSF(uint8_t *pbBuf, uint32_t iATAPILBA)
1154{
1155 iATAPILBA += 150;
1156 pbBuf[0] = (iATAPILBA / 75) / 60;
1157 pbBuf[1] = (iATAPILBA / 75) % 60;
1158 pbBuf[2] = iATAPILBA % 75;
1159}
1160
1161
1162DECLINLINE(uint32_t) ataMSF2LBA(const uint8_t *pbBuf)
1163{
1164 return (pbBuf[0] * 60 + pbBuf[1]) * 75 + pbBuf[2];
1165}
1166
1167/**
1168 * Compares two MSF values.
1169 *
1170 * @returns 1 if the first value is greater than the second value.
1171 * 0 if both are equal
1172 * -1 if the first value is smaller than the second value.
1173 */
1174DECLINLINE(int) atapiCmpMSF(const uint8_t *pbMSF1, const uint8_t *pbMSF2)
1175{
1176 int iRes = 0;
1177
1178 for (unsigned i = 0; i < 3; i++)
1179 {
1180 if (pbMSF1[i] < pbMSF2[i])
1181 {
1182 iRes = -1;
1183 break;
1184 }
1185 else if (pbMSF1[i] > pbMSF2[i])
1186 {
1187 iRes = 1;
1188 break;
1189 }
1190 }
1191
1192 return iRes;
1193}
1194
1195static void ataR3CmdOK(ATADevState *s, uint8_t status)
1196{
1197 s->uATARegError = 0; /* Not needed by ATA spec, but cannot hurt. */
1198 ataSetStatusValue(s, ATA_STAT_READY | status);
1199}
1200
1201
1202static void ataR3CmdError(ATADevState *s, uint8_t uErrorCode)
1203{
1204 Log(("%s: code=%#x\n", __FUNCTION__, uErrorCode));
1205 Assert(uErrorCode);
1206 s->uATARegError = uErrorCode;
1207 ataSetStatusValue(s, ATA_STAT_READY | ATA_STAT_ERR);
1208 s->cbTotalTransfer = 0;
1209 s->cbElementaryTransfer = 0;
1210 s->iIOBufferCur = 0;
1211 s->iIOBufferEnd = 0;
1212 s->uTxDir = PDMBLOCKTXDIR_NONE;
1213 s->iBeginTransfer = ATAFN_BT_NULL;
1214 s->iSourceSink = ATAFN_SS_NULL;
1215}
1216
1217static uint32_t ataR3Checksum(void* ptr, size_t count)
1218{
1219 uint8_t u8Sum = 0xa5, *p = (uint8_t*)ptr;
1220 size_t i;
1221
1222 for (i = 0; i < count; i++)
1223 {
1224 u8Sum += *p++;
1225 }
1226
1227 return (uint8_t)-(int32_t)u8Sum;
1228}
1229
1230static bool ataR3IdentifySS(ATADevState *s)
1231{
1232 uint16_t *p;
1233
1234 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
1235 Assert(s->cbElementaryTransfer == 512);
1236
1237 p = (uint16_t *)s->CTX_SUFF(pbIOBuffer);
1238 memset(p, 0, 512);
1239 p[0] = RT_H2LE_U16(0x0040);
1240 p[1] = RT_H2LE_U16(RT_MIN(s->PCHSGeometry.cCylinders, 16383));
1241 p[3] = RT_H2LE_U16(s->PCHSGeometry.cHeads);
1242 /* Block size; obsolete, but required for the BIOS. */
1243 p[5] = RT_H2LE_U16(s->cbSector);
1244 p[6] = RT_H2LE_U16(s->PCHSGeometry.cSectors);
1245 ataR3PadString((uint8_t *)(p + 10), s->szSerialNumber, ATA_SERIAL_NUMBER_LENGTH); /* serial number */
1246 p[20] = RT_H2LE_U16(3); /* XXX: retired, cache type */
1247 p[21] = RT_H2LE_U16(512); /* XXX: retired, cache size in sectors */
1248 p[22] = RT_H2LE_U16(0); /* ECC bytes per sector */
1249 ataR3PadString((uint8_t *)(p + 23), s->szFirmwareRevision, ATA_FIRMWARE_REVISION_LENGTH); /* firmware version */
1250 ataR3PadString((uint8_t *)(p + 27), s->szModelNumber, ATA_MODEL_NUMBER_LENGTH); /* model */
1251# if ATA_MAX_MULT_SECTORS > 1
1252 p[47] = RT_H2LE_U16(0x8000 | ATA_MAX_MULT_SECTORS);
1253# endif
1254 p[48] = RT_H2LE_U16(1); /* dword I/O, used by the BIOS */
1255 p[49] = RT_H2LE_U16(1 << 11 | 1 << 9 | 1 << 8); /* DMA and LBA supported */
1256 p[50] = RT_H2LE_U16(1 << 14); /* No drive specific standby timer minimum */
1257 p[51] = RT_H2LE_U16(240); /* PIO transfer cycle */
1258 p[52] = RT_H2LE_U16(240); /* DMA transfer cycle */
1259 p[53] = RT_H2LE_U16(1 | 1 << 1 | 1 << 2); /* words 54-58,64-70,88 valid */
1260 p[54] = RT_H2LE_U16(RT_MIN(s->PCHSGeometry.cCylinders, 16383));
1261 p[55] = RT_H2LE_U16(s->PCHSGeometry.cHeads);
1262 p[56] = RT_H2LE_U16(s->PCHSGeometry.cSectors);
1263 p[57] = RT_H2LE_U16( RT_MIN(s->PCHSGeometry.cCylinders, 16383)
1264 * s->PCHSGeometry.cHeads
1265 * s->PCHSGeometry.cSectors);
1266 p[58] = RT_H2LE_U16( RT_MIN(s->PCHSGeometry.cCylinders, 16383)
1267 * s->PCHSGeometry.cHeads
1268 * s->PCHSGeometry.cSectors >> 16);
1269 if (s->cMultSectors)
1270 p[59] = RT_H2LE_U16(0x100 | s->cMultSectors);
1271 if (s->cTotalSectors <= (1 << 28) - 1)
1272 {
1273 p[60] = RT_H2LE_U16(s->cTotalSectors);
1274 p[61] = RT_H2LE_U16(s->cTotalSectors >> 16);
1275 }
1276 else
1277 {
1278 /* Report maximum number of sectors possible with LBA28 */
1279 p[60] = RT_H2LE_U16(((1 << 28) - 1) & 0xffff);
1280 p[61] = RT_H2LE_U16(((1 << 28) - 1) >> 16);
1281 }
1282 p[63] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_MDMA, ATA_MDMA_MODE_MAX, s->uATATransferMode)); /* MDMA modes supported / mode enabled */
1283 p[64] = RT_H2LE_U16(ATA_PIO_MODE_MAX > 2 ? (1 << (ATA_PIO_MODE_MAX - 2)) - 1 : 0); /* PIO modes beyond PIO2 supported */
1284 p[65] = RT_H2LE_U16(120); /* minimum DMA multiword tx cycle time */
1285 p[66] = RT_H2LE_U16(120); /* recommended DMA multiword tx cycle time */
1286 p[67] = RT_H2LE_U16(120); /* minimum PIO cycle time without flow control */
1287 p[68] = RT_H2LE_U16(120); /* minimum PIO cycle time with IORDY flow control */
1288 if ( s->pDrvBlock->pfnDiscard
1289 || s->cbSector != 512
1290 || s->fNonRotational)
1291 {
1292 p[80] = RT_H2LE_U16(0x1f0); /* support everything up to ATA/ATAPI-8 ACS */
1293 p[81] = RT_H2LE_U16(0x28); /* conforms to ATA/ATAPI-8 ACS */
1294 }
1295 else
1296 {
1297 p[80] = RT_H2LE_U16(0x7e); /* support everything up to ATA/ATAPI-6 */
1298 p[81] = RT_H2LE_U16(0x22); /* conforms to ATA/ATAPI-6 */
1299 }
1300 p[82] = RT_H2LE_U16(1 << 3 | 1 << 5 | 1 << 6); /* supports power management, write cache and look-ahead */
1301 if (s->cTotalSectors <= (1 << 28) - 1)
1302 p[83] = RT_H2LE_U16(1 << 14 | 1 << 12); /* supports FLUSH CACHE */
1303 else
1304 p[83] = RT_H2LE_U16(1 << 14 | 1 << 10 | 1 << 12 | 1 << 13); /* supports LBA48, FLUSH CACHE and FLUSH CACHE EXT */
1305 p[84] = RT_H2LE_U16(1 << 14);
1306 p[85] = RT_H2LE_U16(1 << 3 | 1 << 5 | 1 << 6); /* enabled power management, write cache and look-ahead */
1307 if (s->cTotalSectors <= (1 << 28) - 1)
1308 p[86] = RT_H2LE_U16(1 << 12); /* enabled FLUSH CACHE */
1309 else
1310 p[86] = RT_H2LE_U16(1 << 10 | 1 << 12 | 1 << 13); /* enabled LBA48, FLUSH CACHE and FLUSH CACHE EXT */
1311 p[87] = RT_H2LE_U16(1 << 14);
1312 p[88] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_UDMA, ATA_UDMA_MODE_MAX, s->uATATransferMode)); /* UDMA modes supported / mode enabled */
1313 p[93] = RT_H2LE_U16((1 | 1 << 1) << ((s->iLUN & 1) == 0 ? 0 : 8) | 1 << 13 | 1 << 14);
1314 if (s->cTotalSectors > (1 << 28) - 1)
1315 {
1316 p[100] = RT_H2LE_U16(s->cTotalSectors);
1317 p[101] = RT_H2LE_U16(s->cTotalSectors >> 16);
1318 p[102] = RT_H2LE_U16(s->cTotalSectors >> 32);
1319 p[103] = RT_H2LE_U16(s->cTotalSectors >> 48);
1320 }
1321
1322 if (s->cbSector != 512)
1323 {
1324 uint32_t cSectorSizeInWords = s->cbSector / sizeof(uint16_t);
1325 /* Enable reporting of logical sector size. */
1326 p[106] |= RT_H2LE_U16(RT_BIT(12) | RT_BIT(14));
1327 p[117] = RT_H2LE_U16(cSectorSizeInWords);
1328 p[118] = RT_H2LE_U16(cSectorSizeInWords >> 16);
1329 }
1330
1331 if (s->pDrvBlock->pfnDiscard) /** @todo: Set bit 14 in word 69 too? (Deterministic read after TRIM). */
1332 p[169] = RT_H2LE_U16(1); /* DATA SET MANAGEMENT command supported. */
1333 if (s->fNonRotational)
1334 p[217] = RT_H2LE_U16(1); /* Non-rotational medium */
1335 uint32_t uCsum = ataR3Checksum(p, 510);
1336 p[255] = RT_H2LE_U16(0xa5 | (uCsum << 8)); /* Integrity word */
1337 s->iSourceSink = ATAFN_SS_NULL;
1338 ataR3CmdOK(s, ATA_STAT_SEEK);
1339 return false;
1340}
1341
1342
1343static bool ataR3FlushSS(ATADevState *s)
1344{
1345 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
1346 int rc;
1347
1348 Assert(s->uTxDir == PDMBLOCKTXDIR_NONE);
1349 Assert(!s->cbElementaryTransfer);
1350
1351 PDMCritSectLeave(&pCtl->lock);
1352
1353 STAM_PROFILE_START(&s->StatFlushes, f);
1354 rc = s->pDrvBlock->pfnFlush(s->pDrvBlock);
1355 AssertRC(rc);
1356 STAM_PROFILE_STOP(&s->StatFlushes, f);
1357
1358 STAM_PROFILE_START(&pCtl->StatLockWait, a);
1359 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
1360 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
1361 ataR3CmdOK(s, 0);
1362 return false;
1363}
1364
1365static bool atapiR3IdentifySS(ATADevState *s)
1366{
1367 uint16_t *p;
1368
1369 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
1370 Assert(s->cbElementaryTransfer == 512);
1371
1372 p = (uint16_t *)s->CTX_SUFF(pbIOBuffer);
1373 memset(p, 0, 512);
1374 /* Removable CDROM, 3ms response, 12 byte packets */
1375 p[0] = RT_H2LE_U16(2 << 14 | 5 << 8 | 1 << 7 | 0 << 5 | 0 << 0);
1376 ataR3PadString((uint8_t *)(p + 10), s->szSerialNumber, ATA_SERIAL_NUMBER_LENGTH); /* serial number */
1377 p[20] = RT_H2LE_U16(3); /* XXX: retired, cache type */
1378 p[21] = RT_H2LE_U16(512); /* XXX: retired, cache size in sectors */
1379 ataR3PadString((uint8_t *)(p + 23), s->szFirmwareRevision, ATA_FIRMWARE_REVISION_LENGTH); /* firmware version */
1380 ataR3PadString((uint8_t *)(p + 27), s->szModelNumber, ATA_MODEL_NUMBER_LENGTH); /* model */
1381 p[49] = RT_H2LE_U16(1 << 11 | 1 << 9 | 1 << 8); /* DMA and LBA supported */
1382 p[50] = RT_H2LE_U16(1 << 14); /* No drive specific standby timer minimum */
1383 p[51] = RT_H2LE_U16(240); /* PIO transfer cycle */
1384 p[52] = RT_H2LE_U16(240); /* DMA transfer cycle */
1385 p[53] = RT_H2LE_U16(1 << 1 | 1 << 2); /* words 64-70,88 are valid */
1386 p[63] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_MDMA, ATA_MDMA_MODE_MAX, s->uATATransferMode)); /* MDMA modes supported / mode enabled */
1387 p[64] = RT_H2LE_U16(ATA_PIO_MODE_MAX > 2 ? (1 << (ATA_PIO_MODE_MAX - 2)) - 1 : 0); /* PIO modes beyond PIO2 supported */
1388 p[65] = RT_H2LE_U16(120); /* minimum DMA multiword tx cycle time */
1389 p[66] = RT_H2LE_U16(120); /* recommended DMA multiword tx cycle time */
1390 p[67] = RT_H2LE_U16(120); /* minimum PIO cycle time without flow control */
1391 p[68] = RT_H2LE_U16(120); /* minimum PIO cycle time with IORDY flow control */
1392 p[73] = RT_H2LE_U16(0x003e); /* ATAPI CDROM major */
1393 p[74] = RT_H2LE_U16(9); /* ATAPI CDROM minor */
1394 p[75] = RT_H2LE_U16(1); /* queue depth 1 */
1395 p[80] = RT_H2LE_U16(0x7e); /* support everything up to ATA/ATAPI-6 */
1396 p[81] = RT_H2LE_U16(0x22); /* conforms to ATA/ATAPI-6 */
1397 p[82] = RT_H2LE_U16(1 << 4 | 1 << 9); /* supports packet command set and DEVICE RESET */
1398 p[83] = RT_H2LE_U16(1 << 14);
1399 p[84] = RT_H2LE_U16(1 << 14);
1400 p[85] = RT_H2LE_U16(1 << 4 | 1 << 9); /* enabled packet command set and DEVICE RESET */
1401 p[86] = RT_H2LE_U16(0);
1402 p[87] = RT_H2LE_U16(1 << 14);
1403 p[88] = RT_H2LE_U16(ATA_TRANSFER_ID(ATA_MODE_UDMA, ATA_UDMA_MODE_MAX, s->uATATransferMode)); /* UDMA modes supported / mode enabled */
1404 p[93] = RT_H2LE_U16((1 | 1 << 1) << ((s->iLUN & 1) == 0 ? 0 : 8) | 1 << 13 | 1 << 14);
1405 /* According to ATAPI-5 spec:
1406 *
1407 * The use of this word is optional.
1408 * If bits 7:0 of this word contain the signature A5h, bits 15:8
1409 * contain the data
1410 * structure checksum.
1411 * The data structure checksum is the twos complement of the sum of
1412 * all bytes in words 0 through 254 and the byte consisting of
1413 * bits 7:0 in word 255.
1414 * Each byte shall be added with unsigned arithmetic,
1415 * and overflow shall be ignored.
1416 * The sum of all 512 bytes is zero when the checksum is correct.
1417 */
1418 uint32_t uCsum = ataR3Checksum(p, 510);
1419 p[255] = RT_H2LE_U16(0xa5 | (uCsum << 8)); /* Integrity word */
1420
1421 s->iSourceSink = ATAFN_SS_NULL;
1422 ataR3CmdOK(s, ATA_STAT_SEEK);
1423 return false;
1424}
1425
1426
1427static void ataR3SetSignature(ATADevState *s)
1428{
1429 s->uATARegSelect &= 0xf0; /* clear head */
1430 /* put signature */
1431 s->uATARegNSector = 1;
1432 s->uATARegSector = 1;
1433 if (s->fATAPI)
1434 {
1435 s->uATARegLCyl = 0x14;
1436 s->uATARegHCyl = 0xeb;
1437 }
1438 else if (s->pDrvBlock)
1439 {
1440 s->uATARegLCyl = 0;
1441 s->uATARegHCyl = 0;
1442 }
1443 else
1444 {
1445 s->uATARegLCyl = 0xff;
1446 s->uATARegHCyl = 0xff;
1447 }
1448}
1449
1450
1451static uint64_t ataR3GetSector(ATADevState *s)
1452{
1453 uint64_t iLBA;
1454 if (s->uATARegSelect & 0x40)
1455 {
1456 /* any LBA variant */
1457 if (s->fLBA48)
1458 {
1459 /* LBA48 */
1460 iLBA = ((uint64_t)s->uATARegHCylHOB << 40) |
1461 ((uint64_t)s->uATARegLCylHOB << 32) |
1462 ((uint64_t)s->uATARegSectorHOB << 24) |
1463 ((uint64_t)s->uATARegHCyl << 16) |
1464 ((uint64_t)s->uATARegLCyl << 8) |
1465 s->uATARegSector;
1466 }
1467 else
1468 {
1469 /* LBA */
1470 iLBA = ((s->uATARegSelect & 0x0f) << 24) | (s->uATARegHCyl << 16) |
1471 (s->uATARegLCyl << 8) | s->uATARegSector;
1472 }
1473 }
1474 else
1475 {
1476 /* CHS */
1477 iLBA = ((s->uATARegHCyl << 8) | s->uATARegLCyl) * s->PCHSGeometry.cHeads * s->PCHSGeometry.cSectors +
1478 (s->uATARegSelect & 0x0f) * s->PCHSGeometry.cSectors +
1479 (s->uATARegSector - 1);
1480 }
1481 return iLBA;
1482}
1483
1484static void ataR3SetSector(ATADevState *s, uint64_t iLBA)
1485{
1486 uint32_t cyl, r;
1487 if (s->uATARegSelect & 0x40)
1488 {
1489 /* any LBA variant */
1490 if (s->fLBA48)
1491 {
1492 /* LBA48 */
1493 s->uATARegHCylHOB = iLBA >> 40;
1494 s->uATARegLCylHOB = iLBA >> 32;
1495 s->uATARegSectorHOB = iLBA >> 24;
1496 s->uATARegHCyl = iLBA >> 16;
1497 s->uATARegLCyl = iLBA >> 8;
1498 s->uATARegSector = iLBA;
1499 }
1500 else
1501 {
1502 /* LBA */
1503 s->uATARegSelect = (s->uATARegSelect & 0xf0) | (iLBA >> 24);
1504 s->uATARegHCyl = (iLBA >> 16);
1505 s->uATARegLCyl = (iLBA >> 8);
1506 s->uATARegSector = (iLBA);
1507 }
1508 }
1509 else
1510 {
1511 /* CHS */
1512 cyl = iLBA / (s->PCHSGeometry.cHeads * s->PCHSGeometry.cSectors);
1513 r = iLBA % (s->PCHSGeometry.cHeads * s->PCHSGeometry.cSectors);
1514 s->uATARegHCyl = cyl >> 8;
1515 s->uATARegLCyl = cyl;
1516 s->uATARegSelect = (s->uATARegSelect & 0xf0) | ((r / s->PCHSGeometry.cSectors) & 0x0f);
1517 s->uATARegSector = (r % s->PCHSGeometry.cSectors) + 1;
1518 }
1519}
1520
1521
1522static void ataR3WarningDiskFull(PPDMDEVINS pDevIns)
1523{
1524 int rc;
1525 LogRel(("PIIX3 ATA: Host disk full\n"));
1526 rc = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevATA_DISKFULL",
1527 N_("Host system reported disk full. VM execution is suspended. You can resume after freeing some space"));
1528 AssertRC(rc);
1529}
1530
1531static void ataR3WarningFileTooBig(PPDMDEVINS pDevIns)
1532{
1533 int rc;
1534 LogRel(("PIIX3 ATA: File too big\n"));
1535 rc = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevATA_FILETOOBIG",
1536 N_("Host system reported that the file size limit of the host file system has been exceeded. VM execution is suspended. You need to move your virtual hard disk to a filesystem which allows bigger files"));
1537 AssertRC(rc);
1538}
1539
1540static void ataR3WarningISCSI(PPDMDEVINS pDevIns)
1541{
1542 int rc;
1543 LogRel(("PIIX3 ATA: iSCSI target unavailable\n"));
1544 rc = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevATA_ISCSIDOWN",
1545 N_("The iSCSI target has stopped responding. VM execution is suspended. You can resume when it is available again"));
1546 AssertRC(rc);
1547}
1548
1549static bool ataR3IsRedoSetWarning(ATADevState *s, int rc)
1550{
1551 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
1552 Assert(!PDMCritSectIsOwner(&pCtl->lock));
1553 if (rc == VERR_DISK_FULL)
1554 {
1555 pCtl->fRedoIdle = true;
1556 ataR3WarningDiskFull(ATADEVSTATE_2_DEVINS(s));
1557 return true;
1558 }
1559 if (rc == VERR_FILE_TOO_BIG)
1560 {
1561 pCtl->fRedoIdle = true;
1562 ataR3WarningFileTooBig(ATADEVSTATE_2_DEVINS(s));
1563 return true;
1564 }
1565 if (rc == VERR_BROKEN_PIPE || rc == VERR_NET_CONNECTION_REFUSED)
1566 {
1567 pCtl->fRedoIdle = true;
1568 /* iSCSI connection abort (first error) or failure to reestablish
1569 * connection (second error). Pause VM. On resume we'll retry. */
1570 ataR3WarningISCSI(ATADEVSTATE_2_DEVINS(s));
1571 return true;
1572 }
1573 if (rc == VERR_VD_DEK_MISSING)
1574 {
1575 /* Error message already set. */
1576 pCtl->fRedoIdle = true;
1577 return true;
1578 }
1579
1580 return false;
1581}
1582
1583
1584static int ataR3ReadSectors(ATADevState *s, uint64_t u64Sector, void *pvBuf,
1585 uint32_t cSectors, bool *pfRedo)
1586{
1587 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
1588 int rc;
1589
1590 PDMCritSectLeave(&pCtl->lock);
1591
1592 STAM_PROFILE_ADV_START(&s->StatReads, r);
1593 s->Led.Asserted.s.fReading = s->Led.Actual.s.fReading = 1;
1594 rc = s->pDrvBlock->pfnRead(s->pDrvBlock, u64Sector * s->cbSector, pvBuf, cSectors * s->cbSector);
1595 s->Led.Actual.s.fReading = 0;
1596 STAM_PROFILE_ADV_STOP(&s->StatReads, r);
1597 Log4(("ataR3ReadSectors: rc=%Rrc cSectors=%#x u64Sector=%llu\n%.*Rhxd\n",
1598 rc, cSectors, u64Sector, cSectors * s->cbSector, pvBuf));
1599
1600 STAM_REL_COUNTER_ADD(&s->StatBytesRead, cSectors * s->cbSector);
1601
1602 if (RT_SUCCESS(rc))
1603 *pfRedo = false;
1604 else
1605 *pfRedo = ataR3IsRedoSetWarning(s, rc);
1606
1607 STAM_PROFILE_START(&pCtl->StatLockWait, a);
1608 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
1609 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
1610 return rc;
1611}
1612
1613
1614static int ataR3WriteSectors(ATADevState *s, uint64_t u64Sector,
1615 const void *pvBuf, uint32_t cSectors, bool *pfRedo)
1616{
1617 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
1618 int rc;
1619
1620 PDMCritSectLeave(&pCtl->lock);
1621
1622 STAM_PROFILE_ADV_START(&s->StatWrites, w);
1623 s->Led.Asserted.s.fWriting = s->Led.Actual.s.fWriting = 1;
1624# ifdef VBOX_INSTRUMENT_DMA_WRITES
1625 if (s->fDMA)
1626 STAM_PROFILE_ADV_START(&s->StatInstrVDWrites, vw);
1627# endif
1628 rc = s->pDrvBlock->pfnWrite(s->pDrvBlock, u64Sector * s->cbSector, pvBuf, cSectors * s->cbSector);
1629# ifdef VBOX_INSTRUMENT_DMA_WRITES
1630 if (s->fDMA)
1631 STAM_PROFILE_ADV_STOP(&s->StatInstrVDWrites, vw);
1632# endif
1633 s->Led.Actual.s.fWriting = 0;
1634 STAM_PROFILE_ADV_STOP(&s->StatWrites, w);
1635 Log4(("ataR3WriteSectors: rc=%Rrc cSectors=%#x u64Sector=%llu\n%.*Rhxd\n",
1636 rc, cSectors, u64Sector, cSectors * s->cbSector, pvBuf));
1637
1638 STAM_REL_COUNTER_ADD(&s->StatBytesWritten, cSectors * s->cbSector);
1639
1640 if (RT_SUCCESS(rc))
1641 *pfRedo = false;
1642 else
1643 *pfRedo = ataR3IsRedoSetWarning(s, rc);
1644
1645 STAM_PROFILE_START(&pCtl->StatLockWait, a);
1646 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
1647 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
1648 return rc;
1649}
1650
1651
1652static void ataR3ReadWriteSectorsBT(ATADevState *s)
1653{
1654 uint32_t cSectors;
1655
1656 cSectors = s->cbTotalTransfer / s->cbSector;
1657 if (cSectors > s->cSectorsPerIRQ)
1658 s->cbElementaryTransfer = s->cSectorsPerIRQ * s->cbSector;
1659 else
1660 s->cbElementaryTransfer = cSectors * s->cbSector;
1661 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE)
1662 ataR3CmdOK(s, 0);
1663}
1664
1665
1666static bool ataR3ReadSectorsSS(ATADevState *s)
1667{
1668 int rc;
1669 uint32_t cSectors;
1670 uint64_t iLBA;
1671 bool fRedo;
1672
1673 cSectors = s->cbElementaryTransfer / s->cbSector;
1674 Assert(cSectors);
1675 iLBA = ataR3GetSector(s);
1676 Log(("%s: %d sectors at LBA %d\n", __FUNCTION__, cSectors, iLBA));
1677 rc = ataR3ReadSectors(s, iLBA, s->CTX_SUFF(pbIOBuffer), cSectors, &fRedo);
1678 if (RT_SUCCESS(rc))
1679 {
1680 ataR3SetSector(s, iLBA + cSectors);
1681 if (s->cbElementaryTransfer == s->cbTotalTransfer)
1682 s->iSourceSink = ATAFN_SS_NULL;
1683 ataR3CmdOK(s, ATA_STAT_SEEK);
1684 }
1685 else
1686 {
1687 if (fRedo)
1688 return fRedo;
1689 if (s->cErrors++ < MAX_LOG_REL_ERRORS)
1690 LogRel(("PIIX3 ATA: LUN#%d: disk read error (rc=%Rrc iSector=%#RX64 cSectors=%#RX32)\n",
1691 s->iLUN, rc, iLBA, cSectors));
1692
1693 /*
1694 * Check if we got interrupted. We don't need to set status variables
1695 * because the request was aborted.
1696 */
1697 if (rc != VERR_INTERRUPTED)
1698 ataR3CmdError(s, ID_ERR);
1699 }
1700 return false;
1701}
1702
1703
1704static bool ataR3WriteSectorsSS(ATADevState *s)
1705{
1706 int rc;
1707 uint32_t cSectors;
1708 uint64_t iLBA;
1709 bool fRedo;
1710
1711 cSectors = s->cbElementaryTransfer / s->cbSector;
1712 Assert(cSectors);
1713 iLBA = ataR3GetSector(s);
1714 Log(("%s: %d sectors at LBA %d\n", __FUNCTION__, cSectors, iLBA));
1715 rc = ataR3WriteSectors(s, iLBA, s->CTX_SUFF(pbIOBuffer), cSectors, &fRedo);
1716 if (RT_SUCCESS(rc))
1717 {
1718 ataR3SetSector(s, iLBA + cSectors);
1719 if (!s->cbTotalTransfer)
1720 s->iSourceSink = ATAFN_SS_NULL;
1721 ataR3CmdOK(s, ATA_STAT_SEEK);
1722 }
1723 else
1724 {
1725 if (fRedo)
1726 return fRedo;
1727 if (s->cErrors++ < MAX_LOG_REL_ERRORS)
1728 LogRel(("PIIX3 ATA: LUN#%d: disk write error (rc=%Rrc iSector=%#RX64 cSectors=%#RX32)\n",
1729 s->iLUN, rc, iLBA, cSectors));
1730
1731 /*
1732 * Check if we got interrupted. We don't need to set status variables
1733 * because the request was aborted.
1734 */
1735 if (rc != VERR_INTERRUPTED)
1736 ataR3CmdError(s, ID_ERR);
1737 }
1738 return false;
1739}
1740
1741
1742static void atapiR3CmdOK(ATADevState *s)
1743{
1744 s->uATARegError = 0;
1745 ataSetStatusValue(s, ATA_STAT_READY);
1746 s->uATARegNSector = (s->uATARegNSector & ~7)
1747 | ((s->uTxDir != PDMBLOCKTXDIR_TO_DEVICE) ? ATAPI_INT_REASON_IO : 0)
1748 | (!s->cbTotalTransfer ? ATAPI_INT_REASON_CD : 0);
1749 Log2(("%s: interrupt reason %#04x\n", __FUNCTION__, s->uATARegNSector));
1750
1751 memset(s->abATAPISense, '\0', sizeof(s->abATAPISense));
1752 s->abATAPISense[0] = 0x70 | (1 << 7);
1753 s->abATAPISense[7] = 10;
1754}
1755
1756
1757static void atapiR3CmdError(ATADevState *s, const uint8_t *pabATAPISense, size_t cbATAPISense)
1758{
1759 Log(("%s: sense=%#x (%s) asc=%#x ascq=%#x (%s)\n", __FUNCTION__, pabATAPISense[2] & 0x0f, SCSISenseText(pabATAPISense[2] & 0x0f),
1760 pabATAPISense[12], pabATAPISense[13], SCSISenseExtText(pabATAPISense[12], pabATAPISense[13])));
1761 s->uATARegError = pabATAPISense[2] << 4;
1762 ataSetStatusValue(s, ATA_STAT_READY | ATA_STAT_ERR);
1763 s->uATARegNSector = (s->uATARegNSector & ~7) | ATAPI_INT_REASON_IO | ATAPI_INT_REASON_CD;
1764 Log2(("%s: interrupt reason %#04x\n", __FUNCTION__, s->uATARegNSector));
1765 memset(s->abATAPISense, '\0', sizeof(s->abATAPISense));
1766 memcpy(s->abATAPISense, pabATAPISense, RT_MIN(cbATAPISense, sizeof(s->abATAPISense)));
1767 s->cbTotalTransfer = 0;
1768 s->cbElementaryTransfer = 0;
1769 s->cbAtapiPassthroughTransfer = 0;
1770 s->iIOBufferCur = 0;
1771 s->iIOBufferEnd = 0;
1772 s->uTxDir = PDMBLOCKTXDIR_NONE;
1773 s->iBeginTransfer = ATAFN_BT_NULL;
1774 s->iSourceSink = ATAFN_SS_NULL;
1775}
1776
1777
1778/** @todo deprecated function - doesn't provide enough info. Replace by direct
1779 * calls to atapiR3CmdError() with full data. */
1780static void atapiR3CmdErrorSimple(ATADevState *s, uint8_t uATAPISenseKey, uint8_t uATAPIASC)
1781{
1782 uint8_t abATAPISense[ATAPI_SENSE_SIZE];
1783 memset(abATAPISense, '\0', sizeof(abATAPISense));
1784 abATAPISense[0] = 0x70 | (1 << 7);
1785 abATAPISense[2] = uATAPISenseKey & 0x0f;
1786 abATAPISense[7] = 10;
1787 abATAPISense[12] = uATAPIASC;
1788 atapiR3CmdError(s, abATAPISense, sizeof(abATAPISense));
1789}
1790
1791
1792static void atapiR3CmdBT(ATADevState *s)
1793{
1794 s->fATAPITransfer = true;
1795 s->cbElementaryTransfer = s->cbTotalTransfer;
1796 s->cbAtapiPassthroughTransfer = s->cbTotalTransfer;
1797 s->cbPIOTransferLimit = s->uATARegLCyl | (s->uATARegHCyl << 8);
1798 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE)
1799 atapiR3CmdOK(s);
1800}
1801
1802
1803static void atapiR3PassthroughCmdBT(ATADevState *s)
1804{
1805 /** @todo implement an algorithm for correctly determining the read and
1806 * write sector size without sending additional commands to the drive.
1807 * This should be doable by saving processing the configuration requests
1808 * and replies. */
1809# if 0
1810 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE)
1811 {
1812 uint8_t cmd = s->aATAPICmd[0];
1813 if (cmd == SCSI_WRITE_10 || cmd == SCSI_WRITE_12 || cmd == SCSI_WRITE_AND_VERIFY_10)
1814 {
1815 uint8_t aModeSenseCmd[10];
1816 uint8_t aModeSenseResult[16];
1817 uint8_t uDummySense;
1818 uint32_t cbTransfer;
1819 int rc;
1820
1821 cbTransfer = sizeof(aModeSenseResult);
1822 aModeSenseCmd[0] = SCSI_MODE_SENSE_10;
1823 aModeSenseCmd[1] = 0x08; /* disable block descriptor = 1 */
1824 aModeSenseCmd[2] = (SCSI_PAGECONTROL_CURRENT << 6) | SCSI_MODEPAGE_WRITE_PARAMETER;
1825 aModeSenseCmd[3] = 0; /* subpage code */
1826 aModeSenseCmd[4] = 0; /* reserved */
1827 aModeSenseCmd[5] = 0; /* reserved */
1828 aModeSenseCmd[6] = 0; /* reserved */
1829 aModeSenseCmd[7] = cbTransfer >> 8;
1830 aModeSenseCmd[8] = cbTransfer & 0xff;
1831 aModeSenseCmd[9] = 0; /* control */
1832 rc = s->pDrvBlock->pfnSendCmd(s->pDrvBlock, aModeSenseCmd, PDMBLOCKTXDIR_FROM_DEVICE, aModeSenseResult, &cbTransfer, &uDummySense, 500);
1833 if (RT_FAILURE(rc))
1834 {
1835 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_NONE);
1836 return;
1837 }
1838 /* Select sector size based on the current data block type. */
1839 switch (aModeSenseResult[12] & 0x0f)
1840 {
1841 case 0:
1842 s->cbATAPISector = 2352;
1843 break;
1844 case 1:
1845 s->cbATAPISector = 2368;
1846 break;
1847 case 2:
1848 case 3:
1849 s->cbATAPISector = 2448;
1850 break;
1851 case 8:
1852 case 10:
1853 s->cbATAPISector = 2048;
1854 break;
1855 case 9:
1856 s->cbATAPISector = 2336;
1857 break;
1858 case 11:
1859 s->cbATAPISector = 2056;
1860 break;
1861 case 12:
1862 s->cbATAPISector = 2324;
1863 break;
1864 case 13:
1865 s->cbATAPISector = 2332;
1866 break;
1867 default:
1868 s->cbATAPISector = 0;
1869 }
1870 Log2(("%s: sector size %d\n", __FUNCTION__, s->cbATAPISector));
1871 s->cbTotalTransfer *= s->cbATAPISector;
1872 if (s->cbTotalTransfer == 0)
1873 s->uTxDir = PDMBLOCKTXDIR_NONE;
1874 }
1875 }
1876# endif
1877 atapiR3CmdBT(s);
1878}
1879
1880static bool atapiR3ReadSS(ATADevState *s)
1881{
1882 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
1883 int rc = VINF_SUCCESS;
1884 uint32_t cbTransfer, cSectors;
1885
1886 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
1887 cbTransfer = RT_MIN(s->cbTotalTransfer, s->cbIOBuffer);
1888 cSectors = cbTransfer / s->cbATAPISector;
1889 Assert(cSectors * s->cbATAPISector <= cbTransfer);
1890 Log(("%s: %d sectors at LBA %d\n", __FUNCTION__, cSectors, s->iATAPILBA));
1891
1892 PDMCritSectLeave(&pCtl->lock);
1893
1894 STAM_PROFILE_ADV_START(&s->StatReads, r);
1895 s->Led.Asserted.s.fReading = s->Led.Actual.s.fReading = 1;
1896 switch (s->cbATAPISector)
1897 {
1898 case 2048:
1899 rc = s->pDrvBlock->pfnRead(s->pDrvBlock, (uint64_t)s->iATAPILBA * s->cbATAPISector, s->CTX_SUFF(pbIOBuffer), s->cbATAPISector * cSectors);
1900 break;
1901 case 2352:
1902 {
1903 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
1904
1905 for (uint32_t i = s->iATAPILBA; i < s->iATAPILBA + cSectors; i++)
1906 {
1907 /* Sync bytes, see 4.2.3.8 CD Main Channel Block Formats */
1908 *pbBuf++ = 0x00;
1909 memset(pbBuf, 0xff, 10);
1910 pbBuf += 10;
1911 *pbBuf++ = 0x00;
1912 /* MSF */
1913 ataLBA2MSF(pbBuf, i);
1914 pbBuf += 3;
1915 *pbBuf++ = 0x01; /* mode 1 data */
1916 /* data */
1917 rc = s->pDrvBlock->pfnRead(s->pDrvBlock, (uint64_t)i * 2048, pbBuf, 2048);
1918 if (RT_FAILURE(rc))
1919 break;
1920 pbBuf += 2048;
1921 /**
1922 * @todo: maybe compute ECC and parity, layout is:
1923 * 2072 4 EDC
1924 * 2076 172 P parity symbols
1925 * 2248 104 Q parity symbols
1926 */
1927 memset(pbBuf, 0, 280);
1928 pbBuf += 280;
1929 }
1930 break;
1931 }
1932 default:
1933 break;
1934 }
1935 s->Led.Actual.s.fReading = 0;
1936 STAM_PROFILE_ADV_STOP(&s->StatReads, r);
1937
1938 STAM_PROFILE_START(&pCtl->StatLockWait, a);
1939 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
1940 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
1941
1942 if (RT_SUCCESS(rc))
1943 {
1944 STAM_REL_COUNTER_ADD(&s->StatBytesRead, s->cbATAPISector * cSectors);
1945
1946 /* The initial buffer end value has been set up based on the total
1947 * transfer size. But the I/O buffer size limits what can actually be
1948 * done in one transfer, so set the actual value of the buffer end. */
1949 s->cbElementaryTransfer = cbTransfer;
1950 if (cbTransfer >= s->cbTotalTransfer)
1951 s->iSourceSink = ATAFN_SS_NULL;
1952 atapiR3CmdOK(s);
1953 s->iATAPILBA += cSectors;
1954 }
1955 else
1956 {
1957 if (s->cErrors++ < MAX_LOG_REL_ERRORS)
1958 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM read error, %d sectors at LBA %d\n", s->iLUN, cSectors, s->iATAPILBA));
1959
1960 /*
1961 * Check if we got interrupted. We don't need to set status variables
1962 * because the request was aborted.
1963 */
1964 if (rc != VERR_INTERRUPTED)
1965 atapiR3CmdErrorSimple(s, SCSI_SENSE_MEDIUM_ERROR, SCSI_ASC_READ_ERROR);
1966 }
1967 return false;
1968}
1969
1970/**
1971 * Sets the given media track type.
1972 */
1973static uint32_t ataR3MediumTypeSet(ATADevState *s, uint32_t MediaTrackType)
1974{
1975 return ASMAtomicXchgU32(&s->MediaTrackType, MediaTrackType);
1976}
1977
1978static bool atapiR3PassthroughSS(ATADevState *s)
1979{
1980 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
1981 int rc = VINF_SUCCESS;
1982 uint8_t abATAPISense[ATAPI_SENSE_SIZE];
1983 uint32_t cbTransfer;
1984 PSTAMPROFILEADV pProf = NULL;
1985
1986 cbTransfer = RT_MIN(s->cbAtapiPassthroughTransfer, s->cbIOBuffer);
1987
1988 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE)
1989 Log3(("ATAPI PT data write (%d): %.*Rhxs\n", cbTransfer, cbTransfer, s->CTX_SUFF(pbIOBuffer)));
1990
1991 /* Simple heuristics: if there is at least one sector of data
1992 * to transfer, it's worth updating the LEDs. */
1993 if (cbTransfer >= 2048)
1994 {
1995 if (s->uTxDir != PDMBLOCKTXDIR_TO_DEVICE)
1996 {
1997 s->Led.Asserted.s.fReading = s->Led.Actual.s.fReading = 1;
1998 pProf = &s->StatReads;
1999 }
2000 else
2001 {
2002 s->Led.Asserted.s.fWriting = s->Led.Actual.s.fWriting = 1;
2003 pProf = &s->StatWrites;
2004 }
2005 }
2006
2007 PDMCritSectLeave(&pCtl->lock);
2008
2009# if defined(LOG_ENABLED)
2010 char szBuf[1024];
2011
2012 memset(szBuf, 0, sizeof(szBuf));
2013
2014 switch (s->aATAPICmd[0])
2015 {
2016 case SCSI_MODE_SELECT_10:
2017 {
2018 size_t cbBlkDescLength = ataBE2H_U16(&s->CTX_SUFF(pbIOBuffer)[6]);
2019
2020 SCSILogModePage(szBuf, sizeof(szBuf) - 1,
2021 s->CTX_SUFF(pbIOBuffer) + 8 + cbBlkDescLength,
2022 cbTransfer - 8 - cbBlkDescLength);
2023 break;
2024 }
2025 case SCSI_SEND_CUE_SHEET:
2026 {
2027 SCSILogCueSheet(szBuf, sizeof(szBuf) - 1,
2028 s->CTX_SUFF(pbIOBuffer), cbTransfer);
2029 break;
2030 }
2031 default:
2032 break;
2033 }
2034
2035 Log2(("%s\n", szBuf));
2036# endif
2037
2038 if (pProf) { STAM_PROFILE_ADV_START(pProf, b); }
2039 if ( cbTransfer > SCSI_MAX_BUFFER_SIZE
2040 || s->cbElementaryTransfer > s->cbIOBuffer)
2041 {
2042 /* Linux accepts commands with up to 100KB of data, but expects
2043 * us to handle commands with up to 128KB of data. The usual
2044 * imbalance of powers. */
2045 uint8_t aATAPICmd[ATAPI_PACKET_SIZE];
2046 uint32_t iATAPILBA, cSectors, cReqSectors, cbCurrTX;
2047 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2048 uint32_t cSectorsMax; /**< Maximum amount of sectors to read without exceeding the I/O buffer. */
2049
2050 Assert(s->cbATAPISector);
2051 cSectorsMax = cbTransfer / s->cbATAPISector;
2052 Assert(cSectorsMax * s->cbATAPISector <= s->cbIOBuffer);
2053
2054 switch (s->aATAPICmd[0])
2055 {
2056 case SCSI_READ_10:
2057 case SCSI_WRITE_10:
2058 case SCSI_WRITE_AND_VERIFY_10:
2059 iATAPILBA = ataBE2H_U32(s->aATAPICmd + 2);
2060 cSectors = ataBE2H_U16(s->aATAPICmd + 7);
2061 break;
2062 case SCSI_READ_12:
2063 case SCSI_WRITE_12:
2064 iATAPILBA = ataBE2H_U32(s->aATAPICmd + 2);
2065 cSectors = ataBE2H_U32(s->aATAPICmd + 6);
2066 break;
2067 case SCSI_READ_CD:
2068 iATAPILBA = ataBE2H_U32(s->aATAPICmd + 2);
2069 cSectors = ataBE2H_U24(s->aATAPICmd + 6);
2070 break;
2071 case SCSI_READ_CD_MSF:
2072 iATAPILBA = ataMSF2LBA(s->aATAPICmd + 3);
2073 cSectors = ataMSF2LBA(s->aATAPICmd + 6) - iATAPILBA;
2074 break;
2075 default:
2076 AssertMsgFailed(("Don't know how to split command %#04x\n", s->aATAPICmd[0]));
2077 if (s->cErrors++ < MAX_LOG_REL_ERRORS)
2078 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM passthrough split error\n", s->iLUN));
2079 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
2080 {
2081 STAM_PROFILE_START(&pCtl->StatLockWait, a);
2082 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
2083 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
2084 }
2085 return false;
2086 }
2087 cSectorsMax = RT_MIN(cSectorsMax, cSectors);
2088 memcpy(aATAPICmd, s->aATAPICmd, ATAPI_PACKET_SIZE);
2089 cReqSectors = 0;
2090 for (uint32_t i = cSectorsMax; i > 0; i -= cReqSectors)
2091 {
2092 if (i * s->cbATAPISector > SCSI_MAX_BUFFER_SIZE)
2093 cReqSectors = SCSI_MAX_BUFFER_SIZE / s->cbATAPISector;
2094 else
2095 cReqSectors = i;
2096 cbCurrTX = s->cbATAPISector * cReqSectors;
2097 switch (s->aATAPICmd[0])
2098 {
2099 case SCSI_READ_10:
2100 case SCSI_WRITE_10:
2101 case SCSI_WRITE_AND_VERIFY_10:
2102 ataH2BE_U32(aATAPICmd + 2, iATAPILBA);
2103 ataH2BE_U16(aATAPICmd + 7, cReqSectors);
2104 break;
2105 case SCSI_READ_12:
2106 case SCSI_WRITE_12:
2107 ataH2BE_U32(aATAPICmd + 2, iATAPILBA);
2108 ataH2BE_U32(aATAPICmd + 6, cReqSectors);
2109 break;
2110 case SCSI_READ_CD:
2111 ataH2BE_U32(aATAPICmd + 2, iATAPILBA);
2112 ataH2BE_U24(aATAPICmd + 6, cReqSectors);
2113 break;
2114 case SCSI_READ_CD_MSF:
2115 ataLBA2MSF(aATAPICmd + 3, iATAPILBA);
2116 ataLBA2MSF(aATAPICmd + 6, iATAPILBA + cReqSectors);
2117 break;
2118 }
2119 rc = s->pDrvBlock->pfnSendCmd(s->pDrvBlock, aATAPICmd, (PDMBLOCKTXDIR)s->uTxDir, pbBuf, &cbCurrTX, abATAPISense, sizeof(abATAPISense), 30000 /**< @todo timeout */);
2120 if (rc != VINF_SUCCESS)
2121 break;
2122 iATAPILBA += cReqSectors;
2123 pbBuf += s->cbATAPISector * cReqSectors;
2124 }
2125
2126 if (RT_SUCCESS(rc))
2127 {
2128 /* Adjust ATAPI command for the next call. */
2129 switch (s->aATAPICmd[0])
2130 {
2131 case SCSI_READ_10:
2132 case SCSI_WRITE_10:
2133 case SCSI_WRITE_AND_VERIFY_10:
2134 ataH2BE_U32(s->aATAPICmd + 2, iATAPILBA);
2135 ataH2BE_U16(s->aATAPICmd + 7, cSectors - cSectorsMax);
2136 break;
2137 case SCSI_READ_12:
2138 case SCSI_WRITE_12:
2139 ataH2BE_U32(s->aATAPICmd + 2, iATAPILBA);
2140 ataH2BE_U32(s->aATAPICmd + 6, cSectors - cSectorsMax);
2141 break;
2142 case SCSI_READ_CD:
2143 ataH2BE_U32(s->aATAPICmd + 2, iATAPILBA);
2144 ataH2BE_U24(s->aATAPICmd + 6, cSectors - cSectorsMax);
2145 break;
2146 case SCSI_READ_CD_MSF:
2147 ataLBA2MSF(s->aATAPICmd + 3, iATAPILBA);
2148 ataLBA2MSF(s->aATAPICmd + 6, iATAPILBA + cSectors - cSectorsMax);
2149 break;
2150 default:
2151 AssertMsgFailed(("Don't know how to split command %#04x\n", s->aATAPICmd[0]));
2152 if (s->cErrors++ < MAX_LOG_REL_ERRORS)
2153 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM passthrough split error\n", s->iLUN));
2154 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
2155 return false;
2156 }
2157 }
2158 }
2159 else
2160 rc = s->pDrvBlock->pfnSendCmd(s->pDrvBlock, s->aATAPICmd, (PDMBLOCKTXDIR)s->uTxDir, s->CTX_SUFF(pbIOBuffer), &cbTransfer, abATAPISense, sizeof(abATAPISense), 30000 /**< @todo timeout */);
2161 if (pProf) { STAM_PROFILE_ADV_STOP(pProf, b); }
2162
2163 STAM_PROFILE_START(&pCtl->StatLockWait, a);
2164 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
2165 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
2166
2167 /* Update the LEDs and the read/write statistics. */
2168 if (cbTransfer >= 2048)
2169 {
2170 if (s->uTxDir != PDMBLOCKTXDIR_TO_DEVICE)
2171 {
2172 s->Led.Actual.s.fReading = 0;
2173 STAM_REL_COUNTER_ADD(&s->StatBytesRead, cbTransfer);
2174 }
2175 else
2176 {
2177 s->Led.Actual.s.fWriting = 0;
2178 STAM_REL_COUNTER_ADD(&s->StatBytesWritten, cbTransfer);
2179 }
2180 }
2181
2182 if (RT_SUCCESS(rc))
2183 {
2184 /* Do post processing for certain commands. */
2185 switch (s->aATAPICmd[0])
2186 {
2187 case SCSI_SEND_CUE_SHEET:
2188 case SCSI_READ_TOC_PMA_ATIP:
2189 {
2190 if (!s->pTrackList)
2191 rc = ATAPIPassthroughTrackListCreateEmpty(&s->pTrackList);
2192
2193 if (RT_SUCCESS(rc))
2194 rc = ATAPIPassthroughTrackListUpdate(s->pTrackList, s->aATAPICmd, s->CTX_SUFF(pbIOBuffer));
2195
2196 if ( RT_FAILURE(rc)
2197 && s->cErrors++ < MAX_LOG_REL_ERRORS)
2198 LogRel(("ATA: Error (%Rrc) while updating the tracklist during %s, burning the disc might fail\n",
2199 rc, s->aATAPICmd[0] == SCSI_SEND_CUE_SHEET ? "SEND CUE SHEET" : "READ TOC/PMA/ATIP"));
2200 break;
2201 }
2202 case SCSI_SYNCHRONIZE_CACHE:
2203 {
2204 if (s->pTrackList)
2205 ATAPIPassthroughTrackListClear(s->pTrackList);
2206 break;
2207 }
2208 }
2209
2210 if (s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE)
2211 {
2212 Assert(cbTransfer <= s->cbAtapiPassthroughTransfer);
2213 /*
2214 * Reply with the same amount of data as the real drive
2215 * but only if the command wasn't split.
2216 */
2217# if 0 /// @todo This destroys commands where cbTotalTransfer > cbIOBuffer
2218 if (s->cbElementaryTransfer < s->cbIOBuffer)
2219 s->cbTotalTransfer = cbTransfer;
2220# endif
2221
2222 if ( s->aATAPICmd[0] == SCSI_INQUIRY
2223 && s->fOverwriteInquiry)
2224 {
2225 /* Make sure that the real drive cannot be identified.
2226 * Motivation: changing the VM configuration should be as
2227 * invisible as possible to the guest. */
2228 Log3(("ATAPI PT inquiry data before (%d): %.*Rhxs\n", cbTransfer, cbTransfer, s->CTX_SUFF(pbIOBuffer)));
2229 ataR3SCSIPadStr(s->CTX_SUFF(pbIOBuffer) + 8, "VBOX", 8);
2230 ataR3SCSIPadStr(s->CTX_SUFF(pbIOBuffer) + 16, "CD-ROM", 16);
2231 ataR3SCSIPadStr(s->CTX_SUFF(pbIOBuffer) + 32, "1.0", 4);
2232 }
2233
2234 if (cbTransfer)
2235 Log3(("ATAPI PT data read (%d): %.*Rhxs\n", cbTransfer, cbTransfer, s->CTX_SUFF(pbIOBuffer)));
2236 }
2237
2238 /* The initial buffer end value has been set up based on the total
2239 * transfer size. But the I/O buffer size limits what can actually be
2240 * done in one transfer, so set the actual value of the buffer end. */
2241 s->cbElementaryTransfer = cbTransfer;
2242 if (cbTransfer >= s->cbAtapiPassthroughTransfer)
2243 {
2244 s->iSourceSink = ATAFN_SS_NULL;
2245 atapiR3CmdOK(s);
2246 }
2247 }
2248 else
2249 {
2250 if (s->cErrors < MAX_LOG_REL_ERRORS)
2251 {
2252 uint8_t u8Cmd = s->aATAPICmd[0];
2253 do
2254 {
2255 /* don't log superfluous errors */
2256 if ( rc == VERR_DEV_IO_ERROR
2257 && ( u8Cmd == SCSI_TEST_UNIT_READY
2258 || u8Cmd == SCSI_READ_CAPACITY
2259 || u8Cmd == SCSI_READ_DVD_STRUCTURE
2260 || u8Cmd == SCSI_READ_TOC_PMA_ATIP))
2261 break;
2262 s->cErrors++;
2263 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM passthrough cmd=%#04x sense=%d ASC=%#02x ASCQ=%#02x %Rrc\n",
2264 s->iLUN, u8Cmd, abATAPISense[2] & 0x0f, abATAPISense[12], abATAPISense[13], rc));
2265 } while (0);
2266 }
2267 atapiR3CmdError(s, abATAPISense, sizeof(abATAPISense));
2268 }
2269 return false;
2270}
2271
2272/** @todo Revise ASAP. */
2273static bool atapiR3ReadDVDStructureSS(ATADevState *s)
2274{
2275 uint8_t *buf = s->CTX_SUFF(pbIOBuffer);
2276 int media = s->aATAPICmd[1];
2277 int format = s->aATAPICmd[7];
2278
2279 uint16_t max_len = ataBE2H_U16(&s->aATAPICmd[8]);
2280
2281 memset(buf, 0, max_len);
2282
2283 switch (format) {
2284 case 0x00:
2285 case 0x01:
2286 case 0x02:
2287 case 0x03:
2288 case 0x04:
2289 case 0x05:
2290 case 0x06:
2291 case 0x07:
2292 case 0x08:
2293 case 0x09:
2294 case 0x0a:
2295 case 0x0b:
2296 case 0x0c:
2297 case 0x0d:
2298 case 0x0e:
2299 case 0x0f:
2300 case 0x10:
2301 case 0x11:
2302 case 0x30:
2303 case 0x31:
2304 case 0xff:
2305 if (media == 0)
2306 {
2307 int uASC = SCSI_ASC_NONE;
2308
2309 switch (format)
2310 {
2311 case 0x0: /* Physical format information */
2312 {
2313 int layer = s->aATAPICmd[6];
2314 uint64_t total_sectors;
2315
2316 if (layer != 0)
2317 {
2318 uASC = -SCSI_ASC_INV_FIELD_IN_CMD_PACKET;
2319 break;
2320 }
2321
2322 total_sectors = s->cTotalSectors;
2323 total_sectors >>= 2;
2324 if (total_sectors == 0)
2325 {
2326 uASC = -SCSI_ASC_MEDIUM_NOT_PRESENT;
2327 break;
2328 }
2329
2330 buf[4] = 1; /* DVD-ROM, part version 1 */
2331 buf[5] = 0xf; /* 120mm disc, minimum rate unspecified */
2332 buf[6] = 1; /* one layer, read-only (per MMC-2 spec) */
2333 buf[7] = 0; /* default densities */
2334
2335 /* FIXME: 0x30000 per spec? */
2336 ataH2BE_U32(buf + 8, 0); /* start sector */
2337 ataH2BE_U32(buf + 12, total_sectors - 1); /* end sector */
2338 ataH2BE_U32(buf + 16, total_sectors - 1); /* l0 end sector */
2339
2340 /* Size of buffer, not including 2 byte size field */
2341 ataH2BE_U32(&buf[0], 2048 + 2);
2342
2343 /* 2k data + 4 byte header */
2344 uASC = (2048 + 4);
2345 break;
2346 }
2347 case 0x01: /* DVD copyright information */
2348 buf[4] = 0; /* no copyright data */
2349 buf[5] = 0; /* no region restrictions */
2350
2351 /* Size of buffer, not including 2 byte size field */
2352 ataH2BE_U16(buf, 4 + 2);
2353
2354 /* 4 byte header + 4 byte data */
2355 uASC = (4 + 4);
2356 break;
2357
2358 case 0x03: /* BCA information - invalid field for no BCA info */
2359 uASC = -SCSI_ASC_INV_FIELD_IN_CMD_PACKET;
2360 break;
2361
2362 case 0x04: /* DVD disc manufacturing information */
2363 /* Size of buffer, not including 2 byte size field */
2364 ataH2BE_U16(buf, 2048 + 2);
2365
2366 /* 2k data + 4 byte header */
2367 uASC = (2048 + 4);
2368 break;
2369 case 0xff:
2370 /*
2371 * This lists all the command capabilities above. Add new ones
2372 * in order and update the length and buffer return values.
2373 */
2374
2375 buf[4] = 0x00; /* Physical format */
2376 buf[5] = 0x40; /* Not writable, is readable */
2377 ataH2BE_U16((buf + 6), 2048 + 4);
2378
2379 buf[8] = 0x01; /* Copyright info */
2380 buf[9] = 0x40; /* Not writable, is readable */
2381 ataH2BE_U16((buf + 10), 4 + 4);
2382
2383 buf[12] = 0x03; /* BCA info */
2384 buf[13] = 0x40; /* Not writable, is readable */
2385 ataH2BE_U16((buf + 14), 188 + 4);
2386
2387 buf[16] = 0x04; /* Manufacturing info */
2388 buf[17] = 0x40; /* Not writable, is readable */
2389 ataH2BE_U16((buf + 18), 2048 + 4);
2390
2391 /* Size of buffer, not including 2 byte size field */
2392 ataH2BE_U16(buf, 16 + 2);
2393
2394 /* data written + 4 byte header */
2395 uASC = (16 + 4);
2396 break;
2397 default: /* TODO: formats beyond DVD-ROM requires */
2398 uASC = -SCSI_ASC_INV_FIELD_IN_CMD_PACKET;
2399 }
2400
2401 if (uASC < 0)
2402 {
2403 s->iSourceSink = ATAFN_SS_NULL;
2404 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, -uASC);
2405 return false;
2406 }
2407 break;
2408 }
2409 /* TODO: BD support, fall through for now */
2410
2411 /* Generic disk structures */
2412 case 0x80: /* TODO: AACS volume identifier */
2413 case 0x81: /* TODO: AACS media serial number */
2414 case 0x82: /* TODO: AACS media identifier */
2415 case 0x83: /* TODO: AACS media key block */
2416 case 0x90: /* TODO: List of recognized format layers */
2417 case 0xc0: /* TODO: Write protection status */
2418 default:
2419 s->iSourceSink = ATAFN_SS_NULL;
2420 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST,
2421 SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
2422 return false;
2423 }
2424
2425 s->iSourceSink = ATAFN_SS_NULL;
2426 atapiR3CmdOK(s);
2427 return false;
2428}
2429
2430static bool atapiR3ReadSectors(ATADevState *s, uint32_t iATAPILBA, uint32_t cSectors, uint32_t cbSector)
2431{
2432 Assert(cSectors > 0);
2433 s->iATAPILBA = iATAPILBA;
2434 s->cbATAPISector = cbSector;
2435 ataR3StartTransfer(s, cSectors * cbSector, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ, true);
2436 return false;
2437}
2438
2439
2440static bool atapiR3ReadCapacitySS(ATADevState *s)
2441{
2442 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2443
2444 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2445 Assert(s->cbElementaryTransfer <= 8);
2446 ataH2BE_U32(pbBuf, s->cTotalSectors - 1);
2447 ataH2BE_U32(pbBuf + 4, 2048);
2448 s->iSourceSink = ATAFN_SS_NULL;
2449 atapiR3CmdOK(s);
2450 return false;
2451}
2452
2453
2454static bool atapiR3ReadDiscInformationSS(ATADevState *s)
2455{
2456 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2457
2458 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2459 Assert(s->cbElementaryTransfer <= 34);
2460 memset(pbBuf, '\0', 34);
2461 ataH2BE_U16(pbBuf, 32);
2462 pbBuf[2] = (0 << 4) | (3 << 2) | (2 << 0); /* not erasable, complete session, complete disc */
2463 pbBuf[3] = 1; /* number of first track */
2464 pbBuf[4] = 1; /* number of sessions (LSB) */
2465 pbBuf[5] = 1; /* first track number in last session (LSB) */
2466 pbBuf[6] = 1; /* last track number in last session (LSB) */
2467 pbBuf[7] = (0 << 7) | (0 << 6) | (1 << 5) | (0 << 2) | (0 << 0); /* disc id not valid, disc bar code not valid, unrestricted use, not dirty, not RW medium */
2468 pbBuf[8] = 0; /* disc type = CD-ROM */
2469 pbBuf[9] = 0; /* number of sessions (MSB) */
2470 pbBuf[10] = 0; /* number of sessions (MSB) */
2471 pbBuf[11] = 0; /* number of sessions (MSB) */
2472 ataH2BE_U32(pbBuf + 16, 0x00ffffff); /* last session lead-in start time is not available */
2473 ataH2BE_U32(pbBuf + 20, 0x00ffffff); /* last possible start time for lead-out is not available */
2474 s->iSourceSink = ATAFN_SS_NULL;
2475 atapiR3CmdOK(s);
2476 return false;
2477}
2478
2479
2480static bool atapiR3ReadTrackInformationSS(ATADevState *s)
2481{
2482 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2483
2484 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2485 Assert(s->cbElementaryTransfer <= 36);
2486 /* Accept address/number type of 1 only, and only track 1 exists. */
2487 if ((s->aATAPICmd[1] & 0x03) != 1 || ataBE2H_U32(&s->aATAPICmd[2]) != 1)
2488 {
2489 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
2490 return false;
2491 }
2492 memset(pbBuf, '\0', 36);
2493 ataH2BE_U16(pbBuf, 34);
2494 pbBuf[2] = 1; /* track number (LSB) */
2495 pbBuf[3] = 1; /* session number (LSB) */
2496 pbBuf[5] = (0 << 5) | (0 << 4) | (4 << 0); /* not damaged, primary copy, data track */
2497 pbBuf[6] = (0 << 7) | (0 << 6) | (0 << 5) | (0 << 6) | (1 << 0); /* not reserved track, not blank, not packet writing, not fixed packet, data mode 1 */
2498 pbBuf[7] = (0 << 1) | (0 << 0); /* last recorded address not valid, next recordable address not valid */
2499 ataH2BE_U32(pbBuf + 8, 0); /* track start address is 0 */
2500 ataH2BE_U32(pbBuf + 24, s->cTotalSectors); /* track size */
2501 pbBuf[32] = 0; /* track number (MSB) */
2502 pbBuf[33] = 0; /* session number (MSB) */
2503 s->iSourceSink = ATAFN_SS_NULL;
2504 atapiR3CmdOK(s);
2505 return false;
2506}
2507
2508static uint32_t atapiR3GetConfigurationFillFeatureListProfiles(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2509{
2510 if (cbBuf < 3*4)
2511 return 0;
2512
2513 ataH2BE_U16(pbBuf, 0x0); /* feature 0: list of profiles supported */
2514 pbBuf[2] = (0 << 2) | (1 << 1) | (1 << 0); /* version 0, persistent, current */
2515 pbBuf[3] = 8; /* additional bytes for profiles */
2516 /* The MMC-3 spec says that DVD-ROM read capability should be reported
2517 * before CD-ROM read capability. */
2518 ataH2BE_U16(pbBuf + 4, 0x10); /* profile: read-only DVD */
2519 pbBuf[6] = (0 << 0); /* NOT current profile */
2520 ataH2BE_U16(pbBuf + 8, 0x08); /* profile: read only CD */
2521 pbBuf[10] = (1 << 0); /* current profile */
2522
2523 return 3*4; /* Header + 2 profiles entries */
2524}
2525
2526static uint32_t atapiR3GetConfigurationFillFeatureCore(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2527{
2528 if (cbBuf < 12)
2529 return 0;
2530
2531 ataH2BE_U16(pbBuf, 0x1); /* feature 0001h: Core Feature */
2532 pbBuf[2] = (0x2 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2533 pbBuf[3] = 8; /* Additional length */
2534 ataH2BE_U16(pbBuf + 4, 0x00000002); /* Physical interface ATAPI. */
2535 pbBuf[8] = RT_BIT(0); /* DBE */
2536 /* Rest is reserved. */
2537
2538 return 12;
2539}
2540
2541static uint32_t atapiR3GetConfigurationFillFeatureMorphing(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2542{
2543 if (cbBuf < 8)
2544 return 0;
2545
2546 ataH2BE_U16(pbBuf, 0x2); /* feature 0002h: Morphing Feature */
2547 pbBuf[2] = (0x1 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2548 pbBuf[3] = 4; /* Additional length */
2549 pbBuf[4] = RT_BIT(1) | 0x0; /* OCEvent | !ASYNC */
2550 /* Rest is reserved. */
2551
2552 return 8;
2553}
2554
2555static uint32_t atapiR3GetConfigurationFillFeatureRemovableMedium(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2556{
2557 if (cbBuf < 8)
2558 return 0;
2559
2560 ataH2BE_U16(pbBuf, 0x3); /* feature 0003h: Removable Medium Feature */
2561 pbBuf[2] = (0x2 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2562 pbBuf[3] = 4; /* Additional length */
2563 /* Tray type loading | Load | Eject | !Pvnt Jmpr | !DBML | Lock */
2564 pbBuf[4] = (0x2 << 5) | RT_BIT(4) | RT_BIT(3) | (0x0 << 2) | (0x0 << 1) | RT_BIT(0);
2565 /* Rest is reserved. */
2566
2567 return 8;
2568}
2569
2570static uint32_t atapiR3GetConfigurationFillFeatureRandomReadable (ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2571{
2572 if (cbBuf < 12)
2573 return 0;
2574
2575 ataH2BE_U16(pbBuf, 0x10); /* feature 0010h: Random Readable Feature */
2576 pbBuf[2] = (0x0 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2577 pbBuf[3] = 8; /* Additional length */
2578 ataH2BE_U32(pbBuf + 4, 2048); /* Logical block size. */
2579 ataH2BE_U16(pbBuf + 8, 0x10); /* Blocking (0x10 for DVD, CD is not defined). */
2580 pbBuf[10] = 0; /* PP not present */
2581 /* Rest is reserved. */
2582
2583 return 12;
2584}
2585
2586static uint32_t atapiR3GetConfigurationFillFeatureCDRead(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2587{
2588 if (cbBuf < 8)
2589 return 0;
2590
2591 ataH2BE_U16(pbBuf, 0x1e); /* feature 001Eh: CD Read Feature */
2592 pbBuf[2] = (0x2 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2593 pbBuf[3] = 0; /* Additional length */
2594 pbBuf[4] = (0x0 << 7) | (0x0 << 1) | 0x0; /* !DAP | !C2-Flags | !CD-Text. */
2595 /* Rest is reserved. */
2596
2597 return 8;
2598}
2599
2600static uint32_t atapiR3GetConfigurationFillFeaturePowerManagement(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2601{
2602 if (cbBuf < 4)
2603 return 0;
2604
2605 ataH2BE_U16(pbBuf, 0x100); /* feature 0100h: Power Management Feature */
2606 pbBuf[2] = (0x0 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2607 pbBuf[3] = 0; /* Additional length */
2608
2609 return 4;
2610}
2611
2612static uint32_t atapiR3GetConfigurationFillFeatureTimeout(ATADevState *s, uint8_t *pbBuf, size_t cbBuf)
2613{
2614 if (cbBuf < 8)
2615 return 0;
2616
2617 ataH2BE_U16(pbBuf, 0x105); /* feature 0105h: Timeout Feature */
2618 pbBuf[2] = (0x0 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
2619 pbBuf[3] = 4; /* Additional length */
2620 pbBuf[4] = 0x0; /* !Group3 */
2621
2622 return 8;
2623}
2624
2625static bool atapiR3GetConfigurationSS(ATADevState *s)
2626{
2627 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2628 uint32_t cbBuf = s->cbIOBuffer;
2629 uint32_t cbCopied = 0;
2630 uint16_t u16Sfn = ataBE2H_U16(&s->aATAPICmd[2]);
2631
2632 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2633 Assert(s->cbElementaryTransfer <= 80);
2634 /* Accept valid request types only, and only starting feature 0. */
2635 if ((s->aATAPICmd[1] & 0x03) == 3 || u16Sfn != 0)
2636 {
2637 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
2638 return false;
2639 }
2640 memset(pbBuf, '\0', cbBuf);
2641 /** @todo implement switching between CD-ROM and DVD-ROM profile (the only
2642 * way to differentiate them right now is based on the image size). */
2643 if (s->cTotalSectors)
2644 ataH2BE_U16(pbBuf + 6, 0x08); /* current profile: read-only CD */
2645 else
2646 ataH2BE_U16(pbBuf + 6, 0x00); /* current profile: none -> no media */
2647 cbBuf -= 8;
2648 pbBuf += 8;
2649
2650 cbCopied = atapiR3GetConfigurationFillFeatureListProfiles(s, pbBuf, cbBuf);
2651 cbBuf -= cbCopied;
2652 pbBuf += cbCopied;
2653
2654 cbCopied = atapiR3GetConfigurationFillFeatureCore(s, pbBuf, cbBuf);
2655 cbBuf -= cbCopied;
2656 pbBuf += cbCopied;
2657
2658 cbCopied = atapiR3GetConfigurationFillFeatureMorphing(s, pbBuf, cbBuf);
2659 cbBuf -= cbCopied;
2660 pbBuf += cbCopied;
2661
2662 cbCopied = atapiR3GetConfigurationFillFeatureRemovableMedium(s, pbBuf, cbBuf);
2663 cbBuf -= cbCopied;
2664 pbBuf += cbCopied;
2665
2666 cbCopied = atapiR3GetConfigurationFillFeatureRandomReadable (s, pbBuf, cbBuf);
2667 cbBuf -= cbCopied;
2668 pbBuf += cbCopied;
2669
2670 cbCopied = atapiR3GetConfigurationFillFeatureCDRead(s, pbBuf, cbBuf);
2671 cbBuf -= cbCopied;
2672 pbBuf += cbCopied;
2673
2674 cbCopied = atapiR3GetConfigurationFillFeaturePowerManagement(s, pbBuf, cbBuf);
2675 cbBuf -= cbCopied;
2676 pbBuf += cbCopied;
2677
2678 cbCopied = atapiR3GetConfigurationFillFeatureTimeout(s, pbBuf, cbBuf);
2679 cbBuf -= cbCopied;
2680 pbBuf += cbCopied;
2681
2682 /* Set data length now - the field is not included in the final length. */
2683 ataH2BE_U32(s->CTX_SUFF(pbIOBuffer), s->cbIOBuffer - cbBuf - 4);
2684
2685 /* Other profiles we might want to add in the future: 0x40 (BD-ROM) and 0x50 (HDDVD-ROM) */
2686 s->iSourceSink = ATAFN_SS_NULL;
2687 atapiR3CmdOK(s);
2688 return false;
2689}
2690
2691
2692static bool atapiR3GetEventStatusNotificationSS(ATADevState *s)
2693{
2694 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2695
2696 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2697 Assert(s->cbElementaryTransfer <= 8);
2698
2699 if (!(s->aATAPICmd[1] & 1))
2700 {
2701 /* no asynchronous operation supported */
2702 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
2703 return false;
2704 }
2705
2706 uint32_t OldStatus, NewStatus;
2707 do
2708 {
2709 OldStatus = ASMAtomicReadU32(&s->MediaEventStatus);
2710 NewStatus = ATA_EVENT_STATUS_UNCHANGED;
2711 switch (OldStatus)
2712 {
2713 case ATA_EVENT_STATUS_MEDIA_NEW:
2714 /* mount */
2715 ataH2BE_U16(pbBuf + 0, 6);
2716 pbBuf[2] = 0x04; /* media */
2717 pbBuf[3] = 0x5e; /* supported = busy|media|external|power|operational */
2718 pbBuf[4] = 0x02; /* new medium */
2719 pbBuf[5] = 0x02; /* medium present / door closed */
2720 pbBuf[6] = 0x00;
2721 pbBuf[7] = 0x00;
2722 break;
2723
2724 case ATA_EVENT_STATUS_MEDIA_CHANGED:
2725 case ATA_EVENT_STATUS_MEDIA_REMOVED:
2726 /* umount */
2727 ataH2BE_U16(pbBuf + 0, 6);
2728 pbBuf[2] = 0x04; /* media */
2729 pbBuf[3] = 0x5e; /* supported = busy|media|external|power|operational */
2730 pbBuf[4] = 0x03; /* media removal */
2731 pbBuf[5] = 0x00; /* medium absent / door closed */
2732 pbBuf[6] = 0x00;
2733 pbBuf[7] = 0x00;
2734 if (OldStatus == ATA_EVENT_STATUS_MEDIA_CHANGED)
2735 NewStatus = ATA_EVENT_STATUS_MEDIA_NEW;
2736 break;
2737
2738 case ATA_EVENT_STATUS_MEDIA_EJECT_REQUESTED: /* currently unused */
2739 ataH2BE_U16(pbBuf + 0, 6);
2740 pbBuf[2] = 0x04; /* media */
2741 pbBuf[3] = 0x5e; /* supported = busy|media|external|power|operational */
2742 pbBuf[4] = 0x01; /* eject requested (eject button pressed) */
2743 pbBuf[5] = 0x02; /* medium present / door closed */
2744 pbBuf[6] = 0x00;
2745 pbBuf[7] = 0x00;
2746 break;
2747
2748 case ATA_EVENT_STATUS_UNCHANGED:
2749 default:
2750 ataH2BE_U16(pbBuf + 0, 6);
2751 pbBuf[2] = 0x01; /* operational change request / notification */
2752 pbBuf[3] = 0x5e; /* supported = busy|media|external|power|operational */
2753 pbBuf[4] = 0x00;
2754 pbBuf[5] = 0x00;
2755 pbBuf[6] = 0x00;
2756 pbBuf[7] = 0x00;
2757 break;
2758 }
2759 } while (!ASMAtomicCmpXchgU32(&s->MediaEventStatus, NewStatus, OldStatus));
2760
2761 s->iSourceSink = ATAFN_SS_NULL;
2762 atapiR3CmdOK(s);
2763 return false;
2764}
2765
2766
2767static bool atapiR3InquirySS(ATADevState *s)
2768{
2769 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2770
2771 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2772 Assert(s->cbElementaryTransfer <= 36);
2773 pbBuf[0] = 0x05; /* CD-ROM */
2774 pbBuf[1] = 0x80; /* removable */
2775# if 1/*ndef VBOX*/ /** @todo implement MESN + AENC. (async notification on removal and stuff.) */
2776 pbBuf[2] = 0x00; /* ISO */
2777 pbBuf[3] = 0x21; /* ATAPI-2 (XXX: put ATAPI-4 ?) */
2778# else
2779 pbBuf[2] = 0x00; /* ISO */
2780 pbBuf[3] = 0x91; /* format 1, MESN=1, AENC=9 ??? */
2781# endif
2782 pbBuf[4] = 31; /* additional length */
2783 pbBuf[5] = 0; /* reserved */
2784 pbBuf[6] = 0; /* reserved */
2785 pbBuf[7] = 0; /* reserved */
2786 ataR3SCSIPadStr(pbBuf + 8, s->szInquiryVendorId, 8);
2787 ataR3SCSIPadStr(pbBuf + 16, s->szInquiryProductId, 16);
2788 ataR3SCSIPadStr(pbBuf + 32, s->szInquiryRevision, 4);
2789 s->iSourceSink = ATAFN_SS_NULL;
2790 atapiR3CmdOK(s);
2791 return false;
2792}
2793
2794
2795static bool atapiR3ModeSenseErrorRecoverySS(ATADevState *s)
2796{
2797 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2798
2799 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2800 Assert(s->cbElementaryTransfer <= 16);
2801 ataH2BE_U16(&pbBuf[0], 16 + 6);
2802 pbBuf[2] = (uint8_t)s->MediaTrackType;
2803 pbBuf[3] = 0;
2804 pbBuf[4] = 0;
2805 pbBuf[5] = 0;
2806 pbBuf[6] = 0;
2807 pbBuf[7] = 0;
2808
2809 pbBuf[8] = 0x01;
2810 pbBuf[9] = 0x06;
2811 pbBuf[10] = 0x00; /* Maximum error recovery */
2812 pbBuf[11] = 0x05; /* 5 retries */
2813 pbBuf[12] = 0x00;
2814 pbBuf[13] = 0x00;
2815 pbBuf[14] = 0x00;
2816 pbBuf[15] = 0x00;
2817 s->iSourceSink = ATAFN_SS_NULL;
2818 atapiR3CmdOK(s);
2819 return false;
2820}
2821
2822
2823static bool atapiR3ModeSenseCDStatusSS(ATADevState *s)
2824{
2825 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2826
2827 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2828 Assert(s->cbElementaryTransfer <= 40);
2829 ataH2BE_U16(&pbBuf[0], 38);
2830 pbBuf[2] = (uint8_t)s->MediaTrackType;
2831 pbBuf[3] = 0;
2832 pbBuf[4] = 0;
2833 pbBuf[5] = 0;
2834 pbBuf[6] = 0;
2835 pbBuf[7] = 0;
2836
2837 pbBuf[8] = 0x2a;
2838 pbBuf[9] = 30; /* page length */
2839 pbBuf[10] = 0x08; /* DVD-ROM read support */
2840 pbBuf[11] = 0x00; /* no write support */
2841 /* The following claims we support audio play. This is obviously false,
2842 * but the Linux generic CDROM support makes many features depend on this
2843 * capability. If it's not set, this causes many things to be disabled. */
2844 pbBuf[12] = 0x71; /* multisession support, mode 2 form 1/2 support, audio play */
2845 pbBuf[13] = 0x00; /* no subchannel reads supported */
2846 pbBuf[14] = (1 << 0) | (1 << 3) | (1 << 5); /* lock supported, eject supported, tray type loading mechanism */
2847 if (s->pDrvMount->pfnIsLocked(s->pDrvMount))
2848 pbBuf[14] |= 1 << 1; /* report lock state */
2849 pbBuf[15] = 0; /* no subchannel reads supported, no separate audio volume control, no changer etc. */
2850 ataH2BE_U16(&pbBuf[16], 5632); /* (obsolete) claim 32x speed support */
2851 ataH2BE_U16(&pbBuf[18], 2); /* number of audio volume levels */
2852 ataH2BE_U16(&pbBuf[20], s->cbIOBuffer / _1K); /* buffer size supported in Kbyte */
2853 ataH2BE_U16(&pbBuf[22], 5632); /* (obsolete) current read speed 32x */
2854 pbBuf[24] = 0; /* reserved */
2855 pbBuf[25] = 0; /* reserved for digital audio (see idx 15) */
2856 ataH2BE_U16(&pbBuf[26], 0); /* (obsolete) maximum write speed */
2857 ataH2BE_U16(&pbBuf[28], 0); /* (obsolete) current write speed */
2858 ataH2BE_U16(&pbBuf[30], 0); /* copy management revision supported 0=no CSS */
2859 pbBuf[32] = 0; /* reserved */
2860 pbBuf[33] = 0; /* reserved */
2861 pbBuf[34] = 0; /* reserved */
2862 pbBuf[35] = 1; /* rotation control CAV */
2863 ataH2BE_U16(&pbBuf[36], 0); /* current write speed */
2864 ataH2BE_U16(&pbBuf[38], 0); /* number of write speed performance descriptors */
2865 s->iSourceSink = ATAFN_SS_NULL;
2866 atapiR3CmdOK(s);
2867 return false;
2868}
2869
2870
2871static bool atapiR3RequestSenseSS(ATADevState *s)
2872{
2873 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2874
2875 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2876 memset(pbBuf, '\0', s->cbElementaryTransfer);
2877 memcpy(pbBuf, s->abATAPISense, RT_MIN(s->cbElementaryTransfer, sizeof(s->abATAPISense)));
2878 s->iSourceSink = ATAFN_SS_NULL;
2879 atapiR3CmdOK(s);
2880 return false;
2881}
2882
2883
2884static bool atapiR3MechanismStatusSS(ATADevState *s)
2885{
2886 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2887
2888 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2889 Assert(s->cbElementaryTransfer <= 8);
2890 ataH2BE_U16(pbBuf, 0);
2891 /* no current LBA */
2892 pbBuf[2] = 0;
2893 pbBuf[3] = 0;
2894 pbBuf[4] = 0;
2895 pbBuf[5] = 1;
2896 ataH2BE_U16(pbBuf + 6, 0);
2897 s->iSourceSink = ATAFN_SS_NULL;
2898 atapiR3CmdOK(s);
2899 return false;
2900}
2901
2902
2903static bool atapiR3ReadTOCNormalSS(ATADevState *s)
2904{
2905 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer), *q, iStartTrack;
2906 bool fMSF;
2907 uint32_t cbSize;
2908
2909 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2910 fMSF = (s->aATAPICmd[1] >> 1) & 1;
2911 iStartTrack = s->aATAPICmd[6];
2912 if (iStartTrack > 1 && iStartTrack != 0xaa)
2913 {
2914 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
2915 return false;
2916 }
2917 q = pbBuf + 2;
2918 *q++ = 1; /* first session */
2919 *q++ = 1; /* last session */
2920 if (iStartTrack <= 1)
2921 {
2922 *q++ = 0; /* reserved */
2923 *q++ = 0x14; /* ADR, control */
2924 *q++ = 1; /* track number */
2925 *q++ = 0; /* reserved */
2926 if (fMSF)
2927 {
2928 *q++ = 0; /* reserved */
2929 ataLBA2MSF(q, 0);
2930 q += 3;
2931 }
2932 else
2933 {
2934 /* sector 0 */
2935 ataH2BE_U32(q, 0);
2936 q += 4;
2937 }
2938 }
2939 /* lead out track */
2940 *q++ = 0; /* reserved */
2941 *q++ = 0x14; /* ADR, control */
2942 *q++ = 0xaa; /* track number */
2943 *q++ = 0; /* reserved */
2944 if (fMSF)
2945 {
2946 *q++ = 0; /* reserved */
2947 ataLBA2MSF(q, s->cTotalSectors);
2948 q += 3;
2949 }
2950 else
2951 {
2952 ataH2BE_U32(q, s->cTotalSectors);
2953 q += 4;
2954 }
2955 cbSize = q - pbBuf;
2956 ataH2BE_U16(pbBuf, cbSize - 2);
2957 if (cbSize < s->cbTotalTransfer)
2958 s->cbTotalTransfer = cbSize;
2959 s->iSourceSink = ATAFN_SS_NULL;
2960 atapiR3CmdOK(s);
2961 return false;
2962}
2963
2964
2965static bool atapiR3ReadTOCMultiSS(ATADevState *s)
2966{
2967 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer);
2968 bool fMSF;
2969
2970 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
2971 Assert(s->cbElementaryTransfer <= 12);
2972 fMSF = (s->aATAPICmd[1] >> 1) & 1;
2973 /* multi session: only a single session defined */
2974/** @todo double-check this stuff against what a real drive says for a CD-ROM (not a CD-R) with only a single data session. Maybe solve the problem with "cdrdao read-toc" not being able to figure out whether numbers are in BCD or hex. */
2975 memset(pbBuf, 0, 12);
2976 pbBuf[1] = 0x0a;
2977 pbBuf[2] = 0x01;
2978 pbBuf[3] = 0x01;
2979 pbBuf[5] = 0x14; /* ADR, control */
2980 pbBuf[6] = 1; /* first track in last complete session */
2981 if (fMSF)
2982 {
2983 pbBuf[8] = 0; /* reserved */
2984 ataLBA2MSF(&pbBuf[9], 0);
2985 }
2986 else
2987 {
2988 /* sector 0 */
2989 ataH2BE_U32(pbBuf + 8, 0);
2990 }
2991 s->iSourceSink = ATAFN_SS_NULL;
2992 atapiR3CmdOK(s);
2993 return false;
2994}
2995
2996
2997static bool atapiR3ReadTOCRawSS(ATADevState *s)
2998{
2999 uint8_t *pbBuf = s->CTX_SUFF(pbIOBuffer), *q, iStartTrack;
3000 bool fMSF;
3001 uint32_t cbSize;
3002
3003 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
3004 fMSF = (s->aATAPICmd[1] >> 1) & 1;
3005 iStartTrack = s->aATAPICmd[6];
3006
3007 q = pbBuf + 2;
3008 *q++ = 1; /* first session */
3009 *q++ = 1; /* last session */
3010
3011 *q++ = 1; /* session number */
3012 *q++ = 0x14; /* data track */
3013 *q++ = 0; /* track number */
3014 *q++ = 0xa0; /* first track in program area */
3015 *q++ = 0; /* min */
3016 *q++ = 0; /* sec */
3017 *q++ = 0; /* frame */
3018 *q++ = 0;
3019 *q++ = 1; /* first track */
3020 *q++ = 0x00; /* disk type CD-DA or CD data */
3021 *q++ = 0;
3022
3023 *q++ = 1; /* session number */
3024 *q++ = 0x14; /* data track */
3025 *q++ = 0; /* track number */
3026 *q++ = 0xa1; /* last track in program area */
3027 *q++ = 0; /* min */
3028 *q++ = 0; /* sec */
3029 *q++ = 0; /* frame */
3030 *q++ = 0;
3031 *q++ = 1; /* last track */
3032 *q++ = 0;
3033 *q++ = 0;
3034
3035 *q++ = 1; /* session number */
3036 *q++ = 0x14; /* data track */
3037 *q++ = 0; /* track number */
3038 *q++ = 0xa2; /* lead-out */
3039 *q++ = 0; /* min */
3040 *q++ = 0; /* sec */
3041 *q++ = 0; /* frame */
3042 if (fMSF)
3043 {
3044 *q++ = 0; /* reserved */
3045 ataLBA2MSF(q, s->cTotalSectors);
3046 q += 3;
3047 }
3048 else
3049 {
3050 ataH2BE_U32(q, s->cTotalSectors);
3051 q += 4;
3052 }
3053
3054 *q++ = 1; /* session number */
3055 *q++ = 0x14; /* ADR, control */
3056 *q++ = 0; /* track number */
3057 *q++ = 1; /* point */
3058 *q++ = 0; /* min */
3059 *q++ = 0; /* sec */
3060 *q++ = 0; /* frame */
3061 if (fMSF)
3062 {
3063 *q++ = 0; /* reserved */
3064 ataLBA2MSF(q, 0);
3065 q += 3;
3066 }
3067 else
3068 {
3069 /* sector 0 */
3070 ataH2BE_U32(q, 0);
3071 q += 4;
3072 }
3073
3074 cbSize = q - pbBuf;
3075 ataH2BE_U16(pbBuf, cbSize - 2);
3076 if (cbSize < s->cbTotalTransfer)
3077 s->cbTotalTransfer = cbSize;
3078 s->iSourceSink = ATAFN_SS_NULL;
3079 atapiR3CmdOK(s);
3080 return false;
3081}
3082
3083
3084static void atapiR3ParseCmdVirtualATAPI(ATADevState *s)
3085{
3086 const uint8_t *pbPacket;
3087 uint8_t *pbBuf;
3088 uint32_t cbMax;
3089
3090 pbPacket = s->aATAPICmd;
3091 pbBuf = s->CTX_SUFF(pbIOBuffer);
3092 switch (pbPacket[0])
3093 {
3094 case SCSI_TEST_UNIT_READY:
3095 if (s->cNotifiedMediaChange > 0)
3096 {
3097 if (s->cNotifiedMediaChange-- > 2)
3098 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3099 else
3100 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3101 }
3102 else if (s->pDrvMount->pfnIsMounted(s->pDrvMount))
3103 atapiR3CmdOK(s);
3104 else
3105 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3106 break;
3107 case SCSI_GET_EVENT_STATUS_NOTIFICATION:
3108 cbMax = ataBE2H_U16(pbPacket + 7);
3109 ataR3StartTransfer(s, RT_MIN(cbMax, 8), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_GET_EVENT_STATUS_NOTIFICATION, true);
3110 break;
3111 case SCSI_MODE_SENSE_10:
3112 {
3113 uint8_t uPageControl, uPageCode;
3114 cbMax = ataBE2H_U16(pbPacket + 7);
3115 uPageControl = pbPacket[2] >> 6;
3116 uPageCode = pbPacket[2] & 0x3f;
3117 switch (uPageControl)
3118 {
3119 case SCSI_PAGECONTROL_CURRENT:
3120 switch (uPageCode)
3121 {
3122 case SCSI_MODEPAGE_ERROR_RECOVERY:
3123 ataR3StartTransfer(s, RT_MIN(cbMax, 16), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_MODE_SENSE_ERROR_RECOVERY, true);
3124 break;
3125 case SCSI_MODEPAGE_CD_STATUS:
3126 ataR3StartTransfer(s, RT_MIN(cbMax, 40), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_MODE_SENSE_CD_STATUS, true);
3127 break;
3128 default:
3129 goto error_cmd;
3130 }
3131 break;
3132 case SCSI_PAGECONTROL_CHANGEABLE:
3133 goto error_cmd;
3134 case SCSI_PAGECONTROL_DEFAULT:
3135 goto error_cmd;
3136 default:
3137 case SCSI_PAGECONTROL_SAVED:
3138 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_SAVING_PARAMETERS_NOT_SUPPORTED);
3139 break;
3140 }
3141 break;
3142 }
3143 case SCSI_REQUEST_SENSE:
3144 cbMax = pbPacket[4];
3145 ataR3StartTransfer(s, RT_MIN(cbMax, 18), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_REQUEST_SENSE, true);
3146 break;
3147 case SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL:
3148 if (s->pDrvMount->pfnIsMounted(s->pDrvMount))
3149 {
3150 if (pbPacket[4] & 1)
3151 s->pDrvMount->pfnLock(s->pDrvMount);
3152 else
3153 s->pDrvMount->pfnUnlock(s->pDrvMount);
3154 atapiR3CmdOK(s);
3155 }
3156 else
3157 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3158 break;
3159 case SCSI_READ_10:
3160 case SCSI_READ_12:
3161 {
3162 uint32_t cSectors, iATAPILBA;
3163
3164 if (s->cNotifiedMediaChange > 0)
3165 {
3166 s->cNotifiedMediaChange-- ;
3167 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3168 break;
3169 }
3170 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3171 {
3172 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3173 break;
3174 }
3175 if (pbPacket[0] == SCSI_READ_10)
3176 cSectors = ataBE2H_U16(pbPacket + 7);
3177 else
3178 cSectors = ataBE2H_U32(pbPacket + 6);
3179 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3180 if (cSectors == 0)
3181 {
3182 atapiR3CmdOK(s);
3183 break;
3184 }
3185 if ((uint64_t)iATAPILBA + cSectors > s->cTotalSectors)
3186 {
3187 /* Rate limited logging, one log line per second. For
3188 * guests that insist on reading from places outside the
3189 * valid area this often generates too many release log
3190 * entries otherwise. */
3191 static uint64_t uLastLogTS = 0;
3192 if (RTTimeMilliTS() >= uLastLogTS + 1000)
3193 {
3194 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM block number %Ld invalid (READ)\n", s->iLUN, (uint64_t)iATAPILBA + cSectors));
3195 uLastLogTS = RTTimeMilliTS();
3196 }
3197 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_LOGICAL_BLOCK_OOR);
3198 break;
3199 }
3200 atapiR3ReadSectors(s, iATAPILBA, cSectors, 2048);
3201 break;
3202 }
3203 case SCSI_READ_CD:
3204 {
3205 uint32_t cSectors, iATAPILBA;
3206
3207 if (s->cNotifiedMediaChange > 0)
3208 {
3209 s->cNotifiedMediaChange-- ;
3210 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3211 break;
3212 }
3213 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3214 {
3215 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3216 break;
3217 }
3218 cSectors = (pbPacket[6] << 16) | (pbPacket[7] << 8) | pbPacket[8];
3219 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3220 if (cSectors == 0)
3221 {
3222 atapiR3CmdOK(s);
3223 break;
3224 }
3225 if ((uint64_t)iATAPILBA + cSectors > s->cTotalSectors)
3226 {
3227 /* Rate limited logging, one log line per second. For
3228 * guests that insist on reading from places outside the
3229 * valid area this often generates too many release log
3230 * entries otherwise. */
3231 static uint64_t uLastLogTS = 0;
3232 if (RTTimeMilliTS() >= uLastLogTS + 1000)
3233 {
3234 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM block number %Ld invalid (READ CD)\n", s->iLUN, (uint64_t)iATAPILBA + cSectors));
3235 uLastLogTS = RTTimeMilliTS();
3236 }
3237 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_LOGICAL_BLOCK_OOR);
3238 break;
3239 }
3240 switch (pbPacket[9] & 0xf8)
3241 {
3242 case 0x00:
3243 /* nothing */
3244 atapiR3CmdOK(s);
3245 break;
3246 case 0x10:
3247 /* normal read */
3248 atapiR3ReadSectors(s, iATAPILBA, cSectors, 2048);
3249 break;
3250 case 0xf8:
3251 /* read all data */
3252 atapiR3ReadSectors(s, iATAPILBA, cSectors, 2352);
3253 break;
3254 default:
3255 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM sector format not supported (%#x)\n", s->iLUN, pbPacket[9] & 0xf8));
3256 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
3257 break;
3258 }
3259 break;
3260 }
3261 case SCSI_SEEK_10:
3262 {
3263 uint32_t iATAPILBA;
3264 if (s->cNotifiedMediaChange > 0)
3265 {
3266 s->cNotifiedMediaChange-- ;
3267 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3268 break;
3269 }
3270 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3271 {
3272 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3273 break;
3274 }
3275 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3276 if (iATAPILBA > s->cTotalSectors)
3277 {
3278 /* Rate limited logging, one log line per second. For
3279 * guests that insist on seeking to places outside the
3280 * valid area this often generates too many release log
3281 * entries otherwise. */
3282 static uint64_t uLastLogTS = 0;
3283 if (RTTimeMilliTS() >= uLastLogTS + 1000)
3284 {
3285 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM block number %Ld invalid (SEEK)\n", s->iLUN, (uint64_t)iATAPILBA));
3286 uLastLogTS = RTTimeMilliTS();
3287 }
3288 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_LOGICAL_BLOCK_OOR);
3289 break;
3290 }
3291 atapiR3CmdOK(s);
3292 ataSetStatus(s, ATA_STAT_SEEK); /* Linux expects this. */
3293 break;
3294 }
3295 case SCSI_START_STOP_UNIT:
3296 {
3297 int rc = VINF_SUCCESS;
3298 switch (pbPacket[4] & 3)
3299 {
3300 case 0: /* 00 - Stop motor */
3301 case 1: /* 01 - Start motor */
3302 break;
3303 case 2: /* 10 - Eject media */
3304 {
3305 /* This must be done from EMT. */
3306 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
3307 PPDMDEVINS pDevIns = ATADEVSTATE_2_DEVINS(s);
3308 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
3309
3310 PDMCritSectLeave(&pCtl->lock);
3311 rc = VMR3ReqPriorityCallWait(PDMDevHlpGetVM(pDevIns), VMCPUID_ANY,
3312 (PFNRT)s->pDrvMount->pfnUnmount, 3,
3313 s->pDrvMount, false /*=fForce*/, true /*=fEject*/);
3314 Assert(RT_SUCCESS(rc) || rc == VERR_PDM_MEDIA_LOCKED || rc == VERR_PDM_MEDIA_NOT_MOUNTED);
3315 if (RT_SUCCESS(rc) && pThis->pMediaNotify)
3316 {
3317 rc = VMR3ReqCallNoWait(PDMDevHlpGetVM(pDevIns), VMCPUID_ANY,
3318 (PFNRT)pThis->pMediaNotify->pfnEjected, 2,
3319 pThis->pMediaNotify, s->iLUN);
3320 AssertRC(rc);
3321 }
3322 {
3323 STAM_PROFILE_START(&pCtl->StatLockWait, a);
3324 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
3325 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
3326 }
3327 break;
3328 }
3329 case 3: /* 11 - Load media */
3330 /** @todo rc = s->pDrvMount->pfnLoadMedia(s->pDrvMount) */
3331 break;
3332 }
3333 if (RT_SUCCESS(rc))
3334 atapiR3CmdOK(s);
3335 else
3336 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIA_LOAD_OR_EJECT_FAILED);
3337 break;
3338 }
3339 case SCSI_MECHANISM_STATUS:
3340 {
3341 cbMax = ataBE2H_U16(pbPacket + 8);
3342 ataR3StartTransfer(s, RT_MIN(cbMax, 8), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_MECHANISM_STATUS, true);
3343 break;
3344 }
3345 case SCSI_READ_TOC_PMA_ATIP:
3346 {
3347 uint8_t format;
3348
3349 if (s->cNotifiedMediaChange > 0)
3350 {
3351 s->cNotifiedMediaChange-- ;
3352 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3353 break;
3354 }
3355 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3356 {
3357 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3358 break;
3359 }
3360 cbMax = ataBE2H_U16(pbPacket + 7);
3361 /* SCSI MMC-3 spec says format is at offset 2 (lower 4 bits),
3362 * but Linux kernel uses offset 9 (topmost 2 bits). Hope that
3363 * the other field is clear... */
3364 format = (pbPacket[2] & 0xf) | (pbPacket[9] >> 6);
3365 switch (format)
3366 {
3367 case 0:
3368 ataR3StartTransfer(s, cbMax, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_TOC_NORMAL, true);
3369 break;
3370 case 1:
3371 ataR3StartTransfer(s, RT_MIN(cbMax, 12), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_TOC_MULTI, true);
3372 break;
3373 case 2:
3374 ataR3StartTransfer(s, cbMax, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_TOC_RAW, true);
3375 break;
3376 default:
3377 error_cmd:
3378 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
3379 break;
3380 }
3381 break;
3382 }
3383 case SCSI_READ_CAPACITY:
3384 if (s->cNotifiedMediaChange > 0)
3385 {
3386 s->cNotifiedMediaChange-- ;
3387 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3388 break;
3389 }
3390 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3391 {
3392 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3393 break;
3394 }
3395 ataR3StartTransfer(s, 8, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_CAPACITY, true);
3396 break;
3397 case SCSI_READ_DISC_INFORMATION:
3398 if (s->cNotifiedMediaChange > 0)
3399 {
3400 s->cNotifiedMediaChange-- ;
3401 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3402 break;
3403 }
3404 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3405 {
3406 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3407 break;
3408 }
3409 cbMax = ataBE2H_U16(pbPacket + 7);
3410 ataR3StartTransfer(s, RT_MIN(cbMax, 34), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_DISC_INFORMATION, true);
3411 break;
3412 case SCSI_READ_TRACK_INFORMATION:
3413 if (s->cNotifiedMediaChange > 0)
3414 {
3415 s->cNotifiedMediaChange-- ;
3416 atapiR3CmdErrorSimple(s, SCSI_SENSE_UNIT_ATTENTION, SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED); /* media changed */
3417 break;
3418 }
3419 else if (!s->pDrvMount->pfnIsMounted(s->pDrvMount))
3420 {
3421 atapiR3CmdErrorSimple(s, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT);
3422 break;
3423 }
3424 cbMax = ataBE2H_U16(pbPacket + 7);
3425 ataR3StartTransfer(s, RT_MIN(cbMax, 36), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_TRACK_INFORMATION, true);
3426 break;
3427 case SCSI_GET_CONFIGURATION:
3428 /* No media change stuff here, it can confuse Linux guests. */
3429 cbMax = ataBE2H_U16(pbPacket + 7);
3430 ataR3StartTransfer(s, RT_MIN(cbMax, 80), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_GET_CONFIGURATION, true);
3431 break;
3432 case SCSI_INQUIRY:
3433 cbMax = ataBE2H_U16(pbPacket + 3);
3434 ataR3StartTransfer(s, RT_MIN(cbMax, 36), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_INQUIRY, true);
3435 break;
3436 case SCSI_READ_DVD_STRUCTURE:
3437 {
3438 cbMax = ataBE2H_U16(pbPacket + 8);
3439 ataR3StartTransfer(s, RT_MIN(cbMax, 4), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_READ_DVD_STRUCTURE, true);
3440 break;
3441 }
3442 default:
3443 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
3444 break;
3445 }
3446}
3447
3448
3449/*
3450 * Parse ATAPI commands, passing them directly to the CD/DVD drive.
3451 */
3452static void atapiR3ParseCmdPassthrough(ATADevState *s)
3453{
3454 const uint8_t *pbPacket;
3455 uint8_t *pbBuf;
3456 uint32_t cSectors, iATAPILBA;
3457 uint32_t cbTransfer = 0;
3458 PDMBLOCKTXDIR uTxDir = PDMBLOCKTXDIR_NONE;
3459
3460 pbPacket = s->aATAPICmd;
3461 pbBuf = s->CTX_SUFF(pbIOBuffer);
3462 switch (pbPacket[0])
3463 {
3464 case SCSI_BLANK:
3465 goto sendcmd;
3466 case SCSI_CLOSE_TRACK_SESSION:
3467 goto sendcmd;
3468 case SCSI_ERASE_10:
3469 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3470 cbTransfer = ataBE2H_U16(pbPacket + 7);
3471 Log2(("ATAPI PT: lba %d\n", iATAPILBA));
3472 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3473 goto sendcmd;
3474 case SCSI_FORMAT_UNIT:
3475 cbTransfer = s->uATARegLCyl | (s->uATARegHCyl << 8); /* use ATAPI transfer length */
3476 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3477 goto sendcmd;
3478 case SCSI_GET_CONFIGURATION:
3479 cbTransfer = ataBE2H_U16(pbPacket + 7);
3480 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3481 goto sendcmd;
3482 case SCSI_GET_EVENT_STATUS_NOTIFICATION:
3483 cbTransfer = ataBE2H_U16(pbPacket + 7);
3484 if (ASMAtomicReadU32(&s->MediaEventStatus) != ATA_EVENT_STATUS_UNCHANGED)
3485 {
3486 ataR3StartTransfer(s, RT_MIN(cbTransfer, 8), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_GET_EVENT_STATUS_NOTIFICATION, true);
3487 break;
3488 }
3489 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3490 goto sendcmd;
3491 case SCSI_GET_PERFORMANCE:
3492 cbTransfer = s->uATARegLCyl | (s->uATARegHCyl << 8); /* use ATAPI transfer length */
3493 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3494 goto sendcmd;
3495 case SCSI_INQUIRY:
3496 cbTransfer = ataBE2H_U16(pbPacket + 3);
3497 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3498 goto sendcmd;
3499 case SCSI_LOAD_UNLOAD_MEDIUM:
3500 goto sendcmd;
3501 case SCSI_MECHANISM_STATUS:
3502 cbTransfer = ataBE2H_U16(pbPacket + 8);
3503 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3504 goto sendcmd;
3505 case SCSI_MODE_SELECT_10:
3506 cbTransfer = ataBE2H_U16(pbPacket + 7);
3507 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3508 goto sendcmd;
3509 case SCSI_MODE_SENSE_10:
3510 cbTransfer = ataBE2H_U16(pbPacket + 7);
3511 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3512 goto sendcmd;
3513 case SCSI_PAUSE_RESUME:
3514 goto sendcmd;
3515 case SCSI_PLAY_AUDIO_10:
3516 goto sendcmd;
3517 case SCSI_PLAY_AUDIO_12:
3518 goto sendcmd;
3519 case SCSI_PLAY_AUDIO_MSF:
3520 goto sendcmd;
3521 case SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL:
3522 /** @todo do not forget to unlock when a VM is shut down */
3523 goto sendcmd;
3524 case SCSI_READ_10:
3525 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3526 cSectors = ataBE2H_U16(pbPacket + 7);
3527 Log2(("ATAPI PT: lba %d sectors %d\n", iATAPILBA, cSectors));
3528 s->cbATAPISector = 2048;
3529 cbTransfer = cSectors * s->cbATAPISector;
3530 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3531 goto sendcmd;
3532 case SCSI_READ_12:
3533 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3534 cSectors = ataBE2H_U32(pbPacket + 6);
3535 Log2(("ATAPI PT: lba %d sectors %d\n", iATAPILBA, cSectors));
3536 s->cbATAPISector = 2048;
3537 cbTransfer = cSectors * s->cbATAPISector;
3538 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3539 goto sendcmd;
3540 case SCSI_READ_BUFFER:
3541 cbTransfer = ataBE2H_U24(pbPacket + 6);
3542 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3543 goto sendcmd;
3544 case SCSI_READ_BUFFER_CAPACITY:
3545 cbTransfer = ataBE2H_U16(pbPacket + 7);
3546 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3547 goto sendcmd;
3548 case SCSI_READ_CAPACITY:
3549 cbTransfer = 8;
3550 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3551 goto sendcmd;
3552 case SCSI_READ_CD:
3553 case SCSI_READ_CD_MSF:
3554 {
3555 /* Get sector size based on the expected sector type field. */
3556 switch ((pbPacket[1] >> 2) & 0x7)
3557 {
3558 case 0x0: /* All types. */
3559 {
3560 uint32_t iLbaStart;
3561
3562 if (pbPacket[0] == SCSI_READ_CD)
3563 iLbaStart = ataBE2H_U32(&pbPacket[2]);
3564 else
3565 iLbaStart = ataMSF2LBA(&pbPacket[3]);
3566
3567 if (s->pTrackList)
3568 s->cbATAPISector = ATAPIPassthroughTrackListGetSectorSizeFromLba(s->pTrackList, iLbaStart);
3569 else
3570 s->cbATAPISector = 2048; /* Might be incorrect if we couldn't determine the type. */
3571 break;
3572 }
3573 case 0x1: /* CD-DA */
3574 s->cbATAPISector = 2352;
3575 break;
3576 case 0x2: /* Mode 1 */
3577 s->cbATAPISector = 2048;
3578 break;
3579 case 0x3: /* Mode 2 formless */
3580 s->cbATAPISector = 2336;
3581 break;
3582 case 0x4: /* Mode 2 form 1 */
3583 s->cbATAPISector = 2048;
3584 break;
3585 case 0x5: /* Mode 2 form 2 */
3586 s->cbATAPISector = 2324;
3587 break;
3588 default: /* Reserved */
3589 AssertMsgFailed(("Unknown sector type\n"));
3590 s->cbATAPISector = 0; /** @todo we should probably fail the command here already. */
3591 }
3592
3593 if (pbPacket[0] == SCSI_READ_CD)
3594 cbTransfer = ataBE2H_U24(pbPacket + 6) * s->cbATAPISector;
3595 else /* SCSI_READ_MSF */
3596 {
3597 cSectors = ataMSF2LBA(pbPacket + 6) - ataMSF2LBA(pbPacket + 3);
3598 if (cSectors > 32)
3599 cSectors = 32; /* Limit transfer size to 64~74K. Safety first. In any case this can only harm software doing CDDA extraction. */
3600 cbTransfer = cSectors * s->cbATAPISector;
3601 }
3602 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3603 goto sendcmd;
3604 }
3605 case SCSI_READ_DISC_INFORMATION:
3606 cbTransfer = ataBE2H_U16(pbPacket + 7);
3607 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3608 goto sendcmd;
3609 case SCSI_READ_DVD_STRUCTURE:
3610 cbTransfer = ataBE2H_U16(pbPacket + 8);
3611 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3612 goto sendcmd;
3613 case SCSI_READ_FORMAT_CAPACITIES:
3614 cbTransfer = ataBE2H_U16(pbPacket + 7);
3615 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3616 goto sendcmd;
3617 case SCSI_READ_SUBCHANNEL:
3618 cbTransfer = ataBE2H_U16(pbPacket + 7);
3619 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3620 goto sendcmd;
3621 case SCSI_READ_TOC_PMA_ATIP:
3622 cbTransfer = ataBE2H_U16(pbPacket + 7);
3623 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3624 goto sendcmd;
3625 case SCSI_READ_TRACK_INFORMATION:
3626 cbTransfer = ataBE2H_U16(pbPacket + 7);
3627 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3628 goto sendcmd;
3629 case SCSI_REPAIR_TRACK:
3630 goto sendcmd;
3631 case SCSI_REPORT_KEY:
3632 cbTransfer = ataBE2H_U16(pbPacket + 8);
3633 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3634 goto sendcmd;
3635 case SCSI_REQUEST_SENSE:
3636 cbTransfer = pbPacket[4];
3637 if ((s->abATAPISense[2] & 0x0f) != SCSI_SENSE_NONE)
3638 {
3639 ataR3StartTransfer(s, RT_MIN(cbTransfer, 18), PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_ATAPI_CMD, ATAFN_SS_ATAPI_REQUEST_SENSE, true);
3640 break;
3641 }
3642 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3643 goto sendcmd;
3644 case SCSI_RESERVE_TRACK:
3645 goto sendcmd;
3646 case SCSI_SCAN:
3647 goto sendcmd;
3648 case SCSI_SEEK_10:
3649 goto sendcmd;
3650 case SCSI_SEND_CUE_SHEET:
3651 cbTransfer = ataBE2H_U24(pbPacket + 6);
3652 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3653 goto sendcmd;
3654 case SCSI_SEND_DVD_STRUCTURE:
3655 cbTransfer = ataBE2H_U16(pbPacket + 8);
3656 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3657 goto sendcmd;
3658 case SCSI_SEND_EVENT:
3659 cbTransfer = ataBE2H_U16(pbPacket + 8);
3660 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3661 goto sendcmd;
3662 case SCSI_SEND_KEY:
3663 cbTransfer = ataBE2H_U16(pbPacket + 8);
3664 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3665 goto sendcmd;
3666 case SCSI_SEND_OPC_INFORMATION:
3667 cbTransfer = ataBE2H_U16(pbPacket + 7);
3668 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3669 goto sendcmd;
3670 case SCSI_SET_CD_SPEED:
3671 goto sendcmd;
3672 case SCSI_SET_READ_AHEAD:
3673 goto sendcmd;
3674 case SCSI_SET_STREAMING:
3675 cbTransfer = ataBE2H_U16(pbPacket + 9);
3676 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3677 goto sendcmd;
3678 case SCSI_START_STOP_UNIT:
3679 goto sendcmd;
3680 case SCSI_STOP_PLAY_SCAN:
3681 goto sendcmd;
3682 case SCSI_SYNCHRONIZE_CACHE:
3683 goto sendcmd;
3684 case SCSI_TEST_UNIT_READY:
3685 goto sendcmd;
3686 case SCSI_VERIFY_10:
3687 goto sendcmd;
3688 case SCSI_WRITE_10:
3689 case SCSI_WRITE_AND_VERIFY_10:
3690 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3691 cSectors = ataBE2H_U16(pbPacket + 7);
3692 if (s->pTrackList)
3693 s->cbATAPISector = ATAPIPassthroughTrackListGetSectorSizeFromLba(s->pTrackList, iATAPILBA);
3694 else
3695 s->cbATAPISector = 2048;
3696 Log2(("ATAPI PT: lba %d sectors %d sector size %d\n", iATAPILBA, cSectors, s->cbATAPISector));
3697 cbTransfer = cSectors * s->cbATAPISector;
3698 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3699 goto sendcmd;
3700 case SCSI_WRITE_12:
3701 iATAPILBA = ataBE2H_U32(pbPacket + 2);
3702 cSectors = ataBE2H_U32(pbPacket + 6);
3703 if (s->pTrackList)
3704 s->cbATAPISector = ATAPIPassthroughTrackListGetSectorSizeFromLba(s->pTrackList, iATAPILBA);
3705 else
3706 s->cbATAPISector = 2048;
3707 Log2(("ATAPI PT: lba %d sectors %d sector size %d\n", iATAPILBA, cSectors, s->cbATAPISector));
3708 cbTransfer = cSectors * s->cbATAPISector;
3709 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3710 goto sendcmd;
3711 case SCSI_WRITE_BUFFER:
3712 switch (pbPacket[1] & 0x1f)
3713 {
3714 case 0x04: /* download microcode */
3715 case 0x05: /* download microcode and save */
3716 case 0x06: /* download microcode with offsets */
3717 case 0x07: /* download microcode with offsets and save */
3718 case 0x0e: /* download microcode with offsets and defer activation */
3719 case 0x0f: /* activate deferred microcode */
3720 LogRel(("PIIX3 ATA: LUN#%d: CD-ROM passthrough command attempted to update firmware, blocked\n", s->iLUN));
3721 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
3722 break;
3723 default:
3724 cbTransfer = ataBE2H_U16(pbPacket + 6);
3725 uTxDir = PDMBLOCKTXDIR_TO_DEVICE;
3726 goto sendcmd;
3727 }
3728 break;
3729 case SCSI_REPORT_LUNS: /* Not part of MMC-3, but used by Windows. */
3730 cbTransfer = ataBE2H_U32(pbPacket + 6);
3731 uTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
3732 goto sendcmd;
3733 case SCSI_REZERO_UNIT:
3734 /* Obsolete command used by cdrecord. What else would one expect?
3735 * This command is not sent to the drive, it is handled internally,
3736 * as the Linux kernel doesn't like it (message "scsi: unknown
3737 * opcode 0x01" in syslog) and replies with a sense code of 0,
3738 * which sends cdrecord to an endless loop. */
3739 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
3740 break;
3741 default:
3742 LogRel(("PIIX3 ATA: LUN#%d: passthrough unimplemented for command %#x\n", s->iLUN, pbPacket[0]));
3743 atapiR3CmdErrorSimple(s, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
3744 break;
3745 sendcmd:
3746 /*
3747 * Send a command to the drive, passing data in/out as required.
3748 * Commands which exceed the I/O buffer size are split below
3749 * or aborted if splitting is not implemented.
3750 */
3751 Log2(("ATAPI PT: max size %d\n", cbTransfer));
3752 if (cbTransfer == 0)
3753 uTxDir = PDMBLOCKTXDIR_NONE;
3754 ataR3StartTransfer(s, cbTransfer, uTxDir, ATAFN_BT_ATAPI_PASSTHROUGH_CMD, ATAFN_SS_ATAPI_PASSTHROUGH, true);
3755 }
3756}
3757
3758
3759static void atapiR3ParseCmd(ATADevState *s)
3760{
3761 const uint8_t *pbPacket;
3762
3763 pbPacket = s->aATAPICmd;
3764# ifdef DEBUG
3765 Log(("%s: LUN#%d DMA=%d CMD=%#04x \"%s\"\n", __FUNCTION__, s->iLUN, s->fDMA, pbPacket[0], SCSICmdText(pbPacket[0])));
3766# else /* !DEBUG */
3767 Log(("%s: LUN#%d DMA=%d CMD=%#04x\n", __FUNCTION__, s->iLUN, s->fDMA, pbPacket[0]));
3768# endif /* !DEBUG */
3769 Log2(("%s: limit=%#x packet: %.*Rhxs\n", __FUNCTION__, s->uATARegLCyl | (s->uATARegHCyl << 8), ATAPI_PACKET_SIZE, pbPacket));
3770
3771 if (s->fATAPIPassthrough)
3772 atapiR3ParseCmdPassthrough(s);
3773 else
3774 atapiR3ParseCmdVirtualATAPI(s);
3775}
3776
3777
3778static bool ataR3PacketSS(ATADevState *s)
3779{
3780 s->fDMA = !!(s->uATARegFeature & 1);
3781 memcpy(s->aATAPICmd, s->CTX_SUFF(pbIOBuffer), ATAPI_PACKET_SIZE);
3782 s->uTxDir = PDMBLOCKTXDIR_NONE;
3783 s->cbTotalTransfer = 0;
3784 s->cbElementaryTransfer = 0;
3785 s->cbAtapiPassthroughTransfer = 0;
3786 atapiR3ParseCmd(s);
3787 return false;
3788}
3789
3790
3791/**
3792 * SCSI_GET_EVENT_STATUS_NOTIFICATION should return "medium removed" event
3793 * from now on, regardless if there was a medium inserted or not.
3794 */
3795static void ataR3MediumRemoved(ATADevState *s)
3796{
3797 ASMAtomicWriteU32(&s->MediaEventStatus, ATA_EVENT_STATUS_MEDIA_REMOVED);
3798}
3799
3800
3801/**
3802 * SCSI_GET_EVENT_STATUS_NOTIFICATION should return "medium inserted". If
3803 * there was already a medium inserted, don't forget to send the "medium
3804 * removed" event first.
3805 */
3806static void ataR3MediumInserted(ATADevState *s)
3807{
3808 uint32_t OldStatus, NewStatus;
3809 do
3810 {
3811 OldStatus = ASMAtomicReadU32(&s->MediaEventStatus);
3812 switch (OldStatus)
3813 {
3814 case ATA_EVENT_STATUS_MEDIA_CHANGED:
3815 case ATA_EVENT_STATUS_MEDIA_REMOVED:
3816 /* no change, we will send "medium removed" + "medium inserted" */
3817 NewStatus = ATA_EVENT_STATUS_MEDIA_CHANGED;
3818 break;
3819 default:
3820 NewStatus = ATA_EVENT_STATUS_MEDIA_NEW;
3821 break;
3822 }
3823 } while (!ASMAtomicCmpXchgU32(&s->MediaEventStatus, NewStatus, OldStatus));
3824}
3825
3826
3827/**
3828 * @interface_method_impl{PDMIMOUNTNOTIFY, pfnMountNotify}
3829 */
3830static DECLCALLBACK(void) ataR3MountNotify(PPDMIMOUNTNOTIFY pInterface)
3831{
3832 ATADevState *pIf = PDMIMOUNTNOTIFY_2_ATASTATE(pInterface);
3833 Log(("%s: changing LUN#%d\n", __FUNCTION__, pIf->iLUN));
3834
3835 /* Ignore the call if we're called while being attached. */
3836 if (!pIf->pDrvBlock)
3837 return;
3838
3839 if (pIf->fATAPI)
3840 pIf->cTotalSectors = pIf->pDrvBlock->pfnGetSize(pIf->pDrvBlock) / 2048;
3841 else
3842 pIf->cTotalSectors = pIf->pDrvBlock->pfnGetSize(pIf->pDrvBlock) / pIf->cbSector;
3843
3844 LogRel(("PIIX3 ATA: LUN#%d: CD/DVD, total number of sectors %Ld, passthrough unchanged\n", pIf->iLUN, pIf->cTotalSectors));
3845
3846 /* Report media changed in TEST UNIT and other (probably incorrect) places. */
3847 if (pIf->cNotifiedMediaChange < 2)
3848 pIf->cNotifiedMediaChange = 1;
3849 ataR3MediumInserted(pIf);
3850 ataR3MediumTypeSet(pIf, ATA_MEDIA_TYPE_UNKNOWN);
3851}
3852
3853/**
3854 * @interface_method_impl{PDMIMOUNTNOTIFY, pfnUnmountNotify}
3855 */
3856static DECLCALLBACK(void) ataR3UnmountNotify(PPDMIMOUNTNOTIFY pInterface)
3857{
3858 ATADevState *pIf = PDMIMOUNTNOTIFY_2_ATASTATE(pInterface);
3859 Log(("%s:\n", __FUNCTION__));
3860 pIf->cTotalSectors = 0;
3861
3862 /*
3863 * Whatever I do, XP will not use the GET MEDIA STATUS nor the EVENT stuff.
3864 * However, it will respond to TEST UNIT with a 0x6 0x28 (media changed) sense code.
3865 * So, we'll give it 4 TEST UNIT command to catch up, two which the media is not
3866 * present and 2 in which it is changed.
3867 */
3868 pIf->cNotifiedMediaChange = 1;
3869 ataR3MediumRemoved(pIf);
3870 ataR3MediumTypeSet(pIf, ATA_MEDIA_NO_DISC);
3871}
3872
3873static void ataR3PacketBT(ATADevState *s)
3874{
3875 s->cbElementaryTransfer = s->cbTotalTransfer;
3876 s->cbAtapiPassthroughTransfer = s->cbTotalTransfer;
3877 s->uATARegNSector = (s->uATARegNSector & ~7) | ATAPI_INT_REASON_CD;
3878 Log2(("%s: interrupt reason %#04x\n", __FUNCTION__, s->uATARegNSector));
3879 ataSetStatusValue(s, ATA_STAT_READY);
3880}
3881
3882
3883static void ataR3ResetDevice(ATADevState *s)
3884{
3885 s->cMultSectors = ATA_MAX_MULT_SECTORS;
3886 s->cNotifiedMediaChange = 0;
3887 ASMAtomicWriteU32(&s->MediaEventStatus, ATA_EVENT_STATUS_UNCHANGED);
3888 ASMAtomicWriteU32(&s->MediaTrackType, ATA_MEDIA_TYPE_UNKNOWN);
3889 ataUnsetIRQ(s);
3890
3891 s->uATARegSelect = 0x20;
3892 ataSetStatusValue(s, ATA_STAT_READY);
3893 ataR3SetSignature(s);
3894 s->cbTotalTransfer = 0;
3895 s->cbElementaryTransfer = 0;
3896 s->cbAtapiPassthroughTransfer = 0;
3897 s->iIOBufferPIODataStart = 0;
3898 s->iIOBufferPIODataEnd = 0;
3899 s->iBeginTransfer = ATAFN_BT_NULL;
3900 s->iSourceSink = ATAFN_SS_NULL;
3901 s->fDMA = false;
3902 s->fATAPITransfer = false;
3903 s->uATATransferMode = ATA_MODE_UDMA | 2; /* PIIX3 supports only up to UDMA2 */
3904
3905 s->uATARegFeature = 0;
3906}
3907
3908
3909static bool ataR3ExecuteDeviceDiagnosticSS(ATADevState *s)
3910{
3911 ataR3SetSignature(s);
3912 if (s->fATAPI)
3913 ataSetStatusValue(s, 0); /* NOTE: READY is _not_ set */
3914 else
3915 ataSetStatusValue(s, ATA_STAT_READY | ATA_STAT_SEEK);
3916 s->uATARegError = 0x01;
3917 return false;
3918}
3919
3920
3921static int ataR3TrimSectors(ATADevState *s, uint64_t u64Sector, uint32_t cSectors,
3922 bool *pfRedo)
3923{
3924 RTRANGE TrimRange;
3925 PATACONTROLLER pCtl = ATADEVSTATE_2_CONTROLLER(s);
3926 int rc;
3927
3928 PDMCritSectLeave(&pCtl->lock);
3929
3930 TrimRange.offStart = u64Sector * s->cbSector;
3931 TrimRange.cbRange = cSectors * s->cbSector;
3932
3933 s->Led.Asserted.s.fWriting = s->Led.Actual.s.fWriting = 1;
3934 rc = s->pDrvBlock->pfnDiscard(s->pDrvBlock, &TrimRange, 1);
3935 s->Led.Actual.s.fWriting = 0;
3936
3937 if (RT_SUCCESS(rc))
3938 *pfRedo = false;
3939 else
3940 *pfRedo = ataR3IsRedoSetWarning(s, rc);
3941
3942 STAM_PROFILE_START(&pCtl->StatLockWait, a);
3943 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
3944 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
3945 return rc;
3946}
3947
3948
3949static bool ataR3TrimSS(ATADevState *s)
3950{
3951 int rc = VERR_GENERAL_FAILURE;
3952 uint32_t cRangesMax;
3953 uint64_t *pu64Range = (uint64_t *)s->CTX_SUFF(pbIOBuffer);
3954 bool fRedo = false;
3955
3956 cRangesMax = s->cbElementaryTransfer / sizeof(uint64_t);
3957 Assert(cRangesMax);
3958
3959 while (cRangesMax-- > 0)
3960 {
3961 if (ATA_RANGE_LENGTH_GET(*pu64Range) == 0)
3962 break;
3963
3964 rc = ataR3TrimSectors(s, *pu64Range & ATA_RANGE_LBA_MASK,
3965 ATA_RANGE_LENGTH_GET(*pu64Range), &fRedo);
3966 if (RT_FAILURE(rc))
3967 break;
3968
3969 pu64Range++;
3970 }
3971
3972 if (RT_SUCCESS(rc))
3973 {
3974 s->iSourceSink = ATAFN_SS_NULL;
3975 ataR3CmdOK(s, ATA_STAT_SEEK);
3976 }
3977 else
3978 {
3979 if (fRedo)
3980 return fRedo;
3981 if (s->cErrors++ < MAX_LOG_REL_ERRORS)
3982 LogRel(("PIIX3 ATA: LUN#%d: disk trim error (rc=%Rrc iSector=%#RX64 cSectors=%#RX32)\n",
3983 s->iLUN, rc, *pu64Range & ATA_RANGE_LBA_MASK, ATA_RANGE_LENGTH_GET(*pu64Range)));
3984
3985 /*
3986 * Check if we got interrupted. We don't need to set status variables
3987 * because the request was aborted.
3988 */
3989 if (rc != VERR_INTERRUPTED)
3990 ataR3CmdError(s, ID_ERR);
3991 }
3992
3993 return false;
3994}
3995
3996
3997static void ataR3ParseCmd(ATADevState *s, uint8_t cmd)
3998{
3999# ifdef DEBUG
4000 Log(("%s: LUN#%d CMD=%#04x \"%s\"\n", __FUNCTION__, s->iLUN, cmd, ATACmdText(cmd)));
4001# else /* !DEBUG */
4002 Log(("%s: LUN#%d CMD=%#04x\n", __FUNCTION__, s->iLUN, cmd));
4003# endif /* !DEBUG */
4004 s->fLBA48 = false;
4005 s->fDMA = false;
4006 if (cmd == ATA_IDLE_IMMEDIATE)
4007 {
4008 /* Detect Linux timeout recovery, first tries IDLE IMMEDIATE (which
4009 * would overwrite the failing command unfortunately), then RESET. */
4010 int32_t uCmdWait = -1;
4011 uint64_t uNow = RTTimeNanoTS();
4012 if (s->u64CmdTS)
4013 uCmdWait = (uNow - s->u64CmdTS) / 1000;
4014 LogRel(("PIIX3 ATA: LUN#%d: IDLE IMMEDIATE, CmdIf=%#04x (%d usec ago)\n",
4015 s->iLUN, s->uATARegCommand, uCmdWait));
4016 }
4017 s->uATARegCommand = cmd;
4018 switch (cmd)
4019 {
4020 case ATA_IDENTIFY_DEVICE:
4021 if (s->pDrvBlock && !s->fATAPI)
4022 ataR3StartTransfer(s, 512, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_NULL, ATAFN_SS_IDENTIFY, false);
4023 else
4024 {
4025 if (s->fATAPI)
4026 ataR3SetSignature(s);
4027 ataR3CmdError(s, ABRT_ERR);
4028 ataUnsetStatus(s, ATA_STAT_READY);
4029 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4030 }
4031 break;
4032 case ATA_RECALIBRATE:
4033 if (s->fATAPI)
4034 goto abort_cmd;
4035 /* fall through */
4036 case ATA_INITIALIZE_DEVICE_PARAMETERS:
4037 ataR3CmdOK(s, ATA_STAT_SEEK);
4038 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4039 break;
4040 case ATA_SET_MULTIPLE_MODE:
4041 if ( s->uATARegNSector != 0
4042 && ( s->uATARegNSector > ATA_MAX_MULT_SECTORS
4043 || (s->uATARegNSector & (s->uATARegNSector - 1)) != 0))
4044 {
4045 ataR3CmdError(s, ABRT_ERR);
4046 }
4047 else
4048 {
4049 Log2(("%s: set multi sector count to %d\n", __FUNCTION__, s->uATARegNSector));
4050 s->cMultSectors = s->uATARegNSector;
4051 ataR3CmdOK(s, 0);
4052 }
4053 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4054 break;
4055 case ATA_READ_VERIFY_SECTORS_EXT:
4056 s->fLBA48 = true;
4057 case ATA_READ_VERIFY_SECTORS:
4058 case ATA_READ_VERIFY_SECTORS_WITHOUT_RETRIES:
4059 /* do sector number check ? */
4060 ataR3CmdOK(s, ATA_STAT_SEEK);
4061 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4062 break;
4063 case ATA_READ_SECTORS_EXT:
4064 s->fLBA48 = true;
4065 case ATA_READ_SECTORS:
4066 case ATA_READ_SECTORS_WITHOUT_RETRIES:
4067 if (!s->pDrvBlock || s->fATAPI)
4068 goto abort_cmd;
4069 s->cSectorsPerIRQ = 1;
4070 ataR3StartTransfer(s, ataR3GetNSectors(s) * s->cbSector, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_READ_WRITE_SECTORS, ATAFN_SS_READ_SECTORS, false);
4071 break;
4072 case ATA_WRITE_SECTORS_EXT:
4073 s->fLBA48 = true;
4074 case ATA_WRITE_SECTORS:
4075 case ATA_WRITE_SECTORS_WITHOUT_RETRIES:
4076 if (!s->pDrvBlock || s->fATAPI)
4077 goto abort_cmd;
4078 s->cSectorsPerIRQ = 1;
4079 ataR3StartTransfer(s, ataR3GetNSectors(s) * s->cbSector, PDMBLOCKTXDIR_TO_DEVICE, ATAFN_BT_READ_WRITE_SECTORS, ATAFN_SS_WRITE_SECTORS, false);
4080 break;
4081 case ATA_READ_MULTIPLE_EXT:
4082 s->fLBA48 = true;
4083 case ATA_READ_MULTIPLE:
4084 if (!s->pDrvBlock || !s->cMultSectors || s->fATAPI)
4085 goto abort_cmd;
4086 s->cSectorsPerIRQ = s->cMultSectors;
4087 ataR3StartTransfer(s, ataR3GetNSectors(s) * s->cbSector, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_READ_WRITE_SECTORS, ATAFN_SS_READ_SECTORS, false);
4088 break;
4089 case ATA_WRITE_MULTIPLE_EXT:
4090 s->fLBA48 = true;
4091 case ATA_WRITE_MULTIPLE:
4092 if (!s->pDrvBlock || !s->cMultSectors || s->fATAPI)
4093 goto abort_cmd;
4094 s->cSectorsPerIRQ = s->cMultSectors;
4095 ataR3StartTransfer(s, ataR3GetNSectors(s) * s->cbSector, PDMBLOCKTXDIR_TO_DEVICE, ATAFN_BT_READ_WRITE_SECTORS, ATAFN_SS_WRITE_SECTORS, false);
4096 break;
4097 case ATA_READ_DMA_EXT:
4098 s->fLBA48 = true;
4099 case ATA_READ_DMA:
4100 case ATA_READ_DMA_WITHOUT_RETRIES:
4101 if (!s->pDrvBlock || s->fATAPI)
4102 goto abort_cmd;
4103 s->cSectorsPerIRQ = ATA_MAX_MULT_SECTORS;
4104 s->fDMA = true;
4105 ataR3StartTransfer(s, ataR3GetNSectors(s) * s->cbSector, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_READ_WRITE_SECTORS, ATAFN_SS_READ_SECTORS, false);
4106 break;
4107 case ATA_WRITE_DMA_EXT:
4108 s->fLBA48 = true;
4109 case ATA_WRITE_DMA:
4110 case ATA_WRITE_DMA_WITHOUT_RETRIES:
4111 if (!s->pDrvBlock || s->fATAPI)
4112 goto abort_cmd;
4113 s->cSectorsPerIRQ = ATA_MAX_MULT_SECTORS;
4114 s->fDMA = true;
4115 ataR3StartTransfer(s, ataR3GetNSectors(s) * s->cbSector, PDMBLOCKTXDIR_TO_DEVICE, ATAFN_BT_READ_WRITE_SECTORS, ATAFN_SS_WRITE_SECTORS, false);
4116 break;
4117 case ATA_READ_NATIVE_MAX_ADDRESS_EXT:
4118 s->fLBA48 = true;
4119 ataR3SetSector(s, s->cTotalSectors - 1);
4120 ataR3CmdOK(s, 0);
4121 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4122 break;
4123 case ATA_SEEK: /* Used by the SCO OpenServer. Command is marked as obsolete */
4124 ataR3CmdOK(s, 0);
4125 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4126 break;
4127 case ATA_READ_NATIVE_MAX_ADDRESS:
4128 ataR3SetSector(s, RT_MIN(s->cTotalSectors, 1 << 28) - 1);
4129 ataR3CmdOK(s, 0);
4130 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4131 break;
4132 case ATA_CHECK_POWER_MODE:
4133 s->uATARegNSector = 0xff; /* drive active or idle */
4134 ataR3CmdOK(s, 0);
4135 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4136 break;
4137 case ATA_SET_FEATURES:
4138 Log2(("%s: feature=%#x\n", __FUNCTION__, s->uATARegFeature));
4139 if (!s->pDrvBlock)
4140 goto abort_cmd;
4141 switch (s->uATARegFeature)
4142 {
4143 case 0x02: /* write cache enable */
4144 Log2(("%s: write cache enable\n", __FUNCTION__));
4145 ataR3CmdOK(s, ATA_STAT_SEEK);
4146 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4147 break;
4148 case 0xaa: /* read look-ahead enable */
4149 Log2(("%s: read look-ahead enable\n", __FUNCTION__));
4150 ataR3CmdOK(s, ATA_STAT_SEEK);
4151 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4152 break;
4153 case 0x55: /* read look-ahead disable */
4154 Log2(("%s: read look-ahead disable\n", __FUNCTION__));
4155 ataR3CmdOK(s, ATA_STAT_SEEK);
4156 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4157 break;
4158 case 0xcc: /* reverting to power-on defaults enable */
4159 Log2(("%s: revert to power-on defaults enable\n", __FUNCTION__));
4160 ataR3CmdOK(s, ATA_STAT_SEEK);
4161 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4162 break;
4163 case 0x66: /* reverting to power-on defaults disable */
4164 Log2(("%s: revert to power-on defaults disable\n", __FUNCTION__));
4165 ataR3CmdOK(s, ATA_STAT_SEEK);
4166 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4167 break;
4168 case 0x82: /* write cache disable */
4169 Log2(("%s: write cache disable\n", __FUNCTION__));
4170 /* As per the ATA/ATAPI-6 specs, a write cache disable
4171 * command MUST flush the write buffers to disc. */
4172 ataR3StartTransfer(s, 0, PDMBLOCKTXDIR_NONE, ATAFN_BT_NULL, ATAFN_SS_FLUSH, false);
4173 break;
4174 case 0x03: { /* set transfer mode */
4175 Log2(("%s: transfer mode %#04x\n", __FUNCTION__, s->uATARegNSector));
4176 switch (s->uATARegNSector & 0xf8)
4177 {
4178 case 0x00: /* PIO default */
4179 case 0x08: /* PIO mode */
4180 break;
4181 case ATA_MODE_MDMA: /* MDMA mode */
4182 s->uATATransferMode = (s->uATARegNSector & 0xf8) | RT_MIN(s->uATARegNSector & 0x07, ATA_MDMA_MODE_MAX);
4183 break;
4184 case ATA_MODE_UDMA: /* UDMA mode */
4185 s->uATATransferMode = (s->uATARegNSector & 0xf8) | RT_MIN(s->uATARegNSector & 0x07, ATA_UDMA_MODE_MAX);
4186 break;
4187 default:
4188 goto abort_cmd;
4189 }
4190 ataR3CmdOK(s, ATA_STAT_SEEK);
4191 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4192 break;
4193 }
4194 default:
4195 goto abort_cmd;
4196 }
4197 /*
4198 * OS/2 workarond:
4199 * The OS/2 IDE driver from MCP2 appears to rely on the feature register being
4200 * reset here. According to the specification, this is a driver bug as the register
4201 * contents are undefined after the call. This means we can just as well reset it.
4202 */
4203 s->uATARegFeature = 0;
4204 break;
4205 case ATA_FLUSH_CACHE_EXT:
4206 case ATA_FLUSH_CACHE:
4207 if (!s->pDrvBlock || s->fATAPI)
4208 goto abort_cmd;
4209 ataR3StartTransfer(s, 0, PDMBLOCKTXDIR_NONE, ATAFN_BT_NULL, ATAFN_SS_FLUSH, false);
4210 break;
4211 case ATA_STANDBY_IMMEDIATE:
4212 ataR3CmdOK(s, 0);
4213 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4214 break;
4215 case ATA_IDLE_IMMEDIATE:
4216 LogRel(("PIIX3 ATA: LUN#%d: aborting current command\n", s->iLUN));
4217 ataR3AbortCurrentCommand(s, false);
4218 break;
4219 case ATA_SLEEP:
4220 ataR3CmdOK(s, 0);
4221 ataHCSetIRQ(s);
4222 break;
4223 /* ATAPI commands */
4224 case ATA_IDENTIFY_PACKET_DEVICE:
4225 if (s->fATAPI)
4226 ataR3StartTransfer(s, 512, PDMBLOCKTXDIR_FROM_DEVICE, ATAFN_BT_NULL, ATAFN_SS_ATAPI_IDENTIFY, false);
4227 else
4228 {
4229 ataR3CmdError(s, ABRT_ERR);
4230 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4231 }
4232 break;
4233 case ATA_EXECUTE_DEVICE_DIAGNOSTIC:
4234 ataR3StartTransfer(s, 0, PDMBLOCKTXDIR_NONE, ATAFN_BT_NULL, ATAFN_SS_EXECUTE_DEVICE_DIAGNOSTIC, false);
4235 break;
4236 case ATA_DEVICE_RESET:
4237 if (!s->fATAPI)
4238 goto abort_cmd;
4239 LogRel(("PIIX3 ATA: LUN#%d: performing device RESET\n", s->iLUN));
4240 ataR3AbortCurrentCommand(s, true);
4241 break;
4242 case ATA_PACKET:
4243 if (!s->fATAPI)
4244 goto abort_cmd;
4245 /* overlapping commands not supported */
4246 if (s->uATARegFeature & 0x02)
4247 goto abort_cmd;
4248 ataR3StartTransfer(s, ATAPI_PACKET_SIZE, PDMBLOCKTXDIR_TO_DEVICE, ATAFN_BT_PACKET, ATAFN_SS_PACKET, false);
4249 break;
4250 case ATA_DATA_SET_MANAGEMENT:
4251 if (!s->pDrvBlock || !s->pDrvBlock->pfnDiscard)
4252 goto abort_cmd;
4253 if ( !(s->uATARegFeature & UINT8_C(0x01))
4254 || (s->uATARegFeature & ~UINT8_C(0x01)))
4255 goto abort_cmd;
4256 s->fDMA = true;
4257 ataR3StartTransfer(s, (s->uATARegNSectorHOB << 8 | s->uATARegNSector) * s->cbSector, PDMBLOCKTXDIR_TO_DEVICE, ATAFN_BT_NULL, ATAFN_SS_TRIM, false);
4258 break;
4259 default:
4260 abort_cmd:
4261 ataR3CmdError(s, ABRT_ERR);
4262 if (s->fATAPI)
4263 ataUnsetStatus(s, ATA_STAT_READY);
4264 ataHCSetIRQ(s); /* Shortcut, do not use AIO thread. */
4265 break;
4266 }
4267}
4268
4269# endif /* IN_RING3 */
4270#endif /* IN_RING0 || IN_RING3 */
4271
4272/*
4273 * Note: There are four distinct cases of port I/O handling depending on
4274 * which devices (if any) are attached to an IDE channel:
4275 *
4276 * 1) No device attached. No response to writes or reads (i.e. reads return
4277 * all bits set).
4278 *
4279 * 2) Both devices attached. Reads and writes are processed normally.
4280 *
4281 * 3) Device 0 only. If device 0 is selected, normal behavior applies. But
4282 * if Device 1 is selected, writes are still directed to Device 0 (except
4283 * commands are not executed), reads from control/command registers are
4284 * directed to Device 0, but status/alt status reads return 0. If Device 1
4285 * is a PACKET device, all reads return 0. See ATAPI-6 clause 9.16.1 and
4286 * Table 18 in clause 7.1.
4287 *
4288 * 4) Device 1 only - non-standard(!). Device 1 can't tell if Device 0 is
4289 * present or not and behaves the same. That means if Device 0 is selected,
4290 * Device 1 responds to writes (except commands are not executed) but does
4291 * not respond to reads. If Device 1 selected, normal behavior applies.
4292 * See ATAPI-6 clause 9.16.2 and Table 15 in clause 7.1.
4293 */
4294
4295static int ataIOPortWriteU8(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
4296{
4297 Log2(("%s: LUN#%d write addr=%#x val=%#04x\n", __FUNCTION__, pCtl->aIfs[pCtl->iSelectedIf].iLUN, addr, val));
4298 addr &= 7;
4299 switch (addr)
4300 {
4301 case 0:
4302 break;
4303 case 1: /* feature register */
4304 /* NOTE: data is written to the two drives */
4305 pCtl->aIfs[0].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4306 pCtl->aIfs[1].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4307 pCtl->aIfs[0].uATARegFeatureHOB = pCtl->aIfs[0].uATARegFeature;
4308 pCtl->aIfs[1].uATARegFeatureHOB = pCtl->aIfs[1].uATARegFeature;
4309 pCtl->aIfs[0].uATARegFeature = val;
4310 pCtl->aIfs[1].uATARegFeature = val;
4311 break;
4312 case 2: /* sector count */
4313 pCtl->aIfs[0].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4314 pCtl->aIfs[1].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4315 pCtl->aIfs[0].uATARegNSectorHOB = pCtl->aIfs[0].uATARegNSector;
4316 pCtl->aIfs[1].uATARegNSectorHOB = pCtl->aIfs[1].uATARegNSector;
4317 pCtl->aIfs[0].uATARegNSector = val;
4318 pCtl->aIfs[1].uATARegNSector = val;
4319 break;
4320 case 3: /* sector number */
4321 pCtl->aIfs[0].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4322 pCtl->aIfs[1].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4323 pCtl->aIfs[0].uATARegSectorHOB = pCtl->aIfs[0].uATARegSector;
4324 pCtl->aIfs[1].uATARegSectorHOB = pCtl->aIfs[1].uATARegSector;
4325 pCtl->aIfs[0].uATARegSector = val;
4326 pCtl->aIfs[1].uATARegSector = val;
4327 break;
4328 case 4: /* cylinder low */
4329 pCtl->aIfs[0].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4330 pCtl->aIfs[1].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4331 pCtl->aIfs[0].uATARegLCylHOB = pCtl->aIfs[0].uATARegLCyl;
4332 pCtl->aIfs[1].uATARegLCylHOB = pCtl->aIfs[1].uATARegLCyl;
4333 pCtl->aIfs[0].uATARegLCyl = val;
4334 pCtl->aIfs[1].uATARegLCyl = val;
4335 break;
4336 case 5: /* cylinder high */
4337 pCtl->aIfs[0].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4338 pCtl->aIfs[1].uATARegDevCtl &= ~ATA_DEVCTL_HOB;
4339 pCtl->aIfs[0].uATARegHCylHOB = pCtl->aIfs[0].uATARegHCyl;
4340 pCtl->aIfs[1].uATARegHCylHOB = pCtl->aIfs[1].uATARegHCyl;
4341 pCtl->aIfs[0].uATARegHCyl = val;
4342 pCtl->aIfs[1].uATARegHCyl = val;
4343 break;
4344 case 6: /* drive/head */
4345 pCtl->aIfs[0].uATARegSelect = (val & ~0x10) | 0xa0;
4346 pCtl->aIfs[1].uATARegSelect = (val | 0x10) | 0xa0;
4347 if (((val >> 4) & 1) != pCtl->iSelectedIf)
4348 {
4349 PPDMDEVINS pDevIns = CONTROLLER_2_DEVINS(pCtl);
4350
4351 /* select another drive */
4352 pCtl->iSelectedIf = (val >> 4) & 1;
4353 /* The IRQ line is multiplexed between the two drives, so
4354 * update the state when switching to another drive. Only need
4355 * to update interrupt line if it is enabled and there is a
4356 * state change. */
4357 if ( !(pCtl->aIfs[pCtl->iSelectedIf].uATARegDevCtl & ATA_DEVCTL_DISABLE_IRQ)
4358 && ( pCtl->aIfs[pCtl->iSelectedIf].fIrqPending
4359 != pCtl->aIfs[pCtl->iSelectedIf ^ 1].fIrqPending))
4360 {
4361 if (pCtl->aIfs[pCtl->iSelectedIf].fIrqPending)
4362 {
4363 Log2(("%s: LUN#%d asserting IRQ (drive select change)\n", __FUNCTION__, pCtl->aIfs[pCtl->iSelectedIf].iLUN));
4364 /* The BMDMA unit unconditionally sets BM_STATUS_INT if
4365 * the interrupt line is asserted. It monitors the line
4366 * for a rising edge. */
4367 pCtl->BmDma.u8Status |= BM_STATUS_INT;
4368 if (pCtl->irq == 16)
4369 PDMDevHlpPCISetIrq(pDevIns, 0, 1);
4370 else
4371 PDMDevHlpISASetIrq(pDevIns, pCtl->irq, 1);
4372 }
4373 else
4374 {
4375 Log2(("%s: LUN#%d deasserting IRQ (drive select change)\n", __FUNCTION__, pCtl->aIfs[pCtl->iSelectedIf].iLUN));
4376 if (pCtl->irq == 16)
4377 PDMDevHlpPCISetIrq(pDevIns, 0, 0);
4378 else
4379 PDMDevHlpISASetIrq(pDevIns, pCtl->irq, 0);
4380 }
4381 }
4382 }
4383 break;
4384 default:
4385 case 7: /* command */
4386 /* ignore commands to non-existent device */
4387 if (pCtl->iSelectedIf && !pCtl->aIfs[pCtl->iSelectedIf].pDrvBlock)
4388 break;
4389#ifndef IN_RING3
4390 /* Don't do anything complicated in GC */
4391 return VINF_IOM_R3_IOPORT_WRITE;
4392#else /* IN_RING3 */
4393 ataR3ParseCmd(&pCtl->aIfs[pCtl->iSelectedIf], val);
4394#endif /* !IN_RING3 */
4395 }
4396 return VINF_SUCCESS;
4397}
4398
4399
4400static int ataIOPortReadU8(PATACONTROLLER pCtl, uint32_t addr, uint32_t *pu32)
4401{
4402 ATADevState *s = &pCtl->aIfs[pCtl->iSelectedIf];
4403 uint32_t val;
4404 bool fHOB;
4405
4406 /* Check if the guest is reading from a non-existent device. */
4407 if (!s->pDrvBlock)
4408 {
4409 if (pCtl->iSelectedIf) /* Device 1 selected, Device 0 responding for it. */
4410 {
4411 if (!pCtl->aIfs[0].pDrvBlock) /** @todo this case should never get here! */
4412 {
4413 Log2(("%s: addr=%#x: no device on channel\n", __FUNCTION__, addr));
4414 return VERR_IOM_IOPORT_UNUSED;
4415 }
4416 if (((addr & 7) != 1) && pCtl->aIfs[0].fATAPI) {
4417 Log2(("%s: addr=%#x, val=0: LUN#%d not attached/LUN#%d ATAPI\n", __FUNCTION__, addr,
4418 s->iLUN, pCtl->aIfs[0].iLUN));
4419 *pu32 = 0;
4420 return VINF_SUCCESS;
4421 }
4422 /* Else handle normally. */
4423 }
4424 else /* Device 0 selected (but not present). */
4425 {
4426 Log2(("%s: addr=%#x: LUN#%d not attached\n", __FUNCTION__, addr, s->iLUN));
4427 return VERR_IOM_IOPORT_UNUSED;
4428 }
4429 }
4430 fHOB = !!(s->uATARegDevCtl & (1 << 7));
4431 switch (addr & 7)
4432 {
4433 case 0: /* data register */
4434 val = 0xff;
4435 break;
4436 case 1: /* error register */
4437 /* The ATA specification is very terse when it comes to specifying
4438 * the precise effects of reading back the error/feature register.
4439 * The error register (read-only) shares the register number with
4440 * the feature register (write-only), so it seems that it's not
4441 * necessary to support the usual HOB readback here. */
4442 if (!s->pDrvBlock)
4443 val = 0;
4444 else
4445 val = s->uATARegError;
4446 break;
4447 case 2: /* sector count */
4448 if (fHOB)
4449 val = s->uATARegNSectorHOB;
4450 else
4451 val = s->uATARegNSector;
4452 break;
4453 case 3: /* sector number */
4454 if (fHOB)
4455 val = s->uATARegSectorHOB;
4456 else
4457 val = s->uATARegSector;
4458 break;
4459 case 4: /* cylinder low */
4460 if (fHOB)
4461 val = s->uATARegLCylHOB;
4462 else
4463 val = s->uATARegLCyl;
4464 break;
4465 case 5: /* cylinder high */
4466 if (fHOB)
4467 val = s->uATARegHCylHOB;
4468 else
4469 val = s->uATARegHCyl;
4470 break;
4471 case 6: /* drive/head */
4472 /* This register must always work as long as there is at least
4473 * one drive attached to the controller. It is common between
4474 * both drives anyway (completely identical content). */
4475 if (!pCtl->aIfs[0].pDrvBlock && !pCtl->aIfs[1].pDrvBlock)
4476 val = 0;
4477 else
4478 val = s->uATARegSelect;
4479 break;
4480 default:
4481 case 7: /* primary status */
4482 {
4483 /* Counter for number of busy status seen in GC in a row. */
4484 static unsigned cBusy = 0;
4485
4486 if (!s->pDrvBlock)
4487 val = 0;
4488 else
4489 val = s->uATARegStatus;
4490
4491 /* Give the async I/O thread an opportunity to make progress,
4492 * don't let it starve by guests polling frequently. EMT has a
4493 * lower priority than the async I/O thread, but sometimes the
4494 * host OS doesn't care. With some guests we are only allowed to
4495 * be busy for about 5 milliseconds in some situations. Note that
4496 * this is no guarantee for any other VBox thread getting
4497 * scheduled, so this just lowers the CPU load a bit when drives
4498 * are busy. It cannot help with timing problems. */
4499 if (val & ATA_STAT_BUSY)
4500 {
4501#ifdef IN_RING3
4502 cBusy = 0;
4503 PDMCritSectLeave(&pCtl->lock);
4504
4505#ifndef RT_OS_WINDOWS
4506 /*
4507 * The thread might be stuck in an I/O operation
4508 * due to a high I/O load on the host. (see @bugref{3301})
4509 * To perform the reset successfully
4510 * we interrupt the operation by sending a signal to the thread
4511 * if the thread didn't responded in 10ms.
4512 * This works only on POSIX hosts (Windows has a CancelSynchronousIo function which
4513 * does the same but it was introduced with Vista) but so far
4514 * this hang was only observed on Linux and Mac OS X.
4515 *
4516 * This is a workaround and needs to be solved properly.
4517 */
4518 if (pCtl->fReset)
4519 {
4520 uint64_t u64ResetTimeStop = RTTimeMilliTS();
4521
4522 if ((u64ResetTimeStop - pCtl->u64ResetTime) >= 10)
4523 {
4524 LogRel(("PIIX3 ATA LUN#%d: Async I/O thread probably stuck in operation, interrupting\n", s->iLUN));
4525 pCtl->u64ResetTime = u64ResetTimeStop;
4526 RTThreadPoke(pCtl->AsyncIOThread);
4527 }
4528 }
4529#endif
4530
4531 RTThreadYield();
4532
4533 {
4534 STAM_PROFILE_START(&pCtl->StatLockWait, a);
4535 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
4536 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
4537 }
4538
4539 val = s->uATARegStatus;
4540#else /* !IN_RING3 */
4541 /* Cannot yield CPU in raw-mode and ring-0 context. And switching
4542 * to host context for each and every busy status is too costly,
4543 * especially on SMP systems where we don't gain much by
4544 * yielding the CPU to someone else. */
4545 if (++cBusy >= 20)
4546 {
4547 cBusy = 0;
4548 return VINF_IOM_R3_IOPORT_READ;
4549 }
4550#endif /* !IN_RING3 */
4551 }
4552 else
4553 cBusy = 0;
4554 ataUnsetIRQ(s);
4555 break;
4556 }
4557 }
4558 Log2(("%s: LUN#%d addr=%#x val=%#04x\n", __FUNCTION__, s->iLUN, addr, val));
4559 *pu32 = val;
4560 return VINF_SUCCESS;
4561}
4562
4563
4564static uint32_t ataStatusRead(PATACONTROLLER pCtl, uint32_t addr)
4565{
4566 ATADevState *s = &pCtl->aIfs[pCtl->iSelectedIf];
4567 uint32_t val;
4568
4569 //@todo: The handler should not be even registered if there
4570 // is no device on an IDE channel.
4571 if (!pCtl->aIfs[0].pDrvBlock && !pCtl->aIfs[1].pDrvBlock)
4572 val = 0xff;
4573 else if (pCtl->iSelectedIf == 1 && !s->pDrvBlock)
4574 val = 0; /* Device 1 selected, Device 0 responding for it. */
4575 else
4576 val = s->uATARegStatus;
4577 Log2(("%s: addr=%#x val=%#04x\n", __FUNCTION__, addr, val));
4578 return val;
4579}
4580
4581static int ataControlWrite(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
4582{
4583#ifndef IN_RING3
4584 if ((val ^ pCtl->aIfs[0].uATARegDevCtl) & ATA_DEVCTL_RESET)
4585 return VINF_IOM_R3_IOPORT_WRITE; /* The RESET stuff is too complicated for RC+R0. */
4586#endif /* !IN_RING3 */
4587
4588 Log2(("%s: addr=%#x val=%#04x\n", __FUNCTION__, addr, val));
4589 /* RESET is common for both drives attached to a controller. */
4590 if ( !(pCtl->aIfs[0].uATARegDevCtl & ATA_DEVCTL_RESET)
4591 && (val & ATA_DEVCTL_RESET))
4592 {
4593#ifdef IN_RING3
4594 /* Software RESET low to high */
4595 int32_t uCmdWait0 = -1;
4596 int32_t uCmdWait1 = -1;
4597 uint64_t uNow = RTTimeNanoTS();
4598 if (pCtl->aIfs[0].u64CmdTS)
4599 uCmdWait0 = (uNow - pCtl->aIfs[0].u64CmdTS) / 1000;
4600 if (pCtl->aIfs[1].u64CmdTS)
4601 uCmdWait1 = (uNow - pCtl->aIfs[1].u64CmdTS) / 1000;
4602 LogRel(("PIIX3 ATA: Ctl#%d: RESET, DevSel=%d AIOIf=%d CmdIf0=%#04x (%d usec ago) CmdIf1=%#04x (%d usec ago)\n",
4603 ATACONTROLLER_IDX(pCtl), pCtl->iSelectedIf, pCtl->iAIOIf,
4604 pCtl->aIfs[0].uATARegCommand, uCmdWait0,
4605 pCtl->aIfs[1].uATARegCommand, uCmdWait1));
4606 pCtl->fReset = true;
4607 /* Everything must be done after the reset flag is set, otherwise
4608 * there are unavoidable races with the currently executing request
4609 * (which might just finish in the mean time). */
4610 pCtl->fChainedTransfer = false;
4611 for (uint32_t i = 0; i < RT_ELEMENTS(pCtl->aIfs); i++)
4612 {
4613 ataR3ResetDevice(&pCtl->aIfs[i]);
4614 /* The following cannot be done using ataSetStatusValue() since the
4615 * reset flag is already set, which suppresses all status changes. */
4616 pCtl->aIfs[i].uATARegStatus = ATA_STAT_BUSY | ATA_STAT_SEEK;
4617 Log2(("%s: LUN#%d status %#04x\n", __FUNCTION__, pCtl->aIfs[i].iLUN, pCtl->aIfs[i].uATARegStatus));
4618 pCtl->aIfs[i].uATARegError = 0x01;
4619 }
4620 ataR3AsyncIOClearRequests(pCtl);
4621 Log2(("%s: Ctl#%d: message to async I/O thread, resetA\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
4622 if (val & ATA_DEVCTL_HOB)
4623 {
4624 val &= ~ATA_DEVCTL_HOB;
4625 Log2(("%s: ignored setting HOB\n", __FUNCTION__));
4626 }
4627
4628 /* Save the timestamp we started the reset. */
4629 pCtl->u64ResetTime = RTTimeMilliTS();
4630
4631 /* Issue the reset request now. */
4632 ataHCAsyncIOPutRequest(pCtl, &g_ataResetARequest);
4633#else /* !IN_RING3 */
4634 AssertMsgFailed(("RESET handling is too complicated for GC\n"));
4635#endif /* IN_RING3 */
4636 }
4637 else if ( (pCtl->aIfs[0].uATARegDevCtl & ATA_DEVCTL_RESET)
4638 && !(val & ATA_DEVCTL_RESET))
4639 {
4640#ifdef IN_RING3
4641 /* Software RESET high to low */
4642 Log(("%s: deasserting RESET\n", __FUNCTION__));
4643 Log2(("%s: Ctl#%d: message to async I/O thread, resetC\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
4644 if (val & ATA_DEVCTL_HOB)
4645 {
4646 val &= ~ATA_DEVCTL_HOB;
4647 Log2(("%s: ignored setting HOB\n", __FUNCTION__));
4648 }
4649 ataHCAsyncIOPutRequest(pCtl, &g_ataResetCRequest);
4650#else /* !IN_RING3 */
4651 AssertMsgFailed(("RESET handling is too complicated for GC\n"));
4652#endif /* IN_RING3 */
4653 }
4654
4655 /* Change of interrupt disable flag. Update interrupt line if interrupt
4656 * is pending on the current interface. */
4657 if ( ((val ^ pCtl->aIfs[0].uATARegDevCtl) & ATA_DEVCTL_DISABLE_IRQ)
4658 && pCtl->aIfs[pCtl->iSelectedIf].fIrqPending)
4659 {
4660 if (!(val & ATA_DEVCTL_DISABLE_IRQ))
4661 {
4662 Log2(("%s: LUN#%d asserting IRQ (interrupt disable change)\n", __FUNCTION__, pCtl->aIfs[pCtl->iSelectedIf].iLUN));
4663 /* The BMDMA unit unconditionally sets BM_STATUS_INT if the
4664 * interrupt line is asserted. It monitors the line for a rising
4665 * edge. */
4666 pCtl->BmDma.u8Status |= BM_STATUS_INT;
4667 if (pCtl->irq == 16)
4668 PDMDevHlpPCISetIrq(CONTROLLER_2_DEVINS(pCtl), 0, 1);
4669 else
4670 PDMDevHlpISASetIrq(CONTROLLER_2_DEVINS(pCtl), pCtl->irq, 1);
4671 }
4672 else
4673 {
4674 Log2(("%s: LUN#%d deasserting IRQ (interrupt disable change)\n", __FUNCTION__, pCtl->aIfs[pCtl->iSelectedIf].iLUN));
4675 if (pCtl->irq == 16)
4676 PDMDevHlpPCISetIrq(CONTROLLER_2_DEVINS(pCtl), 0, 0);
4677 else
4678 PDMDevHlpISASetIrq(CONTROLLER_2_DEVINS(pCtl), pCtl->irq, 0);
4679 }
4680 }
4681
4682 if (val & ATA_DEVCTL_HOB)
4683 Log2(("%s: set HOB\n", __FUNCTION__));
4684
4685 pCtl->aIfs[0].uATARegDevCtl = val;
4686 pCtl->aIfs[1].uATARegDevCtl = val;
4687
4688 return VINF_SUCCESS;
4689}
4690
4691#if defined(IN_RING0) || defined(IN_RING3)
4692
4693static void ataHCPIOTransfer(PATACONTROLLER pCtl)
4694{
4695 ATADevState *s;
4696
4697 s = &pCtl->aIfs[pCtl->iAIOIf];
4698 Log3(("%s: if=%p\n", __FUNCTION__, s));
4699
4700 if (s->cbTotalTransfer && s->iIOBufferCur > s->iIOBufferEnd)
4701 {
4702# ifdef IN_RING3
4703 LogRel(("PIIX3 ATA: LUN#%d: %s data in the middle of a PIO transfer - VERY SLOW\n", s->iLUN, s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE ? "loading" : "storing"));
4704 /* Any guest OS that triggers this case has a pathetic ATA driver.
4705 * In a real system it would block the CPU via IORDY, here we do it
4706 * very similarly by not continuing with the current instruction
4707 * until the transfer to/from the storage medium is completed. */
4708 if (s->iSourceSink != ATAFN_SS_NULL)
4709 {
4710 bool fRedo;
4711 uint8_t status = s->uATARegStatus;
4712 ataSetStatusValue(s, ATA_STAT_BUSY);
4713 Log2(("%s: calling source/sink function\n", __FUNCTION__));
4714 fRedo = g_apfnSourceSinkFuncs[s->iSourceSink](s);
4715 pCtl->fRedo = fRedo;
4716 if (RT_UNLIKELY(fRedo))
4717 return;
4718 ataSetStatusValue(s, status);
4719 s->iIOBufferCur = 0;
4720 s->iIOBufferEnd = s->cbElementaryTransfer;
4721 }
4722# else
4723 AssertReleaseFailed();
4724# endif
4725 }
4726 if (s->cbTotalTransfer)
4727 {
4728 if (s->fATAPITransfer)
4729 ataHCPIOTransferLimitATAPI(s);
4730
4731 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE && s->cbElementaryTransfer > s->cbTotalTransfer)
4732 s->cbElementaryTransfer = s->cbTotalTransfer;
4733
4734 Log2(("%s: %s tx_size=%d elem_tx_size=%d index=%d end=%d\n",
4735 __FUNCTION__, s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE ? "T2I" : "I2T",
4736 s->cbTotalTransfer, s->cbElementaryTransfer,
4737 s->iIOBufferCur, s->iIOBufferEnd));
4738 ataHCPIOTransferStart(s, s->iIOBufferCur, s->cbElementaryTransfer);
4739 s->cbTotalTransfer -= s->cbElementaryTransfer;
4740 s->iIOBufferCur += s->cbElementaryTransfer;
4741
4742 if (s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE && s->cbElementaryTransfer > s->cbTotalTransfer)
4743 s->cbElementaryTransfer = s->cbTotalTransfer;
4744 }
4745 else
4746 ataHCPIOTransferStop(s);
4747}
4748
4749
4750DECLINLINE(void) ataHCPIOTransferFinish(PATACONTROLLER pCtl, ATADevState *s)
4751{
4752 /* Do not interfere with RESET processing if the PIO transfer finishes
4753 * while the RESET line is asserted. */
4754 if (pCtl->fReset)
4755 {
4756 Log2(("%s: Ctl#%d: suppressed continuing PIO transfer as RESET is active\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
4757 return;
4758 }
4759
4760 if ( s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE
4761 || ( s->iSourceSink != ATAFN_SS_NULL
4762 && s->iIOBufferCur >= s->iIOBufferEnd))
4763 {
4764 /* Need to continue the transfer in the async I/O thread. This is
4765 * the case for write operations or generally for not yet finished
4766 * transfers (some data might need to be read). */
4767 ataUnsetStatus(s, ATA_STAT_READY | ATA_STAT_DRQ);
4768 ataSetStatus(s, ATA_STAT_BUSY);
4769
4770 Log2(("%s: Ctl#%d: message to async I/O thread, continuing PIO transfer\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
4771 ataHCAsyncIOPutRequest(pCtl, &g_ataPIORequest);
4772 }
4773 else
4774 {
4775 /* Either everything finished (though some data might still be pending)
4776 * or some data is pending before the next read is due. */
4777
4778 /* Continue a previously started transfer. */
4779 ataUnsetStatus(s, ATA_STAT_DRQ);
4780 ataSetStatus(s, ATA_STAT_READY);
4781
4782 if (s->cbTotalTransfer)
4783 {
4784 /* There is more to transfer, happens usually for large ATAPI
4785 * reads - the protocol limits the chunk size to 65534 bytes. */
4786 ataHCPIOTransfer(pCtl);
4787 ataHCSetIRQ(s);
4788 }
4789 else
4790 {
4791 Log2(("%s: Ctl#%d: skipping message to async I/O thread, ending PIO transfer\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
4792 /* Finish PIO transfer. */
4793 ataHCPIOTransfer(pCtl);
4794 Assert(!pCtl->fRedo);
4795 }
4796 }
4797}
4798
4799#endif /* IN_RING0 || IN_RING3 */
4800
4801/**
4802 * Fallback for ataCopyPioData124 that handles unaligned and out of bounds cases.
4803 *
4804 * @param pIf The device interface to work with.
4805 * @param pbDst The destination buffer.
4806 * @param pbSrc The source buffer.
4807 * @param cbCopy The number of bytes to copy, either 1, 2 or 4 bytes.
4808 */
4809DECL_NO_INLINE(static, void) ataCopyPioData124Slow(ATADevState *pIf, uint8_t *pbDst, const uint8_t *pbSrc, uint32_t cbCopy)
4810{
4811 uint32_t const offStart = pIf->iIOBufferPIODataStart;
4812 uint32_t const offNext = offStart + cbCopy;
4813
4814 if (offStart + cbCopy > pIf->cbIOBuffer)
4815 {
4816 Log(("%s: cbCopy=%#x offStart=%#x cbIOBuffer=%#x offNext=%#x (iIOBufferPIODataEnd=%#x)\n",
4817 __FUNCTION__, cbCopy, offStart, pIf->cbIOBuffer, offNext, pIf->iIOBufferPIODataEnd));
4818 if (offStart < pIf->cbIOBuffer)
4819 cbCopy = pIf->cbIOBuffer - offStart;
4820 else
4821 cbCopy = 0;
4822 }
4823
4824 switch (cbCopy)
4825 {
4826 case 4: pbDst[3] = pbSrc[3]; /* fall thru */
4827 case 3: pbDst[2] = pbSrc[2]; /* fall thru */
4828 case 2: pbDst[1] = pbSrc[1]; /* fall thru */
4829 case 1: pbDst[0] = pbSrc[0]; /* fall thru */
4830 case 0: break;
4831 default: AssertFailed(); /* impossible */
4832 }
4833
4834 pIf->iIOBufferPIODataStart = offNext;
4835
4836}
4837
4838
4839/**
4840 * Work for ataDataWrite & ataDataRead that copies data without using memcpy.
4841 *
4842 * This also updates pIf->iIOBufferPIODataStart.
4843 *
4844 * The two buffers are either stack (32-bit aligned) or somewhere within
4845 * pIf->pbIOBuffer.
4846 *
4847 * @param pIf The device interface to work with.
4848 * @param pbDst The destination buffer.
4849 * @param pbSrc The source buffer.
4850 * @param cbCopy The number of bytes to copy, either 1, 2 or 4 bytes.
4851 */
4852DECLINLINE(void) ataCopyPioData124(ATADevState *pIf, uint8_t *pbDst, const uint8_t *pbSrc, uint32_t cbCopy)
4853{
4854 /*
4855 * Quick bounds checking can be done by checking that the pbIOBuffer offset
4856 * (iIOBufferPIODataStart) is aligned at the transfer size (which is ASSUMED
4857 * to be 1, 2 or 4). However, since we're paranoid and don't currently
4858 * trust iIOBufferPIODataEnd to be within bounds, we current check against the
4859 * IO buffer size too.
4860 */
4861 Assert(cbCopy == 1 || cbCopy == 2 || cbCopy == 4);
4862 uint32_t const offStart = pIf->iIOBufferPIODataStart;
4863 if (RT_LIKELY( !(offStart & (cbCopy - 1))
4864 && offStart + cbCopy <= pIf->cbIOBuffer))
4865 {
4866 switch (cbCopy)
4867 {
4868 case 4: *(uint32_t *)pbDst = *(uint32_t const *)pbSrc; break;
4869 case 2: *(uint16_t *)pbDst = *(uint16_t const *)pbSrc; break;
4870 case 1: *pbDst = *pbSrc; break;
4871 }
4872 pIf->iIOBufferPIODataStart = offStart + cbCopy;
4873 }
4874 else
4875 ataCopyPioData124Slow(pIf, pbDst, pbSrc, cbCopy);
4876}
4877
4878
4879/**
4880 * Port I/O Handler for primary port range OUT operations.
4881 * @see FNIOMIOPORTOUT for details.
4882 */
4883PDMBOTHCBDECL(int) ataIOPortWrite1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
4884{
4885 uint32_t i = (uint32_t)(uintptr_t)pvUser;
4886 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
4887 PATACONTROLLER pCtl = &pThis->aCts[i];
4888
4889 Assert(i < 2);
4890 Assert(Port == pCtl->IOPortBase1);
4891 Assert(cb == 2 || cb == 4); /* Writes to the data port may be 16-bit or 32-bit. */
4892
4893 int rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_WRITE);
4894 if (rc == VINF_SUCCESS)
4895 {
4896 ATADevState *s = &pCtl->aIfs[pCtl->iSelectedIf];
4897
4898 if (s->iIOBufferPIODataStart < s->iIOBufferPIODataEnd)
4899 {
4900 Assert(s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE);
4901 uint8_t *pbDst = s->CTX_SUFF(pbIOBuffer) + s->iIOBufferPIODataStart;
4902 uint8_t const *pbSrc = (uint8_t const *)&u32;
4903
4904#ifdef IN_RC
4905 /* Raw-mode: The ataHCPIOTransfer following the last transfer unit
4906 requires I/O thread signalling, we must go to ring-3 for that. */
4907 if (s->iIOBufferPIODataStart + cb < s->iIOBufferPIODataEnd)
4908 ataCopyPioData124(s, pbDst, pbSrc, cb);
4909 else
4910 rc = VINF_IOM_R3_IOPORT_WRITE;
4911
4912#elif defined(IN_RING0)
4913 /* Ring-0: We can do I/O thread signalling here, however for paranoid reasons
4914 triggered by a special case in ataHCPIOTransferFinish, we take extra care here. */
4915 if (s->iIOBufferPIODataStart + cb < s->iIOBufferPIODataEnd)
4916 ataCopyPioData124(s, pbDst, pbSrc, cb);
4917 else if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE) /* paranoia */
4918 {
4919 ataCopyPioData124(s, pbDst, pbSrc, cb);
4920 ataHCPIOTransferFinish(pCtl, s);
4921 }
4922 else
4923 {
4924 Log(("%s: Unexpected\n",__FUNCTION__));
4925 rc = VINF_IOM_R3_IOPORT_WRITE;
4926 }
4927
4928#else /* IN_RING 3*/
4929 ataCopyPioData124(s, pbDst, pbSrc, cb);
4930 if (s->iIOBufferPIODataStart >= s->iIOBufferPIODataEnd)
4931 ataHCPIOTransferFinish(pCtl, s);
4932#endif /* IN_RING 3*/
4933 }
4934 else
4935 Log2(("%s: DUMMY data\n", __FUNCTION__));
4936
4937 Log3(("%s: addr=%#x val=%.*Rhxs rc=%d\n", __FUNCTION__, Port, cb, &u32, rc));
4938 PDMCritSectLeave(&pCtl->lock);
4939 }
4940 else
4941 Log3(("%s: addr=%#x -> %d\n", __FUNCTION__, Port, rc));
4942 return rc;
4943}
4944
4945
4946/**
4947 * Port I/O Handler for primary port range IN operations.
4948 * @see FNIOMIOPORTIN for details.
4949 */
4950PDMBOTHCBDECL(int) ataIOPortRead1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
4951{
4952 uint32_t i = (uint32_t)(uintptr_t)pvUser;
4953 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
4954 PATACONTROLLER pCtl = &pThis->aCts[i];
4955
4956 Assert(i < 2);
4957 Assert(Port == pCtl->IOPortBase1);
4958
4959 /* Reads from the data register may be 16-bit or 32-bit. Byte accesses are
4960 upgraded to word. */
4961 Assert(cb == 1 || cb == 2 || cb == 4);
4962 uint32_t cbActual = cb != 1 ? cb : 2;
4963 *pu32 = 0;
4964
4965 int rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_READ);
4966 if (rc == VINF_SUCCESS)
4967 {
4968 ATADevState *s = &pCtl->aIfs[pCtl->iSelectedIf];
4969
4970 if (s->iIOBufferPIODataStart < s->iIOBufferPIODataEnd)
4971 {
4972 Assert(s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE);
4973 uint8_t const *pbSrc = s->CTX_SUFF(pbIOBuffer) + s->iIOBufferPIODataStart;
4974 uint8_t *pbDst = (uint8_t *)pu32;
4975
4976#ifdef IN_RC
4977 /* All but the last transfer unit is simple enough for RC, but
4978 * sending a request to the async IO thread is too complicated. */
4979 if (s->iIOBufferPIODataStart + cbActual < s->iIOBufferPIODataEnd)
4980 ataCopyPioData124(s, pbDst, pbSrc, cbActual);
4981 else
4982 rc = VINF_IOM_R3_IOPORT_READ;
4983
4984#elif defined(IN_RING0)
4985 /* Ring-0: We can do I/O thread signalling here. However there is one
4986 case in ataHCPIOTransfer that does a LogRel and would (but not from
4987 here) call directly into the driver code. We detect that odd case
4988 here cand return to ring-3 to handle it. */
4989 if (s->iIOBufferPIODataStart + cbActual < s->iIOBufferPIODataEnd)
4990 ataCopyPioData124(s, pbDst, pbSrc, cbActual);
4991 else if ( s->cbTotalTransfer == 0
4992 || s->iSourceSink != ATAFN_SS_NULL
4993 || s->iIOBufferCur <= s->iIOBufferEnd)
4994 {
4995 ataCopyPioData124(s, pbDst, pbSrc, cbActual);
4996 ataHCPIOTransferFinish(pCtl, s);
4997 }
4998 else
4999 {
5000 Log(("%s: Unexpected\n",__FUNCTION__));
5001 rc = VINF_IOM_R3_IOPORT_READ;
5002 }
5003
5004#else /* IN_RING3 */
5005 ataCopyPioData124(s, pbDst, pbSrc, cbActual);
5006 if (s->iIOBufferPIODataStart >= s->iIOBufferPIODataEnd)
5007 ataHCPIOTransferFinish(pCtl, s);
5008#endif /* IN_RING3 */
5009
5010 /* Just to be on the safe side (caller takes care of this, really). */
5011 if (cb == 1)
5012 *pu32 &= 0xff;
5013 }
5014 else
5015 {
5016 Log2(("%s: DUMMY data\n", __FUNCTION__));
5017 memset(pu32, 0xff, cb);
5018 }
5019 Log3(("%s: addr=%#x val=%.*Rhxs rc=%d\n", __FUNCTION__, Port, cb, pu32, rc));
5020
5021 PDMCritSectLeave(&pCtl->lock);
5022 }
5023 else
5024 Log3(("%s: addr=%#x -> %d\n", __FUNCTION__, Port, rc));
5025
5026 return rc;
5027}
5028
5029
5030/**
5031 * Port I/O Handler for primary port range IN string operations.
5032 * @see FNIOMIOPORTINSTRING for details.
5033 */
5034PDMBOTHCBDECL(int) ataIOPortReadStr1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint8_t *pbDst,
5035 uint32_t *pcTransfers, unsigned cb)
5036{
5037 uint32_t i = (uint32_t)(uintptr_t)pvUser;
5038 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
5039 PATACONTROLLER pCtl = &pThis->aCts[i];
5040
5041 Assert(i < 2);
5042 Assert(Port == pCtl->IOPortBase1);
5043 Assert(*pcTransfers > 0);
5044
5045 int rc;
5046 if (cb == 2 || cb == 4)
5047 {
5048 rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_READ);
5049 if (rc == VINF_SUCCESS)
5050 {
5051 ATADevState *s = &pCtl->aIfs[pCtl->iSelectedIf];
5052
5053 uint32_t const offStart = s->iIOBufferPIODataStart;
5054 if (offStart < s->iIOBufferPIODataEnd)
5055 {
5056 /*
5057 * Figure how much we can copy. Usually it's the same as the request.
5058 * The last transfer unit cannot be handled in RC, as it involves
5059 * thread communication. In R0 we let the non-string callback handle it,
5060 * and ditto for overflows/dummy data.
5061 */
5062 uint32_t cAvailable = (s->iIOBufferPIODataEnd - offStart) / cb;
5063#ifndef IN_RING3
5064 if (cAvailable > 0)
5065 cAvailable--;
5066#endif
5067 uint32_t const cRequested = *pcTransfers;
5068 if (cAvailable > cRequested)
5069 cAvailable = cRequested;
5070 uint32_t const cbTransfer = cAvailable * cb;
5071 if ( offStart + cbTransfer <= s->cbIOBuffer
5072 && cbTransfer > 0)
5073 {
5074 /*
5075 * Do the transfer.
5076 */
5077 uint8_t const *pbSrc = s->CTX_SUFF(pbIOBuffer) + offStart;
5078 memcpy(pbDst, pbSrc, cbTransfer);
5079 Log3(("%s: addr=%#x cb=%#x cbTransfer=%#x val=%.*Rhxd\n",
5080 __FUNCTION__, Port, cb, cbTransfer, cbTransfer, pbSrc));
5081 s->iIOBufferPIODataStart = offStart + cbTransfer;
5082
5083#ifdef IN_RING3
5084 if (s->iIOBufferPIODataStart >= s->iIOBufferPIODataEnd)
5085 ataHCPIOTransferFinish(pCtl, s);
5086#endif
5087 *pcTransfers = cRequested - cAvailable;
5088 }
5089 else
5090 Log2(("ataIOPortReadStr1Data: DUMMY/Overflow!\n"));
5091 }
5092 else
5093 {
5094 /*
5095 * Dummy read (shouldn't happen) return 0xff like the non-string handler.
5096 */
5097 Log2(("ataIOPortReadStr1Data: DUMMY data (%#x bytes)\n", *pcTransfers * cb));
5098 memset(pbDst, 0xff, *pcTransfers * cb);
5099 *pcTransfers = 0;
5100 }
5101
5102 PDMCritSectLeave(&pCtl->lock);
5103 }
5104 }
5105 /*
5106 * Let the non-string I/O callback handle 1 byte reads.
5107 */
5108 else
5109 {
5110 Log2(("ataIOPortReadStr1Data: 1 byte read (%#x transfers)\n", *pcTransfers));
5111 AssertFailed();
5112 rc = VINF_SUCCESS;
5113 }
5114 return rc;
5115}
5116
5117
5118/**
5119 * Port I/O Handler for primary port range OUT string operations.
5120 * @see FNIOMIOPORTOUTSTRING for details.
5121 */
5122PDMBOTHCBDECL(int) ataIOPortWriteStr1Data(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint8_t const *pbSrc,
5123 uint32_t *pcTransfers, unsigned cb)
5124{
5125 uint32_t i = (uint32_t)(uintptr_t)pvUser;
5126 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
5127 PATACONTROLLER pCtl = &pThis->aCts[i];
5128
5129 Assert(i < 2);
5130 Assert(Port == pCtl->IOPortBase1);
5131 Assert(*pcTransfers > 0);
5132
5133 int rc;
5134 if (cb == 2 || cb == 4)
5135 {
5136 rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_WRITE);
5137 if (rc == VINF_SUCCESS)
5138 {
5139 ATADevState *s = &pCtl->aIfs[pCtl->iSelectedIf];
5140
5141 uint32_t const offStart = s->iIOBufferPIODataStart;
5142 if (offStart < s->iIOBufferPIODataEnd)
5143 {
5144 /*
5145 * Figure how much we can copy. Usually it's the same as the request.
5146 * The last transfer unit cannot be handled in RC, as it involves
5147 * thread communication. In R0 we let the non-string callback handle it,
5148 * and ditto for overflows/dummy data.
5149 */
5150 uint32_t cAvailable = (s->iIOBufferPIODataEnd - offStart) / cb;
5151#ifndef IN_RING3
5152 if (cAvailable)
5153 cAvailable--;
5154#endif
5155 uint32_t const cRequested = *pcTransfers;
5156 if (cAvailable > cRequested)
5157 cAvailable = cRequested;
5158 uint32_t const cbTransfer = cAvailable * cb;
5159 if ( offStart + cbTransfer <= s->cbIOBuffer
5160 && cbTransfer)
5161 {
5162 /*
5163 * Do the transfer.
5164 */
5165 void *pvDst = s->CTX_SUFF(pbIOBuffer) + offStart;
5166 memcpy(pvDst, pbSrc, cbTransfer);
5167 Log3(("%s: addr=%#x val=%.*Rhxs\n", __FUNCTION__, Port, cbTransfer, pvDst));
5168 s->iIOBufferPIODataStart = offStart + cbTransfer;
5169
5170#ifdef IN_RING3
5171 if (s->iIOBufferPIODataStart >= s->iIOBufferPIODataEnd)
5172 ataHCPIOTransferFinish(pCtl, s);
5173#endif
5174 *pcTransfers = cRequested - cAvailable;
5175 }
5176 else
5177 Log2(("ataIOPortWriteStr1Data: DUMMY/Overflow!\n"));
5178 }
5179 else
5180 {
5181 Log2(("ataIOPortWriteStr1Data: DUMMY data (%#x bytes)\n", *pcTransfers * cb));
5182 *pcTransfers = 0;
5183 }
5184
5185 PDMCritSectLeave(&pCtl->lock);
5186 }
5187 }
5188 /*
5189 * Let the non-string I/O callback handle 1 byte reads.
5190 */
5191 else
5192 {
5193 Log2(("ataIOPortWriteStr1Data: 1 byte write (%#x transfers)\n", *pcTransfers));
5194 AssertFailed();
5195 rc = VINF_SUCCESS;
5196 }
5197
5198 return rc;
5199}
5200
5201
5202#ifdef IN_RING3
5203
5204static void ataR3DMATransferStop(ATADevState *s)
5205{
5206 s->cbTotalTransfer = 0;
5207 s->cbElementaryTransfer = 0;
5208 s->iBeginTransfer = ATAFN_BT_NULL;
5209 s->iSourceSink = ATAFN_SS_NULL;
5210}
5211
5212
5213/**
5214 * Perform the entire DMA transfer in one go (unless a source/sink operation
5215 * has to be redone or a RESET comes in between). Unlike the PIO counterpart
5216 * this function cannot handle empty transfers.
5217 *
5218 * @param pCtl Controller for which to perform the transfer.
5219 */
5220static void ataR3DMATransfer(PATACONTROLLER pCtl)
5221{
5222 PPDMDEVINS pDevIns = CONTROLLER_2_DEVINS(pCtl);
5223 ATADevState *s = &pCtl->aIfs[pCtl->iAIOIf];
5224 bool fRedo;
5225 RTGCPHYS32 pDesc;
5226 uint32_t cbTotalTransfer, cbElementaryTransfer;
5227 uint32_t iIOBufferCur, iIOBufferEnd;
5228 uint32_t dmalen;
5229 PDMBLOCKTXDIR uTxDir;
5230 bool fLastDesc = false;
5231
5232 Assert(sizeof(BMDMADesc) == 8);
5233
5234 fRedo = pCtl->fRedo;
5235 if (RT_LIKELY(!fRedo))
5236 Assert(s->cbTotalTransfer);
5237 uTxDir = (PDMBLOCKTXDIR)s->uTxDir;
5238 cbTotalTransfer = s->cbTotalTransfer;
5239 cbElementaryTransfer = s->cbElementaryTransfer;
5240 iIOBufferCur = s->iIOBufferCur;
5241 iIOBufferEnd = s->iIOBufferEnd;
5242
5243 /* The DMA loop is designed to hold the lock only when absolutely
5244 * necessary. This avoids long freezes should the guest access the
5245 * ATA registers etc. for some reason. */
5246 PDMCritSectLeave(&pCtl->lock);
5247
5248 Log2(("%s: %s tx_size=%d elem_tx_size=%d index=%d end=%d\n",
5249 __FUNCTION__, uTxDir == PDMBLOCKTXDIR_FROM_DEVICE ? "T2I" : "I2T",
5250 cbTotalTransfer, cbElementaryTransfer,
5251 iIOBufferCur, iIOBufferEnd));
5252 for (pDesc = pCtl->pFirstDMADesc; pDesc <= pCtl->pLastDMADesc; pDesc += sizeof(BMDMADesc))
5253 {
5254 BMDMADesc DMADesc;
5255 RTGCPHYS32 pBuffer;
5256 uint32_t cbBuffer;
5257
5258 if (RT_UNLIKELY(fRedo))
5259 {
5260 pBuffer = pCtl->pRedoDMABuffer;
5261 cbBuffer = pCtl->cbRedoDMABuffer;
5262 fLastDesc = pCtl->fRedoDMALastDesc;
5263 }
5264 else
5265 {
5266 PDMDevHlpPhysRead(pDevIns, pDesc, &DMADesc, sizeof(BMDMADesc));
5267 pBuffer = RT_LE2H_U32(DMADesc.pBuffer);
5268 cbBuffer = RT_LE2H_U32(DMADesc.cbBuffer);
5269 fLastDesc = !!(cbBuffer & 0x80000000);
5270 cbBuffer &= 0xfffe;
5271 if (cbBuffer == 0)
5272 cbBuffer = 0x10000;
5273 if (cbBuffer > cbTotalTransfer)
5274 cbBuffer = cbTotalTransfer;
5275 }
5276
5277 while (RT_UNLIKELY(fRedo) || (cbBuffer && cbTotalTransfer))
5278 {
5279 if (RT_LIKELY(!fRedo))
5280 {
5281 dmalen = RT_MIN(cbBuffer, iIOBufferEnd - iIOBufferCur);
5282 Log2(("%s: DMA desc %#010x: addr=%#010x size=%#010x orig_size=%#010x\n", __FUNCTION__,
5283 (int)pDesc, pBuffer, cbBuffer, RT_LE2H_U32(DMADesc.cbBuffer) & 0xfffe));
5284
5285 PCIATAState *pATAState = PDMINS_2_DATA(pDevIns, PCIATAState *);
5286 AssertPtr(pATAState);
5287 if (uTxDir == PDMBLOCKTXDIR_FROM_DEVICE)
5288 PDMDevHlpPCIPhysWrite(pDevIns, pBuffer, s->CTX_SUFF(pbIOBuffer) + iIOBufferCur, dmalen);
5289 else
5290 PDMDevHlpPCIPhysRead(pDevIns, pBuffer, s->CTX_SUFF(pbIOBuffer) + iIOBufferCur, dmalen);
5291
5292 iIOBufferCur += dmalen;
5293 cbTotalTransfer -= dmalen;
5294 cbBuffer -= dmalen;
5295 pBuffer += dmalen;
5296 }
5297 if ( iIOBufferCur == iIOBufferEnd
5298 && (uTxDir == PDMBLOCKTXDIR_TO_DEVICE || cbTotalTransfer))
5299 {
5300 if (uTxDir == PDMBLOCKTXDIR_FROM_DEVICE && cbElementaryTransfer > cbTotalTransfer)
5301 cbElementaryTransfer = cbTotalTransfer;
5302
5303 {
5304 STAM_PROFILE_START(&pCtl->StatLockWait, a);
5305 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
5306 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
5307 }
5308
5309 /* The RESET handler could have cleared the DMA transfer
5310 * state (since we didn't hold the lock until just now
5311 * the guest can continue in parallel). If so, the state
5312 * is already set up so the loop is exited immediately. */
5313 if (s->iSourceSink != ATAFN_SS_NULL)
5314 {
5315 s->iIOBufferCur = iIOBufferCur;
5316 s->iIOBufferEnd = iIOBufferEnd;
5317 s->cbElementaryTransfer = cbElementaryTransfer;
5318 s->cbTotalTransfer = cbTotalTransfer;
5319 Log2(("%s: calling source/sink function\n", __FUNCTION__));
5320 fRedo = g_apfnSourceSinkFuncs[s->iSourceSink](s);
5321 if (RT_UNLIKELY(fRedo))
5322 {
5323 pCtl->pFirstDMADesc = pDesc;
5324 pCtl->pRedoDMABuffer = pBuffer;
5325 pCtl->cbRedoDMABuffer = cbBuffer;
5326 pCtl->fRedoDMALastDesc = fLastDesc;
5327 }
5328 else
5329 {
5330 cbTotalTransfer = s->cbTotalTransfer;
5331 cbElementaryTransfer = s->cbElementaryTransfer;
5332
5333 if (uTxDir == PDMBLOCKTXDIR_TO_DEVICE && cbElementaryTransfer > cbTotalTransfer)
5334 cbElementaryTransfer = cbTotalTransfer;
5335 iIOBufferCur = 0;
5336 iIOBufferEnd = cbElementaryTransfer;
5337 }
5338 pCtl->fRedo = fRedo;
5339 }
5340 else
5341 {
5342 /* This forces the loop to exit immediately. */
5343 pDesc = pCtl->pLastDMADesc + 1;
5344 }
5345
5346 PDMCritSectLeave(&pCtl->lock);
5347 if (RT_UNLIKELY(fRedo))
5348 break;
5349 }
5350 }
5351
5352 if (RT_UNLIKELY(fRedo))
5353 break;
5354
5355 /* end of transfer */
5356 if (!cbTotalTransfer || fLastDesc)
5357 break;
5358
5359 {
5360 STAM_PROFILE_START(&pCtl->StatLockWait, a);
5361 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
5362 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
5363 }
5364
5365 if (!(pCtl->BmDma.u8Cmd & BM_CMD_START) || pCtl->fReset)
5366 {
5367 LogRel(("PIIX3 ATA: Ctl#%d: ABORT DMA%s\n", ATACONTROLLER_IDX(pCtl), pCtl->fReset ? " due to RESET" : ""));
5368 if (!pCtl->fReset)
5369 ataR3DMATransferStop(s);
5370 /* This forces the loop to exit immediately. */
5371 pDesc = pCtl->pLastDMADesc + 1;
5372 }
5373
5374 PDMCritSectLeave(&pCtl->lock);
5375 }
5376
5377 {
5378 STAM_PROFILE_START(&pCtl->StatLockWait, a);
5379 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
5380 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
5381 }
5382
5383 if (RT_UNLIKELY(fRedo))
5384 return;
5385
5386 if (fLastDesc)
5387 pCtl->BmDma.u8Status &= ~BM_STATUS_DMAING;
5388 s->cbTotalTransfer = cbTotalTransfer;
5389 s->cbElementaryTransfer = cbElementaryTransfer;
5390 s->iIOBufferCur = iIOBufferCur;
5391 s->iIOBufferEnd = iIOBufferEnd;
5392}
5393
5394/**
5395 * Signal PDM that we're idle (if we actually are).
5396 *
5397 * @param pCtl The controller.
5398 */
5399static void ataR3AsyncSignalIdle(PATACONTROLLER pCtl)
5400{
5401 /*
5402 * Take the lock here and recheck the idle indicator to avoid
5403 * unnecessary work and racing ataR3WaitForAsyncIOIsIdle.
5404 */
5405 int rc = PDMCritSectEnter(&pCtl->AsyncIORequestLock, VINF_SUCCESS);
5406 AssertRC(rc);
5407
5408 if ( pCtl->fSignalIdle
5409 && ataR3AsyncIOIsIdle(pCtl, false /*fStrict*/))
5410 {
5411 PDMDevHlpAsyncNotificationCompleted(pCtl->pDevInsR3);
5412 RTThreadUserSignal(pCtl->AsyncIOThread); /* for ataR3Construct/ataR3ResetCommon. */
5413 }
5414
5415 rc = PDMCritSectLeave(&pCtl->AsyncIORequestLock);
5416 AssertRC(rc);
5417}
5418
5419/**
5420 * Async I/O thread for an interface.
5421 *
5422 * Once upon a time this was readable code with several loops and a different
5423 * semaphore for each purpose. But then came the "how can one save the state in
5424 * the middle of a PIO transfer" question. The solution was to use an ASM,
5425 * which is what's there now.
5426 */
5427static DECLCALLBACK(int) ataR3AsyncIOThread(RTTHREAD ThreadSelf, void *pvUser)
5428{
5429 const ATARequest *pReq;
5430 uint64_t u64TS = 0; /* shut up gcc */
5431 uint64_t uWait;
5432 int rc = VINF_SUCCESS;
5433 PATACONTROLLER pCtl = (PATACONTROLLER)pvUser;
5434 ATADevState *s;
5435
5436 pReq = NULL;
5437 pCtl->fChainedTransfer = false;
5438 while (!pCtl->fShutdown)
5439 {
5440 /* Keep this thread from doing anything as long as EMT is suspended. */
5441 while (pCtl->fRedoIdle)
5442 {
5443 if (pCtl->fSignalIdle)
5444 ataR3AsyncSignalIdle(pCtl);
5445 rc = RTSemEventWait(pCtl->SuspendIOSem, RT_INDEFINITE_WAIT);
5446 /* Continue if we got a signal by RTThreadPoke().
5447 * We will get notified if there is a request to process.
5448 */
5449 if (RT_UNLIKELY(rc == VERR_INTERRUPTED))
5450 continue;
5451 if (RT_FAILURE(rc) || pCtl->fShutdown)
5452 break;
5453
5454 pCtl->fRedoIdle = false;
5455 }
5456
5457 /* Wait for work. */
5458 while (pReq == NULL)
5459 {
5460 if (pCtl->fSignalIdle)
5461 ataR3AsyncSignalIdle(pCtl);
5462 rc = SUPSemEventWaitNoResume(pCtl->pSupDrvSession, pCtl->hAsyncIOSem, RT_INDEFINITE_WAIT);
5463 /* Continue if we got a signal by RTThreadPoke().
5464 * We will get notified if there is a request to process.
5465 */
5466 if (RT_UNLIKELY(rc == VERR_INTERRUPTED))
5467 continue;
5468 if (RT_FAILURE(rc) || RT_UNLIKELY(pCtl->fShutdown))
5469 break;
5470
5471 pReq = ataR3AsyncIOGetCurrentRequest(pCtl);
5472 }
5473
5474 if (RT_FAILURE(rc) || pCtl->fShutdown)
5475 break;
5476
5477 if (pReq == NULL)
5478 continue;
5479
5480 ATAAIO ReqType = pReq->ReqType;
5481
5482 Log2(("%s: Ctl#%d: state=%d, req=%d\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl), pCtl->uAsyncIOState, ReqType));
5483 if (pCtl->uAsyncIOState != ReqType)
5484 {
5485 /* The new state is not the state that was expected by the normal
5486 * state changes. This is either a RESET/ABORT or there's something
5487 * really strange going on. */
5488 if ( (pCtl->uAsyncIOState == ATA_AIO_PIO || pCtl->uAsyncIOState == ATA_AIO_DMA)
5489 && (ReqType == ATA_AIO_PIO || ReqType == ATA_AIO_DMA))
5490 {
5491 /* Incorrect sequence of PIO/DMA states. Dump request queue. */
5492 ataR3AsyncIODumpRequests(pCtl);
5493 }
5494 AssertReleaseMsg(ReqType == ATA_AIO_RESET_ASSERTED || ReqType == ATA_AIO_RESET_CLEARED || ReqType == ATA_AIO_ABORT || pCtl->uAsyncIOState == ReqType, ("I/O state inconsistent: state=%d request=%d\n", pCtl->uAsyncIOState, ReqType));
5495 }
5496
5497 /* Do our work. */
5498 {
5499 STAM_PROFILE_START(&pCtl->StatLockWait, a);
5500 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
5501 STAM_PROFILE_STOP(&pCtl->StatLockWait, a);
5502 }
5503
5504 if (pCtl->uAsyncIOState == ATA_AIO_NEW && !pCtl->fChainedTransfer)
5505 {
5506 u64TS = RTTimeNanoTS();
5507#if defined(DEBUG) || defined(VBOX_WITH_STATISTICS)
5508 STAM_PROFILE_ADV_START(&pCtl->StatAsyncTime, a);
5509#endif
5510 }
5511
5512 switch (ReqType)
5513 {
5514 case ATA_AIO_NEW:
5515
5516 pCtl->iAIOIf = pReq->u.t.iIf;
5517 s = &pCtl->aIfs[pCtl->iAIOIf];
5518 s->cbTotalTransfer = pReq->u.t.cbTotalTransfer;
5519 s->uTxDir = pReq->u.t.uTxDir;
5520 s->iBeginTransfer = pReq->u.t.iBeginTransfer;
5521 s->iSourceSink = pReq->u.t.iSourceSink;
5522 s->iIOBufferEnd = 0;
5523 s->u64CmdTS = u64TS;
5524
5525 if (s->fATAPI)
5526 {
5527 if (pCtl->fChainedTransfer)
5528 {
5529 /* Only count the actual transfers, not the PIO
5530 * transfer of the ATAPI command bytes. */
5531 if (s->fDMA)
5532 STAM_REL_COUNTER_INC(&s->StatATAPIDMA);
5533 else
5534 STAM_REL_COUNTER_INC(&s->StatATAPIPIO);
5535 }
5536 }
5537 else
5538 {
5539 if (s->fDMA)
5540 STAM_REL_COUNTER_INC(&s->StatATADMA);
5541 else
5542 STAM_REL_COUNTER_INC(&s->StatATAPIO);
5543 }
5544
5545 pCtl->fChainedTransfer = false;
5546
5547 if (s->iBeginTransfer != ATAFN_BT_NULL)
5548 {
5549 Log2(("%s: Ctl#%d: calling begin transfer function\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5550 g_apfnBeginTransFuncs[s->iBeginTransfer](s);
5551 s->iBeginTransfer = ATAFN_BT_NULL;
5552 if (s->uTxDir != PDMBLOCKTXDIR_FROM_DEVICE)
5553 s->iIOBufferEnd = s->cbElementaryTransfer;
5554 }
5555 else
5556 {
5557 s->cbElementaryTransfer = s->cbTotalTransfer;
5558 s->iIOBufferEnd = s->cbTotalTransfer;
5559 }
5560 s->iIOBufferCur = 0;
5561
5562 if (s->uTxDir != PDMBLOCKTXDIR_TO_DEVICE)
5563 {
5564 if (s->iSourceSink != ATAFN_SS_NULL)
5565 {
5566 bool fRedo;
5567 Log2(("%s: Ctl#%d: calling source/sink function\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5568 fRedo = g_apfnSourceSinkFuncs[s->iSourceSink](s);
5569 pCtl->fRedo = fRedo;
5570 if (RT_UNLIKELY(fRedo && !pCtl->fReset))
5571 {
5572 /* Operation failed at the initial transfer, restart
5573 * everything from scratch by resending the current
5574 * request. Occurs very rarely, not worth optimizing. */
5575 LogRel(("%s: Ctl#%d: redo entire operation\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5576 ataHCAsyncIOPutRequest(pCtl, pReq);
5577 break;
5578 }
5579 }
5580 else
5581 ataR3CmdOK(s, 0);
5582 s->iIOBufferEnd = s->cbElementaryTransfer;
5583
5584 }
5585
5586 /* Do not go into the transfer phase if RESET is asserted.
5587 * The CritSect is released while waiting for the host OS
5588 * to finish the I/O, thus RESET is possible here. Most
5589 * important: do not change uAsyncIOState. */
5590 if (pCtl->fReset)
5591 break;
5592
5593 if (s->fDMA)
5594 {
5595 if (s->cbTotalTransfer)
5596 {
5597 ataSetStatus(s, ATA_STAT_DRQ);
5598
5599 pCtl->uAsyncIOState = ATA_AIO_DMA;
5600 /* If BMDMA is already started, do the transfer now. */
5601 if (pCtl->BmDma.u8Cmd & BM_CMD_START)
5602 {
5603 Log2(("%s: Ctl#%d: message to async I/O thread, continuing DMA transfer immediately\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5604 ataHCAsyncIOPutRequest(pCtl, &g_ataDMARequest);
5605 }
5606 }
5607 else
5608 {
5609 Assert(s->uTxDir == PDMBLOCKTXDIR_NONE); /* Any transfer which has an initial transfer size of 0 must be marked as such. */
5610 /* Finish DMA transfer. */
5611 ataR3DMATransferStop(s);
5612 ataHCSetIRQ(s);
5613 pCtl->uAsyncIOState = ATA_AIO_NEW;
5614 }
5615 }
5616 else
5617 {
5618 if (s->cbTotalTransfer)
5619 {
5620 ataHCPIOTransfer(pCtl);
5621 Assert(!pCtl->fRedo);
5622 if (s->fATAPITransfer || s->uTxDir != PDMBLOCKTXDIR_TO_DEVICE)
5623 ataHCSetIRQ(s);
5624
5625 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE || s->iSourceSink != ATAFN_SS_NULL)
5626 {
5627 /* Write operations and not yet finished transfers
5628 * must be completed in the async I/O thread. */
5629 pCtl->uAsyncIOState = ATA_AIO_PIO;
5630 }
5631 else
5632 {
5633 /* Finished read operation can be handled inline
5634 * in the end of PIO transfer handling code. Linux
5635 * depends on this, as it waits only briefly for
5636 * devices to become ready after incoming data
5637 * transfer. Cannot find anything in the ATA spec
5638 * that backs this assumption, but as all kernels
5639 * are affected (though most of the time it does
5640 * not cause any harm) this must work. */
5641 pCtl->uAsyncIOState = ATA_AIO_NEW;
5642 }
5643 }
5644 else
5645 {
5646 Assert(s->uTxDir == PDMBLOCKTXDIR_NONE); /* Any transfer which has an initial transfer size of 0 must be marked as such. */
5647 /* Finish PIO transfer. */
5648 ataHCPIOTransfer(pCtl);
5649 Assert(!pCtl->fRedo);
5650 if (!s->fATAPITransfer)
5651 ataHCSetIRQ(s);
5652 pCtl->uAsyncIOState = ATA_AIO_NEW;
5653 }
5654 }
5655 break;
5656
5657 case ATA_AIO_DMA:
5658 {
5659 BMDMAState *bm = &pCtl->BmDma;
5660 s = &pCtl->aIfs[pCtl->iAIOIf]; /* Do not remove or there's an instant crash after loading the saved state */
5661 ATAFNSS iOriginalSourceSink = (ATAFNSS)s->iSourceSink; /* Used by the hack below, but gets reset by then. */
5662
5663 if (s->uTxDir == PDMBLOCKTXDIR_FROM_DEVICE)
5664 AssertRelease(bm->u8Cmd & BM_CMD_WRITE);
5665 else
5666 AssertRelease(!(bm->u8Cmd & BM_CMD_WRITE));
5667
5668 if (RT_LIKELY(!pCtl->fRedo))
5669 {
5670 /* The specs say that the descriptor table must not cross a
5671 * 4K boundary. */
5672 pCtl->pFirstDMADesc = bm->pvAddr;
5673 pCtl->pLastDMADesc = RT_ALIGN_32(bm->pvAddr + 1, _4K) - sizeof(BMDMADesc);
5674 }
5675 ataR3DMATransfer(pCtl);
5676
5677 if (RT_UNLIKELY(pCtl->fRedo && !pCtl->fReset))
5678 {
5679 LogRel(("PIIX3 ATA: Ctl#%d: redo DMA operation\n", ATACONTROLLER_IDX(pCtl)));
5680 ataHCAsyncIOPutRequest(pCtl, &g_ataDMARequest);
5681 break;
5682 }
5683
5684 /* The infamous delay IRQ hack. */
5685 if ( iOriginalSourceSink == ATAFN_SS_WRITE_SECTORS
5686 && s->cbTotalTransfer == 0
5687 && pCtl->DelayIRQMillies)
5688 {
5689 /* Delay IRQ for writing. Required to get the Win2K
5690 * installation work reliably (otherwise it crashes,
5691 * usually during component install). So far no better
5692 * solution has been found. */
5693 Log(("%s: delay IRQ hack\n", __FUNCTION__));
5694 PDMCritSectLeave(&pCtl->lock);
5695 RTThreadSleep(pCtl->DelayIRQMillies);
5696 PDMCritSectEnter(&pCtl->lock, VINF_SUCCESS);
5697 }
5698
5699 ataUnsetStatus(s, ATA_STAT_DRQ);
5700 Assert(!pCtl->fChainedTransfer);
5701 Assert(s->iSourceSink == ATAFN_SS_NULL);
5702 if (s->fATAPITransfer)
5703 {
5704 s->uATARegNSector = (s->uATARegNSector & ~7) | ATAPI_INT_REASON_IO | ATAPI_INT_REASON_CD;
5705 Log2(("%s: Ctl#%d: interrupt reason %#04x\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl), s->uATARegNSector));
5706 s->fATAPITransfer = false;
5707 }
5708 ataHCSetIRQ(s);
5709 pCtl->uAsyncIOState = ATA_AIO_NEW;
5710 break;
5711 }
5712
5713 case ATA_AIO_PIO:
5714 s = &pCtl->aIfs[pCtl->iAIOIf]; /* Do not remove or there's an instant crash after loading the saved state */
5715
5716 if (s->iSourceSink != ATAFN_SS_NULL)
5717 {
5718 bool fRedo;
5719 Log2(("%s: Ctl#%d: calling source/sink function\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5720 fRedo = g_apfnSourceSinkFuncs[s->iSourceSink](s);
5721 pCtl->fRedo = fRedo;
5722 if (RT_UNLIKELY(fRedo && !pCtl->fReset))
5723 {
5724 LogRel(("PIIX3 ATA: Ctl#%d: redo PIO operation\n", ATACONTROLLER_IDX(pCtl)));
5725 ataHCAsyncIOPutRequest(pCtl, &g_ataPIORequest);
5726 break;
5727 }
5728 s->iIOBufferCur = 0;
5729 s->iIOBufferEnd = s->cbElementaryTransfer;
5730 }
5731 else
5732 {
5733 /* Continue a previously started transfer. */
5734 ataUnsetStatus(s, ATA_STAT_BUSY);
5735 ataSetStatus(s, ATA_STAT_READY);
5736 }
5737
5738 /* It is possible that the drives on this controller get RESET
5739 * during the above call to the source/sink function. If that's
5740 * the case, don't restart the transfer and don't finish it the
5741 * usual way. RESET handling took care of all that already.
5742 * Most important: do not change uAsyncIOState. */
5743 if (pCtl->fReset)
5744 break;
5745
5746 if (s->cbTotalTransfer)
5747 {
5748 ataHCPIOTransfer(pCtl);
5749 ataHCSetIRQ(s);
5750
5751 if (s->uTxDir == PDMBLOCKTXDIR_TO_DEVICE || s->iSourceSink != ATAFN_SS_NULL)
5752 {
5753 /* Write operations and not yet finished transfers
5754 * must be completed in the async I/O thread. */
5755 pCtl->uAsyncIOState = ATA_AIO_PIO;
5756 }
5757 else
5758 {
5759 /* Finished read operation can be handled inline
5760 * in the end of PIO transfer handling code. Linux
5761 * depends on this, as it waits only briefly for
5762 * devices to become ready after incoming data
5763 * transfer. Cannot find anything in the ATA spec
5764 * that backs this assumption, but as all kernels
5765 * are affected (though most of the time it does
5766 * not cause any harm) this must work. */
5767 pCtl->uAsyncIOState = ATA_AIO_NEW;
5768 }
5769 }
5770 else
5771 {
5772 /* Finish PIO transfer. */
5773 ataHCPIOTransfer(pCtl);
5774 if ( !pCtl->fChainedTransfer
5775 && !s->fATAPITransfer
5776 && s->uTxDir != PDMBLOCKTXDIR_FROM_DEVICE)
5777 {
5778 ataHCSetIRQ(s);
5779 }
5780 pCtl->uAsyncIOState = ATA_AIO_NEW;
5781 }
5782 break;
5783
5784 case ATA_AIO_RESET_ASSERTED:
5785 pCtl->uAsyncIOState = ATA_AIO_RESET_CLEARED;
5786 ataHCPIOTransferStop(&pCtl->aIfs[0]);
5787 ataHCPIOTransferStop(&pCtl->aIfs[1]);
5788 /* Do not change the DMA registers, they are not affected by the
5789 * ATA controller reset logic. It should be sufficient to issue a
5790 * new command, which is now possible as the state is cleared. */
5791 break;
5792
5793 case ATA_AIO_RESET_CLEARED:
5794 pCtl->uAsyncIOState = ATA_AIO_NEW;
5795 pCtl->fReset = false;
5796 /* Ensure that half-completed transfers are not redone. A reset
5797 * cancels the entire transfer, so continuing is wrong. */
5798 pCtl->fRedo = false;
5799 pCtl->fRedoDMALastDesc = false;
5800 LogRel(("PIIX3 ATA: Ctl#%d: finished processing RESET\n",
5801 ATACONTROLLER_IDX(pCtl)));
5802 for (uint32_t i = 0; i < RT_ELEMENTS(pCtl->aIfs); i++)
5803 {
5804 if (pCtl->aIfs[i].fATAPI)
5805 ataSetStatusValue(&pCtl->aIfs[i], 0); /* NOTE: READY is _not_ set */
5806 else
5807 ataSetStatusValue(&pCtl->aIfs[i], ATA_STAT_READY | ATA_STAT_SEEK);
5808 ataR3SetSignature(&pCtl->aIfs[i]);
5809 }
5810 break;
5811
5812 case ATA_AIO_ABORT:
5813 /* Abort the current command no matter what. There cannot be
5814 * any command activity on the other drive otherwise using
5815 * one thread per controller wouldn't work at all. */
5816 s = &pCtl->aIfs[pReq->u.a.iIf];
5817
5818 pCtl->uAsyncIOState = ATA_AIO_NEW;
5819 /* Do not change the DMA registers, they are not affected by the
5820 * ATA controller reset logic. It should be sufficient to issue a
5821 * new command, which is now possible as the state is cleared. */
5822 if (pReq->u.a.fResetDrive)
5823 {
5824 ataR3ResetDevice(s);
5825 ataR3ExecuteDeviceDiagnosticSS(s);
5826 }
5827 else
5828 {
5829 /* Stop any pending DMA transfer. */
5830 s->fDMA = false;
5831 ataHCPIOTransferStop(s);
5832 ataUnsetStatus(s, ATA_STAT_BUSY | ATA_STAT_DRQ | ATA_STAT_SEEK | ATA_STAT_ERR);
5833 ataSetStatus(s, ATA_STAT_READY);
5834 ataHCSetIRQ(s);
5835 }
5836 break;
5837
5838 default:
5839 AssertMsgFailed(("Undefined async I/O state %d\n", pCtl->uAsyncIOState));
5840 }
5841
5842 ataR3AsyncIORemoveCurrentRequest(pCtl, ReqType);
5843 pReq = ataR3AsyncIOGetCurrentRequest(pCtl);
5844
5845 if (pCtl->uAsyncIOState == ATA_AIO_NEW && !pCtl->fChainedTransfer)
5846 {
5847# if defined(DEBUG) || defined(VBOX_WITH_STATISTICS)
5848 STAM_PROFILE_ADV_STOP(&pCtl->StatAsyncTime, a);
5849# endif
5850
5851 u64TS = RTTimeNanoTS() - u64TS;
5852 uWait = u64TS / 1000;
5853 Log(("%s: Ctl#%d: LUN#%d finished I/O transaction in %d microseconds\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl), pCtl->aIfs[pCtl->iAIOIf].iLUN, (uint32_t)(uWait)));
5854 /* Mark command as finished. */
5855 pCtl->aIfs[pCtl->iAIOIf].u64CmdTS = 0;
5856
5857 /*
5858 * Release logging of command execution times depends on the
5859 * command type. ATAPI commands often take longer (due to CD/DVD
5860 * spin up time etc.) so the threshold is different.
5861 */
5862 if (pCtl->aIfs[pCtl->iAIOIf].uATARegCommand != ATA_PACKET)
5863 {
5864 if (uWait > 8 * 1000 * 1000)
5865 {
5866 /*
5867 * Command took longer than 8 seconds. This is close
5868 * enough or over the guest's command timeout, so place
5869 * an entry in the release log to allow tracking such
5870 * timing errors (which are often caused by the host).
5871 */
5872 LogRel(("PIIX3 ATA: execution time for ATA command %#04x was %d seconds\n", pCtl->aIfs[pCtl->iAIOIf].uATARegCommand, uWait / (1000 * 1000)));
5873 }
5874 }
5875 else
5876 {
5877 if (uWait > 20 * 1000 * 1000)
5878 {
5879 /*
5880 * Command took longer than 20 seconds. This is close
5881 * enough or over the guest's command timeout, so place
5882 * an entry in the release log to allow tracking such
5883 * timing errors (which are often caused by the host).
5884 */
5885 LogRel(("PIIX3 ATA: execution time for ATAPI command %#04x was %d seconds\n", pCtl->aIfs[pCtl->iAIOIf].aATAPICmd[0], uWait / (1000 * 1000)));
5886 }
5887 }
5888
5889# if defined(DEBUG) || defined(VBOX_WITH_STATISTICS)
5890 if (uWait < pCtl->StatAsyncMinWait || !pCtl->StatAsyncMinWait)
5891 pCtl->StatAsyncMinWait = uWait;
5892 if (uWait > pCtl->StatAsyncMaxWait)
5893 pCtl->StatAsyncMaxWait = uWait;
5894
5895 STAM_COUNTER_ADD(&pCtl->StatAsyncTimeUS, uWait);
5896 STAM_COUNTER_INC(&pCtl->StatAsyncOps);
5897# endif /* DEBUG || VBOX_WITH_STATISTICS */
5898 }
5899
5900 PDMCritSectLeave(&pCtl->lock);
5901 }
5902
5903 /* Signal the ultimate idleness. */
5904 RTThreadUserSignal(pCtl->AsyncIOThread);
5905 if (pCtl->fSignalIdle)
5906 PDMDevHlpAsyncNotificationCompleted(pCtl->pDevInsR3);
5907
5908 /* Cleanup the state. */
5909 /* Do not destroy request lock yet, still needed for proper shutdown. */
5910 pCtl->fShutdown = false;
5911
5912 Log2(("%s: Ctl#%d: return %Rrc\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl), rc));
5913 return rc;
5914}
5915
5916#endif /* IN_RING3 */
5917
5918static uint32_t ataBMDMACmdReadB(PATACONTROLLER pCtl, uint32_t addr)
5919{
5920 uint32_t val = pCtl->BmDma.u8Cmd;
5921 Log2(("%s: addr=%#06x val=%#04x\n", __FUNCTION__, addr, val));
5922 return val;
5923}
5924
5925
5926static void ataBMDMACmdWriteB(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
5927{
5928 Log2(("%s: addr=%#06x val=%#04x\n", __FUNCTION__, addr, val));
5929 if (!(val & BM_CMD_START))
5930 {
5931 pCtl->BmDma.u8Status &= ~BM_STATUS_DMAING;
5932 pCtl->BmDma.u8Cmd = val & (BM_CMD_START | BM_CMD_WRITE);
5933 }
5934 else
5935 {
5936#ifndef IN_RC
5937 /* Check whether the guest OS wants to change DMA direction in
5938 * mid-flight. Not allowed, according to the PIIX3 specs. */
5939 Assert(!(pCtl->BmDma.u8Status & BM_STATUS_DMAING) || !((val ^ pCtl->BmDma.u8Cmd) & 0x04));
5940 uint8_t uOldBmDmaStatus = pCtl->BmDma.u8Status;
5941 pCtl->BmDma.u8Status |= BM_STATUS_DMAING;
5942 pCtl->BmDma.u8Cmd = val & (BM_CMD_START | BM_CMD_WRITE);
5943
5944 /* Do not continue DMA transfers while the RESET line is asserted. */
5945 if (pCtl->fReset)
5946 {
5947 Log2(("%s: Ctl#%d: suppressed continuing DMA transfer as RESET is active\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5948 return;
5949 }
5950
5951 /* Do not start DMA transfers if there's a PIO transfer going on,
5952 * or if there is already a transfer started on this controller. */
5953 if ( !pCtl->aIfs[pCtl->iSelectedIf].fDMA
5954 || (uOldBmDmaStatus & BM_STATUS_DMAING))
5955 return;
5956
5957 if (pCtl->aIfs[pCtl->iAIOIf].uATARegStatus & ATA_STAT_DRQ)
5958 {
5959 Log2(("%s: Ctl#%d: message to async I/O thread, continuing DMA transfer\n", __FUNCTION__, ATACONTROLLER_IDX(pCtl)));
5960 ataHCAsyncIOPutRequest(pCtl, &g_ataDMARequest);
5961 }
5962#else /* !IN_RING3 */
5963 AssertMsgFailed(("DMA START handling is too complicated for RC\n"));
5964#endif /* IN_RING3 */
5965 }
5966}
5967
5968static uint32_t ataBMDMAStatusReadB(PATACONTROLLER pCtl, uint32_t addr)
5969{
5970 uint32_t val = pCtl->BmDma.u8Status;
5971 Log2(("%s: addr=%#06x val=%#04x\n", __FUNCTION__, addr, val));
5972 return val;
5973}
5974
5975static void ataBMDMAStatusWriteB(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
5976{
5977 Log2(("%s: addr=%#06x val=%#04x\n", __FUNCTION__, addr, val));
5978 pCtl->BmDma.u8Status = (val & (BM_STATUS_D0DMA | BM_STATUS_D1DMA))
5979 | (pCtl->BmDma.u8Status & BM_STATUS_DMAING)
5980 | (pCtl->BmDma.u8Status & ~val & (BM_STATUS_ERROR | BM_STATUS_INT));
5981}
5982
5983static uint32_t ataBMDMAAddrReadL(PATACONTROLLER pCtl, uint32_t addr)
5984{
5985 uint32_t val = (uint32_t)pCtl->BmDma.pvAddr;
5986 Log2(("%s: addr=%#06x val=%#010x\n", __FUNCTION__, addr, val));
5987 return val;
5988}
5989
5990static void ataBMDMAAddrWriteL(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
5991{
5992 Log2(("%s: addr=%#06x val=%#010x\n", __FUNCTION__, addr, val));
5993 pCtl->BmDma.pvAddr = val & ~3;
5994}
5995
5996static void ataBMDMAAddrWriteLowWord(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
5997{
5998 Log2(("%s: addr=%#06x val=%#010x\n", __FUNCTION__, addr, val));
5999 pCtl->BmDma.pvAddr = (pCtl->BmDma.pvAddr & 0xFFFF0000) | RT_LOWORD(val & ~3);
6000
6001}
6002
6003static void ataBMDMAAddrWriteHighWord(PATACONTROLLER pCtl, uint32_t addr, uint32_t val)
6004{
6005 Log2(("%s: addr=%#06x val=%#010x\n", __FUNCTION__, addr, val));
6006 pCtl->BmDma.pvAddr = (RT_LOWORD(val) << 16) | RT_LOWORD(pCtl->BmDma.pvAddr);
6007}
6008
6009#define VAL(port, size) ( ((port) & 7) | ((size) << 3) )
6010
6011/**
6012 * Port I/O Handler for bus master DMA IN operations.
6013 * @see FNIOMIOPORTIN for details.
6014 */
6015PDMBOTHCBDECL(int) ataBMDMAIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
6016{
6017 uint32_t i = (uint32_t)(uintptr_t)pvUser;
6018 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6019 PATACONTROLLER pCtl = &pThis->aCts[i];
6020
6021 int rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_READ);
6022 if (rc != VINF_SUCCESS)
6023 return rc;
6024 switch (VAL(Port, cb))
6025 {
6026 case VAL(0, 1): *pu32 = ataBMDMACmdReadB(pCtl, Port); break;
6027 case VAL(0, 2): *pu32 = ataBMDMACmdReadB(pCtl, Port); break;
6028 case VAL(2, 1): *pu32 = ataBMDMAStatusReadB(pCtl, Port); break;
6029 case VAL(2, 2): *pu32 = ataBMDMAStatusReadB(pCtl, Port); break;
6030 case VAL(4, 4): *pu32 = ataBMDMAAddrReadL(pCtl, Port); break;
6031 case VAL(0, 4):
6032 /* The SCO OpenServer tries to read 4 bytes starting from offset 0. */
6033 *pu32 = ataBMDMACmdReadB(pCtl, Port) | (ataBMDMAStatusReadB(pCtl, Port) << 16);
6034 break;
6035 default:
6036 AssertMsgFailed(("%s: Unsupported read from port %x size=%d\n", __FUNCTION__, Port, cb));
6037 PDMCritSectLeave(&pCtl->lock);
6038 return VERR_IOM_IOPORT_UNUSED;
6039 }
6040 PDMCritSectLeave(&pCtl->lock);
6041 return rc;
6042}
6043
6044/**
6045 * Port I/O Handler for bus master DMA OUT operations.
6046 * @see FNIOMIOPORTOUT for details.
6047 */
6048PDMBOTHCBDECL(int) ataBMDMAIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
6049{
6050 uint32_t i = (uint32_t)(uintptr_t)pvUser;
6051 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6052 PATACONTROLLER pCtl = &pThis->aCts[i];
6053
6054 int rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_WRITE);
6055 if (rc != VINF_SUCCESS)
6056 return rc;
6057 switch (VAL(Port, cb))
6058 {
6059 case VAL(0, 1):
6060#ifdef IN_RC
6061 if (u32 & BM_CMD_START)
6062 {
6063 rc = VINF_IOM_R3_IOPORT_WRITE;
6064 break;
6065 }
6066#endif
6067 ataBMDMACmdWriteB(pCtl, Port, u32);
6068 break;
6069 case VAL(2, 1): ataBMDMAStatusWriteB(pCtl, Port, u32); break;
6070 case VAL(4, 4): ataBMDMAAddrWriteL(pCtl, Port, u32); break;
6071 case VAL(4, 2): ataBMDMAAddrWriteLowWord(pCtl, Port, u32); break;
6072 case VAL(6, 2): ataBMDMAAddrWriteHighWord(pCtl, Port, u32); break;
6073 default: AssertMsgFailed(("%s: Unsupported write to port %x size=%d val=%x\n", __FUNCTION__, Port, cb, u32)); break;
6074 }
6075 PDMCritSectLeave(&pCtl->lock);
6076 return rc;
6077}
6078
6079#undef VAL
6080
6081#ifdef IN_RING3
6082
6083/**
6084 * Callback function for mapping an PCI I/O region.
6085 *
6086 * @return VBox status code.
6087 * @param pPciDev Pointer to PCI device. Use pPciDev->pDevIns to get the device instance.
6088 * @param iRegion The region number.
6089 * @param GCPhysAddress Physical address of the region. If iType is PCI_ADDRESS_SPACE_IO, this is an
6090 * I/O port, else it's a physical address.
6091 * This address is *NOT* relative to pci_mem_base like earlier!
6092 * @param enmType One of the PCI_ADDRESS_SPACE_* values.
6093 */
6094static DECLCALLBACK(int) ataR3BMDMAIORangeMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
6095 RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
6096{
6097 PCIATAState *pThis = PCIDEV_2_PCIATASTATE(pPciDev);
6098 int rc = VINF_SUCCESS;
6099 Assert(enmType == PCI_ADDRESS_SPACE_IO);
6100 Assert(iRegion == 4);
6101 AssertMsg(RT_ALIGN(GCPhysAddress, 8) == GCPhysAddress, ("Expected 8 byte alignment. GCPhysAddress=%#x\n", GCPhysAddress));
6102
6103 /* Register the port range. */
6104 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6105 {
6106 int rc2 = PDMDevHlpIOPortRegister(pPciDev->pDevIns, (RTIOPORT)GCPhysAddress + i * 8, 8,
6107 (RTHCPTR)(uintptr_t)i, ataBMDMAIOPortWrite, ataBMDMAIOPortRead,
6108 NULL, NULL, "ATA Bus Master DMA");
6109 AssertRC(rc2);
6110 if (rc2 < rc)
6111 rc = rc2;
6112
6113 if (pThis->fRCEnabled)
6114 {
6115 rc2 = PDMDevHlpIOPortRegisterRC(pPciDev->pDevIns, (RTIOPORT)GCPhysAddress + i * 8, 8,
6116 (RTGCPTR)i, "ataBMDMAIOPortWrite", "ataBMDMAIOPortRead",
6117 NULL, NULL, "ATA Bus Master DMA");
6118 AssertRC(rc2);
6119 if (rc2 < rc)
6120 rc = rc2;
6121 }
6122 if (pThis->fR0Enabled)
6123 {
6124 rc2 = PDMDevHlpIOPortRegisterR0(pPciDev->pDevIns, (RTIOPORT)GCPhysAddress + i * 8, 8,
6125 (RTR0PTR)i, "ataBMDMAIOPortWrite", "ataBMDMAIOPortRead",
6126 NULL, NULL, "ATA Bus Master DMA");
6127 AssertRC(rc2);
6128 if (rc2 < rc)
6129 rc = rc2;
6130 }
6131 }
6132 return rc;
6133}
6134
6135
6136/* -=-=-=-=-=- PCIATAState::IBase -=-=-=-=-=- */
6137
6138/**
6139 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
6140 */
6141static DECLCALLBACK(void *) ataR3Status_QueryInterface(PPDMIBASE pInterface, const char *pszIID)
6142{
6143 PCIATAState *pThis = PDMIBASE_2_PCIATASTATE(pInterface);
6144 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
6145 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->ILeds);
6146 return NULL;
6147}
6148
6149
6150/* -=-=-=-=-=- PCIATAState::ILeds -=-=-=-=-=- */
6151
6152/**
6153 * Gets the pointer to the status LED of a unit.
6154 *
6155 * @returns VBox status code.
6156 * @param pInterface Pointer to the interface structure containing the called function pointer.
6157 * @param iLUN The unit which status LED we desire.
6158 * @param ppLed Where to store the LED pointer.
6159 */
6160static DECLCALLBACK(int) ataR3Status_QueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
6161{
6162 PCIATAState *pThis = PDMILEDPORTS_2_PCIATASTATE(pInterface);
6163 if (iLUN < 4)
6164 {
6165 switch (iLUN)
6166 {
6167 case 0: *ppLed = &pThis->aCts[0].aIfs[0].Led; break;
6168 case 1: *ppLed = &pThis->aCts[0].aIfs[1].Led; break;
6169 case 2: *ppLed = &pThis->aCts[1].aIfs[0].Led; break;
6170 case 3: *ppLed = &pThis->aCts[1].aIfs[1].Led; break;
6171 }
6172 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
6173 return VINF_SUCCESS;
6174 }
6175 return VERR_PDM_LUN_NOT_FOUND;
6176}
6177
6178
6179/* -=-=-=-=-=- ATADevState::IBase -=-=-=-=-=- */
6180
6181/**
6182 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
6183 */
6184static DECLCALLBACK(void *) ataR3QueryInterface(PPDMIBASE pInterface, const char *pszIID)
6185{
6186 ATADevState *pIf = PDMIBASE_2_ATASTATE(pInterface);
6187 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pIf->IBase);
6188 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKPORT, &pIf->IPort);
6189 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUNTNOTIFY, &pIf->IMountNotify);
6190 return NULL;
6191}
6192
6193
6194/* -=-=-=-=-=- ATADevState::IPort -=-=-=-=-=- */
6195
6196/**
6197 * @interface_method_impl{PDMIBLOCKPORT,pfnQueryDeviceLocation}
6198 */
6199static DECLCALLBACK(int) ataR3QueryDeviceLocation(PPDMIBLOCKPORT pInterface, const char **ppcszController,
6200 uint32_t *piInstance, uint32_t *piLUN)
6201{
6202 ATADevState *pIf = PDMIBLOCKPORT_2_ATASTATE(pInterface);
6203 PPDMDEVINS pDevIns = pIf->CTX_SUFF(pDevIns);
6204
6205 AssertPtrReturn(ppcszController, VERR_INVALID_POINTER);
6206 AssertPtrReturn(piInstance, VERR_INVALID_POINTER);
6207 AssertPtrReturn(piLUN, VERR_INVALID_POINTER);
6208
6209 *ppcszController = pDevIns->pReg->szName;
6210 *piInstance = pDevIns->iInstance;
6211 *piLUN = pIf->iLUN;
6212
6213 return VINF_SUCCESS;
6214}
6215
6216#endif /* IN_RING3 */
6217
6218/* -=-=-=-=-=- Wrappers -=-=-=-=-=- */
6219
6220
6221/**
6222 * Port I/O Handler for primary port range OUT operations.
6223 * @see FNIOMIOPORTOUT for details.
6224 */
6225PDMBOTHCBDECL(int) ataIOPortWrite1Other(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
6226{
6227 uint32_t i = (uint32_t)(uintptr_t)pvUser;
6228 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6229 PATACONTROLLER pCtl = &pThis->aCts[i];
6230
6231 Assert(i < 2);
6232 Assert(Port != pCtl->IOPortBase1);
6233
6234 int rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_WRITE);
6235 if (rc == VINF_SUCCESS)
6236 {
6237 /* Writes to the other command block ports should be 8-bit only. If they
6238 * are not, the high bits are simply discarded. Undocumented, but observed
6239 * on a real PIIX4 system.
6240 */
6241 if (cb > 1)
6242 Log(("ataIOPortWrite1: suspect write to port %x val=%x size=%d\n", Port, u32, cb));
6243
6244 rc = ataIOPortWriteU8(pCtl, Port, u32);
6245
6246 PDMCritSectLeave(&pCtl->lock);
6247 }
6248 return rc;
6249}
6250
6251
6252/**
6253 * Port I/O Handler for primary port range IN operations.
6254 * @see FNIOMIOPORTIN for details.
6255 */
6256PDMBOTHCBDECL(int) ataIOPortRead1Other(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
6257{
6258 uint32_t i = (uint32_t)(uintptr_t)pvUser;
6259 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6260 PATACONTROLLER pCtl = &pThis->aCts[i];
6261
6262 Assert(i < 2);
6263 Assert(Port != pCtl->IOPortBase1);
6264
6265 int rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_READ);
6266 if (rc == VINF_SUCCESS)
6267 {
6268 /* Reads from the other command block registers should be 8-bit only.
6269 * If they are not, the low byte is propagated to the high bits.
6270 * Undocumented, but observed on a real PIIX4 system.
6271 */
6272 rc = ataIOPortReadU8(pCtl, Port, pu32);
6273 if (cb > 1)
6274 {
6275 uint32_t pad;
6276
6277 /* Replicate the 8-bit result into the upper three bytes. */
6278 pad = *pu32 & 0xff;
6279 pad = pad | (pad << 8);
6280 pad = pad | (pad << 16);
6281 *pu32 = pad;
6282 Log(("ataIOPortRead1: suspect read from port %x size=%d\n", Port, cb));
6283 }
6284 PDMCritSectLeave(&pCtl->lock);
6285 }
6286 return rc;
6287}
6288
6289
6290/**
6291 * Port I/O Handler for secondary port range OUT operations.
6292 * @see FNIOMIOPORTOUT for details.
6293 */
6294PDMBOTHCBDECL(int) ataIOPortWrite2(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
6295{
6296 uint32_t i = (uint32_t)(uintptr_t)pvUser;
6297 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6298 PATACONTROLLER pCtl = &pThis->aCts[i];
6299 int rc;
6300
6301 Assert(i < 2);
6302
6303 if (cb == 1)
6304 {
6305 rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_WRITE);
6306 if (rc == VINF_SUCCESS)
6307 {
6308 rc = ataControlWrite(pCtl, Port, u32);
6309 PDMCritSectLeave(&pCtl->lock);
6310 }
6311 }
6312 else
6313 rc = VINF_SUCCESS;
6314 return rc;
6315}
6316
6317
6318/**
6319 * Port I/O Handler for secondary port range IN operations.
6320 * @see FNIOMIOPORTIN for details.
6321 */
6322PDMBOTHCBDECL(int) ataIOPortRead2(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
6323{
6324 uint32_t i = (uint32_t)(uintptr_t)pvUser;
6325 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6326 PATACONTROLLER pCtl = &pThis->aCts[i];
6327 int rc;
6328
6329 Assert(i < 2);
6330
6331 if (cb == 1)
6332 {
6333 rc = PDMCritSectEnter(&pCtl->lock, VINF_IOM_R3_IOPORT_READ);
6334 if (rc == VINF_SUCCESS)
6335 {
6336 *pu32 = ataStatusRead(pCtl, Port);
6337 PDMCritSectLeave(&pCtl->lock);
6338 }
6339 }
6340 else
6341 rc = VERR_IOM_IOPORT_UNUSED;
6342 return rc;
6343}
6344
6345#ifdef IN_RING3
6346
6347
6348DECLINLINE(void) ataR3RelocBuffer(PPDMDEVINS pDevIns, ATADevState *s)
6349{
6350 if (s->pbIOBufferR3)
6351 s->pbIOBufferRC = MMHyperR3ToRC(PDMDevHlpGetVM(pDevIns), s->pbIOBufferR3);
6352}
6353
6354
6355/**
6356 * Detach notification.
6357 *
6358 * The DVD drive has been unplugged.
6359 *
6360 * @param pDevIns The device instance.
6361 * @param iLUN The logical unit which is being detached.
6362 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
6363 */
6364static DECLCALLBACK(void) ataR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
6365{
6366 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6367 PATACONTROLLER pCtl;
6368 ATADevState *pIf;
6369 unsigned iController;
6370 unsigned iInterface;
6371
6372 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
6373 ("PIIX3IDE: Device does not support hotplugging\n"));
6374
6375 /*
6376 * Locate the controller and stuff.
6377 */
6378 iController = iLUN / RT_ELEMENTS(pThis->aCts[0].aIfs);
6379 AssertReleaseMsg(iController < RT_ELEMENTS(pThis->aCts), ("iController=%d iLUN=%d\n", iController, iLUN));
6380 pCtl = &pThis->aCts[iController];
6381
6382 iInterface = iLUN % RT_ELEMENTS(pThis->aCts[0].aIfs);
6383 pIf = &pCtl->aIfs[iInterface];
6384
6385 /*
6386 * Zero some important members.
6387 */
6388 pIf->pDrvBase = NULL;
6389 pIf->pDrvBlock = NULL;
6390 pIf->pDrvBlockBios = NULL;
6391 pIf->pDrvMount = NULL;
6392
6393 /*
6394 * In case there was a medium inserted.
6395 */
6396 ataR3MediumRemoved(pIf);
6397}
6398
6399
6400/**
6401 * Configure a LUN.
6402 *
6403 * @returns VBox status code.
6404 * @param pDevIns The device instance.
6405 * @param pIf The ATA unit state.
6406 */
6407static int ataR3ConfigLun(PPDMDEVINS pDevIns, ATADevState *pIf)
6408{
6409 int rc = VINF_SUCCESS;
6410 PDMBLOCKTYPE enmType;
6411
6412 /*
6413 * Query Block, Bios and Mount interfaces.
6414 */
6415 pIf->pDrvBlock = PDMIBASE_QUERY_INTERFACE(pIf->pDrvBase, PDMIBLOCK);
6416 if (!pIf->pDrvBlock)
6417 {
6418 AssertMsgFailed(("Configuration error: LUN#%d hasn't a block interface!\n", pIf->iLUN));
6419 return VERR_PDM_MISSING_INTERFACE;
6420 }
6421
6422 /** @todo implement the BIOS invisible code path. */
6423 pIf->pDrvBlockBios = PDMIBASE_QUERY_INTERFACE(pIf->pDrvBase, PDMIBLOCKBIOS);
6424 if (!pIf->pDrvBlockBios)
6425 {
6426 AssertMsgFailed(("Configuration error: LUN#%d hasn't a block BIOS interface!\n", pIf->iLUN));
6427 return VERR_PDM_MISSING_INTERFACE;
6428 }
6429 pIf->pDrvMount = PDMIBASE_QUERY_INTERFACE(pIf->pDrvBase, PDMIMOUNT);
6430
6431 /*
6432 * Validate type.
6433 */
6434 enmType = pIf->pDrvBlock->pfnGetType(pIf->pDrvBlock);
6435 if ( enmType != PDMBLOCKTYPE_CDROM
6436 && enmType != PDMBLOCKTYPE_DVD
6437 && enmType != PDMBLOCKTYPE_HARD_DISK)
6438 {
6439 AssertMsgFailed(("Configuration error: LUN#%d isn't a disk or cd/dvd-rom. enmType=%d\n", pIf->iLUN, enmType));
6440 return VERR_PDM_UNSUPPORTED_BLOCK_TYPE;
6441 }
6442 if ( ( enmType == PDMBLOCKTYPE_DVD
6443 || enmType == PDMBLOCKTYPE_CDROM)
6444 && !pIf->pDrvMount)
6445 {
6446 AssertMsgFailed(("Internal error: cdrom without a mountable interface, WTF???!\n"));
6447 return VERR_INTERNAL_ERROR;
6448 }
6449 pIf->fATAPI = enmType == PDMBLOCKTYPE_DVD || enmType == PDMBLOCKTYPE_CDROM;
6450 pIf->fATAPIPassthrough = pIf->fATAPI ? (pIf->pDrvBlock->pfnSendCmd != NULL) : false;
6451
6452 /*
6453 * Allocate I/O buffer.
6454 */
6455 if (pIf->fATAPI)
6456 pIf->cbSector = 2048;
6457 else
6458 pIf->cbSector = pIf->pDrvBlock->pfnGetSectorSize(pIf->pDrvBlock);
6459
6460 PVM pVM = PDMDevHlpGetVM(pDevIns);
6461 if (pIf->cbIOBuffer)
6462 {
6463 /* Buffer is (probably) already allocated. Validate the fields,
6464 * because memory corruption can also overwrite pIf->cbIOBuffer. */
6465 if (pIf->fATAPI)
6466 AssertRelease(pIf->cbIOBuffer == _128K);
6467 else
6468 AssertRelease(pIf->cbIOBuffer == ATA_MAX_MULT_SECTORS * pIf->cbSector);
6469 Assert(pIf->pbIOBufferR3);
6470 Assert(pIf->pbIOBufferR0 == MMHyperR3ToR0(pVM, pIf->pbIOBufferR3));
6471 Assert(pIf->pbIOBufferRC == MMHyperR3ToRC(pVM, pIf->pbIOBufferR3));
6472 }
6473 else
6474 {
6475 if (pIf->fATAPI)
6476 pIf->cbIOBuffer = _128K;
6477 else
6478 pIf->cbIOBuffer = ATA_MAX_MULT_SECTORS * pIf->cbSector;
6479 Assert(!pIf->pbIOBufferR3);
6480 rc = MMR3HyperAllocOnceNoRel(pVM, pIf->cbIOBuffer, 0, MM_TAG_PDM_DEVICE_USER, (void **)&pIf->pbIOBufferR3);
6481 if (RT_FAILURE(rc))
6482 return VERR_NO_MEMORY;
6483 pIf->pbIOBufferR0 = MMHyperR3ToR0(pVM, pIf->pbIOBufferR3);
6484 pIf->pbIOBufferRC = MMHyperR3ToRC(pVM, pIf->pbIOBufferR3);
6485 }
6486
6487 /*
6488 * Init geometry (only for non-CD/DVD media).
6489 */
6490 if (pIf->fATAPI)
6491 {
6492 pIf->cTotalSectors = pIf->pDrvBlock->pfnGetSize(pIf->pDrvBlock) / pIf->cbSector;
6493 pIf->PCHSGeometry.cCylinders = 0; /* dummy */
6494 pIf->PCHSGeometry.cHeads = 0; /* dummy */
6495 pIf->PCHSGeometry.cSectors = 0; /* dummy */
6496 LogRel(("PIIX3 ATA: LUN#%d: CD/DVD, total number of sectors %Ld, passthrough %s\n", pIf->iLUN, pIf->cTotalSectors, (pIf->fATAPIPassthrough ? "enabled" : "disabled")));
6497 }
6498 else
6499 {
6500 pIf->cTotalSectors = pIf->pDrvBlock->pfnGetSize(pIf->pDrvBlock) / pIf->cbSector;
6501 rc = pIf->pDrvBlockBios->pfnGetPCHSGeometry(pIf->pDrvBlockBios,
6502 &pIf->PCHSGeometry);
6503 if (rc == VERR_PDM_MEDIA_NOT_MOUNTED)
6504 {
6505 pIf->PCHSGeometry.cCylinders = 0;
6506 pIf->PCHSGeometry.cHeads = 16; /*??*/
6507 pIf->PCHSGeometry.cSectors = 63; /*??*/
6508 }
6509 else if (rc == VERR_PDM_GEOMETRY_NOT_SET)
6510 {
6511 pIf->PCHSGeometry.cCylinders = 0; /* autodetect marker */
6512 rc = VINF_SUCCESS;
6513 }
6514 AssertRC(rc);
6515
6516 if ( pIf->PCHSGeometry.cCylinders == 0
6517 || pIf->PCHSGeometry.cHeads == 0
6518 || pIf->PCHSGeometry.cSectors == 0
6519 )
6520 {
6521 uint64_t cCylinders = pIf->cTotalSectors / (16 * 63);
6522 pIf->PCHSGeometry.cCylinders = RT_MAX(RT_MIN(cCylinders, 16383), 1);
6523 pIf->PCHSGeometry.cHeads = 16;
6524 pIf->PCHSGeometry.cSectors = 63;
6525 /* Set the disk geometry information. Ignore errors. */
6526 pIf->pDrvBlockBios->pfnSetPCHSGeometry(pIf->pDrvBlockBios,
6527 &pIf->PCHSGeometry);
6528 rc = VINF_SUCCESS;
6529 }
6530 LogRel(("PIIX3 ATA: LUN#%d: disk, PCHS=%u/%u/%u, total number of sectors %Ld\n", pIf->iLUN, pIf->PCHSGeometry.cCylinders, pIf->PCHSGeometry.cHeads, pIf->PCHSGeometry.cSectors, pIf->cTotalSectors));
6531
6532 if (pIf->pDrvBlock->pfnDiscard)
6533 LogRel(("PIIX3 ATA: LUN#%d: TRIM enabled\n", pIf->iLUN));
6534 }
6535 return rc;
6536}
6537
6538
6539/**
6540 * Attach command.
6541 *
6542 * This is called when we change block driver for the DVD drive.
6543 *
6544 * @returns VBox status code.
6545 * @param pDevIns The device instance.
6546 * @param iLUN The logical unit which is being detached.
6547 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
6548 */
6549static DECLCALLBACK(int) ataR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
6550{
6551 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6552 PATACONTROLLER pCtl;
6553 ATADevState *pIf;
6554 int rc;
6555 unsigned iController;
6556 unsigned iInterface;
6557
6558 AssertMsgReturn(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
6559 ("PIIX3IDE: Device does not support hotplugging\n"),
6560 VERR_INVALID_PARAMETER);
6561
6562 /*
6563 * Locate the controller and stuff.
6564 */
6565 iController = iLUN / RT_ELEMENTS(pThis->aCts[0].aIfs);
6566 AssertReleaseMsg(iController < RT_ELEMENTS(pThis->aCts), ("iController=%d iLUN=%d\n", iController, iLUN));
6567 pCtl = &pThis->aCts[iController];
6568
6569 iInterface = iLUN % RT_ELEMENTS(pThis->aCts[0].aIfs);
6570 pIf = &pCtl->aIfs[iInterface];
6571
6572 /* the usual paranoia */
6573 AssertRelease(!pIf->pDrvBase);
6574 AssertRelease(!pIf->pDrvBlock);
6575 Assert(ATADEVSTATE_2_CONTROLLER(pIf) == pCtl);
6576 Assert(pIf->iLUN == iLUN);
6577
6578 /*
6579 * Try attach the block device and get the interfaces,
6580 * required as well as optional.
6581 */
6582 rc = PDMDevHlpDriverAttach(pDevIns, pIf->iLUN, &pIf->IBase, &pIf->pDrvBase, NULL);
6583 if (RT_SUCCESS(rc))
6584 {
6585 rc = ataR3ConfigLun(pDevIns, pIf);
6586 /*
6587 * In case there is a medium inserted.
6588 */
6589 ataR3MediumInserted(pIf);
6590 ataR3MediumTypeSet(pIf, ATA_MEDIA_TYPE_UNKNOWN);
6591 }
6592 else
6593 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", pIf->iLUN, rc));
6594
6595 if (RT_FAILURE(rc))
6596 {
6597 pIf->pDrvBase = NULL;
6598 pIf->pDrvBlock = NULL;
6599 }
6600 return rc;
6601}
6602
6603
6604/**
6605 * Resume notification.
6606 *
6607 * @returns VBox status.
6608 * @param pDevIns The device instance data.
6609 */
6610static DECLCALLBACK(void) ataR3Resume(PPDMDEVINS pDevIns)
6611{
6612 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6613 int rc;
6614
6615 Log(("%s:\n", __FUNCTION__));
6616 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6617 {
6618 if (pThis->aCts[i].fRedo && pThis->aCts[i].fRedoIdle)
6619 {
6620 rc = RTSemEventSignal(pThis->aCts[i].SuspendIOSem);
6621 AssertRC(rc);
6622 }
6623 }
6624 return;
6625}
6626
6627
6628/**
6629 * Checks if all (both) the async I/O threads have quiesced.
6630 *
6631 * @returns true on success.
6632 * @returns false when one or more threads is still processing.
6633 * @param pThis Pointer to the instance data.
6634 */
6635static bool ataR3AllAsyncIOIsIdle(PPDMDEVINS pDevIns)
6636{
6637 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6638
6639 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6640 if (pThis->aCts[i].AsyncIOThread != NIL_RTTHREAD)
6641 {
6642 bool fRc = ataR3AsyncIOIsIdle(&pThis->aCts[i], false /*fStrict*/);
6643 if (!fRc)
6644 {
6645 /* Make it signal PDM & itself when its done */
6646 PDMCritSectEnter(&pThis->aCts[i].AsyncIORequestLock, VERR_IGNORED);
6647 ASMAtomicWriteBool(&pThis->aCts[i].fSignalIdle, true);
6648 PDMCritSectLeave(&pThis->aCts[i].AsyncIORequestLock);
6649
6650 fRc = ataR3AsyncIOIsIdle(&pThis->aCts[i], false /*fStrict*/);
6651 if (!fRc)
6652 {
6653#if 0 /** @todo Need to do some time tracking here... */
6654 LogRel(("PIIX3 ATA: Ctl#%u is still executing, DevSel=%d AIOIf=%d CmdIf0=%#04x CmdIf1=%#04x\n",
6655 i, pThis->aCts[i].iSelectedIf, pThis->aCts[i].iAIOIf,
6656 pThis->aCts[i].aIfs[0].uATARegCommand, pThis->aCts[i].aIfs[1].uATARegCommand));
6657#endif
6658 return false;
6659 }
6660 }
6661 ASMAtomicWriteBool(&pThis->aCts[i].fSignalIdle, false);
6662 }
6663 return true;
6664}
6665
6666/**
6667 * Prepare state save and load operation.
6668 *
6669 * @returns VBox status code.
6670 * @param pDevIns Device instance of the device which registered the data unit.
6671 * @param pSSM SSM operation handle.
6672 */
6673static DECLCALLBACK(int) ataR3SaveLoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
6674{
6675 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6676
6677 /* sanity - the suspend notification will wait on the async stuff. */
6678 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6679 AssertLogRelMsgReturn(ataR3AsyncIOIsIdle(&pThis->aCts[i], false /*fStrict*/),
6680 ("i=%u\n", i),
6681 VERR_SSM_IDE_ASYNC_TIMEOUT);
6682 return VINF_SUCCESS;
6683}
6684
6685/**
6686 * @copydoc FNSSMDEVLIVEEXEC
6687 */
6688static DECLCALLBACK(int) ataR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
6689{
6690 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6691
6692 SSMR3PutU8(pSSM, pThis->u8Type);
6693 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6694 {
6695 SSMR3PutBool(pSSM, true); /* For controller enabled / disabled. */
6696 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
6697 {
6698 SSMR3PutBool(pSSM, pThis->aCts[i].aIfs[j].pDrvBase != NULL);
6699 SSMR3PutStrZ(pSSM, pThis->aCts[i].aIfs[j].szSerialNumber);
6700 SSMR3PutStrZ(pSSM, pThis->aCts[i].aIfs[j].szFirmwareRevision);
6701 SSMR3PutStrZ(pSSM, pThis->aCts[i].aIfs[j].szModelNumber);
6702 }
6703 }
6704
6705 return VINF_SSM_DONT_CALL_AGAIN;
6706}
6707
6708/**
6709 * @copydoc FNSSMDEVSAVEEXEC
6710 */
6711static DECLCALLBACK(int) ataR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
6712{
6713 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6714
6715 ataR3LiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
6716
6717 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6718 {
6719 SSMR3PutU8(pSSM, pThis->aCts[i].iSelectedIf);
6720 SSMR3PutU8(pSSM, pThis->aCts[i].iAIOIf);
6721 SSMR3PutU8(pSSM, pThis->aCts[i].uAsyncIOState);
6722 SSMR3PutBool(pSSM, pThis->aCts[i].fChainedTransfer);
6723 SSMR3PutBool(pSSM, pThis->aCts[i].fReset);
6724 SSMR3PutBool(pSSM, pThis->aCts[i].fRedo);
6725 SSMR3PutBool(pSSM, pThis->aCts[i].fRedoIdle);
6726 SSMR3PutBool(pSSM, pThis->aCts[i].fRedoDMALastDesc);
6727 SSMR3PutMem(pSSM, &pThis->aCts[i].BmDma, sizeof(pThis->aCts[i].BmDma));
6728 SSMR3PutGCPhys32(pSSM, pThis->aCts[i].pFirstDMADesc);
6729 SSMR3PutGCPhys32(pSSM, pThis->aCts[i].pLastDMADesc);
6730 SSMR3PutGCPhys32(pSSM, pThis->aCts[i].pRedoDMABuffer);
6731 SSMR3PutU32(pSSM, pThis->aCts[i].cbRedoDMABuffer);
6732
6733 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
6734 {
6735 SSMR3PutBool(pSSM, pThis->aCts[i].aIfs[j].fLBA48);
6736 SSMR3PutBool(pSSM, pThis->aCts[i].aIfs[j].fATAPI);
6737 SSMR3PutBool(pSSM, pThis->aCts[i].aIfs[j].fIrqPending);
6738 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].cMultSectors);
6739 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].PCHSGeometry.cCylinders);
6740 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].PCHSGeometry.cHeads);
6741 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].PCHSGeometry.cSectors);
6742 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].cSectorsPerIRQ);
6743 SSMR3PutU64(pSSM, pThis->aCts[i].aIfs[j].cTotalSectors);
6744 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegFeature);
6745 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegFeatureHOB);
6746 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegError);
6747 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegNSector);
6748 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegNSectorHOB);
6749 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegSector);
6750 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegSectorHOB);
6751 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegLCyl);
6752 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegLCylHOB);
6753 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegHCyl);
6754 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegHCylHOB);
6755 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegSelect);
6756 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegStatus);
6757 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegCommand);
6758 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATARegDevCtl);
6759 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uATATransferMode);
6760 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].uTxDir);
6761 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].iBeginTransfer);
6762 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].iSourceSink);
6763 SSMR3PutBool(pSSM, pThis->aCts[i].aIfs[j].fDMA);
6764 SSMR3PutBool(pSSM, pThis->aCts[i].aIfs[j].fATAPITransfer);
6765 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].cbTotalTransfer);
6766 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].cbElementaryTransfer);
6767 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].iIOBufferCur);
6768 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].iIOBufferEnd);
6769 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].iIOBufferPIODataStart);
6770 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].iIOBufferPIODataEnd);
6771 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].iATAPILBA);
6772 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].cbATAPISector);
6773 SSMR3PutMem(pSSM, &pThis->aCts[i].aIfs[j].aATAPICmd, sizeof(pThis->aCts[i].aIfs[j].aATAPICmd));
6774 SSMR3PutMem(pSSM, &pThis->aCts[i].aIfs[j].abATAPISense, sizeof(pThis->aCts[i].aIfs[j].abATAPISense));
6775 SSMR3PutU8(pSSM, pThis->aCts[i].aIfs[j].cNotifiedMediaChange);
6776 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].MediaEventStatus);
6777 SSMR3PutMem(pSSM, &pThis->aCts[i].aIfs[j].Led, sizeof(pThis->aCts[i].aIfs[j].Led));
6778 SSMR3PutU32(pSSM, pThis->aCts[i].aIfs[j].cbIOBuffer);
6779 if (pThis->aCts[i].aIfs[j].cbIOBuffer)
6780 SSMR3PutMem(pSSM, pThis->aCts[i].aIfs[j].CTX_SUFF(pbIOBuffer), pThis->aCts[i].aIfs[j].cbIOBuffer);
6781 else
6782 Assert(pThis->aCts[i].aIfs[j].CTX_SUFF(pbIOBuffer) == NULL);
6783 }
6784 }
6785
6786 return SSMR3PutU32(pSSM, ~0); /* sanity/terminator */
6787}
6788
6789/**
6790 * Converts the LUN number into a message string.
6791 */
6792static const char *ataR3StringifyLun(unsigned iLun)
6793{
6794 switch (iLun)
6795 {
6796 case 0: return "primary master";
6797 case 1: return "primary slave";
6798 case 2: return "secondary master";
6799 case 3: return "secondary slave";
6800 default: AssertFailedReturn("unknown lun");
6801 }
6802}
6803
6804/**
6805 * FNSSMDEVLOADEXEC
6806 */
6807static DECLCALLBACK(int) ataR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
6808{
6809 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
6810 int rc;
6811 uint32_t u32;
6812
6813 if ( uVersion != ATA_SAVED_STATE_VERSION
6814 && uVersion != ATA_SAVED_STATE_VERSION_VBOX_30
6815 && uVersion != ATA_SAVED_STATE_VERSION_WITHOUT_FULL_SENSE
6816 && uVersion != ATA_SAVED_STATE_VERSION_WITHOUT_EVENT_STATUS
6817 && uVersion != ATA_SAVED_STATE_VERSION_WITH_BOOL_TYPE)
6818 {
6819 AssertMsgFailed(("uVersion=%d\n", uVersion));
6820 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
6821 }
6822
6823 /*
6824 * Verify the configuration.
6825 */
6826 if (uVersion > ATA_SAVED_STATE_VERSION_VBOX_30)
6827 {
6828 uint8_t u8Type;
6829 rc = SSMR3GetU8(pSSM, &u8Type);
6830 AssertRCReturn(rc, rc);
6831 if (u8Type != pThis->u8Type)
6832 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch: u8Type - saved=%u config=%u"), u8Type, pThis->u8Type);
6833
6834 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6835 {
6836 bool fEnabled;
6837 rc = SSMR3GetBool(pSSM, &fEnabled);
6838 AssertRCReturn(rc, rc);
6839 if (!fEnabled)
6840 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Ctr#%u onfig mismatch: fEnabled != true"), i);
6841
6842 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
6843 {
6844 ATADevState const *pIf = &pThis->aCts[i].aIfs[j];
6845
6846 bool fInUse;
6847 rc = SSMR3GetBool(pSSM, &fInUse);
6848 AssertRCReturn(rc, rc);
6849 if (fInUse != (pIf->pDrvBase != NULL))
6850 return SSMR3SetCfgError(pSSM, RT_SRC_POS,
6851 N_("The %s VM is missing a %s device. Please make sure the source and target VMs have compatible storage configurations"),
6852 fInUse ? "target" : "source", ataR3StringifyLun(pIf->iLUN) );
6853
6854 char szSerialNumber[ATA_SERIAL_NUMBER_LENGTH+1];
6855 rc = SSMR3GetStrZ(pSSM, szSerialNumber, sizeof(szSerialNumber));
6856 AssertRCReturn(rc, rc);
6857 if (strcmp(szSerialNumber, pIf->szSerialNumber))
6858 LogRel(("PIIX3 ATA: LUN#%u config mismatch: Serial number - saved='%s' config='%s'\n",
6859 pIf->iLUN, szSerialNumber, pIf->szSerialNumber));
6860
6861 char szFirmwareRevision[ATA_FIRMWARE_REVISION_LENGTH+1];
6862 rc = SSMR3GetStrZ(pSSM, szFirmwareRevision, sizeof(szFirmwareRevision));
6863 AssertRCReturn(rc, rc);
6864 if (strcmp(szFirmwareRevision, pIf->szFirmwareRevision))
6865 LogRel(("PIIX3 ATA: LUN#%u config mismatch: Firmware revision - saved='%s' config='%s'\n",
6866 pIf->iLUN, szFirmwareRevision, pIf->szFirmwareRevision));
6867
6868 char szModelNumber[ATA_MODEL_NUMBER_LENGTH+1];
6869 rc = SSMR3GetStrZ(pSSM, szModelNumber, sizeof(szModelNumber));
6870 AssertRCReturn(rc, rc);
6871 if (strcmp(szModelNumber, pIf->szModelNumber))
6872 LogRel(("PIIX3 ATA: LUN#%u config mismatch: Model number - saved='%s' config='%s'\n",
6873 pIf->iLUN, szModelNumber, pIf->szModelNumber));
6874 }
6875 }
6876 }
6877 if (uPass != SSM_PASS_FINAL)
6878 return VINF_SUCCESS;
6879
6880 /*
6881 * Restore valid parts of the PCIATAState structure
6882 */
6883 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
6884 {
6885 /* integrity check */
6886 if (!ataR3AsyncIOIsIdle(&pThis->aCts[i], false))
6887 {
6888 AssertMsgFailed(("Async I/O for controller %d is active\n", i));
6889 return VERR_INTERNAL_ERROR_4;
6890 }
6891
6892 SSMR3GetU8(pSSM, &pThis->aCts[i].iSelectedIf);
6893 SSMR3GetU8(pSSM, &pThis->aCts[i].iAIOIf);
6894 SSMR3GetU8(pSSM, &pThis->aCts[i].uAsyncIOState);
6895 SSMR3GetBool(pSSM, &pThis->aCts[i].fChainedTransfer);
6896 SSMR3GetBool(pSSM, (bool *)&pThis->aCts[i].fReset);
6897 SSMR3GetBool(pSSM, (bool *)&pThis->aCts[i].fRedo);
6898 SSMR3GetBool(pSSM, (bool *)&pThis->aCts[i].fRedoIdle);
6899 SSMR3GetBool(pSSM, (bool *)&pThis->aCts[i].fRedoDMALastDesc);
6900 SSMR3GetMem(pSSM, &pThis->aCts[i].BmDma, sizeof(pThis->aCts[i].BmDma));
6901 SSMR3GetGCPhys32(pSSM, &pThis->aCts[i].pFirstDMADesc);
6902 SSMR3GetGCPhys32(pSSM, &pThis->aCts[i].pLastDMADesc);
6903 SSMR3GetGCPhys32(pSSM, &pThis->aCts[i].pRedoDMABuffer);
6904 SSMR3GetU32(pSSM, &pThis->aCts[i].cbRedoDMABuffer);
6905
6906 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
6907 {
6908 SSMR3GetBool(pSSM, &pThis->aCts[i].aIfs[j].fLBA48);
6909 SSMR3GetBool(pSSM, &pThis->aCts[i].aIfs[j].fATAPI);
6910 SSMR3GetBool(pSSM, &pThis->aCts[i].aIfs[j].fIrqPending);
6911 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].cMultSectors);
6912 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].PCHSGeometry.cCylinders);
6913 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].PCHSGeometry.cHeads);
6914 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].PCHSGeometry.cSectors);
6915 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].cSectorsPerIRQ);
6916 SSMR3GetU64(pSSM, &pThis->aCts[i].aIfs[j].cTotalSectors);
6917 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegFeature);
6918 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegFeatureHOB);
6919 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegError);
6920 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegNSector);
6921 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegNSectorHOB);
6922 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegSector);
6923 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegSectorHOB);
6924 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegLCyl);
6925 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegLCylHOB);
6926 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegHCyl);
6927 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegHCylHOB);
6928 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegSelect);
6929 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegStatus);
6930 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegCommand);
6931 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATARegDevCtl);
6932 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uATATransferMode);
6933 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].uTxDir);
6934 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].iBeginTransfer);
6935 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].iSourceSink);
6936 SSMR3GetBool(pSSM, &pThis->aCts[i].aIfs[j].fDMA);
6937 SSMR3GetBool(pSSM, &pThis->aCts[i].aIfs[j].fATAPITransfer);
6938 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].cbTotalTransfer);
6939 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].cbElementaryTransfer);
6940 /* NB: cbPIOTransferLimit could be saved/restored but it's sufficient
6941 * to re-calculate it here, with a tiny risk that it could be
6942 * unnecessarily low for the current transfer only. Could be changed
6943 * when changing the saved state in the future.
6944 */
6945 pThis->aCts[i].aIfs[j].cbPIOTransferLimit = (pThis->aCts[i].aIfs[j].uATARegHCyl << 8) | pThis->aCts[i].aIfs[j].uATARegLCyl;
6946 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].iIOBufferCur);
6947 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].iIOBufferEnd);
6948 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].iIOBufferPIODataStart);
6949 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].iIOBufferPIODataEnd);
6950 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].iATAPILBA);
6951 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].cbATAPISector);
6952 SSMR3GetMem(pSSM, &pThis->aCts[i].aIfs[j].aATAPICmd, sizeof(pThis->aCts[i].aIfs[j].aATAPICmd));
6953 if (uVersion > ATA_SAVED_STATE_VERSION_WITHOUT_FULL_SENSE)
6954 {
6955 SSMR3GetMem(pSSM, pThis->aCts[i].aIfs[j].abATAPISense, sizeof(pThis->aCts[i].aIfs[j].abATAPISense));
6956 }
6957 else
6958 {
6959 uint8_t uATAPISenseKey, uATAPIASC;
6960 memset(pThis->aCts[i].aIfs[j].abATAPISense, '\0', sizeof(pThis->aCts[i].aIfs[j].abATAPISense));
6961 pThis->aCts[i].aIfs[j].abATAPISense[0] = 0x70 | (1 << 7);
6962 pThis->aCts[i].aIfs[j].abATAPISense[7] = 10;
6963 SSMR3GetU8(pSSM, &uATAPISenseKey);
6964 SSMR3GetU8(pSSM, &uATAPIASC);
6965 pThis->aCts[i].aIfs[j].abATAPISense[2] = uATAPISenseKey & 0x0f;
6966 pThis->aCts[i].aIfs[j].abATAPISense[12] = uATAPIASC;
6967 }
6968 /** @todo triple-check this hack after passthrough is working */
6969 SSMR3GetU8(pSSM, &pThis->aCts[i].aIfs[j].cNotifiedMediaChange);
6970 if (uVersion > ATA_SAVED_STATE_VERSION_WITHOUT_EVENT_STATUS)
6971 SSMR3GetU32(pSSM, (uint32_t*)&pThis->aCts[i].aIfs[j].MediaEventStatus);
6972 else
6973 pThis->aCts[i].aIfs[j].MediaEventStatus = ATA_EVENT_STATUS_UNCHANGED;
6974 SSMR3GetMem(pSSM, &pThis->aCts[i].aIfs[j].Led, sizeof(pThis->aCts[i].aIfs[j].Led));
6975 SSMR3GetU32(pSSM, &pThis->aCts[i].aIfs[j].cbIOBuffer);
6976 if (pThis->aCts[i].aIfs[j].cbIOBuffer)
6977 {
6978 if (pThis->aCts[i].aIfs[j].CTX_SUFF(pbIOBuffer))
6979 SSMR3GetMem(pSSM, pThis->aCts[i].aIfs[j].CTX_SUFF(pbIOBuffer), pThis->aCts[i].aIfs[j].cbIOBuffer);
6980 else
6981 {
6982 LogRel(("ATA: No buffer for %d/%d\n", i, j));
6983 if (SSMR3HandleGetAfter(pSSM) != SSMAFTER_DEBUG_IT)
6984 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("No buffer for %d/%d"), i, j);
6985
6986 /* skip the buffer if we're loading for the debugger / animator. */
6987 uint8_t u8Ignored;
6988 size_t cbLeft = pThis->aCts[i].aIfs[j].cbIOBuffer;
6989 while (cbLeft-- > 0)
6990 SSMR3GetU8(pSSM, &u8Ignored);
6991 }
6992 }
6993 else
6994 Assert(pThis->aCts[i].aIfs[j].CTX_SUFF(pbIOBuffer) == NULL);
6995 }
6996 }
6997 if (uVersion <= ATA_SAVED_STATE_VERSION_VBOX_30)
6998 SSMR3GetU8(pSSM, &pThis->u8Type);
6999
7000 rc = SSMR3GetU32(pSSM, &u32);
7001 if (RT_FAILURE(rc))
7002 return rc;
7003 if (u32 != ~0U)
7004 {
7005 AssertMsgFailed(("u32=%#x expected ~0\n", u32));
7006 rc = VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
7007 return rc;
7008 }
7009
7010 return VINF_SUCCESS;
7011}
7012
7013
7014/**
7015 * Callback employed by ataSuspend and ataR3PowerOff.
7016 *
7017 * @returns true if we've quiesced, false if we're still working.
7018 * @param pDevIns The device instance.
7019 */
7020static DECLCALLBACK(bool) ataR3IsAsyncSuspendOrPowerOffDone(PPDMDEVINS pDevIns)
7021{
7022 return ataR3AllAsyncIOIsIdle(pDevIns);
7023}
7024
7025
7026/**
7027 * Common worker for ataSuspend and ataR3PowerOff.
7028 */
7029static void ataR3SuspendOrPowerOff(PPDMDEVINS pDevIns)
7030{
7031 if (!ataR3AllAsyncIOIsIdle(pDevIns))
7032 PDMDevHlpSetAsyncNotification(pDevIns, ataR3IsAsyncSuspendOrPowerOffDone);
7033}
7034
7035
7036/**
7037 * Power Off notification.
7038 *
7039 * @returns VBox status.
7040 * @param pDevIns The device instance data.
7041 */
7042static DECLCALLBACK(void) ataR3PowerOff(PPDMDEVINS pDevIns)
7043{
7044 Log(("%s:\n", __FUNCTION__));
7045 ataR3SuspendOrPowerOff(pDevIns);
7046}
7047
7048
7049/**
7050 * Suspend notification.
7051 *
7052 * @returns VBox status.
7053 * @param pDevIns The device instance data.
7054 */
7055static DECLCALLBACK(void) ataR3Suspend(PPDMDEVINS pDevIns)
7056{
7057 Log(("%s:\n", __FUNCTION__));
7058 ataR3SuspendOrPowerOff(pDevIns);
7059}
7060
7061
7062/**
7063 * Callback employed by ataR3Reset.
7064 *
7065 * @returns true if we've quiesced, false if we're still working.
7066 * @param pDevIns The device instance.
7067 */
7068static DECLCALLBACK(bool) ataR3IsAsyncResetDone(PPDMDEVINS pDevIns)
7069{
7070 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
7071
7072 if (!ataR3AllAsyncIOIsIdle(pDevIns))
7073 return false;
7074
7075 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7076 {
7077 PDMCritSectEnter(&pThis->aCts[i].lock, VERR_INTERNAL_ERROR);
7078 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
7079 ataR3ResetDevice(&pThis->aCts[i].aIfs[j]);
7080 PDMCritSectLeave(&pThis->aCts[i].lock);
7081 }
7082 return true;
7083}
7084
7085
7086/**
7087 * Common reset worker for ataR3Reset and ataR3Construct.
7088 *
7089 * @returns VBox status.
7090 * @param pDevIns The device instance data.
7091 * @param fConstruct Indicates who is calling.
7092 */
7093static int ataR3ResetCommon(PPDMDEVINS pDevIns, bool fConstruct)
7094{
7095 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
7096
7097 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7098 {
7099 PDMCritSectEnter(&pThis->aCts[i].lock, VERR_INTERNAL_ERROR);
7100
7101 pThis->aCts[i].iSelectedIf = 0;
7102 pThis->aCts[i].iAIOIf = 0;
7103 pThis->aCts[i].BmDma.u8Cmd = 0;
7104 /* Report that both drives present on the bus are in DMA mode. This
7105 * pretends that there is a BIOS that has set it up. Normal reset
7106 * default is 0x00. */
7107 pThis->aCts[i].BmDma.u8Status = (pThis->aCts[i].aIfs[0].pDrvBase != NULL ? BM_STATUS_D0DMA : 0)
7108 | (pThis->aCts[i].aIfs[1].pDrvBase != NULL ? BM_STATUS_D1DMA : 0);
7109 pThis->aCts[i].BmDma.pvAddr = 0;
7110
7111 pThis->aCts[i].fReset = true;
7112 pThis->aCts[i].fRedo = false;
7113 pThis->aCts[i].fRedoIdle = false;
7114 ataR3AsyncIOClearRequests(&pThis->aCts[i]);
7115 Log2(("%s: Ctl#%d: message to async I/O thread, reset controller\n", __FUNCTION__, i));
7116 ataHCAsyncIOPutRequest(&pThis->aCts[i], &g_ataResetARequest);
7117 ataHCAsyncIOPutRequest(&pThis->aCts[i], &g_ataResetCRequest);
7118
7119 PDMCritSectLeave(&pThis->aCts[i].lock);
7120 }
7121
7122 int rcRet = VINF_SUCCESS;
7123 if (!fConstruct)
7124 {
7125 /*
7126 * Setup asynchronous notification completion if the requests haven't
7127 * completed yet.
7128 */
7129 if (!ataR3IsAsyncResetDone(pDevIns))
7130 PDMDevHlpSetAsyncNotification(pDevIns, ataR3IsAsyncResetDone);
7131 }
7132 else
7133 {
7134 /*
7135 * Wait for the requests for complete.
7136 *
7137 * Would be real nice if we could do it all from EMT(0) and not
7138 * involve the worker threads, then we could dispense with all the
7139 * waiting and semaphore ping-pong here...
7140 */
7141 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7142 {
7143 if (pThis->aCts[i].AsyncIOThread != NIL_RTTHREAD)
7144 {
7145 int rc = PDMCritSectEnter(&pThis->aCts[i].AsyncIORequestLock, VERR_IGNORED);
7146 AssertRC(rc);
7147
7148 ASMAtomicWriteBool(&pThis->aCts[i].fSignalIdle, true);
7149 rc = RTThreadUserReset(pThis->aCts[i].AsyncIOThread);
7150 AssertRC(rc);
7151
7152 rc = PDMCritSectLeave(&pThis->aCts[i].AsyncIORequestLock);
7153 AssertRC(rc);
7154
7155 if (!ataR3AsyncIOIsIdle(&pThis->aCts[i], false /*fStrict*/))
7156 {
7157 rc = RTThreadUserWait(pThis->aCts[i].AsyncIOThread, 30*1000 /*ms*/);
7158 if (RT_FAILURE(rc))
7159 rc = RTThreadUserWait(pThis->aCts[i].AsyncIOThread, 1000 /*ms*/);
7160 if (RT_FAILURE(rc))
7161 {
7162 AssertRC(rc);
7163 rcRet = rc;
7164 }
7165 }
7166 }
7167 ASMAtomicWriteBool(&pThis->aCts[i].fSignalIdle, false);
7168 }
7169 if (RT_SUCCESS(rcRet))
7170 {
7171 rcRet = ataR3IsAsyncResetDone(pDevIns) ? VINF_SUCCESS : VERR_INTERNAL_ERROR;
7172 AssertRC(rcRet);
7173 }
7174 }
7175 return rcRet;
7176}
7177
7178/**
7179 * Reset notification.
7180 *
7181 * @param pDevIns The device instance data.
7182 */
7183static DECLCALLBACK(void) ataR3Reset(PPDMDEVINS pDevIns)
7184{
7185 ataR3ResetCommon(pDevIns, false /*fConstruct*/);
7186}
7187
7188/**
7189 * @copydoc FNPDMDEVRELOCATE
7190 */
7191static DECLCALLBACK(void) ataR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
7192{
7193 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
7194
7195 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7196 {
7197 pThis->aCts[i].pDevInsRC += offDelta;
7198 pThis->aCts[i].aIfs[0].pDevInsRC += offDelta;
7199 pThis->aCts[i].aIfs[0].pControllerRC += offDelta;
7200 ataR3RelocBuffer(pDevIns, &pThis->aCts[i].aIfs[0]);
7201 pThis->aCts[i].aIfs[1].pDevInsRC += offDelta;
7202 pThis->aCts[i].aIfs[1].pControllerRC += offDelta;
7203 ataR3RelocBuffer(pDevIns, &pThis->aCts[i].aIfs[1]);
7204 }
7205}
7206
7207/**
7208 * Destroy a driver instance.
7209 *
7210 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
7211 * resources can be freed correctly.
7212 *
7213 * @param pDevIns The device instance data.
7214 */
7215static DECLCALLBACK(int) ataR3Destruct(PPDMDEVINS pDevIns)
7216{
7217 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
7218 int rc;
7219
7220 Log(("ataR3Destruct\n"));
7221 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
7222
7223 /*
7224 * Tell the async I/O threads to terminate.
7225 */
7226 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7227 {
7228 if (pThis->aCts[i].AsyncIOThread != NIL_RTTHREAD)
7229 {
7230 ASMAtomicWriteU32(&pThis->aCts[i].fShutdown, true);
7231 rc = SUPSemEventSignal(pThis->aCts[i].pSupDrvSession, pThis->aCts[i].hAsyncIOSem);
7232 AssertRC(rc);
7233 rc = RTSemEventSignal(pThis->aCts[i].SuspendIOSem);
7234 AssertRC(rc);
7235 }
7236 }
7237
7238 /*
7239 * Wait for the threads to terminate before destroying their resources.
7240 */
7241 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7242 {
7243 if (pThis->aCts[i].AsyncIOThread != NIL_RTTHREAD)
7244 {
7245 rc = RTThreadWait(pThis->aCts[i].AsyncIOThread, 30000 /* 30 s*/, NULL);
7246 if (RT_SUCCESS(rc))
7247 pThis->aCts[i].AsyncIOThread = NIL_RTTHREAD;
7248 else
7249 LogRel(("PIIX3 ATA Dtor: Ctl#%u is still executing, DevSel=%d AIOIf=%d CmdIf0=%#04x CmdIf1=%#04x rc=%Rrc\n",
7250 i, pThis->aCts[i].iSelectedIf, pThis->aCts[i].iAIOIf,
7251 pThis->aCts[i].aIfs[0].uATARegCommand, pThis->aCts[i].aIfs[1].uATARegCommand, rc));
7252 }
7253 }
7254
7255 /*
7256 * Free resources.
7257 */
7258 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7259 {
7260 if (PDMCritSectIsInitialized(&pThis->aCts[i].AsyncIORequestLock))
7261 PDMR3CritSectDelete(&pThis->aCts[i].AsyncIORequestLock);
7262 if (pThis->aCts[i].hAsyncIOSem != NIL_SUPSEMEVENT)
7263 {
7264 SUPSemEventClose(pThis->aCts[i].pSupDrvSession, pThis->aCts[i].hAsyncIOSem);
7265 pThis->aCts[i].hAsyncIOSem = NIL_SUPSEMEVENT;
7266 }
7267 if (pThis->aCts[i].SuspendIOSem != NIL_RTSEMEVENT)
7268 {
7269 RTSemEventDestroy(pThis->aCts[i].SuspendIOSem);
7270 pThis->aCts[i].SuspendIOSem = NIL_RTSEMEVENT;
7271 }
7272
7273 /* try one final time */
7274 if (pThis->aCts[i].AsyncIOThread != NIL_RTTHREAD)
7275 {
7276 rc = RTThreadWait(pThis->aCts[i].AsyncIOThread, 1 /*ms*/, NULL);
7277 if (RT_SUCCESS(rc))
7278 {
7279 pThis->aCts[i].AsyncIOThread = NIL_RTTHREAD;
7280 LogRel(("PIIX3 ATA Dtor: Ctl#%u actually completed.\n", i));
7281 }
7282 }
7283
7284 for (uint32_t iIf = 0; iIf < RT_ELEMENTS(pThis->aCts[i].aIfs); iIf++)
7285 {
7286 if (pThis->aCts[i].aIfs[iIf].pTrackList)
7287 {
7288 ATAPIPassthroughTrackListDestroy(pThis->aCts[i].aIfs[iIf].pTrackList);
7289 pThis->aCts[i].aIfs[iIf].pTrackList = NULL;
7290 }
7291 }
7292 }
7293
7294 return VINF_SUCCESS;
7295}
7296
7297/**
7298 * Convert config value to DEVPCBIOSBOOT.
7299 *
7300 * @returns VBox status code.
7301 * @param pDevIns The device instance data.
7302 * @param pCfg Configuration handle.
7303 * @param penmChipset Where to store the chipset type.
7304 */
7305static int ataR3ControllerFromCfg(PPDMDEVINS pDevIns, PCFGMNODE pCfg, CHIPSET *penmChipset)
7306{
7307 char szType[20];
7308
7309 int rc = CFGMR3QueryStringDef(pCfg, "Type", &szType[0], sizeof(szType), "PIIX4");
7310 if (RT_FAILURE(rc))
7311 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
7312 N_("Configuration error: Querying \"Type\" as a string failed"));
7313 if (!strcmp(szType, "PIIX3"))
7314 *penmChipset = CHIPSET_PIIX3;
7315 else if (!strcmp(szType, "PIIX4"))
7316 *penmChipset = CHIPSET_PIIX4;
7317 else if (!strcmp(szType, "ICH6"))
7318 *penmChipset = CHIPSET_ICH6;
7319 else
7320 {
7321 PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
7322 N_("Configuration error: The \"Type\" value \"%s\" is unknown"),
7323 szType);
7324 rc = VERR_INTERNAL_ERROR;
7325 }
7326 return rc;
7327}
7328
7329/**
7330 * @interface_method_impl{PDMDEVREG,pfnConstruct}
7331 */
7332static DECLCALLBACK(int) ataR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
7333{
7334 PCIATAState *pThis = PDMINS_2_DATA(pDevIns, PCIATAState *);
7335 PPDMIBASE pBase;
7336 int rc;
7337 bool fRCEnabled;
7338 bool fR0Enabled;
7339 uint32_t DelayIRQMillies;
7340
7341 Assert(iInstance == 0);
7342 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
7343
7344 /*
7345 * Initialize NIL handle values (for the destructor).
7346 */
7347 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7348 {
7349 pThis->aCts[i].hAsyncIOSem = NIL_SUPSEMEVENT;
7350 pThis->aCts[i].SuspendIOSem = NIL_RTSEMEVENT;
7351 pThis->aCts[i].AsyncIOThread = NIL_RTTHREAD;
7352 }
7353
7354 /*
7355 * Validate and read configuration.
7356 */
7357 if (!CFGMR3AreValuesValid(pCfg,
7358 "GCEnabled\0"
7359 "R0Enabled\0"
7360 "IRQDelay\0"
7361 "Type\0")
7362 /** @todo || invalid keys */)
7363 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
7364 N_("PIIX3 configuration error: unknown option specified"));
7365
7366 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &fRCEnabled, true);
7367 if (RT_FAILURE(rc))
7368 return PDMDEV_SET_ERROR(pDevIns, rc,
7369 N_("PIIX3 configuration error: failed to read GCEnabled as boolean"));
7370 Log(("%s: fRCEnabled=%d\n", __FUNCTION__, fRCEnabled));
7371
7372 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &fR0Enabled, true);
7373 if (RT_FAILURE(rc))
7374 return PDMDEV_SET_ERROR(pDevIns, rc,
7375 N_("PIIX3 configuration error: failed to read R0Enabled as boolean"));
7376 Log(("%s: fR0Enabled=%d\n", __FUNCTION__, fR0Enabled));
7377
7378 rc = CFGMR3QueryU32Def(pCfg, "IRQDelay", &DelayIRQMillies, 0);
7379 if (RT_FAILURE(rc))
7380 return PDMDEV_SET_ERROR(pDevIns, rc,
7381 N_("PIIX3 configuration error: failed to read IRQDelay as integer"));
7382 Log(("%s: DelayIRQMillies=%d\n", __FUNCTION__, DelayIRQMillies));
7383 Assert(DelayIRQMillies < 50);
7384
7385 CHIPSET enmChipset = CHIPSET_PIIX3;
7386 rc = ataR3ControllerFromCfg(pDevIns, pCfg, &enmChipset);
7387 if (RT_FAILURE(rc))
7388 return rc;
7389 pThis->u8Type = (uint8_t)enmChipset;
7390
7391 /*
7392 * Initialize data (most of it anyway).
7393 */
7394 /* Status LUN. */
7395 pThis->IBase.pfnQueryInterface = ataR3Status_QueryInterface;
7396 pThis->ILeds.pfnQueryStatusLed = ataR3Status_QueryStatusLed;
7397
7398 /* PCI configuration space. */
7399 PCIDevSetVendorId(&pThis->dev, 0x8086); /* Intel */
7400
7401 /*
7402 * When adding more IDE chipsets, don't forget to update pci_bios_init_device()
7403 * as it explicitly checks for PCI id for IDE controllers.
7404 */
7405 switch (pThis->u8Type)
7406 {
7407 case CHIPSET_ICH6:
7408 PCIDevSetDeviceId(&pThis->dev, 0x269e); /* ICH6 IDE */
7409 /** @todo: do we need it? Do we need anything else? */
7410 pThis->dev.config[0x48] = 0x00; /* UDMACTL */
7411 pThis->dev.config[0x4A] = 0x00; /* UDMATIM */
7412 pThis->dev.config[0x4B] = 0x00;
7413 {
7414 /*
7415 * See www.intel.com/Assets/PDF/manual/298600.pdf p. 30
7416 * Report
7417 * WR_Ping-Pong_EN: must be set
7418 * PCR0, PCR1: 80-pin primary cable reporting for both disks
7419 * SCR0, SCR1: 80-pin secondary cable reporting for both disks
7420 */
7421 uint16_t u16Config = (1<<10) | (1<<7) | (1<<6) | (1<<5) | (1<<4) ;
7422 pThis->dev.config[0x54] = u16Config & 0xff;
7423 pThis->dev.config[0x55] = u16Config >> 8;
7424 }
7425 break;
7426 case CHIPSET_PIIX4:
7427 PCIDevSetDeviceId(&pThis->dev, 0x7111); /* PIIX4 IDE */
7428 PCIDevSetRevisionId(&pThis->dev, 0x01); /* PIIX4E */
7429 pThis->dev.config[0x48] = 0x00; /* UDMACTL */
7430 pThis->dev.config[0x4A] = 0x00; /* UDMATIM */
7431 pThis->dev.config[0x4B] = 0x00;
7432 break;
7433 case CHIPSET_PIIX3:
7434 PCIDevSetDeviceId(&pThis->dev, 0x7010); /* PIIX3 IDE */
7435 break;
7436 default:
7437 AssertMsgFailed(("Unsupported IDE chipset type: %d\n", pThis->u8Type));
7438 }
7439
7440 PCIDevSetCommand( &pThis->dev, PCI_COMMAND_IOACCESS | PCI_COMMAND_MEMACCESS | PCI_COMMAND_BUSMASTER);
7441 PCIDevSetClassProg( &pThis->dev, 0x8a); /* programming interface = PCI_IDE bus master is supported */
7442 PCIDevSetClassSub( &pThis->dev, 0x01); /* class_sub = PCI_IDE */
7443 PCIDevSetClassBase( &pThis->dev, 0x01); /* class_base = PCI_mass_storage */
7444 PCIDevSetHeaderType(&pThis->dev, 0x00);
7445
7446 pThis->pDevIns = pDevIns;
7447 pThis->fRCEnabled = fRCEnabled;
7448 pThis->fR0Enabled = fR0Enabled;
7449 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7450 {
7451 pThis->aCts[i].pDevInsR3 = pDevIns;
7452 pThis->aCts[i].pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
7453 pThis->aCts[i].pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
7454 pThis->aCts[i].DelayIRQMillies = (uint32_t)DelayIRQMillies;
7455 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
7456 {
7457 ATADevState *pIf = &pThis->aCts[i].aIfs[j];
7458
7459 pIf->iLUN = i * RT_ELEMENTS(pThis->aCts) + j;
7460 pIf->pDevInsR3 = pDevIns;
7461 pIf->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
7462 pIf->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
7463 pIf->pControllerR3 = &pThis->aCts[i];
7464 pIf->pControllerR0 = MMHyperR3ToR0(PDMDevHlpGetVM(pDevIns), &pThis->aCts[i]);
7465 pIf->pControllerRC = MMHyperR3ToRC(PDMDevHlpGetVM(pDevIns), &pThis->aCts[i]);
7466 pIf->IBase.pfnQueryInterface = ataR3QueryInterface;
7467 pIf->IMountNotify.pfnMountNotify = ataR3MountNotify;
7468 pIf->IMountNotify.pfnUnmountNotify = ataR3UnmountNotify;
7469 pIf->IPort.pfnQueryDeviceLocation = ataR3QueryDeviceLocation;
7470 pIf->Led.u32Magic = PDMLED_MAGIC;
7471 }
7472 }
7473
7474 Assert(RT_ELEMENTS(pThis->aCts) == 2);
7475 pThis->aCts[0].irq = 14;
7476 pThis->aCts[0].IOPortBase1 = 0x1f0;
7477 pThis->aCts[0].IOPortBase2 = 0x3f6;
7478 pThis->aCts[1].irq = 15;
7479 pThis->aCts[1].IOPortBase1 = 0x170;
7480 pThis->aCts[1].IOPortBase2 = 0x376;
7481
7482 /*
7483 * Set the default critical section to NOP as we lock on controller level.
7484 */
7485 rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
7486 AssertRCReturn(rc, rc);
7487
7488 /*
7489 * Register the PCI device.
7490 * N.B. There's a hack in the PIIX3 PCI bridge device to assign this
7491 * device the slot next to itself.
7492 */
7493 rc = PDMDevHlpPCIRegister(pDevIns, &pThis->dev);
7494 if (RT_FAILURE(rc))
7495 return PDMDEV_SET_ERROR(pDevIns, rc,
7496 N_("PIIX3 cannot register PCI device"));
7497 //AssertMsg(pThis->dev.devfn == 9 || iInstance != 0, ("pThis->dev.devfn=%d\n", pThis->dev.devfn));
7498 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 4, 0x10, PCI_ADDRESS_SPACE_IO, ataR3BMDMAIORangeMap);
7499 if (RT_FAILURE(rc))
7500 return PDMDEV_SET_ERROR(pDevIns, rc,
7501 N_("PIIX3 cannot register PCI I/O region for BMDMA"));
7502
7503 /*
7504 * Register the I/O ports.
7505 * The ports are all hardcoded and enforced by the PIIX3 host bridge controller.
7506 */
7507 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7508 {
7509 rc = PDMDevHlpIOPortRegister(pDevIns, pThis->aCts[i].IOPortBase1, 1, (RTHCPTR)(uintptr_t)i,
7510 ataIOPortWrite1Data, ataIOPortRead1Data,
7511 ataIOPortWriteStr1Data, ataIOPortReadStr1Data, "ATA I/O Base 1 - Data");
7512 AssertLogRelRCReturn(rc, rc);
7513 rc = PDMDevHlpIOPortRegister(pDevIns, pThis->aCts[i].IOPortBase1 + 1, 7, (RTHCPTR)(uintptr_t)i,
7514 ataIOPortWrite1Other, ataIOPortRead1Other, NULL, NULL, "ATA I/O Base 1 - Other");
7515
7516 AssertLogRelRCReturn(rc, rc);
7517 if (fRCEnabled)
7518 {
7519 rc = PDMDevHlpIOPortRegisterRC(pDevIns, pThis->aCts[i].IOPortBase1, 1, (RTGCPTR)i,
7520 "ataIOPortWrite1Data", "ataIOPortRead1Data",
7521 "ataIOPortWriteStr1Data", "ataIOPortReadStr1Data", "ATA I/O Base 1 - Data");
7522 AssertLogRelRCReturn(rc, rc);
7523 rc = PDMDevHlpIOPortRegisterRC(pDevIns, pThis->aCts[i].IOPortBase1 + 1, 7, (RTGCPTR)i,
7524 "ataIOPortWrite1Other", "ataIOPortRead1Other", NULL, NULL, "ATA I/O Base 1 - Other");
7525 AssertLogRelRCReturn(rc, rc);
7526 }
7527
7528 if (fR0Enabled)
7529 {
7530#if 0
7531 rc = PDMDevHlpIOPortRegisterR0(pDevIns, pThis->aCts[i].IOPortBase1, 1, (RTR0PTR)i,
7532 "ataIOPortWrite1Data", "ataIOPortRead1Data", NULL, NULL, "ATA I/O Base 1 - Data");
7533#else
7534 rc = PDMDevHlpIOPortRegisterR0(pDevIns, pThis->aCts[i].IOPortBase1, 1, (RTR0PTR)i,
7535 "ataIOPortWrite1Data", "ataIOPortRead1Data",
7536 "ataIOPortWriteStr1Data", "ataIOPortReadStr1Data", "ATA I/O Base 1 - Data");
7537#endif
7538 AssertLogRelRCReturn(rc, rc);
7539 rc = PDMDevHlpIOPortRegisterR0(pDevIns, pThis->aCts[i].IOPortBase1 + 1, 7, (RTR0PTR)i,
7540 "ataIOPortWrite1Other", "ataIOPortRead1Other", NULL, NULL, "ATA I/O Base 1 - Other");
7541 AssertLogRelRCReturn(rc, rc);
7542 }
7543
7544 rc = PDMDevHlpIOPortRegister(pDevIns, pThis->aCts[i].IOPortBase2, 1, (RTHCPTR)(uintptr_t)i,
7545 ataIOPortWrite2, ataIOPortRead2, NULL, NULL, "ATA I/O Base 2");
7546 if (RT_FAILURE(rc))
7547 return PDMDEV_SET_ERROR(pDevIns, rc, N_("PIIX3 cannot register base2 I/O handlers"));
7548
7549 if (fRCEnabled)
7550 {
7551 rc = PDMDevHlpIOPortRegisterRC(pDevIns, pThis->aCts[i].IOPortBase2, 1, (RTGCPTR)i,
7552 "ataIOPortWrite2", "ataIOPortRead2", NULL, NULL, "ATA I/O Base 2");
7553 if (RT_FAILURE(rc))
7554 return PDMDEV_SET_ERROR(pDevIns, rc, N_("PIIX3 cannot register base2 I/O handlers (GC)"));
7555 }
7556 if (fR0Enabled)
7557 {
7558 rc = PDMDevHlpIOPortRegisterR0(pDevIns, pThis->aCts[i].IOPortBase2, 1, (RTR0PTR)i,
7559 "ataIOPortWrite2", "ataIOPortRead2", NULL, NULL, "ATA I/O Base 2");
7560 if (RT_FAILURE(rc))
7561 return PDMDEV_SET_ERROR(pDevIns, rc, N_("PIIX3 cannot register base2 I/O handlers (R0)"));
7562 }
7563
7564 for (uint32_t j = 0; j < RT_ELEMENTS(pThis->aCts[i].aIfs); j++)
7565 {
7566 ATADevState *pIf = &pThis->aCts[i].aIfs[j];
7567 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatATADMA, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
7568 "Number of ATA DMA transfers.", "/Devices/IDE%d/ATA%d/Unit%d/DMA", iInstance, i, j);
7569 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatATAPIO, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
7570 "Number of ATA PIO transfers.", "/Devices/IDE%d/ATA%d/Unit%d/PIO", iInstance, i, j);
7571 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatATAPIDMA, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
7572 "Number of ATAPI DMA transfers.", "/Devices/IDE%d/ATA%d/Unit%d/AtapiDMA", iInstance, i, j);
7573 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatATAPIPIO, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
7574 "Number of ATAPI PIO transfers.", "/Devices/IDE%d/ATA%d/Unit%d/AtapiPIO", iInstance, i, j);
7575#ifdef VBOX_WITH_STATISTICS /** @todo release too. */
7576 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatReads, STAMTYPE_PROFILE_ADV, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL,
7577 "Profiling of the read operations.", "/Devices/IDE%d/ATA%d/Unit%d/Reads", iInstance, i, j);
7578#endif
7579 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatBytesRead, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
7580 "Amount of data read.", "/Devices/IDE%d/ATA%d/Unit%d/ReadBytes", iInstance, i, j);
7581#ifdef VBOX_INSTRUMENT_DMA_WRITES
7582 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatInstrVDWrites,STAMTYPE_PROFILE_ADV, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL,
7583 "Profiling of the VD DMA write operations.", "/Devices/IDE%d/ATA%d/Unit%d/InstrVDWrites", iInstance, i, j);
7584#endif
7585#ifdef VBOX_WITH_STATISTICS
7586 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatWrites, STAMTYPE_PROFILE_ADV, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL,
7587 "Profiling of the write operations.", "/Devices/IDE%d/ATA%d/Unit%d/Writes", iInstance, i, j);
7588#endif
7589 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatBytesWritten, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
7590 "Amount of data written.", "/Devices/IDE%d/ATA%d/Unit%d/WrittenBytes", iInstance, i, j);
7591#ifdef VBOX_WITH_STATISTICS
7592 PDMDevHlpSTAMRegisterF(pDevIns, &pIf->StatFlushes, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL,
7593 "Profiling of the flush operations.", "/Devices/IDE%d/ATA%d/Unit%d/Flushes", iInstance, i, j);
7594#endif
7595 }
7596#ifdef VBOX_WITH_STATISTICS /** @todo release too. */
7597 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aCts[i].StatAsyncOps, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
7598 "The number of async operations.", "/Devices/IDE%d/ATA%d/Async/Operations", iInstance, i);
7599 /** @todo STAMUNIT_MICROSECS */
7600 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aCts[i].StatAsyncMinWait, STAMTYPE_U64_RESET, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE,
7601 "Minimum wait in microseconds.", "/Devices/IDE%d/ATA%d/Async/MinWait", iInstance, i);
7602 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aCts[i].StatAsyncMaxWait, STAMTYPE_U64_RESET, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE,
7603 "Maximum wait in microseconds.", "/Devices/IDE%d/ATA%d/Async/MaxWait", iInstance, i);
7604 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aCts[i].StatAsyncTimeUS, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE,
7605 "Total time spent in microseconds.", "/Devices/IDE%d/ATA%d/Async/TotalTimeUS", iInstance, i);
7606 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aCts[i].StatAsyncTime, STAMTYPE_PROFILE_ADV, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL,
7607 "Profiling of async operations.", "/Devices/IDE%d/ATA%d/Async/Time", iInstance, i);
7608 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aCts[i].StatLockWait, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL,
7609 "Profiling of locks.", "/Devices/IDE%d/ATA%d/Async/LockWait", iInstance, i);
7610#endif /* VBOX_WITH_STATISTICS */
7611
7612 /* Initialize per-controller critical section. */
7613 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->aCts[i].lock, RT_SRC_POS, "ATA#%u-Ctl", i);
7614 AssertLogRelRCReturn(rc, rc);
7615
7616 /* Initialize per-controller async I/O request critical section. */
7617 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->aCts[i].AsyncIORequestLock, RT_SRC_POS, "ATA#%u-Req", i);
7618 AssertLogRelRCReturn(rc, rc);
7619 }
7620
7621 /*
7622 * Attach status driver (optional).
7623 */
7624 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
7625 if (RT_SUCCESS(rc))
7626 {
7627 pThis->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
7628 pThis->pMediaNotify = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIANOTIFY);
7629 }
7630 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
7631 {
7632 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
7633 return PDMDEV_SET_ERROR(pDevIns, rc, N_("PIIX3 cannot attach to status driver"));
7634 }
7635
7636 /*
7637 * Attach the units.
7638 */
7639 uint32_t cbTotalBuffer = 0;
7640 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aCts); i++)
7641 {
7642 PATACONTROLLER pCtl = &pThis->aCts[i];
7643
7644 /*
7645 * Start the worker thread.
7646 */
7647 pCtl->uAsyncIOState = ATA_AIO_NEW;
7648 pCtl->pSupDrvSession = PDMDevHlpGetSupDrvSession(pDevIns);
7649 rc = SUPSemEventCreate(pCtl->pSupDrvSession, &pCtl->hAsyncIOSem);
7650 AssertLogRelRCReturn(rc, rc);
7651 rc = RTSemEventCreate(&pCtl->SuspendIOSem);
7652 AssertLogRelRCReturn(rc, rc);
7653
7654 ataR3AsyncIOClearRequests(pCtl);
7655 rc = RTThreadCreateF(&pCtl->AsyncIOThread, ataR3AsyncIOThread, (void *)pCtl, 128*1024 /*cbStack*/,
7656 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "ATA-%u", i);
7657 AssertLogRelRCReturn(rc, rc);
7658 Assert( pCtl->AsyncIOThread != NIL_RTTHREAD && pCtl->hAsyncIOSem != NIL_SUPSEMEVENT
7659 && pCtl->SuspendIOSem != NIL_RTSEMEVENT && PDMCritSectIsInitialized(&pCtl->AsyncIORequestLock));
7660 Log(("%s: controller %d AIO thread id %#x; sem %p susp_sem %p\n", __FUNCTION__, i, pCtl->AsyncIOThread, pCtl->hAsyncIOSem, pCtl->SuspendIOSem));
7661
7662 for (uint32_t j = 0; j < RT_ELEMENTS(pCtl->aIfs); j++)
7663 {
7664 static const char *s_apszDescs[RT_ELEMENTS(pThis->aCts)][RT_ELEMENTS(pCtl->aIfs)] =
7665 {
7666 { "Primary Master", "Primary Slave" },
7667 { "Secondary Master", "Secondary Slave" }
7668 };
7669
7670 /*
7671 * Try attach the block device and get the interfaces,
7672 * required as well as optional.
7673 */
7674 ATADevState *pIf = &pCtl->aIfs[j];
7675
7676 rc = PDMDevHlpDriverAttach(pDevIns, pIf->iLUN, &pIf->IBase, &pIf->pDrvBase, s_apszDescs[i][j]);
7677 if (RT_SUCCESS(rc))
7678 {
7679 rc = ataR3ConfigLun(pDevIns, pIf);
7680 if (RT_SUCCESS(rc))
7681 {
7682 /*
7683 * Init vendor product data.
7684 */
7685 static const char *s_apszCFGMKeys[RT_ELEMENTS(pThis->aCts)][RT_ELEMENTS(pCtl->aIfs)] =
7686 {
7687 { "PrimaryMaster", "PrimarySlave" },
7688 { "SecondaryMaster", "SecondarySlave" }
7689 };
7690
7691 /* Generate a default serial number. */
7692 char szSerial[ATA_SERIAL_NUMBER_LENGTH+1];
7693 RTUUID Uuid;
7694 if (pIf->pDrvBlock)
7695 rc = pIf->pDrvBlock->pfnGetUuid(pIf->pDrvBlock, &Uuid);
7696 else
7697 RTUuidClear(&Uuid);
7698
7699 if (RT_FAILURE(rc) || RTUuidIsNull(&Uuid))
7700 {
7701 /* Generate a predictable serial for drives which don't have a UUID. */
7702 RTStrPrintf(szSerial, sizeof(szSerial), "VB%x-%04x%04x",
7703 pIf->iLUN + pDevIns->iInstance * 32,
7704 pThis->aCts[i].IOPortBase1, pThis->aCts[i].IOPortBase2);
7705 }
7706 else
7707 RTStrPrintf(szSerial, sizeof(szSerial), "VB%08x-%08x", Uuid.au32[0], Uuid.au32[3]);
7708
7709 /* Get user config if present using defaults otherwise. */
7710 PCFGMNODE pCfgNode = CFGMR3GetChild(pCfg, s_apszCFGMKeys[i][j]);
7711 rc = CFGMR3QueryStringDef(pCfgNode, "SerialNumber", pIf->szSerialNumber, sizeof(pIf->szSerialNumber),
7712 szSerial);
7713 if (RT_FAILURE(rc))
7714 {
7715 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
7716 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
7717 N_("PIIX3 configuration error: \"SerialNumber\" is longer than 20 bytes"));
7718 return PDMDEV_SET_ERROR(pDevIns, rc,
7719 N_("PIIX3 configuration error: failed to read \"SerialNumber\" as string"));
7720 }
7721
7722 rc = CFGMR3QueryStringDef(pCfgNode, "FirmwareRevision", pIf->szFirmwareRevision, sizeof(pIf->szFirmwareRevision),
7723 "1.0");
7724 if (RT_FAILURE(rc))
7725 {
7726 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
7727 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
7728 N_("PIIX3 configuration error: \"FirmwareRevision\" is longer than 8 bytes"));
7729 return PDMDEV_SET_ERROR(pDevIns, rc,
7730 N_("PIIX3 configuration error: failed to read \"FirmwareRevision\" as string"));
7731 }
7732
7733 rc = CFGMR3QueryStringDef(pCfgNode, "ModelNumber", pIf->szModelNumber, sizeof(pIf->szModelNumber),
7734 pIf->fATAPI ? "VBOX CD-ROM" : "VBOX HARDDISK");
7735 if (RT_FAILURE(rc))
7736 {
7737 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
7738 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
7739 N_("PIIX3 configuration error: \"ModelNumber\" is longer than 40 bytes"));
7740 return PDMDEV_SET_ERROR(pDevIns, rc,
7741 N_("PIIX3 configuration error: failed to read \"ModelNumber\" as string"));
7742 }
7743
7744 rc = CFGMR3QueryBoolDef(pCfgNode, "NonRotationalMedium", &pIf->fNonRotational, false);
7745 if (RT_FAILURE(rc))
7746 return PDMDEV_SET_ERROR(pDevIns, rc,
7747 N_("PIIX3 configuration error: failed to read \"NonRotationalMedium\" as boolean"));
7748
7749 /* There are three other identification strings for CD drives used for INQUIRY */
7750 if (pIf->fATAPI)
7751 {
7752 rc = CFGMR3QueryStringDef(pCfgNode, "ATAPIVendorId", pIf->szInquiryVendorId, sizeof(pIf->szInquiryVendorId),
7753 "VBOX");
7754 if (RT_FAILURE(rc))
7755 {
7756 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
7757 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
7758 N_("PIIX3 configuration error: \"ATAPIVendorId\" is longer than 16 bytes"));
7759 return PDMDEV_SET_ERROR(pDevIns, rc,
7760 N_("PIIX3 configuration error: failed to read \"ATAPIVendorId\" as string"));
7761 }
7762
7763 rc = CFGMR3QueryStringDef(pCfgNode, "ATAPIProductId", pIf->szInquiryProductId, sizeof(pIf->szInquiryProductId),
7764 "CD-ROM");
7765 if (RT_FAILURE(rc))
7766 {
7767 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
7768 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
7769 N_("PIIX3 configuration error: \"ATAPIProductId\" is longer than 16 bytes"));
7770 return PDMDEV_SET_ERROR(pDevIns, rc,
7771 N_("PIIX3 configuration error: failed to read \"ATAPIProductId\" as string"));
7772 }
7773
7774 rc = CFGMR3QueryStringDef(pCfgNode, "ATAPIRevision", pIf->szInquiryRevision, sizeof(pIf->szInquiryRevision),
7775 "1.0");
7776 if (RT_FAILURE(rc))
7777 {
7778 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
7779 return PDMDEV_SET_ERROR(pDevIns, VERR_INVALID_PARAMETER,
7780 N_("PIIX3 configuration error: \"ATAPIRevision\" is longer than 4 bytes"));
7781 return PDMDEV_SET_ERROR(pDevIns, rc,
7782 N_("PIIX3 configuration error: failed to read \"ATAPIRevision\" as string"));
7783 }
7784
7785 rc = CFGMR3QueryBoolDef(pCfgNode, "OverwriteInquiry", &pIf->fOverwriteInquiry, true);
7786 if (RT_FAILURE(rc))
7787 return PDMDEV_SET_ERROR(pDevIns, rc,
7788 N_("PIIX3 configuration error: failed to read \"OverwriteInquiry\" as boolean"));
7789 }
7790 }
7791
7792 }
7793 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
7794 {
7795 pIf->pDrvBase = NULL;
7796 pIf->pDrvBlock = NULL;
7797 pIf->cbIOBuffer = 0;
7798 pIf->pbIOBufferR3 = NULL;
7799 pIf->pbIOBufferR0 = NIL_RTR0PTR;
7800 pIf->pbIOBufferRC = NIL_RTGCPTR;
7801 LogRel(("PIIX3 ATA: LUN#%d: no unit\n", pIf->iLUN));
7802 }
7803 else
7804 {
7805 switch (rc)
7806 {
7807 case VERR_ACCESS_DENIED:
7808 /* Error already cached by DrvHostBase */
7809 return rc;
7810 default:
7811 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
7812 N_("PIIX3 cannot attach drive to the %s"),
7813 s_apszDescs[i][j]);
7814 }
7815 }
7816 cbTotalBuffer += pIf->cbIOBuffer;
7817 }
7818 }
7819
7820 rc = PDMDevHlpSSMRegisterEx(pDevIns, ATA_SAVED_STATE_VERSION, sizeof(*pThis) + cbTotalBuffer, NULL,
7821 NULL, ataR3LiveExec, NULL,
7822 ataR3SaveLoadPrep, ataR3SaveExec, NULL,
7823 ataR3SaveLoadPrep, ataR3LoadExec, NULL);
7824 if (RT_FAILURE(rc))
7825 return PDMDEV_SET_ERROR(pDevIns, rc, N_("PIIX3 cannot register save state handlers"));
7826
7827 /*
7828 * Initialize the device state.
7829 */
7830 return ataR3ResetCommon(pDevIns, true /*fConstruct*/);
7831}
7832
7833
7834/**
7835 * The device registration structure.
7836 */
7837const PDMDEVREG g_DevicePIIX3IDE =
7838{
7839 /* u32Version */
7840 PDM_DEVREG_VERSION,
7841 /* szName */
7842 "piix3ide",
7843 /* szRCMod */
7844 "VBoxDDRC.rc",
7845 /* szR0Mod */
7846 "VBoxDDR0.r0",
7847 /* pszDescription */
7848 "Intel PIIX3 ATA controller.\n"
7849 " LUN #0 is primary master.\n"
7850 " LUN #1 is primary slave.\n"
7851 " LUN #2 is secondary master.\n"
7852 " LUN #3 is secondary slave.\n"
7853 " LUN #999 is the LED/Status connector.",
7854 /* fFlags */
7855 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0 |
7856 PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION | PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION |
7857 PDM_DEVREG_FLAGS_FIRST_RESET_NOTIFICATION,
7858 /* fClass */
7859 PDM_DEVREG_CLASS_STORAGE,
7860 /* cMaxInstances */
7861 1,
7862 /* cbInstance */
7863 sizeof(PCIATAState),
7864 /* pfnConstruct */
7865 ataR3Construct,
7866 /* pfnDestruct */
7867 ataR3Destruct,
7868 /* pfnRelocate */
7869 ataR3Relocate,
7870 /* pfnMemSetup */
7871 NULL,
7872 /* pfnPowerOn */
7873 NULL,
7874 /* pfnReset */
7875 ataR3Reset,
7876 /* pfnSuspend */
7877 ataR3Suspend,
7878 /* pfnResume */
7879 ataR3Resume,
7880 /* pfnAttach */
7881 ataR3Attach,
7882 /* pfnDetach */
7883 ataR3Detach,
7884 /* pfnQueryInterface. */
7885 NULL,
7886 /* pfnInitComplete */
7887 NULL,
7888 /* pfnPowerOff */
7889 ataR3PowerOff,
7890 /* pfnSoftReset */
7891 NULL,
7892 /* u32VersionEnd */
7893 PDM_DEVREG_VERSION
7894};
7895#endif /* IN_RING3 */
7896#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