VirtualBox

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

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

*: scm cleanup run.

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