VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DevBusLogic.cpp@ 47143

Last change on this file since 47143 was 46601, checked in by vboxsync, 12 years ago

BusLogic: Reset a little more.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 159.0 KB
Line 
1/* $Id: DevBusLogic.cpp 46601 2013-06-17 16:05:23Z vboxsync $ */
2/** @file
3 * VBox storage devices - BusLogic SCSI host adapter BT-958.
4 *
5 * Based on the Multi-Master Ultra SCSI Systems Technical Reference Manual.
6 */
7
8/*
9 * Copyright (C) 2006-2013 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20
21/*******************************************************************************
22* Header Files *
23*******************************************************************************/
24#define LOG_GROUP LOG_GROUP_DEV_BUSLOGIC
25#include <VBox/vmm/pdmdev.h>
26#include <VBox/vmm/pdmifs.h>
27#include <VBox/vmm/pdmcritsect.h>
28#include <VBox/scsi.h>
29#include <iprt/asm.h>
30#include <iprt/assert.h>
31#include <iprt/string.h>
32#include <iprt/log.h>
33#ifdef IN_RING3
34# include <iprt/alloc.h>
35# include <iprt/memcache.h>
36# include <iprt/param.h>
37# include <iprt/uuid.h>
38#endif
39
40#include "VBoxSCSI.h"
41#include "VBoxDD.h"
42
43
44/*******************************************************************************
45* Defined Constants And Macros *
46*******************************************************************************/
47/** Maximum number of attached devices the adapter can handle. */
48#define BUSLOGIC_MAX_DEVICES 16
49
50/** Maximum number of scatter gather elements this device can handle. */
51#define BUSLOGIC_MAX_SCATTER_GATHER_LIST_SIZE 128
52
53/** Size of the command buffer. */
54#define BUSLOGIC_COMMAND_SIZE_MAX 5
55
56/** Size of the reply buffer. */
57#define BUSLOGIC_REPLY_SIZE_MAX 64
58
59/** Custom fixed I/O ports for BIOS controller access.
60 * Note that these should not be in the ISA range (below 400h) to avoid
61 * conflicts with ISA device probing. Addresses in the 300h-340h range should be
62 * especially avoided.
63 */
64#define BUSLOGIC_BIOS_IO_PORT 0x430
65
66/** State saved version. */
67#define BUSLOGIC_SAVED_STATE_MINOR_VERSION 3
68
69/** Saved state version before the suspend on error feature was implemented. */
70#define BUSLOGIC_SAVED_STATE_MINOR_PRE_ERROR_HANDLING 1
71/** Saved state version before 24-bit mailbox support was implemented. */
72#define BUSLOGIC_SAVED_STATE_MINOR_PRE_24BIT_MBOX 2
73
74/** The duration of software-initiated reset (in nano seconds).
75 * Not documented, set to 50 ms. */
76#define BUSLOGIC_RESET_DURATION_NS UINT64_C(50000000)
77
78
79/*******************************************************************************
80* Structures and Typedefs *
81*******************************************************************************/
82/**
83 * State of a device attached to the buslogic host adapter.
84 *
85 * @implements PDMIBASE
86 * @implements PDMISCSIPORT
87 * @implements PDMILEDPORTS
88 */
89typedef struct BUSLOGICDEVICE
90{
91 /** Pointer to the owning buslogic device instance. - R3 pointer */
92 R3PTRTYPE(struct BUSLOGIC *) pBusLogicR3;
93 /** Pointer to the owning buslogic device instance. - R0 pointer */
94 R0PTRTYPE(struct BUSLOGIC *) pBusLogicR0;
95 /** Pointer to the owning buslogic device instance. - RC pointer */
96 RCPTRTYPE(struct BUSLOGIC *) pBusLogicRC;
97
98 /** Flag whether device is present. */
99 bool fPresent;
100 /** LUN of the device. */
101 RTUINT iLUN;
102
103#if HC_ARCH_BITS == 64
104 uint32_t Alignment0;
105#endif
106
107 /** Our base interface. */
108 PDMIBASE IBase;
109 /** SCSI port interface. */
110 PDMISCSIPORT ISCSIPort;
111 /** Led interface. */
112 PDMILEDPORTS ILed;
113 /** Pointer to the attached driver's base interface. */
114 R3PTRTYPE(PPDMIBASE) pDrvBase;
115 /** Pointer to the underlying SCSI connector interface. */
116 R3PTRTYPE(PPDMISCSICONNECTOR) pDrvSCSIConnector;
117 /** The status LED state for this device. */
118 PDMLED Led;
119
120#if HC_ARCH_BITS == 64
121 uint32_t Alignment1;
122#endif
123
124 /** Number of outstanding tasks on the port. */
125 volatile uint32_t cOutstandingRequests;
126
127} BUSLOGICDEVICE, *PBUSLOGICDEVICE;
128
129/**
130 * Commands the BusLogic adapter supports.
131 */
132enum BUSLOGICCOMMAND
133{
134 BUSLOGICCOMMAND_TEST_CMDC_INTERRUPT = 0x00,
135 BUSLOGICCOMMAND_INITIALIZE_MAILBOX = 0x01,
136 BUSLOGICCOMMAND_EXECUTE_MAILBOX_COMMAND = 0x02,
137 BUSLOGICCOMMAND_EXECUTE_BIOS_COMMAND = 0x03,
138 BUSLOGICCOMMAND_INQUIRE_BOARD_ID = 0x04,
139 BUSLOGICCOMMAND_ENABLE_OUTGOING_MAILBOX_AVAILABLE_INTERRUPT = 0x05,
140 BUSLOGICCOMMAND_SET_SCSI_SELECTION_TIMEOUT = 0x06,
141 BUSLOGICCOMMAND_SET_PREEMPT_TIME_ON_BUS = 0x07,
142 BUSLOGICCOMMAND_SET_TIME_OFF_BUS = 0x08,
143 BUSLOGICCOMMAND_SET_BUS_TRANSFER_RATE = 0x09,
144 BUSLOGICCOMMAND_INQUIRE_INSTALLED_DEVICES_ID_0_TO_7 = 0x0a,
145 BUSLOGICCOMMAND_INQUIRE_CONFIGURATION = 0x0b,
146 BUSLOGICCOMMAND_ENABLE_TARGET_MODE = 0x0c,
147 BUSLOGICCOMMAND_INQUIRE_SETUP_INFORMATION = 0x0d,
148 BUSLOGICCOMMAND_WRITE_ADAPTER_LOCAL_RAM = 0x1a,
149 BUSLOGICCOMMAND_READ_ADAPTER_LOCAL_RAM = 0x1b,
150 BUSLOGICCOMMAND_WRITE_BUSMASTER_CHIP_FIFO = 0x1c,
151 BUSLOGICCOMMAND_READ_BUSMASTER_CHIP_FIFO = 0x1d,
152 BUSLOGICCOMMAND_ECHO_COMMAND_DATA = 0x1f,
153 BUSLOGICCOMMAND_HOST_ADAPTER_DIAGNOSTIC = 0x20,
154 BUSLOGICCOMMAND_SET_ADAPTER_OPTIONS = 0x21,
155 BUSLOGICCOMMAND_INQUIRE_INSTALLED_DEVICES_ID_8_TO_15 = 0x23,
156 BUSLOGICCOMMAND_INQUIRE_TARGET_DEVICES = 0x24,
157 BUSLOGICCOMMAND_DISABLE_HOST_ADAPTER_INTERRUPT = 0x25,
158 BUSLOGICCOMMAND_EXT_BIOS_INFO = 0x28,
159 BUSLOGICCOMMAND_UNLOCK_MAILBOX = 0x29,
160 BUSLOGICCOMMAND_INITIALIZE_EXTENDED_MAILBOX = 0x81,
161 BUSLOGICCOMMAND_EXECUTE_SCSI_COMMAND = 0x83,
162 BUSLOGICCOMMAND_INQUIRE_FIRMWARE_VERSION_3RD_LETTER = 0x84,
163 BUSLOGICCOMMAND_INQUIRE_FIRMWARE_VERSION_LETTER = 0x85,
164 BUSLOGICCOMMAND_INQUIRE_PCI_HOST_ADAPTER_INFORMATION = 0x86,
165 BUSLOGICCOMMAND_INQUIRE_HOST_ADAPTER_MODEL_NUMBER = 0x8b,
166 BUSLOGICCOMMAND_INQUIRE_SYNCHRONOUS_PERIOD = 0x8c,
167 BUSLOGICCOMMAND_INQUIRE_EXTENDED_SETUP_INFORMATION = 0x8d,
168 BUSLOGICCOMMAND_ENABLE_STRICT_ROUND_ROBIN_MODE = 0x8f,
169 BUSLOGICCOMMAND_STORE_HOST_ADAPTER_LOCAL_RAM = 0x90,
170 BUSLOGICCOMMAND_FETCH_HOST_ADAPTER_LOCAL_RAM = 0x91,
171 BUSLOGICCOMMAND_STORE_LOCAL_DATA_IN_EEPROM = 0x92,
172 BUSLOGICCOMMAND_UPLOAD_AUTO_SCSI_CODE = 0x94,
173 BUSLOGICCOMMAND_MODIFY_IO_ADDRESS = 0x95,
174 BUSLOGICCOMMAND_SET_CCB_FORMAT = 0x96,
175 BUSLOGICCOMMAND_WRITE_INQUIRY_BUFFER = 0x9a,
176 BUSLOGICCOMMAND_READ_INQUIRY_BUFFER = 0x9b,
177 BUSLOGICCOMMAND_FLASH_ROM_UPLOAD_DOWNLOAD = 0xa7,
178 BUSLOGICCOMMAND_READ_SCAM_DATA = 0xa8,
179 BUSLOGICCOMMAND_WRITE_SCAM_DATA = 0xa9
180} BUSLOGICCOMMAND;
181
182#pragma pack(1)
183/**
184 * Auto SCSI structure which is located
185 * in host adapter RAM and contains several
186 * configuration parameters.
187 */
188typedef struct AutoSCSIRam
189{
190 uint8_t aInternalSignature[2];
191 uint8_t cbInformation;
192 uint8_t aHostAdaptertype[6];
193 uint8_t uReserved1;
194 bool fFloppyEnabled : 1;
195 bool fFloppySecondary : 1;
196 bool fLevelSensitiveInterrupt : 1;
197 unsigned char uReserved2 : 2;
198 unsigned char uSystemRAMAreForBIOS : 3;
199 unsigned char uDMAChannel : 7;
200 bool fDMAAutoConfiguration : 1;
201 unsigned char uIrqChannel : 7;
202 bool fIrqAutoConfiguration : 1;
203 uint8_t uDMATransferRate;
204 uint8_t uSCSIId;
205 bool fLowByteTerminated : 1;
206 bool fParityCheckingEnabled : 1;
207 bool fHighByteTerminated : 1;
208 bool fNoisyCablingEnvironment : 1;
209 bool fFastSynchronousNeogtiation : 1;
210 bool fBusResetEnabled : 1;
211 bool fReserved3 : 1;
212 bool fActiveNegotiationEnabled : 1;
213 uint8_t uBusOnDelay;
214 uint8_t uBusOffDelay;
215 bool fHostAdapterBIOSEnabled : 1;
216 bool fBIOSRedirectionOfInt19 : 1;
217 bool fExtendedTranslation : 1;
218 bool fMapRemovableAsFixed : 1;
219 bool fReserved4 : 1;
220 bool fBIOSSupportsMoreThan2Drives : 1;
221 bool fBIOSInterruptMode : 1;
222 bool fFlopticalSupport : 1;
223 uint16_t u16DeviceEnabledMask;
224 uint16_t u16WidePermittedMask;
225 uint16_t u16FastPermittedMask;
226 uint16_t u16SynchronousPermittedMask;
227 uint16_t u16DisconnectPermittedMask;
228 uint16_t u16SendStartUnitCommandMask;
229 uint16_t u16IgnoreInBIOSScanMask;
230 unsigned char uPCIInterruptPin : 2;
231 unsigned char uHostAdapterIoPortAddress : 2;
232 bool fStrictRoundRobinMode : 1;
233 bool fVesaBusSpeedGreaterThan33MHz : 1;
234 bool fVesaBurstWrite : 1;
235 bool fVesaBurstRead : 1;
236 uint16_t u16UltraPermittedMask;
237 uint32_t uReserved5;
238 uint8_t uReserved6;
239 uint8_t uAutoSCSIMaximumLUN;
240 bool fReserved7 : 1;
241 bool fSCAMDominant : 1;
242 bool fSCAMenabled : 1;
243 bool fSCAMLevel2 : 1;
244 unsigned char uReserved8 : 4;
245 bool fInt13Extension : 1;
246 bool fReserved9 : 1;
247 bool fCDROMBoot : 1;
248 unsigned char uReserved10 : 5;
249 unsigned char uBootTargetId : 4;
250 unsigned char uBootChannel : 4;
251 bool fForceBusDeviceScanningOrder : 1;
252 unsigned char uReserved11 : 7;
253 uint16_t u16NonTaggedToAlternateLunPermittedMask;
254 uint16_t u16RenegotiateSyncAfterCheckConditionMask;
255 uint8_t aReserved12[10];
256 uint8_t aManufacturingDiagnostic[2];
257 uint16_t u16Checksum;
258} AutoSCSIRam, *PAutoSCSIRam;
259AssertCompileSize(AutoSCSIRam, 64);
260#pragma pack()
261
262/**
263 * The local Ram.
264 */
265typedef union HostAdapterLocalRam
266{
267 /** Byte view. */
268 uint8_t u8View[256];
269 /** Structured view. */
270 struct
271 {
272 /** Offset 0 - 63 is for BIOS. */
273 uint8_t u8Bios[64];
274 /** Auto SCSI structure. */
275 AutoSCSIRam autoSCSIData;
276 } structured;
277} HostAdapterLocalRam, *PHostAdapterLocalRam;
278AssertCompileSize(HostAdapterLocalRam, 256);
279
280
281/** Ugly 24-bit big-endian addressing. */
282typedef struct
283{
284 uint8_t hi;
285 uint8_t mid;
286 uint8_t lo;
287} Addr24, Len24;
288AssertCompileSize(Addr24, 3);
289
290#define ADDR_TO_U32(x) (((x).hi << 16) | ((x).mid << 8) | (x).lo)
291#define LEN_TO_U32 ADDR_TO_U32
292#define U32_TO_ADDR(a, x) do {(a).hi = (x) >> 16; (a).mid = (x) >> 8; (a).lo = (x);} while(0)
293#define U32_TO_LEN U32_TO_ADDR
294
295/** @name Compatible ISA base I/O port addresses. Disabled if zero.
296 * @{ */
297#define NUM_ISA_BASES 8
298#define MAX_ISA_BASE (NUM_ISA_BASES - 1)
299#define ISA_BASE_DISABLED 6
300
301static uint16_t const g_aISABases[NUM_ISA_BASES] =
302{
303 0x330, 0x334, 0x230, 0x234, 0x130, 0x134, 0, 0
304};
305/** @} */
306
307/** Pointer to a task state structure. */
308typedef struct BUSLOGICTASKSTATE *PBUSLOGICTASKSTATE;
309
310/**
311 * Main BusLogic device state.
312 *
313 * @extends PCIDEVICE
314 * @implements PDMILEDPORTS
315 */
316typedef struct BUSLOGIC
317{
318 /** The PCI device structure. */
319 PCIDEVICE dev;
320 /** Pointer to the device instance - HC ptr */
321 PPDMDEVINSR3 pDevInsR3;
322 /** Pointer to the device instance - R0 ptr */
323 PPDMDEVINSR0 pDevInsR0;
324 /** Pointer to the device instance - RC ptr. */
325 PPDMDEVINSRC pDevInsRC;
326
327 /** Whether R0 is enabled. */
328 bool fR0Enabled;
329 /** Whether RC is enabled. */
330 bool fGCEnabled;
331
332 /** Base address of the I/O ports. */
333 RTIOPORT IOPortBase;
334 /** Base address of the memory mapping. */
335 RTGCPHYS MMIOBase;
336 /** Status register - Readonly. */
337 volatile uint8_t regStatus;
338 /** Interrupt register - Readonly. */
339 volatile uint8_t regInterrupt;
340 /** Geometry register - Readonly. */
341 volatile uint8_t regGeometry;
342 /** Pending (delayed) interrupt. */
343 uint8_t uPendingIntr;
344
345 /** Local RAM for the fetch hostadapter local RAM request.
346 * I don't know how big the buffer really is but the maximum
347 * seems to be 256 bytes because the offset and count field in the command request
348 * are only one byte big.
349 */
350 HostAdapterLocalRam LocalRam;
351
352 /** Command code the guest issued. */
353 uint8_t uOperationCode;
354 /** Buffer for the command parameters the adapter is currently receiving from the guest.
355 * Size of the largest command which is possible.
356 */
357 uint8_t aCommandBuffer[BUSLOGIC_COMMAND_SIZE_MAX]; /* Size of the biggest request. */
358 /** Current position in the command buffer. */
359 uint8_t iParameter;
360 /** Parameters left until the command is complete. */
361 uint8_t cbCommandParametersLeft;
362
363 /** Whether we are using the RAM or reply buffer. */
364 bool fUseLocalRam;
365 /** Buffer to store reply data from the controller to the guest. */
366 uint8_t aReplyBuffer[BUSLOGIC_REPLY_SIZE_MAX]; /* Size of the biggest reply. */
367 /** Position in the buffer we are reading next. */
368 uint8_t iReply;
369 /** Bytes left until the reply buffer is empty. */
370 uint8_t cbReplyParametersLeft;
371
372 /** Flag whether IRQs are enabled. */
373 bool fIRQEnabled;
374 /** Flag whether the ISA I/O port range is disabled
375 * to prevent the BIOS to access the device. */
376 bool fISAEnabled; /**< @todo unused, to be removed */
377 /** Flag whether 24-bit mailboxes are in use (default is 32-bit). */
378 bool fMbxIs24Bit;
379 /** ISA I/O port base (encoded in FW-compatible format). */
380 uint8_t uISABaseCode;
381
382 /** ISA I/O port base (disabled if zero). */
383 RTIOPORT IOISABase;
384 /** Default ISA I/O port base in FW-compatible format. */
385 uint8_t uDefaultISABaseCode;
386
387 /** Number of mailboxes the guest set up. */
388 uint32_t cMailbox;
389
390#if HC_ARCH_BITS == 64
391 uint32_t Alignment0;
392#endif
393
394 /** Time when HBA reset was last initiated. */ /**< @todo does this need to be saved? */
395 uint64_t u64ResetTime;
396 /** Physical base address of the outgoing mailboxes. */
397 RTGCPHYS GCPhysAddrMailboxOutgoingBase;
398 /** Current outgoing mailbox position. */
399 uint32_t uMailboxOutgoingPositionCurrent;
400 /** Number of mailboxes ready. */
401 volatile uint32_t cMailboxesReady;
402 /** Whether a notification to R3 was send. */
403 volatile bool fNotificationSend;
404
405#if HC_ARCH_BITS == 64
406 uint32_t Alignment1;
407#endif
408
409 /** Physical base address of the incoming mailboxes. */
410 RTGCPHYS GCPhysAddrMailboxIncomingBase;
411 /** Current incoming mailbox position. */
412 uint32_t uMailboxIncomingPositionCurrent;
413
414 /** Whether strict round robin is enabled. */
415 bool fStrictRoundRobinMode;
416 /** Whether the extended LUN CCB format is enabled for 32 possible logical units. */
417 bool fExtendedLunCCBFormat;
418
419 /** Queue to send tasks to R3. - HC ptr */
420 R3PTRTYPE(PPDMQUEUE) pNotifierQueueR3;
421 /** Queue to send tasks to R3. - HC ptr */
422 R0PTRTYPE(PPDMQUEUE) pNotifierQueueR0;
423 /** Queue to send tasks to R3. - RC ptr */
424 RCPTRTYPE(PPDMQUEUE) pNotifierQueueRC;
425
426 uint32_t Alignment2;
427
428 /** Critical section protecting access to the interrupt status register. */
429 PDMCRITSECT CritSectIntr;
430
431 /** Cache for task states. */
432 R3PTRTYPE(RTMEMCACHE) hTaskCache;
433
434 /** Device state for BIOS access. */
435 VBOXSCSI VBoxSCSI;
436
437 /** BusLogic device states. */
438 BUSLOGICDEVICE aDeviceStates[BUSLOGIC_MAX_DEVICES];
439
440 /** The base interface.
441 * @todo use PDMDEVINS::IBase */
442 PDMIBASE IBase;
443 /** Status Port - Leds interface. */
444 PDMILEDPORTS ILeds;
445 /** Partner of ILeds. */
446 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
447
448#if HC_ARCH_BITS == 64
449 uint32_t Alignment3;
450#endif
451
452 /** Indicates that PDMDevHlpAsyncNotificationCompleted should be called when
453 * a port is entering the idle state. */
454 bool volatile fSignalIdle;
455 /** Flag whether we have tasks which need to be processed again. */
456 bool volatile fRedo;
457 /** List of tasks which can be redone. */
458 R3PTRTYPE(volatile PBUSLOGICTASKSTATE) pTasksRedoHead;
459
460#ifdef LOG_ENABLED
461# if HC_ARCH_BITS == 64
462 uint32_t Alignment4;
463# endif
464
465 volatile uint32_t cInMailboxesReady;
466#endif
467
468} BUSLOGIC, *PBUSLOGIC;
469
470/** Register offsets in the I/O port space. */
471#define BUSLOGIC_REGISTER_CONTROL 0 /**< Writeonly */
472/** Fields for the control register. */
473# define BUSLOGIC_REGISTER_CONTROL_SCSI_BUSRESET RT_BIT(4)
474# define BUSLOGIC_REGISTER_CONTROL_INTERRUPT_RESET RT_BIT(5)
475# define BUSLOGIC_REGISTER_CONTROL_SOFT_RESET RT_BIT(6)
476# define BUSLOGIC_REGISTER_CONTROL_HARD_RESET RT_BIT(7)
477
478#define BUSLOGIC_REGISTER_STATUS 0 /**< Readonly */
479/** Fields for the status register. */
480# define BUSLOGIC_REGISTER_STATUS_COMMAND_INVALID RT_BIT(0)
481# define BUSLOGIC_REGISTER_STATUS_DATA_IN_REGISTER_READY RT_BIT(2)
482# define BUSLOGIC_REGISTER_STATUS_COMMAND_PARAMETER_REGISTER_BUSY RT_BIT(3)
483# define BUSLOGIC_REGISTER_STATUS_HOST_ADAPTER_READY RT_BIT(4)
484# define BUSLOGIC_REGISTER_STATUS_INITIALIZATION_REQUIRED RT_BIT(5)
485# define BUSLOGIC_REGISTER_STATUS_DIAGNOSTIC_FAILURE RT_BIT(6)
486# define BUSLOGIC_REGISTER_STATUS_DIAGNOSTIC_ACTIVE RT_BIT(7)
487
488#define BUSLOGIC_REGISTER_COMMAND 1 /**< Writeonly */
489#define BUSLOGIC_REGISTER_DATAIN 1 /**< Readonly */
490#define BUSLOGIC_REGISTER_INTERRUPT 2 /**< Readonly */
491/** Fields for the interrupt register. */
492# define BUSLOGIC_REGISTER_INTERRUPT_INCOMING_MAILBOX_LOADED RT_BIT(0)
493# define BUSLOGIC_REGISTER_INTERRUPT_OUTGOING_MAILBOX_AVAILABLE RT_BIT(1)
494# define BUSLOGIC_REGISTER_INTERRUPT_COMMAND_COMPLETE RT_BIT(2)
495# define BUSLOGIC_REGISTER_INTERRUPT_EXTERNAL_BUS_RESET RT_BIT(3)
496# define BUSLOGIC_REGISTER_INTERRUPT_INTERRUPT_VALID RT_BIT(7)
497
498#define BUSLOGIC_REGISTER_GEOMETRY 3 /* Readonly */
499# define BUSLOGIC_REGISTER_GEOMETRY_EXTENTED_TRANSLATION_ENABLED RT_BIT(7)
500
501/** Structure for the INQUIRE_PCI_HOST_ADAPTER_INFORMATION reply. */
502typedef struct ReplyInquirePCIHostAdapterInformation
503{
504 uint8_t IsaIOPort;
505 uint8_t IRQ;
506 unsigned char LowByteTerminated : 1;
507 unsigned char HighByteTerminated : 1;
508 unsigned char uReserved : 2; /* Reserved. */
509 unsigned char JP1 : 1; /* Whatever that means. */
510 unsigned char JP2 : 1; /* Whatever that means. */
511 unsigned char JP3 : 1; /* Whatever that means. */
512 /** Whether the provided info is valid. */
513 unsigned char InformationIsValid: 1;
514 uint8_t uReserved2; /* Reserved. */
515} ReplyInquirePCIHostAdapterInformation, *PReplyInquirePCIHostAdapterInformation;
516AssertCompileSize(ReplyInquirePCIHostAdapterInformation, 4);
517
518/** Structure for the INQUIRE_CONFIGURATION reply. */
519typedef struct ReplyInquireConfiguration
520{
521 unsigned char uReserved1 : 5;
522 bool fDmaChannel5 : 1;
523 bool fDmaChannel6 : 1;
524 bool fDmaChannel7 : 1;
525 bool fIrqChannel9 : 1;
526 bool fIrqChannel10 : 1;
527 bool fIrqChannel11 : 1;
528 bool fIrqChannel12 : 1;
529 unsigned char uReserved2 : 1;
530 bool fIrqChannel14 : 1;
531 bool fIrqChannel15 : 1;
532 unsigned char uReserved3 : 1;
533 unsigned char uHostAdapterId : 4;
534 unsigned char uReserved4 : 4;
535} ReplyInquireConfiguration, *PReplyInquireConfiguration;
536AssertCompileSize(ReplyInquireConfiguration, 3);
537
538/** Structure for the INQUIRE_SETUP_INFORMATION reply. */
539typedef struct ReplyInquireSetupInformationSynchronousValue
540{
541 unsigned char uOffset : 4;
542 unsigned char uTransferPeriod : 3;
543 bool fSynchronous : 1;
544}ReplyInquireSetupInformationSynchronousValue, *PReplyInquireSetupInformationSynchronousValue;
545AssertCompileSize(ReplyInquireSetupInformationSynchronousValue, 1);
546
547typedef struct ReplyInquireSetupInformation
548{
549 bool fSynchronousInitiationEnabled : 1;
550 bool fParityCheckingEnabled : 1;
551 unsigned char uReserved1 : 6;
552 uint8_t uBusTransferRate;
553 uint8_t uPreemptTimeOnBus;
554 uint8_t uTimeOffBus;
555 uint8_t cMailbox;
556 Addr24 MailboxAddress;
557 ReplyInquireSetupInformationSynchronousValue SynchronousValuesId0To7[8];
558 uint8_t uDisconnectPermittedId0To7;
559 uint8_t uSignature;
560 uint8_t uCharacterD;
561 uint8_t uHostBusType;
562 uint8_t uWideTransferPermittedId0To7;
563 uint8_t uWideTransfersActiveId0To7;
564 ReplyInquireSetupInformationSynchronousValue SynchronousValuesId8To15[8];
565 uint8_t uDisconnectPermittedId8To15;
566 uint8_t uReserved2;
567 uint8_t uWideTransferPermittedId8To15;
568 uint8_t uWideTransfersActiveId8To15;
569} ReplyInquireSetupInformation, *PReplyInquireSetupInformation;
570AssertCompileSize(ReplyInquireSetupInformation, 34);
571
572/** Structure for the INQUIRE_EXTENDED_SETUP_INFORMATION. */
573#pragma pack(1)
574typedef struct ReplyInquireExtendedSetupInformation
575{
576 uint8_t uBusType;
577 uint8_t uBiosAddress;
578 uint16_t u16ScatterGatherLimit;
579 uint8_t cMailbox;
580 uint32_t uMailboxAddressBase;
581 unsigned char uReserved1 : 2;
582 bool fFastEISA : 1;
583 unsigned char uReserved2 : 3;
584 bool fLevelSensitiveInterrupt : 1;
585 unsigned char uReserved3 : 1;
586 unsigned char aFirmwareRevision[3];
587 bool fHostWideSCSI : 1;
588 bool fHostDifferentialSCSI : 1;
589 bool fHostSupportsSCAM : 1;
590 bool fHostUltraSCSI : 1;
591 bool fHostSmartTermination : 1;
592 unsigned char uReserved4 : 3;
593} ReplyInquireExtendedSetupInformation, *PReplyInquireExtendedSetupInformation;
594AssertCompileSize(ReplyInquireExtendedSetupInformation, 14);
595#pragma pack()
596
597/** Structure for the INITIALIZE EXTENDED MAILBOX request. */
598#pragma pack(1)
599typedef struct RequestInitializeExtendedMailbox
600{
601 /** Number of mailboxes in guest memory. */
602 uint8_t cMailbox;
603 /** Physical address of the first mailbox. */
604 uint32_t uMailboxBaseAddress;
605} RequestInitializeExtendedMailbox, *PRequestInitializeExtendedMailbox;
606AssertCompileSize(RequestInitializeExtendedMailbox, 5);
607#pragma pack()
608
609/** Structure for the INITIALIZE MAILBOX request. */
610typedef struct
611{
612 /** Number of mailboxes to set up. */
613 uint8_t cMailbox;
614 /** Physical address of the first mailbox. */
615 Addr24 aMailboxBaseAddr;
616} RequestInitMbx, *PRequestInitMbx;
617AssertCompileSize(RequestInitMbx, 4);
618
619/**
620 * Structure of a mailbox in guest memory.
621 * The incoming and outgoing mailbox have the same size
622 * but the incoming one has some more fields defined which
623 * are marked as reserved in the outgoing one.
624 * The last field is also different from the type.
625 * For outgoing mailboxes it is the action and
626 * for incoming ones the completion status code for the task.
627 * We use one structure for both types.
628 */
629typedef struct Mailbox32
630{
631 /** Physical address of the CCB structure in the guest memory. */
632 uint32_t u32PhysAddrCCB;
633 /** Type specific data. */
634 union
635 {
636 /** For outgoing mailboxes. */
637 struct
638 {
639 /** Reserved */
640 uint8_t uReserved[3];
641 /** Action code. */
642 uint8_t uActionCode;
643 } out;
644 /** For incoming mailboxes. */
645 struct
646 {
647 /** The host adapter status after finishing the request. */
648 uint8_t uHostAdapterStatus;
649 /** The status of the device which executed the request after executing it. */
650 uint8_t uTargetDeviceStatus;
651 /** Reserved. */
652 uint8_t uReserved;
653 /** The completion status code of the request. */
654 uint8_t uCompletionCode;
655 } in;
656 } u;
657} Mailbox32, *PMailbox32;
658AssertCompileSize(Mailbox32, 8);
659
660/** Old style 24-bit mailbox entry. */
661typedef struct Mailbox24
662{
663 /** Mailbox command (incoming) or state (outgoing). */
664 uint8_t uCmdState;
665 /** Physical address of the CCB structure in the guest memory. */
666 Addr24 aPhysAddrCCB;
667} Mailbox24, *PMailbox24;
668AssertCompileSize(Mailbox24, 4);
669
670/**
671 * Action codes for outgoing mailboxes.
672 */
673enum BUSLOGIC_MAILBOX_OUTGOING_ACTION
674{
675 BUSLOGIC_MAILBOX_OUTGOING_ACTION_FREE = 0x00,
676 BUSLOGIC_MAILBOX_OUTGOING_ACTION_START_COMMAND = 0x01,
677 BUSLOGIC_MAILBOX_OUTGOING_ACTION_ABORT_COMMAND = 0x02
678};
679
680/**
681 * Completion codes for incoming mailboxes.
682 */
683enum BUSLOGIC_MAILBOX_INCOMING_COMPLETION
684{
685 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_FREE = 0x00,
686 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_WITHOUT_ERROR = 0x01,
687 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_ABORTED = 0x02,
688 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_ABORTED_NOT_FOUND = 0x03,
689 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_WITH_ERROR = 0x04,
690 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_INVALID_CCB = 0x05
691};
692
693/**
694 * Host adapter status for incoming mailboxes.
695 */
696enum BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS
697{
698 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_CMD_COMPLETED = 0x00,
699 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_LINKED_CMD_COMPLETED = 0x0a,
700 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_LINKED_CMD_COMPLETED_WITH_FLAG = 0x0b,
701 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_DATA_UNDERUN = 0x0c,
702 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_SCSI_SELECTION_TIMEOUT = 0x11,
703 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_DATA_OVERRUN = 0x12,
704 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_UNEXPECTED_BUS_FREE = 0x13,
705 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_INVALID_BUS_PHASE_REQUESTED = 0x14,
706 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_INVALID_OUTGOING_MAILBOX_ACTION_CODE = 0x15,
707 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_INVALID_COMMAND_OPERATION_CODE = 0x16,
708 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_LINKED_CCB_HAS_INVALID_LUN = 0x17,
709 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_INVALID_COMMAND_PARAMETER = 0x1a,
710 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_AUTO_REQUEST_SENSE_FAILED = 0x1b,
711 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_TAGGED_QUEUING_MESSAGE_REJECTED = 0x1c,
712 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_UNSUPPORTED_MESSAGE_RECEIVED = 0x1d,
713 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_HOST_ADAPTER_HARDWARE_FAILED = 0x20,
714 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_TARGET_FAILED_RESPONSE_TO_ATN = 0x21,
715 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_HOST_ADAPTER_ASSERTED_RST = 0x22,
716 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_OTHER_DEVICE_ASSERTED_RST = 0x23,
717 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_TARGET_DEVICE_RECONNECTED_IMPROPERLY = 0x24,
718 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_HOST_ADAPTER_ASSERTED_BUS_DEVICE_RESET = 0x25,
719 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_ABORT_QUEUE_GENERATED = 0x26,
720 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_HOST_ADAPTER_SOFTWARE_ERROR = 0x27,
721 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_HOST_ADAPTER_HARDWARE_TIMEOUT_ERROR = 0x30,
722 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_SCSI_PARITY_ERROR_DETECTED = 0x34
723};
724
725/**
726 * Device status codes for incoming mailboxes.
727 */
728enum BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS
729{
730 BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS_OPERATION_GOOD = 0x00,
731 BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS_CHECK_CONDITION = 0x02,
732 BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS_DEVICE_BUSY = 0x08
733};
734
735/**
736 * Opcode types for CCB.
737 */
738enum BUSLOGIC_CCB_OPCODE
739{
740 BUSLOGIC_CCB_OPCODE_INITIATOR_CCB = 0x00,
741 BUSLOGIC_CCB_OPCODE_TARGET_CCB = 0x01,
742 BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_SCATTER_GATHER = 0x02,
743 BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_DATA_LENGTH = 0x03,
744 BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_SCATTER_GATHER = 0x04,
745 BUSLOGIC_CCB_OPCODE_BUS_DEVICE_RESET = 0x81
746};
747
748/**
749 * Data transfer direction.
750 */
751enum BUSLOGIC_CCB_DIRECTION
752{
753 BUSLOGIC_CCB_DIRECTION_UNKNOWN = 0x00,
754 BUSLOGIC_CCB_DIRECTION_IN = 0x01,
755 BUSLOGIC_CCB_DIRECTION_OUT = 0x02,
756 BUSLOGIC_CCB_DIRECTION_NO_DATA = 0x03
757};
758
759/**
760 * The command control block for a SCSI request.
761 */
762typedef struct CCB32
763{
764 /** Opcode. */
765 uint8_t uOpcode;
766 /** Reserved */
767 unsigned char uReserved1 : 3;
768 /** Data direction for the request. */
769 unsigned char uDataDirection : 2;
770 /** Whether the request is tag queued. */
771 bool fTagQueued : 1;
772 /** Queue tag mode. */
773 unsigned char uQueueTag : 2;
774 /** Length of the SCSI CDB. */
775 uint8_t cbCDB;
776 /** Sense data length. */
777 uint8_t cbSenseData;
778 /** Data length. */
779 uint32_t cbData;
780 /** Data pointer.
781 * This points to the data region or a scatter gather list based on the opcode.
782 */
783 uint32_t u32PhysAddrData;
784 /** Reserved. */
785 uint8_t uReserved2[2];
786 /** Host adapter status. */
787 uint8_t uHostAdapterStatus;
788 /** Device adapter status. */
789 uint8_t uDeviceStatus;
790 /** The device the request is sent to. */
791 uint8_t uTargetId;
792 /**The LUN in the device. */
793 unsigned char uLogicalUnit : 5;
794 /** Legacy tag. */
795 bool fLegacyTagEnable : 1;
796 /** Legacy queue tag. */
797 unsigned char uLegacyQueueTag : 2;
798 /** The SCSI CDB. (A CDB can be 12 bytes long.) */
799 uint8_t abCDB[12];
800 /** Reserved. */
801 uint8_t uReserved3[6];
802 /** Sense data pointer. */
803 uint32_t u32PhysAddrSenseData;
804} CCB32, *PCCB32;
805AssertCompileSize(CCB32, 40);
806
807
808/**
809 * The 24-bit command control block.
810 */
811typedef struct CCB24
812{
813 /** Opcode. */
814 uint8_t uOpcode;
815 /** The LUN in the device. */
816 unsigned char uLogicalUnit : 3;
817 /** Data direction for the request. */
818 unsigned char uDataDirection : 2;
819 /** The target device ID. */
820 unsigned char uTargetId : 3;
821 /** Length of the SCSI CDB. */
822 uint8_t cbCDB;
823 /** Sense data length. */
824 uint8_t cbSenseData;
825 /** Data length. */
826 Len24 acbData;
827 /** Data pointer.
828 * This points to the data region or a scatter gather list based on the opc
829 */
830 Addr24 aPhysAddrData;
831 /** Pointer to next CCB for linked commands. */
832 Addr24 aPhysAddrLink;
833 /** Command linking identifier. */
834 uint8_t uLinkId;
835 /** Host adapter status. */
836 uint8_t uHostAdapterStatus;
837 /** Device adapter status. */
838 uint8_t uDeviceStatus;
839 /** Two unused bytes. */
840 uint8_t aReserved[2];
841 /** The SCSI CDB. (A CDB can be 12 bytes long.) */
842 uint8_t abCDB[12];
843} CCB24, *PCCB24;
844AssertCompileSize(CCB24, 30);
845
846/**
847 * The common 24-bit/32-bit command control block. The 32-bit CCB is laid out
848 * such that many fields are in the same location as in the older 24-bit CCB.
849 */
850typedef struct CCBC
851{
852 /** Opcode. */
853 uint8_t uOpcode;
854 /** The LUN in the device. */
855 unsigned char uPad1 : 3;
856 /** Data direction for the request. */
857 unsigned char uDataDirection : 2;
858 /** The target device ID. */
859 unsigned char uPad2 : 3;
860 /** Length of the SCSI CDB. */
861 uint8_t cbCDB;
862 /** Sense data length. */
863 uint8_t cbSenseData;
864 uint8_t aPad1[10];
865 /** Host adapter status. */
866 uint8_t uHostAdapterStatus;
867 /** Device adapter status. */
868 uint8_t uDeviceStatus;
869 uint8_t aPad2[2];
870 /** The SCSI CDB (up to 12 bytes). */
871 uint8_t abCDB[12];
872} CCBC, *PCCBC;
873AssertCompileSize(CCB24, 30);
874
875/* Make sure that the 24-bit/32-bit/common CCB offsets match. */
876AssertCompileMemberOffset(CCBC, cbCDB, 2);
877AssertCompileMemberOffset(CCB24, cbCDB, 2);
878AssertCompileMemberOffset(CCB32, cbCDB, 2);
879AssertCompileMemberOffset(CCBC, uHostAdapterStatus, 14);
880AssertCompileMemberOffset(CCB24, uHostAdapterStatus, 14);
881AssertCompileMemberOffset(CCB32, uHostAdapterStatus, 14);
882AssertCompileMemberOffset(CCBC, abCDB, 18);
883AssertCompileMemberOffset(CCB24, abCDB, 18);
884AssertCompileMemberOffset(CCB32, abCDB, 18);
885
886/** A union of all CCB types (24-bit/32-bit/common). */
887typedef union CCBU
888{
889 CCB32 n; /**< New 32-bit CCB. */
890 CCB24 o; /**< Old 24-bit CCB. */
891 CCBC c; /**< Common CCB subset. */
892} CCBU, *PCCBU;
893
894/** 32-bit scatter-gather list entry. */
895typedef struct SGE32
896{
897 uint32_t cbSegment;
898 uint32_t u32PhysAddrSegmentBase;
899} SGE32, *PSGE32;
900AssertCompileSize(SGE32, 8);
901
902/** 24-bit scatter-gather list entry. */
903typedef struct SGE24
904{
905 Len24 acbSegment;
906 Addr24 aPhysAddrSegmentBase;
907} SGE24, *PSGE24;
908AssertCompileSize(SGE24, 6);
909
910/**
911 * The structure for the "Execute SCSI Command" command.
912 */
913typedef struct ESCMD
914{
915 /** Data length. */
916 uint32_t cbData;
917 /** Data pointer. */
918 uint32_t u32PhysAddrData;
919 /** The device the request is sent to. */
920 uint8_t uTargetId;
921 /** The LUN in the device. */
922 uint8_t uLogicalUnit;
923 /** Reserved */
924 unsigned char uReserved1 : 3;
925 /** Data direction for the request. */
926 unsigned char uDataDirection : 2;
927 /** Reserved */
928 unsigned char uReserved2 : 3;
929 /** Length of the SCSI CDB. */
930 uint8_t cbCDB;
931 /** The SCSI CDB. (A CDB can be 12 bytes long.) */
932 uint8_t abCDB[12];
933} ESCMD, *PESCMD;
934AssertCompileSize(ESCMD, 24);
935
936/**
937 * Task state for a CCB request.
938 */
939typedef struct BUSLOGICTASKSTATE
940{
941 /** Next in the redo list. */
942 PBUSLOGICTASKSTATE pRedoNext;
943 /** Device this task is assigned to. */
944 R3PTRTYPE(PBUSLOGICDEVICE) pTargetDeviceR3;
945 /** The command control block from the guest. */
946 CCBU CommandControlBlockGuest;
947 /** Mailbox read from guest memory. */
948 Mailbox32 MailboxGuest;
949 /** The SCSI request we pass to the underlying SCSI engine. */
950 PDMSCSIREQUEST PDMScsiRequest;
951 /** Data buffer segment */
952 RTSGSEG DataSeg;
953 /** Pointer to the R3 sense buffer. */
954 uint8_t *pbSenseBuffer;
955 /** Flag whether this is a request from the BIOS. */
956 bool fBIOS;
957 /** 24-bit request flag (default is 32-bit). */
958 bool fIs24Bit;
959 /** S/G entry size (depends on the above flag). */
960 uint8_t cbSGEntry;
961} BUSLOGICTASKSTATE;
962
963#ifndef VBOX_DEVICE_STRUCT_TESTCASE
964
965#define PDMIBASE_2_PBUSLOGICDEVICE(pInterface) ( (PBUSLOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(BUSLOGICDEVICE, IBase)) )
966#define PDMISCSIPORT_2_PBUSLOGICDEVICE(pInterface) ( (PBUSLOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(BUSLOGICDEVICE, ISCSIPort)) )
967#define PDMILEDPORTS_2_PBUSLOGICDEVICE(pInterface) ( (PBUSLOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(BUSLOGICDEVICE, ILed)) )
968#define PDMIBASE_2_PBUSLOGIC(pInterface) ( (PBUSLOGIC)((uintptr_t)(pInterface) - RT_OFFSETOF(BUSLOGIC, IBase)) )
969#define PDMILEDPORTS_2_PBUSLOGIC(pInterface) ( (PBUSLOGIC)((uintptr_t)(pInterface) - RT_OFFSETOF(BUSLOGIC, ILeds)) )
970
971/*******************************************************************************
972* Internal Functions *
973*******************************************************************************/
974static int buslogicR3RegisterISARange(PBUSLOGIC pBusLogic, uint8_t uBaseCode);
975
976
977/**
978 * Assert IRQ line of the BusLogic adapter.
979 *
980 * @returns nothing.
981 * @param pBusLogic Pointer to the BusLogic device instance.
982 * @param fSuppressIrq Flag to suppress IRQ generation regardless of fIRQEnabled
983 * @param uFlag Type of interrupt being generated.
984 */
985static void buslogicSetInterrupt(PBUSLOGIC pBusLogic, bool fSuppressIrq, uint8_t uIrqType)
986{
987 LogFlowFunc(("pBusLogic=%#p\n", pBusLogic));
988
989 /* The CMDC interrupt has priority over IMBL and MBOR. */
990 if (uIrqType & (BUSLOGIC_REGISTER_INTERRUPT_INCOMING_MAILBOX_LOADED | BUSLOGIC_REGISTER_INTERRUPT_OUTGOING_MAILBOX_AVAILABLE))
991 {
992 if (!(pBusLogic->regInterrupt & BUSLOGIC_REGISTER_INTERRUPT_COMMAND_COMPLETE))
993 pBusLogic->regInterrupt |= uIrqType; /* Report now. */
994 else
995 pBusLogic->uPendingIntr |= uIrqType; /* Report later. */
996 }
997 else if (uIrqType & BUSLOGIC_REGISTER_INTERRUPT_COMMAND_COMPLETE)
998 {
999 Assert(!pBusLogic->regInterrupt);
1000 pBusLogic->regInterrupt |= uIrqType;
1001 }
1002 else
1003 AssertMsgFailed(("Invalid interrupt state!\n"));
1004
1005 pBusLogic->regInterrupt |= BUSLOGIC_REGISTER_INTERRUPT_INTERRUPT_VALID;
1006 if (pBusLogic->fIRQEnabled && !fSuppressIrq)
1007 PDMDevHlpPCISetIrq(pBusLogic->CTX_SUFF(pDevIns), 0, 1);
1008}
1009
1010/**
1011 * Deasserts the interrupt line of the BusLogic adapter.
1012 *
1013 * @returns nothing
1014 * @param pBuslogic Pointer to the BusLogic device instance.
1015 */
1016static void buslogicClearInterrupt(PBUSLOGIC pBusLogic)
1017{
1018 LogFlowFunc(("pBusLogic=%#p, clearing %#02x (pending %#02x)\n",
1019 pBusLogic, pBusLogic->regInterrupt, pBusLogic->uPendingIntr));
1020 pBusLogic->regInterrupt = 0;
1021 PDMDevHlpPCISetIrq(pBusLogic->CTX_SUFF(pDevIns), 0, 0);
1022 /* If there's another pending interrupt, report it now. */
1023 if (pBusLogic->uPendingIntr)
1024 {
1025 buslogicSetInterrupt(pBusLogic, false, pBusLogic->uPendingIntr);
1026 pBusLogic->uPendingIntr = 0;
1027 }
1028}
1029
1030#if defined(IN_RING3)
1031
1032/**
1033 * Advances the mailbox pointer to the next slot.
1034 */
1035DECLINLINE(void) buslogicR3OutgoingMailboxAdvance(PBUSLOGIC pBusLogic)
1036{
1037 pBusLogic->uMailboxOutgoingPositionCurrent = (pBusLogic->uMailboxOutgoingPositionCurrent + 1) % pBusLogic->cMailbox;
1038}
1039
1040/**
1041 * Initialize local RAM of host adapter with default values.
1042 *
1043 * @returns nothing.
1044 * @param pBusLogic.
1045 */
1046static void buslogicR3InitializeLocalRam(PBUSLOGIC pBusLogic)
1047{
1048 /*
1049 * These values are mostly from what I think is right
1050 * looking at the dmesg output from a Linux guest inside
1051 * a VMware server VM.
1052 *
1053 * So they don't have to be right :)
1054 */
1055 memset(pBusLogic->LocalRam.u8View, 0, sizeof(HostAdapterLocalRam));
1056 pBusLogic->LocalRam.structured.autoSCSIData.fLevelSensitiveInterrupt = true;
1057 pBusLogic->LocalRam.structured.autoSCSIData.fParityCheckingEnabled = true;
1058 pBusLogic->LocalRam.structured.autoSCSIData.fExtendedTranslation = true; /* Same as in geometry register. */
1059 pBusLogic->LocalRam.structured.autoSCSIData.u16DeviceEnabledMask = ~0; /* All enabled. Maybe mask out non present devices? */
1060 pBusLogic->LocalRam.structured.autoSCSIData.u16WidePermittedMask = ~0;
1061 pBusLogic->LocalRam.structured.autoSCSIData.u16FastPermittedMask = ~0;
1062 pBusLogic->LocalRam.structured.autoSCSIData.u16SynchronousPermittedMask = ~0;
1063 pBusLogic->LocalRam.structured.autoSCSIData.u16DisconnectPermittedMask = ~0;
1064 pBusLogic->LocalRam.structured.autoSCSIData.fStrictRoundRobinMode = pBusLogic->fStrictRoundRobinMode;
1065 pBusLogic->LocalRam.structured.autoSCSIData.u16UltraPermittedMask = ~0;
1066 /** @todo calculate checksum? */
1067}
1068
1069/**
1070 * Do a hardware reset of the buslogic adapter.
1071 *
1072 * @returns VBox status code.
1073 * @param pBusLogic Pointer to the BusLogic device instance.
1074 * @param fResetIO Flag determining whether ISA I/O should be reset.
1075 */
1076static int buslogicR3HwReset(PBUSLOGIC pBusLogic, bool fResetIO)
1077{
1078 LogFlowFunc(("pBusLogic=%#p\n", pBusLogic));
1079
1080 /* Reset registers to default values. */
1081 pBusLogic->regStatus = BUSLOGIC_REGISTER_STATUS_HOST_ADAPTER_READY | BUSLOGIC_REGISTER_STATUS_INITIALIZATION_REQUIRED;
1082 pBusLogic->regGeometry = BUSLOGIC_REGISTER_GEOMETRY_EXTENTED_TRANSLATION_ENABLED;
1083 pBusLogic->uOperationCode = 0xff; /* No command executing. */
1084 pBusLogic->iParameter = 0;
1085 pBusLogic->cbCommandParametersLeft = 0;
1086 pBusLogic->fIRQEnabled = true;
1087 pBusLogic->fStrictRoundRobinMode = false;
1088 pBusLogic->fExtendedLunCCBFormat = false;
1089 pBusLogic->uMailboxOutgoingPositionCurrent = 0;
1090 pBusLogic->uMailboxIncomingPositionCurrent = 0;
1091
1092 /* Clear any active/pending interrupts. */
1093 pBusLogic->uPendingIntr = 0;
1094 buslogicClearInterrupt(pBusLogic);
1095
1096 /* Guest-initiated HBA reset does not affect ISA port I/O. */
1097 if (fResetIO)
1098 {
1099 buslogicR3RegisterISARange(pBusLogic, pBusLogic->uDefaultISABaseCode);
1100 }
1101 buslogicR3InitializeLocalRam(pBusLogic);
1102 vboxscsiInitialize(&pBusLogic->VBoxSCSI);
1103
1104 return VINF_SUCCESS;
1105}
1106
1107#endif /* IN_RING3 */
1108
1109/**
1110 * Resets the command state machine for the next command and notifies the guest.
1111 *
1112 * @returns nothing.
1113 * @param pBusLogic Pointer to the BusLogic device instance
1114 * @param fSuppressIrq Flag to suppress IRQ generation regardless of current state
1115 */
1116static void buslogicCommandComplete(PBUSLOGIC pBusLogic, bool fSuppressIrq)
1117{
1118 LogFlowFunc(("pBusLogic=%#p\n", pBusLogic));
1119
1120 pBusLogic->fUseLocalRam = false;
1121 pBusLogic->regStatus |= BUSLOGIC_REGISTER_STATUS_HOST_ADAPTER_READY;
1122 pBusLogic->iReply = 0;
1123
1124 /* Modify I/O address does not generate an interrupt. */
1125 if (pBusLogic->uOperationCode != BUSLOGICCOMMAND_EXECUTE_MAILBOX_COMMAND)
1126 {
1127 /* Notify that the command is complete. */
1128 pBusLogic->regStatus &= ~BUSLOGIC_REGISTER_STATUS_DATA_IN_REGISTER_READY;
1129 buslogicSetInterrupt(pBusLogic, fSuppressIrq, BUSLOGIC_REGISTER_INTERRUPT_COMMAND_COMPLETE);
1130 }
1131
1132 pBusLogic->uOperationCode = 0xff;
1133 pBusLogic->iParameter = 0;
1134}
1135
1136#if defined(IN_RING3)
1137
1138/**
1139 * Initiates a hard reset which was issued from the guest.
1140 *
1141 * @returns nothing
1142 * @param pBusLogic Pointer to the BusLogic device instance.
1143 * @param fHardReset Flag initiating a hard (vs. soft) reset.
1144 */
1145static void buslogicR3InitiateReset(PBUSLOGIC pBusLogic, bool fHardReset)
1146{
1147 LogFlowFunc(("pBusLogic=%#p fHardReset=%d\n", pBusLogic, fHardReset));
1148
1149 buslogicR3HwReset(pBusLogic, false);
1150
1151 if (fHardReset)
1152 {
1153 /* Set the diagnostic active bit in the status register and clear the ready state. */
1154 pBusLogic->regStatus |= BUSLOGIC_REGISTER_STATUS_DIAGNOSTIC_ACTIVE;
1155 pBusLogic->regStatus &= ~BUSLOGIC_REGISTER_STATUS_HOST_ADAPTER_READY;
1156
1157 /* Remember when the guest initiated a reset (after we're done resetting). */
1158 pBusLogic->u64ResetTime = PDMDevHlpTMTimeVirtGetNano(pBusLogic->CTX_SUFF(pDevIns));
1159 }
1160}
1161
1162/**
1163 * Send a mailbox with set status codes to the guest.
1164 *
1165 * @returns nothing.
1166 * @param pBusLogic Pointer to the BusLogic device instance.
1167 * @param pTaskState Pointer to the task state with the mailbox to send.
1168 * @param uHostAdapterStatus The host adapter status code to set.
1169 * @param uDeviceStatus The target device status to set.
1170 * @param uMailboxCompletionCode Completion status code to set in the mailbox.
1171 */
1172static void buslogicR3SendIncomingMailbox(PBUSLOGIC pBusLogic, PBUSLOGICTASKSTATE pTaskState,
1173 uint8_t uHostAdapterStatus, uint8_t uDeviceStatus,
1174 uint8_t uMailboxCompletionCode)
1175{
1176 pTaskState->MailboxGuest.u.in.uHostAdapterStatus = uHostAdapterStatus;
1177 pTaskState->MailboxGuest.u.in.uTargetDeviceStatus = uDeviceStatus;
1178 pTaskState->MailboxGuest.u.in.uCompletionCode = uMailboxCompletionCode;
1179
1180 int rc = PDMCritSectEnter(&pBusLogic->CritSectIntr, VINF_SUCCESS);
1181 AssertRC(rc);
1182 RTGCPHYS GCPhysAddrCCB = pTaskState->MailboxGuest.u32PhysAddrCCB;
1183 RTGCPHYS GCPhysAddrMailboxIncoming = pBusLogic->GCPhysAddrMailboxIncomingBase
1184 + ( pBusLogic->uMailboxIncomingPositionCurrent
1185 * (pTaskState->fIs24Bit ? sizeof(Mailbox24) : sizeof(Mailbox32)) );
1186 LogFlowFunc(("Completing CCB %RGp hstat=%u, dstat=%u, outgoing mailbox at %RGp\n", GCPhysAddrCCB,
1187 uHostAdapterStatus, uDeviceStatus, GCPhysAddrMailboxIncoming));
1188
1189 /* Update CCB. */
1190 pTaskState->CommandControlBlockGuest.c.uHostAdapterStatus = uHostAdapterStatus;
1191 pTaskState->CommandControlBlockGuest.c.uDeviceStatus = uDeviceStatus;
1192 /* Rewrite CCB up to the CDB; perhaps more than necessary. */
1193 PDMDevHlpPCIPhysWrite(pBusLogic->CTX_SUFF(pDevIns), GCPhysAddrCCB,
1194 &pTaskState->CommandControlBlockGuest, RT_OFFSETOF(CCBC, abCDB));
1195
1196# ifdef RT_STRICT
1197 uint8_t uCode;
1198 unsigned uCodeOffs = pTaskState->fIs24Bit ? RT_OFFSETOF(Mailbox24, uCmdState) : RT_OFFSETOF(Mailbox32, u.out.uActionCode);
1199 PDMDevHlpPhysRead(pBusLogic->CTX_SUFF(pDevIns), GCPhysAddrMailboxIncoming + uCodeOffs, &uCode, sizeof(uCode));
1200 Assert(uCode == BUSLOGIC_MAILBOX_INCOMING_COMPLETION_FREE);
1201# endif
1202
1203 /* Update mailbox. */
1204 if (pTaskState->fIs24Bit)
1205 {
1206 Mailbox24 Mbx24;
1207
1208 Mbx24.uCmdState = pTaskState->MailboxGuest.u.in.uCompletionCode;
1209 U32_TO_ADDR(Mbx24.aPhysAddrCCB, pTaskState->MailboxGuest.u32PhysAddrCCB);
1210 Log(("24-bit mailbox: completion code=%u, CCB at %RGp\n", Mbx24.uCmdState, (RTGCPHYS)ADDR_TO_U32(Mbx24.aPhysAddrCCB)));
1211 PDMDevHlpPCIPhysWrite(pBusLogic->CTX_SUFF(pDevIns), GCPhysAddrMailboxIncoming, &Mbx24, sizeof(Mailbox24));
1212 }
1213 else
1214 {
1215 Log(("32-bit mailbox: completion code=%u, CCB at %RGp\n", pTaskState->MailboxGuest.u.in.uCompletionCode, (RTGCPHYS)pTaskState->MailboxGuest.u32PhysAddrCCB));
1216 PDMDevHlpPCIPhysWrite(pBusLogic->CTX_SUFF(pDevIns), GCPhysAddrMailboxIncoming,
1217 &pTaskState->MailboxGuest, sizeof(Mailbox32));
1218 }
1219
1220 /* Advance to next mailbox position. */
1221 pBusLogic->uMailboxIncomingPositionCurrent++;
1222 if (pBusLogic->uMailboxIncomingPositionCurrent >= pBusLogic->cMailbox)
1223 pBusLogic->uMailboxIncomingPositionCurrent = 0;
1224
1225# ifdef LOG_ENABLED
1226 ASMAtomicIncU32(&pBusLogic->cInMailboxesReady);
1227# endif
1228
1229 buslogicSetInterrupt(pBusLogic, false, BUSLOGIC_REGISTER_INTERRUPT_INCOMING_MAILBOX_LOADED);
1230
1231 PDMCritSectLeave(&pBusLogic->CritSectIntr);
1232}
1233
1234# ifdef LOG_ENABLED
1235
1236/**
1237 * Dumps the content of a mailbox for debugging purposes.
1238 *
1239 * @return nothing
1240 * @param pMailbox The mailbox to dump.
1241 * @param fOutgoing true if dumping the outgoing state.
1242 * false if dumping the incoming state.
1243 */
1244static void buslogicR3DumpMailboxInfo(PMailbox32 pMailbox, bool fOutgoing)
1245{
1246 Log(("%s: Dump for %s mailbox:\n", __FUNCTION__, fOutgoing ? "outgoing" : "incoming"));
1247 Log(("%s: u32PhysAddrCCB=%#x\n", __FUNCTION__, pMailbox->u32PhysAddrCCB));
1248 if (fOutgoing)
1249 {
1250 Log(("%s: uActionCode=%u\n", __FUNCTION__, pMailbox->u.out.uActionCode));
1251 }
1252 else
1253 {
1254 Log(("%s: uHostAdapterStatus=%u\n", __FUNCTION__, pMailbox->u.in.uHostAdapterStatus));
1255 Log(("%s: uTargetDeviceStatus=%u\n", __FUNCTION__, pMailbox->u.in.uTargetDeviceStatus));
1256 Log(("%s: uCompletionCode=%u\n", __FUNCTION__, pMailbox->u.in.uCompletionCode));
1257 }
1258}
1259
1260/**
1261 * Dumps the content of a command control block for debugging purposes.
1262 *
1263 * @returns nothing.
1264 * @param pCCB Pointer to the command control block to dump.
1265 * @param fIs24BitCCB Flag to determine CCB format.
1266 */
1267static void buslogicR3DumpCCBInfo(PCCBU pCCB, bool fIs24BitCCB)
1268{
1269 Log(("%s: Dump for %s Command Control Block:\n", __FUNCTION__, fIs24BitCCB ? "24-bit" : "32-bit"));
1270 Log(("%s: uOpCode=%#x\n", __FUNCTION__, pCCB->c.uOpcode));
1271 Log(("%s: uDataDirection=%u\n", __FUNCTION__, pCCB->c.uDataDirection));
1272 Log(("%s: cbCDB=%u\n", __FUNCTION__, pCCB->c.cbCDB));
1273 Log(("%s: cbSenseData=%u\n", __FUNCTION__, pCCB->c.cbSenseData));
1274 Log(("%s: uHostAdapterStatus=%u\n", __FUNCTION__, pCCB->c.uHostAdapterStatus));
1275 Log(("%s: uDeviceStatus=%u\n", __FUNCTION__, pCCB->c.uDeviceStatus));
1276 if (fIs24BitCCB)
1277 {
1278 Log(("%s: cbData=%u\n", __FUNCTION__, LEN_TO_U32(pCCB->o.acbData)));
1279 Log(("%s: PhysAddrData=%#x\n", __FUNCTION__, ADDR_TO_U32(pCCB->o.aPhysAddrData)));
1280 Log(("%s: uTargetId=%u\n", __FUNCTION__, pCCB->o.uTargetId));
1281 Log(("%s: uLogicalUnit=%u\n", __FUNCTION__, pCCB->o.uLogicalUnit));
1282 }
1283 else
1284 {
1285 Log(("%s: cbData=%u\n", __FUNCTION__, pCCB->n.cbData));
1286 Log(("%s: PhysAddrData=%#x\n", __FUNCTION__, pCCB->n.u32PhysAddrData));
1287 Log(("%s: uTargetId=%u\n", __FUNCTION__, pCCB->n.uTargetId));
1288 Log(("%s: uLogicalUnit=%u\n", __FUNCTION__, pCCB->n.uLogicalUnit));
1289 Log(("%s: fTagQueued=%d\n", __FUNCTION__, pCCB->n.fTagQueued));
1290 Log(("%s: uQueueTag=%u\n", __FUNCTION__, pCCB->n.uQueueTag));
1291 Log(("%s: fLegacyTagEnable=%u\n", __FUNCTION__, pCCB->n.fLegacyTagEnable));
1292 Log(("%s: uLegacyQueueTag=%u\n", __FUNCTION__, pCCB->n.uLegacyQueueTag));
1293 Log(("%s: PhysAddrSenseData=%#x\n", __FUNCTION__, pCCB->n.u32PhysAddrSenseData));
1294 }
1295 Log(("%s: uCDB[0]=%#x\n", __FUNCTION__, pCCB->c.abCDB[0]));
1296 for (int i = 1; i < pCCB->c.cbCDB; i++)
1297 Log(("%s: uCDB[%d]=%u\n", __FUNCTION__, i, pCCB->c.abCDB[i]));
1298}
1299
1300# endif /* LOG_ENABLED */
1301
1302/**
1303 * Allocate data buffer.
1304 *
1305 * @param pTaskState Pointer to the task state.
1306 * @param GCSGList Guest physical address of S/G list.
1307 * @param cEntries Number of list entries to read.
1308 * @param pSGEList Pointer to 32-bit S/G list storage.
1309 */
1310static void buslogicR3ReadSGEntries(PBUSLOGICTASKSTATE pTaskState, RTGCPHYS GCSGList, uint32_t cEntries, SGE32 *pSGEList)
1311{
1312 PPDMDEVINS pDevIns = pTaskState->CTX_SUFF(pTargetDevice)->CTX_SUFF(pBusLogic)->CTX_SUFF(pDevIns);
1313 SGE24 aSGE24[32];
1314 Assert(cEntries <= RT_ELEMENTS(aSGE24));
1315
1316 /* Read the S/G entries. Convert 24-bit entries to 32-bit format. */
1317 if (pTaskState->fIs24Bit)
1318 {
1319 Log2(("Converting %u 24-bit S/G entries to 32-bit\n", cEntries));
1320 PDMDevHlpPhysRead(pDevIns, GCSGList, &aSGE24, cEntries * sizeof(SGE24));
1321 for (uint32_t i = 0; i < cEntries; ++i)
1322 {
1323 pSGEList[i].cbSegment = LEN_TO_U32(aSGE24[i].acbSegment);
1324 pSGEList[i].u32PhysAddrSegmentBase = ADDR_TO_U32(aSGE24[i].aPhysAddrSegmentBase);
1325 }
1326 }
1327 else
1328 PDMDevHlpPhysRead(pDevIns, GCSGList, pSGEList, cEntries * sizeof(SGE32));
1329}
1330
1331/**
1332 * Allocate data buffer.
1333 *
1334 * @returns VBox status code.
1335 * @param pTaskState Pointer to the task state.
1336 */
1337static int buslogicR3DataBufferAlloc(PBUSLOGICTASKSTATE pTaskState)
1338{
1339 PPDMDEVINS pDevIns = pTaskState->CTX_SUFF(pTargetDevice)->CTX_SUFF(pBusLogic)->CTX_SUFF(pDevIns);
1340 uint32_t cbDataCCB;
1341 uint32_t u32PhysAddrCCB;
1342
1343 /* Extract the data length and physical address from the CCB. */
1344 if (pTaskState->fIs24Bit)
1345 {
1346 u32PhysAddrCCB = ADDR_TO_U32(pTaskState->CommandControlBlockGuest.o.aPhysAddrData);
1347 cbDataCCB = LEN_TO_U32(pTaskState->CommandControlBlockGuest.o.acbData);
1348 }
1349 else
1350 {
1351 u32PhysAddrCCB = pTaskState->CommandControlBlockGuest.n.u32PhysAddrData;
1352 cbDataCCB = pTaskState->CommandControlBlockGuest.n.cbData;
1353 }
1354
1355 if ( (pTaskState->CommandControlBlockGuest.c.uDataDirection != BUSLOGIC_CCB_DIRECTION_NO_DATA)
1356 && cbDataCCB)
1357 {
1358 /** @todo Check following assumption and what residual means. */
1359 /*
1360 * The BusLogic adapter can handle two different data buffer formats.
1361 * The first one is that the data pointer entry in the CCB points to
1362 * the buffer directly. In second mode the data pointer points to a
1363 * scatter gather list which describes the buffer.
1364 */
1365 if ( (pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_SCATTER_GATHER)
1366 || (pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_SCATTER_GATHER))
1367 {
1368 uint32_t cScatterGatherGCRead;
1369 uint32_t iScatterGatherEntry;
1370 SGE32 aScatterGatherReadGC[32]; /* A buffer for scatter gather list entries read from guest memory. */
1371 uint32_t cScatterGatherGCLeft = cbDataCCB / pTaskState->cbSGEntry;
1372 RTGCPHYS GCPhysAddrScatterGatherCurrent = u32PhysAddrCCB;
1373 size_t cbDataToTransfer = 0;
1374
1375 /* Count number of bytes to transfer. */
1376 do
1377 {
1378 cScatterGatherGCRead = (cScatterGatherGCLeft < RT_ELEMENTS(aScatterGatherReadGC))
1379 ? cScatterGatherGCLeft
1380 : RT_ELEMENTS(aScatterGatherReadGC);
1381 cScatterGatherGCLeft -= cScatterGatherGCRead;
1382
1383 buslogicR3ReadSGEntries(pTaskState, GCPhysAddrScatterGatherCurrent, cScatterGatherGCRead, aScatterGatherReadGC);
1384
1385 for (iScatterGatherEntry = 0; iScatterGatherEntry < cScatterGatherGCRead; iScatterGatherEntry++)
1386 {
1387 RTGCPHYS GCPhysAddrDataBase;
1388
1389 Log(("%s: iScatterGatherEntry=%u\n", __FUNCTION__, iScatterGatherEntry));
1390
1391 GCPhysAddrDataBase = (RTGCPHYS)aScatterGatherReadGC[iScatterGatherEntry].u32PhysAddrSegmentBase;
1392 cbDataToTransfer += aScatterGatherReadGC[iScatterGatherEntry].cbSegment;
1393
1394 Log(("%s: GCPhysAddrDataBase=%RGp cbDataToTransfer=%u\n",
1395 __FUNCTION__, GCPhysAddrDataBase,
1396 aScatterGatherReadGC[iScatterGatherEntry].cbSegment));
1397 }
1398
1399 /* Set address to the next entries to read. */
1400 GCPhysAddrScatterGatherCurrent += cScatterGatherGCRead * pTaskState->cbSGEntry;
1401 } while (cScatterGatherGCLeft > 0);
1402
1403 Log(("%s: cbDataToTransfer=%d\n", __FUNCTION__, cbDataToTransfer));
1404
1405 /* Allocate buffer */
1406 pTaskState->DataSeg.cbSeg = cbDataToTransfer;
1407 pTaskState->DataSeg.pvSeg = RTMemAlloc(pTaskState->DataSeg.cbSeg);
1408 if (!pTaskState->DataSeg.pvSeg)
1409 return VERR_NO_MEMORY;
1410
1411 /* Copy the data if needed */
1412 if ( (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_OUT)
1413 || (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_UNKNOWN))
1414 {
1415 cScatterGatherGCLeft = cbDataCCB / pTaskState->cbSGEntry;
1416 GCPhysAddrScatterGatherCurrent = u32PhysAddrCCB;
1417 uint8_t *pbData = (uint8_t *)pTaskState->DataSeg.pvSeg;
1418
1419 do
1420 {
1421 cScatterGatherGCRead = (cScatterGatherGCLeft < RT_ELEMENTS(aScatterGatherReadGC))
1422 ? cScatterGatherGCLeft
1423 : RT_ELEMENTS(aScatterGatherReadGC);
1424 cScatterGatherGCLeft -= cScatterGatherGCRead;
1425
1426 buslogicR3ReadSGEntries(pTaskState, GCPhysAddrScatterGatherCurrent, cScatterGatherGCRead, aScatterGatherReadGC);
1427
1428 for (iScatterGatherEntry = 0; iScatterGatherEntry < cScatterGatherGCRead; iScatterGatherEntry++)
1429 {
1430 RTGCPHYS GCPhysAddrDataBase;
1431
1432 Log(("%s: iScatterGatherEntry=%u\n", __FUNCTION__, iScatterGatherEntry));
1433
1434 GCPhysAddrDataBase = (RTGCPHYS)aScatterGatherReadGC[iScatterGatherEntry].u32PhysAddrSegmentBase;
1435 cbDataToTransfer = aScatterGatherReadGC[iScatterGatherEntry].cbSegment;
1436
1437 Log(("%s: GCPhysAddrDataBase=%RGp cbDataToTransfer=%u\n", __FUNCTION__, GCPhysAddrDataBase, cbDataToTransfer));
1438
1439 PDMDevHlpPhysRead(pDevIns, GCPhysAddrDataBase, pbData, cbDataToTransfer);
1440 pbData += cbDataToTransfer;
1441 }
1442
1443 /* Set address to the next entries to read. */
1444 GCPhysAddrScatterGatherCurrent += cScatterGatherGCRead * pTaskState->cbSGEntry;
1445 } while (cScatterGatherGCLeft > 0);
1446 }
1447
1448 }
1449 else if ( pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB
1450 || pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_DATA_LENGTH)
1451 {
1452 /* The buffer is not scattered. */
1453 RTGCPHYS GCPhysAddrDataBase = u32PhysAddrCCB;
1454
1455 AssertMsg(GCPhysAddrDataBase != 0, ("Physical address is 0\n"));
1456
1457 pTaskState->DataSeg.cbSeg = cbDataCCB;
1458 pTaskState->DataSeg.pvSeg = RTMemAlloc(pTaskState->DataSeg.cbSeg);
1459 if (!pTaskState->DataSeg.pvSeg)
1460 return VERR_NO_MEMORY;
1461
1462 Log(("Non scattered buffer:\n"));
1463 Log(("u32PhysAddrData=%#x\n", u32PhysAddrCCB));
1464 Log(("cbData=%u\n", cbDataCCB));
1465 Log(("GCPhysAddrDataBase=0x%RGp\n", GCPhysAddrDataBase));
1466
1467 /* Copy the data into the buffer. */
1468 PDMDevHlpPhysRead(pDevIns, GCPhysAddrDataBase, pTaskState->DataSeg.pvSeg, pTaskState->DataSeg.cbSeg);
1469 }
1470 }
1471
1472 return VINF_SUCCESS;
1473}
1474
1475/**
1476 * Free allocated resources used for the scatter gather list.
1477 *
1478 * @returns nothing.
1479 * @param pTaskState Pointer to the task state.
1480 */
1481static void buslogicR3DataBufferFree(PBUSLOGICTASKSTATE pTaskState)
1482{
1483 PPDMDEVINS pDevIns = pTaskState->CTX_SUFF(pTargetDevice)->CTX_SUFF(pBusLogic)->CTX_SUFF(pDevIns);
1484 uint32_t cbDataCCB;
1485 uint32_t u32PhysAddrCCB;
1486
1487 /* Extract the data length and physical address from the CCB. */
1488 if (pTaskState->fIs24Bit)
1489 {
1490 u32PhysAddrCCB = ADDR_TO_U32(pTaskState->CommandControlBlockGuest.o.aPhysAddrData);
1491 cbDataCCB = LEN_TO_U32(pTaskState->CommandControlBlockGuest.o.acbData);
1492 }
1493 else
1494 {
1495 u32PhysAddrCCB = pTaskState->CommandControlBlockGuest.n.u32PhysAddrData;
1496 cbDataCCB = pTaskState->CommandControlBlockGuest.n.cbData;
1497 }
1498
1499#if 1
1500 /* Hack for NT 10/91: A CCB describes a 2K buffer, but TEST UNIT READY is executed. This command
1501 * returns no data, hence the buffer must be left alone!
1502 */
1503 if (pTaskState->CommandControlBlockGuest.c.abCDB[0] == 0)
1504 cbDataCCB = 0;
1505#endif
1506
1507 LogFlowFunc(("pTaskState=%#p cbDataCCB=%u direction=%u cbSeg=%u\n", pTaskState, cbDataCCB,
1508 pTaskState->CommandControlBlockGuest.c.uDataDirection, pTaskState->DataSeg.cbSeg));
1509
1510 if ( (cbDataCCB > 0)
1511 && ( (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_IN)
1512 || (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_UNKNOWN)))
1513 {
1514 if ( (pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_SCATTER_GATHER)
1515 || (pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_SCATTER_GATHER))
1516 {
1517 uint32_t cScatterGatherGCRead;
1518 uint32_t iScatterGatherEntry;
1519 SGE32 aScatterGatherReadGC[32]; /* Number of scatter gather list entries read from guest memory. */
1520 uint32_t cScatterGatherGCLeft = cbDataCCB / pTaskState->cbSGEntry;
1521 RTGCPHYS GCPhysAddrScatterGatherCurrent = u32PhysAddrCCB;
1522 uint8_t *pbData = (uint8_t *)pTaskState->DataSeg.pvSeg;
1523
1524 do
1525 {
1526 cScatterGatherGCRead = (cScatterGatherGCLeft < RT_ELEMENTS(aScatterGatherReadGC))
1527 ? cScatterGatherGCLeft
1528 : RT_ELEMENTS(aScatterGatherReadGC);
1529 cScatterGatherGCLeft -= cScatterGatherGCRead;
1530
1531 buslogicR3ReadSGEntries(pTaskState, GCPhysAddrScatterGatherCurrent, cScatterGatherGCRead, aScatterGatherReadGC);
1532
1533 for (iScatterGatherEntry = 0; iScatterGatherEntry < cScatterGatherGCRead; iScatterGatherEntry++)
1534 {
1535 RTGCPHYS GCPhysAddrDataBase;
1536 size_t cbDataToTransfer;
1537
1538 Log(("%s: iScatterGatherEntry=%u\n", __FUNCTION__, iScatterGatherEntry));
1539
1540 GCPhysAddrDataBase = (RTGCPHYS)aScatterGatherReadGC[iScatterGatherEntry].u32PhysAddrSegmentBase;
1541 cbDataToTransfer = aScatterGatherReadGC[iScatterGatherEntry].cbSegment;
1542
1543 Log(("%s: GCPhysAddrDataBase=%RGp cbDataToTransfer=%u\n", __FUNCTION__, GCPhysAddrDataBase, cbDataToTransfer));
1544
1545 PDMDevHlpPCIPhysWrite(pDevIns, GCPhysAddrDataBase, pbData, cbDataToTransfer);
1546 pbData += cbDataToTransfer;
1547 }
1548
1549 /* Set address to the next entries to read. */
1550 GCPhysAddrScatterGatherCurrent += cScatterGatherGCRead * pTaskState->cbSGEntry;
1551 } while (cScatterGatherGCLeft > 0);
1552
1553 }
1554 else if ( pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB
1555 || pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_DATA_LENGTH)
1556 {
1557 /* The buffer is not scattered. */
1558 RTGCPHYS GCPhysAddrDataBase = u32PhysAddrCCB;
1559
1560 AssertMsg(GCPhysAddrDataBase != 0, ("Physical address is 0\n"));
1561
1562 Log(("Non-scattered buffer:\n"));
1563 Log(("u32PhysAddrData=%#x\n", u32PhysAddrCCB));
1564 Log(("cbData=%u\n", cbDataCCB));
1565 Log(("GCPhysAddrDataBase=0x%RGp\n", GCPhysAddrDataBase));
1566
1567 /* Copy the data into the guest memory. */
1568 PDMDevHlpPCIPhysWrite(pDevIns, GCPhysAddrDataBase, pTaskState->DataSeg.pvSeg, pTaskState->DataSeg.cbSeg);
1569 }
1570
1571 }
1572 /* Update residual data length. */
1573 if ( (pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_DATA_LENGTH)
1574 || (pTaskState->CommandControlBlockGuest.c.uOpcode == BUSLOGIC_CCB_OPCODE_INITIATOR_CCB_RESIDUAL_SCATTER_GATHER))
1575 {
1576 uint32_t cbResidual;
1577
1578 /** @todo we need to get the actual transfer length from the VSCSI layer?! */
1579 cbResidual = 0; //LEN_TO_U32(pTaskState->CCBGuest.acbData) - ???;
1580 if (pTaskState->fIs24Bit)
1581 U32_TO_LEN(pTaskState->CommandControlBlockGuest.o.acbData, cbResidual);
1582 else
1583 pTaskState->CommandControlBlockGuest.n.cbData = cbResidual;
1584 }
1585
1586 RTMemFree(pTaskState->DataSeg.pvSeg);
1587 pTaskState->DataSeg.pvSeg = NULL;
1588 pTaskState->DataSeg.cbSeg = 0;
1589}
1590
1591/** Convert sense buffer length taking into account shortcut values. */
1592static uint32_t buslogicR3ConvertSenseBufferLength(uint32_t cbSense)
1593{
1594 /* Convert special sense buffer length values. */
1595 if (cbSense == 0)
1596 cbSense = 14; /* 0 means standard 14-byte buffer. */
1597 else if (cbSense == 1)
1598 cbSense = 0; /* 1 means no sense data. */
1599 else if (cbSense < 8)
1600 AssertMsgFailed(("Reserved cbSense value of %d used!\n", cbSense));
1601
1602 return cbSense;
1603}
1604
1605/**
1606 * Free the sense buffer.
1607 *
1608 * @returns nothing.
1609 * @param pTaskState Pointer to the task state.
1610 * @param fCopy If sense data should be copied to guest memory.
1611 */
1612static void buslogicR3SenseBufferFree(PBUSLOGICTASKSTATE pTaskState, bool fCopy)
1613{
1614 uint32_t cbSenseBuffer;
1615
1616 cbSenseBuffer = buslogicR3ConvertSenseBufferLength(pTaskState->CommandControlBlockGuest.c.cbSenseData);
1617
1618 /* Copy the sense buffer into guest memory if requested. */
1619 if (fCopy && cbSenseBuffer)
1620 {
1621 PPDMDEVINS pDevIns = pTaskState->CTX_SUFF(pTargetDevice)->CTX_SUFF(pBusLogic)->CTX_SUFF(pDevIns);
1622 RTGCPHYS GCPhysAddrSenseBuffer;
1623
1624 /* With 32-bit CCBs, the (optional) sense buffer physical address is provided separately.
1625 * On the other hand, with 24-bit CCBs, the sense buffer is simply located at the end of
1626 * the CCB, right after the variable-length CDB.
1627 */
1628 if (pTaskState->fIs24Bit)
1629 {
1630 GCPhysAddrSenseBuffer = pTaskState->MailboxGuest.u32PhysAddrCCB;
1631 GCPhysAddrSenseBuffer += pTaskState->CommandControlBlockGuest.c.cbCDB + RT_OFFSETOF(CCB24, abCDB);
1632 }
1633 else
1634 GCPhysAddrSenseBuffer = pTaskState->CommandControlBlockGuest.n.u32PhysAddrSenseData;
1635
1636 Log3(("%s: sense buffer: %.*Rhxs\n", __FUNCTION__, cbSenseBuffer, pTaskState->pbSenseBuffer));
1637 PDMDevHlpPCIPhysWrite(pDevIns, GCPhysAddrSenseBuffer, pTaskState->pbSenseBuffer, cbSenseBuffer);
1638 }
1639
1640 RTMemFree(pTaskState->pbSenseBuffer);
1641 pTaskState->pbSenseBuffer = NULL;
1642}
1643
1644/**
1645 * Alloc the sense buffer.
1646 *
1647 * @returns VBox status code.
1648 * @param pTaskState Pointer to the task state.
1649 * @note Current assumption is that the sense buffer is not scattered and does not cross a page boundary.
1650 */
1651static int buslogicR3SenseBufferAlloc(PBUSLOGICTASKSTATE pTaskState)
1652{
1653 PPDMDEVINS pDevIns = pTaskState->CTX_SUFF(pTargetDevice)->CTX_SUFF(pBusLogic)->CTX_SUFF(pDevIns);
1654 uint32_t cbSenseBuffer;
1655
1656 pTaskState->pbSenseBuffer = NULL;
1657
1658 cbSenseBuffer = buslogicR3ConvertSenseBufferLength(pTaskState->CommandControlBlockGuest.c.cbSenseData);
1659 if (cbSenseBuffer)
1660 {
1661 pTaskState->pbSenseBuffer = (uint8_t *)RTMemAllocZ(cbSenseBuffer);
1662 if (!pTaskState->pbSenseBuffer)
1663 return VERR_NO_MEMORY;
1664 }
1665
1666 return VINF_SUCCESS;
1667}
1668
1669#endif /* IN_RING3 */
1670
1671/**
1672 * Parses the command buffer and executes it.
1673 *
1674 * @returns VBox status code.
1675 * @param pBusLogic Pointer to the BusLogic device instance.
1676 */
1677static int buslogicProcessCommand(PBUSLOGIC pBusLogic)
1678{
1679 int rc = VINF_SUCCESS;
1680 bool fSuppressIrq = false;
1681
1682 LogFlowFunc(("pBusLogic=%#p\n", pBusLogic));
1683 AssertMsg(pBusLogic->uOperationCode != 0xff, ("There is no command to execute\n"));
1684
1685 switch (pBusLogic->uOperationCode)
1686 {
1687 case BUSLOGICCOMMAND_TEST_CMDC_INTERRUPT:
1688 /* Valid command, no reply. */
1689 pBusLogic->cbReplyParametersLeft = 0;
1690 break;
1691 case BUSLOGICCOMMAND_INQUIRE_PCI_HOST_ADAPTER_INFORMATION:
1692 {
1693 PReplyInquirePCIHostAdapterInformation pReply = (PReplyInquirePCIHostAdapterInformation)pBusLogic->aReplyBuffer;
1694 memset(pReply, 0, sizeof(ReplyInquirePCIHostAdapterInformation));
1695
1696 /* It seems VMware does not provide valid information here too, lets do the same :) */
1697 pReply->InformationIsValid = 0;
1698 pReply->IsaIOPort = pBusLogic->uISABaseCode;
1699 pReply->IRQ = PCIDevGetInterruptLine(&pBusLogic->dev);
1700 pBusLogic->cbReplyParametersLeft = sizeof(ReplyInquirePCIHostAdapterInformation);
1701 break;
1702 }
1703 case BUSLOGICCOMMAND_MODIFY_IO_ADDRESS:
1704 {
1705 /* Modify the ISA-compatible I/O port base. Note that this technically
1706 * violates the PCI spec, as this address is not reported through PCI.
1707 * However, it is required for compatibility with old drivers.
1708 */
1709#ifdef IN_RING3
1710 Log(("ISA I/O for PCI (code %x)\n", pBusLogic->aCommandBuffer[0]));
1711 buslogicR3RegisterISARange(pBusLogic, pBusLogic->aCommandBuffer[0]);
1712 pBusLogic->cbReplyParametersLeft = 0;
1713 fSuppressIrq = true;
1714 break;
1715#else
1716 AssertMsgFailed(("Must never get here!\n"));
1717#endif
1718 }
1719 case BUSLOGICCOMMAND_INQUIRE_BOARD_ID:
1720 {
1721 /* The special option byte is important: If it is '0' or 'B', Windows NT drivers
1722 * for Adaptec AHA-154x may claim the adapter. The BusLogic drivers will claim
1723 * the adapter only when the byte is *not* '0' or 'B'.
1724 */
1725 pBusLogic->aReplyBuffer[0] = 'A'; /* Firmware option bytes */
1726 pBusLogic->aReplyBuffer[1] = 'A'; /* Special option byte */
1727
1728 /* We report version 5.07B. This reply will provide the first two digits. */
1729 pBusLogic->aReplyBuffer[2] = '5'; /* Major version 5 */
1730 pBusLogic->aReplyBuffer[3] = '0'; /* Minor version 0 */
1731 pBusLogic->cbReplyParametersLeft = 4; /* Reply is 4 bytes long */
1732 break;
1733 }
1734 case BUSLOGICCOMMAND_INQUIRE_FIRMWARE_VERSION_3RD_LETTER:
1735 {
1736 pBusLogic->aReplyBuffer[0] = '7';
1737 pBusLogic->cbReplyParametersLeft = 1;
1738 break;
1739 }
1740 case BUSLOGICCOMMAND_INQUIRE_FIRMWARE_VERSION_LETTER:
1741 {
1742 pBusLogic->aReplyBuffer[0] = 'B';
1743 pBusLogic->cbReplyParametersLeft = 1;
1744 break;
1745 }
1746 case BUSLOGICCOMMAND_SET_ADAPTER_OPTIONS:
1747 /* The parameter list length is determined by the first byte of the command buffer. */
1748 if (pBusLogic->iParameter == 1)
1749 {
1750 /* First pass - set the number of following parameter bytes. */
1751 pBusLogic->cbCommandParametersLeft = pBusLogic->aCommandBuffer[0];
1752 Log(("Set HA options: %u bytes follow\n", pBusLogic->cbCommandParametersLeft));
1753 }
1754 else
1755 {
1756 /* Second pass - process received data. */
1757 Log(("Set HA options: received %u bytes\n", pBusLogic->aCommandBuffer[0]));
1758 /* We ignore the data - it only concerns the SCSI hardware protocol. */
1759 }
1760 pBusLogic->cbReplyParametersLeft = 0;
1761 break;
1762
1763 case BUSLOGICCOMMAND_INQUIRE_HOST_ADAPTER_MODEL_NUMBER:
1764 {
1765 /* The reply length is set by the guest and is found in the first byte of the command buffer. */
1766 pBusLogic->cbReplyParametersLeft = pBusLogic->aCommandBuffer[0];
1767 memset(pBusLogic->aReplyBuffer, ' ', pBusLogic->cbReplyParametersLeft);
1768 const char aModelName[] = "958";
1769 int cCharsToTransfer = (pBusLogic->cbReplyParametersLeft <= (sizeof(aModelName) - 1))
1770 ? pBusLogic->cbReplyParametersLeft
1771 : sizeof(aModelName) - 1;
1772
1773 for (int i = 0; i < cCharsToTransfer; i++)
1774 pBusLogic->aReplyBuffer[i] = aModelName[i];
1775
1776 break;
1777 }
1778 case BUSLOGICCOMMAND_INQUIRE_CONFIGURATION:
1779 {
1780 uint8_t uPciIrq = PCIDevGetInterruptLine(&pBusLogic->dev);
1781
1782 pBusLogic->cbReplyParametersLeft = sizeof(ReplyInquireConfiguration);
1783 PReplyInquireConfiguration pReply = (PReplyInquireConfiguration)pBusLogic->aReplyBuffer;
1784 memset(pReply, 0, sizeof(ReplyInquireConfiguration));
1785
1786 pReply->uHostAdapterId = 7; /* The controller has always 7 as ID. */
1787 pReply->fDmaChannel6 = 1; /* DMA channel 6 is a good default. */
1788 /* The PCI IRQ is not necessarily representable in this structure.
1789 * If that is the case, the guest likely won't function correctly,
1790 * therefore we log a warning.
1791 */
1792 switch (uPciIrq)
1793 {
1794 case 9: pReply->fIrqChannel9 = 1; break;
1795 case 10: pReply->fIrqChannel10 = 1; break;
1796 case 11: pReply->fIrqChannel11 = 1; break;
1797 case 12: pReply->fIrqChannel12 = 1; break;
1798 case 14: pReply->fIrqChannel14 = 1; break;
1799 case 15: pReply->fIrqChannel15 = 1; break;
1800 default:
1801 LogRel(("Warning: PCI IRQ %d cannot be represented as ISA!\n", uPciIrq));
1802 break;
1803 }
1804 break;
1805 }
1806 case BUSLOGICCOMMAND_INQUIRE_EXTENDED_SETUP_INFORMATION:
1807 {
1808 /* Some Adaptec AHA-154x drivers (e.g. OS/2) execute this command and expect
1809 * it to fail. If it succeeds, the drivers refuse to load. However, some newer
1810 * Adaptec 154x models supposedly support it too??
1811 */
1812
1813 /* The reply length is set by the guest and is found in the first byte of the command buffer. */
1814 pBusLogic->cbReplyParametersLeft = pBusLogic->aCommandBuffer[0];
1815 PReplyInquireExtendedSetupInformation pReply = (PReplyInquireExtendedSetupInformation)pBusLogic->aReplyBuffer;
1816 memset(pReply, 0, sizeof(ReplyInquireExtendedSetupInformation));
1817
1818 /** @todo should this reflect the RAM contents (AutoSCSIRam)? */
1819 pReply->uBusType = 'E'; /* EISA style */
1820 pReply->u16ScatterGatherLimit = 8192;
1821 pReply->cMailbox = pBusLogic->cMailbox;
1822 pReply->uMailboxAddressBase = (uint32_t)pBusLogic->GCPhysAddrMailboxOutgoingBase;
1823 pReply->fLevelSensitiveInterrupt = true;
1824 pReply->fHostWideSCSI = true;
1825 pReply->fHostUltraSCSI = true;
1826 memcpy(pReply->aFirmwareRevision, "07B", sizeof(pReply->aFirmwareRevision));
1827
1828 break;
1829 }
1830 case BUSLOGICCOMMAND_INQUIRE_SETUP_INFORMATION:
1831 {
1832 /* The reply length is set by the guest and is found in the first byte of the command buffer. */
1833 pBusLogic->cbReplyParametersLeft = pBusLogic->aCommandBuffer[0];
1834 PReplyInquireSetupInformation pReply = (PReplyInquireSetupInformation)pBusLogic->aReplyBuffer;
1835 memset(pReply, 0, sizeof(ReplyInquireSetupInformation));
1836 pReply->fSynchronousInitiationEnabled = true;
1837 pReply->fParityCheckingEnabled = true;
1838 pReply->cMailbox = pBusLogic->cMailbox;
1839 U32_TO_ADDR(pReply->MailboxAddress, pBusLogic->GCPhysAddrMailboxOutgoingBase);
1840 pReply->uSignature = 'B';
1841 /* The 'D' signature prevents Adaptec's OS/2 drivers from getting too
1842 * friendly with BusLogic hardware and upsetting the HBA state.
1843 */
1844 pReply->uCharacterD = 'D'; /* BusLogic model. */
1845 pReply->uHostBusType = 'F'; /* PCI bus. */
1846 break;
1847 }
1848 case BUSLOGICCOMMAND_FETCH_HOST_ADAPTER_LOCAL_RAM:
1849 {
1850 /*
1851 * First element in the command buffer contains start offset to read from
1852 * and second one the number of bytes to read.
1853 */
1854 uint8_t uOffset = pBusLogic->aCommandBuffer[0];
1855 pBusLogic->cbReplyParametersLeft = pBusLogic->aCommandBuffer[1];
1856
1857 pBusLogic->fUseLocalRam = true;
1858 pBusLogic->iReply = uOffset;
1859 break;
1860 }
1861 case BUSLOGICCOMMAND_INITIALIZE_MAILBOX:
1862 {
1863 PRequestInitMbx pRequest = (PRequestInitMbx)pBusLogic->aCommandBuffer;
1864
1865 pBusLogic->fMbxIs24Bit = true;
1866 pBusLogic->cMailbox = pRequest->cMailbox;
1867 pBusLogic->GCPhysAddrMailboxOutgoingBase = (RTGCPHYS)ADDR_TO_U32(pRequest->aMailboxBaseAddr);
1868 /* The area for incoming mailboxes is right after the last entry of outgoing mailboxes. */
1869 pBusLogic->GCPhysAddrMailboxIncomingBase = pBusLogic->GCPhysAddrMailboxOutgoingBase + (pBusLogic->cMailbox * sizeof(Mailbox24));
1870
1871 Log(("GCPhysAddrMailboxOutgoingBase=%RGp\n", pBusLogic->GCPhysAddrMailboxOutgoingBase));
1872 Log(("GCPhysAddrMailboxIncomingBase=%RGp\n", pBusLogic->GCPhysAddrMailboxIncomingBase));
1873 Log(("cMailboxes=%u (24-bit mode)\n", pBusLogic->cMailbox));
1874 LogRel(("Initialized 24-bit mailbox, %d entries at %08x\n", pRequest->cMailbox, ADDR_TO_U32(pRequest->aMailboxBaseAddr)));
1875
1876 pBusLogic->regStatus &= ~BUSLOGIC_REGISTER_STATUS_INITIALIZATION_REQUIRED;
1877 pBusLogic->cbReplyParametersLeft = 0;
1878 break;
1879 }
1880 case BUSLOGICCOMMAND_INITIALIZE_EXTENDED_MAILBOX:
1881 {
1882 PRequestInitializeExtendedMailbox pRequest = (PRequestInitializeExtendedMailbox)pBusLogic->aCommandBuffer;
1883
1884 pBusLogic->fMbxIs24Bit = false;
1885 pBusLogic->cMailbox = pRequest->cMailbox;
1886 pBusLogic->GCPhysAddrMailboxOutgoingBase = (RTGCPHYS)pRequest->uMailboxBaseAddress;
1887 /* The area for incoming mailboxes is right after the last entry of outgoing mailboxes. */
1888 pBusLogic->GCPhysAddrMailboxIncomingBase = (RTGCPHYS)pRequest->uMailboxBaseAddress + (pBusLogic->cMailbox * sizeof(Mailbox32));
1889
1890 Log(("GCPhysAddrMailboxOutgoingBase=%RGp\n", pBusLogic->GCPhysAddrMailboxOutgoingBase));
1891 Log(("GCPhysAddrMailboxIncomingBase=%RGp\n", pBusLogic->GCPhysAddrMailboxIncomingBase));
1892 Log(("cMailboxes=%u (32-bit mode)\n", pBusLogic->cMailbox));
1893 LogRel(("Initialized 32-bit mailbox, %d entries at %08x\n", pRequest->cMailbox, pRequest->uMailboxBaseAddress));
1894
1895 pBusLogic->regStatus &= ~BUSLOGIC_REGISTER_STATUS_INITIALIZATION_REQUIRED;
1896 pBusLogic->cbReplyParametersLeft = 0;
1897 break;
1898 }
1899 case BUSLOGICCOMMAND_ENABLE_STRICT_ROUND_ROBIN_MODE:
1900 {
1901 if (pBusLogic->aCommandBuffer[0] == 0)
1902 pBusLogic->fStrictRoundRobinMode = false;
1903 else if (pBusLogic->aCommandBuffer[0] == 1)
1904 pBusLogic->fStrictRoundRobinMode = true;
1905 else
1906 AssertMsgFailed(("Invalid round robin mode %d\n", pBusLogic->aCommandBuffer[0]));
1907
1908 pBusLogic->cbReplyParametersLeft = 0;
1909 break;
1910 }
1911 case BUSLOGICCOMMAND_SET_CCB_FORMAT:
1912 {
1913 if (pBusLogic->aCommandBuffer[0] == 0)
1914 pBusLogic->fExtendedLunCCBFormat = false;
1915 else if (pBusLogic->aCommandBuffer[0] == 1)
1916 pBusLogic->fExtendedLunCCBFormat = true;
1917 else
1918 AssertMsgFailed(("Invalid CCB format %d\n", pBusLogic->aCommandBuffer[0]));
1919
1920 pBusLogic->cbReplyParametersLeft = 0;
1921 break;
1922 }
1923 case BUSLOGICCOMMAND_INQUIRE_INSTALLED_DEVICES_ID_0_TO_7:
1924 /* This is supposed to send TEST UNIT READY to each target/LUN.
1925 * We cheat and skip that, since we already know what's attached
1926 */
1927 memset(pBusLogic->aReplyBuffer, 0, 8);
1928 for (int i = 0; i < 8; ++i)
1929 {
1930 if (pBusLogic->aDeviceStates[i].fPresent)
1931 pBusLogic->aReplyBuffer[i] = 1;
1932 }
1933 pBusLogic->aReplyBuffer[7] = 0; /* HA hardcoded at ID 7. */
1934 pBusLogic->cbReplyParametersLeft = 8;
1935 break;
1936 case BUSLOGICCOMMAND_INQUIRE_INSTALLED_DEVICES_ID_8_TO_15:
1937 /* See note about cheating above. */
1938 memset(pBusLogic->aReplyBuffer, 0, 8);
1939 for (int i = 0; i < 8; ++i)
1940 {
1941 if (pBusLogic->aDeviceStates[i + 8].fPresent)
1942 pBusLogic->aReplyBuffer[i] = 1;
1943 }
1944 pBusLogic->cbReplyParametersLeft = 8;
1945 break;
1946 case BUSLOGICCOMMAND_INQUIRE_TARGET_DEVICES:
1947 {
1948 /* Each bit which is set in the 16bit wide variable means a present device. */
1949 uint16_t u16TargetsPresentMask = 0;
1950
1951 for (uint8_t i = 0; i < RT_ELEMENTS(pBusLogic->aDeviceStates); i++)
1952 {
1953 if (pBusLogic->aDeviceStates[i].fPresent)
1954 u16TargetsPresentMask |= (1 << i);
1955 }
1956 pBusLogic->aReplyBuffer[0] = (uint8_t)u16TargetsPresentMask;
1957 pBusLogic->aReplyBuffer[1] = (uint8_t)(u16TargetsPresentMask >> 8);
1958 pBusLogic->cbReplyParametersLeft = 2;
1959 break;
1960 }
1961 case BUSLOGICCOMMAND_INQUIRE_SYNCHRONOUS_PERIOD:
1962 {
1963 pBusLogic->cbReplyParametersLeft = pBusLogic->aCommandBuffer[0];
1964
1965 for (uint8_t i = 0; i < pBusLogic->cbReplyParametersLeft; i++)
1966 pBusLogic->aReplyBuffer[i] = 0; /** @todo Figure if we need something other here. It's not needed for the linux driver */
1967
1968 break;
1969 }
1970 case BUSLOGICCOMMAND_DISABLE_HOST_ADAPTER_INTERRUPT:
1971 {
1972 if (pBusLogic->aCommandBuffer[0] == 0)
1973 pBusLogic->fIRQEnabled = false;
1974 else
1975 pBusLogic->fIRQEnabled = true;
1976 /* No interrupt signaled regardless of enable/disable. */
1977 fSuppressIrq = true;
1978 break;
1979 }
1980 case BUSLOGICCOMMAND_ECHO_COMMAND_DATA:
1981 {
1982 pBusLogic->aReplyBuffer[0] = pBusLogic->aCommandBuffer[0];
1983 pBusLogic->cbReplyParametersLeft = 1;
1984 break;
1985 }
1986 case BUSLOGICCOMMAND_SET_PREEMPT_TIME_ON_BUS:
1987 {
1988 pBusLogic->cbReplyParametersLeft = 0;
1989 pBusLogic->LocalRam.structured.autoSCSIData.uBusOnDelay = pBusLogic->aCommandBuffer[0];
1990 Log(("Bus-on time: %d\n", pBusLogic->aCommandBuffer[0]));
1991 break;
1992 }
1993 case BUSLOGICCOMMAND_SET_TIME_OFF_BUS:
1994 {
1995 pBusLogic->cbReplyParametersLeft = 0;
1996 pBusLogic->LocalRam.structured.autoSCSIData.uBusOffDelay = pBusLogic->aCommandBuffer[0];
1997 Log(("Bus-off time: %d\n", pBusLogic->aCommandBuffer[0]));
1998 break;
1999 }
2000 case BUSLOGICCOMMAND_SET_BUS_TRANSFER_RATE:
2001 {
2002 pBusLogic->cbReplyParametersLeft = 0;
2003 pBusLogic->LocalRam.structured.autoSCSIData.uDMATransferRate = pBusLogic->aCommandBuffer[0];
2004 Log(("Bus transfer rate: %02X\n", pBusLogic->aCommandBuffer[0]));
2005 break;
2006 }
2007 case BUSLOGICCOMMAND_WRITE_BUSMASTER_CHIP_FIFO:
2008 {
2009 RTGCPHYS GCPhysFifoBuf;
2010 Addr24 addr;
2011
2012 pBusLogic->cbReplyParametersLeft = 0;
2013 addr.hi = pBusLogic->aCommandBuffer[0];
2014 addr.mid = pBusLogic->aCommandBuffer[1];
2015 addr.lo = pBusLogic->aCommandBuffer[2];
2016 GCPhysFifoBuf = (RTGCPHYS)ADDR_TO_U32(addr);
2017 Log(("Write busmaster FIFO at: %04X\n", ADDR_TO_U32(addr)));
2018 PDMDevHlpPhysRead(pBusLogic->CTX_SUFF(pDevIns), GCPhysFifoBuf,
2019 &pBusLogic->LocalRam.u8View[64], 64);
2020 break;
2021 }
2022 case BUSLOGICCOMMAND_READ_BUSMASTER_CHIP_FIFO:
2023 {
2024 RTGCPHYS GCPhysFifoBuf;
2025 Addr24 addr;
2026
2027 pBusLogic->cbReplyParametersLeft = 0;
2028 addr.hi = pBusLogic->aCommandBuffer[0];
2029 addr.mid = pBusLogic->aCommandBuffer[1];
2030 addr.lo = pBusLogic->aCommandBuffer[2];
2031 GCPhysFifoBuf = (RTGCPHYS)ADDR_TO_U32(addr);
2032 Log(("Read busmaster FIFO at: %04X\n", ADDR_TO_U32(addr)));
2033 PDMDevHlpPCIPhysWrite(pBusLogic->CTX_SUFF(pDevIns), GCPhysFifoBuf,
2034 &pBusLogic->LocalRam.u8View[64], 64);
2035 break;
2036 }
2037 default:
2038 AssertMsgFailed(("Invalid command %#x\n", pBusLogic->uOperationCode));
2039 case BUSLOGICCOMMAND_EXT_BIOS_INFO:
2040 case BUSLOGICCOMMAND_UNLOCK_MAILBOX:
2041 /* Commands valid for Adaptec 154xC which we don't handle since
2042 * we pretend being 154xB compatible. Just mark the command as invalid.
2043 */
2044 Log(("Command %#x not valid for this adapter\n", pBusLogic->uOperationCode));
2045 pBusLogic->cbReplyParametersLeft = 0;
2046 pBusLogic->regStatus |= BUSLOGIC_REGISTER_STATUS_COMMAND_INVALID;
2047 break;
2048 case BUSLOGICCOMMAND_EXECUTE_MAILBOX_COMMAND: /* Should be handled already. */
2049 AssertMsgFailed(("Invalid mailbox execute state!\n"));
2050 }
2051
2052 Log(("uOperationCode=%#x, cbReplyParametersLeft=%d\n", pBusLogic->uOperationCode, pBusLogic->cbReplyParametersLeft));
2053
2054 /* Set the data in ready bit in the status register in case the command has a reply. */
2055 if (pBusLogic->cbReplyParametersLeft)
2056 pBusLogic->regStatus |= BUSLOGIC_REGISTER_STATUS_DATA_IN_REGISTER_READY;
2057 else if (!pBusLogic->cbCommandParametersLeft)
2058 buslogicCommandComplete(pBusLogic, fSuppressIrq);
2059
2060 return rc;
2061}
2062
2063/**
2064 * Read a register from the BusLogic adapter.
2065 *
2066 * @returns VBox status code.
2067 * @param pBusLogic Pointer to the BusLogic instance data.
2068 * @param iRegister The index of the register to read.
2069 * @param pu32 Where to store the register content.
2070 */
2071static int buslogicRegisterRead(PBUSLOGIC pBusLogic, unsigned iRegister, uint32_t *pu32)
2072{
2073 int rc = VINF_SUCCESS;
2074
2075 switch (iRegister)
2076 {
2077 case BUSLOGIC_REGISTER_STATUS:
2078 {
2079 *pu32 = pBusLogic->regStatus;
2080
2081 /* If the diagnostic active bit is set, we are in a guest-initiated
2082 * hard reset. If the guest reads the status register and waits for
2083 * the host adapter ready bit to be set, we terminate the reset right
2084 * away. However, guests may also expect the reset condition to clear
2085 * automatically after a period of time, in which case we can't show
2086 * the DIAG bit at all.
2087 */
2088 if (pBusLogic->regStatus & BUSLOGIC_REGISTER_STATUS_DIAGNOSTIC_ACTIVE)
2089 {
2090 uint64_t u64AccessTime = PDMDevHlpTMTimeVirtGetNano(pBusLogic->CTX_SUFF(pDevIns));
2091
2092 pBusLogic->regStatus &= ~BUSLOGIC_REGISTER_STATUS_DIAGNOSTIC_ACTIVE;
2093 pBusLogic->regStatus |= BUSLOGIC_REGISTER_STATUS_HOST_ADAPTER_READY;
2094
2095 if (u64AccessTime - pBusLogic->u64ResetTime > BUSLOGIC_RESET_DURATION_NS)
2096 {
2097 /* If reset already expired, let the guest see that right away. */
2098 *pu32 = pBusLogic->regStatus;
2099 pBusLogic->u64ResetTime = 0;
2100 }
2101 }
2102 break;
2103 }
2104 case BUSLOGIC_REGISTER_DATAIN:
2105 {
2106 if (pBusLogic->fUseLocalRam)
2107 *pu32 = pBusLogic->LocalRam.u8View[pBusLogic->iReply];
2108 else
2109 *pu32 = pBusLogic->aReplyBuffer[pBusLogic->iReply];
2110
2111 /* Careful about underflow - guest can read data register even if
2112 * no data is available.
2113 */
2114 if (pBusLogic->cbReplyParametersLeft)
2115 {
2116 pBusLogic->iReply++;
2117 pBusLogic->cbReplyParametersLeft--;
2118 if (!pBusLogic->cbReplyParametersLeft)
2119 {
2120 /*
2121 * Reply finished, set command complete bit, unset data-in ready bit and
2122 * interrupt the guest if enabled.
2123 */
2124 buslogicCommandComplete(pBusLogic, false);
2125 }
2126 }
2127 LogFlowFunc(("data=%02x, iReply=%d, cbReplyParametersLeft=%u\n", *pu32,
2128 pBusLogic->iReply, pBusLogic->cbReplyParametersLeft));
2129 break;
2130 }
2131 case BUSLOGIC_REGISTER_INTERRUPT:
2132 {
2133 *pu32 = pBusLogic->regInterrupt;
2134 break;
2135 }
2136 case BUSLOGIC_REGISTER_GEOMETRY:
2137 {
2138 *pu32 = pBusLogic->regGeometry;
2139 break;
2140 }
2141 default:
2142 *pu32 = UINT32_C(0xffffffff);
2143 }
2144
2145 Log2(("%s: pu32=%p:{%.*Rhxs} iRegister=%d rc=%Rrc\n",
2146 __FUNCTION__, pu32, 1, pu32, iRegister, rc));
2147
2148 return rc;
2149}
2150
2151/**
2152 * Write a value to a register.
2153 *
2154 * @returns VBox status code.
2155 * @param pBusLogic Pointer to the BusLogic instance data.
2156 * @param iRegister The index of the register to read.
2157 * @param uVal The value to write.
2158 */
2159static int buslogicRegisterWrite(PBUSLOGIC pBusLogic, unsigned iRegister, uint8_t uVal)
2160{
2161 int rc = VINF_SUCCESS;
2162
2163 switch (iRegister)
2164 {
2165 case BUSLOGIC_REGISTER_CONTROL:
2166 {
2167 if ((uVal & BUSLOGIC_REGISTER_CONTROL_HARD_RESET) || (uVal & BUSLOGIC_REGISTER_CONTROL_SOFT_RESET))
2168 {
2169#ifdef IN_RING3
2170 bool fHardReset = !!(uVal & BUSLOGIC_REGISTER_CONTROL_HARD_RESET);
2171
2172 LogRel(("BusLogic: %s reset\n", fHardReset ? "hard" : "soft"));
2173 buslogicR3InitiateReset(pBusLogic, fHardReset);
2174#else
2175 rc = VINF_IOM_R3_IOPORT_WRITE;
2176#endif
2177 break;
2178 }
2179
2180 rc = PDMCritSectEnter(&pBusLogic->CritSectIntr, VINF_IOM_R3_IOPORT_WRITE);
2181 if (rc != VINF_SUCCESS)
2182 return rc;
2183
2184#ifdef LOG_ENABLED
2185 uint32_t cMailboxesReady = ASMAtomicXchgU32(&pBusLogic->cInMailboxesReady, 0);
2186 Log(("%u incoming mailboxes were ready when this interrupt was cleared\n", cMailboxesReady));
2187#endif
2188
2189 if (uVal & BUSLOGIC_REGISTER_CONTROL_INTERRUPT_RESET)
2190 buslogicClearInterrupt(pBusLogic);
2191
2192 PDMCritSectLeave(&pBusLogic->CritSectIntr);
2193
2194 break;
2195 }
2196 case BUSLOGIC_REGISTER_COMMAND:
2197 {
2198 /* Fast path for mailbox execution command. */
2199 if ((uVal == BUSLOGICCOMMAND_EXECUTE_MAILBOX_COMMAND) && (pBusLogic->uOperationCode == 0xff))
2200 {
2201 /* If there are no mailboxes configured, don't even try to do anything. */
2202 if (pBusLogic->cMailbox) {
2203 ASMAtomicIncU32(&pBusLogic->cMailboxesReady);
2204 if (!ASMAtomicXchgBool(&pBusLogic->fNotificationSend, true))
2205 {
2206 /* Send new notification to the queue. */
2207 PPDMQUEUEITEMCORE pItem = PDMQueueAlloc(pBusLogic->CTX_SUFF(pNotifierQueue));
2208 AssertMsg(pItem, ("Allocating item for queue failed\n"));
2209 PDMQueueInsert(pBusLogic->CTX_SUFF(pNotifierQueue), (PPDMQUEUEITEMCORE)pItem);
2210 }
2211 }
2212
2213 return rc;
2214 }
2215
2216 /*
2217 * Check if we are already fetch command parameters from the guest.
2218 * If not we initialize executing a new command.
2219 */
2220 if (pBusLogic->uOperationCode == 0xff)
2221 {
2222 pBusLogic->uOperationCode = uVal;
2223 pBusLogic->iParameter = 0;
2224
2225 /* Mark host adapter as busy and clear the invalid status bit. */
2226 pBusLogic->regStatus &= ~(BUSLOGIC_REGISTER_STATUS_HOST_ADAPTER_READY | BUSLOGIC_REGISTER_STATUS_COMMAND_INVALID);
2227
2228 /* Get the number of bytes for parameters from the command code. */
2229 switch (pBusLogic->uOperationCode)
2230 {
2231 case BUSLOGICCOMMAND_TEST_CMDC_INTERRUPT:
2232 case BUSLOGICCOMMAND_INQUIRE_FIRMWARE_VERSION_LETTER:
2233 case BUSLOGICCOMMAND_INQUIRE_BOARD_ID:
2234 case BUSLOGICCOMMAND_INQUIRE_FIRMWARE_VERSION_3RD_LETTER:
2235 case BUSLOGICCOMMAND_INQUIRE_PCI_HOST_ADAPTER_INFORMATION:
2236 case BUSLOGICCOMMAND_INQUIRE_CONFIGURATION:
2237 case BUSLOGICCOMMAND_INQUIRE_INSTALLED_DEVICES_ID_0_TO_7:
2238 case BUSLOGICCOMMAND_INQUIRE_INSTALLED_DEVICES_ID_8_TO_15:
2239 case BUSLOGICCOMMAND_INQUIRE_TARGET_DEVICES:
2240 pBusLogic->cbCommandParametersLeft = 0;
2241 break;
2242 case BUSLOGICCOMMAND_MODIFY_IO_ADDRESS:
2243 case BUSLOGICCOMMAND_INQUIRE_EXTENDED_SETUP_INFORMATION:
2244 case BUSLOGICCOMMAND_INQUIRE_SETUP_INFORMATION:
2245 case BUSLOGICCOMMAND_INQUIRE_HOST_ADAPTER_MODEL_NUMBER:
2246 case BUSLOGICCOMMAND_ENABLE_STRICT_ROUND_ROBIN_MODE:
2247 case BUSLOGICCOMMAND_SET_CCB_FORMAT:
2248 case BUSLOGICCOMMAND_INQUIRE_SYNCHRONOUS_PERIOD:
2249 case BUSLOGICCOMMAND_DISABLE_HOST_ADAPTER_INTERRUPT:
2250 case BUSLOGICCOMMAND_ECHO_COMMAND_DATA:
2251 case BUSLOGICCOMMAND_SET_PREEMPT_TIME_ON_BUS:
2252 case BUSLOGICCOMMAND_SET_TIME_OFF_BUS:
2253 case BUSLOGICCOMMAND_SET_BUS_TRANSFER_RATE:
2254 pBusLogic->cbCommandParametersLeft = 1;
2255 break;
2256 case BUSLOGICCOMMAND_FETCH_HOST_ADAPTER_LOCAL_RAM:
2257 pBusLogic->cbCommandParametersLeft = 2;
2258 break;
2259 case BUSLOGICCOMMAND_READ_BUSMASTER_CHIP_FIFO:
2260 case BUSLOGICCOMMAND_WRITE_BUSMASTER_CHIP_FIFO:
2261 pBusLogic->cbCommandParametersLeft = 3;
2262 break;
2263 case BUSLOGICCOMMAND_INITIALIZE_MAILBOX:
2264 pBusLogic->cbCommandParametersLeft = sizeof(RequestInitMbx);
2265 break;
2266 case BUSLOGICCOMMAND_INITIALIZE_EXTENDED_MAILBOX:
2267 pBusLogic->cbCommandParametersLeft = sizeof(RequestInitializeExtendedMailbox);
2268 break;
2269 case BUSLOGICCOMMAND_SET_ADAPTER_OPTIONS:
2270 /* There must be at least one byte following this command. */
2271 pBusLogic->cbCommandParametersLeft = 1;
2272 break;
2273 case BUSLOGICCOMMAND_EXT_BIOS_INFO:
2274 case BUSLOGICCOMMAND_UNLOCK_MAILBOX:
2275 /* Invalid commands. */
2276 pBusLogic->cbCommandParametersLeft = 0;
2277 break;
2278 case BUSLOGICCOMMAND_EXECUTE_MAILBOX_COMMAND: /* Should not come here anymore. */
2279 default:
2280 AssertMsgFailed(("Invalid operation code %#x\n", uVal));
2281 }
2282 }
2283 else
2284 {
2285#ifndef IN_RING3
2286 /* This command must be executed in R3 as it rehooks the ISA I/O port. */
2287 if (pBusLogic->uOperationCode == BUSLOGICCOMMAND_MODIFY_IO_ADDRESS)
2288 {
2289 rc = VINF_IOM_R3_IOPORT_WRITE;
2290 break;
2291 }
2292#endif
2293 /*
2294 * The real adapter would set the Command register busy bit in the status register.
2295 * The guest has to wait until it is unset.
2296 * We don't need to do it because the guest does not continue execution while we are in this
2297 * function.
2298 */
2299 pBusLogic->aCommandBuffer[pBusLogic->iParameter] = uVal;
2300 pBusLogic->iParameter++;
2301 pBusLogic->cbCommandParametersLeft--;
2302 }
2303
2304 /* Start execution of command if there are no parameters left. */
2305 if (!pBusLogic->cbCommandParametersLeft)
2306 {
2307 rc = buslogicProcessCommand(pBusLogic);
2308 AssertMsgRC(rc, ("Processing command failed rc=%Rrc\n", rc));
2309 }
2310 break;
2311 }
2312
2313 /* On BusLogic adapters, the interrupt and geometry registers are R/W.
2314 * That is different from Adaptec 154x where those are read only.
2315 */
2316 case BUSLOGIC_REGISTER_INTERRUPT:
2317 pBusLogic->regInterrupt = uVal;
2318 break;
2319
2320 case BUSLOGIC_REGISTER_GEOMETRY:
2321 pBusLogic->regGeometry = uVal;
2322 break;
2323
2324 default:
2325 AssertMsgFailed(("Register not available\n"));
2326 rc = VERR_IOM_IOPORT_UNUSED;
2327 }
2328
2329 return rc;
2330}
2331
2332/**
2333 * Memory mapped I/O Handler for read operations.
2334 *
2335 * @returns VBox status code.
2336 *
2337 * @param pDevIns The device instance.
2338 * @param pvUser User argument.
2339 * @param GCPhysAddr Physical address (in GC) where the read starts.
2340 * @param pv Where to store the result.
2341 * @param cb Number of bytes read.
2342 */
2343PDMBOTHCBDECL(int) buslogicMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
2344{
2345 /* the linux driver does not make use of the MMIO area. */
2346 AssertMsgFailed(("MMIO Read\n"));
2347 return VINF_SUCCESS;
2348}
2349
2350/**
2351 * Memory mapped I/O Handler for write operations.
2352 *
2353 * @returns VBox status code.
2354 *
2355 * @param pDevIns The device instance.
2356 * @param pvUser User argument.
2357 * @param GCPhysAddr Physical address (in GC) where the read starts.
2358 * @param pv Where to fetch the result.
2359 * @param cb Number of bytes to write.
2360 */
2361PDMBOTHCBDECL(int) buslogicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void const *pv, unsigned cb)
2362{
2363 /* the linux driver does not make use of the MMIO area. */
2364 AssertMsgFailed(("MMIO Write\n"));
2365 return VINF_SUCCESS;
2366}
2367
2368/**
2369 * Port I/O Handler for IN operations.
2370 *
2371 * @returns VBox status code.
2372 *
2373 * @param pDevIns The device instance.
2374 * @param pvUser User argument.
2375 * @param uPort Port number used for the IN operation.
2376 * @param pu32 Where to store the result.
2377 * @param cb Number of bytes read.
2378 */
2379PDMBOTHCBDECL(int) buslogicIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
2380{
2381 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2382 unsigned iRegister = Port % 4;
2383
2384 Assert(cb == 1);
2385
2386 return buslogicRegisterRead(pBusLogic, iRegister, pu32);
2387}
2388
2389/**
2390 * Port I/O Handler for OUT operations.
2391 *
2392 * @returns VBox status code.
2393 *
2394 * @param pDevIns The device instance.
2395 * @param pvUser User argument.
2396 * @param uPort Port number used for the IN operation.
2397 * @param u32 The value to output.
2398 * @param cb The value size in bytes.
2399 */
2400PDMBOTHCBDECL(int) buslogicIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2401{
2402 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2403 int rc = VINF_SUCCESS;
2404 unsigned iRegister = Port % 4;
2405 uint8_t uVal = (uint8_t)u32;
2406
2407 Assert(cb == 1);
2408
2409 rc = buslogicRegisterWrite(pBusLogic, iRegister, (uint8_t)uVal);
2410
2411 Log2(("#%d %s: pvUser=%#p cb=%d u32=%#x Port=%#x rc=%Rrc\n",
2412 pDevIns->iInstance, __FUNCTION__, pvUser, cb, u32, Port, rc));
2413
2414 return rc;
2415}
2416
2417#ifdef IN_RING3
2418
2419static int buslogicR3PrepareBIOSSCSIRequest(PBUSLOGIC pBusLogic)
2420{
2421 int rc;
2422 PBUSLOGICTASKSTATE pTaskState;
2423 uint32_t uTargetDevice;
2424
2425 rc = RTMemCacheAllocEx(pBusLogic->hTaskCache, (void **)&pTaskState);
2426 AssertMsgRCReturn(rc, ("Getting task from cache failed rc=%Rrc\n", rc), rc);
2427
2428 pTaskState->fBIOS = true;
2429
2430 rc = vboxscsiSetupRequest(&pBusLogic->VBoxSCSI, &pTaskState->PDMScsiRequest, &uTargetDevice);
2431 AssertMsgRCReturn(rc, ("Setting up SCSI request failed rc=%Rrc\n", rc), rc);
2432
2433 pTaskState->PDMScsiRequest.pvUser = pTaskState;
2434
2435 pTaskState->CTX_SUFF(pTargetDevice) = &pBusLogic->aDeviceStates[uTargetDevice];
2436
2437 if (!pTaskState->CTX_SUFF(pTargetDevice)->fPresent)
2438 {
2439 /* Device is not present. */
2440 AssertMsg(pTaskState->PDMScsiRequest.pbCDB[0] == SCSI_INQUIRY,
2441 ("Device is not present but command is not inquiry\n"));
2442
2443 SCSIINQUIRYDATA ScsiInquiryData;
2444
2445 memset(&ScsiInquiryData, 0, sizeof(SCSIINQUIRYDATA));
2446 ScsiInquiryData.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_UNKNOWN;
2447 ScsiInquiryData.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_NOT_CONNECTED_NOT_SUPPORTED;
2448
2449 memcpy(pBusLogic->VBoxSCSI.pbBuf, &ScsiInquiryData, 5);
2450
2451 rc = vboxscsiRequestFinished(&pBusLogic->VBoxSCSI, &pTaskState->PDMScsiRequest, SCSI_STATUS_OK);
2452 AssertMsgRCReturn(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc), rc);
2453
2454 RTMemCacheFree(pBusLogic->hTaskCache, pTaskState);
2455 }
2456 else
2457 {
2458 LogFlowFunc(("before increment %u\n", pTaskState->CTX_SUFF(pTargetDevice)->cOutstandingRequests));
2459 ASMAtomicIncU32(&pTaskState->CTX_SUFF(pTargetDevice)->cOutstandingRequests);
2460 LogFlowFunc(("after increment %u\n", pTaskState->CTX_SUFF(pTargetDevice)->cOutstandingRequests));
2461
2462 rc = pTaskState->CTX_SUFF(pTargetDevice)->pDrvSCSIConnector->pfnSCSIRequestSend(pTaskState->CTX_SUFF(pTargetDevice)->pDrvSCSIConnector,
2463 &pTaskState->PDMScsiRequest);
2464 AssertMsgRC(rc, ("Sending request to SCSI layer failed rc=%Rrc\n", rc));
2465 }
2466
2467 return rc;
2468}
2469
2470
2471/**
2472 * Port I/O Handler for IN operations - BIOS port.
2473 *
2474 * @returns VBox status code.
2475 *
2476 * @param pDevIns The device instance.
2477 * @param pvUser User argument.
2478 * @param uPort Port number used for the IN operation.
2479 * @param pu32 Where to store the result.
2480 * @param cb Number of bytes read.
2481 */
2482static DECLCALLBACK(int) buslogicR3BiosIoPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
2483{
2484 int rc;
2485 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2486
2487 Assert(cb == 1);
2488
2489 rc = vboxscsiReadRegister(&pBusLogic->VBoxSCSI, (Port - BUSLOGIC_BIOS_IO_PORT), pu32);
2490
2491 //Log2(("%s: pu32=%p:{%.*Rhxs} iRegister=%d rc=%Rrc\n",
2492 // __FUNCTION__, pu32, 1, pu32, (Port - BUSLOGIC_BIOS_IO_PORT), rc));
2493
2494 return rc;
2495}
2496
2497/**
2498 * Port I/O Handler for OUT operations - BIOS port.
2499 *
2500 * @returns VBox status code.
2501 *
2502 * @param pDevIns The device instance.
2503 * @param pvUser User argument.
2504 * @param uPort Port number used for the IN operation.
2505 * @param u32 The value to output.
2506 * @param cb The value size in bytes.
2507 */
2508static DECLCALLBACK(int) buslogicR3BiosIoPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2509{
2510 int rc;
2511 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2512
2513 Log2(("#%d %s: pvUser=%#p cb=%d u32=%#x Port=%#x\n",
2514 pDevIns->iInstance, __FUNCTION__, pvUser, cb, u32, Port));
2515
2516 Assert(cb == 1);
2517
2518 rc = vboxscsiWriteRegister(&pBusLogic->VBoxSCSI, (Port - BUSLOGIC_BIOS_IO_PORT), (uint8_t)u32);
2519 if (rc == VERR_MORE_DATA)
2520 {
2521 rc = buslogicR3PrepareBIOSSCSIRequest(pBusLogic);
2522 AssertRC(rc);
2523 }
2524 else if (RT_FAILURE(rc))
2525 AssertMsgFailed(("Writing BIOS register failed %Rrc\n", rc));
2526
2527 return VINF_SUCCESS;
2528}
2529
2530/**
2531 * Port I/O Handler for primary port range OUT string operations.
2532 * @see FNIOMIOPORTOUTSTRING for details.
2533 */
2534static DECLCALLBACK(int) buslogicR3BiosIoPortWriteStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, RTGCPTR *pGCPtrSrc,
2535 PRTGCUINTREG pcTransfer, unsigned cb)
2536{
2537 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2538 int rc;
2539
2540 Log2(("#%d %s: pvUser=%#p cb=%d Port=%#x\n",
2541 pDevIns->iInstance, __FUNCTION__, pvUser, cb, Port));
2542
2543 rc = vboxscsiWriteString(pDevIns, &pBusLogic->VBoxSCSI, (Port - BUSLOGIC_BIOS_IO_PORT),
2544 pGCPtrSrc, pcTransfer, cb);
2545 if (rc == VERR_MORE_DATA)
2546 {
2547 rc = buslogicR3PrepareBIOSSCSIRequest(pBusLogic);
2548 AssertRC(rc);
2549 }
2550 else if (RT_FAILURE(rc))
2551 AssertMsgFailed(("Writing BIOS register failed %Rrc\n", rc));
2552
2553 return rc;
2554}
2555
2556/**
2557 * Port I/O Handler for primary port range IN string operations.
2558 * @see FNIOMIOPORTINSTRING for details.
2559 */
2560static DECLCALLBACK(int) buslogicR3BiosIoPortReadStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, RTGCPTR *pGCPtrDst,
2561 PRTGCUINTREG pcTransfer, unsigned cb)
2562{
2563 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2564
2565 LogFlowFunc(("#%d %s: pvUser=%#p cb=%d Port=%#x\n",
2566 pDevIns->iInstance, __FUNCTION__, pvUser, cb, Port));
2567
2568 return vboxscsiReadString(pDevIns, &pBusLogic->VBoxSCSI, (Port - BUSLOGIC_BIOS_IO_PORT),
2569 pGCPtrDst, pcTransfer, cb);
2570}
2571
2572/**
2573 * Update the ISA I/O range.
2574 *
2575 * @returns nothing.
2576 * @param pBusLogic Pointer to the BusLogic device instance.
2577 * @param uBaseCode Encoded ISA I/O base; only low 3 bits are used.
2578 */
2579static int buslogicR3RegisterISARange(PBUSLOGIC pBusLogic, uint8_t uBaseCode)
2580{
2581 uint8_t uCode = uBaseCode & MAX_ISA_BASE;
2582 uint16_t uNewBase = g_aISABases[uCode];
2583 int rc = VINF_SUCCESS;
2584
2585 LogFlowFunc(("ISA I/O code %02X, new base %X\n", uBaseCode, uNewBase));
2586
2587 /* Check if the same port range is already registered. */
2588 if (uNewBase != pBusLogic->IOISABase)
2589 {
2590 /* Unregister the old range, if any. */
2591 if (pBusLogic->IOISABase)
2592 rc = PDMDevHlpIOPortDeregister(pBusLogic->CTX_SUFF(pDevIns), pBusLogic->IOISABase, 4);
2593
2594 if (RT_SUCCESS(rc))
2595 {
2596 pBusLogic->IOISABase = 0; /* First mark as unregistered. */
2597 pBusLogic->uISABaseCode = ISA_BASE_DISABLED;
2598
2599 if (uNewBase)
2600 {
2601 /* Register the new range if requested. */
2602 rc = PDMDevHlpIOPortRegister(pBusLogic->CTX_SUFF(pDevIns), uNewBase, 4, NULL,
2603 buslogicIOPortWrite, buslogicIOPortRead,
2604 NULL, NULL,
2605 "BusLogic ISA");
2606 if (RT_SUCCESS(rc))
2607 {
2608 pBusLogic->IOISABase = uNewBase;
2609 pBusLogic->uISABaseCode = uCode;
2610 }
2611 }
2612 }
2613 if (RT_SUCCESS(rc))
2614 {
2615 if (uNewBase)
2616 {
2617 Log(("ISA I/O base: %x\n", uNewBase));
2618 LogRel(("BusLogic: ISA I/O base: %x\n", uNewBase));
2619 }
2620 else
2621 {
2622 Log(("Disabling ISA I/O ports.\n"));
2623 LogRel(("BusLogic: ISA I/O disabled\n"));
2624 }
2625 }
2626
2627 }
2628 return rc;
2629}
2630
2631static void buslogicR3WarningDiskFull(PPDMDEVINS pDevIns)
2632{
2633 int rc;
2634 LogRel(("BusLogic#%d: Host disk full\n", pDevIns->iInstance));
2635 rc = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevBusLogic_DISKFULL",
2636 N_("Host system reported disk full. VM execution is suspended. You can resume after freeing some space"));
2637 AssertRC(rc);
2638}
2639
2640static void buslogicR3WarningFileTooBig(PPDMDEVINS pDevIns)
2641{
2642 int rc;
2643 LogRel(("BusLogic#%d: File too big\n", pDevIns->iInstance));
2644 rc = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevBusLogic_FILETOOBIG",
2645 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"));
2646 AssertRC(rc);
2647}
2648
2649static void buslogicR3WarningISCSI(PPDMDEVINS pDevIns)
2650{
2651 int rc;
2652 LogRel(("BusLogic#%d: iSCSI target unavailable\n", pDevIns->iInstance));
2653 rc = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevBusLogic_ISCSIDOWN",
2654 N_("The iSCSI target has stopped responding. VM execution is suspended. You can resume when it is available again"));
2655 AssertRC(rc);
2656}
2657
2658static void buslogicR3WarningUnknown(PPDMDEVINS pDevIns, int rc)
2659{
2660 int rc2;
2661 LogRel(("BusLogic#%d: Unknown but recoverable error has occurred (rc=%Rrc)\n", pDevIns->iInstance, rc));
2662 rc2 = PDMDevHlpVMSetRuntimeError(pDevIns, VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_NO_WAIT, "DevBusLogic_UNKNOWN",
2663 N_("An unknown but recoverable I/O error has occurred (rc=%Rrc). VM execution is suspended. You can resume when the error is fixed"), rc);
2664 AssertRC(rc2);
2665}
2666
2667static void buslogicR3RedoSetWarning(PBUSLOGIC pThis, int rc)
2668{
2669 if (rc == VERR_DISK_FULL)
2670 buslogicR3WarningDiskFull(pThis->CTX_SUFF(pDevIns));
2671 else if (rc == VERR_FILE_TOO_BIG)
2672 buslogicR3WarningFileTooBig(pThis->CTX_SUFF(pDevIns));
2673 else if (rc == VERR_BROKEN_PIPE || rc == VERR_NET_CONNECTION_REFUSED)
2674 {
2675 /* iSCSI connection abort (first error) or failure to reestablish
2676 * connection (second error). Pause VM. On resume we'll retry. */
2677 buslogicR3WarningISCSI(pThis->CTX_SUFF(pDevIns));
2678 }
2679 else
2680 buslogicR3WarningUnknown(pThis->CTX_SUFF(pDevIns), rc);
2681}
2682
2683
2684static DECLCALLBACK(int) buslogicR3MmioMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
2685 RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
2686{
2687 PPDMDEVINS pDevIns = pPciDev->pDevIns;
2688 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
2689 int rc = VINF_SUCCESS;
2690
2691 Log2(("%s: registering MMIO area at GCPhysAddr=%RGp cb=%u\n", __FUNCTION__, GCPhysAddress, cb));
2692
2693 Assert(cb >= 32);
2694
2695 if (enmType == PCI_ADDRESS_SPACE_MEM)
2696 {
2697 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
2698 rc = PDMDevHlpMMIORegister(pDevIns, GCPhysAddress, cb, NULL /*pvUser*/,
2699 IOMMMIO_FLAGS_READ_PASSTHRU | IOMMMIO_FLAGS_WRITE_PASSTHRU,
2700 buslogicMMIOWrite, buslogicMMIORead, "BusLogic MMIO");
2701 if (RT_FAILURE(rc))
2702 return rc;
2703
2704 if (pThis->fR0Enabled)
2705 {
2706 rc = PDMDevHlpMMIORegisterR0(pDevIns, GCPhysAddress, cb, NIL_RTR0PTR /*pvUser*/,
2707 "buslogicMMIOWrite", "buslogicMMIORead");
2708 if (RT_FAILURE(rc))
2709 return rc;
2710 }
2711
2712 if (pThis->fGCEnabled)
2713 {
2714 rc = PDMDevHlpMMIORegisterRC(pDevIns, GCPhysAddress, cb, NIL_RTRCPTR /*pvUser*/,
2715 "buslogicMMIOWrite", "buslogicMMIORead");
2716 if (RT_FAILURE(rc))
2717 return rc;
2718 }
2719
2720 pThis->MMIOBase = GCPhysAddress;
2721 }
2722 else if (enmType == PCI_ADDRESS_SPACE_IO)
2723 {
2724 rc = PDMDevHlpIOPortRegister(pDevIns, (RTIOPORT)GCPhysAddress, 32,
2725 NULL, buslogicIOPortWrite, buslogicIOPortRead, NULL, NULL, "BusLogic PCI");
2726 if (RT_FAILURE(rc))
2727 return rc;
2728
2729 if (pThis->fR0Enabled)
2730 {
2731 rc = PDMDevHlpIOPortRegisterR0(pDevIns, (RTIOPORT)GCPhysAddress, 32,
2732 0, "buslogicIOPortWrite", "buslogicIOPortRead", NULL, NULL, "BusLogic PCI");
2733 if (RT_FAILURE(rc))
2734 return rc;
2735 }
2736
2737 if (pThis->fGCEnabled)
2738 {
2739 rc = PDMDevHlpIOPortRegisterRC(pDevIns, (RTIOPORT)GCPhysAddress, 32,
2740 0, "buslogicIOPortWrite", "buslogicIOPortRead", NULL, NULL, "BusLogic PCI");
2741 if (RT_FAILURE(rc))
2742 return rc;
2743 }
2744
2745 pThis->IOPortBase = (RTIOPORT)GCPhysAddress;
2746 }
2747 else
2748 AssertMsgFailed(("Invalid enmType=%d\n", enmType));
2749
2750 return rc;
2751}
2752
2753static DECLCALLBACK(int) buslogicR3DeviceSCSIRequestCompleted(PPDMISCSIPORT pInterface, PPDMSCSIREQUEST pSCSIRequest,
2754 int rcCompletion, bool fRedo, int rcReq)
2755{
2756 int rc;
2757 PBUSLOGICTASKSTATE pTaskState = (PBUSLOGICTASKSTATE)pSCSIRequest->pvUser;
2758 PBUSLOGICDEVICE pBusLogicDevice = pTaskState->CTX_SUFF(pTargetDevice);
2759 PBUSLOGIC pBusLogic = pBusLogicDevice->CTX_SUFF(pBusLogic);
2760
2761 LogFlowFunc(("before decrement %u\n", pBusLogicDevice->cOutstandingRequests));
2762 ASMAtomicDecU32(&pBusLogicDevice->cOutstandingRequests);
2763 LogFlowFunc(("after decrement %u\n", pBusLogicDevice->cOutstandingRequests));
2764
2765 if (fRedo)
2766 {
2767 if (!pTaskState->fBIOS)
2768 {
2769 buslogicR3DataBufferFree(pTaskState);
2770
2771 if (pTaskState->pbSenseBuffer)
2772 buslogicR3SenseBufferFree(pTaskState, false /* fCopy */);
2773 }
2774
2775 /* Add to the list. */
2776 do
2777 {
2778 pTaskState->pRedoNext = ASMAtomicReadPtrT(&pBusLogic->pTasksRedoHead, PBUSLOGICTASKSTATE);
2779 } while (!ASMAtomicCmpXchgPtr(&pBusLogic->pTasksRedoHead, pTaskState, pTaskState->pRedoNext));
2780
2781 /* Suspend the VM if not done already. */
2782 if (!ASMAtomicXchgBool(&pBusLogic->fRedo, true))
2783 buslogicR3RedoSetWarning(pBusLogic, rcReq);
2784 }
2785 else
2786 {
2787 if (pTaskState->fBIOS)
2788 {
2789 rc = vboxscsiRequestFinished(&pBusLogic->VBoxSCSI, pSCSIRequest, rcCompletion);
2790 AssertMsgRC(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc));
2791 }
2792 else
2793 {
2794 buslogicR3DataBufferFree(pTaskState);
2795
2796 if (pTaskState->pbSenseBuffer)
2797 buslogicR3SenseBufferFree(pTaskState, (rcCompletion != SCSI_STATUS_OK));
2798
2799 if (rcCompletion == SCSI_STATUS_OK)
2800 buslogicR3SendIncomingMailbox(pBusLogic, pTaskState,
2801 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_CMD_COMPLETED,
2802 BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS_OPERATION_GOOD,
2803 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_WITHOUT_ERROR);
2804 else if (rcCompletion == SCSI_STATUS_CHECK_CONDITION)
2805 buslogicR3SendIncomingMailbox(pBusLogic, pTaskState,
2806 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_CMD_COMPLETED,
2807 BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS_CHECK_CONDITION,
2808 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_WITH_ERROR);
2809 else
2810 AssertMsgFailed(("invalid completion status %d\n", rcCompletion));
2811 }
2812#ifdef LOG_ENABLED
2813 buslogicR3DumpCCBInfo(&pTaskState->CommandControlBlockGuest, pTaskState->fIs24Bit);
2814#endif
2815
2816 /* Remove task from the cache. */
2817 RTMemCacheFree(pBusLogic->hTaskCache, pTaskState);
2818 }
2819
2820 if (pBusLogicDevice->cOutstandingRequests == 0 && pBusLogic->fSignalIdle)
2821 PDMDevHlpAsyncNotificationCompleted(pBusLogic->pDevInsR3);
2822
2823 return VINF_SUCCESS;
2824}
2825
2826static DECLCALLBACK(int) buslogicR3QueryDeviceLocation(PPDMISCSIPORT pInterface, const char **ppcszController,
2827 uint32_t *piInstance, uint32_t *piLUN)
2828{
2829 PBUSLOGICDEVICE pBusLogicDevice = PDMISCSIPORT_2_PBUSLOGICDEVICE(pInterface);
2830 PPDMDEVINS pDevIns = pBusLogicDevice->CTX_SUFF(pBusLogic)->CTX_SUFF(pDevIns);
2831
2832 AssertPtrReturn(ppcszController, VERR_INVALID_POINTER);
2833 AssertPtrReturn(piInstance, VERR_INVALID_POINTER);
2834 AssertPtrReturn(piLUN, VERR_INVALID_POINTER);
2835
2836 *ppcszController = pDevIns->pReg->szName;
2837 *piInstance = pDevIns->iInstance;
2838 *piLUN = pBusLogicDevice->iLUN;
2839
2840 return VINF_SUCCESS;
2841}
2842
2843static int buslogicR3DeviceSCSIRequestSetup(PBUSLOGIC pBusLogic, PBUSLOGICTASKSTATE pTaskState)
2844{
2845 int rc = VINF_SUCCESS;
2846 uint8_t uTargetIdCCB;
2847 PBUSLOGICDEVICE pTargetDevice;
2848
2849 /* Fetch the CCB from guest memory. */
2850 /** @todo How much do we really have to read? */
2851 RTGCPHYS GCPhysAddrCCB = (RTGCPHYS)pTaskState->MailboxGuest.u32PhysAddrCCB;
2852 PDMDevHlpPhysRead(pBusLogic->CTX_SUFF(pDevIns), GCPhysAddrCCB,
2853 &pTaskState->CommandControlBlockGuest, sizeof(CCB32));
2854
2855 uTargetIdCCB = pTaskState->fIs24Bit ? pTaskState->CommandControlBlockGuest.o.uTargetId : pTaskState->CommandControlBlockGuest.n.uTargetId;
2856 pTargetDevice = &pBusLogic->aDeviceStates[uTargetIdCCB];
2857 pTaskState->CTX_SUFF(pTargetDevice) = pTargetDevice;
2858
2859#ifdef LOG_ENABLED
2860 buslogicR3DumpCCBInfo(&pTaskState->CommandControlBlockGuest, pTaskState->fIs24Bit);
2861#endif
2862
2863 /* Alloc required buffers. */
2864 rc = buslogicR3DataBufferAlloc(pTaskState);
2865 AssertMsgRC(rc, ("Alloc failed rc=%Rrc\n", rc));
2866
2867 rc = buslogicR3SenseBufferAlloc(pTaskState);
2868 AssertMsgRC(rc, ("Mapping sense buffer failed rc=%Rrc\n", rc));
2869
2870 /* Check if device is present on bus. If not return error immediately and don't process this further. */
2871 if (!pBusLogic->aDeviceStates[uTargetIdCCB].fPresent)
2872 {
2873 buslogicR3DataBufferFree(pTaskState);
2874
2875 if (pTaskState->pbSenseBuffer)
2876 buslogicR3SenseBufferFree(pTaskState, true);
2877
2878 buslogicR3SendIncomingMailbox(pBusLogic, pTaskState,
2879 BUSLOGIC_MAILBOX_INCOMING_ADAPTER_STATUS_SCSI_SELECTION_TIMEOUT,
2880 BUSLOGIC_MAILBOX_INCOMING_DEVICE_STATUS_OPERATION_GOOD,
2881 BUSLOGIC_MAILBOX_INCOMING_COMPLETION_WITH_ERROR);
2882
2883 RTMemCacheFree(pBusLogic->hTaskCache, pTaskState);
2884 }
2885 else
2886 {
2887 /* Setup SCSI request. */
2888 pTaskState->PDMScsiRequest.uLogicalUnit = pTaskState->fIs24Bit ? pTaskState->CommandControlBlockGuest.o.uLogicalUnit
2889 : pTaskState->CommandControlBlockGuest.n.uLogicalUnit;
2890
2891 if (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_UNKNOWN)
2892 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_UNKNOWN;
2893 else if (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_IN)
2894 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_FROM_DEVICE;
2895 else if (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_OUT)
2896 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_TO_DEVICE;
2897 else if (pTaskState->CommandControlBlockGuest.c.uDataDirection == BUSLOGIC_CCB_DIRECTION_NO_DATA)
2898 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_NONE;
2899 else
2900 AssertMsgFailed(("Invalid data direction type %d\n", pTaskState->CommandControlBlockGuest.c.uDataDirection));
2901
2902 pTaskState->PDMScsiRequest.cbCDB = pTaskState->CommandControlBlockGuest.c.cbCDB;
2903 pTaskState->PDMScsiRequest.pbCDB = pTaskState->CommandControlBlockGuest.c.abCDB;
2904 if (pTaskState->DataSeg.cbSeg)
2905 {
2906 pTaskState->PDMScsiRequest.cbScatterGather = pTaskState->DataSeg.cbSeg;
2907 pTaskState->PDMScsiRequest.cScatterGatherEntries = 1;
2908 pTaskState->PDMScsiRequest.paScatterGatherHead = &pTaskState->DataSeg;
2909 }
2910 else
2911 {
2912 pTaskState->PDMScsiRequest.cbScatterGather = 0;
2913 pTaskState->PDMScsiRequest.cScatterGatherEntries = 0;
2914 pTaskState->PDMScsiRequest.paScatterGatherHead = NULL;
2915 }
2916 pTaskState->PDMScsiRequest.cbSenseBuffer = buslogicR3ConvertSenseBufferLength(pTaskState->CommandControlBlockGuest.c.cbSenseData);
2917 pTaskState->PDMScsiRequest.pbSenseBuffer = pTaskState->pbSenseBuffer;
2918 pTaskState->PDMScsiRequest.pvUser = pTaskState;
2919
2920 ASMAtomicIncU32(&pTargetDevice->cOutstandingRequests);
2921 rc = pTargetDevice->pDrvSCSIConnector->pfnSCSIRequestSend(pTargetDevice->pDrvSCSIConnector, &pTaskState->PDMScsiRequest);
2922 AssertMsgRC(rc, ("Sending request to SCSI layer failed rc=%Rrc\n", rc));
2923 }
2924
2925 return rc;
2926}
2927
2928/**
2929 * Read a mailbox from guest memory. Convert 24-bit mailboxes to
2930 * 32-bit format.
2931 *
2932 * @returns Mailbox guest physical address.
2933 * @param pBusLogic Pointer to the BusLogic instance data.
2934 * @param pTaskStat Pointer to the task state being set up.
2935 */
2936static RTGCPHYS buslogicR3ReadOutgoingMailbox(PBUSLOGIC pBusLogic, PBUSLOGICTASKSTATE pTaskState)
2937{
2938 RTGCPHYS GCMailbox;
2939
2940 if (pBusLogic->fMbxIs24Bit)
2941 {
2942 Mailbox24 Mbx24;
2943
2944 GCMailbox = pBusLogic->GCPhysAddrMailboxOutgoingBase + (pBusLogic->uMailboxOutgoingPositionCurrent * sizeof(Mailbox24));
2945 PDMDevHlpPhysRead(pBusLogic->CTX_SUFF(pDevIns), GCMailbox, &Mbx24, sizeof(Mailbox24));
2946 pTaskState->MailboxGuest.u32PhysAddrCCB = ADDR_TO_U32(Mbx24.aPhysAddrCCB);
2947 pTaskState->MailboxGuest.u.out.uActionCode = Mbx24.uCmdState;
2948 }
2949 else
2950 {
2951 GCMailbox = pBusLogic->GCPhysAddrMailboxOutgoingBase + (pBusLogic->uMailboxOutgoingPositionCurrent * sizeof(Mailbox32));
2952 PDMDevHlpPhysRead(pBusLogic->CTX_SUFF(pDevIns), GCMailbox, &pTaskState->MailboxGuest, sizeof(Mailbox32));
2953 }
2954
2955 return GCMailbox;
2956}
2957
2958/**
2959 * Read mailbox from the guest and execute command.
2960 *
2961 * @returns VBox status code.
2962 * @param pBusLogic Pointer to the BusLogic instance data.
2963 */
2964static int buslogicR3ProcessMailboxNext(PBUSLOGIC pBusLogic)
2965{
2966 PBUSLOGICTASKSTATE pTaskState = NULL;
2967 RTGCPHYS GCPhysAddrMailboxCurrent;
2968 int rc;
2969
2970 rc = RTMemCacheAllocEx(pBusLogic->hTaskCache, (void **)&pTaskState);
2971 AssertMsgReturn(RT_SUCCESS(rc) && (pTaskState != NULL), ("Failed to get task state from cache\n"), rc);
2972
2973 pTaskState->fBIOS = false;
2974 pTaskState->fIs24Bit = pBusLogic->fMbxIs24Bit;
2975 pTaskState->cbSGEntry = pBusLogic->fMbxIs24Bit ? sizeof(SGE24) : sizeof(SGE32);
2976
2977 if (!pBusLogic->fStrictRoundRobinMode)
2978 {
2979 /* Search for a filled mailbox - stop if we have scanned all mailboxes. */
2980 uint8_t uMailboxPosCur = pBusLogic->uMailboxOutgoingPositionCurrent;
2981
2982 do
2983 {
2984 /* Fetch mailbox from guest memory. */
2985 GCPhysAddrMailboxCurrent = buslogicR3ReadOutgoingMailbox(pBusLogic,pTaskState);
2986
2987 /* Check the next mailbox. */
2988 buslogicR3OutgoingMailboxAdvance(pBusLogic);
2989 } while ( pTaskState->MailboxGuest.u.out.uActionCode == BUSLOGIC_MAILBOX_OUTGOING_ACTION_FREE
2990 && uMailboxPosCur != pBusLogic->uMailboxOutgoingPositionCurrent);
2991 }
2992 else
2993 {
2994 /* Fetch mailbox from guest memory. */
2995 GCPhysAddrMailboxCurrent = buslogicR3ReadOutgoingMailbox(pBusLogic,pTaskState);
2996 }
2997
2998 /*
2999 * Check if the mailbox is actually loaded.
3000 * It might be possible that the guest notified us without
3001 * a loaded mailbox. Do nothing in that case but leave a
3002 * log entry.
3003 */
3004 if (pTaskState->MailboxGuest.u.out.uActionCode == BUSLOGIC_MAILBOX_OUTGOING_ACTION_FREE)
3005 {
3006 Log(("No loaded mailbox left\n"));
3007 RTMemCacheFree(pBusLogic->hTaskCache, pTaskState);
3008 return VERR_NO_DATA;
3009 }
3010
3011 LogFlow(("Got loaded mailbox at slot %u, CCB phys %RGp\n", pBusLogic->uMailboxOutgoingPositionCurrent, (RTGCPHYS)pTaskState->MailboxGuest.u32PhysAddrCCB));
3012#ifdef LOG_ENABLED
3013 buslogicR3DumpMailboxInfo(&pTaskState->MailboxGuest, true);
3014#endif
3015
3016 /* We got the mailbox, mark it as free in the guest. */
3017 uint8_t uActionCode = BUSLOGIC_MAILBOX_OUTGOING_ACTION_FREE;
3018 unsigned uCodeOffs = pTaskState->fIs24Bit ? RT_OFFSETOF(Mailbox24, uCmdState) : RT_OFFSETOF(Mailbox32, u.out.uActionCode);
3019 PDMDevHlpPCIPhysWrite(pBusLogic->CTX_SUFF(pDevIns), GCPhysAddrMailboxCurrent + uCodeOffs, &uActionCode, sizeof(uActionCode));
3020
3021 if (pTaskState->MailboxGuest.u.out.uActionCode == BUSLOGIC_MAILBOX_OUTGOING_ACTION_START_COMMAND)
3022 rc = buslogicR3DeviceSCSIRequestSetup(pBusLogic, pTaskState);
3023 else if (pTaskState->MailboxGuest.u.out.uActionCode == BUSLOGIC_MAILBOX_OUTGOING_ACTION_ABORT_COMMAND)
3024 {
3025 AssertMsgFailed(("Not implemented yet\n"));
3026 }
3027 else
3028 AssertMsgFailed(("Invalid outgoing mailbox action code %u\n", pTaskState->MailboxGuest.u.out.uActionCode));
3029
3030 AssertRC(rc);
3031
3032 /* Advance to the next mailbox. */
3033 if (pBusLogic->fStrictRoundRobinMode)
3034 buslogicR3OutgoingMailboxAdvance(pBusLogic);
3035
3036 return rc;
3037}
3038
3039/**
3040 * Transmit queue consumer
3041 * Queue a new async task.
3042 *
3043 * @returns Success indicator.
3044 * If false the item will not be removed and the flushing will stop.
3045 * @param pDevIns The device instance.
3046 * @param pItem The item to consume. Upon return this item will be freed.
3047 */
3048static DECLCALLBACK(bool) buslogicR3NotifyQueueConsumer(PPDMDEVINS pDevIns, PPDMQUEUEITEMCORE pItem)
3049{
3050 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3051
3052 /* Reset notification send flag now. */
3053 Assert(pBusLogic->fNotificationSend);
3054 ASMAtomicXchgBool(&pBusLogic->fNotificationSend, false);
3055 ASMAtomicXchgU32(&pBusLogic->cMailboxesReady, 0); /** @todo Actually not required anymore but to stay compatible with older saved states. */
3056
3057 /* Process mailboxes. */
3058 int rc;
3059 do
3060 {
3061 rc = buslogicR3ProcessMailboxNext(pBusLogic);
3062 AssertMsg(RT_SUCCESS(rc) || rc == VERR_NO_DATA, ("Processing mailbox failed rc=%Rrc\n", rc));
3063 } while (RT_SUCCESS(rc));
3064
3065 return true;
3066}
3067
3068/**
3069 * Kicks the controller to process pending tasks after the VM was resumed
3070 * or loaded from a saved state.
3071 *
3072 * @returns nothing.
3073 * @param pThis The BusLogic device instance.
3074 */
3075static void buslogicR3Kick(PBUSLOGIC pThis)
3076{
3077 if (pThis->fRedo)
3078 {
3079 pThis->fRedo = false;
3080 if (pThis->VBoxSCSI.fBusy)
3081 {
3082
3083 /* The BIOS had a request active when we got suspended. Resume it. */
3084 int rc = buslogicR3PrepareBIOSSCSIRequest(pThis);
3085 AssertRC(rc);
3086 }
3087 else
3088 {
3089 /* Queue all pending tasks again. */
3090 PBUSLOGICTASKSTATE pTaskState = pThis->pTasksRedoHead;
3091
3092 pThis->pTasksRedoHead = NULL;
3093
3094 while (pTaskState)
3095 {
3096 PBUSLOGICTASKSTATE pCur = pTaskState;
3097
3098 int rc = buslogicR3DeviceSCSIRequestSetup(pThis, pCur);
3099 AssertRC(rc);
3100
3101 pTaskState = pTaskState->pRedoNext;
3102 }
3103 }
3104 }
3105}
3106
3107/** @callback_method_impl{FNSSMDEVLIVEEXEC} */
3108static DECLCALLBACK(int) buslogicR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
3109{
3110 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3111
3112 /* Save the device config. */
3113 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aDeviceStates); i++)
3114 SSMR3PutBool(pSSM, pThis->aDeviceStates[i].fPresent);
3115
3116 return VINF_SSM_DONT_CALL_AGAIN;
3117}
3118
3119/** @callback_method_impl{FNSSMDEVSAVEEXEC} */
3120static DECLCALLBACK(int) buslogicR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3121{
3122 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3123
3124 /* Every device first. */
3125 for (unsigned i = 0; i < RT_ELEMENTS(pBusLogic->aDeviceStates); i++)
3126 {
3127 PBUSLOGICDEVICE pDevice = &pBusLogic->aDeviceStates[i];
3128
3129 AssertMsg(!pDevice->cOutstandingRequests,
3130 ("There are still outstanding requests on this device\n"));
3131 SSMR3PutBool(pSSM, pDevice->fPresent);
3132 SSMR3PutU32(pSSM, pDevice->cOutstandingRequests);
3133 }
3134 /* Now the main device state. */
3135 SSMR3PutU8 (pSSM, pBusLogic->regStatus);
3136 SSMR3PutU8 (pSSM, pBusLogic->regInterrupt);
3137 SSMR3PutU8 (pSSM, pBusLogic->regGeometry);
3138 SSMR3PutMem (pSSM, &pBusLogic->LocalRam, sizeof(pBusLogic->LocalRam));
3139 SSMR3PutU8 (pSSM, pBusLogic->uOperationCode);
3140 SSMR3PutMem (pSSM, &pBusLogic->aCommandBuffer, sizeof(pBusLogic->aCommandBuffer));
3141 SSMR3PutU8 (pSSM, pBusLogic->iParameter);
3142 SSMR3PutU8 (pSSM, pBusLogic->cbCommandParametersLeft);
3143 SSMR3PutBool (pSSM, pBusLogic->fUseLocalRam);
3144 SSMR3PutMem (pSSM, pBusLogic->aReplyBuffer, sizeof(pBusLogic->aReplyBuffer));
3145 SSMR3PutU8 (pSSM, pBusLogic->iReply);
3146 SSMR3PutU8 (pSSM, pBusLogic->cbReplyParametersLeft);
3147 SSMR3PutBool (pSSM, pBusLogic->fIRQEnabled);
3148 SSMR3PutU8 (pSSM, pBusLogic->uISABaseCode);
3149 SSMR3PutU32 (pSSM, pBusLogic->cMailbox);
3150 SSMR3PutBool (pSSM, pBusLogic->fMbxIs24Bit);
3151 SSMR3PutGCPhys(pSSM, pBusLogic->GCPhysAddrMailboxOutgoingBase);
3152 SSMR3PutU32 (pSSM, pBusLogic->uMailboxOutgoingPositionCurrent);
3153 SSMR3PutU32 (pSSM, pBusLogic->cMailboxesReady);
3154 SSMR3PutBool (pSSM, pBusLogic->fNotificationSend);
3155 SSMR3PutGCPhys(pSSM, pBusLogic->GCPhysAddrMailboxIncomingBase);
3156 SSMR3PutU32 (pSSM, pBusLogic->uMailboxIncomingPositionCurrent);
3157 SSMR3PutBool (pSSM, pBusLogic->fStrictRoundRobinMode);
3158 SSMR3PutBool (pSSM, pBusLogic->fExtendedLunCCBFormat);
3159 /* Now the data for the BIOS interface. */
3160 SSMR3PutU8 (pSSM, pBusLogic->VBoxSCSI.regIdentify);
3161 SSMR3PutU8 (pSSM, pBusLogic->VBoxSCSI.uTargetDevice);
3162 SSMR3PutU8 (pSSM, pBusLogic->VBoxSCSI.uTxDir);
3163 SSMR3PutU8 (pSSM, pBusLogic->VBoxSCSI.cbCDB);
3164 SSMR3PutMem (pSSM, pBusLogic->VBoxSCSI.abCDB, sizeof(pBusLogic->VBoxSCSI.abCDB));
3165 SSMR3PutU8 (pSSM, pBusLogic->VBoxSCSI.iCDB);
3166 SSMR3PutU32 (pSSM, pBusLogic->VBoxSCSI.cbBuf);
3167 SSMR3PutU32 (pSSM, pBusLogic->VBoxSCSI.iBuf);
3168 SSMR3PutBool (pSSM, pBusLogic->VBoxSCSI.fBusy);
3169 SSMR3PutU8 (pSSM, pBusLogic->VBoxSCSI.enmState);
3170 if (pBusLogic->VBoxSCSI.cbBuf)
3171 SSMR3PutMem(pSSM, pBusLogic->VBoxSCSI.pbBuf, pBusLogic->VBoxSCSI.cbBuf);
3172
3173 /*
3174 * Save the physical addresses of the command control blocks of still pending tasks.
3175 * They are processed again on resume.
3176 *
3177 * The number of pending tasks needs to be determined first.
3178 */
3179 uint32_t cTasks = 0;
3180
3181 PBUSLOGICTASKSTATE pTaskState = pBusLogic->pTasksRedoHead;
3182 if (pBusLogic->fRedo)
3183 {
3184 while (pTaskState)
3185 {
3186 cTasks++;
3187 pTaskState = pTaskState->pRedoNext;
3188 }
3189 }
3190 SSMR3PutU32(pSSM, cTasks);
3191
3192 /* Write the address of every task now. */
3193 pTaskState = pBusLogic->pTasksRedoHead;
3194 while (pTaskState)
3195 {
3196 SSMR3PutU32(pSSM, pTaskState->MailboxGuest.u32PhysAddrCCB);
3197 pTaskState = pTaskState->pRedoNext;
3198 }
3199
3200 return SSMR3PutU32(pSSM, ~0);
3201}
3202
3203/** @callback_method_impl{FNSSMDEVLOADDONE} */
3204static DECLCALLBACK(int) buslogicR3LoadDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3205{
3206 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3207
3208 buslogicR3RegisterISARange(pThis, pThis->uISABaseCode);
3209 buslogicR3Kick(pThis);
3210 return VINF_SUCCESS;
3211}
3212
3213/** @callback_method_impl{FNSSMDEVLOADEXEC} */
3214static DECLCALLBACK(int) buslogicR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
3215{
3216 PBUSLOGIC pBusLogic = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3217 int rc = VINF_SUCCESS;
3218
3219 /* We support saved states only from this and older versions. */
3220 if (uVersion > BUSLOGIC_SAVED_STATE_MINOR_VERSION)
3221 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
3222
3223 /* Every device first. */
3224 for (unsigned i = 0; i < RT_ELEMENTS(pBusLogic->aDeviceStates); i++)
3225 {
3226 PBUSLOGICDEVICE pDevice = &pBusLogic->aDeviceStates[i];
3227
3228 AssertMsg(!pDevice->cOutstandingRequests,
3229 ("There are still outstanding requests on this device\n"));
3230 bool fPresent;
3231 rc = SSMR3GetBool(pSSM, &fPresent);
3232 AssertRCReturn(rc, rc);
3233 if (pDevice->fPresent != fPresent)
3234 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Target %u config mismatch: config=%RTbool state=%RTbool"), i, pDevice->fPresent, fPresent);
3235
3236 if (uPass == SSM_PASS_FINAL)
3237 SSMR3GetU32(pSSM, (uint32_t *)&pDevice->cOutstandingRequests);
3238 }
3239
3240 if (uPass != SSM_PASS_FINAL)
3241 return VINF_SUCCESS;
3242
3243 /* Now the main device state. */
3244 SSMR3GetU8 (pSSM, (uint8_t *)&pBusLogic->regStatus);
3245 SSMR3GetU8 (pSSM, (uint8_t *)&pBusLogic->regInterrupt);
3246 SSMR3GetU8 (pSSM, (uint8_t *)&pBusLogic->regGeometry);
3247 SSMR3GetMem (pSSM, &pBusLogic->LocalRam, sizeof(pBusLogic->LocalRam));
3248 SSMR3GetU8 (pSSM, &pBusLogic->uOperationCode);
3249 SSMR3GetMem (pSSM, &pBusLogic->aCommandBuffer, sizeof(pBusLogic->aCommandBuffer));
3250 SSMR3GetU8 (pSSM, &pBusLogic->iParameter);
3251 SSMR3GetU8 (pSSM, &pBusLogic->cbCommandParametersLeft);
3252 SSMR3GetBool (pSSM, &pBusLogic->fUseLocalRam);
3253 SSMR3GetMem (pSSM, pBusLogic->aReplyBuffer, sizeof(pBusLogic->aReplyBuffer));
3254 SSMR3GetU8 (pSSM, &pBusLogic->iReply);
3255 SSMR3GetU8 (pSSM, &pBusLogic->cbReplyParametersLeft);
3256 SSMR3GetBool (pSSM, &pBusLogic->fIRQEnabled);
3257 SSMR3GetU8 (pSSM, &pBusLogic->uISABaseCode);
3258 SSMR3GetU32 (pSSM, &pBusLogic->cMailbox);
3259 if (uVersion > BUSLOGIC_SAVED_STATE_MINOR_PRE_24BIT_MBOX)
3260 SSMR3GetBool (pSSM, &pBusLogic->fMbxIs24Bit);
3261 SSMR3GetGCPhys(pSSM, &pBusLogic->GCPhysAddrMailboxOutgoingBase);
3262 SSMR3GetU32 (pSSM, &pBusLogic->uMailboxOutgoingPositionCurrent);
3263 SSMR3GetU32 (pSSM, (uint32_t *)&pBusLogic->cMailboxesReady);
3264 SSMR3GetBool (pSSM, (bool *)&pBusLogic->fNotificationSend);
3265 SSMR3GetGCPhys(pSSM, &pBusLogic->GCPhysAddrMailboxIncomingBase);
3266 SSMR3GetU32 (pSSM, &pBusLogic->uMailboxIncomingPositionCurrent);
3267 SSMR3GetBool (pSSM, &pBusLogic->fStrictRoundRobinMode);
3268 SSMR3GetBool (pSSM, &pBusLogic->fExtendedLunCCBFormat);
3269 /* Now the data for the BIOS interface. */
3270 SSMR3GetU8 (pSSM, &pBusLogic->VBoxSCSI.regIdentify);
3271 SSMR3GetU8 (pSSM, &pBusLogic->VBoxSCSI.uTargetDevice);
3272 SSMR3GetU8 (pSSM, &pBusLogic->VBoxSCSI.uTxDir);
3273 SSMR3GetU8 (pSSM, &pBusLogic->VBoxSCSI.cbCDB);
3274 SSMR3GetMem (pSSM, pBusLogic->VBoxSCSI.abCDB, sizeof(pBusLogic->VBoxSCSI.abCDB));
3275 SSMR3GetU8 (pSSM, &pBusLogic->VBoxSCSI.iCDB);
3276 SSMR3GetU32 (pSSM, &pBusLogic->VBoxSCSI.cbBuf);
3277 SSMR3GetU32 (pSSM, &pBusLogic->VBoxSCSI.iBuf);
3278 SSMR3GetBool(pSSM, (bool *)&pBusLogic->VBoxSCSI.fBusy);
3279 SSMR3GetU8 (pSSM, (uint8_t *)&pBusLogic->VBoxSCSI.enmState);
3280 if (pBusLogic->VBoxSCSI.cbBuf)
3281 {
3282 pBusLogic->VBoxSCSI.pbBuf = (uint8_t *)RTMemAllocZ(pBusLogic->VBoxSCSI.cbBuf);
3283 if (!pBusLogic->VBoxSCSI.pbBuf)
3284 {
3285 LogRel(("BusLogic: Out of memory during restore.\n"));
3286 return PDMDEV_SET_ERROR(pDevIns, VERR_NO_MEMORY,
3287 N_("BusLogic: Out of memory during restore\n"));
3288 }
3289 SSMR3GetMem(pSSM, pBusLogic->VBoxSCSI.pbBuf, pBusLogic->VBoxSCSI.cbBuf);
3290 }
3291
3292 if (pBusLogic->VBoxSCSI.fBusy)
3293 pBusLogic->fRedo = true;
3294
3295 if (uVersion > BUSLOGIC_SAVED_STATE_MINOR_PRE_ERROR_HANDLING)
3296 {
3297 /* Check if there are pending tasks saved. */
3298 uint32_t cTasks = 0;
3299
3300 SSMR3GetU32(pSSM, &cTasks);
3301
3302 if (cTasks)
3303 pBusLogic->fRedo = true;
3304
3305 for (uint32_t i = 0; i < cTasks; i++)
3306 {
3307 PBUSLOGICTASKSTATE pTaskState = (PBUSLOGICTASKSTATE)RTMemCacheAlloc(pBusLogic->hTaskCache);
3308 if (!pTaskState)
3309 {
3310 rc = VERR_NO_MEMORY;
3311 break;
3312 }
3313
3314 rc = SSMR3GetU32(pSSM, &pTaskState->MailboxGuest.u32PhysAddrCCB);
3315 if (RT_FAILURE(rc))
3316 {
3317 RTMemCacheFree(pBusLogic->hTaskCache, pTaskState);
3318 break;
3319 }
3320
3321 /* Link into the list. */
3322 pTaskState->pRedoNext = pBusLogic->pTasksRedoHead;
3323 pBusLogic->pTasksRedoHead = pTaskState;
3324 }
3325 }
3326
3327 if (RT_SUCCESS(rc))
3328 {
3329 uint32_t u32;
3330 rc = SSMR3GetU32(pSSM, &u32);
3331 if (RT_SUCCESS(rc))
3332 AssertMsgReturn(u32 == ~0U, ("%#x\n", u32), VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
3333 }
3334
3335 return rc;
3336}
3337
3338/**
3339 * Gets the pointer to the status LED of a device - called from the SCSI driver.
3340 *
3341 * @returns VBox status code.
3342 * @param pInterface Pointer to the interface structure containing the called function pointer.
3343 * @param iLUN The unit which status LED we desire. Always 0 here as the driver
3344 * doesn't know about other LUN's.
3345 * @param ppLed Where to store the LED pointer.
3346 */
3347static DECLCALLBACK(int) buslogicR3DeviceQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
3348{
3349 PBUSLOGICDEVICE pDevice = PDMILEDPORTS_2_PBUSLOGICDEVICE(pInterface);
3350 if (iLUN == 0)
3351 {
3352 *ppLed = &pDevice->Led;
3353 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
3354 return VINF_SUCCESS;
3355 }
3356 return VERR_PDM_LUN_NOT_FOUND;
3357}
3358
3359/**
3360 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
3361 */
3362static DECLCALLBACK(void *) buslogicR3DeviceQueryInterface(PPDMIBASE pInterface, const char *pszIID)
3363{
3364 PBUSLOGICDEVICE pDevice = PDMIBASE_2_PBUSLOGICDEVICE(pInterface);
3365 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDevice->IBase);
3366 PDMIBASE_RETURN_INTERFACE(pszIID, PDMISCSIPORT, &pDevice->ISCSIPort);
3367 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pDevice->ILed);
3368 return NULL;
3369}
3370
3371/**
3372 * Gets the pointer to the status LED of a unit.
3373 *
3374 * @returns VBox status code.
3375 * @param pInterface Pointer to the interface structure containing the called function pointer.
3376 * @param iLUN The unit which status LED we desire.
3377 * @param ppLed Where to store the LED pointer.
3378 */
3379static DECLCALLBACK(int) buslogicR3StatusQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
3380{
3381 PBUSLOGIC pBusLogic = PDMILEDPORTS_2_PBUSLOGIC(pInterface);
3382 if (iLUN < BUSLOGIC_MAX_DEVICES)
3383 {
3384 *ppLed = &pBusLogic->aDeviceStates[iLUN].Led;
3385 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
3386 return VINF_SUCCESS;
3387 }
3388 return VERR_PDM_LUN_NOT_FOUND;
3389}
3390
3391/**
3392 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
3393 */
3394static DECLCALLBACK(void *) buslogicR3StatusQueryInterface(PPDMIBASE pInterface, const char *pszIID)
3395{
3396 PBUSLOGIC pThis = PDMIBASE_2_PBUSLOGIC(pInterface);
3397 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
3398 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->ILeds);
3399 return NULL;
3400}
3401
3402/**
3403 * BusLogic debugger info callback.
3404 *
3405 * @param pDevIns The device instance.
3406 * @param pHlp The output helpers.
3407 * @param pszArgs The arguments.
3408 */
3409static DECLCALLBACK(void) buslogicR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
3410{
3411 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3412 unsigned i;
3413 bool fVerbose = false;
3414
3415 /* Parse arguments. */
3416 if (pszArgs)
3417 fVerbose = strstr(pszArgs, "verbose") != NULL;
3418
3419 /* Show basic information. */
3420 pHlp->pfnPrintf(pHlp,
3421 "%s#%d: PCI I/O=%RTiop ISA I/O=%RTiop MMIO=%RGp IRQ=%u GC=%RTbool R0=%RTbool\n",
3422 pDevIns->pReg->szName,
3423 pDevIns->iInstance,
3424 pThis->IOPortBase, pThis->IOISABase, pThis->MMIOBase,
3425 PCIDevGetInterruptLine(&pThis->dev),
3426 !!pThis->fGCEnabled, !!pThis->fR0Enabled);
3427
3428 /* Print mailbox state. */
3429 if (pThis->regStatus & BUSLOGIC_REGISTER_STATUS_INITIALIZATION_REQUIRED)
3430 pHlp->pfnPrintf(pHlp, "Mailbox not initialized\n");
3431 else
3432 pHlp->pfnPrintf(pHlp, "%u-bit mailbox with %u entries at %RGp\n",
3433 pThis->fMbxIs24Bit ? 24 : 32, pThis->cMailbox,
3434 pThis->GCPhysAddrMailboxOutgoingBase);
3435
3436 /* Print register contents. */
3437 pHlp->pfnPrintf(pHlp, "Registers: STAT=%02x INTR=%02x GEOM=%02x\n",
3438 pThis->regStatus, pThis->regInterrupt, pThis->regGeometry);
3439
3440 /* Print the current command, if any. */
3441 if (pThis->uOperationCode != 0xff )
3442 pHlp->pfnPrintf(pHlp, "Current command: %02X\n", pThis->uOperationCode);
3443
3444 if (fVerbose && (pThis->regStatus & BUSLOGIC_REGISTER_STATUS_INITIALIZATION_REQUIRED) == 0)
3445 {
3446 RTGCPHYS GCMailbox;
3447
3448 /* Dump the mailbox contents. */
3449 if (pThis->fMbxIs24Bit)
3450 {
3451 Mailbox24 Mbx24;
3452
3453 /* Outgoing mailbox, 24-bit format. */
3454 GCMailbox = pThis->GCPhysAddrMailboxOutgoingBase;
3455 pHlp->pfnPrintf(pHlp, " Outgoing mailbox entries (24-bit) at %06X:\n", GCMailbox);
3456 for (i = 0; i < pThis->cMailbox; ++i)
3457 {
3458 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCMailbox, &Mbx24, sizeof(Mailbox24));
3459 pHlp->pfnPrintf(pHlp, " slot %03d: CCB at %06X action code %02X", i, ADDR_TO_U32(Mbx24.aPhysAddrCCB), Mbx24.uCmdState);
3460 pHlp->pfnPrintf(pHlp, "%s\n", pThis->uMailboxOutgoingPositionCurrent == i ? " *" : "");
3461 GCMailbox += sizeof(Mailbox24);
3462 }
3463
3464 /* Incoming mailbox, 24-bit format. */
3465 GCMailbox = pThis->GCPhysAddrMailboxOutgoingBase + (pThis->cMailbox * sizeof(Mailbox24));
3466 pHlp->pfnPrintf(pHlp, " Incoming mailbox entries (24-bit) at %06X:\n", GCMailbox);
3467 for (i = 0; i < pThis->cMailbox; ++i)
3468 {
3469 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCMailbox, &Mbx24, sizeof(Mailbox24));
3470 pHlp->pfnPrintf(pHlp, " slot %03d: CCB at %06X completion code %02X", i, ADDR_TO_U32(Mbx24.aPhysAddrCCB), Mbx24.uCmdState);
3471 pHlp->pfnPrintf(pHlp, "%s\n", pThis->uMailboxIncomingPositionCurrent == i ? " *" : "");
3472 GCMailbox += sizeof(Mailbox24);
3473 }
3474
3475 }
3476 else
3477 {
3478 Mailbox32 Mbx32;
3479
3480 /* Outgoing mailbox, 32-bit format. */
3481 GCMailbox = pThis->GCPhysAddrMailboxOutgoingBase;
3482 pHlp->pfnPrintf(pHlp, " Outgoing mailbox entries (32-bit) at %08X:\n", (uint32_t)GCMailbox);
3483 for (i = 0; i < pThis->cMailbox; ++i)
3484 {
3485 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCMailbox, &Mbx32, sizeof(Mailbox32));
3486 pHlp->pfnPrintf(pHlp, " slot %03d: CCB at %08X action code %02X", i, Mbx32.u32PhysAddrCCB, Mbx32.u.out.uActionCode);
3487 pHlp->pfnPrintf(pHlp, "%s\n", pThis->uMailboxOutgoingPositionCurrent == i ? " *" : "");
3488 GCMailbox += sizeof(Mailbox32);
3489 }
3490
3491 /* Incoming mailbox, 32-bit format. */
3492 GCMailbox = pThis->GCPhysAddrMailboxOutgoingBase + (pThis->cMailbox * sizeof(Mailbox32));
3493 pHlp->pfnPrintf(pHlp, " Outgoing mailbox entries (32-bit) at %08X:\n", (uint32_t)GCMailbox);
3494 for (i = 0; i < pThis->cMailbox; ++i)
3495 {
3496 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCMailbox, &Mbx32, sizeof(Mailbox32));
3497 pHlp->pfnPrintf(pHlp, " slot %03d: CCB at %08X completion code %02X BTSTAT %02X SDSTAT %02X", i,
3498 Mbx32.u32PhysAddrCCB, Mbx32.u.in.uCompletionCode, Mbx32.u.in.uHostAdapterStatus, Mbx32.u.in.uTargetDeviceStatus);
3499 pHlp->pfnPrintf(pHlp, "%s\n", pThis->uMailboxOutgoingPositionCurrent == i ? " *" : "");
3500 GCMailbox += sizeof(Mailbox32);
3501 }
3502
3503 }
3504 }
3505}
3506
3507/* -=-=-=-=- Helper -=-=-=-=- */
3508
3509 /**
3510 * Checks if all asynchronous I/O is finished.
3511 *
3512 * Used by buslogicR3Reset, buslogicR3Suspend and buslogicR3PowerOff.
3513 *
3514 * @returns true if quiesced, false if busy.
3515 * @param pDevIns The device instance.
3516 */
3517static bool buslogicR3AllAsyncIOIsFinished(PPDMDEVINS pDevIns)
3518{
3519 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3520
3521 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aDeviceStates); i++)
3522 {
3523 PBUSLOGICDEVICE pThisDevice = &pThis->aDeviceStates[i];
3524 if (pThisDevice->pDrvBase)
3525 {
3526 if (pThisDevice->cOutstandingRequests != 0)
3527 return false;
3528 }
3529 }
3530
3531 return true;
3532}
3533
3534/**
3535 * Callback employed by buslogicR3Suspend and buslogicR3PowerOff..
3536 *
3537 * @returns true if we've quiesced, false if we're still working.
3538 * @param pDevIns The device instance.
3539 */
3540static DECLCALLBACK(bool) buslogicR3IsAsyncSuspendOrPowerOffDone(PPDMDEVINS pDevIns)
3541{
3542 if (!buslogicR3AllAsyncIOIsFinished(pDevIns))
3543 return false;
3544
3545 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3546 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
3547 return true;
3548}
3549
3550/**
3551 * Common worker for ahciR3Suspend and ahciR3PowerOff.
3552 */
3553static void buslogicR3SuspendOrPowerOff(PPDMDEVINS pDevIns, bool fPowerOff)
3554{
3555 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3556
3557 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
3558 if (!buslogicR3AllAsyncIOIsFinished(pDevIns))
3559 PDMDevHlpSetAsyncNotification(pDevIns, buslogicR3IsAsyncSuspendOrPowerOffDone);
3560 else
3561 {
3562 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
3563
3564 AssertMsg(!pThis->fNotificationSend, ("The PDM Queue should be empty at this point\n"));
3565
3566 if (pThis->fRedo)
3567 {
3568 if (fPowerOff)
3569 {
3570 /* Free tasks which would have been queued again on resume. */
3571 PBUSLOGICTASKSTATE pTaskState = pThis->pTasksRedoHead;
3572
3573 pThis->pTasksRedoHead = NULL;
3574
3575 while (pTaskState)
3576 {
3577 PBUSLOGICTASKSTATE pFree;
3578
3579 pFree = pTaskState;
3580 pTaskState = pTaskState->pRedoNext;
3581
3582 RTMemCacheFree(pThis->hTaskCache, pFree);
3583 }
3584 pThis->fRedo = false;
3585 }
3586 else if (pThis->VBoxSCSI.fBusy)
3587 {
3588 /* Destroy the task because the BIOS interface has all necessary information. */
3589 Assert(pThis->pTasksRedoHead->fBIOS);
3590 Assert(!pThis->pTasksRedoHead->pRedoNext);
3591
3592 RTMemCacheFree(pThis->hTaskCache, pThis->pTasksRedoHead);
3593 pThis->pTasksRedoHead = NULL;
3594 }
3595 }
3596 }
3597}
3598
3599/**
3600 * Suspend notification.
3601 *
3602 * @param pDevIns The device instance data.
3603 */
3604static DECLCALLBACK(void) buslogicR3Suspend(PPDMDEVINS pDevIns)
3605{
3606 Log(("buslogicR3Suspend\n"));
3607 buslogicR3SuspendOrPowerOff(pDevIns, false /* fPoweroff */);
3608}
3609
3610/**
3611 * Resume notification.
3612 *
3613 * @param pDevIns The device instance data.
3614 */
3615static DECLCALLBACK(void) buslogicR3Resume(PPDMDEVINS pDevIns)
3616{
3617 Log(("buslogicR3Resume\n"));
3618 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3619 buslogicR3Kick(pThis);
3620}
3621
3622
3623/**
3624 * Detach notification.
3625 *
3626 * One harddisk at one port has been unplugged.
3627 * The VM is suspended at this point.
3628 *
3629 * @param pDevIns The device instance.
3630 * @param iLUN The logical unit which is being detached.
3631 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
3632 */
3633static DECLCALLBACK(void) buslogicR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
3634{
3635 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3636 PBUSLOGICDEVICE pDevice = &pThis->aDeviceStates[iLUN];
3637
3638 Log(("%s:\n", __FUNCTION__));
3639
3640 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
3641 ("BusLogic: Device does not support hotplugging\n"));
3642
3643 /*
3644 * Zero some important members.
3645 */
3646 pDevice->pDrvBase = NULL;
3647 pDevice->fPresent = false;
3648 pDevice->pDrvSCSIConnector = NULL;
3649}
3650
3651/**
3652 * Attach command.
3653 *
3654 * This is called when we change block driver.
3655 *
3656 * @returns VBox status code.
3657 * @param pDevIns The device instance.
3658 * @param iLUN The logical unit which is being detached.
3659 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
3660 */
3661static DECLCALLBACK(int) buslogicR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
3662{
3663 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3664 PBUSLOGICDEVICE pDevice = &pThis->aDeviceStates[iLUN];
3665 int rc;
3666
3667 AssertMsgReturn(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
3668 ("BusLogic: Device does not support hotplugging\n"),
3669 VERR_INVALID_PARAMETER);
3670
3671 /* the usual paranoia */
3672 AssertRelease(!pDevice->pDrvBase);
3673 AssertRelease(!pDevice->pDrvSCSIConnector);
3674 Assert(pDevice->iLUN == iLUN);
3675
3676 /*
3677 * Try attach the block device and get the interfaces,
3678 * required as well as optional.
3679 */
3680 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, NULL);
3681 if (RT_SUCCESS(rc))
3682 {
3683 /* Get SCSI connector interface. */
3684 pDevice->pDrvSCSIConnector = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMISCSICONNECTOR);
3685 AssertMsgReturn(pDevice->pDrvSCSIConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
3686 pDevice->fPresent = true;
3687 }
3688 else
3689 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", pDevice->iLUN, rc));
3690
3691 if (RT_FAILURE(rc))
3692 {
3693 pDevice->pDrvBase = NULL;
3694 pDevice->pDrvSCSIConnector = NULL;
3695 }
3696 return rc;
3697}
3698
3699/**
3700 * Callback employed by buslogicR3Reset.
3701 *
3702 * @returns true if we've quiesced, false if we're still working.
3703 * @param pDevIns The device instance.
3704 */
3705static DECLCALLBACK(bool) buslogicR3IsAsyncResetDone(PPDMDEVINS pDevIns)
3706{
3707 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3708
3709 if (!buslogicR3AllAsyncIOIsFinished(pDevIns))
3710 return false;
3711 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
3712
3713 buslogicR3HwReset(pThis, true);
3714 return true;
3715}
3716
3717/**
3718 * @copydoc FNPDMDEVRESET
3719 */
3720static DECLCALLBACK(void) buslogicR3Reset(PPDMDEVINS pDevIns)
3721{
3722 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3723
3724 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
3725 if (!buslogicR3AllAsyncIOIsFinished(pDevIns))
3726 PDMDevHlpSetAsyncNotification(pDevIns, buslogicR3IsAsyncResetDone);
3727 else
3728 {
3729 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
3730 buslogicR3HwReset(pThis, true);
3731 }
3732}
3733
3734static DECLCALLBACK(void) buslogicR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
3735{
3736 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3737
3738 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
3739 pThis->pNotifierQueueRC = PDMQueueRCPtr(pThis->pNotifierQueueR3);
3740
3741 for (uint32_t i = 0; i < BUSLOGIC_MAX_DEVICES; i++)
3742 {
3743 PBUSLOGICDEVICE pDevice = &pThis->aDeviceStates[i];
3744
3745 pDevice->pBusLogicRC = PDMINS_2_DATA_RCPTR(pDevIns);
3746 }
3747
3748}
3749
3750/**
3751 * Poweroff notification.
3752 *
3753 * @param pDevIns Pointer to the device instance
3754 */
3755static DECLCALLBACK(void) buslogicR3PowerOff(PPDMDEVINS pDevIns)
3756{
3757 Log(("buslogicR3PowerOff\n"));
3758 buslogicR3SuspendOrPowerOff(pDevIns, true /* fPoweroff */);
3759}
3760
3761/**
3762 * Destroy a driver instance.
3763 *
3764 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
3765 * resources can be freed correctly.
3766 *
3767 * @param pDevIns The device instance data.
3768 */
3769static DECLCALLBACK(int) buslogicR3Destruct(PPDMDEVINS pDevIns)
3770{
3771 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3772 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
3773
3774 PDMR3CritSectDelete(&pThis->CritSectIntr);
3775
3776 /*
3777 * Free all tasks which are still hanging around
3778 * (Power off after the VM was suspended).
3779 */
3780 if (pThis->fRedo)
3781 {
3782 /* Free tasks which would have been queued again on resume. */
3783 PBUSLOGICTASKSTATE pTaskState = pThis->pTasksRedoHead;
3784
3785 pThis->pTasksRedoHead = NULL;
3786
3787 while (pTaskState)
3788 {
3789 PBUSLOGICTASKSTATE pFree;
3790
3791 pFree = pTaskState;
3792 pTaskState = pTaskState->pRedoNext;
3793
3794 RTMemCacheFree(pThis->hTaskCache, pFree);
3795 }
3796 pThis->fRedo = false;
3797 }
3798
3799 int rc = RTMemCacheDestroy(pThis->hTaskCache);
3800 AssertMsgRC(rc, ("Destroying task cache failed rc=%Rrc\n", rc));
3801
3802 return rc;
3803}
3804
3805/**
3806 * @interface_method_impl{PDMDEVREG,pfnConstruct}
3807 */
3808static DECLCALLBACK(int) buslogicR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
3809{
3810 PBUSLOGIC pThis = PDMINS_2_DATA(pDevIns, PBUSLOGIC);
3811 int rc = VINF_SUCCESS;
3812 bool fBootable = true;
3813 char achISACompat[16];
3814 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
3815
3816 /*
3817 * Init instance data (do early because of constructor).
3818 */
3819 pThis->hTaskCache = NIL_RTMEMCACHE;
3820 pThis->pDevInsR3 = pDevIns;
3821 pThis->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
3822 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
3823 pThis->IBase.pfnQueryInterface = buslogicR3StatusQueryInterface;
3824 pThis->ILeds.pfnQueryStatusLed = buslogicR3StatusQueryStatusLed;
3825
3826 PCIDevSetVendorId (&pThis->dev, 0x104b); /* BusLogic */
3827 PCIDevSetDeviceId (&pThis->dev, 0x1040); /* BT-958 */
3828 PCIDevSetCommand (&pThis->dev, 0x0003);
3829 PCIDevSetRevisionId (&pThis->dev, 0x01);
3830 PCIDevSetClassProg (&pThis->dev, 0x00); /* SCSI */
3831 PCIDevSetClassSub (&pThis->dev, 0x00); /* SCSI */
3832 PCIDevSetClassBase (&pThis->dev, 0x01); /* Mass storage */
3833 PCIDevSetBaseAddress (&pThis->dev, 0, true /*IO*/, false /*Pref*/, false /*64-bit*/, 0x00000000);
3834 PCIDevSetBaseAddress (&pThis->dev, 1, false /*IO*/, false /*Pref*/, false /*64-bit*/, 0x00000000);
3835 PCIDevSetSubSystemVendorId(&pThis->dev, 0x104b);
3836 PCIDevSetSubSystemId (&pThis->dev, 0x1040);
3837 PCIDevSetInterruptLine (&pThis->dev, 0x00);
3838 PCIDevSetInterruptPin (&pThis->dev, 0x01);
3839
3840 /*
3841 * Validate and read configuration.
3842 */
3843 if (!CFGMR3AreValuesValid(pCfg,
3844 "GCEnabled\0"
3845 "R0Enabled\0"
3846 "Bootable\0"
3847 "ISACompat\0"))
3848 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
3849 N_("BusLogic configuration error: unknown option specified"));
3850
3851 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &pThis->fGCEnabled, true);
3852 if (RT_FAILURE(rc))
3853 return PDMDEV_SET_ERROR(pDevIns, rc,
3854 N_("BusLogic configuration error: failed to read GCEnabled as boolean"));
3855 Log(("%s: fGCEnabled=%d\n", __FUNCTION__, pThis->fGCEnabled));
3856
3857 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &pThis->fR0Enabled, true);
3858 if (RT_FAILURE(rc))
3859 return PDMDEV_SET_ERROR(pDevIns, rc,
3860 N_("BusLogic configuration error: failed to read R0Enabled as boolean"));
3861 Log(("%s: fR0Enabled=%d\n", __FUNCTION__, pThis->fR0Enabled));
3862 rc = CFGMR3QueryBoolDef(pCfg, "Bootable", &fBootable, true);
3863 if (RT_FAILURE(rc))
3864 return PDMDEV_SET_ERROR(pDevIns, rc,
3865 N_("BusLogic configuration error: failed to read Bootable as boolean"));
3866 Log(("%s: fBootable=%RTbool\n", __FUNCTION__, fBootable));
3867 rc = CFGMR3QueryStringDef(pCfg, "ISACompat", achISACompat, sizeof(achISACompat), "Alternate");
3868 if (RT_FAILURE(rc))
3869 return PDMDEV_SET_ERROR(pDevIns, rc,
3870 N_("BusLogic configuration error: failed to read ISACompat as string"));
3871 Log(("%s: ISACompat=%s\n", __FUNCTION__, achISACompat));
3872
3873 /* Grok the ISACompat setting. */
3874 if (!strcmp(achISACompat, "Disabled"))
3875 pThis->uDefaultISABaseCode = ISA_BASE_DISABLED;
3876 else if (!strcmp(achISACompat, "Primary"))
3877 pThis->uDefaultISABaseCode = 0; /* I/O base at 330h. */
3878 else if (!strcmp(achISACompat, "Alternate"))
3879 pThis->uDefaultISABaseCode = 1; /* I/O base at 334h. */
3880 else
3881 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
3882 N_("BusLogic configuration error: invalid ISACompat setting"));
3883
3884 /*
3885 * Register the PCI device and its I/O regions.
3886 */
3887 rc = PDMDevHlpPCIRegister(pDevIns, &pThis->dev);
3888 if (RT_FAILURE(rc))
3889 return rc;
3890
3891 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, 32, PCI_ADDRESS_SPACE_IO, buslogicR3MmioMap);
3892 if (RT_FAILURE(rc))
3893 return rc;
3894
3895 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, 32, PCI_ADDRESS_SPACE_MEM, buslogicR3MmioMap);
3896 if (RT_FAILURE(rc))
3897 return rc;
3898
3899 if (fBootable)
3900 {
3901 /* Register I/O port space for BIOS access. */
3902 rc = PDMDevHlpIOPortRegister(pDevIns, BUSLOGIC_BIOS_IO_PORT, 4, NULL,
3903 buslogicR3BiosIoPortWrite, buslogicR3BiosIoPortRead,
3904 buslogicR3BiosIoPortWriteStr, buslogicR3BiosIoPortReadStr,
3905 "BusLogic BIOS");
3906 if (RT_FAILURE(rc))
3907 return PDMDEV_SET_ERROR(pDevIns, rc, N_("BusLogic cannot register BIOS I/O handlers"));
3908 }
3909
3910 /* Set up the compatibility I/O range. */
3911 rc = buslogicR3RegisterISARange(pThis, pThis->uDefaultISABaseCode);
3912 if (RT_FAILURE(rc))
3913 return PDMDEV_SET_ERROR(pDevIns, rc, N_("BusLogic cannot register ISA I/O handlers"));
3914
3915 /* Initialize task cache. */
3916 rc = RTMemCacheCreate(&pThis->hTaskCache, sizeof(BUSLOGICTASKSTATE), 0, UINT32_MAX,
3917 NULL, NULL, NULL, 0);
3918 if (RT_FAILURE(rc))
3919 return PDMDEV_SET_ERROR(pDevIns, rc,
3920 N_("BusLogic: Failed to initialize task cache\n"));
3921
3922 /* Initialize task queue. */
3923 rc = PDMDevHlpQueueCreate(pDevIns, sizeof(PDMQUEUEITEMCORE), 5, 0,
3924 buslogicR3NotifyQueueConsumer, true, "BusLogicTask", &pThis->pNotifierQueueR3);
3925 if (RT_FAILURE(rc))
3926 return rc;
3927 pThis->pNotifierQueueR0 = PDMQueueR0Ptr(pThis->pNotifierQueueR3);
3928 pThis->pNotifierQueueRC = PDMQueueRCPtr(pThis->pNotifierQueueR3);
3929
3930 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->CritSectIntr, RT_SRC_POS, "BusLogic-Intr#%u", pDevIns->iInstance);
3931 if (RT_FAILURE(rc))
3932 return PDMDEV_SET_ERROR(pDevIns, rc, N_("BusLogic: cannot create critical section"));
3933
3934 /* Initialize per device state. */
3935 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aDeviceStates); i++)
3936 {
3937 char szName[24];
3938 PBUSLOGICDEVICE pDevice = &pThis->aDeviceStates[i];
3939
3940 RTStrPrintf(szName, sizeof(szName), "Device%u", i);
3941
3942 /* Initialize static parts of the device. */
3943 pDevice->iLUN = i;
3944 pDevice->pBusLogicR3 = pThis;
3945 pDevice->pBusLogicR0 = PDMINS_2_DATA_R0PTR(pDevIns);
3946 pDevice->pBusLogicRC = PDMINS_2_DATA_RCPTR(pDevIns);
3947 pDevice->Led.u32Magic = PDMLED_MAGIC;
3948 pDevice->IBase.pfnQueryInterface = buslogicR3DeviceQueryInterface;
3949 pDevice->ISCSIPort.pfnSCSIRequestCompleted = buslogicR3DeviceSCSIRequestCompleted;
3950 pDevice->ISCSIPort.pfnQueryDeviceLocation = buslogicR3QueryDeviceLocation;
3951 pDevice->ILed.pfnQueryStatusLed = buslogicR3DeviceQueryStatusLed;
3952
3953 /* Attach SCSI driver. */
3954 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, szName);
3955 if (RT_SUCCESS(rc))
3956 {
3957 /* Get SCSI connector interface. */
3958 pDevice->pDrvSCSIConnector = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMISCSICONNECTOR);
3959 AssertMsgReturn(pDevice->pDrvSCSIConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
3960
3961 pDevice->fPresent = true;
3962 }
3963 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
3964 {
3965 pDevice->pDrvBase = NULL;
3966 pDevice->fPresent = false;
3967 rc = VINF_SUCCESS;
3968 Log(("BusLogic: no driver attached to device %s\n", szName));
3969 }
3970 else
3971 {
3972 AssertLogRelMsgFailed(("BusLogic: Failed to attach %s\n", szName));
3973 return rc;
3974 }
3975 }
3976
3977 /*
3978 * Attach status driver (optional).
3979 */
3980 PPDMIBASE pBase;
3981 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
3982 if (RT_SUCCESS(rc))
3983 pThis->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
3984 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
3985 {
3986 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
3987 return PDMDEV_SET_ERROR(pDevIns, rc, N_("BusLogic cannot attach to status driver"));
3988 }
3989
3990 rc = PDMDevHlpSSMRegisterEx(pDevIns, BUSLOGIC_SAVED_STATE_MINOR_VERSION, sizeof(*pThis), NULL,
3991 NULL, buslogicR3LiveExec, NULL,
3992 NULL, buslogicR3SaveExec, NULL,
3993 NULL, buslogicR3LoadExec, buslogicR3LoadDone);
3994 if (RT_FAILURE(rc))
3995 return PDMDEV_SET_ERROR(pDevIns, rc, N_("BusLogic cannot register save state handlers"));
3996
3997 /*
3998 * Register the debugger info callback.
3999 */
4000 char szTmp[128];
4001 RTStrPrintf(szTmp, sizeof(szTmp), "%s%d", pDevIns->pReg->szName, pDevIns->iInstance);
4002 PDMDevHlpDBGFInfoRegister(pDevIns, szTmp, "BusLogic HBA info", buslogicR3Info);
4003
4004 rc = buslogicR3HwReset(pThis, true);
4005 AssertMsgRC(rc, ("hardware reset of BusLogic host adapter failed rc=%Rrc\n", rc));
4006
4007 return rc;
4008}
4009
4010/**
4011 * The device registration structure.
4012 */
4013const PDMDEVREG g_DeviceBusLogic =
4014{
4015 /* u32Version */
4016 PDM_DEVREG_VERSION,
4017 /* szName */
4018 "buslogic",
4019 /* szRCMod */
4020 "VBoxDDGC.gc",
4021 /* szR0Mod */
4022 "VBoxDDR0.r0",
4023 /* pszDescription */
4024 "BusLogic BT-958 SCSI host adapter.\n",
4025 /* fFlags */
4026 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0 |
4027 PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION | PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION |
4028 PDM_DEVREG_FLAGS_FIRST_RESET_NOTIFICATION,
4029 /* fClass */
4030 PDM_DEVREG_CLASS_STORAGE,
4031 /* cMaxInstances */
4032 ~0U,
4033 /* cbInstance */
4034 sizeof(BUSLOGIC),
4035 /* pfnConstruct */
4036 buslogicR3Construct,
4037 /* pfnDestruct */
4038 buslogicR3Destruct,
4039 /* pfnRelocate */
4040 buslogicR3Relocate,
4041 /* pfnMemSetup */
4042 NULL,
4043 /* pfnPowerOn */
4044 NULL,
4045 /* pfnReset */
4046 buslogicR3Reset,
4047 /* pfnSuspend */
4048 buslogicR3Suspend,
4049 /* pfnResume */
4050 buslogicR3Resume,
4051 /* pfnAttach */
4052 buslogicR3Attach,
4053 /* pfnDetach */
4054 buslogicR3Detach,
4055 /* pfnQueryInterface. */
4056 NULL,
4057 /* pfnInitComplete */
4058 NULL,
4059 /* pfnPowerOff */
4060 buslogicR3PowerOff,
4061 /* pfnSoftReset */
4062 NULL,
4063 /* u32VersionEnd */
4064 PDM_DEVREG_VERSION
4065};
4066
4067#endif /* IN_RING3 */
4068#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