VirtualBox

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

Last change on this file since 65620 was 65445, checked in by vboxsync, 8 years ago

LsiLogic: Fix compatibility with Windows 10

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