VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DevLsiLogicSCSI.cpp@ 88662

Last change on this file since 88662 was 88643, checked in by vboxsync, 4 years ago

Devices/Storage/DevLsiLogicSCSI: Fix crashing on NULL pointer dereference during device construction when the SAS variant is used (regression from r143802), bugref:9914

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 252.2 KB
Line 
1/* $Id: DevLsiLogicSCSI.cpp 88643 2021-04-22 07:57:03Z vboxsync $ */
2/** @file
3 * DevLsiLogicSCSI - LsiLogic LSI53c1030 SCSI controller.
4 */
5
6/*
7 * Copyright (C) 2006-2020 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DEV_LSILOGICSCSI
23#include <VBox/vmm/pdmdev.h>
24#include <VBox/vmm/pdmstorageifs.h>
25#include <VBox/vmm/pdmqueue.h>
26#include <VBox/vmm/pdmthread.h>
27#include <VBox/vmm/pdmcritsect.h>
28#include <VBox/AssertGuest.h>
29#include <VBox/scsi.h>
30#include <VBox/sup.h>
31#include <iprt/assert.h>
32#include <iprt/asm.h>
33#include <iprt/string.h>
34#include <iprt/list.h>
35#ifdef IN_RING3
36# include <iprt/memcache.h>
37# include <iprt/mem.h>
38# include <iprt/param.h>
39# include <iprt/uuid.h>
40# include <iprt/time.h>
41#endif
42
43#include "DevLsiLogicSCSI.h"
44#include "VBoxSCSI.h"
45
46#include "VBoxDD.h"
47
48
49/*********************************************************************************************************************************
50* Defined Constants And Macros *
51*********************************************************************************************************************************/
52/** The current saved state version. */
53#define LSILOGIC_SAVED_STATE_VERSION 5
54/** The saved state version used by VirtualBox before the diagnostic
55 * memory access was implemented. */
56#define LSILOGIC_SAVED_STATE_VERSION_PRE_DIAG_MEM 4
57/** The saved state version used by VirtualBox before the doorbell status flag
58 * was changed from bool to a 32bit enum. */
59#define LSILOGIC_SAVED_STATE_VERSION_BOOL_DOORBELL 3
60/** The saved state version used by VirtualBox before SAS support was added. */
61#define LSILOGIC_SAVED_STATE_VERSION_PRE_SAS 2
62/** The saved state version used by VirtualBox 3.0 and earlier. It does not
63 * include the device config part. */
64#define LSILOGIC_SAVED_STATE_VERSION_VBOX_30 1
65
66/** Maximum number of entries in the release log. */
67#define MAX_REL_LOG_ERRORS 1024
68
69#define LSILOGIC_RTGCPHYS_FROM_U32(Hi, Lo) ( (RTGCPHYS)RT_MAKE_U64(Lo, Hi) )
70
71/** Upper number a buffer is freed if it was too big before. */
72#define LSILOGIC_MAX_ALLOC_TOO_MUCH 20
73
74/** Maximum size of the memory regions (prevents teh guest from DOSing the host by
75 * allocating loadds of memory). */
76#define LSILOGIC_MEMORY_REGIONS_MAX _1M
77
78
79/*********************************************************************************************************************************
80* Structures and Typedefs *
81*********************************************************************************************************************************/
82
83/** Pointer to the shared instance data for the LsiLogic emulation. */
84typedef struct LSILOGICSCSI *PLSILOGICSCSI;
85
86#ifdef IN_RING3
87/**
88 * Memory buffer callback.
89 *
90 * @returns nothing.
91 * @param pDevIns The device instance.
92 * @param GCPhys The guest physical address of the memory buffer.
93 * @param pSgBuf The pointer to the host R3 S/G buffer.
94 * @param cbCopy How many bytes to copy between the two buffers.
95 * @param pcbSkip Initially contains the amount of bytes to skip
96 * starting from the guest physical address before
97 * accessing the S/G buffer and start copying data.
98 * On return this contains the remaining amount if
99 * cbCopy < *pcbSkip or 0 otherwise.
100 */
101typedef DECLCALLBACKTYPE(void, FNLSILOGICR3MEMCOPYCALLBACK,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys,
102 PRTSGBUF pSgBuf, size_t cbCopy, size_t *pcbSkip));
103/** Pointer to a memory copy buffer callback. */
104typedef FNLSILOGICR3MEMCOPYCALLBACK *PFNLSILOGICR3MEMCOPYCALLBACK;
105#endif
106
107/**
108 * Reply data.
109 */
110typedef struct LSILOGICSCSIREPLY
111{
112 /** Lower 32 bits of the reply address in memory. */
113 uint32_t u32HostMFALowAddress;
114 /** Full address of the reply in guest memory. */
115 RTGCPHYS GCPhysReplyAddress;
116 /** Size of the reply. */
117 uint32_t cbReply;
118 /** Different views to the reply depending on the request type. */
119 MptReplyUnion Reply;
120} LSILOGICSCSIREPLY;
121/** Pointer to reply data. */
122typedef LSILOGICSCSIREPLY *PLSILOGICSCSIREPLY;
123
124/**
125 * Memory region of the IOC.
126 */
127typedef struct LSILOGICMEMREGN
128{
129 /** List node. */
130 RTLISTNODE NodeList;
131 /** 32bit address the region starts to describe. */
132 uint32_t u32AddrStart;
133 /** 32bit address the region ends (inclusive). */
134 uint32_t u32AddrEnd;
135 /** Data for this region - variable. */
136 uint32_t au32Data[1];
137} LSILOGICMEMREGN;
138/** Pointer to a memory region. */
139typedef LSILOGICMEMREGN *PLSILOGICMEMREGN;
140
141/**
142 * State of a device attached to the buslogic host adapter.
143 *
144 * @implements PDMIBASE
145 * @implements PDMISCSIPORT
146 * @implements PDMILEDPORTS
147 */
148typedef struct LSILOGICDEVICE
149{
150 /** Pointer to the owning lsilogic device instance - R3 pointer */
151 PPDMDEVINSR3 pDevIns;
152
153 /** LUN of the device. */
154 uint32_t iLUN;
155 /** Number of outstanding tasks on the port. */
156 volatile uint32_t cOutstandingRequests;
157
158 /** Our base interface. */
159 PDMIBASE IBase;
160 /** Media port interface. */
161 PDMIMEDIAPORT IMediaPort;
162 /** Extended media port interface. */
163 PDMIMEDIAEXPORT IMediaExPort;
164 /** Led interface. */
165 PDMILEDPORTS ILed;
166 /** Pointer to the attached driver's base interface. */
167 R3PTRTYPE(PPDMIBASE) pDrvBase;
168 /** Pointer to the attached driver's media interface. */
169 R3PTRTYPE(PPDMIMEDIA) pDrvMedia;
170 /** Pointer to the attached driver's extended media interface. */
171 R3PTRTYPE(PPDMIMEDIAEX) pDrvMediaEx;
172 /** The status LED state for this device. */
173 PDMLED Led;
174 /** Device name. */
175 char szName[16];
176} LSILOGICDEVICE;
177/** Pointer to a device state. */
178typedef LSILOGICDEVICE *PLSILOGICDEVICE;
179
180/** Pointer to a task state. */
181typedef struct LSILOGICREQ *PLSILOGICREQ;
182
183
184/**
185 * Shared instance data for the LsiLogic emulation.
186 */
187typedef struct LSILOGICSCSI
188{
189 /** The state the controller is currently in. */
190 LSILOGICSTATE enmState;
191 /** Who needs to init the driver to get into operational state. */
192 LSILOGICWHOINIT enmWhoInit;
193 /** Flag whether we are in doorbell function. */
194 LSILOGICDOORBELLSTATE enmDoorbellState;
195 /** Flag whether diagnostic access is enabled. */
196 bool fDiagnosticEnabled;
197 /** Flag whether a notification was send to R3. */
198 bool fNotificationSent;
199 /** Flag whether the guest enabled event notification from the IOC. */
200 bool fEventNotificationEnabled;
201 /** Flag whether the diagnostic address and RW registers are enabled. */
202 bool fDiagRegsEnabled;
203
204 /** Number of device states allocated. */
205 uint32_t cDeviceStates;
206 uint32_t u32Padding1;
207
208 /** Interrupt mask. */
209 volatile uint32_t uInterruptMask;
210 /** Interrupt status register. */
211 volatile uint32_t uInterruptStatus;
212
213 /** Buffer for messages which are passed through the doorbell using the
214 * handshake method. */
215 uint32_t aMessage[sizeof(MptConfigurationRequest)]; /** @todo r=bird: Looks like 4 times the required size? Please explain in comment if this correct... */
216 /** Actual position in the buffer. */
217 uint32_t iMessage;
218 /** Size of the message which is given in the doorbell message in dwords. */
219 uint32_t cMessage;
220
221 /** Reply buffer.
222 * @note 60 bytes */
223 MptReplyUnion ReplyBuffer;
224 /** Next entry to read. */
225 uint32_t uNextReplyEntryRead;
226 /** Size of the reply in the buffer in 16bit words. */
227 uint32_t cReplySize;
228
229 /** The fault code of the I/O controller if we are in the fault state. */
230 uint16_t u16IOCFaultCode;
231 uint16_t u16Padding2;
232
233 /** Upper 32 bits of the message frame address to locate requests in guest memory. */
234 uint32_t u32HostMFAHighAddr;
235 /** Upper 32 bits of the sense buffer address. */
236 uint32_t u32SenseBufferHighAddr;
237 /** Maximum number of devices the driver reported he can handle. */
238 uint8_t cMaxDevices;
239 /** Maximum number of buses the driver reported he can handle. */
240 uint8_t cMaxBuses;
241 /** Current size of reply message frames in the guest. */
242 uint16_t cbReplyFrame;
243
244 /** Next key to write in the sequence to get access
245 * to diagnostic memory. */
246 uint32_t iDiagnosticAccess;
247
248 /** Number entries configured for the reply queue. */
249 uint32_t cReplyQueueEntries;
250 /** Number entries configured for the outstanding request queue. */
251 uint32_t cRequestQueueEntries;
252
253 /** Critical section protecting the reply post queue. */
254 PDMCRITSECT ReplyPostQueueCritSect;
255 /** Critical section protecting the reply free queue. */
256 PDMCRITSECT ReplyFreeQueueCritSect;
257 /** Critical section protecting the request queue against
258 * concurrent access from the guest. */
259 PDMCRITSECT RequestQueueCritSect;
260 /** Critical section protecting the reply free queue against
261 * concurrent write access from the guest. */
262 PDMCRITSECT ReplyFreeQueueWriteCritSect;
263
264 /** The reply free qeueue (only the first cReplyQueueEntries are used). */
265 uint32_t volatile aReplyFreeQueue[LSILOGICSCSI_REPLY_QUEUE_DEPTH_MAX];
266 /** The reply post qeueue (only the first cReplyQueueEntries are used). */
267 uint32_t volatile aReplyPostQueue[LSILOGICSCSI_REPLY_QUEUE_DEPTH_MAX];
268 /** The request qeueue (only the first cRequestQueueEntries are used). */
269 uint32_t volatile aRequestQueue[LSILOGICSCSI_REQUEST_QUEUE_DEPTH_MAX];
270
271 /** Next free entry in the reply queue the guest can write a address to. */
272 volatile uint32_t uReplyFreeQueueNextEntryFreeWrite;
273 /** Next valid entry the controller can read a valid address for reply frames from. */
274 volatile uint32_t uReplyFreeQueueNextAddressRead;
275
276 /** Next free entry in the reply queue the guest can write a address to. */
277 volatile uint32_t uReplyPostQueueNextEntryFreeWrite;
278 /** Next valid entry the controller can read a valid address for reply frames from. */
279 volatile uint32_t uReplyPostQueueNextAddressRead;
280
281 /** Next free entry the guest can write a address to a request frame to. */
282 volatile uint32_t uRequestQueueNextEntryFreeWrite;
283 /** Next valid entry the controller can read a valid address for request frames from. */
284 volatile uint32_t uRequestQueueNextAddressRead;
285
286 /** Indicates that PDMDevHlpAsyncNotificationCompleted should be called when
287 * a port is entering the idle state. */
288 bool volatile fSignalIdle;
289 /** Flag whether the worker thread is sleeping. */
290 volatile bool fWrkThreadSleeping;
291 /** Flag whether a request from the BIOS is pending which the
292 * worker thread needs to process. */
293 volatile bool fBiosReqPending;
294 bool fPadding3;
295
296 /** Current address to read from or write to in the diagnostic memory region. */
297 uint32_t u32DiagMemAddr;
298
299 /** Emulated controller type */
300 LSILOGICCTRLTYPE enmCtrlType;
301 /** Handle counter */
302 uint16_t u16NextHandle;
303
304 /** Number of ports this controller has. */
305 uint8_t cPorts;
306 uint8_t afPadding4;
307
308 /** The event semaphore the processing thread waits on. */
309 SUPSEMEVENT hEvtProcess;
310
311 /** PCI Region \#0: I/O ports register access. */
312 IOMIOPORTHANDLE hIoPortsReg;
313 /** PCI Region \#1: MMIO register access. */
314 IOMMMIOHANDLE hMmioReg;
315 /** PCI Region \#2: MMIO diag. */
316 IOMMMIOHANDLE hMmioDiag;
317 /** ISA Ports for the BIOS (when booting is configured). */
318 IOMIOPORTHANDLE hIoPortsBios;
319} LSILOGICSCSI;
320AssertCompileMemberAlignment(LSILOGICSCSI, ReplyPostQueueCritSect, 8);
321
322/**
323 * Ring-3 instance data for the LsiLogic emulation.
324 */
325typedef struct LSILOGICSCSIR3
326{
327 /** States for attached devices. */
328 R3PTRTYPE(PLSILOGICDEVICE) paDeviceStates;
329 /** Status LUN: The base interface. */
330 PDMIBASE IBase;
331 /** Status LUN: Leds interface. */
332 PDMILEDPORTS ILeds;
333 /** Status LUN: Partner of ILeds. */
334 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
335 /** Status LUN: Media Notifys. */
336 R3PTRTYPE(PPDMIMEDIANOTIFY) pMediaNotify;
337 /** Pointer to the configuration page area. */
338 R3PTRTYPE(PMptConfigurationPagesSupported) pConfigurationPages;
339
340 /** BIOS emulation. */
341 VBOXSCSI VBoxSCSI;
342
343 /** Current size of the memory regions. */
344 uint32_t cbMemRegns;
345 uint32_t u32Padding3;
346
347 /** Critical section protecting the memory regions. */
348 RTCRITSECT CritSectMemRegns;
349 /** List of memory regions - PLSILOGICMEMREGN. */
350 RTLISTANCHORR3 ListMemRegns;
351
352 /** Worker thread. */
353 R3PTRTYPE(PPDMTHREAD) pThreadWrk;
354
355 /** The device instace - only for getting bearings in interface methods. */
356 PPDMDEVINSR3 pDevIns;
357} LSILOGICSCSIR3;
358/** Pointer to the ring-3 instance data for the LsiLogic emulation. */
359typedef LSILOGICSCSIR3 *PLSILOGICSCSIR3;
360
361
362/**
363 * Ring-0 instance data for the LsiLogic emulation.
364 */
365typedef struct LSILOGICSCSIR0
366{
367 uint64_t u64Unused;
368} LSILOGICSCSIR0;
369/** Pointer to the ring-0 instance data for the LsiLogic emulation. */
370typedef LSILOGICSCSIR0 *PLSILOGICSCSIR0;
371
372
373/**
374 * Raw-mode instance data for the LsiLogic emulation.
375 */
376typedef struct LSILOGICSCSIRC
377{
378 uint64_t u64Unused;
379} LSILOGICSCSIRC;
380/** Pointer to the raw-mode instance data for the LsiLogic emulation. */
381typedef LSILOGICSCSIRC *PLSILOGICSCSIRC;
382
383
384/** The current context instance data for the LsiLogic emulation. */
385typedef CTX_SUFF(LSILOGICSCSI) LSILOGICSCSICC;
386/** Pointer to the current context instance data for the LsiLogic emulation. */
387typedef CTX_SUFF(PLSILOGICSCSI) PLSILOGICSCSICC;
388
389
390/**
391 * Task state object which holds all necessary data while
392 * processing the request from the guest.
393 */
394typedef struct LSILOGICREQ
395{
396 /** I/O request handle. */
397 PDMMEDIAEXIOREQ hIoReq;
398 /** Next in the redo list. */
399 PLSILOGICREQ pRedoNext;
400 /** Target device. */
401 PLSILOGICDEVICE pTargetDevice;
402 /** The message request from the guest. */
403 MptRequestUnion GuestRequest;
404 /** Address of the message request frame in guests memory.
405 * Used to read the S/G entries in the second step. */
406 RTGCPHYS GCPhysMessageFrameAddr;
407 /** Physical start address of the S/G list. */
408 RTGCPHYS GCPhysSgStart;
409 /** Chain offset */
410 uint32_t cChainOffset;
411 /** Pointer to the sense buffer. */
412 uint8_t abSenseBuffer[18];
413 /** Flag whether the request was issued from the BIOS. */
414 bool fBIOS;
415 /** SCSI status code. */
416 uint8_t u8ScsiSts;
417} LSILOGICREQ;
418
419
420#ifndef VBOX_DEVICE_STRUCT_TESTCASE
421
422
423/*********************************************************************************************************************************
424* Internal Functions *
425*********************************************************************************************************************************/
426RT_C_DECLS_BEGIN
427#ifdef IN_RING3
428static void lsilogicR3InitializeConfigurationPages(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC);
429static void lsilogicR3ConfigurationPagesFree(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC);
430static int lsilogicR3ProcessConfigurationRequest(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC,
431 PMptConfigurationRequest pConfigurationReq, PMptConfigurationReply pReply);
432#endif
433RT_C_DECLS_END
434
435
436/*********************************************************************************************************************************
437* Global Variables *
438*********************************************************************************************************************************/
439/** Key sequence the guest has to write to enable access
440 * to diagnostic memory. */
441static const uint8_t g_lsilogicDiagnosticAccess[] = {0x04, 0x0b, 0x02, 0x07, 0x0d};
442
443/**
444 * Updates the status of the interrupt pin of the device.
445 *
446 * @returns nothing.
447 * @param pDevIns The device instance.
448 * @param pThis Pointer to the shared LsiLogic device state.
449 */
450static void lsilogicUpdateInterrupt(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis)
451{
452 uint32_t uIntSts;
453
454 LogFlowFunc(("Updating interrupts\n"));
455
456 /* Mask out doorbell status so that it does not affect interrupt updating. */
457 uIntSts = (ASMAtomicReadU32(&pThis->uInterruptStatus) & ~LSILOGIC_REG_HOST_INTR_STATUS_DOORBELL_STS);
458 /* Check maskable interrupts. */
459 uIntSts &= ~(ASMAtomicReadU32(&pThis->uInterruptMask) & ~LSILOGIC_REG_HOST_INTR_MASK_IRQ_ROUTING);
460
461 if (uIntSts)
462 {
463 LogFlowFunc(("Setting interrupt\n"));
464 PDMDevHlpPCISetIrq(pDevIns, 0, 1);
465 }
466 else
467 {
468 LogFlowFunc(("Clearing interrupt\n"));
469 PDMDevHlpPCISetIrq(pDevIns, 0, 0);
470 }
471}
472
473/**
474 * Sets a given interrupt status bit in the status register and
475 * updates the interrupt status.
476 *
477 * @returns nothing.
478 * @param pDevIns The device instance.
479 * @param pThis Pointer to the shared LsiLogic device state.
480 * @param uStatus The status bit to set.
481 */
482DECLINLINE(void) lsilogicSetInterrupt(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, uint32_t uStatus)
483{
484 ASMAtomicOrU32(&pThis->uInterruptStatus, uStatus);
485 lsilogicUpdateInterrupt(pDevIns, pThis);
486}
487
488/**
489 * Clears a given interrupt status bit in the status register and
490 * updates the interrupt status.
491 *
492 * @returns nothing.
493 * @param pDevIns The device instance.
494 * @param pThis Pointer to the shared LsiLogic device state.
495 * @param uStatus The status bit to set.
496 */
497DECLINLINE(void) lsilogicClearInterrupt(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, uint32_t uStatus)
498{
499 ASMAtomicAndU32(&pThis->uInterruptStatus, ~uStatus);
500 lsilogicUpdateInterrupt(pDevIns, pThis);
501}
502
503
504#ifdef IN_RING3
505/**
506 * Sets the I/O controller into fault state and sets the fault code.
507 *
508 * @returns nothing
509 * @param pThis Pointer to the shared LsiLogic device state.
510 * @param uIOCFaultCode Fault code to set.
511 */
512DECLINLINE(void) lsilogicSetIOCFaultCode(PLSILOGICSCSI pThis, uint16_t uIOCFaultCode)
513{
514 if (pThis->enmState != LSILOGICSTATE_FAULT)
515 {
516 LogFunc(("Setting I/O controller into FAULT state: uIOCFaultCode=%u\n", uIOCFaultCode));
517 pThis->enmState = LSILOGICSTATE_FAULT;
518 pThis->u16IOCFaultCode = uIOCFaultCode;
519 }
520 else
521 LogFunc(("We are already in FAULT state\n"));
522}
523#endif /* IN_RING3 */
524
525
526/**
527 * Returns the number of frames in the reply free queue.
528 *
529 * @returns Number of frames in the reply free queue.
530 * @param pThis Pointer to the shared LsiLogic device state.
531 */
532DECLINLINE(uint32_t) lsilogicReplyFreeQueueGetFrameCount(PLSILOGICSCSI pThis)
533{
534 uint32_t cReplyFrames = 0;
535
536 if (pThis->uReplyFreeQueueNextAddressRead <= pThis->uReplyFreeQueueNextEntryFreeWrite)
537 cReplyFrames = pThis->uReplyFreeQueueNextEntryFreeWrite - pThis->uReplyFreeQueueNextAddressRead;
538 else
539 cReplyFrames = pThis->cReplyQueueEntries - pThis->uReplyFreeQueueNextAddressRead + pThis->uReplyFreeQueueNextEntryFreeWrite;
540
541 return cReplyFrames;
542}
543
544#ifdef IN_RING3
545
546/**
547 * Returns the number of free entries in the reply post queue.
548 *
549 * @returns Number of frames in the reply free queue.
550 * @param pThis Pointer to the shared LsiLogic device state.
551 */
552DECLINLINE(uint32_t) lsilogicReplyPostQueueGetFrameCount(PLSILOGICSCSI pThis)
553{
554 uint32_t cReplyFrames = 0;
555
556 if (pThis->uReplyPostQueueNextAddressRead <= pThis->uReplyPostQueueNextEntryFreeWrite)
557 cReplyFrames = pThis->cReplyQueueEntries - pThis->uReplyPostQueueNextEntryFreeWrite + pThis->uReplyPostQueueNextAddressRead;
558 else
559 cReplyFrames = pThis->uReplyPostQueueNextEntryFreeWrite - pThis->uReplyPostQueueNextAddressRead;
560
561 return cReplyFrames;
562}
563
564
565/**
566 * Performs a hard reset on the controller.
567 *
568 * @returns VBox status code.
569 * @param pDevIns The device instance.
570 * @param pThis Pointer to the shared LsiLogic device state.
571 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
572 */
573static int lsilogicR3HardReset(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
574{
575 pThis->enmState = LSILOGICSTATE_RESET;
576 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_NOT_IN_USE;
577
578 /* The interrupts are masked out. */
579 pThis->uInterruptMask |= LSILOGIC_REG_HOST_INTR_MASK_DOORBELL
580 | LSILOGIC_REG_HOST_INTR_MASK_REPLY;
581 /* Reset interrupt states. */
582 pThis->uInterruptStatus = 0;
583 lsilogicUpdateInterrupt(pDevIns, pThis);
584
585 /* Reset the queues. */
586 pThis->uReplyFreeQueueNextEntryFreeWrite = 0;
587 pThis->uReplyFreeQueueNextAddressRead = 0;
588 pThis->uReplyPostQueueNextEntryFreeWrite = 0;
589 pThis->uReplyPostQueueNextAddressRead = 0;
590 pThis->uRequestQueueNextEntryFreeWrite = 0;
591 pThis->uRequestQueueNextAddressRead = 0;
592
593 /* Disable diagnostic access. */
594 pThis->iDiagnosticAccess = 0;
595 pThis->fDiagnosticEnabled = false;
596 pThis->fDiagRegsEnabled = false;
597
598 /* Set default values. */
599 pThis->cMaxDevices = pThis->cDeviceStates;
600 pThis->cMaxBuses = 1;
601 pThis->cbReplyFrame = 128; /** @todo Figure out where it is needed. */
602 pThis->u16NextHandle = 1;
603 pThis->u32DiagMemAddr = 0;
604
605 lsilogicR3InitializeConfigurationPages(pDevIns, pThis, pThisCC);
606
607 /* Mark that we finished performing the reset. */
608 pThis->enmState = LSILOGICSTATE_READY;
609 return VINF_SUCCESS;
610}
611
612/**
613 * Allocates the configuration pages based on the device.
614 *
615 * @returns nothing.
616 * @param pThis Pointer to the shared LsiLogic device state.
617 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
618 */
619static int lsilogicR3ConfigurationPagesAlloc(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
620{
621 pThisCC->pConfigurationPages = (PMptConfigurationPagesSupported)RTMemAllocZ(sizeof(MptConfigurationPagesSupported));
622 if (!pThisCC->pConfigurationPages)
623 return VERR_NO_MEMORY;
624
625 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
626 {
627 PMptConfigurationPagesSas pPages = &pThisCC->pConfigurationPages->u.SasPages;
628
629 pPages->cbManufacturingPage7 = LSILOGICSCSI_MANUFACTURING7_GET_SIZE(pThis->cPorts);
630 PMptConfigurationPageManufacturing7 pManufacturingPage7 = (PMptConfigurationPageManufacturing7)RTMemAllocZ(pPages->cbManufacturingPage7);
631 AssertPtrReturn(pManufacturingPage7, VERR_NO_MEMORY);
632 pPages->pManufacturingPage7 = pManufacturingPage7;
633
634 /* SAS I/O unit page 0 - Port specific information. */
635 pPages->cbSASIOUnitPage0 = LSILOGICSCSI_SASIOUNIT0_GET_SIZE(pThis->cPorts);
636 PMptConfigurationPageSASIOUnit0 pSASPage0 = (PMptConfigurationPageSASIOUnit0)RTMemAllocZ(pPages->cbSASIOUnitPage0);
637 AssertPtrReturn(pSASPage0, VERR_NO_MEMORY);
638 pPages->pSASIOUnitPage0 = pSASPage0;
639
640 /* SAS I/O unit page 1 - Port specific settings. */
641 pPages->cbSASIOUnitPage1 = LSILOGICSCSI_SASIOUNIT1_GET_SIZE(pThis->cPorts);
642 PMptConfigurationPageSASIOUnit1 pSASPage1 = (PMptConfigurationPageSASIOUnit1)RTMemAllocZ(pPages->cbSASIOUnitPage1);
643 AssertPtrReturn(pSASPage1, VERR_NO_MEMORY);
644 pPages->pSASIOUnitPage1 = pSASPage1;
645
646 pPages->cPHYs = pThis->cPorts;
647 pPages->paPHYs = (PMptPHY)RTMemAllocZ(pPages->cPHYs * sizeof(MptPHY));
648 AssertPtrReturn(pPages->paPHYs, VERR_NO_MEMORY);
649
650 /* Initialize the PHY configuration */
651 for (unsigned i = 0; i < pThis->cPorts; i++)
652 {
653 /* Settings for present devices. */
654 if (pThisCC->paDeviceStates[i].pDrvBase)
655 {
656 PMptSASDevice pSASDevice = (PMptSASDevice)RTMemAllocZ(sizeof(MptSASDevice));
657 AssertPtrReturn(pSASDevice, VERR_NO_MEMORY);
658
659 /* Link into device list. */
660 if (!pPages->cDevices)
661 {
662 pPages->pSASDeviceHead = pSASDevice;
663 pPages->pSASDeviceTail = pSASDevice;
664 pPages->cDevices = 1;
665 }
666 else
667 {
668 pSASDevice->pPrev = pPages->pSASDeviceTail;
669 pPages->pSASDeviceTail->pNext = pSASDevice;
670 pPages->pSASDeviceTail = pSASDevice;
671 pPages->cDevices++;
672 }
673 }
674 }
675 }
676
677 return VINF_SUCCESS;
678}
679
680/**
681 * Frees the configuration pages if allocated.
682 *
683 * @returns nothing.
684 * @param pThis Pointer to the shared LsiLogic device state.
685 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
686 */
687static void lsilogicR3ConfigurationPagesFree(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
688{
689
690 if (pThisCC->pConfigurationPages)
691 {
692 /* Destroy device list if we emulate a SAS controller. */
693 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
694 {
695 PMptConfigurationPagesSas pSasPages = &pThisCC->pConfigurationPages->u.SasPages;
696 PMptSASDevice pSASDeviceCurr = pSasPages->pSASDeviceHead;
697
698 while (pSASDeviceCurr)
699 {
700 PMptSASDevice pFree = pSASDeviceCurr;
701
702 pSASDeviceCurr = pSASDeviceCurr->pNext;
703 RTMemFree(pFree);
704 }
705 if (pSasPages->paPHYs)
706 RTMemFree(pSasPages->paPHYs);
707 if (pSasPages->pManufacturingPage7)
708 RTMemFree(pSasPages->pManufacturingPage7);
709 if (pSasPages->pSASIOUnitPage0)
710 RTMemFree(pSasPages->pSASIOUnitPage0);
711 if (pSasPages->pSASIOUnitPage1)
712 RTMemFree(pSasPages->pSASIOUnitPage1);
713
714 pSasPages->pSASDeviceHead = NULL;
715 pSasPages->paPHYs = NULL;
716 pSasPages->pManufacturingPage7 = NULL;
717 pSasPages->pSASIOUnitPage0 = NULL;
718 pSasPages->pSASIOUnitPage1 = NULL;
719 }
720
721 RTMemFree(pThisCC->pConfigurationPages);
722 pThisCC->pConfigurationPages = NULL;
723 }
724}
725
726/**
727 * Finishes a context reply.
728 *
729 * @returns nothing
730 * @param pDevIns The device instance.
731 * @param pThis Pointer to the shared LsiLogic device state.
732 * @param u32MessageContext The message context ID to post.
733 */
734static void lsilogicR3FinishContextReply(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, uint32_t u32MessageContext)
735{
736 int rc;
737
738 LogFlowFunc(("pThis=%#p u32MessageContext=%#x\n", pThis, u32MessageContext));
739
740 AssertMsg(pThis->enmDoorbellState == LSILOGICDOORBELLSTATE_NOT_IN_USE, ("We are in a doorbell function\n"));
741
742 /* Write message context ID into reply post queue. */
743 rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->ReplyPostQueueCritSect, VINF_SUCCESS);
744 AssertRC(rc);
745
746 /* Check for a entry in the queue. */
747 if (!lsilogicReplyPostQueueGetFrameCount(pThis))
748 {
749 /* Set error code. */
750 lsilogicSetIOCFaultCode(pThis, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
751 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyPostQueueCritSect);
752 return;
753 }
754
755 /* We have a context reply. */
756 ASMAtomicWriteU32(&pThis->aReplyPostQueue[pThis->uReplyPostQueueNextEntryFreeWrite], u32MessageContext);
757 ASMAtomicIncU32(&pThis->uReplyPostQueueNextEntryFreeWrite);
758 pThis->uReplyPostQueueNextEntryFreeWrite %= pThis->cReplyQueueEntries;
759
760 /* Set interrupt. */
761 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
762
763 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyPostQueueCritSect);
764}
765
766
767/**
768 * Takes necessary steps to finish a reply frame.
769 *
770 * @returns nothing
771 * @param pDevIns The device instance.
772 * @param pThis Pointer to the shared LsiLogic device state.
773 * @param pReply Pointer to the reply message.
774 * @param fForceReplyFifo Flag whether the use of the reply post fifo is forced.
775 */
776static void lsilogicFinishAddressReply(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PMptReplyUnion pReply, bool fForceReplyFifo)
777{
778 /*
779 * If we are in a doorbell function we set the reply size now and
780 * set the system doorbell status interrupt to notify the guest that
781 * we are ready to send the reply.
782 */
783 if (pThis->enmDoorbellState != LSILOGICDOORBELLSTATE_NOT_IN_USE && !fForceReplyFifo)
784 {
785 /* Set size of the reply in 16bit words. The size in the reply is in 32bit dwords. */
786 pThis->cReplySize = pReply->Header.u8MessageLength * 2;
787 Log(("%s: cReplySize=%u\n", __FUNCTION__, pThis->cReplySize));
788 pThis->uNextReplyEntryRead = 0;
789 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
790 }
791 else
792 {
793 /*
794 * The reply queues are only used if the request was fetched from the request queue.
795 * Requests from the request queue are always transferred to R3. So it is not possible
796 * that this case happens in R0 or GC.
797 */
798# ifdef IN_RING3
799 int rc;
800 /* Grab a free reply message from the queue. */
801 rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->ReplyFreeQueueCritSect, VINF_SUCCESS);
802 AssertRC(rc);
803
804 /* Check for a free reply frame. */
805 if (!lsilogicReplyFreeQueueGetFrameCount(pThis))
806 {
807 /* Set error code. */
808 lsilogicSetIOCFaultCode(pThis, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
809 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyFreeQueueCritSect);
810 return;
811 }
812
813 uint32_t u32ReplyFrameAddressLow = pThis->aReplyFreeQueue[pThis->uReplyFreeQueueNextAddressRead];
814
815 pThis->uReplyFreeQueueNextAddressRead++;
816 pThis->uReplyFreeQueueNextAddressRead %= pThis->cReplyQueueEntries;
817
818 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyFreeQueueCritSect);
819
820 /* Build 64bit physical address. */
821 RTGCPHYS GCPhysReplyMessage = LSILOGIC_RTGCPHYS_FROM_U32(pThis->u32HostMFAHighAddr, u32ReplyFrameAddressLow);
822 size_t cbReplyCopied = (pThis->cbReplyFrame < sizeof(MptReplyUnion)) ? pThis->cbReplyFrame : sizeof(MptReplyUnion);
823
824 /* Write reply to guest memory. */
825 PDMDevHlpPCIPhysWriteMeta(pDevIns, GCPhysReplyMessage, pReply, cbReplyCopied);
826
827 /* Write low 32bits of reply frame into post reply queue. */
828 rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->ReplyPostQueueCritSect, VINF_SUCCESS);
829 AssertRC(rc);
830
831 /* Check for a entry in the queue. */
832 if (!lsilogicReplyPostQueueGetFrameCount(pThis))
833 {
834 /* Set error code. */
835 lsilogicSetIOCFaultCode(pThis, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
836 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyPostQueueCritSect);
837 return;
838 }
839
840 /* We have a address reply. Set the 31th bit to indicate that. */
841 ASMAtomicWriteU32(&pThis->aReplyPostQueue[pThis->uReplyPostQueueNextEntryFreeWrite],
842 RT_BIT(31) | (u32ReplyFrameAddressLow >> 1));
843 ASMAtomicIncU32(&pThis->uReplyPostQueueNextEntryFreeWrite);
844 pThis->uReplyPostQueueNextEntryFreeWrite %= pThis->cReplyQueueEntries;
845
846 if (fForceReplyFifo)
847 {
848 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_NOT_IN_USE;
849 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
850 }
851
852 /* Set interrupt. */
853 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
854
855 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyPostQueueCritSect);
856# else
857 AssertMsgFailed(("This is not allowed to happen.\n"));
858# endif
859 }
860}
861
862
863/**
864 * Tries to find a memory region which covers the given address.
865 *
866 * @returns Pointer to memory region or NULL if not found.
867 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
868 * @param u32Addr The 32bit address to search for.
869 */
870static PLSILOGICMEMREGN lsilogicR3MemRegionFindByAddr(PLSILOGICSCSICC pThisCC, uint32_t u32Addr)
871{
872 PLSILOGICMEMREGN pRegion = NULL;
873
874 PLSILOGICMEMREGN pIt;
875 RTListForEach(&pThisCC->ListMemRegns, pIt, LSILOGICMEMREGN, NodeList)
876 {
877 if ( u32Addr >= pIt->u32AddrStart
878 && u32Addr <= pIt->u32AddrEnd)
879 {
880 pRegion = pIt;
881 break;
882 }
883 }
884
885 return pRegion;
886}
887
888/**
889 * Frees all allocated memory regions.
890 *
891 * @returns nothing.
892 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
893 */
894static void lsilogicR3MemRegionsFree(PLSILOGICSCSICC pThisCC)
895{
896 PLSILOGICMEMREGN pItNext;
897
898 PLSILOGICMEMREGN pIt;
899 RTListForEachSafe(&pThisCC->ListMemRegns, pIt, pItNext, LSILOGICMEMREGN, NodeList)
900 {
901 RTListNodeRemove(&pIt->NodeList);
902 RTMemFree(pIt);
903 }
904 pThisCC->cbMemRegns = 0;
905}
906
907/**
908 * Inserts a given memory region into the list.
909 *
910 * @returns nothing.
911 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
912 * @param pRegion The region to insert.
913 */
914static void lsilogicR3MemRegionInsert(PLSILOGICSCSICC pThisCC, PLSILOGICMEMREGN pRegion)
915{
916 bool fInserted = false;
917
918 /* Insert at the right position. */
919 PLSILOGICMEMREGN pIt;
920 RTListForEach(&pThisCC->ListMemRegns, pIt, LSILOGICMEMREGN, NodeList)
921 {
922 if (pRegion->u32AddrEnd < pIt->u32AddrStart)
923 {
924 RTListNodeInsertBefore(&pIt->NodeList, &pRegion->NodeList);
925 fInserted = true;
926 break;
927 }
928 }
929 if (!fInserted)
930 RTListAppend(&pThisCC->ListMemRegns, &pRegion->NodeList);
931}
932
933/**
934 * Count number of memory regions.
935 *
936 * @returns Number of memory regions.
937 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
938 */
939static uint32_t lsilogicR3MemRegionsCount(PLSILOGICSCSICC pThisCC)
940{
941 uint32_t cRegions = 0;
942
943 PLSILOGICMEMREGN pIt;
944 RTListForEach(&pThisCC->ListMemRegns, pIt, LSILOGICMEMREGN, NodeList)
945 {
946 cRegions++;
947 }
948
949 return cRegions;
950}
951
952/**
953 * Handles a write to the diagnostic data register.
954 *
955 * @returns nothing.
956 * @param pThis Pointer to the shared LsiLogic device state.
957 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
958 * @param u32Data Data to write.
959 */
960static void lsilogicR3DiagRegDataWrite(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC, uint32_t u32Data)
961{
962 RTCritSectEnter(&pThisCC->CritSectMemRegns);
963
964 PLSILOGICMEMREGN pRegion = lsilogicR3MemRegionFindByAddr(pThisCC, pThis->u32DiagMemAddr);
965 if (pRegion)
966 {
967 uint32_t offRegion = pThis->u32DiagMemAddr - pRegion->u32AddrStart;
968
969 AssertMsg( offRegion % 4 == 0
970 && pThis->u32DiagMemAddr <= pRegion->u32AddrEnd,
971 ("Region offset not on a word boundary or crosses memory region\n"));
972
973 offRegion /= 4;
974 pRegion->au32Data[offRegion] = u32Data;
975 }
976 else
977 {
978 pRegion = NULL;
979
980 /* Create new region, first check whether we can extend another region. */
981 PLSILOGICMEMREGN pIt;
982 RTListForEach(&pThisCC->ListMemRegns, pIt, LSILOGICMEMREGN, NodeList)
983 {
984 if (pThis->u32DiagMemAddr == pIt->u32AddrEnd + sizeof(uint32_t))
985 {
986 pRegion = pIt;
987 break;
988 }
989 }
990
991 if (pRegion)
992 {
993 /* Reallocate. */
994 RTListNodeRemove(&pRegion->NodeList);
995
996 uint32_t cRegionSizeOld = (pRegion->u32AddrEnd - pRegion->u32AddrStart) / 4 + 1;
997 uint32_t cRegionSizeNew = cRegionSizeOld + 512;
998
999 if (pThisCC->cbMemRegns + 512 * sizeof(uint32_t) < LSILOGIC_MEMORY_REGIONS_MAX)
1000 {
1001 PLSILOGICMEMREGN pRegionNew;
1002 pRegionNew = (PLSILOGICMEMREGN)RTMemRealloc(pRegion, RT_UOFFSETOF_DYN(LSILOGICMEMREGN, au32Data[cRegionSizeNew]));
1003 if (pRegionNew)
1004 {
1005 pRegion = pRegionNew;
1006 memset(&pRegion->au32Data[cRegionSizeOld], 0, 512 * sizeof(uint32_t));
1007 pRegion->au32Data[cRegionSizeOld] = u32Data;
1008 pRegion->u32AddrEnd = pRegion->u32AddrStart + (cRegionSizeNew - 1) * sizeof(uint32_t);
1009 pThisCC->cbMemRegns += 512 * sizeof(uint32_t);
1010 }
1011 /* else: Silently fail, there is nothing we can do here and the guest might work nevertheless. */
1012
1013 lsilogicR3MemRegionInsert(pThisCC, pRegion);
1014 }
1015 }
1016 else
1017 {
1018 if (pThisCC->cbMemRegns + 512 * sizeof(uint32_t) < LSILOGIC_MEMORY_REGIONS_MAX)
1019 {
1020 /* Create completely new. */
1021 pRegion = (PLSILOGICMEMREGN)RTMemAllocZ(RT_OFFSETOF(LSILOGICMEMREGN, au32Data[512]));
1022 if (pRegion)
1023 {
1024 pRegion->u32AddrStart = pThis->u32DiagMemAddr;
1025 pRegion->u32AddrEnd = pRegion->u32AddrStart + (512 - 1) * sizeof(uint32_t);
1026 pRegion->au32Data[0] = u32Data;
1027 pThisCC->cbMemRegns += 512 * sizeof(uint32_t);
1028
1029 lsilogicR3MemRegionInsert(pThisCC, pRegion);
1030 }
1031 /* else: Silently fail, there is nothing we can do here and the guest might work nevertheless. */
1032 }
1033 }
1034
1035 }
1036
1037 /* Memory access is always 32bit big. */
1038 pThis->u32DiagMemAddr += sizeof(uint32_t);
1039 RTCritSectLeave(&pThisCC->CritSectMemRegns);
1040}
1041
1042/**
1043 * Handles a read from the diagnostic data register.
1044 *
1045 * @returns nothing.
1046 * @param pThis Pointer to the shared LsiLogic device state.
1047 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
1048 * @param pu32Data Where to store the data.
1049 */
1050static void lsilogicR3DiagRegDataRead(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC, uint32_t *pu32Data)
1051{
1052 RTCritSectEnter(&pThisCC->CritSectMemRegns);
1053
1054 PLSILOGICMEMREGN pRegion = lsilogicR3MemRegionFindByAddr(pThisCC, pThis->u32DiagMemAddr);
1055 if (pRegion)
1056 {
1057 uint32_t offRegion = pThis->u32DiagMemAddr - pRegion->u32AddrStart;
1058
1059 AssertMsg( offRegion % 4 == 0
1060 && pThis->u32DiagMemAddr <= pRegion->u32AddrEnd,
1061 ("Region offset not on a word boundary or crosses memory region\n"));
1062
1063 offRegion /= 4;
1064 *pu32Data = pRegion->au32Data[offRegion];
1065 }
1066 else /* No region, default value 0. */
1067 *pu32Data = 0;
1068
1069 /* Memory access is always 32bit big. */
1070 pThis->u32DiagMemAddr += sizeof(uint32_t);
1071 RTCritSectLeave(&pThisCC->CritSectMemRegns);
1072}
1073
1074/**
1075 * Handles a write to the diagnostic memory address register.
1076 *
1077 * @returns nothing.
1078 * @param pThis Pointer to the shared LsiLogic device state.
1079 * @param u32Addr Address to write.
1080 */
1081static void lsilogicR3DiagRegAddressWrite(PLSILOGICSCSI pThis, uint32_t u32Addr)
1082{
1083 pThis->u32DiagMemAddr = u32Addr & ~UINT32_C(0x3); /* 32bit alignment. */
1084}
1085
1086/**
1087 * Handles a read from the diagnostic memory address register.
1088 *
1089 * @returns nothing.
1090 * @param pThis Pointer to the shared LsiLogic device state.
1091 * @param pu32Addr Where to store the current address.
1092 */
1093static void lsilogicR3DiagRegAddressRead(PLSILOGICSCSI pThis, uint32_t *pu32Addr)
1094{
1095 *pu32Addr = pThis->u32DiagMemAddr;
1096}
1097
1098/**
1099 * Processes a given Request from the guest
1100 *
1101 * @returns VBox status code.
1102 * @param pDevIns The device instance.
1103 * @param pThis Pointer to the shared LsiLogic device state.
1104 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
1105 * @param pMessageHdr Pointer to the message header of the request.
1106 * @param pReply Pointer to the reply.
1107 */
1108static int lsilogicR3ProcessMessageRequest(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC,
1109 PMptMessageHdr pMessageHdr, PMptReplyUnion pReply)
1110{
1111 int rc = VINF_SUCCESS;
1112 bool fForceReplyPostFifo = false;
1113
1114# ifdef LOG_ENABLED
1115 if (pMessageHdr->u8Function < RT_ELEMENTS(g_apszMPTFunctionNames))
1116 Log(("Message request function: %s\n", g_apszMPTFunctionNames[pMessageHdr->u8Function]));
1117 else
1118 Log(("Message request function: <unknown>\n"));
1119# endif
1120
1121 memset(pReply, 0, sizeof(MptReplyUnion));
1122
1123 switch (pMessageHdr->u8Function)
1124 {
1125 case MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT:
1126 {
1127 PMptSCSITaskManagementRequest pTaskMgmtReq = (PMptSCSITaskManagementRequest)pMessageHdr;
1128
1129 LogFlow(("u8TaskType=%u\n", pTaskMgmtReq->u8TaskType));
1130 LogFlow(("u32TaskMessageContext=%#x\n", pTaskMgmtReq->u32TaskMessageContext));
1131
1132 pReply->SCSITaskManagement.u8MessageLength = 6; /* 6 32bit dwords. */
1133 pReply->SCSITaskManagement.u8TaskType = pTaskMgmtReq->u8TaskType;
1134 pReply->SCSITaskManagement.u32TerminationCount = 0;
1135 fForceReplyPostFifo = true;
1136 break;
1137 }
1138 case MPT_MESSAGE_HDR_FUNCTION_IOC_INIT:
1139 {
1140 /*
1141 * This request sets the I/O controller to the
1142 * operational state.
1143 */
1144 PMptIOCInitRequest pIOCInitReq = (PMptIOCInitRequest)pMessageHdr;
1145
1146 /* Update configuration values. */
1147 pThis->enmWhoInit = (LSILOGICWHOINIT)pIOCInitReq->u8WhoInit;
1148 pThis->cbReplyFrame = pIOCInitReq->u16ReplyFrameSize;
1149 pThis->cMaxBuses = pIOCInitReq->u8MaxBuses;
1150 pThis->cMaxDevices = pIOCInitReq->u8MaxDevices;
1151 pThis->u32HostMFAHighAddr = pIOCInitReq->u32HostMfaHighAddr;
1152 pThis->u32SenseBufferHighAddr = pIOCInitReq->u32SenseBufferHighAddr;
1153
1154 if (pThis->enmState == LSILOGICSTATE_READY)
1155 {
1156 pThis->enmState = LSILOGICSTATE_OPERATIONAL;
1157 }
1158
1159 /* Return reply. */
1160 pReply->IOCInit.u8MessageLength = 5;
1161 pReply->IOCInit.u8WhoInit = pThis->enmWhoInit;
1162 pReply->IOCInit.u8MaxDevices = pThis->cMaxDevices;
1163 pReply->IOCInit.u8MaxBuses = pThis->cMaxBuses;
1164 break;
1165 }
1166 case MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS:
1167 {
1168 pReply->IOCFacts.u8MessageLength = 15; /* 15 32bit dwords. */
1169
1170 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
1171 {
1172 pReply->IOCFacts.u16MessageVersion = 0x0102; /* Version from the specification. */
1173 pReply->IOCFacts.u8NumberOfPorts = pThis->cPorts;
1174 }
1175 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
1176 {
1177 pReply->IOCFacts.u16MessageVersion = 0x0105; /* Version from the specification. */
1178 pReply->IOCFacts.u8NumberOfPorts = pThis->cPorts;
1179 }
1180 else
1181 AssertMsgFailed(("Invalid controller type %d\n", pThis->enmCtrlType));
1182
1183 pReply->IOCFacts.u8IOCNumber = 0; /* PCI function number. */
1184 pReply->IOCFacts.u16IOCExceptions = 0;
1185 pReply->IOCFacts.u8MaxChainDepth = LSILOGICSCSI_MAXIMUM_CHAIN_DEPTH;
1186 pReply->IOCFacts.u8WhoInit = pThis->enmWhoInit;
1187 pReply->IOCFacts.u8BlockSize = 12; /* Block size in 32bit dwords. This is the largest request we can get (SCSI I/O). */
1188 pReply->IOCFacts.u8Flags = 0; /* Bit 0 is set if the guest must upload the FW prior to using the controller. Obviously not needed here. */
1189 pReply->IOCFacts.u16ReplyQueueDepth = pThis->cReplyQueueEntries - 1; /* One entry is always free. */
1190 pReply->IOCFacts.u16RequestFrameSize = 128; /** @todo Figure out where it is needed. */
1191 pReply->IOCFacts.u32CurrentHostMFAHighAddr = pThis->u32HostMFAHighAddr;
1192 pReply->IOCFacts.u16GlobalCredits = pThis->cRequestQueueEntries - 1; /* One entry is always free. */
1193
1194 pReply->IOCFacts.u8EventState = 0; /* Event notifications not enabled. */
1195 pReply->IOCFacts.u32CurrentSenseBufferHighAddr = pThis->u32SenseBufferHighAddr;
1196 pReply->IOCFacts.u16CurReplyFrameSize = pThis->cbReplyFrame;
1197 pReply->IOCFacts.u8MaxDevices = pThis->cMaxDevices;
1198 pReply->IOCFacts.u8MaxBuses = pThis->cMaxBuses;
1199
1200 pReply->IOCFacts.u16ProductID = 0xcafe; /* Our own product ID :) */
1201 pReply->IOCFacts.u32FwImageSize = 0; /* No image needed. */
1202 pReply->IOCFacts.u32FWVersion = 0;
1203
1204 /* Check for a valid firmware image in the IOC memory which was downloaded by the guest earlier and use that. */
1205 RTCritSectEnter(&pThisCC->CritSectMemRegns);
1206 PLSILOGICMEMREGN pRegion = lsilogicR3MemRegionFindByAddr(pThisCC, LSILOGIC_FWIMGHDR_LOAD_ADDRESS);
1207 if (pRegion)
1208 {
1209 uint32_t offImgHdr = (LSILOGIC_FWIMGHDR_LOAD_ADDRESS - pRegion->u32AddrStart);
1210 if (pRegion->u32AddrEnd - offImgHdr + 1 >= sizeof(FwImageHdr)) /* End address is inclusive. */
1211 {
1212 PFwImageHdr pFwImgHdr = (PFwImageHdr)&pRegion->au32Data[offImgHdr / 4];
1213
1214 /* Check for the signature. */
1215 /** @todo Checksum validation. */
1216 if ( pFwImgHdr->u32Signature1 == LSILOGIC_FWIMGHDR_SIGNATURE1
1217 && pFwImgHdr->u32Signature2 == LSILOGIC_FWIMGHDR_SIGNATURE2
1218 && pFwImgHdr->u32Signature3 == LSILOGIC_FWIMGHDR_SIGNATURE3)
1219 {
1220 LogFlowFunc(("IOC Facts: Found valid firmware image header in memory, using version (%#x), size (%d) and product ID (%#x) from there\n",
1221 pFwImgHdr->u32FwVersion, pFwImgHdr->u32ImageSize, pFwImgHdr->u16ProductId));
1222
1223 pReply->IOCFacts.u16ProductID = pFwImgHdr->u16ProductId;
1224 pReply->IOCFacts.u32FwImageSize = pFwImgHdr->u32ImageSize;
1225 pReply->IOCFacts.u32FWVersion = pFwImgHdr->u32FwVersion;
1226 }
1227 }
1228 }
1229 RTCritSectLeave(&pThisCC->CritSectMemRegns);
1230 break;
1231 }
1232 case MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS:
1233 {
1234 PMptPortFactsRequest pPortFactsReq = (PMptPortFactsRequest)pMessageHdr;
1235
1236 pReply->PortFacts.u8MessageLength = 10;
1237 pReply->PortFacts.u8PortNumber = pPortFactsReq->u8PortNumber;
1238
1239 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
1240 {
1241 /* This controller only supports one bus with bus number 0. */
1242 if (pPortFactsReq->u8PortNumber >= pThis->cPorts)
1243 {
1244 pReply->PortFacts.u8PortType = 0; /* Not existant. */
1245 }
1246 else
1247 {
1248 pReply->PortFacts.u8PortType = 0x01; /* SCSI Port. */
1249 pReply->PortFacts.u16MaxDevices = LSILOGICSCSI_PCI_SPI_DEVICES_PER_BUS_MAX;
1250 pReply->PortFacts.u16ProtocolFlags = RT_BIT(3) | RT_BIT(0); /* SCSI initiator and LUN supported. */
1251 pReply->PortFacts.u16PortSCSIID = 7; /* Default */
1252 pReply->PortFacts.u16MaxPersistentIDs = 0;
1253 pReply->PortFacts.u16MaxPostedCmdBuffers = 0; /* Only applies for target mode which we dont support. */
1254 pReply->PortFacts.u16MaxLANBuckets = 0; /* Only for the LAN controller. */
1255 }
1256 }
1257 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
1258 {
1259 if (pPortFactsReq->u8PortNumber >= pThis->cPorts)
1260 {
1261 pReply->PortFacts.u8PortType = 0; /* Not existant. */
1262 }
1263 else
1264 {
1265 pReply->PortFacts.u8PortType = 0x30; /* SAS Port. */
1266 pReply->PortFacts.u16MaxDevices = pThis->cPorts;
1267 pReply->PortFacts.u16ProtocolFlags = RT_BIT(3) | RT_BIT(0); /* SCSI initiator and LUN supported. */
1268 pReply->PortFacts.u16PortSCSIID = pThis->cPorts;
1269 pReply->PortFacts.u16MaxPersistentIDs = 0;
1270 pReply->PortFacts.u16MaxPostedCmdBuffers = 0; /* Only applies for target mode which we dont support. */
1271 pReply->PortFacts.u16MaxLANBuckets = 0; /* Only for the LAN controller. */
1272 }
1273 }
1274 else
1275 AssertMsgFailed(("Invalid controller type %d\n", pThis->enmCtrlType));
1276 break;
1277 }
1278 case MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE:
1279 {
1280 /*
1281 * The port enable request notifies the IOC to make the port available and perform
1282 * appropriate discovery on the associated link.
1283 */
1284 PMptPortEnableRequest pPortEnableReq = (PMptPortEnableRequest)pMessageHdr;
1285
1286 pReply->PortEnable.u8MessageLength = 5;
1287 pReply->PortEnable.u8PortNumber = pPortEnableReq->u8PortNumber;
1288 break;
1289 }
1290 case MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION:
1291 {
1292 PMptEventNotificationRequest pEventNotificationReq = (PMptEventNotificationRequest)pMessageHdr;
1293
1294 if (pEventNotificationReq->u8Switch)
1295 pThis->fEventNotificationEnabled = true;
1296 else
1297 pThis->fEventNotificationEnabled = false;
1298
1299 pReply->EventNotification.u16EventDataLength = 1; /* 1 32bit D-Word. */
1300 pReply->EventNotification.u8MessageLength = 8;
1301 pReply->EventNotification.u8MessageFlags = (1 << 7);
1302 pReply->EventNotification.u8AckRequired = 0;
1303 pReply->EventNotification.u32Event = MPT_EVENT_EVENT_CHANGE;
1304 pReply->EventNotification.u32EventContext = 0;
1305 pReply->EventNotification.u32EventData = pThis->fEventNotificationEnabled ? 1 : 0;
1306
1307 break;
1308 }
1309 case MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK:
1310 {
1311 AssertMsgFailed(("todo"));
1312 break;
1313 }
1314 case MPT_MESSAGE_HDR_FUNCTION_CONFIG:
1315 {
1316 PMptConfigurationRequest pConfigurationReq = (PMptConfigurationRequest)pMessageHdr;
1317
1318 rc = lsilogicR3ProcessConfigurationRequest(pDevIns, pThis, pThisCC, pConfigurationReq, &pReply->Configuration);
1319 AssertRC(rc);
1320 break;
1321 }
1322 case MPT_MESSAGE_HDR_FUNCTION_FW_UPLOAD:
1323 {
1324 PMptFWUploadRequest pFWUploadReq = (PMptFWUploadRequest)pMessageHdr;
1325
1326 pReply->FWUpload.u8ImageType = pFWUploadReq->u8ImageType;
1327 pReply->FWUpload.u8MessageLength = 6;
1328 pReply->FWUpload.u32ActualImageSize = 0;
1329 break;
1330 }
1331 case MPT_MESSAGE_HDR_FUNCTION_FW_DOWNLOAD:
1332 {
1333 //PMptFWDownloadRequest pFWDownloadReq = (PMptFWDownloadRequest)pMessageHdr;
1334
1335 pReply->FWDownload.u8MessageLength = 5;
1336 LogFlowFunc(("FW Download request issued\n"));
1337 break;
1338 }
1339 case MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST: /* Should be handled already. */
1340 default:
1341 AssertMsgFailed(("Invalid request function %#x\n", pMessageHdr->u8Function));
1342 }
1343
1344 /* Copy common bits from request message frame to reply. */
1345 pReply->Header.u8Function = pMessageHdr->u8Function;
1346 pReply->Header.u32MessageContext = pMessageHdr->u32MessageContext;
1347
1348 lsilogicFinishAddressReply(pDevIns, pThis, pReply, fForceReplyPostFifo);
1349 return rc;
1350}
1351
1352#endif /* IN_RING3 */
1353
1354/**
1355 * Writes a value to a register at a given offset.
1356 *
1357 * @returns Strict VBox status code.
1358 * @param pDevIns The devie instance.
1359 * @param pThis Pointer to the shared LsiLogic device state.
1360 * @param offReg Offset of the register to write.
1361 * @param u32 The value being written.
1362 */
1363static VBOXSTRICTRC lsilogicRegisterWrite(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, uint32_t offReg, uint32_t u32)
1364{
1365 LogFlowFunc(("pThis=%#p offReg=%#x u32=%#x\n", pThis, offReg, u32));
1366 switch (offReg)
1367 {
1368 case LSILOGIC_REG_REPLY_QUEUE:
1369 {
1370 int rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->ReplyFreeQueueWriteCritSect, VINF_IOM_R3_MMIO_WRITE);
1371 if (rc != VINF_SUCCESS)
1372 return rc;
1373 /* Add the entry to the reply free queue. */
1374 ASMAtomicWriteU32(&pThis->aReplyFreeQueue[pThis->uReplyFreeQueueNextEntryFreeWrite], u32);
1375 pThis->uReplyFreeQueueNextEntryFreeWrite++;
1376 pThis->uReplyFreeQueueNextEntryFreeWrite %= pThis->cReplyQueueEntries;
1377 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyFreeQueueWriteCritSect);
1378 break;
1379 }
1380 case LSILOGIC_REG_REQUEST_QUEUE:
1381 {
1382 int rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->RequestQueueCritSect, VINF_IOM_R3_MMIO_WRITE);
1383 if (rc != VINF_SUCCESS)
1384 return rc;
1385
1386 uint32_t uNextWrite = ASMAtomicReadU32(&pThis->uRequestQueueNextEntryFreeWrite);
1387
1388 ASMAtomicWriteU32(&pThis->aRequestQueue[uNextWrite], u32);
1389
1390 /*
1391 * Don't update the value in place. It can happen that we get preempted
1392 * after the increment but before the modulo.
1393 * Another EMT will read the wrong value when processing the queues
1394 * and hang in an endless loop creating thousands of requests.
1395 */
1396 uNextWrite++;
1397 uNextWrite %= pThis->cRequestQueueEntries;
1398 ASMAtomicWriteU32(&pThis->uRequestQueueNextEntryFreeWrite, uNextWrite);
1399 PDMDevHlpCritSectLeave(pDevIns, &pThis->RequestQueueCritSect);
1400
1401 /* Send notification to R3 if there is not one sent already. Do this
1402 * only if the worker thread is not sleeping or might go sleeping. */
1403 if (!ASMAtomicXchgBool(&pThis->fNotificationSent, true))
1404 {
1405 if (ASMAtomicReadBool(&pThis->fWrkThreadSleeping))
1406 {
1407 LogFlowFunc(("Signal event semaphore\n"));
1408 rc = PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEvtProcess);
1409 AssertRC(rc);
1410 }
1411 }
1412 break;
1413 }
1414 case LSILOGIC_REG_DOORBELL:
1415 {
1416 /*
1417 * When the guest writes to this register a real device would set the
1418 * doorbell status bit in the interrupt status register to indicate that the IOP
1419 * has still to process the message.
1420 * The guest needs to wait with posting new messages here until the bit is cleared.
1421 * Because the guest is not continuing execution while we are here we can skip this.
1422 */
1423 if (pThis->enmDoorbellState == LSILOGICDOORBELLSTATE_NOT_IN_USE)
1424 {
1425 uint32_t uFunction = LSILOGIC_REG_DOORBELL_GET_FUNCTION(u32);
1426
1427 switch (uFunction)
1428 {
1429 case LSILOGIC_DOORBELL_FUNCTION_IO_UNIT_RESET:
1430 case LSILOGIC_DOORBELL_FUNCTION_IOC_MSG_UNIT_RESET:
1431 {
1432 /*
1433 * The I/O unit reset does much more on real hardware like
1434 * reloading the firmware, nothing we need to do here,
1435 * so this is like the IOC message unit reset.
1436 */
1437 pThis->enmState = LSILOGICSTATE_RESET;
1438
1439 /* Reset interrupt status. */
1440 pThis->uInterruptStatus = 0;
1441 lsilogicUpdateInterrupt(pDevIns, pThis);
1442
1443 /* Reset the queues. */
1444 pThis->uReplyFreeQueueNextEntryFreeWrite = 0;
1445 pThis->uReplyFreeQueueNextAddressRead = 0;
1446 pThis->uReplyPostQueueNextEntryFreeWrite = 0;
1447 pThis->uReplyPostQueueNextAddressRead = 0;
1448 pThis->uRequestQueueNextEntryFreeWrite = 0;
1449 pThis->uRequestQueueNextAddressRead = 0;
1450
1451 /* Only the IOC message unit reset transisionts to the ready state. */
1452 if (uFunction == LSILOGIC_DOORBELL_FUNCTION_IOC_MSG_UNIT_RESET)
1453 pThis->enmState = LSILOGICSTATE_READY;
1454 break;
1455 }
1456 case LSILOGIC_DOORBELL_FUNCTION_HANDSHAKE:
1457 {
1458 pThis->cMessage = LSILOGIC_REG_DOORBELL_GET_SIZE(u32);
1459 pThis->iMessage = 0;
1460
1461 /* This is not supposed to happen and the result is undefined, just stay in the current state. */
1462 AssertMsgReturn(pThis->cMessage <= RT_ELEMENTS(pThis->aMessage),
1463 ("Message doesn't fit into the buffer, cMessage=%u", pThis->cMessage),
1464 VINF_SUCCESS);
1465
1466 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_FN_HANDSHAKE;
1467 /* Update the interrupt status to notify the guest that a doorbell function was started. */
1468 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1469 break;
1470 }
1471 case LSILOGIC_DOORBELL_FUNCTION_REPLY_FRAME_REMOVAL:
1472 {
1473 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_RFR_FRAME_COUNT_LOW;
1474 /* Update the interrupt status to notify the guest that a doorbell function was started. */
1475 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1476 break;
1477 }
1478 default:
1479 AssertMsgFailed(("Unknown function %u to perform\n", uFunction));
1480 }
1481 }
1482 else if (pThis->enmDoorbellState == LSILOGICDOORBELLSTATE_FN_HANDSHAKE)
1483 {
1484 /*
1485 * We are already performing a doorbell function.
1486 * Get the remaining parameters, ignore any excess writes.
1487 */
1488 AssertMsgReturn(pThis->iMessage < pThis->cMessage,
1489 ("Guest is trying to write more than was indicated in the handshake\n"),
1490 VINF_SUCCESS);
1491
1492 /*
1493 * If the last byte of the message is written, force a switch to R3 because some requests might force
1494 * a reply through the FIFO which cannot be handled in GC or R0.
1495 */
1496#ifndef IN_RING3
1497 if (pThis->iMessage == pThis->cMessage - 1)
1498 return VINF_IOM_R3_MMIO_WRITE;
1499#endif
1500 pThis->aMessage[pThis->iMessage++] = u32;
1501#ifdef IN_RING3
1502 if (pThis->iMessage == pThis->cMessage)
1503 {
1504 int rc = lsilogicR3ProcessMessageRequest(pDevIns, pThis, PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC),
1505 (PMptMessageHdr)pThis->aMessage, &pThis->ReplyBuffer);
1506 AssertRC(rc);
1507 }
1508#endif
1509 }
1510 break;
1511 }
1512 case LSILOGIC_REG_HOST_INTR_STATUS:
1513 {
1514 /*
1515 * Clear the bits the guest wants except the system doorbell interrupt and the IO controller
1516 * status bit.
1517 * The former bit is always cleared no matter what the guest writes to the register and
1518 * the latter one is read only.
1519 */
1520 ASMAtomicAndU32(&pThis->uInterruptStatus, ~LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1521
1522 /*
1523 * Check if there is still a doorbell function in progress. Set the
1524 * system doorbell interrupt bit again if it is.
1525 * We do not use lsilogicSetInterrupt here because the interrupt status
1526 * is updated afterwards anyway.
1527 */
1528 if ( (pThis->enmDoorbellState == LSILOGICDOORBELLSTATE_FN_HANDSHAKE)
1529 && (pThis->cMessage == pThis->iMessage))
1530 {
1531 if (pThis->uNextReplyEntryRead == pThis->cReplySize)
1532 {
1533 /* Reply finished. Reset doorbell in progress status. */
1534 Log(("%s: Doorbell function finished\n", __FUNCTION__));
1535 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_NOT_IN_USE;
1536 }
1537 ASMAtomicOrU32(&pThis->uInterruptStatus, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1538 }
1539 else if ( pThis->enmDoorbellState != LSILOGICDOORBELLSTATE_NOT_IN_USE
1540 && pThis->enmDoorbellState != LSILOGICDOORBELLSTATE_FN_HANDSHAKE)
1541 {
1542 /* Reply frame removal, check whether the reply free queue is empty. */
1543 if ( pThis->uReplyFreeQueueNextAddressRead == pThis->uReplyFreeQueueNextEntryFreeWrite
1544 && pThis->enmDoorbellState == LSILOGICDOORBELLSTATE_RFR_NEXT_FRAME_LOW)
1545 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_NOT_IN_USE;
1546 ASMAtomicOrU32(&pThis->uInterruptStatus, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1547 }
1548
1549 lsilogicUpdateInterrupt(pDevIns, pThis);
1550 break;
1551 }
1552 case LSILOGIC_REG_HOST_INTR_MASK:
1553 {
1554 ASMAtomicWriteU32(&pThis->uInterruptMask, u32 & LSILOGIC_REG_HOST_INTR_MASK_W_MASK);
1555 lsilogicUpdateInterrupt(pDevIns, pThis);
1556 break;
1557 }
1558 case LSILOGIC_REG_WRITE_SEQUENCE:
1559 {
1560 if (pThis->fDiagnosticEnabled)
1561 {
1562 /* Any value will cause a reset and disabling access. */
1563 pThis->fDiagnosticEnabled = false;
1564 pThis->iDiagnosticAccess = 0;
1565 pThis->fDiagRegsEnabled = false;
1566 }
1567 else if ((u32 & 0xf) == g_lsilogicDiagnosticAccess[pThis->iDiagnosticAccess])
1568 {
1569 pThis->iDiagnosticAccess++;
1570 if (pThis->iDiagnosticAccess == RT_ELEMENTS(g_lsilogicDiagnosticAccess))
1571 {
1572 /*
1573 * Key sequence successfully written. Enable access to diagnostic
1574 * memory and register.
1575 */
1576 pThis->fDiagnosticEnabled = true;
1577 }
1578 }
1579 else
1580 {
1581 /* Wrong value written - reset to beginning. */
1582 pThis->iDiagnosticAccess = 0;
1583 }
1584 break;
1585 }
1586 case LSILOGIC_REG_HOST_DIAGNOSTIC:
1587 {
1588 if (pThis->fDiagnosticEnabled)
1589 {
1590#ifndef IN_RING3
1591 return VINF_IOM_R3_MMIO_WRITE;
1592#else
1593 if (u32 & LSILOGIC_REG_HOST_DIAGNOSTIC_RESET_ADAPTER)
1594 lsilogicR3HardReset(pDevIns, pThis, PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC));
1595 else if (u32 & LSILOGIC_REG_HOST_DIAGNOSTIC_DIAG_RW_ENABLE)
1596 pThis->fDiagRegsEnabled = true;
1597#endif
1598 }
1599 break;
1600 }
1601 case LSILOGIC_REG_DIAG_RW_DATA:
1602 {
1603 if (pThis->fDiagRegsEnabled)
1604 {
1605#ifndef IN_RING3
1606 return VINF_IOM_R3_MMIO_WRITE;
1607#else
1608 lsilogicR3DiagRegDataWrite(pThis, PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC), u32);
1609#endif
1610 }
1611 break;
1612 }
1613 case LSILOGIC_REG_DIAG_RW_ADDRESS:
1614 {
1615 if (pThis->fDiagRegsEnabled)
1616 {
1617#ifndef IN_RING3
1618 return VINF_IOM_R3_MMIO_WRITE;
1619#else
1620 lsilogicR3DiagRegAddressWrite(pThis, u32);
1621#endif
1622 }
1623 break;
1624 }
1625 default: /* Ignore. */
1626 {
1627 break;
1628 }
1629 }
1630 return VINF_SUCCESS;
1631}
1632
1633/**
1634 * Reads the content of a register at a given offset.
1635 *
1636 * @returns VBox status code.
1637 * @param pDevIns The device instance.
1638 * @param pThis Pointer to the shared LsiLogic device state.
1639 * @param offReg Offset of the register to read.
1640 * @param pu32 Where to store the content of the register.
1641 */
1642static VBOXSTRICTRC lsilogicRegisterRead(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, uint32_t offReg, uint32_t *pu32)
1643{
1644 int rc = VINF_SUCCESS;
1645 uint32_t u32 = 0;
1646 Assert(!(offReg & 3));
1647
1648 /* Align to a 4 byte offset. */
1649 switch (offReg)
1650 {
1651 case LSILOGIC_REG_REPLY_QUEUE:
1652 {
1653 rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->ReplyPostQueueCritSect, VINF_IOM_R3_MMIO_READ);
1654 if (rc != VINF_SUCCESS)
1655 break;
1656
1657 uint32_t idxReplyPostQueueWrite = ASMAtomicUoReadU32(&pThis->uReplyPostQueueNextEntryFreeWrite);
1658 uint32_t idxReplyPostQueueRead = ASMAtomicUoReadU32(&pThis->uReplyPostQueueNextAddressRead);
1659
1660 if (idxReplyPostQueueWrite != idxReplyPostQueueRead)
1661 {
1662 u32 = pThis->aReplyPostQueue[idxReplyPostQueueRead];
1663 idxReplyPostQueueRead++;
1664 idxReplyPostQueueRead %= pThis->cReplyQueueEntries;
1665 ASMAtomicWriteU32(&pThis->uReplyPostQueueNextAddressRead, idxReplyPostQueueRead);
1666 }
1667 else
1668 {
1669 /* The reply post queue is empty. Reset interrupt. */
1670 u32 = UINT32_C(0xffffffff);
1671 lsilogicClearInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
1672 }
1673 PDMDevHlpCritSectLeave(pDevIns, &pThis->ReplyPostQueueCritSect);
1674
1675 Log(("%s: Returning address %#x\n", __FUNCTION__, u32));
1676 break;
1677 }
1678 case LSILOGIC_REG_DOORBELL:
1679 {
1680 u32 = LSILOGIC_REG_DOORBELL_SET_STATE(pThis->enmState);
1681 u32 |= LSILOGIC_REG_DOORBELL_SET_USED(pThis->enmDoorbellState);
1682 u32 |= LSILOGIC_REG_DOORBELL_SET_WHOINIT(pThis->enmWhoInit);
1683 /*
1684 * If there is a doorbell function in progress we pass the return value
1685 * instead of the status code. We transfer 16bit of the reply
1686 * during one read.
1687 */
1688 switch (pThis->enmDoorbellState)
1689 {
1690 case LSILOGICDOORBELLSTATE_NOT_IN_USE:
1691 /* We return the status code of the I/O controller. */
1692 u32 |= pThis->u16IOCFaultCode;
1693 break;
1694 case LSILOGICDOORBELLSTATE_FN_HANDSHAKE:
1695 /* Return next 16bit value. */
1696 if (pThis->uNextReplyEntryRead < pThis->cReplySize)
1697 u32 |= pThis->ReplyBuffer.au16Reply[pThis->uNextReplyEntryRead++];
1698 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1699 break;
1700 case LSILOGICDOORBELLSTATE_RFR_FRAME_COUNT_LOW:
1701 {
1702 uint32_t cReplyFrames = lsilogicReplyFreeQueueGetFrameCount(pThis);
1703
1704 u32 |= cReplyFrames & UINT32_C(0xffff);
1705 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_RFR_FRAME_COUNT_HIGH;
1706 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1707 break;
1708 }
1709 case LSILOGICDOORBELLSTATE_RFR_FRAME_COUNT_HIGH:
1710 {
1711 uint32_t cReplyFrames = lsilogicReplyFreeQueueGetFrameCount(pThis);
1712
1713 u32 |= cReplyFrames >> 16;
1714 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_RFR_NEXT_FRAME_LOW;
1715 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1716 break;
1717 }
1718 case LSILOGICDOORBELLSTATE_RFR_NEXT_FRAME_LOW:
1719 if (pThis->uReplyFreeQueueNextEntryFreeWrite != pThis->uReplyFreeQueueNextAddressRead)
1720 {
1721 u32 |= pThis->aReplyFreeQueue[pThis->uReplyFreeQueueNextAddressRead] & UINT32_C(0xffff);
1722 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_RFR_NEXT_FRAME_HIGH;
1723 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1724 }
1725 break;
1726 case LSILOGICDOORBELLSTATE_RFR_NEXT_FRAME_HIGH:
1727 u32 |= pThis->aReplyFreeQueue[pThis->uReplyFreeQueueNextAddressRead] >> 16;
1728 pThis->uReplyFreeQueueNextAddressRead++;
1729 pThis->uReplyFreeQueueNextAddressRead %= pThis->cReplyQueueEntries;
1730 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_RFR_NEXT_FRAME_LOW;
1731 lsilogicSetInterrupt(pDevIns, pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1732 break;
1733 default:
1734 AssertMsgFailed(("Invalid doorbell state %d\n", pThis->enmDoorbellState));
1735 }
1736
1737 break;
1738 }
1739 case LSILOGIC_REG_HOST_INTR_STATUS:
1740 {
1741 u32 = ASMAtomicReadU32(&pThis->uInterruptStatus);
1742 break;
1743 }
1744 case LSILOGIC_REG_HOST_INTR_MASK:
1745 {
1746 u32 = ASMAtomicReadU32(&pThis->uInterruptMask);
1747 break;
1748 }
1749 case LSILOGIC_REG_HOST_DIAGNOSTIC:
1750 {
1751 if (pThis->fDiagnosticEnabled)
1752 u32 |= LSILOGIC_REG_HOST_DIAGNOSTIC_DRWE;
1753 if (pThis->fDiagRegsEnabled)
1754 u32 |= LSILOGIC_REG_HOST_DIAGNOSTIC_DIAG_RW_ENABLE;
1755 break;
1756 }
1757 case LSILOGIC_REG_DIAG_RW_DATA:
1758 {
1759 if (pThis->fDiagRegsEnabled)
1760 {
1761#ifndef IN_RING3
1762 return VINF_IOM_R3_MMIO_READ;
1763#else
1764 lsilogicR3DiagRegDataRead(pThis, PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC), &u32);
1765#endif
1766 }
1767 }
1768 RT_FALL_THRU();
1769 case LSILOGIC_REG_DIAG_RW_ADDRESS:
1770 {
1771 if (pThis->fDiagRegsEnabled)
1772 {
1773#ifndef IN_RING3
1774 return VINF_IOM_R3_MMIO_READ;
1775#else
1776 lsilogicR3DiagRegAddressRead(pThis, &u32);
1777#endif
1778 }
1779 }
1780 RT_FALL_THRU();
1781 case LSILOGIC_REG_TEST_BASE_ADDRESS: /* The spec doesn't say anything about these registers, so we just ignore them */
1782 default: /* Ignore. */
1783 {
1784 /** @todo LSILOGIC_REG_DIAG_* should return all F's when accessed by MMIO. We
1785 * return 0. Likely to apply to undefined offsets as well. */
1786 break;
1787 }
1788 }
1789
1790 *pu32 = u32;
1791 LogFlowFunc(("pThis=%#p offReg=%#x u32=%#x\n", pThis, offReg, u32));
1792 return rc;
1793}
1794
1795/**
1796 * @callback_method_impl{FNIOMIOPORTNEWOUT}
1797 */
1798static DECLCALLBACK(VBOXSTRICTRC)
1799lsilogicIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb)
1800{
1801 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
1802 VBOXSTRICTRC rcStrict;
1803 RT_NOREF2(pvUser, cb);
1804
1805 if (!(offPort & 3))
1806 {
1807 rcStrict = lsilogicRegisterWrite(pDevIns, pThis, offPort, u32);
1808 if (rcStrict == VINF_IOM_R3_MMIO_WRITE)
1809 rcStrict = VINF_IOM_R3_IOPORT_WRITE;
1810 }
1811 else
1812 {
1813 Log(("lsilogicIOPortWrite: Ignoring misaligned write - offPort=%#x u32=%#x cb=%#x\n", offPort, u32, cb));
1814 rcStrict = VINF_SUCCESS;
1815 }
1816
1817 return rcStrict;
1818}
1819
1820/**
1821 * @callback_method_impl{FNIOMIOPORTNEWIN}
1822 */
1823static DECLCALLBACK(VBOXSTRICTRC)
1824lsilogicIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb)
1825{
1826 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
1827 RT_NOREF_PV(pvUser);
1828 RT_NOREF_PV(cb);
1829
1830 VBOXSTRICTRC rcStrict = lsilogicRegisterRead(pDevIns, pThis, offPort & ~(uint32_t)3, pu32);
1831 if (rcStrict == VINF_IOM_R3_MMIO_READ)
1832 rcStrict = VINF_IOM_R3_IOPORT_READ;
1833
1834 return rcStrict;
1835}
1836
1837/**
1838 * @callback_method_impl{FNIOMMMIONEWWRITE}
1839 */
1840static DECLCALLBACK(VBOXSTRICTRC) lsilogicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void const *pv, unsigned cb)
1841{
1842 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
1843 uint32_t u32;
1844 RT_NOREF_PV(pvUser);
1845
1846 /* See comments in lsilogicR3Construct regarding size and alignment. */
1847 if (cb == 4)
1848 u32 = *(uint32_t const *)pv;
1849 else
1850 {
1851 if (cb > 4)
1852 u32 = *(uint32_t const *)pv;
1853 else if (cb >= 2)
1854 u32 = *(uint16_t const *)pv;
1855 else
1856 u32 = *(uint8_t const *)pv;
1857 Log(("lsilogicMMIOWrite: Non-DWORD write access - off=%#RGp u32=%#x cb=%#x\n", off, u32, cb));
1858 }
1859
1860 VBOXSTRICTRC rcStrict;
1861 if (!(off & 3))
1862 rcStrict = lsilogicRegisterWrite(pDevIns, pThis, (uint32_t)off, u32);
1863 else
1864 {
1865 Log(("lsilogicMMIOWrite: Ignoring misaligned write - off=%#RGp u32=%#x cb=%#x\n", off, u32, cb));
1866 rcStrict = VINF_SUCCESS;
1867 }
1868 return rcStrict;
1869}
1870
1871/**
1872 * @callback_method_impl{FNIOMMMIONEWREAD}
1873 */
1874static DECLCALLBACK(VBOXSTRICTRC) lsilogicMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void *pv, unsigned cb)
1875{
1876 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
1877 Assert(!(off & 3)); Assert(cb == 4); /* If any of these trigger you've changed the registration flags or IOM is busted. */
1878 RT_NOREF2(pvUser, cb);
1879
1880 return lsilogicRegisterRead(pDevIns, pThis, off, (uint32_t *)pv);
1881}
1882
1883/**
1884 * @callback_method_impl{FNIOMMMIONEWWRITE}
1885 */
1886static DECLCALLBACK(VBOXSTRICTRC)
1887lsilogicDiagnosticWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void const *pv, unsigned cb)
1888{
1889 RT_NOREF(pDevIns, pvUser, off, pv, cb);
1890 LogFlowFunc(("pThis=%#p GCPhysAddr=%RGp pv=%#p{%.*Rhxs} cb=%u\n", PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI), off, pv, cb, pv, cb));
1891 return VINF_SUCCESS;
1892}
1893
1894/**
1895 * @callback_method_impl{FNIOMMMIONEWREAD}
1896 */
1897static DECLCALLBACK(VBOXSTRICTRC) lsilogicDiagnosticRead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void *pv, unsigned cb)
1898{
1899 RT_NOREF(pDevIns, pvUser, off, pv, cb);
1900 LogFlowFunc(("pThis=%#p off=%RGp pv=%#p{%.*Rhxs} cb=%u\n", PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI), off, pv, cb, pv, cb));
1901 return VINF_SUCCESS;
1902}
1903
1904#ifdef IN_RING3
1905
1906# ifdef LOG_ENABLED
1907/**
1908 * Dump an SG entry.
1909 *
1910 * @returns nothing.
1911 * @param pSGEntry Pointer to the SG entry to dump
1912 */
1913static void lsilogicDumpSGEntry(PMptSGEntryUnion pSGEntry)
1914{
1915 if (LogIsEnabled())
1916 {
1917 switch (pSGEntry->Simple32.u2ElementType)
1918 {
1919 case MPTSGENTRYTYPE_SIMPLE:
1920 {
1921 Log(("%s: Dumping info for SIMPLE SG entry:\n", __FUNCTION__));
1922 Log(("%s: u24Length=%u\n", __FUNCTION__, pSGEntry->Simple32.u24Length));
1923 Log(("%s: fEndOfList=%d\n", __FUNCTION__, pSGEntry->Simple32.fEndOfList));
1924 Log(("%s: f64BitAddress=%d\n", __FUNCTION__, pSGEntry->Simple32.f64BitAddress));
1925 Log(("%s: fBufferContainsData=%d\n", __FUNCTION__, pSGEntry->Simple32.fBufferContainsData));
1926 Log(("%s: fLocalAddress=%d\n", __FUNCTION__, pSGEntry->Simple32.fLocalAddress));
1927 Log(("%s: fEndOfBuffer=%d\n", __FUNCTION__, pSGEntry->Simple32.fEndOfBuffer));
1928 Log(("%s: fLastElement=%d\n", __FUNCTION__, pSGEntry->Simple32.fLastElement));
1929 Log(("%s: u32DataBufferAddressLow=%u\n", __FUNCTION__, pSGEntry->Simple32.u32DataBufferAddressLow));
1930 if (pSGEntry->Simple32.f64BitAddress)
1931 {
1932 Log(("%s: u32DataBufferAddressHigh=%u\n", __FUNCTION__, pSGEntry->Simple64.u32DataBufferAddressHigh));
1933 Log(("%s: GCDataBufferAddress=%RGp\n", __FUNCTION__,
1934 ((uint64_t)pSGEntry->Simple64.u32DataBufferAddressHigh << 32)
1935 | pSGEntry->Simple64.u32DataBufferAddressLow));
1936 }
1937 else
1938 Log(("%s: GCDataBufferAddress=%RGp\n", __FUNCTION__, pSGEntry->Simple32.u32DataBufferAddressLow));
1939
1940 break;
1941 }
1942 case MPTSGENTRYTYPE_CHAIN:
1943 {
1944 Log(("%s: Dumping info for CHAIN SG entry:\n", __FUNCTION__));
1945 Log(("%s: u16Length=%u\n", __FUNCTION__, pSGEntry->Chain.u16Length));
1946 Log(("%s: u8NExtChainOffset=%d\n", __FUNCTION__, pSGEntry->Chain.u8NextChainOffset));
1947 Log(("%s: f64BitAddress=%d\n", __FUNCTION__, pSGEntry->Chain.f64BitAddress));
1948 Log(("%s: fLocalAddress=%d\n", __FUNCTION__, pSGEntry->Chain.fLocalAddress));
1949 Log(("%s: u32SegmentAddressLow=%u\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressLow));
1950 Log(("%s: u32SegmentAddressHigh=%u\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressHigh));
1951 if (pSGEntry->Chain.f64BitAddress)
1952 Log(("%s: GCSegmentAddress=%RGp\n", __FUNCTION__,
1953 ((uint64_t)pSGEntry->Chain.u32SegmentAddressHigh << 32) | pSGEntry->Chain.u32SegmentAddressLow));
1954 else
1955 Log(("%s: GCSegmentAddress=%RGp\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressLow));
1956 break;
1957 }
1958 }
1959 }
1960}
1961# endif /* LOG_ENABLED */
1962
1963/**
1964 * Copy from guest to host memory worker.
1965 *
1966 * @copydoc FNLSILOGICR3MEMCOPYCALLBACK
1967 */
1968static DECLCALLBACK(void) lsilogicR3CopyBufferFromGuestWorker(PPDMDEVINS pDevIns, RTGCPHYS GCPhys,
1969 PRTSGBUF pSgBuf, size_t cbCopy, size_t *pcbSkip)
1970{
1971 size_t cbSkipped = RT_MIN(cbCopy, *pcbSkip);
1972 cbCopy -= cbSkipped;
1973 GCPhys += cbSkipped;
1974 *pcbSkip -= cbSkipped;
1975
1976 while (cbCopy)
1977 {
1978 size_t cbSeg = cbCopy;
1979 void *pvSeg = RTSgBufGetNextSegment(pSgBuf, &cbSeg);
1980
1981 AssertPtr(pvSeg);
1982 PDMDevHlpPCIPhysReadUser(pDevIns, GCPhys, pvSeg, cbSeg);
1983 GCPhys += cbSeg;
1984 cbCopy -= cbSeg;
1985 }
1986}
1987
1988/**
1989 * Copy from host to guest memory worker.
1990 *
1991 * @copydoc FNLSILOGICR3MEMCOPYCALLBACK
1992 */
1993static DECLCALLBACK(void) lsilogicR3CopyBufferToGuestWorker(PPDMDEVINS pDevIns, RTGCPHYS GCPhys,
1994 PRTSGBUF pSgBuf, size_t cbCopy, size_t *pcbSkip)
1995{
1996 size_t cbSkipped = RT_MIN(cbCopy, *pcbSkip);
1997 cbCopy -= cbSkipped;
1998 GCPhys += cbSkipped;
1999 *pcbSkip -= cbSkipped;
2000
2001 while (cbCopy)
2002 {
2003 size_t cbSeg = cbCopy;
2004 void *pvSeg = RTSgBufGetNextSegment(pSgBuf, &cbSeg);
2005
2006 AssertPtr(pvSeg);
2007 PDMDevHlpPCIPhysWriteUser(pDevIns, GCPhys, pvSeg, cbSeg);
2008 GCPhys += cbSeg;
2009 cbCopy -= cbSeg;
2010 }
2011}
2012
2013/**
2014 * Walks the guest S/G buffer calling the given copy worker for every buffer.
2015 *
2016 * @returns The amout of bytes actually copied.
2017 * @param pDevIns The device instance.
2018 * @param pLsiReq LSI request state.
2019 * @param pfnCopyWorker The copy method to apply for each guest buffer.
2020 * @param pSgBuf The host S/G buffer.
2021 * @param cbSkip How many bytes to skip in advance before starting to
2022 * copy.
2023 * @param cbCopy How many bytes to copy.
2024 */
2025static size_t lsilogicSgBufWalker(PPDMDEVINS pDevIns, PLSILOGICREQ pLsiReq,
2026 PFNLSILOGICR3MEMCOPYCALLBACK pfnCopyWorker,
2027 PRTSGBUF pSgBuf, size_t cbSkip, size_t cbCopy)
2028{
2029 bool fEndOfList = false;
2030 RTGCPHYS GCPhysSgEntryNext = pLsiReq->GCPhysSgStart;
2031 RTGCPHYS GCPhysSegmentStart = pLsiReq->GCPhysSgStart;
2032 uint32_t cChainOffsetNext = pLsiReq->cChainOffset;
2033 size_t cbCopied = 0;
2034
2035 /*
2036 * Add the amount to skip to the host buffer size to avoid a
2037 * few conditionals later on.
2038 */
2039 cbCopy += cbSkip;
2040
2041 /* Go through the list until we reach the end. */
2042 while ( !fEndOfList
2043 && cbCopy)
2044 {
2045 bool fEndOfSegment = false;
2046
2047 while ( !fEndOfSegment
2048 && cbCopy)
2049 {
2050 MptSGEntryUnion SGEntry;
2051
2052 Log(("%s: Reading SG entry from %RGp\n", __FUNCTION__, GCPhysSgEntryNext));
2053
2054 /* Read the entry. */
2055 PDMDevHlpPCIPhysReadMeta(pDevIns, GCPhysSgEntryNext, &SGEntry, sizeof(MptSGEntryUnion));
2056
2057# ifdef LOG_ENABLED
2058 lsilogicDumpSGEntry(&SGEntry);
2059# endif
2060
2061 AssertMsg(SGEntry.Simple32.u2ElementType == MPTSGENTRYTYPE_SIMPLE, ("Invalid SG entry type\n"));
2062
2063 /* Check if this is a zero element and abort. */
2064 if ( !SGEntry.Simple32.u24Length
2065 && SGEntry.Simple32.fEndOfList
2066 && SGEntry.Simple32.fEndOfBuffer)
2067 return cbCopied - RT_MIN(cbSkip, cbCopied);
2068
2069 size_t cbCopyThis = RT_MIN(SGEntry.Simple32.u24Length, cbCopy);
2070 RTGCPHYS GCPhysAddrDataBuffer = SGEntry.Simple32.u32DataBufferAddressLow;
2071
2072 if (SGEntry.Simple32.f64BitAddress)
2073 {
2074 GCPhysAddrDataBuffer |= ((uint64_t)SGEntry.Simple64.u32DataBufferAddressHigh) << 32;
2075 GCPhysSgEntryNext += sizeof(MptSGEntrySimple64);
2076 }
2077 else
2078 GCPhysSgEntryNext += sizeof(MptSGEntrySimple32);
2079
2080 pfnCopyWorker(pDevIns, GCPhysAddrDataBuffer, pSgBuf, cbCopyThis, &cbSkip);
2081 cbCopy -= cbCopyThis;
2082 cbCopied += cbCopyThis;
2083
2084 /* Check if we reached the end of the list. */
2085 if (SGEntry.Simple32.fEndOfList)
2086 {
2087 /* We finished. */
2088 fEndOfSegment = true;
2089 fEndOfList = true;
2090 }
2091 else if (SGEntry.Simple32.fLastElement)
2092 fEndOfSegment = true;
2093 } /* while (!fEndOfSegment) */
2094
2095 /* Get next chain element. */
2096 if (cChainOffsetNext)
2097 {
2098 MptSGEntryChain SGEntryChain;
2099
2100 PDMDevHlpPCIPhysReadMeta(pDevIns, GCPhysSegmentStart + cChainOffsetNext, &SGEntryChain, sizeof(MptSGEntryChain));
2101
2102 AssertMsg(SGEntryChain.u2ElementType == MPTSGENTRYTYPE_CHAIN, ("Invalid SG entry type\n"));
2103
2104 /* Set the next address now. */
2105 GCPhysSgEntryNext = SGEntryChain.u32SegmentAddressLow;
2106 if (SGEntryChain.f64BitAddress)
2107 GCPhysSgEntryNext |= ((uint64_t)SGEntryChain.u32SegmentAddressHigh) << 32;
2108
2109 GCPhysSegmentStart = GCPhysSgEntryNext;
2110 cChainOffsetNext = SGEntryChain.u8NextChainOffset * sizeof(uint32_t);
2111 }
2112 } /* while (!fEndOfList) */
2113
2114 return cbCopied - RT_MIN(cbSkip, cbCopied);
2115}
2116
2117/**
2118 * Copies a data buffer into the S/G buffer set up by the guest.
2119 *
2120 * @returns Amount of bytes copied to the guest.
2121 * @param pDevIns The device instance.
2122 * @param pReq Request structure.
2123 * @param pSgBuf The S/G buffer to copy from.
2124 * @param cbSkip How many bytes to skip in advance before starting to copy.
2125 * @param cbCopy How many bytes to copy.
2126 */
2127static size_t lsilogicR3CopySgBufToGuest(PPDMDEVINS pDevIns, PLSILOGICREQ pReq, PRTSGBUF pSgBuf,
2128 size_t cbSkip, size_t cbCopy)
2129{
2130 return lsilogicSgBufWalker(pDevIns, pReq, lsilogicR3CopyBufferToGuestWorker, pSgBuf, cbSkip, cbCopy);
2131}
2132
2133/**
2134 * Copies the guest S/G buffer into a host data buffer.
2135 *
2136 * @returns Amount of bytes copied from the guest.
2137 * @param pDevIns The device instance.
2138 * @param pReq Request structure.
2139 * @param pSgBuf The S/G buffer to copy into.
2140 * @param cbSkip How many bytes to skip in advance before starting to copy.
2141 * @param cbCopy How many bytes to copy.
2142 */
2143static size_t lsilogicR3CopySgBufFromGuest(PPDMDEVINS pDevIns, PLSILOGICREQ pReq, PRTSGBUF pSgBuf,
2144 size_t cbSkip, size_t cbCopy)
2145{
2146 return lsilogicSgBufWalker(pDevIns, pReq, lsilogicR3CopyBufferFromGuestWorker, pSgBuf, cbSkip, cbCopy);
2147}
2148
2149#if 0 /* unused */
2150/**
2151 * Copy a simple memory buffer to the guest memory buffer.
2152 *
2153 * @returns Amount of bytes copied to the guest.
2154 * @param pThis The LsiLogic controller device instance.
2155 * @param pReq Request structure.
2156 * @param pvSrc The buffer to copy from.
2157 * @param cbSrc How many bytes to copy.
2158 * @param cbSkip How many bytes to skip initially.
2159 */
2160static size_t lsilogicR3CopyBufferToGuest(PLSILOGICSCSI pThis, PLSILOGICREQ pReq, const void *pvSrc,
2161 size_t cbSrc, size_t cbSkip)
2162{
2163 RTSGSEG Seg;
2164 RTSGBUF SgBuf;
2165 Seg.pvSeg = (void *)pvSrc;
2166 Seg.cbSeg = cbSrc;
2167 RTSgBufInit(&SgBuf, &Seg, 1);
2168 return lsilogicR3CopySgBufToGuest(pThis, pReq, &SgBuf, cbSkip, cbSrc);
2169}
2170
2171/**
2172 * Copy a guest memry buffe into simple host memory buffer.
2173 *
2174 * @returns Amount of bytes copied to the guest.
2175 * @param pThis The LsiLogic controller device instance.
2176 * @param pReq Request structure.
2177 * @param pvSrc The buffer to copy from.
2178 * @param cbSrc How many bytes to copy.
2179 * @param cbSkip How many bytes to skip initially.
2180 */
2181static size_t lsilogicR3CopyBufferFromGuest(PLSILOGICSCSI pThis, PLSILOGICREQ pReq, void *pvDst,
2182 size_t cbDst, size_t cbSkip)
2183{
2184 RTSGSEG Seg;
2185 RTSGBUF SgBuf;
2186 Seg.pvSeg = (void *)pvDst;
2187 Seg.cbSeg = cbDst;
2188 RTSgBufInit(&SgBuf, &Seg, 1);
2189 return lsilogicR3CopySgBufFromGuest(pThis, pReq, &SgBuf, cbSkip, cbDst);
2190}
2191#endif
2192
2193# ifdef LOG_ENABLED
2194static void lsilogicR3DumpSCSIIORequest(PMptSCSIIORequest pSCSIIORequest)
2195{
2196 if (LogIsEnabled())
2197 {
2198 Log(("%s: u8TargetID=%d\n", __FUNCTION__, pSCSIIORequest->u8TargetID));
2199 Log(("%s: u8Bus=%d\n", __FUNCTION__, pSCSIIORequest->u8Bus));
2200 Log(("%s: u8ChainOffset=%d\n", __FUNCTION__, pSCSIIORequest->u8ChainOffset));
2201 Log(("%s: u8Function=%d\n", __FUNCTION__, pSCSIIORequest->u8Function));
2202 Log(("%s: u8CDBLength=%d\n", __FUNCTION__, pSCSIIORequest->u8CDBLength));
2203 Log(("%s: u8SenseBufferLength=%d\n", __FUNCTION__, pSCSIIORequest->u8SenseBufferLength));
2204 Log(("%s: u8MessageFlags=%d\n", __FUNCTION__, pSCSIIORequest->u8MessageFlags));
2205 Log(("%s: u32MessageContext=%#x\n", __FUNCTION__, pSCSIIORequest->u32MessageContext));
2206 for (unsigned i = 0; i < RT_ELEMENTS(pSCSIIORequest->au8LUN); i++)
2207 Log(("%s: u8LUN[%d]=%d\n", __FUNCTION__, i, pSCSIIORequest->au8LUN[i]));
2208 Log(("%s: u32Control=%#x\n", __FUNCTION__, pSCSIIORequest->u32Control));
2209 for (unsigned i = 0; i < RT_ELEMENTS(pSCSIIORequest->au8CDB); i++)
2210 Log(("%s: u8CDB[%d]=%d\n", __FUNCTION__, i, pSCSIIORequest->au8CDB[i]));
2211 Log(("%s: u32DataLength=%#x\n", __FUNCTION__, pSCSIIORequest->u32DataLength));
2212 Log(("%s: u32SenseBufferLowAddress=%#x\n", __FUNCTION__, pSCSIIORequest->u32SenseBufferLowAddress));
2213 }
2214}
2215# endif
2216
2217/**
2218 * Handles the completion of th given request.
2219 *
2220 * @returns nothing.
2221 * @param pDevIns The device instance.
2222 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
2223 * @param pThis Pointer to the shared LsiLogic device state.
2224 * @param pReq The request to complete.
2225 * @param rcReq Status code of the request.
2226 */
2227static void lsilogicR3ReqComplete(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC, PLSILOGICREQ pReq, int rcReq)
2228{
2229 PLSILOGICDEVICE pTgtDev = pReq->pTargetDevice;
2230
2231 if (!pReq->fBIOS)
2232 {
2233 RTGCPHYS GCPhysAddrSenseBuffer;
2234
2235 GCPhysAddrSenseBuffer = pReq->GuestRequest.SCSIIO.u32SenseBufferLowAddress;
2236 GCPhysAddrSenseBuffer |= ((uint64_t)pThis->u32SenseBufferHighAddr << 32);
2237
2238 /* Copy the sense buffer over. */
2239 if (pReq->GuestRequest.SCSIIO.u8SenseBufferLength > 0)
2240 PDMDevHlpPCIPhysWriteMeta(pDevIns, GCPhysAddrSenseBuffer, pReq->abSenseBuffer,
2241 RT_UNLIKELY( pReq->GuestRequest.SCSIIO.u8SenseBufferLength
2242 < sizeof(pReq->abSenseBuffer))
2243 ? pReq->GuestRequest.SCSIIO.u8SenseBufferLength
2244 : sizeof(pReq->abSenseBuffer));
2245
2246 if (RT_SUCCESS(rcReq) && RT_LIKELY(pReq->u8ScsiSts == SCSI_STATUS_OK))
2247 {
2248 uint32_t u32MsgCtx = pReq->GuestRequest.SCSIIO.u32MessageContext;
2249
2250 /* Free the request before posting completion. */
2251 pTgtDev->pDrvMediaEx->pfnIoReqFree(pTgtDev->pDrvMediaEx, pReq->hIoReq);
2252 lsilogicR3FinishContextReply(pDevIns, pThis, u32MsgCtx);
2253 }
2254 else
2255 {
2256 MptReplyUnion IOCReply;
2257 RT_ZERO(IOCReply);
2258
2259 /* The SCSI target encountered an error during processing post a reply. */
2260 IOCReply.SCSIIOError.u8TargetID = pReq->GuestRequest.SCSIIO.u8TargetID;
2261 IOCReply.SCSIIOError.u8Bus = pReq->GuestRequest.SCSIIO.u8Bus;
2262 IOCReply.SCSIIOError.u8MessageLength = 8;
2263 IOCReply.SCSIIOError.u8Function = pReq->GuestRequest.SCSIIO.u8Function;
2264 IOCReply.SCSIIOError.u8CDBLength = pReq->GuestRequest.SCSIIO.u8CDBLength;
2265 IOCReply.SCSIIOError.u8SenseBufferLength = pReq->GuestRequest.SCSIIO.u8SenseBufferLength;
2266 IOCReply.SCSIIOError.u8MessageFlags = pReq->GuestRequest.SCSIIO.u8MessageFlags;
2267 IOCReply.SCSIIOError.u32MessageContext = pReq->GuestRequest.SCSIIO.u32MessageContext;
2268 IOCReply.SCSIIOError.u8SCSIStatus = pReq->u8ScsiSts;
2269 IOCReply.SCSIIOError.u8SCSIState = MPT_SCSI_IO_ERROR_SCSI_STATE_AUTOSENSE_VALID;
2270 IOCReply.SCSIIOError.u16IOCStatus = 0;
2271 IOCReply.SCSIIOError.u32IOCLogInfo = 0;
2272 IOCReply.SCSIIOError.u32TransferCount = 0;
2273 IOCReply.SCSIIOError.u32SenseCount = sizeof(pReq->abSenseBuffer);
2274 IOCReply.SCSIIOError.u32ResponseInfo = 0;
2275
2276 /* Free the request before posting completion. */
2277 pTgtDev->pDrvMediaEx->pfnIoReqFree(pTgtDev->pDrvMediaEx, pReq->hIoReq);
2278 lsilogicFinishAddressReply(pDevIns, pThis, &IOCReply, false);
2279 }
2280 }
2281 else
2282 {
2283 uint8_t u8ScsiSts = pReq->u8ScsiSts;
2284 pTgtDev->pDrvMediaEx->pfnIoReqFree(pTgtDev->pDrvMediaEx, pReq->hIoReq);
2285 int rc = vboxscsiRequestFinished(&pThisCC->VBoxSCSI, u8ScsiSts);
2286 AssertMsgRC(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc));
2287 }
2288
2289 ASMAtomicDecU32(&pTgtDev->cOutstandingRequests);
2290
2291 if (pTgtDev->cOutstandingRequests == 0 && pThis->fSignalIdle)
2292 PDMDevHlpAsyncNotificationCompleted(pDevIns);
2293}
2294
2295/**
2296 * Processes a SCSI I/O request by setting up the request
2297 * and sending it to the underlying SCSI driver.
2298 * Steps needed to complete request are done in the
2299 * callback called by the driver below upon completion of
2300 * the request.
2301 *
2302 * @returns VBox status code.
2303 * @param pDevIns The device instance.
2304 * @param pThis Pointer to the shared LsiLogic device state.
2305 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
2306 * @param GCPhysMessageFrameAddr Guest physical address where the request is located.
2307 * @param pGuestReq The request read fro th guest memory.
2308 */
2309static int lsilogicR3ProcessSCSIIORequest(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC,
2310 RTGCPHYS GCPhysMessageFrameAddr, PMptRequestUnion pGuestReq)
2311{
2312 MptReplyUnion IOCReply;
2313 int rc = VINF_SUCCESS;
2314
2315# ifdef LOG_ENABLED
2316 lsilogicR3DumpSCSIIORequest(&pGuestReq->SCSIIO);
2317# endif
2318
2319 if (RT_LIKELY( (pGuestReq->SCSIIO.u8TargetID < pThis->cDeviceStates)
2320 && (pGuestReq->SCSIIO.u8Bus == 0)))
2321 {
2322 PLSILOGICDEVICE pTgtDev = &pThisCC->paDeviceStates[pGuestReq->SCSIIO.u8TargetID];
2323
2324 if (pTgtDev->pDrvBase)
2325 {
2326 /* Allocate and prepare a new request. */
2327 PDMMEDIAEXIOREQ hIoReq;
2328 PLSILOGICREQ pLsiReq = NULL;
2329 rc = pTgtDev->pDrvMediaEx->pfnIoReqAlloc(pTgtDev->pDrvMediaEx, &hIoReq, (void **)&pLsiReq,
2330 pGuestReq->SCSIIO.u32MessageContext,
2331 PDMIMEDIAEX_F_SUSPEND_ON_RECOVERABLE_ERR);
2332 if (RT_SUCCESS(rc))
2333 {
2334 pLsiReq->hIoReq = hIoReq;
2335 pLsiReq->pTargetDevice = pTgtDev;
2336 pLsiReq->GCPhysMessageFrameAddr = GCPhysMessageFrameAddr;
2337 pLsiReq->fBIOS = false;
2338 pLsiReq->GCPhysSgStart = GCPhysMessageFrameAddr + sizeof(MptSCSIIORequest);
2339 pLsiReq->cChainOffset = pGuestReq->SCSIIO.u8ChainOffset;
2340 if (pLsiReq->cChainOffset)
2341 pLsiReq->cChainOffset = pLsiReq->cChainOffset * sizeof(uint32_t) - sizeof(MptSCSIIORequest);
2342 memcpy(&pLsiReq->GuestRequest, pGuestReq, sizeof(MptRequestUnion));
2343 RT_BZERO(&pLsiReq->abSenseBuffer[0], sizeof(pLsiReq->abSenseBuffer));
2344
2345 PDMMEDIAEXIOREQSCSITXDIR enmXferDir = PDMMEDIAEXIOREQSCSITXDIR_UNKNOWN;
2346 uint8_t uDataDirection = MPT_SCSIIO_REQUEST_CONTROL_TXDIR_GET(pLsiReq->GuestRequest.SCSIIO.u32Control);
2347
2348 /*
2349 * Keep the direction to unknown if there is a mismatch between the data length
2350 * and the transfer direction bit.
2351 * The Solaris 9 driver is buggy and sets it to none for INQUIRY requests.
2352 */
2353 if ( uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_NONE
2354 && pLsiReq->GuestRequest.SCSIIO.u32DataLength == 0)
2355 enmXferDir = PDMMEDIAEXIOREQSCSITXDIR_NONE;
2356 else if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_WRITE)
2357 enmXferDir = PDMMEDIAEXIOREQSCSITXDIR_TO_DEVICE;
2358 else if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_READ)
2359 enmXferDir = PDMMEDIAEXIOREQSCSITXDIR_FROM_DEVICE;
2360
2361 ASMAtomicIncU32(&pTgtDev->cOutstandingRequests);
2362 rc = pTgtDev->pDrvMediaEx->pfnIoReqSendScsiCmd(pTgtDev->pDrvMediaEx, pLsiReq->hIoReq, pLsiReq->GuestRequest.SCSIIO.au8LUN[1],
2363 &pLsiReq->GuestRequest.SCSIIO.au8CDB[0], pLsiReq->GuestRequest.SCSIIO.u8CDBLength,
2364 enmXferDir, NULL, pLsiReq->GuestRequest.SCSIIO.u32DataLength,
2365 &pLsiReq->abSenseBuffer[0], sizeof(pLsiReq->abSenseBuffer), NULL,
2366 &pLsiReq->u8ScsiSts, 30 * RT_MS_1SEC);
2367 if (rc != VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS)
2368 lsilogicR3ReqComplete(pDevIns, pThis, pThisCC, pLsiReq, rc);
2369
2370 return VINF_SUCCESS;
2371 }
2372 else
2373 IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_DEVICE_NOT_THERE;
2374 }
2375 else
2376 {
2377 /* Device is not present report SCSI selection timeout. */
2378 IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_DEVICE_NOT_THERE;
2379 }
2380 }
2381 else
2382 {
2383 /* Report out of bounds target ID or bus. */
2384 if (pGuestReq->SCSIIO.u8Bus != 0)
2385 IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_BUS;
2386 else
2387 IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_TARGETID;
2388 }
2389
2390 static int g_cLogged = 0;
2391
2392 if (g_cLogged++ < MAX_REL_LOG_ERRORS)
2393 {
2394 LogRel(("LsiLogic#%d: %d/%d (Bus/Target) doesn't exist\n", pDevIns->iInstance,
2395 pGuestReq->SCSIIO.u8TargetID, pGuestReq->SCSIIO.u8Bus));
2396 /* Log the CDB too */
2397 LogRel(("LsiLogic#%d: Guest issued CDB {%#x",
2398 pDevIns->iInstance, pGuestReq->SCSIIO.au8CDB[0]));
2399 for (unsigned i = 1; i < pGuestReq->SCSIIO.u8CDBLength; i++)
2400 LogRel((", %#x", pGuestReq->SCSIIO.au8CDB[i]));
2401 LogRel(("}\n"));
2402 }
2403
2404 /* The rest is equal to both errors. */
2405 IOCReply.SCSIIOError.u8TargetID = pGuestReq->SCSIIO.u8TargetID;
2406 IOCReply.SCSIIOError.u8Bus = pGuestReq->SCSIIO.u8Bus;
2407 IOCReply.SCSIIOError.u8MessageLength = sizeof(MptSCSIIOErrorReply) / 4;
2408 IOCReply.SCSIIOError.u8Function = pGuestReq->SCSIIO.u8Function;
2409 IOCReply.SCSIIOError.u8CDBLength = pGuestReq->SCSIIO.u8CDBLength;
2410 IOCReply.SCSIIOError.u8SenseBufferLength = pGuestReq->SCSIIO.u8SenseBufferLength;
2411 IOCReply.SCSIIOError.u8Reserved = 0;
2412 IOCReply.SCSIIOError.u8MessageFlags = 0;
2413 IOCReply.SCSIIOError.u32MessageContext = pGuestReq->SCSIIO.u32MessageContext;
2414 IOCReply.SCSIIOError.u8SCSIStatus = SCSI_STATUS_OK;
2415 IOCReply.SCSIIOError.u8SCSIState = MPT_SCSI_IO_ERROR_SCSI_STATE_TERMINATED;
2416 IOCReply.SCSIIOError.u32IOCLogInfo = 0;
2417 IOCReply.SCSIIOError.u32TransferCount = 0;
2418 IOCReply.SCSIIOError.u32SenseCount = 0;
2419 IOCReply.SCSIIOError.u32ResponseInfo = 0;
2420
2421 lsilogicFinishAddressReply(pDevIns, pThis, &IOCReply, false);
2422
2423 return rc;
2424}
2425
2426
2427/**
2428 * @interface_method_impl{PDMIMEDIAPORT,pfnQueryDeviceLocation}
2429 */
2430static DECLCALLBACK(int) lsilogicR3QueryDeviceLocation(PPDMIMEDIAPORT pInterface, const char **ppcszController,
2431 uint32_t *piInstance, uint32_t *piLUN)
2432{
2433 PLSILOGICDEVICE pTgtDev = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IMediaPort);
2434 PPDMDEVINS pDevIns = pTgtDev->pDevIns;
2435
2436 AssertPtrReturn(ppcszController, VERR_INVALID_POINTER);
2437 AssertPtrReturn(piInstance, VERR_INVALID_POINTER);
2438 AssertPtrReturn(piLUN, VERR_INVALID_POINTER);
2439
2440 *ppcszController = pDevIns->pReg->szName;
2441 *piInstance = pDevIns->iInstance;
2442 *piLUN = pTgtDev->iLUN;
2443
2444 return VINF_SUCCESS;
2445}
2446
2447
2448/**
2449 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqCopyFromBuf}
2450 */
2451static DECLCALLBACK(int) lsilogicR3IoReqCopyFromBuf(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
2452 void *pvIoReqAlloc, uint32_t offDst, PRTSGBUF pSgBuf,
2453 size_t cbCopy)
2454{
2455 RT_NOREF1(hIoReq);
2456 PLSILOGICDEVICE pTgtDev = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IMediaExPort);
2457 PPDMDEVINS pDevIns = pTgtDev->pDevIns;
2458 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
2459 PLSILOGICREQ pReq = (PLSILOGICREQ)pvIoReqAlloc;
2460
2461 size_t cbCopied = 0;
2462 if (!pReq->fBIOS)
2463 cbCopied = lsilogicR3CopySgBufToGuest(pDevIns, pReq, pSgBuf, offDst, cbCopy);
2464 else
2465 cbCopied = vboxscsiCopyToBuf(&pThisCC->VBoxSCSI, pSgBuf, offDst, cbCopy);
2466 return cbCopied == cbCopy ? VINF_SUCCESS : VERR_PDM_MEDIAEX_IOBUF_OVERFLOW;
2467}
2468
2469/**
2470 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqCopyToBuf}
2471 */
2472static DECLCALLBACK(int) lsilogicR3IoReqCopyToBuf(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
2473 void *pvIoReqAlloc, uint32_t offSrc, PRTSGBUF pSgBuf,
2474 size_t cbCopy)
2475{
2476 RT_NOREF1(hIoReq);
2477 PLSILOGICDEVICE pTgtDev = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IMediaExPort);
2478 PPDMDEVINS pDevIns = pTgtDev->pDevIns;
2479 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
2480 PLSILOGICREQ pReq = (PLSILOGICREQ)pvIoReqAlloc;
2481
2482 size_t cbCopied = 0;
2483 if (!pReq->fBIOS)
2484 cbCopied = lsilogicR3CopySgBufFromGuest(pDevIns, pReq, pSgBuf, offSrc, cbCopy);
2485 else
2486 cbCopied = vboxscsiCopyFromBuf(&pThisCC->VBoxSCSI, pSgBuf, offSrc, cbCopy);
2487 return cbCopied == cbCopy ? VINF_SUCCESS : VERR_PDM_MEDIAEX_IOBUF_UNDERRUN;
2488}
2489
2490/**
2491 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqCompleteNotify}
2492 */
2493static DECLCALLBACK(int) lsilogicR3IoReqCompleteNotify(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
2494 void *pvIoReqAlloc, int rcReq)
2495{
2496 RT_NOREF(hIoReq);
2497 PLSILOGICDEVICE pTgtDev = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IMediaExPort);
2498 PPDMDEVINS pDevIns = pTgtDev->pDevIns;
2499 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
2500 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
2501 lsilogicR3ReqComplete(pDevIns, pThis, pThisCC, (PLSILOGICREQ)pvIoReqAlloc, rcReq);
2502 return VINF_SUCCESS;
2503}
2504
2505/**
2506 * @interface_method_impl{PDMIMEDIAEXPORT,pfnIoReqStateChanged}
2507 */
2508static DECLCALLBACK(void) lsilogicR3IoReqStateChanged(PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
2509 void *pvIoReqAlloc, PDMMEDIAEXIOREQSTATE enmState)
2510{
2511 RT_NOREF3(hIoReq, pvIoReqAlloc, enmState);
2512 PLSILOGICDEVICE pTgtDev = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IMediaExPort);
2513
2514 switch (enmState)
2515 {
2516 case PDMMEDIAEXIOREQSTATE_SUSPENDED:
2517 {
2518 /* Make sure the request is not accounted for so the VM can suspend successfully. */
2519 PPDMDEVINS pDevIns = pTgtDev->pDevIns;
2520 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
2521 uint32_t cTasksActive = ASMAtomicDecU32(&pTgtDev->cOutstandingRequests);
2522 if (!cTasksActive && pThis->fSignalIdle)
2523 PDMDevHlpAsyncNotificationCompleted(pDevIns);
2524 break;
2525 }
2526 case PDMMEDIAEXIOREQSTATE_ACTIVE:
2527 /* Make sure the request is accounted for so the VM suspends only when the request is complete. */
2528 ASMAtomicIncU32(&pTgtDev->cOutstandingRequests);
2529 break;
2530 default:
2531 AssertMsgFailed(("Invalid request state given %u\n", enmState));
2532 }
2533}
2534
2535/**
2536 * @interface_method_impl{PDMIMEDIAEXPORT,pfnMediumEjected}
2537 */
2538static DECLCALLBACK(void) lsilogicR3MediumEjected(PPDMIMEDIAEXPORT pInterface)
2539{
2540 PLSILOGICDEVICE pTgtDev = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IMediaExPort);
2541 PPDMDEVINS pDevIns = pTgtDev->pDevIns;
2542 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
2543
2544 if (pThisCC->pMediaNotify)
2545 {
2546 int rc = VMR3ReqCallNoWait(PDMDevHlpGetVM(pDevIns), VMCPUID_ANY,
2547 (PFNRT)pThisCC->pMediaNotify->pfnEjected, 2,
2548 pThisCC->pMediaNotify, pTgtDev->iLUN);
2549 AssertRC(rc);
2550 }
2551}
2552
2553
2554/**
2555 * Return the configuration page header and data
2556 * which matches the given page type and number.
2557 *
2558 * @returns VINF_SUCCESS if successful
2559 * VERR_NOT_FOUND if the requested page could be found.
2560 * @param pThis Pointer to the shared LsiLogic device state. data.
2561 * @param pPages The pages supported by the controller.
2562 * @param u8PageNumber Number of the page to get.
2563 * @param ppPageHeader Where to store the pointer to the page header.
2564 * @param ppbPageData Where to store the pointer to the page data.
2565 * @param pcbPage Where to store the size of the page data in bytes on success.
2566 */
2567static int lsilogicR3ConfigurationIOUnitPageGetFromNumber(PLSILOGICSCSI pThis,
2568 PMptConfigurationPagesSupported pPages,
2569 uint8_t u8PageNumber,
2570 PMptConfigurationPageHeader *ppPageHeader,
2571 uint8_t **ppbPageData, size_t *pcbPage)
2572{
2573 RT_NOREF(pThis);
2574 int rc = VINF_SUCCESS;
2575
2576 AssertPtr(ppPageHeader); Assert(ppbPageData);
2577
2578 switch (u8PageNumber)
2579 {
2580 case 0:
2581 *ppPageHeader = &pPages->IOUnitPage0.u.fields.Header;
2582 *ppbPageData = pPages->IOUnitPage0.u.abPageData;
2583 *pcbPage = sizeof(pPages->IOUnitPage0);
2584 break;
2585 case 1:
2586 *ppPageHeader = &pPages->IOUnitPage1.u.fields.Header;
2587 *ppbPageData = pPages->IOUnitPage1.u.abPageData;
2588 *pcbPage = sizeof(pPages->IOUnitPage1);
2589 break;
2590 case 2:
2591 *ppPageHeader = &pPages->IOUnitPage2.u.fields.Header;
2592 *ppbPageData = pPages->IOUnitPage2.u.abPageData;
2593 *pcbPage = sizeof(pPages->IOUnitPage2);
2594 break;
2595 case 3:
2596 *ppPageHeader = &pPages->IOUnitPage3.u.fields.Header;
2597 *ppbPageData = pPages->IOUnitPage3.u.abPageData;
2598 *pcbPage = sizeof(pPages->IOUnitPage3);
2599 break;
2600 case 4:
2601 *ppPageHeader = &pPages->IOUnitPage4.u.fields.Header;
2602 *ppbPageData = pPages->IOUnitPage4.u.abPageData;
2603 *pcbPage = sizeof(pPages->IOUnitPage4);
2604 break;
2605 default:
2606 rc = VERR_NOT_FOUND;
2607 }
2608
2609 return rc;
2610}
2611
2612/**
2613 * Return the configuration page header and data
2614 * which matches the given page type and number.
2615 *
2616 * @returns VINF_SUCCESS if successful
2617 * VERR_NOT_FOUND if the requested page could be found.
2618 * @param pThis Pointer to the shared LsiLogic device state. data.
2619 * @param pPages The pages supported by the controller.
2620 * @param u8PageNumber Number of the page to get.
2621 * @param ppPageHeader Where to store the pointer to the page header.
2622 * @param ppbPageData Where to store the pointer to the page data.
2623 * @param pcbPage Where to store the size of the page data in bytes on success.
2624 */
2625static int lsilogicR3ConfigurationIOCPageGetFromNumber(PLSILOGICSCSI pThis,
2626 PMptConfigurationPagesSupported pPages,
2627 uint8_t u8PageNumber,
2628 PMptConfigurationPageHeader *ppPageHeader,
2629 uint8_t **ppbPageData, size_t *pcbPage)
2630{
2631 RT_NOREF(pThis);
2632 int rc = VINF_SUCCESS;
2633
2634 AssertPtr(ppPageHeader); Assert(ppbPageData);
2635
2636 switch (u8PageNumber)
2637 {
2638 case 0:
2639 *ppPageHeader = &pPages->IOCPage0.u.fields.Header;
2640 *ppbPageData = pPages->IOCPage0.u.abPageData;
2641 *pcbPage = sizeof(pPages->IOCPage0);
2642 break;
2643 case 1:
2644 *ppPageHeader = &pPages->IOCPage1.u.fields.Header;
2645 *ppbPageData = pPages->IOCPage1.u.abPageData;
2646 *pcbPage = sizeof(pPages->IOCPage1);
2647 break;
2648 case 2:
2649 *ppPageHeader = &pPages->IOCPage2.u.fields.Header;
2650 *ppbPageData = pPages->IOCPage2.u.abPageData;
2651 *pcbPage = sizeof(pPages->IOCPage2);
2652 break;
2653 case 3:
2654 *ppPageHeader = &pPages->IOCPage3.u.fields.Header;
2655 *ppbPageData = pPages->IOCPage3.u.abPageData;
2656 *pcbPage = sizeof(pPages->IOCPage3);
2657 break;
2658 case 4:
2659 *ppPageHeader = &pPages->IOCPage4.u.fields.Header;
2660 *ppbPageData = pPages->IOCPage4.u.abPageData;
2661 *pcbPage = sizeof(pPages->IOCPage4);
2662 break;
2663 case 6:
2664 *ppPageHeader = &pPages->IOCPage6.u.fields.Header;
2665 *ppbPageData = pPages->IOCPage6.u.abPageData;
2666 *pcbPage = sizeof(pPages->IOCPage6);
2667 break;
2668 default:
2669 rc = VERR_NOT_FOUND;
2670 }
2671
2672 return rc;
2673}
2674
2675/**
2676 * Return the configuration page header and data
2677 * which matches the given page type and number.
2678 *
2679 * @returns VINF_SUCCESS if successful
2680 * VERR_NOT_FOUND if the requested page could be found.
2681 * @param pThis Pointer to the shared LsiLogic device state. data.
2682 * @param pPages The pages supported by the controller.
2683 * @param u8PageNumber Number of the page to get.
2684 * @param ppPageHeader Where to store the pointer to the page header.
2685 * @param ppbPageData Where to store the pointer to the page data.
2686 * @param pcbPage Where to store the size of the page data in bytes on success.
2687 */
2688static int lsilogicR3ConfigurationManufacturingPageGetFromNumber(PLSILOGICSCSI pThis,
2689 PMptConfigurationPagesSupported pPages,
2690 uint8_t u8PageNumber,
2691 PMptConfigurationPageHeader *ppPageHeader,
2692 uint8_t **ppbPageData, size_t *pcbPage)
2693{
2694 int rc = VINF_SUCCESS;
2695
2696 AssertPtr(ppPageHeader); Assert(ppbPageData);
2697
2698 switch (u8PageNumber)
2699 {
2700 case 0:
2701 *ppPageHeader = &pPages->ManufacturingPage0.u.fields.Header;
2702 *ppbPageData = pPages->ManufacturingPage0.u.abPageData;
2703 *pcbPage = sizeof(pPages->ManufacturingPage0);
2704 break;
2705 case 1:
2706 *ppPageHeader = &pPages->ManufacturingPage1.u.fields.Header;
2707 *ppbPageData = pPages->ManufacturingPage1.u.abPageData;
2708 *pcbPage = sizeof(pPages->ManufacturingPage1);
2709 break;
2710 case 2:
2711 *ppPageHeader = &pPages->ManufacturingPage2.u.fields.Header;
2712 *ppbPageData = pPages->ManufacturingPage2.u.abPageData;
2713 *pcbPage = sizeof(pPages->ManufacturingPage2);
2714 break;
2715 case 3:
2716 *ppPageHeader = &pPages->ManufacturingPage3.u.fields.Header;
2717 *ppbPageData = pPages->ManufacturingPage3.u.abPageData;
2718 *pcbPage = sizeof(pPages->ManufacturingPage3);
2719 break;
2720 case 4:
2721 *ppPageHeader = &pPages->ManufacturingPage4.u.fields.Header;
2722 *ppbPageData = pPages->ManufacturingPage4.u.abPageData;
2723 *pcbPage = sizeof(pPages->ManufacturingPage4);
2724 break;
2725 case 5:
2726 *ppPageHeader = &pPages->ManufacturingPage5.u.fields.Header;
2727 *ppbPageData = pPages->ManufacturingPage5.u.abPageData;
2728 *pcbPage = sizeof(pPages->ManufacturingPage5);
2729 break;
2730 case 6:
2731 *ppPageHeader = &pPages->ManufacturingPage6.u.fields.Header;
2732 *ppbPageData = pPages->ManufacturingPage6.u.abPageData;
2733 *pcbPage = sizeof(pPages->ManufacturingPage6);
2734 break;
2735 case 7:
2736 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
2737 {
2738 *ppPageHeader = &pPages->u.SasPages.pManufacturingPage7->u.fields.Header;
2739 *ppbPageData = pPages->u.SasPages.pManufacturingPage7->u.abPageData;
2740 *pcbPage = pPages->u.SasPages.cbManufacturingPage7;
2741 }
2742 else
2743 rc = VERR_NOT_FOUND;
2744 break;
2745 case 8:
2746 *ppPageHeader = &pPages->ManufacturingPage8.u.fields.Header;
2747 *ppbPageData = pPages->ManufacturingPage8.u.abPageData;
2748 *pcbPage = sizeof(pPages->ManufacturingPage8);
2749 break;
2750 case 9:
2751 *ppPageHeader = &pPages->ManufacturingPage9.u.fields.Header;
2752 *ppbPageData = pPages->ManufacturingPage9.u.abPageData;
2753 *pcbPage = sizeof(pPages->ManufacturingPage9);
2754 break;
2755 case 10:
2756 *ppPageHeader = &pPages->ManufacturingPage10.u.fields.Header;
2757 *ppbPageData = pPages->ManufacturingPage10.u.abPageData;
2758 *pcbPage = sizeof(pPages->ManufacturingPage10);
2759 break;
2760 default:
2761 rc = VERR_NOT_FOUND;
2762 }
2763
2764 return rc;
2765}
2766
2767/**
2768 * Return the configuration page header and data
2769 * which matches the given page type and number.
2770 *
2771 * @returns VINF_SUCCESS if successful
2772 * VERR_NOT_FOUND if the requested page could be found.
2773 * @param pThis Pointer to the shared LsiLogic device state. data.
2774 * @param pPages The pages supported by the controller.
2775 * @param u8PageNumber Number of the page to get.
2776 * @param ppPageHeader Where to store the pointer to the page header.
2777 * @param ppbPageData Where to store the pointer to the page data.
2778 * @param pcbPage Where to store the size of the page data in bytes on success.
2779 */
2780static int lsilogicR3ConfigurationBiosPageGetFromNumber(PLSILOGICSCSI pThis,
2781 PMptConfigurationPagesSupported pPages,
2782 uint8_t u8PageNumber,
2783 PMptConfigurationPageHeader *ppPageHeader,
2784 uint8_t **ppbPageData, size_t *pcbPage)
2785{
2786 RT_NOREF(pThis);
2787 int rc = VINF_SUCCESS;
2788
2789 AssertPtr(ppPageHeader); Assert(ppbPageData);
2790
2791 switch (u8PageNumber)
2792 {
2793 case 1:
2794 *ppPageHeader = &pPages->BIOSPage1.u.fields.Header;
2795 *ppbPageData = pPages->BIOSPage1.u.abPageData;
2796 *pcbPage = sizeof(pPages->BIOSPage1);
2797 break;
2798 case 2:
2799 *ppPageHeader = &pPages->BIOSPage2.u.fields.Header;
2800 *ppbPageData = pPages->BIOSPage2.u.abPageData;
2801 *pcbPage = sizeof(pPages->BIOSPage2);
2802 break;
2803 case 4:
2804 *ppPageHeader = &pPages->BIOSPage4.u.fields.Header;
2805 *ppbPageData = pPages->BIOSPage4.u.abPageData;
2806 *pcbPage = sizeof(pPages->BIOSPage4);
2807 break;
2808 default:
2809 rc = VERR_NOT_FOUND;
2810 }
2811
2812 return rc;
2813}
2814
2815/**
2816 * Return the configuration page header and data
2817 * which matches the given page type and number.
2818 *
2819 * @returns VINF_SUCCESS if successful
2820 * VERR_NOT_FOUND if the requested page could be found.
2821 * @param pThis Pointer to the shared LsiLogic device state. data.
2822 * @param pPages The pages supported by the controller.
2823 * @param u8Port The port to retrieve the page for.
2824 * @param u8PageNumber Number of the page to get.
2825 * @param ppPageHeader Where to store the pointer to the page header.
2826 * @param ppbPageData Where to store the pointer to the page data.
2827 * @param pcbPage Where to store the size of the page data in bytes on success.
2828 */
2829static int lsilogicR3ConfigurationSCSISPIPortPageGetFromNumber(PLSILOGICSCSI pThis,
2830 PMptConfigurationPagesSupported pPages,
2831 uint8_t u8Port,
2832 uint8_t u8PageNumber,
2833 PMptConfigurationPageHeader *ppPageHeader,
2834 uint8_t **ppbPageData, size_t *pcbPage)
2835{
2836 RT_NOREF(pThis);
2837 int rc = VINF_SUCCESS;
2838 AssertPtr(ppPageHeader); Assert(ppbPageData);
2839
2840
2841 if (u8Port >= RT_ELEMENTS(pPages->u.SpiPages.aPortPages))
2842 return VERR_NOT_FOUND;
2843
2844 switch (u8PageNumber)
2845 {
2846 case 0:
2847 *ppPageHeader = &pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage0.u.fields.Header;
2848 *ppbPageData = pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage0.u.abPageData;
2849 *pcbPage = sizeof(pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage0);
2850 break;
2851 case 1:
2852 *ppPageHeader = &pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage1.u.fields.Header;
2853 *ppbPageData = pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage1.u.abPageData;
2854 *pcbPage = sizeof(pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage1);
2855 break;
2856 case 2:
2857 *ppPageHeader = &pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage2.u.fields.Header;
2858 *ppbPageData = pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage2.u.abPageData;
2859 *pcbPage = sizeof(pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage2);
2860 break;
2861 default:
2862 rc = VERR_NOT_FOUND;
2863 }
2864
2865 return rc;
2866}
2867
2868/**
2869 * Return the configuration page header and data
2870 * which matches the given page type and number.
2871 *
2872 * @returns VINF_SUCCESS if successful
2873 * VERR_NOT_FOUND if the requested page could be found.
2874 * @param pThis Pointer to the shared LsiLogic device state. data.
2875 * @param pPages The pages supported by the controller.
2876 * @param u8Bus The bus the device is on the page should be returned.
2877 * @param u8TargetID The target ID of the device to return the page for.
2878 * @param u8PageNumber Number of the page to get.
2879 * @param ppPageHeader Where to store the pointer to the page header.
2880 * @param ppbPageData Where to store the pointer to the page data.
2881 * @param pcbPage Where to store the size of the page data in bytes on success.
2882 */
2883static int lsilogicR3ConfigurationSCSISPIDevicePageGetFromNumber(PLSILOGICSCSI pThis,
2884 PMptConfigurationPagesSupported pPages,
2885 uint8_t u8Bus,
2886 uint8_t u8TargetID, uint8_t u8PageNumber,
2887 PMptConfigurationPageHeader *ppPageHeader,
2888 uint8_t **ppbPageData, size_t *pcbPage)
2889{
2890 RT_NOREF(pThis);
2891 int rc = VINF_SUCCESS;
2892 AssertPtr(ppPageHeader); Assert(ppbPageData);
2893
2894 if (u8Bus >= RT_ELEMENTS(pPages->u.SpiPages.aBuses))
2895 return VERR_NOT_FOUND;
2896
2897 if (u8TargetID >= RT_ELEMENTS(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages))
2898 return VERR_NOT_FOUND;
2899
2900 switch (u8PageNumber)
2901 {
2902 case 0:
2903 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0.u.fields.Header;
2904 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0.u.abPageData;
2905 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0);
2906 break;
2907 case 1:
2908 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1.u.fields.Header;
2909 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1.u.abPageData;
2910 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1);
2911 break;
2912 case 2:
2913 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2.u.fields.Header;
2914 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2.u.abPageData;
2915 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2);
2916 break;
2917 case 3:
2918 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3.u.fields.Header;
2919 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3.u.abPageData;
2920 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3);
2921 break;
2922 default:
2923 rc = VERR_NOT_FOUND;
2924 }
2925
2926 return rc;
2927}
2928
2929static int lsilogicR3ConfigurationSASIOUnitPageGetFromNumber(PMptConfigurationPagesSupported pPages,
2930 uint8_t u8PageNumber,
2931 PMptExtendedConfigurationPageHeader *ppPageHeader,
2932 uint8_t **ppbPageData, size_t *pcbPage)
2933{
2934 int rc = VINF_SUCCESS;
2935
2936 switch (u8PageNumber)
2937 {
2938 case 0:
2939 *ppPageHeader = &pPages->u.SasPages.pSASIOUnitPage0->u.fields.ExtHeader;
2940 *ppbPageData = pPages->u.SasPages.pSASIOUnitPage0->u.abPageData;
2941 *pcbPage = pPages->u.SasPages.cbSASIOUnitPage0;
2942 break;
2943 case 1:
2944 *ppPageHeader = &pPages->u.SasPages.pSASIOUnitPage1->u.fields.ExtHeader;
2945 *ppbPageData = pPages->u.SasPages.pSASIOUnitPage1->u.abPageData;
2946 *pcbPage = pPages->u.SasPages.cbSASIOUnitPage1;
2947 break;
2948 case 2:
2949 *ppPageHeader = &pPages->u.SasPages.SASIOUnitPage2.u.fields.ExtHeader;
2950 *ppbPageData = pPages->u.SasPages.SASIOUnitPage2.u.abPageData;
2951 *pcbPage = sizeof(pPages->u.SasPages.SASIOUnitPage2);
2952 break;
2953 case 3:
2954 *ppPageHeader = &pPages->u.SasPages.SASIOUnitPage3.u.fields.ExtHeader;
2955 *ppbPageData = pPages->u.SasPages.SASIOUnitPage3.u.abPageData;
2956 *pcbPage = sizeof(pPages->u.SasPages.SASIOUnitPage3);
2957 break;
2958 default:
2959 rc = VERR_NOT_FOUND;
2960 }
2961
2962 return rc;
2963}
2964
2965static int lsilogicR3ConfigurationSASPHYPageGetFromNumber(PMptConfigurationPagesSupported pPages,
2966 uint8_t u8PageNumber,
2967 MptConfigurationPageAddress PageAddress,
2968 PMptExtendedConfigurationPageHeader *ppPageHeader,
2969 uint8_t **ppbPageData, size_t *pcbPage)
2970{
2971 int rc = VINF_SUCCESS;
2972 uint8_t uAddressForm = MPT_CONFIGURATION_PAGE_ADDRESS_GET_SAS_FORM(PageAddress);
2973 PMptConfigurationPagesSas pPagesSas = &pPages->u.SasPages;
2974 PMptPHY pPHYPages = NULL;
2975
2976 Log(("Address form %d\n", uAddressForm));
2977
2978 if (uAddressForm == 0) /* PHY number */
2979 {
2980 uint8_t u8PhyNumber = PageAddress.SASPHY.Form0.u8PhyNumber;
2981
2982 Log(("PHY number %d\n", u8PhyNumber));
2983
2984 if (u8PhyNumber >= pPagesSas->cPHYs)
2985 return VERR_NOT_FOUND;
2986
2987 pPHYPages = &pPagesSas->paPHYs[u8PhyNumber];
2988 }
2989 else if (uAddressForm == 1) /* Index form */
2990 {
2991 uint16_t u16Index = PageAddress.SASPHY.Form1.u16Index;
2992
2993 Log(("PHY index %d\n", u16Index));
2994
2995 if (u16Index >= pPagesSas->cPHYs)
2996 return VERR_NOT_FOUND;
2997
2998 pPHYPages = &pPagesSas->paPHYs[u16Index];
2999 }
3000 else
3001 rc = VERR_NOT_FOUND; /* Correct? */
3002
3003 if (pPHYPages)
3004 {
3005 switch (u8PageNumber)
3006 {
3007 case 0:
3008 *ppPageHeader = &pPHYPages->SASPHYPage0.u.fields.ExtHeader;
3009 *ppbPageData = pPHYPages->SASPHYPage0.u.abPageData;
3010 *pcbPage = sizeof(pPHYPages->SASPHYPage0);
3011 break;
3012 case 1:
3013 *ppPageHeader = &pPHYPages->SASPHYPage1.u.fields.ExtHeader;
3014 *ppbPageData = pPHYPages->SASPHYPage1.u.abPageData;
3015 *pcbPage = sizeof(pPHYPages->SASPHYPage1);
3016 break;
3017 default:
3018 rc = VERR_NOT_FOUND;
3019 }
3020 }
3021 else
3022 rc = VERR_NOT_FOUND;
3023
3024 return rc;
3025}
3026
3027static int lsilogicR3ConfigurationSASDevicePageGetFromNumber(PMptConfigurationPagesSupported pPages,
3028 uint8_t u8PageNumber,
3029 MptConfigurationPageAddress PageAddress,
3030 PMptExtendedConfigurationPageHeader *ppPageHeader,
3031 uint8_t **ppbPageData, size_t *pcbPage)
3032{
3033 int rc = VINF_SUCCESS;
3034 uint8_t uAddressForm = MPT_CONFIGURATION_PAGE_ADDRESS_GET_SAS_FORM(PageAddress);
3035 PMptConfigurationPagesSas pPagesSas = &pPages->u.SasPages;
3036 PMptSASDevice pSASDevice = NULL;
3037
3038 Log(("Address form %d\n", uAddressForm));
3039
3040 if (uAddressForm == 0)
3041 {
3042 uint16_t u16Handle = PageAddress.SASDevice.Form0And2.u16Handle;
3043
3044 Log(("Get next handle %#x\n", u16Handle));
3045
3046 pSASDevice = pPagesSas->pSASDeviceHead;
3047
3048 /* Get the first device? */
3049 if (u16Handle != 0xffff)
3050 {
3051 /* No, search for the right one. */
3052
3053 while ( pSASDevice
3054 && pSASDevice->SASDevicePage0.u.fields.u16DevHandle != u16Handle)
3055 pSASDevice = pSASDevice->pNext;
3056
3057 if (pSASDevice)
3058 pSASDevice = pSASDevice->pNext;
3059 }
3060 }
3061 else if (uAddressForm == 1)
3062 {
3063 uint8_t u8TargetID = PageAddress.SASDevice.Form1.u8TargetID;
3064 uint8_t u8Bus = PageAddress.SASDevice.Form1.u8Bus;
3065
3066 Log(("u8TargetID=%d u8Bus=%d\n", u8TargetID, u8Bus));
3067
3068 pSASDevice = pPagesSas->pSASDeviceHead;
3069
3070 while ( pSASDevice
3071 && ( pSASDevice->SASDevicePage0.u.fields.u8TargetID != u8TargetID
3072 || pSASDevice->SASDevicePage0.u.fields.u8Bus != u8Bus))
3073 pSASDevice = pSASDevice->pNext;
3074 }
3075 else if (uAddressForm == 2)
3076 {
3077 uint16_t u16Handle = PageAddress.SASDevice.Form0And2.u16Handle;
3078
3079 Log(("Handle %#x\n", u16Handle));
3080
3081 pSASDevice = pPagesSas->pSASDeviceHead;
3082
3083 while ( pSASDevice
3084 && pSASDevice->SASDevicePage0.u.fields.u16DevHandle != u16Handle)
3085 pSASDevice = pSASDevice->pNext;
3086 }
3087
3088 if (pSASDevice)
3089 {
3090 switch (u8PageNumber)
3091 {
3092 case 0:
3093 *ppPageHeader = &pSASDevice->SASDevicePage0.u.fields.ExtHeader;
3094 *ppbPageData = pSASDevice->SASDevicePage0.u.abPageData;
3095 *pcbPage = sizeof(pSASDevice->SASDevicePage0);
3096 break;
3097 case 1:
3098 *ppPageHeader = &pSASDevice->SASDevicePage1.u.fields.ExtHeader;
3099 *ppbPageData = pSASDevice->SASDevicePage1.u.abPageData;
3100 *pcbPage = sizeof(pSASDevice->SASDevicePage1);
3101 break;
3102 case 2:
3103 *ppPageHeader = &pSASDevice->SASDevicePage2.u.fields.ExtHeader;
3104 *ppbPageData = pSASDevice->SASDevicePage2.u.abPageData;
3105 *pcbPage = sizeof(pSASDevice->SASDevicePage2);
3106 break;
3107 default:
3108 rc = VERR_NOT_FOUND;
3109 }
3110 }
3111 else
3112 rc = VERR_NOT_FOUND;
3113
3114 return rc;
3115}
3116
3117/**
3118 * Returns the extended configuration page header and data.
3119 * @returns VINF_SUCCESS if successful
3120 * VERR_NOT_FOUND if the requested page could be found.
3121 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
3122 * @param pConfigurationReq The configuration request.
3123 * @param ppPageHeader Where to return the pointer to the page header on success.
3124 * @param ppbPageData Where to store the pointer to the page data.
3125 * @param pcbPage Where to store the size of the page in bytes.
3126 */
3127static int lsilogicR3ConfigurationPageGetExtended(PLSILOGICSCSICC pThisCC, PMptConfigurationRequest pConfigurationReq,
3128 PMptExtendedConfigurationPageHeader *ppPageHeader,
3129 uint8_t **ppbPageData, size_t *pcbPage)
3130{
3131 int rc = VINF_SUCCESS;
3132
3133 Log(("Extended page requested:\n"));
3134 Log(("u8ExtPageType=%#x\n", pConfigurationReq->u8ExtPageType));
3135 Log(("u8ExtPageLength=%d\n", pConfigurationReq->u16ExtPageLength));
3136
3137 switch (pConfigurationReq->u8ExtPageType)
3138 {
3139 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT:
3140 {
3141 rc = lsilogicR3ConfigurationSASIOUnitPageGetFromNumber(pThisCC->pConfigurationPages,
3142 pConfigurationReq->u8PageNumber,
3143 ppPageHeader, ppbPageData, pcbPage);
3144 break;
3145 }
3146 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASPHYS:
3147 {
3148 rc = lsilogicR3ConfigurationSASPHYPageGetFromNumber(pThisCC->pConfigurationPages,
3149 pConfigurationReq->u8PageNumber,
3150 pConfigurationReq->PageAddress,
3151 ppPageHeader, ppbPageData, pcbPage);
3152 break;
3153 }
3154 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE:
3155 {
3156 rc = lsilogicR3ConfigurationSASDevicePageGetFromNumber(pThisCC->pConfigurationPages,
3157 pConfigurationReq->u8PageNumber,
3158 pConfigurationReq->PageAddress,
3159 ppPageHeader, ppbPageData, pcbPage);
3160 break;
3161 }
3162 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASEXPANDER: /* No expanders supported */
3163 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_ENCLOSURE: /* No enclosures supported */
3164 default:
3165 rc = VERR_NOT_FOUND;
3166 }
3167
3168 return rc;
3169}
3170
3171/**
3172 * Processes a Configuration request.
3173 *
3174 * @returns VBox status code.
3175 * @param pDevIns The device instance.
3176 * @param pThis Pointer to the shared LsiLogic device state.
3177 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
3178 * @param pConfigurationReq Pointer to the request structure.
3179 * @param pReply Pointer to the reply message frame
3180 */
3181static int lsilogicR3ProcessConfigurationRequest(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC,
3182 PMptConfigurationRequest pConfigurationReq, PMptConfigurationReply pReply)
3183{
3184 int rc = VINF_SUCCESS;
3185 uint8_t *pbPageData = NULL;
3186 PMptConfigurationPageHeader pPageHeader = NULL;
3187 PMptExtendedConfigurationPageHeader pExtPageHeader = NULL;
3188 uint8_t u8PageType;
3189 uint8_t u8PageAttribute;
3190 size_t cbPage = 0;
3191
3192 LogFlowFunc(("pThis=%#p\n", pThis));
3193
3194 u8PageType = MPT_CONFIGURATION_PAGE_TYPE_GET(pConfigurationReq->u8PageType);
3195 u8PageAttribute = MPT_CONFIGURATION_PAGE_ATTRIBUTE_GET(pConfigurationReq->u8PageType);
3196
3197 Log(("GuestRequest:\n"));
3198 Log(("u8Action=%#x\n", pConfigurationReq->u8Action));
3199 Log(("u8PageType=%#x\n", u8PageType));
3200 Log(("u8PageNumber=%d\n", pConfigurationReq->u8PageNumber));
3201 Log(("u8PageLength=%d\n", pConfigurationReq->u8PageLength));
3202 Log(("u8PageVersion=%d\n", pConfigurationReq->u8PageVersion));
3203
3204 /* Copy common bits from the request into the reply. */
3205 pReply->u8MessageLength = 6; /* 6 32bit D-Words. */
3206 pReply->u8Action = pConfigurationReq->u8Action;
3207 pReply->u8Function = pConfigurationReq->u8Function;
3208 pReply->u32MessageContext = pConfigurationReq->u32MessageContext;
3209
3210 switch (u8PageType)
3211 {
3212 case MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT:
3213 {
3214 /* Get the page data. */
3215 rc = lsilogicR3ConfigurationIOUnitPageGetFromNumber(pThis,
3216 pThisCC->pConfigurationPages,
3217 pConfigurationReq->u8PageNumber,
3218 &pPageHeader, &pbPageData, &cbPage);
3219 break;
3220 }
3221 case MPT_CONFIGURATION_PAGE_TYPE_IOC:
3222 {
3223 /* Get the page data. */
3224 rc = lsilogicR3ConfigurationIOCPageGetFromNumber(pThis,
3225 pThisCC->pConfigurationPages,
3226 pConfigurationReq->u8PageNumber,
3227 &pPageHeader, &pbPageData, &cbPage);
3228 break;
3229 }
3230 case MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING:
3231 {
3232 /* Get the page data. */
3233 rc = lsilogicR3ConfigurationManufacturingPageGetFromNumber(pThis,
3234 pThisCC->pConfigurationPages,
3235 pConfigurationReq->u8PageNumber,
3236 &pPageHeader, &pbPageData, &cbPage);
3237 break;
3238 }
3239 case MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT:
3240 {
3241 /* Get the page data. */
3242 rc = lsilogicR3ConfigurationSCSISPIPortPageGetFromNumber(pThis,
3243 pThisCC->pConfigurationPages,
3244 pConfigurationReq->PageAddress.MPIPortNumber.u8PortNumber,
3245 pConfigurationReq->u8PageNumber,
3246 &pPageHeader, &pbPageData, &cbPage);
3247 break;
3248 }
3249 case MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE:
3250 {
3251 /* Get the page data. */
3252 rc = lsilogicR3ConfigurationSCSISPIDevicePageGetFromNumber(pThis,
3253 pThisCC->pConfigurationPages,
3254 pConfigurationReq->PageAddress.BusAndTargetId.u8Bus,
3255 pConfigurationReq->PageAddress.BusAndTargetId.u8TargetID,
3256 pConfigurationReq->u8PageNumber,
3257 &pPageHeader, &pbPageData, &cbPage);
3258 break;
3259 }
3260 case MPT_CONFIGURATION_PAGE_TYPE_BIOS:
3261 {
3262 rc = lsilogicR3ConfigurationBiosPageGetFromNumber(pThis,
3263 pThisCC->pConfigurationPages,
3264 pConfigurationReq->u8PageNumber,
3265 &pPageHeader, &pbPageData, &cbPage);
3266 break;
3267 }
3268 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED:
3269 {
3270 rc = lsilogicR3ConfigurationPageGetExtended(pThisCC,
3271 pConfigurationReq,
3272 &pExtPageHeader, &pbPageData, &cbPage);
3273 break;
3274 }
3275 default:
3276 rc = VERR_NOT_FOUND;
3277 }
3278
3279 if (rc == VERR_NOT_FOUND)
3280 {
3281 Log(("Page not found\n"));
3282 pReply->u8PageType = pConfigurationReq->u8PageType;
3283 pReply->u8PageNumber = pConfigurationReq->u8PageNumber;
3284 pReply->u8PageLength = pConfigurationReq->u8PageLength;
3285 pReply->u8PageVersion = pConfigurationReq->u8PageVersion;
3286 pReply->u16IOCStatus = MPT_IOCSTATUS_CONFIG_INVALID_PAGE;
3287 return VINF_SUCCESS;
3288 }
3289
3290 if (u8PageType == MPT_CONFIGURATION_PAGE_TYPE_EXTENDED)
3291 {
3292 pReply->u8PageType = pExtPageHeader->u8PageType;
3293 pReply->u8PageNumber = pExtPageHeader->u8PageNumber;
3294 pReply->u8PageVersion = pExtPageHeader->u8PageVersion;
3295 pReply->u8ExtPageType = pExtPageHeader->u8ExtPageType;
3296 pReply->u16ExtPageLength = pExtPageHeader->u16ExtPageLength;
3297
3298 for (int i = 0; i < pExtPageHeader->u16ExtPageLength; i++)
3299 LogFlowFunc(("PageData[%d]=%#x\n", i, ((uint32_t *)pbPageData)[i]));
3300 }
3301 else
3302 {
3303 pReply->u8PageType = pPageHeader->u8PageType;
3304 pReply->u8PageNumber = pPageHeader->u8PageNumber;
3305 pReply->u8PageLength = pPageHeader->u8PageLength;
3306 pReply->u8PageVersion = pPageHeader->u8PageVersion;
3307
3308 for (int i = 0; i < pReply->u8PageLength; i++)
3309 LogFlowFunc(("PageData[%d]=%#x\n", i, ((uint32_t *)pbPageData)[i]));
3310 }
3311
3312 /*
3313 * Don't use the scatter gather handling code as the configuration request always have only one
3314 * simple element.
3315 */
3316 switch (pConfigurationReq->u8Action)
3317 {
3318 case MPT_CONFIGURATION_REQUEST_ACTION_DEFAULT: /* Nothing to do. We are always using the defaults. */
3319 case MPT_CONFIGURATION_REQUEST_ACTION_HEADER:
3320 {
3321 /* Already copied above nothing to do. */
3322 break;
3323 }
3324 case MPT_CONFIGURATION_REQUEST_ACTION_READ_NVRAM:
3325 case MPT_CONFIGURATION_REQUEST_ACTION_READ_CURRENT:
3326 case MPT_CONFIGURATION_REQUEST_ACTION_READ_DEFAULT:
3327 {
3328 uint32_t cbBuffer = pConfigurationReq->SimpleSGElement.u24Length;
3329 if (cbBuffer != 0)
3330 {
3331 RTGCPHYS GCPhysAddrPageBuffer = pConfigurationReq->SimpleSGElement.u32DataBufferAddressLow;
3332 if (pConfigurationReq->SimpleSGElement.f64BitAddress)
3333 GCPhysAddrPageBuffer |= (uint64_t)pConfigurationReq->SimpleSGElement.u32DataBufferAddressHigh << 32;
3334
3335 PDMDevHlpPCIPhysWriteMeta(pDevIns, GCPhysAddrPageBuffer, pbPageData, RT_MIN(cbBuffer, cbPage));
3336 }
3337 break;
3338 }
3339 case MPT_CONFIGURATION_REQUEST_ACTION_WRITE_CURRENT:
3340 case MPT_CONFIGURATION_REQUEST_ACTION_WRITE_NVRAM:
3341 {
3342 uint32_t cbBuffer = pConfigurationReq->SimpleSGElement.u24Length;
3343 if (cbBuffer != 0)
3344 {
3345 RTGCPHYS GCPhysAddrPageBuffer = pConfigurationReq->SimpleSGElement.u32DataBufferAddressLow;
3346 if (pConfigurationReq->SimpleSGElement.f64BitAddress)
3347 GCPhysAddrPageBuffer |= (uint64_t)pConfigurationReq->SimpleSGElement.u32DataBufferAddressHigh << 32;
3348
3349 LogFlow(("cbBuffer=%u cbPage=%u\n", cbBuffer, cbPage));
3350
3351 PDMDevHlpPCIPhysReadMeta(pDevIns, GCPhysAddrPageBuffer, pbPageData, RT_MIN(cbBuffer, cbPage));
3352 }
3353 break;
3354 }
3355 default:
3356 AssertMsgFailed(("todo\n"));
3357 }
3358
3359 return VINF_SUCCESS;
3360}
3361
3362/**
3363 * Initializes the configuration pages for the SPI SCSI controller.
3364 *
3365 * @returns nothing
3366 * @param pThis Pointer to the shared LsiLogic device state.
3367 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
3368 */
3369static void lsilogicR3InitializeConfigurationPagesSpi(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
3370{
3371 PMptConfigurationPagesSpi pPages = &pThisCC->pConfigurationPages->u.SpiPages;
3372
3373 AssertMsg(pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI, ("Controller is not the SPI SCSI one\n"));
3374 LogFlowFunc(("pThis=%#p\n", pThis));
3375 RT_NOREF(pThis);
3376
3377 /* Clear everything first. */
3378 memset(pPages, 0, sizeof(MptConfigurationPagesSpi));
3379
3380 for (unsigned i = 0; i < RT_ELEMENTS(pPages->aPortPages); i++)
3381 {
3382 /* SCSI-SPI port page 0. */
3383 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3384 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
3385 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageNumber = 0;
3386 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort0) / 4;
3387 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fInformationUnitTransfersCapable = true;
3388 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fDTCapable = true;
3389 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fQASCapable = true;
3390 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u8MinimumSynchronousTransferPeriod = 0;
3391 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u8MaximumSynchronousOffset = 0xff;
3392 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fWide = true;
3393 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fAIPCapable = true;
3394 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u2SignalingType = 0x3; /* Single Ended. */
3395
3396 /* SCSI-SPI port page 1. */
3397 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
3398 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
3399 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageNumber = 1;
3400 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort1) / 4;
3401 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u8SCSIID = 7;
3402 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u16PortResponseIDsBitmask = (1 << 7);
3403 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u32OnBusTimerValue = 0;
3404
3405 /* SCSI-SPI port page 2. */
3406 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
3407 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
3408 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageNumber = 2;
3409 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort2) / 4;
3410 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.u4HostSCSIID = 7;
3411 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.u2InitializeHBA = 0x3;
3412 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.fTerminationDisabled = true;
3413 for (unsigned iDevice = 0; iDevice < RT_ELEMENTS(pPages->aPortPages[i].SCSISPIPortPage2.u.fields.aDeviceSettings); iDevice++)
3414 {
3415 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.aDeviceSettings[iDevice].fBootChoice = true;
3416 }
3417 /* Everything else 0 for now. */
3418 }
3419
3420 for (unsigned uBusCurr = 0; uBusCurr < RT_ELEMENTS(pPages->aBuses); uBusCurr++)
3421 {
3422 for (unsigned uDeviceCurr = 0; uDeviceCurr < RT_ELEMENTS(pPages->aBuses[uBusCurr].aDevicePages); uDeviceCurr++)
3423 {
3424 /* SCSI-SPI device page 0. */
3425 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3426 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
3427 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageNumber = 0;
3428 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice0) / 4;
3429 /* Everything else 0 for now. */
3430
3431 /* SCSI-SPI device page 1. */
3432 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
3433 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
3434 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageNumber = 1;
3435 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice1) / 4;
3436 /* Everything else 0 for now. */
3437
3438 /* SCSI-SPI device page 2. */
3439 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
3440 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
3441 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageNumber = 2;
3442 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice2) / 4;
3443 /* Everything else 0 for now. */
3444
3445 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3446 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
3447 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageNumber = 3;
3448 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice3) / 4;
3449 /* Everything else 0 for now. */
3450 }
3451 }
3452}
3453
3454/**
3455 * Generates a handle.
3456 *
3457 * @returns the handle.
3458 * @param pThis Pointer to the shared LsiLogic device state.
3459 */
3460DECLINLINE(uint16_t) lsilogicGetHandle(PLSILOGICSCSI pThis)
3461{
3462 uint16_t u16Handle = pThis->u16NextHandle++;
3463 return u16Handle;
3464}
3465
3466/**
3467 * Generates a SAS address (WWID)
3468 *
3469 * @returns nothing.
3470 * @param pSASAddress Pointer to an unitialised SAS address.
3471 * @param iId iId which will go into the address.
3472 *
3473 * @todo Generate better SAS addresses. (Request a block from SUN probably)
3474 */
3475void lsilogicSASAddressGenerate(PSASADDRESS pSASAddress, unsigned iId)
3476{
3477 pSASAddress->u8Address[0] = (0x5 << 5);
3478 pSASAddress->u8Address[1] = 0x01;
3479 pSASAddress->u8Address[2] = 0x02;
3480 pSASAddress->u8Address[3] = 0x03;
3481 pSASAddress->u8Address[4] = 0x04;
3482 pSASAddress->u8Address[5] = 0x05;
3483 pSASAddress->u8Address[6] = 0x06;
3484 pSASAddress->u8Address[7] = iId;
3485}
3486
3487/**
3488 * Initializes the configuration pages for the SAS SCSI controller.
3489 *
3490 * @returns nothing
3491 * @param pThis Pointer to the shared LsiLogic device state.
3492 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
3493 */
3494static void lsilogicR3InitializeConfigurationPagesSas(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
3495{
3496 PMptConfigurationPagesSas pPages = &pThisCC->pConfigurationPages->u.SasPages;
3497
3498 AssertMsg(pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS, ("Controller is not the SAS SCSI one\n"));
3499
3500 LogFlowFunc(("pThis=%#p\n", pThis));
3501
3502 /* Manufacturing Page 7 - Connector settings. */
3503 PMptConfigurationPageManufacturing7 pManufacturingPage7 = pPages->pManufacturingPage7;
3504 AssertPtr(pManufacturingPage7);
3505
3506 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(pManufacturingPage7,
3507 0, 7,
3508 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3509 /* Set size manually. */
3510 if (pPages->cbManufacturingPage7 / 4 > 255)
3511 pManufacturingPage7->u.fields.Header.u8PageLength = 255;
3512 else
3513 pManufacturingPage7->u.fields.Header.u8PageLength = pPages->cbManufacturingPage7 / 4;
3514 pManufacturingPage7->u.fields.u8NumPhys = pThis->cPorts;
3515
3516 /* SAS I/O unit page 0 - Port specific information. */
3517 PMptConfigurationPageSASIOUnit0 pSASPage0 = pPages->pSASIOUnitPage0;
3518 AssertPtr(pSASPage0);
3519
3520 MPT_CONFIG_EXTENDED_PAGE_HEADER_INIT(pSASPage0, pPages->cbSASIOUnitPage0,
3521 0, MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY,
3522 MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT);
3523 pSASPage0->u.fields.u8NumPhys = pThis->cPorts;
3524 pPages->pSASIOUnitPage0 = pSASPage0;
3525
3526 /* SAS I/O unit page 1 - Port specific settings. */
3527 PMptConfigurationPageSASIOUnit1 pSASPage1 = pPages->pSASIOUnitPage1;
3528 AssertPtr(pSASPage1);
3529
3530 MPT_CONFIG_EXTENDED_PAGE_HEADER_INIT(pSASPage1, pPages->cbSASIOUnitPage1,
3531 1, MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE,
3532 MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT);
3533 pSASPage1->u.fields.u8NumPhys = pSASPage0->u.fields.u8NumPhys;
3534 pSASPage1->u.fields.u16ControlFlags = 0;
3535 pSASPage1->u.fields.u16AdditionalControlFlags = 0;
3536
3537 /* SAS I/O unit page 2 - Port specific information. */
3538 pPages->SASIOUnitPage2.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3539 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3540 pPages->SASIOUnitPage2.u.fields.ExtHeader.u8PageNumber = 2;
3541 pPages->SASIOUnitPage2.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT;
3542 pPages->SASIOUnitPage2.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASIOUnit2) / 4;
3543
3544 /* SAS I/O unit page 3 - Port specific information. */
3545 pPages->SASIOUnitPage3.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3546 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3547 pPages->SASIOUnitPage3.u.fields.ExtHeader.u8PageNumber = 3;
3548 pPages->SASIOUnitPage3.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT;
3549 pPages->SASIOUnitPage3.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASIOUnit3) / 4;
3550
3551 Assert(pPages->cPHYs == pThis->cPorts);
3552 AssertPtr(pPages->paPHYs);
3553
3554 /* Initialize the PHY configuration */
3555 PMptSASDevice pSASDevice = pPages->pSASDeviceHead;
3556 for (unsigned i = 0; i < pThis->cPorts; i++)
3557 {
3558 PMptPHY pPHYPages = &pPages->paPHYs[i];
3559 uint16_t u16ControllerHandle = lsilogicGetHandle(pThis);
3560
3561 pManufacturingPage7->u.fields.aPHY[i].u8Location = LSILOGICSCSI_MANUFACTURING7_LOCATION_AUTO;
3562
3563 pSASPage0->u.fields.aPHY[i].u8Port = i;
3564 pSASPage0->u.fields.aPHY[i].u8PortFlags = 0;
3565 pSASPage0->u.fields.aPHY[i].u8PhyFlags = 0;
3566 pSASPage0->u.fields.aPHY[i].u8NegotiatedLinkRate = LSILOGICSCSI_SASIOUNIT0_NEGOTIATED_RATE_FAILED;
3567 pSASPage0->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_NO);
3568 pSASPage0->u.fields.aPHY[i].u16ControllerDevHandle = u16ControllerHandle;
3569 pSASPage0->u.fields.aPHY[i].u16AttachedDevHandle = 0; /* No device attached. */
3570 pSASPage0->u.fields.aPHY[i].u32DiscoveryStatus = 0; /* No errors */
3571
3572 pSASPage1->u.fields.aPHY[i].u8Port = i;
3573 pSASPage1->u.fields.aPHY[i].u8PortFlags = 0;
3574 pSASPage1->u.fields.aPHY[i].u8PhyFlags = 0;
3575 pSASPage1->u.fields.aPHY[i].u8MaxMinLinkRate = LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MIN_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_15GB)
3576 | LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MAX_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_30GB);
3577 pSASPage1->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_NO);
3578
3579 /* SAS PHY page 0. */
3580 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3581 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3582 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u8PageNumber = 0;
3583 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASPHYS;
3584 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASPHY0) / 4;
3585 pPHYPages->SASPHYPage0.u.fields.u8AttachedPhyIdentifier = i;
3586 pPHYPages->SASPHYPage0.u.fields.u32AttachedDeviceInfo = LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_SET(LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_NO);
3587 pPHYPages->SASPHYPage0.u.fields.u8ProgrammedLinkRate = LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MIN_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_15GB)
3588 | LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MAX_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_30GB);
3589 pPHYPages->SASPHYPage0.u.fields.u8HwLinkRate = LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MIN_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_15GB)
3590 | LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MAX_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_30GB);
3591
3592 /* SAS PHY page 1. */
3593 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3594 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3595 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u8PageNumber = 1;
3596 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASPHYS;
3597 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASPHY1) / 4;
3598
3599 /* Settings for present devices. */
3600 if (pThisCC->paDeviceStates[i].pDrvBase)
3601 {
3602 uint16_t u16DeviceHandle = lsilogicGetHandle(pThis);
3603 SASADDRESS SASAddress;
3604 AssertPtr(pSASDevice);
3605
3606 memset(&SASAddress, 0, sizeof(SASADDRESS));
3607 lsilogicSASAddressGenerate(&SASAddress, i);
3608
3609 pSASPage0->u.fields.aPHY[i].u8NegotiatedLinkRate = LSILOGICSCSI_SASIOUNIT0_NEGOTIATED_RATE_SET(LSILOGICSCSI_SASIOUNIT0_NEGOTIATED_RATE_30GB);
3610 pSASPage0->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_END)
3611 | LSILOGICSCSI_SASIOUNIT0_DEVICE_SSP_TARGET;
3612 pSASPage0->u.fields.aPHY[i].u16AttachedDevHandle = u16DeviceHandle;
3613 pSASPage1->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_END)
3614 | LSILOGICSCSI_SASIOUNIT0_DEVICE_SSP_TARGET;
3615 pSASPage0->u.fields.aPHY[i].u16ControllerDevHandle = u16DeviceHandle;
3616
3617 pPHYPages->SASPHYPage0.u.fields.u32AttachedDeviceInfo = LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_SET(LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_END);
3618 pPHYPages->SASPHYPage0.u.fields.SASAddress = SASAddress;
3619 pPHYPages->SASPHYPage0.u.fields.u16OwnerDevHandle = u16DeviceHandle;
3620 pPHYPages->SASPHYPage0.u.fields.u16AttachedDevHandle = u16DeviceHandle;
3621
3622 /* SAS device page 0. */
3623 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3624 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3625 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u8PageNumber = 0;
3626 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE;
3627 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASDevice0) / 4;
3628 pSASDevice->SASDevicePage0.u.fields.SASAddress = SASAddress;
3629 pSASDevice->SASDevicePage0.u.fields.u16ParentDevHandle = u16ControllerHandle;
3630 pSASDevice->SASDevicePage0.u.fields.u8PhyNum = i;
3631 pSASDevice->SASDevicePage0.u.fields.u8AccessStatus = LSILOGICSCSI_SASDEVICE0_STATUS_NO_ERRORS;
3632 pSASDevice->SASDevicePage0.u.fields.u16DevHandle = u16DeviceHandle;
3633 pSASDevice->SASDevicePage0.u.fields.u8TargetID = i;
3634 pSASDevice->SASDevicePage0.u.fields.u8Bus = 0;
3635 pSASDevice->SASDevicePage0.u.fields.u32DeviceInfo = LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_SET(LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_END)
3636 | LSILOGICSCSI_SASIOUNIT0_DEVICE_SSP_TARGET;
3637 pSASDevice->SASDevicePage0.u.fields.u16Flags = LSILOGICSCSI_SASDEVICE0_FLAGS_DEVICE_PRESENT
3638 | LSILOGICSCSI_SASDEVICE0_FLAGS_DEVICE_MAPPED_TO_BUS_AND_TARGET_ID
3639 | LSILOGICSCSI_SASDEVICE0_FLAGS_DEVICE_MAPPING_PERSISTENT;
3640 pSASDevice->SASDevicePage0.u.fields.u8PhysicalPort = i;
3641
3642 /* SAS device page 1. */
3643 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3644 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3645 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u8PageNumber = 1;
3646 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE;
3647 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASDevice1) / 4;
3648 pSASDevice->SASDevicePage1.u.fields.SASAddress = SASAddress;
3649 pSASDevice->SASDevicePage1.u.fields.u16DevHandle = u16DeviceHandle;
3650 pSASDevice->SASDevicePage1.u.fields.u8TargetID = i;
3651 pSASDevice->SASDevicePage1.u.fields.u8Bus = 0;
3652
3653 /* SAS device page 2. */
3654 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3655 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3656 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u8PageNumber = 2;
3657 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE;
3658 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASDevice2) / 4;
3659 pSASDevice->SASDevicePage2.u.fields.SASAddress = SASAddress;
3660
3661 pSASDevice = pSASDevice->pNext;
3662 }
3663 }
3664}
3665
3666/**
3667 * Initializes the configuration pages.
3668 *
3669 * @returns nothing
3670 * @param pDevIns The device instance.
3671 * @param pThis Pointer to the shared LsiLogic device state.
3672 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
3673 */
3674static void lsilogicR3InitializeConfigurationPages(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
3675{
3676 /* Initialize the common pages. */
3677
3678 LogFlowFunc(("pThis=%#p\n", pThis));
3679
3680 AssertPtrReturnVoid(pThisCC->pConfigurationPages);
3681 PMptConfigurationPagesSupported pPages = pThisCC->pConfigurationPages;
3682
3683 /* Manufacturing Page 0. */
3684 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage0,
3685 MptConfigurationPageManufacturing0, 0,
3686 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3687 strncpy((char *)pPages->ManufacturingPage0.u.fields.abChipName, "VBox MPT Fusion", 16);
3688 strncpy((char *)pPages->ManufacturingPage0.u.fields.abChipRevision, "1.0", 8);
3689 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardName, "VBox MPT Fusion", 16);
3690 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardAssembly, "SUN", 8);
3691 memcpy(pPages->ManufacturingPage0.u.fields.abBoardTracerNumber, "CAFECAFECAFECAFE", 16);
3692
3693 /* Manufacturing Page 1 - I don't know what this contains so we leave it 0 for now. */
3694 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage1,
3695 MptConfigurationPageManufacturing1, 1,
3696 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3697
3698 /* Manufacturing Page 2. */
3699 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage2,
3700 MptConfigurationPageManufacturing2, 2,
3701 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3702
3703 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3704 {
3705 pPages->ManufacturingPage2.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SPI_DEVICE_ID;
3706 pPages->ManufacturingPage2.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SPI_REVISION_ID;
3707 }
3708 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3709 {
3710 pPages->ManufacturingPage2.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SAS_DEVICE_ID;
3711 pPages->ManufacturingPage2.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SAS_REVISION_ID;
3712 }
3713
3714 /* Manufacturing Page 3. */
3715 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage3,
3716 MptConfigurationPageManufacturing3, 3,
3717 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3718
3719 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3720 {
3721 pPages->ManufacturingPage3.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SPI_DEVICE_ID;
3722 pPages->ManufacturingPage3.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SPI_REVISION_ID;
3723 }
3724 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3725 {
3726 pPages->ManufacturingPage3.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SAS_DEVICE_ID;
3727 pPages->ManufacturingPage3.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SAS_REVISION_ID;
3728 }
3729
3730 /* Manufacturing Page 4 - I don't know what this contains so we leave it 0 for now. */
3731 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage4,
3732 MptConfigurationPageManufacturing4, 4,
3733 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3734
3735 /* Manufacturing Page 5 - WWID settings. */
3736 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage5,
3737 MptConfigurationPageManufacturing5, 5,
3738 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3739
3740 /* Manufacturing Page 6 - Product specific settings. */
3741 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage6,
3742 MptConfigurationPageManufacturing6, 6,
3743 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3744
3745 /* Manufacturing Page 8 - Product specific settings. */
3746 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage8,
3747 MptConfigurationPageManufacturing8, 8,
3748 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3749
3750 /* Manufacturing Page 9 - Product specific settings. */
3751 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage9,
3752 MptConfigurationPageManufacturing9, 9,
3753 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3754
3755 /* Manufacturing Page 10 - Product specific settings. */
3756 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage10,
3757 MptConfigurationPageManufacturing10, 10,
3758 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3759
3760 /* I/O Unit page 0. */
3761 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage0,
3762 MptConfigurationPageIOUnit0, 0,
3763 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3764 pPages->IOUnitPage0.u.fields.u64UniqueIdentifier = 0xcafe;
3765
3766 /* I/O Unit page 1. */
3767 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage1,
3768 MptConfigurationPageIOUnit1, 1,
3769 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3770 pPages->IOUnitPage1.u.fields.fSingleFunction = true;
3771 pPages->IOUnitPage1.u.fields.fAllPathsMapped = false;
3772 pPages->IOUnitPage1.u.fields.fIntegratedRAIDDisabled = true;
3773 pPages->IOUnitPage1.u.fields.f32BitAccessForced = false;
3774
3775 /* I/O Unit page 2. */
3776 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage2,
3777 MptConfigurationPageIOUnit2, 2,
3778 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT);
3779 pPages->IOUnitPage2.u.fields.fPauseOnError = false;
3780 pPages->IOUnitPage2.u.fields.fVerboseModeEnabled = false;
3781 pPages->IOUnitPage2.u.fields.fDisableColorVideo = false;
3782 pPages->IOUnitPage2.u.fields.fNotHookInt40h = false;
3783 pPages->IOUnitPage2.u.fields.u32BIOSVersion = 0xcafecafe;
3784 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].fAdapterEnabled = true;
3785 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].fAdapterEmbedded = true;
3786 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].u8PCIBusNumber = 0;
3787 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].u8PCIDevFn = pDevIns->apPciDevs[0]->uDevFn;
3788
3789 /* I/O Unit page 3. */
3790 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage3,
3791 MptConfigurationPageIOUnit3, 3,
3792 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3793 pPages->IOUnitPage3.u.fields.u8GPIOCount = 0;
3794
3795 /* I/O Unit page 4. */
3796 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage4,
3797 MptConfigurationPageIOUnit4, 4,
3798 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3799
3800 /* IOC page 0. */
3801 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage0,
3802 MptConfigurationPageIOC0, 0,
3803 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3804 pPages->IOCPage0.u.fields.u32TotalNVStore = 0;
3805 pPages->IOCPage0.u.fields.u32FreeNVStore = 0;
3806
3807 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3808 {
3809 pPages->IOCPage0.u.fields.u16VendorId = LSILOGICSCSI_PCI_VENDOR_ID;
3810 pPages->IOCPage0.u.fields.u16DeviceId = LSILOGICSCSI_PCI_SPI_DEVICE_ID;
3811 pPages->IOCPage0.u.fields.u8RevisionId = LSILOGICSCSI_PCI_SPI_REVISION_ID;
3812 pPages->IOCPage0.u.fields.u32ClassCode = LSILOGICSCSI_PCI_SPI_CLASS_CODE;
3813 pPages->IOCPage0.u.fields.u16SubsystemVendorId = LSILOGICSCSI_PCI_SPI_SUBSYSTEM_VENDOR_ID;
3814 pPages->IOCPage0.u.fields.u16SubsystemId = LSILOGICSCSI_PCI_SPI_SUBSYSTEM_ID;
3815 }
3816 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3817 {
3818 pPages->IOCPage0.u.fields.u16VendorId = LSILOGICSCSI_PCI_VENDOR_ID;
3819 pPages->IOCPage0.u.fields.u16DeviceId = LSILOGICSCSI_PCI_SAS_DEVICE_ID;
3820 pPages->IOCPage0.u.fields.u8RevisionId = LSILOGICSCSI_PCI_SAS_REVISION_ID;
3821 pPages->IOCPage0.u.fields.u32ClassCode = LSILOGICSCSI_PCI_SAS_CLASS_CODE;
3822 pPages->IOCPage0.u.fields.u16SubsystemVendorId = LSILOGICSCSI_PCI_SAS_SUBSYSTEM_VENDOR_ID;
3823 pPages->IOCPage0.u.fields.u16SubsystemId = LSILOGICSCSI_PCI_SAS_SUBSYSTEM_ID;
3824 }
3825
3826 /* IOC page 1. */
3827 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage1,
3828 MptConfigurationPageIOC1, 1,
3829 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3830 pPages->IOCPage1.u.fields.fReplyCoalescingEnabled = false;
3831 pPages->IOCPage1.u.fields.u32CoalescingTimeout = 0;
3832 pPages->IOCPage1.u.fields.u8CoalescingDepth = 0;
3833
3834 /* IOC page 2. */
3835 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage2,
3836 MptConfigurationPageIOC2, 2,
3837 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3838 /* Everything else here is 0. */
3839
3840 /* IOC page 3. */
3841 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage3,
3842 MptConfigurationPageIOC3, 3,
3843 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3844 /* Everything else here is 0. */
3845
3846 /* IOC page 4. */
3847 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage4,
3848 MptConfigurationPageIOC4, 4,
3849 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3850 /* Everything else here is 0. */
3851
3852 /* IOC page 6. */
3853 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage6,
3854 MptConfigurationPageIOC6, 6,
3855 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3856 /* Everything else here is 0. */
3857
3858 /* BIOS page 1. */
3859 MPT_CONFIG_PAGE_HEADER_INIT_BIOS(&pPages->BIOSPage1,
3860 MptConfigurationPageBIOS1, 1,
3861 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3862
3863 /* BIOS page 2. */
3864 MPT_CONFIG_PAGE_HEADER_INIT_BIOS(&pPages->BIOSPage2,
3865 MptConfigurationPageBIOS2, 2,
3866 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3867
3868 /* BIOS page 4. */
3869 MPT_CONFIG_PAGE_HEADER_INIT_BIOS(&pPages->BIOSPage4,
3870 MptConfigurationPageBIOS4, 4,
3871 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3872
3873 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3874 lsilogicR3InitializeConfigurationPagesSpi(pThis, pThisCC);
3875 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3876 lsilogicR3InitializeConfigurationPagesSas(pThis, pThisCC);
3877 else
3878 AssertMsgFailed(("Invalid controller type %d\n", pThis->enmCtrlType));
3879}
3880
3881/**
3882 * Sets the emulated controller type from a given string.
3883 *
3884 * @returns VBox status code.
3885 *
3886 * @param pThis Pointer to the shared LsiLogic device state.
3887 * @param pcszCtrlType The string to use.
3888 */
3889static int lsilogicR3GetCtrlTypeFromString(PLSILOGICSCSI pThis, const char *pcszCtrlType)
3890{
3891 int rc = VERR_INVALID_PARAMETER;
3892
3893 if (!RTStrCmp(pcszCtrlType, LSILOGICSCSI_PCI_SPI_CTRLNAME))
3894 {
3895 pThis->enmCtrlType = LSILOGICCTRLTYPE_SCSI_SPI;
3896 rc = VINF_SUCCESS;
3897 }
3898 else if (!RTStrCmp(pcszCtrlType, LSILOGICSCSI_PCI_SAS_CTRLNAME))
3899 {
3900 pThis->enmCtrlType = LSILOGICCTRLTYPE_SCSI_SAS;
3901 rc = VINF_SUCCESS;
3902 }
3903
3904 return rc;
3905}
3906
3907/**
3908 * @callback_method_impl{FNIOMIOPORTIN, Legacy ISA port.}
3909 */
3910static DECLCALLBACK(VBOXSTRICTRC)
3911lsilogicR3IsaIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb)
3912{
3913 RT_NOREF(pvUser, cb);
3914 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
3915
3916 ASSERT_GUEST(cb == 1);
3917
3918 int rc = vboxscsiReadRegister(&pThisCC->VBoxSCSI, offPort, pu32);
3919 AssertMsg(rc == VINF_SUCCESS, ("Unexpected BIOS register read status: %Rrc\n", rc));
3920
3921 Log2(("%s: pu32=%p:{%.*Rhxs} offPort=%d rc=%Rrc\n", __FUNCTION__, pu32, 1, pu32, offPort, rc));
3922
3923 return rc;
3924}
3925
3926/**
3927 * Prepares a request from the BIOS.
3928 *
3929 * @returns VBox status code.
3930 * @param pThis Pointer to the shared LsiLogic device state.
3931 * @param pThisCC Pointer to the ring-3 LsiLogic device state.
3932 */
3933static int lsilogicR3PrepareBiosScsiRequest(PLSILOGICSCSI pThis, PLSILOGICSCSICC pThisCC)
3934{
3935 int rc;
3936 uint32_t uTargetDevice;
3937 uint32_t uLun;
3938 uint8_t *pbCdb;
3939 size_t cbCdb;
3940 size_t cbBuf;
3941
3942 rc = vboxscsiSetupRequest(&pThisCC->VBoxSCSI, &uLun, &pbCdb, &cbCdb, &cbBuf, &uTargetDevice);
3943 AssertMsgRCReturn(rc, ("Setting up SCSI request failed rc=%Rrc\n", rc), rc);
3944
3945 if ( uTargetDevice < pThis->cDeviceStates
3946 && pThisCC->paDeviceStates[uTargetDevice].pDrvBase)
3947 {
3948 PLSILOGICDEVICE pTgtDev = &pThisCC->paDeviceStates[uTargetDevice];
3949 PDMMEDIAEXIOREQ hIoReq;
3950 PLSILOGICREQ pReq;
3951
3952 rc = pTgtDev->pDrvMediaEx->pfnIoReqAlloc(pTgtDev->pDrvMediaEx, &hIoReq, (void **)&pReq,
3953 0, PDMIMEDIAEX_F_SUSPEND_ON_RECOVERABLE_ERR);
3954 AssertMsgRCReturn(rc, ("Getting task from cache failed rc=%Rrc\n", rc), rc);
3955
3956 pReq->fBIOS = true;
3957 pReq->hIoReq = hIoReq;
3958 pReq->pTargetDevice = pTgtDev;
3959
3960 ASMAtomicIncU32(&pTgtDev->cOutstandingRequests);
3961
3962 rc = pTgtDev->pDrvMediaEx->pfnIoReqSendScsiCmd(pTgtDev->pDrvMediaEx, pReq->hIoReq, uLun,
3963 pbCdb, cbCdb, PDMMEDIAEXIOREQSCSITXDIR_UNKNOWN, NULL,
3964 cbBuf, NULL, 0, NULL, &pReq->u8ScsiSts, 30 * RT_MS_1SEC);
3965 if (rc == VINF_SUCCESS || rc != VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS)
3966 {
3967 uint8_t u8ScsiSts = pReq->u8ScsiSts;
3968 pTgtDev->pDrvMediaEx->pfnIoReqFree(pTgtDev->pDrvMediaEx, pReq->hIoReq);
3969 rc = vboxscsiRequestFinished(&pThisCC->VBoxSCSI, u8ScsiSts);
3970 }
3971 else if (rc == VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS)
3972 rc = VINF_SUCCESS;
3973
3974 return rc;
3975 }
3976
3977 /* Device is not present. */
3978 AssertMsg(pbCdb[0] == SCSI_INQUIRY,
3979 ("Device is not present but command is not inquiry\n"));
3980
3981 SCSIINQUIRYDATA ScsiInquiryData;
3982
3983 memset(&ScsiInquiryData, 0, sizeof(SCSIINQUIRYDATA));
3984 ScsiInquiryData.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_UNKNOWN;
3985 ScsiInquiryData.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_NOT_CONNECTED_NOT_SUPPORTED;
3986
3987 memcpy(pThisCC->VBoxSCSI.pbBuf, &ScsiInquiryData, 5);
3988
3989 rc = vboxscsiRequestFinished(&pThisCC->VBoxSCSI, SCSI_STATUS_OK);
3990 AssertMsgRCReturn(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc), rc);
3991
3992 return rc;
3993}
3994
3995/**
3996 * @callback_method_impl{FNIOMIOPORTNEWOUT, Legacy ISA port.}
3997 */
3998static DECLCALLBACK(VBOXSTRICTRC)
3999lsilogicR3IsaIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb)
4000{
4001 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4002 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4003 Log2(("#%d %s: pvUser=%#p cb=%d u32=%#x offPort=%#x\n", pDevIns->iInstance, __FUNCTION__, pvUser, cb, u32, offPort));
4004 RT_NOREF(pvUser, cb);
4005
4006 ASSERT_GUEST(cb == 1);
4007
4008 /*
4009 * If there is already a request form the BIOS pending ignore this write
4010 * because it should not happen.
4011 */
4012 if (ASMAtomicReadBool(&pThis->fBiosReqPending))
4013 return VINF_SUCCESS;
4014
4015 int rc = vboxscsiWriteRegister(&pThisCC->VBoxSCSI, offPort, (uint8_t)u32);
4016 if (rc == VERR_MORE_DATA)
4017 {
4018 ASMAtomicXchgBool(&pThis->fBiosReqPending, true);
4019 /* Notify the worker thread that there are pending requests. */
4020 LogFlowFunc(("Signal event semaphore\n"));
4021 rc = PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEvtProcess);
4022 AssertRC(rc);
4023 }
4024 else
4025 AssertMsg(rc == VINF_SUCCESS, ("Unexpected BIOS register write status: %Rrc\n", rc));
4026
4027 return VINF_SUCCESS;
4028}
4029
4030/**
4031 * @callback_method_impl{FNIOMIOPORTNEWOUTSTRING,
4032 * Port I/O Handler for primary port range OUT string operations.}
4033 */
4034static DECLCALLBACK(VBOXSTRICTRC) lsilogicR3IsaIOPortWriteStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort,
4035 uint8_t const *pbSrc, uint32_t *pcTransfers, unsigned cb)
4036{
4037 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4038 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4039 Log2(("#%d %s: pvUser=%#p cb=%d offPort=%#x\n", pDevIns->iInstance, __FUNCTION__, pvUser, cb, offPort));
4040 RT_NOREF(pvUser);
4041
4042 int rc = vboxscsiWriteString(pDevIns, &pThisCC->VBoxSCSI, offPort, pbSrc, pcTransfers, cb);
4043 if (rc == VERR_MORE_DATA)
4044 {
4045 ASMAtomicXchgBool(&pThis->fBiosReqPending, true);
4046 /* Notify the worker thread that there are pending requests. */
4047 LogFlowFunc(("Signal event semaphore\n"));
4048 rc = PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEvtProcess);
4049 AssertRC(rc);
4050 }
4051 else
4052 AssertMsg(rc == VINF_SUCCESS, ("Unexpected BIOS register write status: %Rrc\n", rc));
4053
4054 return VINF_SUCCESS;
4055}
4056
4057/**
4058 * @callback_method_impl{FNIOMIOPORTINSTRING,
4059 * Port I/O Handler for primary port range IN string operations.}
4060 */
4061static DECLCALLBACK(VBOXSTRICTRC) lsilogicR3IsaIOPortReadStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort,
4062 uint8_t *pbDst, uint32_t *pcTransfers, unsigned cb)
4063{
4064 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4065 LogFlowFunc(("#%d %s: pvUser=%#p cb=%d offPort=%#x\n", pDevIns->iInstance, __FUNCTION__, pvUser, cb, offPort));
4066 RT_NOREF(pvUser);
4067
4068 int rc = vboxscsiReadString(pDevIns, &pThisCC->VBoxSCSI, offPort, pbDst, pcTransfers, cb);
4069 AssertMsg(rc == VINF_SUCCESS, ("Unexpected BIOS register read status: %Rrc\n", rc));
4070 return rc;
4071}
4072
4073/**
4074 * @callback_method_impl{PFNDBGFHANDLERDEV}
4075 */
4076static DECLCALLBACK(void) lsilogicR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
4077{
4078 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4079 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4080
4081 /*
4082 * Parse args.
4083 */
4084 bool const fVerbose = pszArgs && strstr(pszArgs, "verbose") != NULL;
4085
4086 /*
4087 * Show info.
4088 */
4089 pHlp->pfnPrintf(pHlp,
4090 "%s#%d: port=%04x mmio=%RGp max-devices=%u GC=%RTbool R0=%RTbool\n",
4091 pDevIns->pReg->szName, pDevIns->iInstance,
4092 PDMDevHlpIoPortGetMappingAddress(pDevIns, pThis->hIoPortsReg),
4093 PDMDevHlpMmioGetMappingAddress(pDevIns, pThis->hMmioReg),
4094 pThis->cDeviceStates, pDevIns->fRCEnabled, pDevIns->fR0Enabled);
4095
4096 /*
4097 * Show general state.
4098 */
4099 pHlp->pfnPrintf(pHlp, "enmState=%u\n", pThis->enmState);
4100 pHlp->pfnPrintf(pHlp, "enmWhoInit=%u\n", pThis->enmWhoInit);
4101 pHlp->pfnPrintf(pHlp, "enmDoorbellState=%d\n", pThis->enmDoorbellState);
4102 pHlp->pfnPrintf(pHlp, "fDiagnosticEnabled=%RTbool\n", pThis->fDiagnosticEnabled);
4103 pHlp->pfnPrintf(pHlp, "fNotificationSent=%RTbool\n", pThis->fNotificationSent);
4104 pHlp->pfnPrintf(pHlp, "fEventNotificationEnabled=%RTbool\n", pThis->fEventNotificationEnabled);
4105 pHlp->pfnPrintf(pHlp, "uInterruptMask=%#x\n", pThis->uInterruptMask);
4106 pHlp->pfnPrintf(pHlp, "uInterruptStatus=%#x\n", pThis->uInterruptStatus);
4107 pHlp->pfnPrintf(pHlp, "u16IOCFaultCode=%#06x\n", pThis->u16IOCFaultCode);
4108 pHlp->pfnPrintf(pHlp, "u32HostMFAHighAddr=%#x\n", pThis->u32HostMFAHighAddr);
4109 pHlp->pfnPrintf(pHlp, "u32SenseBufferHighAddr=%#x\n", pThis->u32SenseBufferHighAddr);
4110 pHlp->pfnPrintf(pHlp, "cMaxDevices=%u\n", pThis->cMaxDevices);
4111 pHlp->pfnPrintf(pHlp, "cMaxBuses=%u\n", pThis->cMaxBuses);
4112 pHlp->pfnPrintf(pHlp, "cbReplyFrame=%u\n", pThis->cbReplyFrame);
4113 pHlp->pfnPrintf(pHlp, "cReplyQueueEntries=%u\n", pThis->cReplyQueueEntries);
4114 pHlp->pfnPrintf(pHlp, "cRequestQueueEntries=%u\n", pThis->cRequestQueueEntries);
4115 pHlp->pfnPrintf(pHlp, "cPorts=%u\n", pThis->cPorts);
4116
4117 /*
4118 * Show queue status.
4119 */
4120 pHlp->pfnPrintf(pHlp, "uReplyFreeQueueNextEntryFreeWrite=%u\n", pThis->uReplyFreeQueueNextEntryFreeWrite);
4121 pHlp->pfnPrintf(pHlp, "uReplyFreeQueueNextAddressRead=%u\n", pThis->uReplyFreeQueueNextAddressRead);
4122 pHlp->pfnPrintf(pHlp, "uReplyPostQueueNextEntryFreeWrite=%u\n", pThis->uReplyPostQueueNextEntryFreeWrite);
4123 pHlp->pfnPrintf(pHlp, "uReplyPostQueueNextAddressRead=%u\n", pThis->uReplyPostQueueNextAddressRead);
4124 pHlp->pfnPrintf(pHlp, "uRequestQueueNextEntryFreeWrite=%u\n", pThis->uRequestQueueNextEntryFreeWrite);
4125 pHlp->pfnPrintf(pHlp, "uRequestQueueNextAddressRead=%u\n", pThis->uRequestQueueNextAddressRead);
4126
4127 /*
4128 * Show queue content if verbose
4129 */
4130 if (fVerbose)
4131 {
4132 for (unsigned i = 0; i < pThis->cReplyQueueEntries; i++)
4133 pHlp->pfnPrintf(pHlp, "RFQ[%u]=%#x\n", i, pThis->aReplyFreeQueue[i]);
4134
4135 pHlp->pfnPrintf(pHlp, "\n");
4136
4137 for (unsigned i = 0; i < pThis->cReplyQueueEntries; i++)
4138 pHlp->pfnPrintf(pHlp, "RPQ[%u]=%#x\n", i, pThis->aReplyPostQueue[i]);
4139
4140 pHlp->pfnPrintf(pHlp, "\n");
4141
4142 for (unsigned i = 0; i < pThis->cRequestQueueEntries; i++)
4143 pHlp->pfnPrintf(pHlp, "ReqQ[%u]=%#x\n", i, pThis->aRequestQueue[i]);
4144 }
4145
4146 /*
4147 * Print the device status.
4148 */
4149 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
4150 {
4151 PLSILOGICDEVICE pDevice = &pThisCC->paDeviceStates[i];
4152
4153 pHlp->pfnPrintf(pHlp, "\n");
4154
4155 pHlp->pfnPrintf(pHlp, "Device[%u]: device-attached=%RTbool cOutstandingRequests=%u\n",
4156 i, pDevice->pDrvBase != NULL, pDevice->cOutstandingRequests);
4157 }
4158}
4159
4160
4161/**
4162 * @callback_method_impl{FNPDMTHREADDEV}
4163 */
4164static DECLCALLBACK(int) lsilogicR3Worker(PPDMDEVINS pDevIns, PPDMTHREAD pThread)
4165{
4166 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4167 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4168 int rc = VINF_SUCCESS;
4169
4170 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
4171 return VINF_SUCCESS;
4172
4173 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
4174 {
4175 ASMAtomicWriteBool(&pThis->fWrkThreadSleeping, true);
4176 bool fNotificationSent = ASMAtomicXchgBool(&pThis->fNotificationSent, false);
4177 if (!fNotificationSent)
4178 {
4179 Assert(ASMAtomicReadBool(&pThis->fWrkThreadSleeping));
4180 rc = PDMDevHlpSUPSemEventWaitNoResume(pDevIns, pThis->hEvtProcess, RT_INDEFINITE_WAIT);
4181 AssertLogRelMsgReturn(RT_SUCCESS(rc) || rc == VERR_INTERRUPTED, ("%Rrc\n", rc), rc);
4182 if (RT_UNLIKELY(pThread->enmState != PDMTHREADSTATE_RUNNING))
4183 break;
4184 LogFlowFunc(("Woken up with rc=%Rrc\n", rc));
4185 ASMAtomicWriteBool(&pThis->fNotificationSent, false);
4186 }
4187
4188 ASMAtomicWriteBool(&pThis->fWrkThreadSleeping, false);
4189
4190 /* Check whether there is a BIOS request pending and process it first. */
4191 if (ASMAtomicReadBool(&pThis->fBiosReqPending))
4192 {
4193 rc = lsilogicR3PrepareBiosScsiRequest(pThis, pThisCC);
4194 AssertRC(rc);
4195 ASMAtomicXchgBool(&pThis->fBiosReqPending, false);
4196 }
4197
4198 /* Only process request which arrived before we received the notification. */
4199 uint32_t uRequestQueueNextEntryWrite = ASMAtomicReadU32(&pThis->uRequestQueueNextEntryFreeWrite);
4200
4201 /* Go through the messages now and process them. */
4202 while ( RT_LIKELY(pThis->enmState == LSILOGICSTATE_OPERATIONAL)
4203 && (pThis->uRequestQueueNextAddressRead != uRequestQueueNextEntryWrite))
4204 {
4205 MptRequestUnion GuestRequest;
4206 uint32_t u32RequestMessageFrameDesc = pThis->aRequestQueue[pThis->uRequestQueueNextAddressRead];
4207 RTGCPHYS GCPhysMessageFrameAddr = LSILOGIC_RTGCPHYS_FROM_U32(pThis->u32HostMFAHighAddr,
4208 (u32RequestMessageFrameDesc & ~0x07));
4209
4210 /* Read the message header from the guest first. */
4211 PDMDevHlpPCIPhysReadMeta(pDevIns, GCPhysMessageFrameAddr, &GuestRequest, sizeof(MptMessageHdr));
4212
4213 /* Determine the size of the request. */
4214 uint32_t cbRequest = 0;
4215 switch (GuestRequest.Header.u8Function)
4216 {
4217 case MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST:
4218 cbRequest = sizeof(MptSCSIIORequest);
4219 break;
4220 case MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT:
4221 cbRequest = sizeof(MptSCSITaskManagementRequest);
4222 break;
4223 case MPT_MESSAGE_HDR_FUNCTION_IOC_INIT:
4224 cbRequest = sizeof(MptIOCInitRequest);
4225 break;
4226 case MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS:
4227 cbRequest = sizeof(MptIOCFactsRequest);
4228 break;
4229 case MPT_MESSAGE_HDR_FUNCTION_CONFIG:
4230 cbRequest = sizeof(MptConfigurationRequest);
4231 break;
4232 case MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS:
4233 cbRequest = sizeof(MptPortFactsRequest);
4234 break;
4235 case MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE:
4236 cbRequest = sizeof(MptPortEnableRequest);
4237 break;
4238 case MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION:
4239 cbRequest = sizeof(MptEventNotificationRequest);
4240 break;
4241 case MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK:
4242 AssertMsgFailed(("todo\n"));
4243 //cbRequest = sizeof(MptEventAckRequest);
4244 break;
4245 case MPT_MESSAGE_HDR_FUNCTION_FW_DOWNLOAD:
4246 cbRequest = sizeof(MptFWDownloadRequest);
4247 break;
4248 case MPT_MESSAGE_HDR_FUNCTION_FW_UPLOAD:
4249 cbRequest = sizeof(MptFWUploadRequest);
4250 break;
4251 default:
4252 AssertMsgFailed(("Unknown function issued %u\n", GuestRequest.Header.u8Function));
4253 lsilogicSetIOCFaultCode(pThis, LSILOGIC_IOCSTATUS_INVALID_FUNCTION);
4254 }
4255
4256 if (cbRequest != 0)
4257 {
4258 /* Read the complete message frame from guest memory now. */
4259 PDMDevHlpPCIPhysReadMeta(pDevIns, GCPhysMessageFrameAddr, &GuestRequest, cbRequest);
4260
4261 /* Handle SCSI I/O requests now. */
4262 if (GuestRequest.Header.u8Function == MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST)
4263 {
4264 rc = lsilogicR3ProcessSCSIIORequest(pDevIns, pThis, pThisCC, GCPhysMessageFrameAddr, &GuestRequest);
4265 AssertRC(rc);
4266 }
4267 else
4268 {
4269 MptReplyUnion Reply;
4270 rc = lsilogicR3ProcessMessageRequest(pDevIns, pThis, pThisCC, &GuestRequest.Header, &Reply);
4271 AssertRC(rc);
4272 }
4273
4274 pThis->uRequestQueueNextAddressRead++;
4275 pThis->uRequestQueueNextAddressRead %= pThis->cRequestQueueEntries;
4276 }
4277 } /* While request frames available. */
4278 } /* While running */
4279
4280 return VINF_SUCCESS;
4281}
4282
4283
4284/**
4285 * @callback_method_impl{FNPDMTHREADWAKEUPDEV}
4286 */
4287static DECLCALLBACK(int) lsilogicR3WorkerWakeUp(PPDMDEVINS pDevIns, PPDMTHREAD pThread)
4288{
4289 RT_NOREF(pThread);
4290 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4291 return PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEvtProcess);
4292}
4293
4294
4295/**
4296 * Kicks the controller to process pending tasks after the VM was resumed
4297 * or loaded from a saved state.
4298 *
4299 * @returns nothing.
4300 * @param pDevIns The device instance.
4301 * @param pThis Pointer to the shared LsiLogic device state.
4302 */
4303static void lsilogicR3Kick(PPDMDEVINS pDevIns, PLSILOGICSCSI pThis)
4304{
4305 if (pThis->fNotificationSent)
4306 {
4307 /* Notify the worker thread that there are pending requests. */
4308 LogFlowFunc(("Signal event semaphore\n"));
4309 int rc = PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEvtProcess);
4310 AssertRC(rc);
4311 }
4312}
4313
4314
4315/*
4316 * Saved state.
4317 */
4318
4319/**
4320 * @callback_method_impl{FNSSMDEVLIVEEXEC}
4321 */
4322static DECLCALLBACK(int) lsilogicR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
4323{
4324 RT_NOREF(uPass);
4325 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4326 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4327 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
4328
4329 pHlp->pfnSSMPutU32(pSSM, pThis->enmCtrlType);
4330 pHlp->pfnSSMPutU32(pSSM, pThis->cDeviceStates);
4331 pHlp->pfnSSMPutU32(pSSM, pThis->cPorts);
4332
4333 /* Save the device config. */
4334 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
4335 pHlp->pfnSSMPutBool(pSSM, pThisCC->paDeviceStates[i].pDrvBase != NULL);
4336
4337 return VINF_SSM_DONT_CALL_AGAIN;
4338}
4339
4340/**
4341 * @callback_method_impl{FNSSMDEVSAVEEXEC}
4342 */
4343static DECLCALLBACK(int) lsilogicR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4344{
4345 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4346 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4347 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
4348
4349 /* Every device first. */
4350 lsilogicR3LiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
4351 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
4352 {
4353 PLSILOGICDEVICE pDevice = &pThisCC->paDeviceStates[i];
4354
4355 AssertMsg(!pDevice->cOutstandingRequests,
4356 ("There are still outstanding requests on this device\n"));
4357 pHlp->pfnSSMPutU32(pSSM, pDevice->cOutstandingRequests);
4358
4359 /* Query all suspended requests and store them in the request queue. */
4360 if (pDevice->pDrvMediaEx)
4361 {
4362 uint32_t cReqsRedo = pDevice->pDrvMediaEx->pfnIoReqGetSuspendedCount(pDevice->pDrvMediaEx);
4363 if (cReqsRedo)
4364 {
4365 PDMMEDIAEXIOREQ hIoReq;
4366 PLSILOGICREQ pReq;
4367 int rc = pDevice->pDrvMediaEx->pfnIoReqQuerySuspendedStart(pDevice->pDrvMediaEx, &hIoReq,
4368 (void **)&pReq);
4369 AssertRCBreak(rc);
4370
4371 for (;;)
4372 {
4373 if (!pReq->fBIOS)
4374 {
4375 /* Write only the lower 32bit part of the address. */
4376 ASMAtomicWriteU32(&pThis->aRequestQueue[pThis->uRequestQueueNextEntryFreeWrite],
4377 pReq->GCPhysMessageFrameAddr & UINT32_C(0xffffffff));
4378
4379 pThis->uRequestQueueNextEntryFreeWrite++;
4380 pThis->uRequestQueueNextEntryFreeWrite %= pThis->cRequestQueueEntries;
4381 }
4382 else
4383 {
4384 AssertMsg(!pReq->pRedoNext, ("Only one BIOS task can be active!\n"));
4385 vboxscsiSetRequestRedo(&pThisCC->VBoxSCSI);
4386 }
4387
4388 cReqsRedo--;
4389 if (!cReqsRedo)
4390 break;
4391
4392 rc = pDevice->pDrvMediaEx->pfnIoReqQuerySuspendedNext(pDevice->pDrvMediaEx, hIoReq,
4393 &hIoReq, (void **)&pReq);
4394 AssertRCBreak(rc);
4395 }
4396 }
4397 }
4398 }
4399
4400 /* Now the main device state. */
4401 pHlp->pfnSSMPutU32(pSSM, pThis->enmState);
4402 pHlp->pfnSSMPutU32(pSSM, pThis->enmWhoInit);
4403 pHlp->pfnSSMPutU32(pSSM, pThis->enmDoorbellState);
4404 pHlp->pfnSSMPutBool(pSSM, pThis->fDiagnosticEnabled);
4405 pHlp->pfnSSMPutBool(pSSM, pThis->fNotificationSent);
4406 pHlp->pfnSSMPutBool(pSSM, pThis->fEventNotificationEnabled);
4407 pHlp->pfnSSMPutU32(pSSM, pThis->uInterruptMask);
4408 pHlp->pfnSSMPutU32(pSSM, pThis->uInterruptStatus);
4409 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aMessage); i++)
4410 pHlp->pfnSSMPutU32(pSSM, pThis->aMessage[i]);
4411 pHlp->pfnSSMPutU32(pSSM, pThis->iMessage);
4412 pHlp->pfnSSMPutU32(pSSM, pThis->cMessage);
4413 pHlp->pfnSSMPutMem(pSSM, &pThis->ReplyBuffer, sizeof(pThis->ReplyBuffer));
4414 pHlp->pfnSSMPutU32(pSSM, pThis->uNextReplyEntryRead);
4415 pHlp->pfnSSMPutU32(pSSM, pThis->cReplySize);
4416 pHlp->pfnSSMPutU16(pSSM, pThis->u16IOCFaultCode);
4417 pHlp->pfnSSMPutU32(pSSM, pThis->u32HostMFAHighAddr);
4418 pHlp->pfnSSMPutU32(pSSM, pThis->u32SenseBufferHighAddr);
4419 pHlp->pfnSSMPutU8(pSSM, pThis->cMaxDevices);
4420 pHlp->pfnSSMPutU8(pSSM, pThis->cMaxBuses);
4421 pHlp->pfnSSMPutU16(pSSM, pThis->cbReplyFrame);
4422 pHlp->pfnSSMPutU32(pSSM, pThis->iDiagnosticAccess);
4423 pHlp->pfnSSMPutU32(pSSM, pThis->cReplyQueueEntries);
4424 pHlp->pfnSSMPutU32(pSSM, pThis->cRequestQueueEntries);
4425 pHlp->pfnSSMPutU32(pSSM, pThis->uReplyFreeQueueNextEntryFreeWrite);
4426 pHlp->pfnSSMPutU32(pSSM, pThis->uReplyFreeQueueNextAddressRead);
4427 pHlp->pfnSSMPutU32(pSSM, pThis->uReplyPostQueueNextEntryFreeWrite);
4428 pHlp->pfnSSMPutU32(pSSM, pThis->uReplyPostQueueNextAddressRead);
4429 pHlp->pfnSSMPutU32(pSSM, pThis->uRequestQueueNextEntryFreeWrite);
4430 pHlp->pfnSSMPutU32(pSSM, pThis->uRequestQueueNextAddressRead);
4431
4432 for (unsigned i = 0; i < pThis->cReplyQueueEntries; i++)
4433 pHlp->pfnSSMPutU32(pSSM, pThis->aReplyFreeQueue[i]);
4434 for (unsigned i = 0; i < pThis->cReplyQueueEntries; i++)
4435 pHlp->pfnSSMPutU32(pSSM, pThis->aReplyPostQueue[i]);
4436 for (unsigned i = 0; i < pThis->cRequestQueueEntries; i++)
4437 pHlp->pfnSSMPutU32(pSSM, pThis->aRequestQueue[i]);
4438
4439 pHlp->pfnSSMPutU16(pSSM, pThis->u16NextHandle);
4440
4441 /* Save diagnostic memory register and data regions. */
4442 pHlp->pfnSSMPutU32(pSSM, pThis->u32DiagMemAddr);
4443 pHlp->pfnSSMPutU32(pSSM, lsilogicR3MemRegionsCount(pThisCC));
4444
4445 PLSILOGICMEMREGN pIt;
4446 RTListForEach(&pThisCC->ListMemRegns, pIt, LSILOGICMEMREGN, NodeList)
4447 {
4448 pHlp->pfnSSMPutU32(pSSM, pIt->u32AddrStart);
4449 pHlp->pfnSSMPutU32(pSSM, pIt->u32AddrEnd);
4450 pHlp->pfnSSMPutMem(pSSM, &pIt->au32Data[0], (pIt->u32AddrEnd - pIt->u32AddrStart + 1) * sizeof(uint32_t));
4451 }
4452
4453 PMptConfigurationPagesSupported pPages = pThisCC->pConfigurationPages;
4454
4455 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage0, sizeof(MptConfigurationPageManufacturing0));
4456 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage1, sizeof(MptConfigurationPageManufacturing1));
4457 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage2, sizeof(MptConfigurationPageManufacturing2));
4458 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage3, sizeof(MptConfigurationPageManufacturing3));
4459 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage4, sizeof(MptConfigurationPageManufacturing4));
4460 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage5, sizeof(MptConfigurationPageManufacturing5));
4461 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage6, sizeof(MptConfigurationPageManufacturing6));
4462 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage8, sizeof(MptConfigurationPageManufacturing8));
4463 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage9, sizeof(MptConfigurationPageManufacturing9));
4464 pHlp->pfnSSMPutMem(pSSM, &pPages->ManufacturingPage10, sizeof(MptConfigurationPageManufacturing10));
4465 pHlp->pfnSSMPutMem(pSSM, &pPages->IOUnitPage0, sizeof(MptConfigurationPageIOUnit0));
4466 pHlp->pfnSSMPutMem(pSSM, &pPages->IOUnitPage1, sizeof(MptConfigurationPageIOUnit1));
4467 pHlp->pfnSSMPutMem(pSSM, &pPages->IOUnitPage2, sizeof(MptConfigurationPageIOUnit2));
4468 pHlp->pfnSSMPutMem(pSSM, &pPages->IOUnitPage3, sizeof(MptConfigurationPageIOUnit3));
4469 pHlp->pfnSSMPutMem(pSSM, &pPages->IOUnitPage4, sizeof(MptConfigurationPageIOUnit4));
4470 pHlp->pfnSSMPutMem(pSSM, &pPages->IOCPage0, sizeof(MptConfigurationPageIOC0));
4471 pHlp->pfnSSMPutMem(pSSM, &pPages->IOCPage1, sizeof(MptConfigurationPageIOC1));
4472 pHlp->pfnSSMPutMem(pSSM, &pPages->IOCPage2, sizeof(MptConfigurationPageIOC2));
4473 pHlp->pfnSSMPutMem(pSSM, &pPages->IOCPage3, sizeof(MptConfigurationPageIOC3));
4474 pHlp->pfnSSMPutMem(pSSM, &pPages->IOCPage4, sizeof(MptConfigurationPageIOC4));
4475 pHlp->pfnSSMPutMem(pSSM, &pPages->IOCPage6, sizeof(MptConfigurationPageIOC6));
4476 pHlp->pfnSSMPutMem(pSSM, &pPages->BIOSPage1, sizeof(MptConfigurationPageBIOS1));
4477 pHlp->pfnSSMPutMem(pSSM, &pPages->BIOSPage2, sizeof(MptConfigurationPageBIOS2));
4478 pHlp->pfnSSMPutMem(pSSM, &pPages->BIOSPage4, sizeof(MptConfigurationPageBIOS4));
4479
4480 /* Device dependent pages */
4481 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
4482 {
4483 PMptConfigurationPagesSpi pSpiPages = &pPages->u.SpiPages;
4484
4485 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage0, sizeof(MptConfigurationPageSCSISPIPort0));
4486 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage1, sizeof(MptConfigurationPageSCSISPIPort1));
4487 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage2, sizeof(MptConfigurationPageSCSISPIPort2));
4488
4489 for (unsigned i = 0; i < RT_ELEMENTS(pSpiPages->aBuses[0].aDevicePages); i++)
4490 {
4491 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage0, sizeof(MptConfigurationPageSCSISPIDevice0));
4492 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage1, sizeof(MptConfigurationPageSCSISPIDevice1));
4493 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage2, sizeof(MptConfigurationPageSCSISPIDevice2));
4494 pHlp->pfnSSMPutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage3, sizeof(MptConfigurationPageSCSISPIDevice3));
4495 }
4496 }
4497 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
4498 {
4499 PMptConfigurationPagesSas pSasPages = &pPages->u.SasPages;
4500
4501 pHlp->pfnSSMPutU32(pSSM, pSasPages->cbManufacturingPage7);
4502 pHlp->pfnSSMPutU32(pSSM, pSasPages->cbSASIOUnitPage0);
4503 pHlp->pfnSSMPutU32(pSSM, pSasPages->cbSASIOUnitPage1);
4504
4505 pHlp->pfnSSMPutMem(pSSM, pSasPages->pManufacturingPage7, pSasPages->cbManufacturingPage7);
4506 pHlp->pfnSSMPutMem(pSSM, pSasPages->pSASIOUnitPage0, pSasPages->cbSASIOUnitPage0);
4507 pHlp->pfnSSMPutMem(pSSM, pSasPages->pSASIOUnitPage1, pSasPages->cbSASIOUnitPage1);
4508
4509 pHlp->pfnSSMPutMem(pSSM, &pSasPages->SASIOUnitPage2, sizeof(MptConfigurationPageSASIOUnit2));
4510 pHlp->pfnSSMPutMem(pSSM, &pSasPages->SASIOUnitPage3, sizeof(MptConfigurationPageSASIOUnit3));
4511
4512 pHlp->pfnSSMPutU32(pSSM, pSasPages->cPHYs);
4513 for (unsigned i = 0; i < pSasPages->cPHYs; i++)
4514 {
4515 pHlp->pfnSSMPutMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage0, sizeof(MptConfigurationPageSASPHY0));
4516 pHlp->pfnSSMPutMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage1, sizeof(MptConfigurationPageSASPHY1));
4517 }
4518
4519 /* The number of devices first. */
4520 pHlp->pfnSSMPutU32(pSSM, pSasPages->cDevices);
4521
4522 for (PMptSASDevice pCurr = pSasPages->pSASDeviceHead; pCurr; pCurr = pCurr->pNext)
4523 {
4524 pHlp->pfnSSMPutMem(pSSM, &pCurr->SASDevicePage0, sizeof(MptConfigurationPageSASDevice0));
4525 pHlp->pfnSSMPutMem(pSSM, &pCurr->SASDevicePage1, sizeof(MptConfigurationPageSASDevice1));
4526 pHlp->pfnSSMPutMem(pSSM, &pCurr->SASDevicePage2, sizeof(MptConfigurationPageSASDevice2));
4527 }
4528 }
4529 else
4530 AssertMsgFailed(("Invalid controller type %d\n", pThis->enmCtrlType));
4531
4532 vboxscsiR3SaveExec(pHlp, &pThisCC->VBoxSCSI, pSSM);
4533 return pHlp->pfnSSMPutU32(pSSM, UINT32_MAX);
4534}
4535
4536/**
4537 * @callback_method_impl{FNSSMDEVLOADDONE}
4538 */
4539static DECLCALLBACK(int) lsilogicR3LoadDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4540{
4541 RT_NOREF(pSSM);
4542 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4543
4544 lsilogicR3Kick(pDevIns, pThis);
4545 return VINF_SUCCESS;
4546}
4547
4548/**
4549 * @callback_method_impl{FNSSMDEVLOADEXEC}
4550 */
4551static DECLCALLBACK(int) lsilogicR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
4552{
4553 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4554 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4555 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
4556 int rc;
4557
4558 if ( uVersion != LSILOGIC_SAVED_STATE_VERSION
4559 && uVersion != LSILOGIC_SAVED_STATE_VERSION_PRE_DIAG_MEM
4560 && uVersion != LSILOGIC_SAVED_STATE_VERSION_BOOL_DOORBELL
4561 && uVersion != LSILOGIC_SAVED_STATE_VERSION_PRE_SAS
4562 && uVersion != LSILOGIC_SAVED_STATE_VERSION_VBOX_30)
4563 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
4564
4565 /* device config */
4566 if (uVersion > LSILOGIC_SAVED_STATE_VERSION_PRE_SAS)
4567 {
4568 LSILOGICCTRLTYPE enmCtrlType;
4569 uint32_t cDeviceStates, cPorts;
4570
4571 PDMDEVHLP_SSM_GET_ENUM32_RET(pHlp, pSSM, enmCtrlType, LSILOGICCTRLTYPE);
4572 pHlp->pfnSSMGetU32(pSSM, &cDeviceStates);
4573 rc = pHlp->pfnSSMGetU32(pSSM, &cPorts);
4574 AssertRCReturn(rc, rc);
4575
4576 if (enmCtrlType != pThis->enmCtrlType)
4577 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Target config mismatch (Controller type): config=%d state=%d"),
4578 pThis->enmCtrlType, enmCtrlType);
4579 if (cDeviceStates != pThis->cDeviceStates)
4580 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Target config mismatch (Device states): config=%u state=%u"),
4581 pThis->cDeviceStates, cDeviceStates);
4582 if (cPorts != pThis->cPorts)
4583 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Target config mismatch (Ports): config=%u state=%u"),
4584 pThis->cPorts, cPorts);
4585 }
4586 if (uVersion > LSILOGIC_SAVED_STATE_VERSION_VBOX_30)
4587 {
4588 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
4589 {
4590 bool fPresent;
4591 rc = pHlp->pfnSSMGetBool(pSSM, &fPresent);
4592 AssertRCReturn(rc, rc);
4593 if (fPresent != (pThisCC->paDeviceStates[i].pDrvBase != NULL))
4594 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Target %u config mismatch: config=%RTbool state=%RTbool"),
4595 i, pThisCC->paDeviceStates[i].pDrvBase != NULL, fPresent);
4596 }
4597 }
4598 if (uPass != SSM_PASS_FINAL)
4599 return VINF_SUCCESS;
4600
4601 /* Every device first. */
4602 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
4603 {
4604 PLSILOGICDEVICE pDevice = &pThisCC->paDeviceStates[i];
4605
4606 AssertMsg(!pDevice->cOutstandingRequests,
4607 ("There are still outstanding requests on this device\n"));
4608 pHlp->pfnSSMGetU32V(pSSM, &pDevice->cOutstandingRequests);
4609 }
4610 /* Now the main device state. */
4611 PDMDEVHLP_SSM_GET_ENUM32_RET(pHlp, pSSM, pThis->enmState, LSILOGICSTATE);
4612 PDMDEVHLP_SSM_GET_ENUM32_RET(pHlp, pSSM, pThis->enmWhoInit, LSILOGICWHOINIT);
4613 if (uVersion <= LSILOGIC_SAVED_STATE_VERSION_BOOL_DOORBELL)
4614 {
4615 /*
4616 * The doorbell status flag distinguishes only between
4617 * doorbell not in use or a Function handshake is currently in progress.
4618 */
4619 bool fDoorbellInProgress = false;
4620 rc = pHlp->pfnSSMGetBool(pSSM, &fDoorbellInProgress);
4621 AssertRCReturn(rc, rc);
4622 if (fDoorbellInProgress)
4623 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_FN_HANDSHAKE;
4624 else
4625 pThis->enmDoorbellState = LSILOGICDOORBELLSTATE_NOT_IN_USE;
4626 }
4627 else
4628 PDMDEVHLP_SSM_GET_ENUM32_RET(pHlp, pSSM, pThis->enmDoorbellState, LSILOGICDOORBELLSTATE);
4629 pHlp->pfnSSMGetBool(pSSM, &pThis->fDiagnosticEnabled);
4630 pHlp->pfnSSMGetBool(pSSM, &pThis->fNotificationSent);
4631 pHlp->pfnSSMGetBool(pSSM, &pThis->fEventNotificationEnabled);
4632 pHlp->pfnSSMGetU32V(pSSM, &pThis->uInterruptMask);
4633 pHlp->pfnSSMGetU32V(pSSM, &pThis->uInterruptStatus);
4634 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aMessage); i++)
4635 pHlp->pfnSSMGetU32(pSSM, &pThis->aMessage[i]);
4636 pHlp->pfnSSMGetU32(pSSM, &pThis->iMessage);
4637 pHlp->pfnSSMGetU32(pSSM, &pThis->cMessage);
4638 pHlp->pfnSSMGetMem(pSSM, &pThis->ReplyBuffer, sizeof(pThis->ReplyBuffer));
4639 pHlp->pfnSSMGetU32(pSSM, &pThis->uNextReplyEntryRead);
4640 pHlp->pfnSSMGetU32(pSSM, &pThis->cReplySize);
4641 pHlp->pfnSSMGetU16(pSSM, &pThis->u16IOCFaultCode);
4642 pHlp->pfnSSMGetU32(pSSM, &pThis->u32HostMFAHighAddr);
4643 pHlp->pfnSSMGetU32(pSSM, &pThis->u32SenseBufferHighAddr);
4644 pHlp->pfnSSMGetU8(pSSM, &pThis->cMaxDevices);
4645 pHlp->pfnSSMGetU8(pSSM, &pThis->cMaxBuses);
4646 pHlp->pfnSSMGetU16(pSSM, &pThis->cbReplyFrame);
4647 pHlp->pfnSSMGetU32(pSSM, &pThis->iDiagnosticAccess);
4648
4649 uint32_t cReplyQueueEntries, cRequestQueueEntries;
4650 pHlp->pfnSSMGetU32(pSSM, &cReplyQueueEntries);
4651 rc = pHlp->pfnSSMGetU32(pSSM, &cRequestQueueEntries);
4652 AssertRCReturn(rc, rc);
4653
4654 if ( cReplyQueueEntries != pThis->cReplyQueueEntries
4655 || cRequestQueueEntries != pThis->cRequestQueueEntries)
4656 {
4657 LogRel(("Changing queue sizes: cReplyQueueEntries=%u cRequestQueuEntries=%u\n", cReplyQueueEntries, cRequestQueueEntries));
4658 if ( cReplyQueueEntries > RT_ELEMENTS(pThis->aReplyFreeQueue)
4659 || cReplyQueueEntries < LSILOGICSCSI_REQUEST_QUEUE_DEPTH_MIN
4660 || cRequestQueueEntries > RT_ELEMENTS(pThis->aRequestQueue)
4661 || cRequestQueueEntries < LSILOGICSCSI_REPLY_QUEUE_DEPTH_MIN)
4662 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Out of bounds: cReplyQueueEntries=%u cRequestQueueEntries=%u"),
4663 cReplyQueueEntries, cRequestQueueEntries);
4664 pThis->cReplyQueueEntries = cReplyQueueEntries;
4665 pThis->cRequestQueueEntries = cRequestQueueEntries;
4666 }
4667
4668 pHlp->pfnSSMGetU32V(pSSM, &pThis->uReplyFreeQueueNextEntryFreeWrite);
4669 pHlp->pfnSSMGetU32V(pSSM, &pThis->uReplyFreeQueueNextAddressRead);
4670 pHlp->pfnSSMGetU32V(pSSM, &pThis->uReplyPostQueueNextEntryFreeWrite);
4671 pHlp->pfnSSMGetU32V(pSSM, &pThis->uReplyPostQueueNextAddressRead);
4672 pHlp->pfnSSMGetU32V(pSSM, &pThis->uRequestQueueNextEntryFreeWrite);
4673 pHlp->pfnSSMGetU32V(pSSM, &pThis->uRequestQueueNextAddressRead);
4674
4675 PMptConfigurationPagesSupported pPages = pThisCC->pConfigurationPages;
4676
4677 if (uVersion <= LSILOGIC_SAVED_STATE_VERSION_PRE_SAS)
4678 {
4679 PMptConfigurationPagesSpi pSpiPages = &pPages->u.SpiPages;
4680 MptConfigurationPagesSupported_SSM_V2 ConfigPagesV2;
4681
4682 if (pThis->enmCtrlType != LSILOGICCTRLTYPE_SCSI_SPI)
4683 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch: Expected SPI SCSI controller"));
4684
4685 pHlp->pfnSSMGetMem(pSSM, &ConfigPagesV2, sizeof(MptConfigurationPagesSupported_SSM_V2));
4686
4687 pPages->ManufacturingPage0 = ConfigPagesV2.ManufacturingPage0;
4688 pPages->ManufacturingPage1 = ConfigPagesV2.ManufacturingPage1;
4689 pPages->ManufacturingPage2 = ConfigPagesV2.ManufacturingPage2;
4690 pPages->ManufacturingPage3 = ConfigPagesV2.ManufacturingPage3;
4691 pPages->ManufacturingPage4 = ConfigPagesV2.ManufacturingPage4;
4692 pPages->IOUnitPage0 = ConfigPagesV2.IOUnitPage0;
4693 pPages->IOUnitPage1 = ConfigPagesV2.IOUnitPage1;
4694 pPages->IOUnitPage2 = ConfigPagesV2.IOUnitPage2;
4695 pPages->IOUnitPage3 = ConfigPagesV2.IOUnitPage3;
4696 pPages->IOCPage0 = ConfigPagesV2.IOCPage0;
4697 pPages->IOCPage1 = ConfigPagesV2.IOCPage1;
4698 pPages->IOCPage2 = ConfigPagesV2.IOCPage2;
4699 pPages->IOCPage3 = ConfigPagesV2.IOCPage3;
4700 pPages->IOCPage4 = ConfigPagesV2.IOCPage4;
4701 pPages->IOCPage6 = ConfigPagesV2.IOCPage6;
4702
4703 pSpiPages->aPortPages[0].SCSISPIPortPage0 = ConfigPagesV2.aPortPages[0].SCSISPIPortPage0;
4704 pSpiPages->aPortPages[0].SCSISPIPortPage1 = ConfigPagesV2.aPortPages[0].SCSISPIPortPage1;
4705 pSpiPages->aPortPages[0].SCSISPIPortPage2 = ConfigPagesV2.aPortPages[0].SCSISPIPortPage2;
4706
4707 for (unsigned i = 0; i < RT_ELEMENTS(pPages->u.SpiPages.aBuses[0].aDevicePages); i++)
4708 {
4709 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage0 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage0;
4710 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage1 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage1;
4711 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage2 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage2;
4712 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage3 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage3;
4713 }
4714 }
4715 else
4716 {
4717 /* Queue content */
4718 for (unsigned i = 0; i < pThis->cReplyQueueEntries; i++)
4719 pHlp->pfnSSMGetU32V(pSSM, &pThis->aReplyFreeQueue[i]);
4720 for (unsigned i = 0; i < pThis->cReplyQueueEntries; i++)
4721 pHlp->pfnSSMGetU32V(pSSM, &pThis->aReplyPostQueue[i]);
4722 for (unsigned i = 0; i < pThis->cRequestQueueEntries; i++)
4723 pHlp->pfnSSMGetU32V(pSSM, &pThis->aRequestQueue[i]);
4724
4725 pHlp->pfnSSMGetU16(pSSM, &pThis->u16NextHandle);
4726
4727 if (uVersion > LSILOGIC_SAVED_STATE_VERSION_PRE_DIAG_MEM)
4728 {
4729
4730 /* Save diagnostic memory register and data regions. */
4731 pHlp->pfnSSMGetU32(pSSM, &pThis->u32DiagMemAddr);
4732 uint32_t cMemRegions = 0;
4733 rc = pHlp->pfnSSMGetU32(pSSM, &cMemRegions);
4734 AssertLogRelRCReturn(rc, rc);
4735
4736 while (cMemRegions)
4737 {
4738 uint32_t u32AddrStart = 0;
4739 pHlp->pfnSSMGetU32(pSSM, &u32AddrStart);
4740 uint32_t u32AddrEnd = 0;
4741 rc = pHlp->pfnSSMGetU32(pSSM, &u32AddrEnd);
4742 AssertLogRelRCReturn(rc, rc);
4743
4744 uint32_t cRegion = u32AddrEnd - u32AddrStart + 1;
4745 PLSILOGICMEMREGN pRegion = (PLSILOGICMEMREGN)RTMemAllocZ(RT_UOFFSETOF_DYN(LSILOGICMEMREGN, au32Data[cRegion]));
4746 if (pRegion)
4747 {
4748 pRegion->u32AddrStart = u32AddrStart;
4749 pRegion->u32AddrEnd = u32AddrEnd;
4750 pHlp->pfnSSMGetMem(pSSM, &pRegion->au32Data[0], cRegion * sizeof(uint32_t));
4751 lsilogicR3MemRegionInsert(pThisCC, pRegion);
4752 pThisCC->cbMemRegns += cRegion * sizeof(uint32_t);
4753 }
4754 else
4755 {
4756 /* Leave a log message but continue. */
4757 LogRel(("LsiLogic: Out of memory while restoring the state, might not work as expected\n"));
4758 pHlp->pfnSSMSkip(pSSM, cRegion * sizeof(uint32_t));
4759 }
4760 cMemRegions--;
4761 }
4762 }
4763
4764 /* Configuration pages */
4765 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage0, sizeof(MptConfigurationPageManufacturing0));
4766 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage1, sizeof(MptConfigurationPageManufacturing1));
4767 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage2, sizeof(MptConfigurationPageManufacturing2));
4768 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage3, sizeof(MptConfigurationPageManufacturing3));
4769 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage4, sizeof(MptConfigurationPageManufacturing4));
4770 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage5, sizeof(MptConfigurationPageManufacturing5));
4771 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage6, sizeof(MptConfigurationPageManufacturing6));
4772 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage8, sizeof(MptConfigurationPageManufacturing8));
4773 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage9, sizeof(MptConfigurationPageManufacturing9));
4774 pHlp->pfnSSMGetMem(pSSM, &pPages->ManufacturingPage10, sizeof(MptConfigurationPageManufacturing10));
4775 pHlp->pfnSSMGetMem(pSSM, &pPages->IOUnitPage0, sizeof(MptConfigurationPageIOUnit0));
4776 pHlp->pfnSSMGetMem(pSSM, &pPages->IOUnitPage1, sizeof(MptConfigurationPageIOUnit1));
4777 pHlp->pfnSSMGetMem(pSSM, &pPages->IOUnitPage2, sizeof(MptConfigurationPageIOUnit2));
4778 pHlp->pfnSSMGetMem(pSSM, &pPages->IOUnitPage3, sizeof(MptConfigurationPageIOUnit3));
4779 pHlp->pfnSSMGetMem(pSSM, &pPages->IOUnitPage4, sizeof(MptConfigurationPageIOUnit4));
4780 pHlp->pfnSSMGetMem(pSSM, &pPages->IOCPage0, sizeof(MptConfigurationPageIOC0));
4781 pHlp->pfnSSMGetMem(pSSM, &pPages->IOCPage1, sizeof(MptConfigurationPageIOC1));
4782 pHlp->pfnSSMGetMem(pSSM, &pPages->IOCPage2, sizeof(MptConfigurationPageIOC2));
4783 pHlp->pfnSSMGetMem(pSSM, &pPages->IOCPage3, sizeof(MptConfigurationPageIOC3));
4784 pHlp->pfnSSMGetMem(pSSM, &pPages->IOCPage4, sizeof(MptConfigurationPageIOC4));
4785 pHlp->pfnSSMGetMem(pSSM, &pPages->IOCPage6, sizeof(MptConfigurationPageIOC6));
4786 pHlp->pfnSSMGetMem(pSSM, &pPages->BIOSPage1, sizeof(MptConfigurationPageBIOS1));
4787 pHlp->pfnSSMGetMem(pSSM, &pPages->BIOSPage2, sizeof(MptConfigurationPageBIOS2));
4788 pHlp->pfnSSMGetMem(pSSM, &pPages->BIOSPage4, sizeof(MptConfigurationPageBIOS4));
4789
4790 /* Device dependent pages */
4791 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
4792 {
4793 PMptConfigurationPagesSpi pSpiPages = &pPages->u.SpiPages;
4794
4795 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage0, sizeof(MptConfigurationPageSCSISPIPort0));
4796 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage1, sizeof(MptConfigurationPageSCSISPIPort1));
4797 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage2, sizeof(MptConfigurationPageSCSISPIPort2));
4798
4799 for (unsigned i = 0; i < RT_ELEMENTS(pSpiPages->aBuses[0].aDevicePages); i++)
4800 {
4801 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage0, sizeof(MptConfigurationPageSCSISPIDevice0));
4802 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage1, sizeof(MptConfigurationPageSCSISPIDevice1));
4803 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage2, sizeof(MptConfigurationPageSCSISPIDevice2));
4804 pHlp->pfnSSMGetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage3, sizeof(MptConfigurationPageSCSISPIDevice3));
4805 }
4806 }
4807 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
4808 {
4809 uint32_t cbPage0, cbPage1, cPHYs, cbManufacturingPage7;
4810 PMptConfigurationPagesSas pSasPages = &pPages->u.SasPages;
4811
4812 pHlp->pfnSSMGetU32(pSSM, &cbManufacturingPage7);
4813 pHlp->pfnSSMGetU32(pSSM, &cbPage0);
4814 rc = pHlp->pfnSSMGetU32(pSSM, &cbPage1);
4815 AssertRCReturn(rc, rc);
4816
4817 if ( (cbPage0 != pSasPages->cbSASIOUnitPage0)
4818 || (cbPage1 != pSasPages->cbSASIOUnitPage1)
4819 || (cbManufacturingPage7 != pSasPages->cbManufacturingPage7))
4820 return VERR_SSM_LOAD_CONFIG_MISMATCH;
4821
4822 AssertPtr(pSasPages->pManufacturingPage7);
4823 AssertPtr(pSasPages->pSASIOUnitPage0);
4824 AssertPtr(pSasPages->pSASIOUnitPage1);
4825
4826 pHlp->pfnSSMGetMem(pSSM, pSasPages->pManufacturingPage7, pSasPages->cbManufacturingPage7);
4827 pHlp->pfnSSMGetMem(pSSM, pSasPages->pSASIOUnitPage0, pSasPages->cbSASIOUnitPage0);
4828 pHlp->pfnSSMGetMem(pSSM, pSasPages->pSASIOUnitPage1, pSasPages->cbSASIOUnitPage1);
4829
4830 pHlp->pfnSSMGetMem(pSSM, &pSasPages->SASIOUnitPage2, sizeof(MptConfigurationPageSASIOUnit2));
4831 pHlp->pfnSSMGetMem(pSSM, &pSasPages->SASIOUnitPage3, sizeof(MptConfigurationPageSASIOUnit3));
4832
4833 rc = pHlp->pfnSSMGetU32(pSSM, &cPHYs);
4834 AssertRCReturn(rc, rc);
4835 if (cPHYs != pSasPages->cPHYs)
4836 return VERR_SSM_LOAD_CONFIG_MISMATCH;
4837
4838 AssertPtr(pSasPages->paPHYs);
4839 for (unsigned i = 0; i < pSasPages->cPHYs; i++)
4840 {
4841 pHlp->pfnSSMGetMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage0, sizeof(MptConfigurationPageSASPHY0));
4842 pHlp->pfnSSMGetMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage1, sizeof(MptConfigurationPageSASPHY1));
4843 }
4844
4845 /* The number of devices first. */
4846 rc = pHlp->pfnSSMGetU32(pSSM, &pSasPages->cDevices);
4847 AssertRCReturn(rc, rc);
4848
4849 PMptSASDevice pCurr = pSasPages->pSASDeviceHead;
4850
4851 for (unsigned i = 0; i < pSasPages->cDevices; i++)
4852 {
4853 AssertReturn(pCurr, VERR_SSM_LOAD_CONFIG_MISMATCH);
4854
4855 pHlp->pfnSSMGetMem(pSSM, &pCurr->SASDevicePage0, sizeof(MptConfigurationPageSASDevice0));
4856 pHlp->pfnSSMGetMem(pSSM, &pCurr->SASDevicePage1, sizeof(MptConfigurationPageSASDevice1));
4857 rc = pHlp->pfnSSMGetMem(pSSM, &pCurr->SASDevicePage2, sizeof(MptConfigurationPageSASDevice2));
4858 AssertRCReturn(rc, rc);
4859
4860 pCurr = pCurr->pNext;
4861 }
4862
4863 Assert(!pCurr);
4864 }
4865 else
4866 AssertMsgFailed(("Invalid controller type %d\n", pThis->enmCtrlType));
4867 }
4868
4869 rc = vboxscsiR3LoadExec(pHlp, &pThisCC->VBoxSCSI, pSSM);
4870 if (RT_FAILURE(rc))
4871 {
4872 LogRel(("LsiLogic: Failed to restore BIOS state: %Rrc.\n", rc));
4873 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: Failed to restore BIOS state\n"));
4874 }
4875
4876 uint32_t u32;
4877 rc = pHlp->pfnSSMGetU32(pSSM, &u32);
4878 if (RT_FAILURE(rc))
4879 return rc;
4880 AssertMsgReturn(u32 == UINT32_MAX, ("%#x\n", u32), VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
4881
4882 return VINF_SUCCESS;
4883}
4884
4885
4886/*
4887 * The device level IBASE and LED interfaces.
4888 */
4889
4890/**
4891 * @interface_method_impl{PDMILEDPORTS,pfnQueryStatusLed, For a SCSI device.}
4892 *
4893 * @remarks Called by the scsi driver, proxying the main calls.
4894 */
4895static DECLCALLBACK(int) lsilogicR3DeviceQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
4896{
4897 PLSILOGICDEVICE pDevice = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, ILed);
4898 if (iLUN == 0)
4899 {
4900 *ppLed = &pDevice->Led;
4901 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
4902 return VINF_SUCCESS;
4903 }
4904 return VERR_PDM_LUN_NOT_FOUND;
4905}
4906
4907
4908/**
4909 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
4910 */
4911static DECLCALLBACK(void *) lsilogicR3DeviceQueryInterface(PPDMIBASE pInterface, const char *pszIID)
4912{
4913 PLSILOGICDEVICE pDevice = RT_FROM_MEMBER(pInterface, LSILOGICDEVICE, IBase);
4914
4915 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDevice->IBase);
4916 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAPORT, &pDevice->IMediaPort);
4917 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAEXPORT, &pDevice->IMediaExPort);
4918 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pDevice->ILed);
4919 return NULL;
4920}
4921
4922
4923/*
4924 * The controller level IBASE and LED interfaces.
4925 */
4926
4927/**
4928 * Gets the pointer to the status LED of a unit.
4929 *
4930 * @returns VBox status code.
4931 * @param pInterface Pointer to the interface structure containing the called function pointer.
4932 * @param iLUN The unit which status LED we desire.
4933 * @param ppLed Where to store the LED pointer.
4934 */
4935static DECLCALLBACK(int) lsilogicR3StatusQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
4936{
4937 PLSILOGICSCSICC pThisCC = RT_FROM_MEMBER(pInterface, LSILOGICSCSICC, ILeds);
4938 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pThisCC->pDevIns, PLSILOGICSCSI);
4939 if (iLUN < pThis->cDeviceStates)
4940 {
4941 *ppLed = &pThisCC->paDeviceStates[iLUN].Led;
4942 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
4943 return VINF_SUCCESS;
4944 }
4945 return VERR_PDM_LUN_NOT_FOUND;
4946}
4947
4948/**
4949 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
4950 */
4951static DECLCALLBACK(void *) lsilogicR3StatusQueryInterface(PPDMIBASE pInterface, const char *pszIID)
4952{
4953 PLSILOGICSCSICC pThisCC = RT_FROM_MEMBER(pInterface, LSILOGICSCSICC, IBase);
4954 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThisCC->IBase);
4955 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThisCC->ILeds);
4956 return NULL;
4957}
4958
4959
4960/*
4961 * The PDM device interface and some helpers.
4962 */
4963
4964/**
4965 * Checks if all asynchronous I/O is finished.
4966 *
4967 * Used by lsilogicR3Reset, lsilogicR3Suspend and lsilogicR3PowerOff.
4968 *
4969 * @returns true if quiesced, false if busy.
4970 * @param pDevIns The device instance.
4971 */
4972static bool lsilogicR3AllAsyncIOIsFinished(PPDMDEVINS pDevIns)
4973{
4974 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
4975 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
4976
4977 for (uint32_t i = 0; i < pThis->cDeviceStates; i++)
4978 {
4979 PLSILOGICDEVICE pThisDevice = &pThisCC->paDeviceStates[i];
4980 if (pThisDevice->pDrvBase)
4981 {
4982 if (pThisDevice->cOutstandingRequests != 0)
4983 return false;
4984 }
4985 }
4986
4987 return true;
4988}
4989
4990/**
4991 * @callback_method_impl{FNPDMDEVASYNCNOTIFY,
4992 * Callback employed by lsilogicR3Suspend and lsilogicR3PowerOff.}
4993 */
4994static DECLCALLBACK(bool) lsilogicR3IsAsyncSuspendOrPowerOffDone(PPDMDEVINS pDevIns)
4995{
4996 if (!lsilogicR3AllAsyncIOIsFinished(pDevIns))
4997 return false;
4998
4999 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5000 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5001 return true;
5002}
5003
5004/**
5005 * Common worker for ahciR3Suspend and ahciR3PowerOff.
5006 */
5007static void lsilogicR3SuspendOrPowerOff(PPDMDEVINS pDevIns)
5008{
5009 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5010 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
5011
5012 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
5013 if (!lsilogicR3AllAsyncIOIsFinished(pDevIns))
5014 PDMDevHlpSetAsyncNotification(pDevIns, lsilogicR3IsAsyncSuspendOrPowerOffDone);
5015 else
5016 {
5017 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5018
5019 AssertMsg(!pThis->fNotificationSent, ("The PDM Queue should be empty at this point\n"));
5020 }
5021
5022 for (uint32_t i = 0; i < pThis->cDeviceStates; i++)
5023 {
5024 PLSILOGICDEVICE pThisDevice = &pThisCC->paDeviceStates[i];
5025 if (pThisDevice->pDrvMediaEx)
5026 pThisDevice->pDrvMediaEx->pfnNotifySuspend(pThisDevice->pDrvMediaEx);
5027 }
5028}
5029
5030/**
5031 * @interface_method_impl{PDMDEVREG,pfnSuspend}
5032 */
5033static DECLCALLBACK(void) lsilogicR3Suspend(PPDMDEVINS pDevIns)
5034{
5035 Log(("lsilogicR3Suspend\n"));
5036 lsilogicR3SuspendOrPowerOff(pDevIns);
5037}
5038
5039/**
5040 * @interface_method_impl{PDMDEVREG,pfnResume}
5041 */
5042static DECLCALLBACK(void) lsilogicR3Resume(PPDMDEVINS pDevIns)
5043{
5044 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5045
5046 Log(("lsilogicR3Resume\n"));
5047
5048 lsilogicR3Kick(pDevIns, pThis);
5049}
5050
5051/**
5052 * @interface_method_impl{PDMDEVREG,pfnDetach}
5053 *
5054 * One harddisk at one port has been unplugged.
5055 * The VM is suspended at this point.
5056 */
5057static DECLCALLBACK(void) lsilogicR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
5058{
5059 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5060 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
5061 Log(("%s: iLUN=%#x\n", __FUNCTION__, iLUN));
5062 RT_NOREF(fFlags);
5063
5064 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG, ("LsiLogic: Device does not support hotplugging\n"));
5065
5066 if (iLUN >= pThis->cDeviceStates)
5067 return;
5068
5069 /*
5070 * Zero some important members.
5071 */
5072 PLSILOGICDEVICE pDevice = &pThisCC->paDeviceStates[iLUN];
5073 pDevice->pDrvBase = NULL;
5074 pDevice->pDrvMedia = NULL;
5075 pDevice->pDrvMediaEx = NULL;
5076}
5077
5078/**
5079 * @interface_method_impl{PDMDEVREG,pfnAttach}
5080 */
5081static DECLCALLBACK(int) lsilogicR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
5082{
5083 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5084 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
5085 PLSILOGICDEVICE pDevice = &pThisCC->paDeviceStates[iLUN];
5086 int rc;
5087
5088 if (iLUN >= pThis->cDeviceStates)
5089 return VERR_PDM_LUN_NOT_FOUND;
5090
5091 AssertMsgReturn(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
5092 ("LsiLogic: Device does not support hotplugging\n"),
5093 VERR_INVALID_PARAMETER);
5094
5095 /* the usual paranoia */
5096 AssertRelease(!pDevice->pDrvBase);
5097 AssertRelease(!pDevice->pDrvMedia);
5098 AssertRelease(!pDevice->pDrvMediaEx);
5099 Assert(pDevice->iLUN == iLUN);
5100
5101 /*
5102 * Try attach the block device and get the interfaces,
5103 * required as well as optional.
5104 */
5105 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, NULL);
5106 if (RT_SUCCESS(rc))
5107 {
5108 /* Query the media interface. */
5109 pDevice->pDrvMedia = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMIMEDIA);
5110 AssertMsgReturn(VALID_PTR(pDevice->pDrvMedia),
5111 ("LsiLogic configuration error: LUN#%d misses the basic media interface!\n", pDevice->iLUN),
5112 VERR_PDM_MISSING_INTERFACE);
5113
5114 /* Get the extended media interface. */
5115 pDevice->pDrvMediaEx = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMIMEDIAEX);
5116 AssertMsgReturn(VALID_PTR(pDevice->pDrvMediaEx),
5117 ("LsiLogic configuration error: LUN#%d misses the extended media interface!\n", pDevice->iLUN),
5118 VERR_PDM_MISSING_INTERFACE);
5119
5120 rc = pDevice->pDrvMediaEx->pfnIoReqAllocSizeSet(pDevice->pDrvMediaEx, sizeof(LSILOGICREQ));
5121 AssertMsgRCReturn(rc, ("LsiLogic configuration error: LUN#%u: Failed to set I/O request size!", pDevice->iLUN),
5122 rc);
5123 }
5124 else
5125 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", pDevice->iLUN, rc));
5126
5127 if (RT_FAILURE(rc))
5128 {
5129 pDevice->pDrvBase = NULL;
5130 pDevice->pDrvMedia = NULL;
5131 pDevice->pDrvMediaEx = NULL;
5132 }
5133 return rc;
5134}
5135
5136/**
5137 * Common reset worker.
5138 *
5139 * @param pDevIns The device instance data.
5140 */
5141static void lsilogicR3ResetCommon(PPDMDEVINS pDevIns)
5142{
5143 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5144 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
5145 int rc;
5146
5147 rc = lsilogicR3HardReset(pDevIns, pThis, pThisCC);
5148 AssertRC(rc);
5149
5150 vboxscsiHwReset(&pThisCC->VBoxSCSI);
5151}
5152
5153/**
5154 * @callback_method_impl{FNPDMDEVASYNCNOTIFY,
5155 * Callback employed by lsilogicR3Reset.}
5156 */
5157static DECLCALLBACK(bool) lsilogicR3IsAsyncResetDone(PPDMDEVINS pDevIns)
5158{
5159 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5160
5161 if (!lsilogicR3AllAsyncIOIsFinished(pDevIns))
5162 return false;
5163 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5164
5165 lsilogicR3ResetCommon(pDevIns);
5166 return true;
5167}
5168
5169/**
5170 * @interface_method_impl{PDMDEVREG,pfnReset}
5171 */
5172static DECLCALLBACK(void) lsilogicR3Reset(PPDMDEVINS pDevIns)
5173{
5174 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5175
5176 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
5177 if (!lsilogicR3AllAsyncIOIsFinished(pDevIns))
5178 PDMDevHlpSetAsyncNotification(pDevIns, lsilogicR3IsAsyncResetDone);
5179 else
5180 {
5181 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
5182 lsilogicR3ResetCommon(pDevIns);
5183 }
5184}
5185
5186/**
5187 * @interface_method_impl{PDMDEVREG,pfnPowerOff}
5188 */
5189static DECLCALLBACK(void) lsilogicR3PowerOff(PPDMDEVINS pDevIns)
5190{
5191 Log(("lsilogicR3PowerOff\n"));
5192 lsilogicR3SuspendOrPowerOff(pDevIns);
5193}
5194
5195/**
5196 * @interface_method_impl{PDMDEVREG,pfnDestruct}
5197 */
5198static DECLCALLBACK(int) lsilogicR3Destruct(PPDMDEVINS pDevIns)
5199{
5200 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
5201 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5202 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
5203
5204 PDMDevHlpCritSectDelete(pDevIns, &pThis->ReplyFreeQueueCritSect);
5205 PDMDevHlpCritSectDelete(pDevIns, &pThis->ReplyPostQueueCritSect);
5206 PDMDevHlpCritSectDelete(pDevIns, &pThis->RequestQueueCritSect);
5207 PDMDevHlpCritSectDelete(pDevIns, &pThis->ReplyFreeQueueWriteCritSect);
5208
5209 if (RTCritSectIsInitialized(&pThisCC->CritSectMemRegns))
5210 RTCritSectDelete(&pThisCC->CritSectMemRegns);
5211
5212 RTMemFree(pThisCC->paDeviceStates);
5213 pThisCC->paDeviceStates = NULL;
5214
5215 if (pThis->hEvtProcess != NIL_SUPSEMEVENT)
5216 {
5217 PDMDevHlpSUPSemEventClose(pDevIns, pThis->hEvtProcess);
5218 pThis->hEvtProcess = NIL_SUPSEMEVENT;
5219 }
5220
5221 lsilogicR3ConfigurationPagesFree(pThis, pThisCC);
5222 lsilogicR3MemRegionsFree(pThisCC);
5223 vboxscsiDestroy(&pThisCC->VBoxSCSI);
5224
5225 return VINF_SUCCESS;
5226}
5227
5228/**
5229 * @interface_method_impl{PDMDEVREG,pfnConstruct}
5230 */
5231static DECLCALLBACK(int) lsilogicR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
5232{
5233 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
5234 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5235 PLSILOGICSCSICC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PLSILOGICSCSICC);
5236 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
5237 int rc = VINF_SUCCESS;
5238
5239 /*
5240 * Initialize enought of the state to make the destructure not trip up.
5241 */
5242 pThis->hEvtProcess = NIL_SUPSEMEVENT;
5243 pThis->fBiosReqPending = false;
5244 RTListInit(&pThisCC->ListMemRegns);
5245 pThis->hMmioReg = NIL_IOMMMIOHANDLE;
5246 pThis->hMmioDiag = NIL_IOMMMIOHANDLE;
5247 pThis->hIoPortsReg = NIL_IOMIOPORTHANDLE;
5248 pThis->hIoPortsBios = NIL_IOMIOPORTHANDLE;
5249 pThisCC->pDevIns = pDevIns;
5250 pThisCC->IBase.pfnQueryInterface = lsilogicR3StatusQueryInterface;
5251 pThisCC->ILeds.pfnQueryStatusLed = lsilogicR3StatusQueryStatusLed;
5252
5253
5254 /*
5255 * Validate and read configuration.
5256 */
5257 PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns,
5258 "ReplyQueueDepth|"
5259 "RequestQueueDepth|"
5260 "ControllerType|"
5261 "NumPorts|"
5262 "Bootable",
5263 "");
5264
5265 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "ReplyQueueDepth",
5266 &pThis->cReplyQueueEntries, LSILOGICSCSI_REPLY_QUEUE_DEPTH_DEFAULT);
5267 if (RT_FAILURE(rc))
5268 return PDMDEV_SET_ERROR(pDevIns, rc,
5269 N_("LsiLogic configuration error: failed to read ReplyQueue as integer"));
5270 if ( pThis->cReplyQueueEntries < LSILOGICSCSI_REPLY_QUEUE_DEPTH_MIN
5271 || pThis->cReplyQueueEntries > LSILOGICSCSI_REPLY_QUEUE_DEPTH_MAX - 1 /* see +1 later in the function */)
5272 return PDMDevHlpVMSetError(pDevIns, VERR_OUT_OF_RANGE, RT_SRC_POS,
5273 N_("LsiLogic configuration error: 'ReplyQueueDepth' = %u is out of ranage (%u..%u)"),
5274 pThis->cReplyQueueEntries, LSILOGICSCSI_REPLY_QUEUE_DEPTH_MIN,
5275 LSILOGICSCSI_REPLY_QUEUE_DEPTH_MAX - 1);
5276 Log(("%s: ReplyQueueDepth=%u\n", __FUNCTION__, pThis->cReplyQueueEntries));
5277
5278 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "RequestQueueDepth",
5279 &pThis->cRequestQueueEntries, LSILOGICSCSI_REQUEST_QUEUE_DEPTH_DEFAULT);
5280 if (RT_FAILURE(rc))
5281 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic configuration error: failed to read RequestQueue as integer"));
5282 if ( pThis->cRequestQueueEntries < LSILOGICSCSI_REQUEST_QUEUE_DEPTH_MIN
5283 || pThis->cRequestQueueEntries > LSILOGICSCSI_REQUEST_QUEUE_DEPTH_MAX - 1 /* see +1 later in the function */)
5284 return PDMDevHlpVMSetError(pDevIns, VERR_OUT_OF_RANGE, RT_SRC_POS,
5285 N_("LsiLogic configuration error: 'RequestQueue' = %u is out of ranage (%u..%u)"),
5286 pThis->cRequestQueueEntries, LSILOGICSCSI_REQUEST_QUEUE_DEPTH_MIN,
5287 LSILOGICSCSI_REQUEST_QUEUE_DEPTH_MIN - 1);
5288 Log(("%s: RequestQueueDepth=%u\n", __FUNCTION__, pThis->cRequestQueueEntries));
5289
5290 char szCtrlType[64];
5291 rc = pHlp->pfnCFGMQueryStringDef(pCfg, "ControllerType", szCtrlType, sizeof(szCtrlType), LSILOGICSCSI_PCI_SPI_CTRLNAME);
5292 if (RT_FAILURE(rc))
5293 return PDMDEV_SET_ERROR(pDevIns, rc,
5294 N_("LsiLogic configuration error: failed to read ControllerType as string"));
5295 Log(("%s: ControllerType=%s\n", __FUNCTION__, szCtrlType));
5296 rc = lsilogicR3GetCtrlTypeFromString(pThis, szCtrlType);
5297 if (RT_FAILURE(rc))
5298 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic configuration error: failed to determine controller type from string"));
5299
5300 char szDevTag[20];
5301 RTStrPrintf(szDevTag, sizeof(szDevTag), "LSILOGIC%s-%u",
5302 pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI ? "SPI" : "SAS",
5303 iInstance);
5304
5305 rc = pHlp->pfnCFGMQueryU8(pCfg, "NumPorts", &pThis->cPorts);
5306 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
5307 {
5308 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
5309 pThis->cPorts = LSILOGICSCSI_PCI_SPI_PORTS_MAX;
5310 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
5311 pThis->cPorts = LSILOGICSCSI_PCI_SAS_PORTS_DEFAULT;
5312 else
5313 AssertMsgFailed(("Invalid controller type: %d\n", pThis->enmCtrlType));
5314 }
5315 else if (RT_FAILURE(rc))
5316 return PDMDEV_SET_ERROR(pDevIns, rc,
5317 N_("LsiLogic configuration error: failed to read NumPorts as integer"));
5318
5319 bool fBootable;
5320 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "Bootable", &fBootable, true);
5321 if (RT_FAILURE(rc))
5322 return PDMDEV_SET_ERROR(pDevIns, rc,
5323 N_("LsiLogic configuration error: failed to read Bootable as boolean"));
5324 Log(("%s: Bootable=%RTbool\n", __FUNCTION__, fBootable));
5325
5326 /* Init static parts. */
5327 PPDMPCIDEV pPciDev = pDevIns->apPciDevs[0];
5328 PDMPCIDEV_ASSERT_VALID(pDevIns, pPciDev);
5329
5330 PDMPciDevSetVendorId(pPciDev, LSILOGICSCSI_PCI_VENDOR_ID); /* LsiLogic */
5331 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
5332 {
5333 PDMPciDevSetDeviceId(pPciDev, LSILOGICSCSI_PCI_SPI_DEVICE_ID); /* LSI53C1030 */
5334 PDMPciDevSetSubSystemVendorId(pPciDev, LSILOGICSCSI_PCI_SPI_SUBSYSTEM_VENDOR_ID);
5335 PDMPciDevSetSubSystemId(pPciDev, LSILOGICSCSI_PCI_SPI_SUBSYSTEM_ID);
5336 }
5337 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
5338 {
5339 PDMPciDevSetDeviceId(pPciDev, LSILOGICSCSI_PCI_SAS_DEVICE_ID); /* SAS1068 */
5340 PDMPciDevSetSubSystemVendorId(pPciDev, LSILOGICSCSI_PCI_SAS_SUBSYSTEM_VENDOR_ID);
5341 PDMPciDevSetSubSystemId(pPciDev, LSILOGICSCSI_PCI_SAS_SUBSYSTEM_ID);
5342 }
5343 else
5344 AssertMsgFailed(("Invalid controller type: %d\n", pThis->enmCtrlType));
5345
5346 PDMPciDevSetClassProg(pPciDev, 0x00); /* SCSI */
5347 PDMPciDevSetClassSub(pPciDev, 0x00); /* SCSI */
5348 PDMPciDevSetClassBase(pPciDev, 0x01); /* Mass storage */
5349 PDMPciDevSetInterruptPin(pPciDev, 0x01); /* Interrupt pin A */
5350
5351# ifdef VBOX_WITH_MSI_DEVICES
5352 PDMPciDevSetStatus(pPciDev, VBOX_PCI_STATUS_CAP_LIST);
5353 PDMPciDevSetCapabilityList(pPciDev, 0x80);
5354# endif
5355
5356 /*
5357 * Create critical sections protecting the reply post and free queues.
5358 */
5359 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyFreeQueueCritSect, RT_SRC_POS, "%sRFQ", szDevTag);
5360 if (RT_FAILURE(rc))
5361 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: cannot create critical section for reply free queue"));
5362
5363 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyPostQueueCritSect, RT_SRC_POS, "%sRPQ", szDevTag);
5364 if (RT_FAILURE(rc))
5365 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: cannot create critical section for reply post queue"));
5366
5367 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->RequestQueueCritSect, RT_SRC_POS, "%sRQ", szDevTag);
5368 if (RT_FAILURE(rc))
5369 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: cannot create critical section for request queue"));
5370
5371 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyFreeQueueWriteCritSect, RT_SRC_POS, "%sRFQW", szDevTag);
5372 if (RT_FAILURE(rc))
5373 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: cannot create critical section for reply free queue write access"));
5374
5375 /*
5376 * Critical section protecting the memory regions.
5377 */
5378 rc = RTCritSectInit(&pThisCC->CritSectMemRegns);
5379 if (RT_FAILURE(rc))
5380 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: Failed to initialize critical section protecting the memory regions"));
5381
5382 /*
5383 * Register the PCI device, it's I/O regions.
5384 */
5385 rc = PDMDevHlpPCIRegister(pDevIns, pPciDev);
5386 if (RT_FAILURE(rc))
5387 return rc;
5388
5389# ifdef VBOX_WITH_MSI_DEVICES
5390 PDMMSIREG MsiReg;
5391 RT_ZERO(MsiReg);
5392 /* use this code for MSI-X support */
5393# if 0
5394 MsiReg.cMsixVectors = 1;
5395 MsiReg.iMsixCapOffset = 0x80;
5396 MsiReg.iMsixNextOffset = 0x00;
5397 MsiReg.iMsixBar = 3;
5398 Assert(pDevIns->pReg->cMaxMsixVectors >= MsiReg.cMsixVectors); /* fix device registration when enabling this */
5399# else
5400 MsiReg.cMsiVectors = 1;
5401 MsiReg.iMsiCapOffset = 0x80;
5402 MsiReg.iMsiNextOffset = 0x00;
5403# endif
5404 rc = PDMDevHlpPCIRegisterMsi(pDevIns, &MsiReg);
5405 if (RT_FAILURE (rc))
5406 {
5407 /* That's OK, we can work without MSI */
5408 PDMPciDevSetCapabilityList(pPciDev, 0x0);
5409 }
5410# endif
5411
5412 /* Region #0: I/O ports. */
5413 rc = PDMDevHlpPCIIORegionCreateIo(pDevIns, 0 /*iPciRegion*/, LSILOGIC_PCI_SPACE_IO_SIZE,
5414 lsilogicIOPortWrite, lsilogicIOPortRead, NULL /*pvUser*/,
5415 pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI ? "LsiLogic" : "LsiLogicSas",
5416 NULL /*paExtDesc*/, &pThis->hIoPortsReg);
5417 AssertRCReturn(rc, rc);
5418
5419 /* Region #1: MMIO.
5420 *
5421 * Non-4-byte read access to LSILOGIC_REG_REPLY_QUEUE may cause real strange behavior
5422 * because the data is part of a physical guest address. But some drivers use 1-byte
5423 * access to scan for SCSI controllers. So, we simplify our code by telling IOM to
5424 * read DWORDs.
5425 *
5426 * Regarding writes, we couldn't find anything specific in the specs about what should
5427 * happen. So far we've ignored unaligned writes and assumed the missing bytes of
5428 * byte and word access to be zero. We suspect that IOMMMIO_FLAGS_WRITE_ONLY_DWORD
5429 * or IOMMMIO_FLAGS_WRITE_DWORD_ZEROED would be the most appropriate here, but since we
5430 * don't have real hw to test one, the old behavior is kept exactly like it used to be.
5431 */
5432 /** @todo Check out unaligned writes and non-dword writes on real LsiLogic
5433 * hardware. */
5434 rc = PDMDevHlpPCIIORegionCreateMmio(pDevIns, 1 /*iPciRegion*/, LSILOGIC_PCI_SPACE_MEM_SIZE, PCI_ADDRESS_SPACE_MEM,
5435 lsilogicMMIOWrite, lsilogicMMIORead, NULL /*pvUser*/,
5436 IOMMMIO_FLAGS_READ_DWORD | IOMMMIO_FLAGS_WRITE_PASSTHRU,
5437 pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI ? "LsiLogic" : "LsiLogicSas",
5438 &pThis->hMmioReg);
5439 AssertRCReturn(rc, rc);
5440
5441 /* Region #2: MMIO - Diag. */
5442 rc = PDMDevHlpPCIIORegionCreateMmio(pDevIns, 2 /*iPciRegion*/, LSILOGIC_PCI_SPACE_MEM_SIZE, PCI_ADDRESS_SPACE_MEM,
5443 lsilogicDiagnosticWrite, lsilogicDiagnosticRead, NULL /*pvUser*/,
5444 IOMMMIO_FLAGS_READ_PASSTHRU | IOMMMIO_FLAGS_WRITE_PASSTHRU,
5445 pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI ? "LsiLogicDiag" : "LsiLogicSasDiag",
5446 &pThis->hMmioDiag);
5447 AssertRCReturn(rc, rc);
5448
5449 /*
5450 * We need one entry free in the queue.
5451 */
5452 pThis->cReplyQueueEntries++;
5453 AssertLogRelReturn(pThis->cReplyQueueEntries <= RT_ELEMENTS(pThis->aReplyFreeQueue), VERR_INTERNAL_ERROR_3);
5454 AssertLogRelReturn(pThis->cReplyQueueEntries <= RT_ELEMENTS(pThis->aReplyPostQueue), VERR_INTERNAL_ERROR_3);
5455
5456 pThis->cRequestQueueEntries++;
5457 AssertLogRelReturn(pThis->cRequestQueueEntries <= RT_ELEMENTS(pThis->aRequestQueue), VERR_INTERNAL_ERROR_3);
5458
5459 /*
5460 * Device states.
5461 */
5462 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
5463 pThis->cDeviceStates = pThis->cPorts * LSILOGICSCSI_PCI_SPI_DEVICES_PER_BUS_MAX;
5464 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
5465 pThis->cDeviceStates = pThis->cPorts * LSILOGICSCSI_PCI_SAS_DEVICES_PER_PORT_MAX;
5466 else
5467 AssertLogRelMsgFailedReturn(("Invalid controller type: %d\n", pThis->enmCtrlType), VERR_INTERNAL_ERROR_4);
5468
5469 /*
5470 * Create event semaphore and worker thread.
5471 */
5472 rc = PDMDevHlpThreadCreate(pDevIns, &pThisCC->pThreadWrk, pThis, lsilogicR3Worker,
5473 lsilogicR3WorkerWakeUp, 0, RTTHREADTYPE_IO, szDevTag);
5474 if (RT_FAILURE(rc))
5475 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5476 N_("LsiLogic: Failed to create worker thread %s"), szDevTag);
5477
5478 rc = PDMDevHlpSUPSemEventCreate(pDevIns, &pThis->hEvtProcess);
5479 if (RT_FAILURE(rc))
5480 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5481 N_("LsiLogic: Failed to create SUP event semaphore"));
5482
5483 /*
5484 * Allocate device states.
5485 */
5486 pThisCC->paDeviceStates = (PLSILOGICDEVICE)RTMemAllocZ(sizeof(LSILOGICDEVICE) * pThis->cDeviceStates);
5487 if (!pThisCC->paDeviceStates)
5488 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to allocate memory for device states"));
5489
5490 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
5491 {
5492 PLSILOGICDEVICE pDevice = &pThisCC->paDeviceStates[i];
5493
5494 /* Initialize static parts of the device. */
5495 pDevice->iLUN = i;
5496 pDevice->pDevIns = pDevIns;
5497 pDevice->Led.u32Magic = PDMLED_MAGIC;
5498 pDevice->IBase.pfnQueryInterface = lsilogicR3DeviceQueryInterface;
5499 pDevice->IMediaPort.pfnQueryDeviceLocation = lsilogicR3QueryDeviceLocation;
5500 pDevice->IMediaExPort.pfnIoReqCompleteNotify = lsilogicR3IoReqCompleteNotify;
5501 pDevice->IMediaExPort.pfnIoReqCopyFromBuf = lsilogicR3IoReqCopyFromBuf;
5502 pDevice->IMediaExPort.pfnIoReqCopyToBuf = lsilogicR3IoReqCopyToBuf;
5503 pDevice->IMediaExPort.pfnIoReqQueryBuf = NULL;
5504 pDevice->IMediaExPort.pfnIoReqQueryDiscardRanges = NULL;
5505 pDevice->IMediaExPort.pfnIoReqStateChanged = lsilogicR3IoReqStateChanged;
5506 pDevice->IMediaExPort.pfnMediumEjected = lsilogicR3MediumEjected;
5507 pDevice->ILed.pfnQueryStatusLed = lsilogicR3DeviceQueryStatusLed;
5508 RTStrPrintf(pDevice->szName, sizeof(pDevice->szName), "Device%u", i);
5509
5510 /* Attach SCSI driver. */
5511 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, pDevice->szName);
5512 if (RT_SUCCESS(rc))
5513 {
5514 /* Query the media interface. */
5515 pDevice->pDrvMedia = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMIMEDIA);
5516 AssertMsgReturn(VALID_PTR(pDevice->pDrvMedia),
5517 ("LsiLogic configuration error: LUN#%d misses the basic media interface!\n", pDevice->iLUN),
5518 VERR_PDM_MISSING_INTERFACE);
5519
5520 /* Get the extended media interface. */
5521 pDevice->pDrvMediaEx = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMIMEDIAEX);
5522 AssertMsgReturn(VALID_PTR(pDevice->pDrvMediaEx),
5523 ("LsiLogic configuration error: LUN#%d misses the extended media interface!\n", pDevice->iLUN),
5524 VERR_PDM_MISSING_INTERFACE);
5525
5526 rc = pDevice->pDrvMediaEx->pfnIoReqAllocSizeSet(pDevice->pDrvMediaEx, sizeof(LSILOGICREQ));
5527 if (RT_FAILURE(rc))
5528 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5529 N_("LsiLogic configuration error: LUN#%u: Failed to set I/O request size!"),
5530 pDevice->iLUN);
5531 }
5532 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
5533 {
5534 pDevice->pDrvBase = NULL;
5535 rc = VINF_SUCCESS;
5536 Log(("LsiLogic: no driver attached to device %s\n", pDevice->szName));
5537 }
5538 else
5539 {
5540 AssertLogRelMsgFailed(("LsiLogic: Failed to attach %s\n", pDevice->szName));
5541 return rc;
5542 }
5543 }
5544
5545 /*
5546 * Attach status driver (optional).
5547 */
5548 PPDMIBASE pBase;
5549 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThisCC->IBase, &pBase, "Status Port");
5550 if (RT_SUCCESS(rc))
5551 {
5552 pThisCC->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
5553 pThisCC->pMediaNotify = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIANOTIFY);
5554 }
5555 else
5556 AssertMsgReturn(rc == VERR_PDM_NO_ATTACHED_DRIVER,
5557 ("Failed to attach to status driver. rc=%Rrc\n", rc),
5558 PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot attach to status driver")));
5559
5560 /* Initialize the SCSI emulation for the BIOS. */
5561 rc = vboxscsiInitialize(&pThisCC->VBoxSCSI);
5562 if (RT_FAILURE(rc))
5563 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic failed to initialize BIOS SCSI interface"));
5564
5565 /*
5566 * Register I/O port space in ISA region for BIOS access
5567 * if the controller is marked as bootable.
5568 */
5569 if (fBootable)
5570 {
5571 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
5572 rc = PDMDevHlpIoPortCreateExAndMap(pDevIns, LSILOGIC_BIOS_IO_PORT, 4 /*cPorts*/, 0 /*fFlags*/,
5573 lsilogicR3IsaIOPortWrite, lsilogicR3IsaIOPortRead,
5574 lsilogicR3IsaIOPortWriteStr, lsilogicR3IsaIOPortReadStr, NULL /*pvUser*/,
5575 "LsiLogic BIOS", NULL /*paExtDesc*/, &pThis->hIoPortsBios);
5576 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
5577 rc = PDMDevHlpIoPortCreateExAndMap(pDevIns, LSILOGIC_SAS_BIOS_IO_PORT, 4 /*cPorts*/, 0 /*fFlags*/,
5578 lsilogicR3IsaIOPortWrite, lsilogicR3IsaIOPortRead,
5579 lsilogicR3IsaIOPortWriteStr, lsilogicR3IsaIOPortReadStr, NULL /*pvUser*/,
5580 "LsiLogic SAS BIOS", NULL /*paExtDesc*/, &pThis->hIoPortsBios);
5581 else
5582 AssertMsgFailedReturn(("Invalid controller type %d\n", pThis->enmCtrlType), VERR_INTERNAL_ERROR_3);
5583 AssertRCReturn(rc, PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot register legacy I/O handlers")));
5584 }
5585
5586 /* Register save state handlers. */
5587 rc = PDMDevHlpSSMRegisterEx(pDevIns, LSILOGIC_SAVED_STATE_VERSION, sizeof(*pThis), NULL,
5588 NULL, lsilogicR3LiveExec, NULL,
5589 NULL, lsilogicR3SaveExec, NULL,
5590 NULL, lsilogicR3LoadExec, lsilogicR3LoadDone);
5591 if (RT_FAILURE(rc))
5592 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot register save state handlers"));
5593
5594 pThis->enmWhoInit = LSILOGICWHOINIT_SYSTEM_BIOS;
5595
5596 /*
5597 * Register the info item.
5598 */
5599 char szTmp[128];
5600 RTStrPrintf(szTmp, sizeof(szTmp), "%s%u", pDevIns->pReg->szName, pDevIns->iInstance);
5601 PDMDevHlpDBGFInfoRegister(pDevIns, szTmp,
5602 pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI
5603 ? "LsiLogic SPI info."
5604 : "LsiLogic SAS info.", lsilogicR3Info);
5605
5606 /* Allocate configuration pages. */
5607 rc = lsilogicR3ConfigurationPagesAlloc(pThis, pThisCC);
5608 if (RT_FAILURE(rc))
5609 PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic: Failed to allocate memory for configuration pages"));
5610
5611 /* Perform hard reset. */
5612 rc = lsilogicR3HardReset(pDevIns, pThis, pThisCC);
5613 AssertRC(rc);
5614
5615 return rc;
5616}
5617
5618#else /* !IN_RING3 */
5619
5620/**
5621 * @callback_method_impl{PDMDEVREGR0,pfnConstruct}
5622 */
5623static DECLCALLBACK(int) lsilogicRZConstruct(PPDMDEVINS pDevIns)
5624{
5625 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
5626 PLSILOGICSCSI pThis = PDMDEVINS_2_DATA(pDevIns, PLSILOGICSCSI);
5627
5628 /* Setup callbacks for this context: */
5629 int rc = PDMDevHlpIoPortSetUpContext(pDevIns, pThis->hIoPortsReg, lsilogicIOPortWrite, lsilogicIOPortRead, NULL /*pvUser*/);
5630 AssertRCReturn(rc, rc);
5631
5632 rc = PDMDevHlpMmioSetUpContext(pDevIns, pThis->hMmioReg, lsilogicMMIOWrite, lsilogicMMIORead, NULL /*pvUser*/);
5633 AssertRCReturn(rc, rc);
5634
5635 rc = PDMDevHlpMmioSetUpContext(pDevIns, pThis->hMmioDiag, lsilogicDiagnosticWrite, lsilogicDiagnosticRead, NULL /*pvUser*/);
5636 AssertRCReturn(rc, rc);
5637
5638 return VINF_SUCCESS;
5639}
5640
5641#endif /* !IN_RING3 */
5642
5643/**
5644 * The device registration structure - SPI SCSI controller.
5645 */
5646const PDMDEVREG g_DeviceLsiLogicSCSI =
5647{
5648 /* .u32Version = */ PDM_DEVREG_VERSION,
5649 /* .uReserved0 = */ 0,
5650 /* .szName = */ "lsilogicscsi",
5651 /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RZ | PDM_DEVREG_FLAGS_NEW_STYLE
5652 | PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION | PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION,
5653 /* .fClass = */ PDM_DEVREG_CLASS_STORAGE,
5654 /* .cMaxInstances = */ ~0U,
5655 /* .uSharedVersion = */ 42,
5656 /* .cbInstanceShared = */ sizeof(LSILOGICSCSI),
5657 /* .cbInstanceCC = */ sizeof(LSILOGICSCSICC),
5658 /* .cbInstanceRC = */ sizeof(LSILOGICSCSIRC),
5659 /* .cMaxPciDevices = */ 1,
5660 /* .cMaxMsixVectors = */ 0,
5661 /* .pszDescription = */ "LSI Logic 53c1030 SCSI controller.\n",
5662#if defined(IN_RING3)
5663 /* .pszRCMod = */ "VBoxDDRC.rc",
5664 /* .pszR0Mod = */ "VBoxDDR0.r0",
5665 /* .pfnConstruct = */ lsilogicR3Construct,
5666 /* .pfnDestruct = */ lsilogicR3Destruct,
5667 /* .pfnRelocate = */ NULL,
5668 /* .pfnMemSetup = */ NULL,
5669 /* .pfnPowerOn = */ NULL,
5670 /* .pfnReset = */ lsilogicR3Reset,
5671 /* .pfnSuspend = */ lsilogicR3Suspend,
5672 /* .pfnResume = */ lsilogicR3Resume,
5673 /* .pfnAttach = */ lsilogicR3Attach,
5674 /* .pfnDetach = */ lsilogicR3Detach,
5675 /* .pfnQueryInterface = */ NULL,
5676 /* .pfnInitComplete = */ NULL,
5677 /* .pfnPowerOff = */ lsilogicR3PowerOff,
5678 /* .pfnSoftReset = */ NULL,
5679 /* .pfnReserved0 = */ NULL,
5680 /* .pfnReserved1 = */ NULL,
5681 /* .pfnReserved2 = */ NULL,
5682 /* .pfnReserved3 = */ NULL,
5683 /* .pfnReserved4 = */ NULL,
5684 /* .pfnReserved5 = */ NULL,
5685 /* .pfnReserved6 = */ NULL,
5686 /* .pfnReserved7 = */ NULL,
5687#elif defined(IN_RING0)
5688 /* .pfnEarlyConstruct = */ NULL,
5689 /* .pfnConstruct = */ lsilogicRZConstruct,
5690 /* .pfnDestruct = */ NULL,
5691 /* .pfnFinalDestruct = */ NULL,
5692 /* .pfnRequest = */ NULL,
5693 /* .pfnReserved0 = */ NULL,
5694 /* .pfnReserved1 = */ NULL,
5695 /* .pfnReserved2 = */ NULL,
5696 /* .pfnReserved3 = */ NULL,
5697 /* .pfnReserved4 = */ NULL,
5698 /* .pfnReserved5 = */ NULL,
5699 /* .pfnReserved6 = */ NULL,
5700 /* .pfnReserved7 = */ NULL,
5701#elif defined(IN_RC)
5702 /* .pfnConstruct = */ lsilogicRZConstruct,
5703 /* .pfnReserved0 = */ NULL,
5704 /* .pfnReserved1 = */ NULL,
5705 /* .pfnReserved2 = */ NULL,
5706 /* .pfnReserved3 = */ NULL,
5707 /* .pfnReserved4 = */ NULL,
5708 /* .pfnReserved5 = */ NULL,
5709 /* .pfnReserved6 = */ NULL,
5710 /* .pfnReserved7 = */ NULL,
5711#else
5712# error "Not in IN_RING3, IN_RING0 or IN_RC!"
5713#endif
5714 /* .u32VersionEnd = */ PDM_DEVREG_VERSION
5715};
5716
5717/**
5718 * The device registration structure - SAS controller.
5719 */
5720const PDMDEVREG g_DeviceLsiLogicSAS =
5721{
5722 /* .u32Version = */ PDM_DEVREG_VERSION,
5723 /* .uReserved0 = */ 0,
5724 /* .szName = */ "lsilogicsas",
5725 /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RZ | PDM_DEVREG_FLAGS_NEW_STYLE
5726 | PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION | PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION
5727 | PDM_DEVREG_FLAGS_FIRST_RESET_NOTIFICATION,
5728 /* .fClass = */ PDM_DEVREG_CLASS_STORAGE,
5729 /* .cMaxInstances = */ ~0U,
5730 /* .uSharedVersion = */ 42,
5731 /* .cbInstanceShared = */ sizeof(LSILOGICSCSI),
5732 /* .cbInstanceCC = */ sizeof(LSILOGICSCSICC),
5733 /* .cbInstanceRC = */ sizeof(LSILOGICSCSIRC),
5734 /* .cMaxPciDevices = */ 1,
5735 /* .cMaxMsixVectors = */ 0,
5736 /* .pszDescription = */ "LSI Logic SAS1068 controller.\n",
5737#if defined(IN_RING3)
5738 /* .pszRCMod = */ "VBoxDDRC.rc",
5739 /* .pszR0Mod = */ "VBoxDDR0.r0",
5740 /* .pfnConstruct = */ lsilogicR3Construct,
5741 /* .pfnDestruct = */ lsilogicR3Destruct,
5742 /* .pfnRelocate = */ NULL,
5743 /* .pfnMemSetup = */ NULL,
5744 /* .pfnPowerOn = */ NULL,
5745 /* .pfnReset = */ lsilogicR3Reset,
5746 /* .pfnSuspend = */ lsilogicR3Suspend,
5747 /* .pfnResume = */ lsilogicR3Resume,
5748 /* .pfnAttach = */ lsilogicR3Attach,
5749 /* .pfnDetach = */ lsilogicR3Detach,
5750 /* .pfnQueryInterface = */ NULL,
5751 /* .pfnInitComplete = */ NULL,
5752 /* .pfnPowerOff = */ lsilogicR3PowerOff,
5753 /* .pfnSoftReset = */ NULL,
5754 /* .pfnReserved0 = */ NULL,
5755 /* .pfnReserved1 = */ NULL,
5756 /* .pfnReserved2 = */ NULL,
5757 /* .pfnReserved3 = */ NULL,
5758 /* .pfnReserved4 = */ NULL,
5759 /* .pfnReserved5 = */ NULL,
5760 /* .pfnReserved6 = */ NULL,
5761 /* .pfnReserved7 = */ NULL,
5762#elif defined(IN_RING0)
5763 /* .pfnEarlyConstruct = */ NULL,
5764 /* .pfnConstruct = */ lsilogicRZConstruct,
5765 /* .pfnDestruct = */ NULL,
5766 /* .pfnFinalDestruct = */ NULL,
5767 /* .pfnRequest = */ NULL,
5768 /* .pfnReserved0 = */ NULL,
5769 /* .pfnReserved1 = */ NULL,
5770 /* .pfnReserved2 = */ NULL,
5771 /* .pfnReserved3 = */ NULL,
5772 /* .pfnReserved4 = */ NULL,
5773 /* .pfnReserved5 = */ NULL,
5774 /* .pfnReserved6 = */ NULL,
5775 /* .pfnReserved7 = */ NULL,
5776#elif defined(IN_RC)
5777 /* .pfnConstruct = */ lsilogicRZConstruct,
5778 /* .pfnReserved0 = */ NULL,
5779 /* .pfnReserved1 = */ NULL,
5780 /* .pfnReserved2 = */ NULL,
5781 /* .pfnReserved3 = */ NULL,
5782 /* .pfnReserved4 = */ NULL,
5783 /* .pfnReserved5 = */ NULL,
5784 /* .pfnReserved6 = */ NULL,
5785 /* .pfnReserved7 = */ NULL,
5786#else
5787# error "Not in IN_RING3, IN_RING0 or IN_RC!"
5788#endif
5789 /* .u32VersionEnd = */ PDM_DEVREG_VERSION
5790};
5791
5792#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