VirtualBox

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

Last change on this file since 27411 was 27302, checked in by vboxsync, 15 years ago

IPRT: RIP RTCache*, RTMemCache* is way better...

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 202.0 KB
Line 
1/* $Id: DevLsiLogicSCSI.cpp 27302 2010-03-11 20:11:17Z vboxsync $ */
2/** @file
3 * VBox storage devices: LsiLogic LSI53c1030 SCSI controller.
4 */
5
6/*
7 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22//#define DEBUG
23#define LOG_GROUP LOG_GROUP_DEV_LSILOGICSCSI
24#include <VBox/pdmdev.h>
25#include <VBox/pdmqueue.h>
26#include <VBox/pdmcritsect.h>
27#include <VBox/scsi.h>
28#include <iprt/assert.h>
29#include <iprt/asm.h>
30#include <iprt/string.h>
31#ifdef IN_RING3
32# include <iprt/memcache.h>
33# include <iprt/mem.h>
34# include <iprt/param.h>
35# include <iprt/uuid.h>
36#endif
37
38#include "DevLsiLogicSCSI.h"
39#include "VBoxSCSI.h"
40
41#include "../Builtins.h"
42
43/** The current saved state version. */
44#define LSILOGIC_SAVED_STATE_VERSION 3
45/** The saved state version used by VirtualBox before SAS support was added. */
46#define LSILOGIC_SAVED_STATE_VERSION_PRE_SAS 2
47/** The saved state version used by VirtualBox 3.0 and earlier. It does not
48 * include the device config part. */
49#define LSILOGIC_SAVED_STATE_VERSION_VBOX_30 1
50
51/**
52 * Reply data.
53 */
54typedef struct LSILOGICSCSIREPLY
55{
56 /** Lower 32 bits of the reply address in memory. */
57 uint32_t u32HostMFALowAddress;
58 /** Full address of the reply in guest memory. */
59 RTGCPHYS GCPhysReplyAddress;
60 /** Size of the reply. */
61 uint32_t cbReply;
62 /** Different views to the reply depending on the request type. */
63 MptReplyUnion Reply;
64} LSILOGICSCSIREPLY, *PLSILOGICSCSIREPLY;
65
66/**
67 * State of a device attached to the buslogic host adapter.
68 *
69 * @implements PDMIBASE
70 * @implements PDMISCSIPORT
71 * @implements PDMILEDPORTS
72 */
73typedef struct LSILOGICDEVICE
74{
75 /** Pointer to the owning lsilogic device instance. - R3 pointer */
76 R3PTRTYPE(struct LSILOGICSCSI *) pLsiLogicR3;
77
78 /** LUN of the device. */
79 RTUINT iLUN;
80 /** Number of outstanding tasks on the port. */
81 volatile uint32_t cOutstandingRequests;
82
83#if HC_ARCH_BITS == 64
84 uint32_t Alignment0;
85#endif
86
87 /** Our base interace. */
88 PDMIBASE IBase;
89 /** SCSI port interface. */
90 PDMISCSIPORT ISCSIPort;
91 /** Led interface. */
92 PDMILEDPORTS ILed;
93 /** Pointer to the attached driver's base interface. */
94 R3PTRTYPE(PPDMIBASE) pDrvBase;
95 /** Pointer to the underlying SCSI connector interface. */
96 R3PTRTYPE(PPDMISCSICONNECTOR) pDrvSCSIConnector;
97 /** The status LED state for this device. */
98 PDMLED Led;
99
100} LSILOGICDEVICE, *PLSILOGICDEVICE;
101
102/**
103 * Device instance data for the emulated
104 * SCSI controller.
105 */
106typedef struct LSILOGICSCSI
107{
108 /** PCI device structure. */
109 PCIDEVICE PciDev;
110 /** Pointer to the device instance. - R3 ptr. */
111 PPDMDEVINSR3 pDevInsR3;
112 /** Pointer to the device instance. - R0 ptr. */
113 PPDMDEVINSR0 pDevInsR0;
114 /** Pointer to the device instance. - RC ptr. */
115 PPDMDEVINSRC pDevInsRC;
116
117 /** Flag whether the GC part of the device is enabled. */
118 bool fGCEnabled;
119 /** Flag whether the R0 part of the device is enabled. */
120 bool fR0Enabled;
121
122 /** The state the controller is currently in. */
123 LSILOGICSTATE enmState;
124 /** Who needs to init the driver to get into operational state. */
125 LSILOGICWHOINIT enmWhoInit;
126 /** Flag whether we are in doorbell function. */
127 bool fDoorbellInProgress;
128 /** Flag whether diagnostic access is enabled. */
129 bool fDiagnosticEnabled;
130
131 /** Flag whether a notification was send to R3. */
132 bool fNotificationSend;
133
134 /** Flag whether the guest enabled event notification from the IOC. */
135 bool fEventNotificationEnabled;
136
137#if HC_ARCH_BITS == 64
138 uint32_t Alignment0;
139#endif
140
141 /** Queue to send tasks to R3. - R3 ptr */
142 R3PTRTYPE(PPDMQUEUE) pNotificationQueueR3;
143 /** Queue to send tasks to R3. - R0 ptr */
144 R0PTRTYPE(PPDMQUEUE) pNotificationQueueR0;
145 /** Queue to send tasks to R3. - RC ptr */
146 RCPTRTYPE(PPDMQUEUE) pNotificationQueueRC;
147
148#if HC_ARCH_BITS == 64
149 uint32_t Alignment1;
150#endif
151
152 /** Number of device states allocated. */
153 uint32_t cDeviceStates;
154
155#if HC_ARCH_BITS == 64
156 uint32_t Alignment2;
157#endif
158
159 /** States for attached devices. */
160 R3PTRTYPE(PLSILOGICDEVICE) paDeviceStates;
161
162 /** MMIO address the device is mapped to. */
163 RTGCPHYS GCPhysMMIOBase;
164 /** I/O port address the device is mapped to. */
165 RTIOPORT IOPortBase;
166
167 /** Interrupt mask. */
168 volatile uint32_t uInterruptMask;
169 /** Interrupt status register. */
170 volatile uint32_t uInterruptStatus;
171
172 /** Buffer for messages which are passed
173 * through the doorbell using the
174 * handshake method. */
175 uint32_t aMessage[sizeof(MptConfigurationRequest)];
176 /** Actual position in the buffer. */
177 uint32_t iMessage;
178 /** Size of the message which is given in the doorbell message in dwords. */
179 uint32_t cMessage;
180
181 /** Reply buffer. */
182 MptReplyUnion ReplyBuffer;
183 /** Next entry to read. */
184 uint32_t uNextReplyEntryRead;
185 /** Size of the reply in the buffer in 16bit words. */
186 uint32_t cReplySize;
187
188 /** The fault code of the I/O controller if we are in the fault state. */
189 uint16_t u16IOCFaultCode;
190
191 /** Upper 32 bits of the message frame address to locate requests in guest memory. */
192 uint32_t u32HostMFAHighAddr;
193 /** Upper 32 bits of the sense buffer address. */
194 uint32_t u32SenseBufferHighAddr;
195 /** Maximum number of devices the driver reported he can handle. */
196 uint8_t cMaxDevices;
197 /** Maximum number of buses the driver reported he can handle. */
198 uint8_t cMaxBuses;
199 /** Current size of reply message frames in the guest. */
200 uint16_t cbReplyFrame;
201
202 /** Next key to write in the sequence to get access
203 * to diagnostic memory. */
204 uint32_t iDiagnosticAccess;
205
206 /** Number entries allocated for the reply queue. */
207 uint32_t cReplyQueueEntries;
208 /** Number entries allocated for the outstanding request queue. */
209 uint32_t cRequestQueueEntries;
210
211 uint32_t Alignment3;
212
213 /** Critical section protecting the reply post queue. */
214 PDMCRITSECT ReplyPostQueueCritSect;
215 /** Critical section protecting the reply free queue. */
216 PDMCRITSECT ReplyFreeQueueCritSect;
217
218 /** Pointer to the start of the reply free queue - R3. */
219 R3PTRTYPE(volatile uint32_t *) pReplyFreeQueueBaseR3;
220 /** Pointer to the start of the reply post queue - R3. */
221 R3PTRTYPE(volatile uint32_t *) pReplyPostQueueBaseR3;
222 /** Pointer to the start of the request queue - R3. */
223 R3PTRTYPE(volatile uint32_t *) pRequestQueueBaseR3;
224
225 /** Pointer to the start of the reply queue - R0. */
226 R0PTRTYPE(volatile uint32_t *) pReplyFreeQueueBaseR0;
227 /** Pointer to the start of the reply queue - R0. */
228 R0PTRTYPE(volatile uint32_t *) pReplyPostQueueBaseR0;
229 /** Pointer to the start of the request queue - R0. */
230 R0PTRTYPE(volatile uint32_t *) pRequestQueueBaseR0;
231
232 /** Pointer to the start of the reply queue - RC. */
233 RCPTRTYPE(volatile uint32_t *) pReplyFreeQueueBaseRC;
234 /** Pointer to the start of the reply queue - RC. */
235 RCPTRTYPE(volatile uint32_t *) pReplyPostQueueBaseRC;
236 /** Pointer to the start of the request queue - RC. */
237 RCPTRTYPE(volatile uint32_t *) pRequestQueueBaseRC;
238
239 /** Next free entry in the reply queue the guest can write a address to. */
240 volatile uint32_t uReplyFreeQueueNextEntryFreeWrite;
241 /** Next valid entry the controller can read a valid address for reply frames from. */
242 volatile uint32_t uReplyFreeQueueNextAddressRead;
243
244 /** Next free entry in the reply queue the guest can write a address to. */
245 volatile uint32_t uReplyPostQueueNextEntryFreeWrite;
246 /** Next valid entry the controller can read a valid address for reply frames from. */
247 volatile uint32_t uReplyPostQueueNextAddressRead;
248
249 /** Next free entry the guest can write a address to a request frame to. */
250 volatile uint32_t uRequestQueueNextEntryFreeWrite;
251 /** Next valid entry the controller can read a valid address for request frames from. */
252 volatile uint32_t uRequestQueueNextAddressRead;
253
254 /** Emulated controller type */
255 LSILOGICCTRLTYPE enmCtrlType;
256 /** Handle counter */
257 uint16_t u16NextHandle;
258
259 uint16_t u16Alignment4;
260 uint32_t u32Alignment5;
261
262 /** Number of ports this controller has. */
263 uint8_t cPorts;
264
265#if HC_ARCH_BITS == 64
266 uint32_t Alignment6;
267#endif
268
269 /** BIOS emulation. */
270 VBOXSCSI VBoxSCSI;
271 /** Cache for allocated tasks. */
272 R3PTRTYPE(RTMEMCACHE) hTaskCache;
273 /** Status LUN: The base interface. */
274 PDMIBASE IBase;
275 /** Status LUN: Leds interface. */
276 PDMILEDPORTS ILeds;
277 /** Status LUN: Partner of ILeds. */
278 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
279
280 R3PTRTYPE(PMptConfigurationPagesSupported) pConfigurationPages;
281} LSILOGISCSI, *PLSILOGICSCSI;
282
283/**
284 * Scatter gather list entry data.
285 */
286typedef struct LSILOGICTASKSTATESGENTRY
287{
288 /** Flag whether the buffer in the list is from the guest or an
289 * allocated temporary buffer because the segments in the guest
290 * are not sector aligned.
291 */
292 bool fGuestMemory;
293 /** Flag whether the buffer contains data or is the destination for the transfer. */
294 bool fBufferContainsData;
295 /** Pointer to the start of the buffer. */
296 void *pvBuf;
297 /** Size of the buffer. */
298 uint32_t cbBuf;
299 /** Flag dependent data. */
300 union
301 {
302 /** Data to handle direct mappings of guest buffers. */
303 PGMPAGEMAPLOCK PageLock;
304 /** The segment in the guest which is not sector aligned. */
305 RTGCPHYS GCPhysAddrBufferUnaligned;
306 } u;
307} LSILOGICTASKSTATESGENTRY, *PLSILOGICTASKSTATESGENTRY;
308
309/**
310 * Task state object which holds all neccessary data while
311 * processing the request from the guest.
312 */
313typedef struct LSILOGICTASKSTATE
314{
315 /** Target device. */
316 PLSILOGICDEVICE pTargetDevice;
317 /** The message request from the guest. */
318 MptRequestUnion GuestRequest;
319 /** Reply message if the request produces one. */
320 MptReplyUnion IOCReply;
321 /** SCSI request structure for the SCSI driver. */
322 PDMSCSIREQUEST PDMScsiRequest;
323 /** Address of the message request frame in guests memory.
324 * Used to read the S/G entries in the second step. */
325 RTGCPHYS GCPhysMessageFrameAddr;
326 /** Number of scatter gather list entries. */
327 uint32_t cSGListEntries;
328 /** How many entries would fit into the sg list. */
329 uint32_t cSGListSize;
330 /** How many times the list was too big. */
331 uint32_t cSGListTooBig;
332 /** Pointer to the first entry of the scatter gather list. */
333 PPDMDATASEG pSGListHead;
334 /** How many entries would fit into the sg info list. */
335 uint32_t cSGInfoSize;
336 /** Number of entries for the information entries. */
337 uint32_t cSGInfoEntries;
338 /** How many times the list was too big. */
339 uint32_t cSGInfoTooBig;
340 /** Pointer to the first mapping information entry. */
341 PLSILOGICTASKSTATESGENTRY paSGEntries;
342 /** Size of the temporary buffer for unaligned guest segments. */
343 uint32_t cbBufferUnaligned;
344 /** Pointer to the temporary buffer. */
345 void *pvBufferUnaligned;
346 /** Pointer to the sense buffer. */
347 uint8_t abSenseBuffer[18];
348 /** Flag whether the request was issued from the BIOS. */
349 bool fBIOS;
350} LSILOGICTASKSTATE, *PLSILOGICTASKSTATE;
351
352#ifndef VBOX_DEVICE_STRUCT_TESTCASE
353
354RT_C_DECLS_BEGIN
355PDMBOTHCBDECL(int) lsilogicIOPortWrite (PPDMDEVINS pDevIns, void *pvUser,
356 RTIOPORT Port, uint32_t u32, unsigned cb);
357PDMBOTHCBDECL(int) lsilogicIOPortRead (PPDMDEVINS pDevIns, void *pvUser,
358 RTIOPORT Port, uint32_t *pu32, unsigned cb);
359PDMBOTHCBDECL(int) lsilogicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser,
360 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
361PDMBOTHCBDECL(int) lsilogicMMIORead(PPDMDEVINS pDevIns, void *pvUser,
362 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
363PDMBOTHCBDECL(int) lsilogicDiagnosticWrite(PPDMDEVINS pDevIns, void *pvUser,
364 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
365PDMBOTHCBDECL(int) lsilogicDiagnosticRead(PPDMDEVINS pDevIns, void *pvUser,
366 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
367#ifdef IN_RING3
368static void lsilogicInitializeConfigurationPages(PLSILOGICSCSI pLsiLogic);
369static void lsilogicConfigurationPagesFree(PLSILOGICSCSI pThis);
370static int lsilogicProcessConfigurationRequest(PLSILOGICSCSI pLsiLogic, PMptConfigurationRequest pConfigurationReq,
371 PMptConfigurationReply pReply);
372#endif
373RT_C_DECLS_END
374
375#define PDMIBASE_2_PLSILOGICDEVICE(pInterface) ( (PLSILOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICDEVICE, IBase)) )
376#define PDMISCSIPORT_2_PLSILOGICDEVICE(pInterface) ( (PLSILOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICDEVICE, ISCSIPort)) )
377#define PDMILEDPORTS_2_PLSILOGICDEVICE(pInterface) ( (PLSILOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICDEVICE, ILed)) )
378#define LSILOGIC_RTGCPHYS_FROM_U32(Hi, Lo) ( (RTGCPHYS)RT_MAKE_U64(Lo, Hi) )
379#define PDMIBASE_2_PLSILOGICSCSI(pInterface) ( (PLSILOGICSCSI)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICSCSI, IBase)) )
380#define PDMILEDPORTS_2_PLSILOGICSCSI(pInterface) ( (PLSILOGICSCSI)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICSCSI, ILeds)) )
381
382/** Key sequence the guest has to write to enable access
383 * to diagnostic memory. */
384static const uint8_t g_lsilogicDiagnosticAccess[] = {0x04, 0x0b, 0x02, 0x07, 0x0d};
385
386/**
387 * Updates the status of the interrupt pin of the device.
388 *
389 * @returns nothing.
390 * @param pThis Pointer to the device instance data.
391 */
392static void lsilogicUpdateInterrupt(PLSILOGICSCSI pThis)
393{
394 uint32_t uIntSts;
395
396 LogFlowFunc(("Updating interrupts\n"));
397
398 /* Mask out doorbell status so that it does not affect interrupt updating. */
399 uIntSts = (ASMAtomicReadU32(&pThis->uInterruptStatus) & ~LSILOGIC_REG_HOST_INTR_STATUS_DOORBELL_STS);
400 /* Check maskable interrupts. */
401 uIntSts &= ~(pThis->uInterruptMask & ~LSILOGIC_REG_HOST_INTR_MASK_IRQ_ROUTING);
402
403 if (uIntSts)
404 {
405 LogFlowFunc(("Setting interrupt\n"));
406 PDMDevHlpPCISetIrq(pThis->CTX_SUFF(pDevIns), 0, 1);
407 }
408 else
409 {
410 LogFlowFunc(("Clearing interrupt\n"));
411 PDMDevHlpPCISetIrq(pThis->CTX_SUFF(pDevIns), 0, 0);
412 }
413}
414
415/**
416 * Sets a given interrupt status bit in the status register and
417 * updates the interupt status.
418 *
419 * @returns nothing.
420 * @param pLsiLogic Pointer to the device instance.
421 * @param uStatus The status bit to set.
422 */
423DECLINLINE(void) lsilogicSetInterrupt(PLSILOGICSCSI pLsiLogic, uint32_t uStatus)
424{
425 ASMAtomicOrU32(&pLsiLogic->uInterruptStatus, uStatus);
426 lsilogicUpdateInterrupt(pLsiLogic);
427}
428
429/**
430 * Clears a given interrupt status bit in the status register and
431 * updates the interupt status.
432 *
433 * @returns nothing.
434 * @param pLsiLogic Pointer to the device instance.
435 * @param uStatus The status bit to set.
436 */
437DECLINLINE(void) lsilogicClearInterrupt(PLSILOGICSCSI pLsiLogic, uint32_t uStatus)
438{
439 ASMAtomicAndU32(&pLsiLogic->uInterruptStatus, ~uStatus);
440 lsilogicUpdateInterrupt(pLsiLogic);
441}
442
443/**
444 * Sets the I/O controller into fault state and sets the fault code.
445 *
446 * @returns nothing
447 * @param pLsiLogic Pointer to the controller device instance.
448 * @param uIOCFaultCode Fault code to set.
449 */
450DECLINLINE(void) lsilogicSetIOCFaultCode(PLSILOGICSCSI pLsiLogic, uint16_t uIOCFaultCode)
451{
452 if (pLsiLogic->enmState != LSILOGICSTATE_FAULT)
453 {
454 Log(("%s: Setting I/O controller into FAULT state: uIOCFaultCode=%u\n", __FUNCTION__, uIOCFaultCode));
455 pLsiLogic->enmState = LSILOGICSTATE_FAULT;
456 pLsiLogic->u16IOCFaultCode = uIOCFaultCode;
457 }
458 else
459 {
460 Log(("%s: We are already in FAULT state\n"));
461 }
462}
463
464#ifdef IN_RING3
465/**
466 * Performs a hard reset on the controller.
467 *
468 * @returns VBox status code.
469 * @param pThis Pointer to the device instance to initialize.
470 */
471static int lsilogicHardReset(PLSILOGICSCSI pThis)
472{
473 pThis->enmState = LSILOGICSTATE_RESET;
474
475 /* The interrupts are masked out. */
476 pThis->uInterruptMask |= LSILOGIC_REG_HOST_INTR_MASK_DOORBELL |
477 LSILOGIC_REG_HOST_INTR_MASK_REPLY;
478 /* Reset interrupt states. */
479 pThis->uInterruptStatus = 0;
480 lsilogicUpdateInterrupt(pThis);
481
482 /* Reset the queues. */
483 pThis->uReplyFreeQueueNextEntryFreeWrite = 0;
484 pThis->uReplyFreeQueueNextAddressRead = 0;
485 pThis->uReplyPostQueueNextEntryFreeWrite = 0;
486 pThis->uReplyPostQueueNextAddressRead = 0;
487 pThis->uRequestQueueNextEntryFreeWrite = 0;
488 pThis->uRequestQueueNextAddressRead = 0;
489
490 /* Disable diagnostic access. */
491 pThis->iDiagnosticAccess = 0;
492
493 /* Set default values. */
494 pThis->cMaxDevices = pThis->cDeviceStates;
495 pThis->cMaxBuses = 1;
496 pThis->cbReplyFrame = 128; /* @todo Figure out where it is needed. */
497 pThis->u16NextHandle = 1;
498 /** @todo: Put stuff to reset here. */
499
500 lsilogicConfigurationPagesFree(pThis);
501 lsilogicInitializeConfigurationPages(pThis);
502
503 /* Mark that we finished performing the reset. */
504 pThis->enmState = LSILOGICSTATE_READY;
505 return VINF_SUCCESS;
506}
507
508/**
509 * Frees the configuration pages if allocated.
510 *
511 * @returns nothing.
512 * @param pThis The LsiLogic controller instance
513 */
514static void lsilogicConfigurationPagesFree(PLSILOGICSCSI pThis)
515{
516
517 if (pThis->pConfigurationPages)
518 {
519 /* Destroy device list if we emulate a SAS controller. */
520 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
521 {
522 PMptConfigurationPagesSas pSasPages = &pThis->pConfigurationPages->u.SasPages;
523 PMptSASDevice pSASDeviceCurr = pSasPages->pSASDeviceHead;
524
525 while (pSASDeviceCurr)
526 {
527 PMptSASDevice pFree = pSASDeviceCurr;
528
529 pSASDeviceCurr = pSASDeviceCurr->pNext;
530 RTMemFree(pFree);
531 }
532 if (pSasPages->paPHYs)
533 RTMemFree(pSasPages->paPHYs);
534 if (pSasPages->pManufacturingPage7)
535 RTMemFree(pSasPages->pManufacturingPage7);
536 if (pSasPages->pSASIOUnitPage0)
537 RTMemFree(pSasPages->pSASIOUnitPage0);
538 if (pSasPages->pSASIOUnitPage1)
539 RTMemFree(pSasPages->pSASIOUnitPage1);
540 }
541
542 RTMemFree(pThis->pConfigurationPages);
543 }
544}
545
546/**
547 * Finishes a context reply.
548 *
549 * @returns nothing
550 * @param pLsiLogic Pointer to the device instance
551 * @param u32MessageContext The message context ID to post.
552 */
553static void lsilogicFinishContextReply(PLSILOGICSCSI pLsiLogic, uint32_t u32MessageContext)
554{
555 int rc;
556 AssertMsg(!pLsiLogic->fDoorbellInProgress, ("We are in a doorbell function\n"));
557
558 /* Write message context ID into reply post queue. */
559 rc = PDMCritSectEnter(&pLsiLogic->ReplyPostQueueCritSect, VINF_SUCCESS);
560 AssertRC(rc);
561
562#if 0
563 /* Check for a entry in the queue. */
564 if (RT_UNLIKELY(pLsiLogic->uReplyPostQueueNextAddressRead != pLsiLogic->uReplyPostQueueNextEntryFreeWrite))
565 {
566 /* Set error code. */
567 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
568 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
569 return;
570 }
571#endif
572
573 /* We have a context reply. */
574 ASMAtomicWriteU32(&pLsiLogic->CTX_SUFF(pReplyPostQueueBase)[pLsiLogic->uReplyPostQueueNextEntryFreeWrite], u32MessageContext);
575 ASMAtomicIncU32(&pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
576 pLsiLogic->uReplyPostQueueNextEntryFreeWrite %= pLsiLogic->cReplyQueueEntries;
577
578 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
579
580 /* Set interrupt. */
581 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
582}
583#endif /* IN_RING3 */
584
585/**
586 * Takes neccessary steps to finish a reply frame.
587 *
588 * @returns nothing
589 * @param pLsiLogic Pointer to the device instance
590 * @param pReply Pointer to the reply message.
591 * @param fForceReplyFifo Flag whether the use of the reply post fifo is forced.
592 */
593static void lsilogicFinishAddressReply(PLSILOGICSCSI pLsiLogic, PMptReplyUnion pReply, bool fForceReplyFifo)
594{
595 /*
596 * If we are in a doorbell function we set the reply size now and
597 * set the system doorbell status interrupt to notify the guest that
598 * we are ready to send the reply.
599 */
600 if (pLsiLogic->fDoorbellInProgress && !fForceReplyFifo)
601 {
602 /* Set size of the reply in 16bit words. The size in the reply is in 32bit dwords. */
603 pLsiLogic->cReplySize = pReply->Header.u8MessageLength * 2;
604 Log(("%s: cReplySize=%u\n", __FUNCTION__, pLsiLogic->cReplySize));
605 pLsiLogic->uNextReplyEntryRead = 0;
606 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
607 }
608 else
609 {
610 /*
611 * The reply queues are only used if the request was fetched from the request queue.
612 * Requests from the request queue are always transferred to R3. So it is not possible
613 * that this case happens in R0 or GC.
614 */
615#ifdef IN_RING3
616 int rc;
617 /* Grab a free reply message from the queue. */
618 rc = PDMCritSectEnter(&pLsiLogic->ReplyFreeQueueCritSect, VINF_SUCCESS);
619 AssertRC(rc);
620
621#if 0
622 /* Check for a free reply frame. */
623 if (RT_UNLIKELY(pLsiLogic->uReplyFreeQueueNextAddressRead != pLsiLogic->uReplyFreeQueueNextEntryFreeWrite))
624 {
625 /* Set error code. */
626 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
627 PDMCritSectLeave(&pLsiLogic->ReplyFreeQueueCritSect);
628 return;
629 }
630#endif
631
632 uint32_t u32ReplyFrameAddressLow = pLsiLogic->CTX_SUFF(pReplyFreeQueueBase)[pLsiLogic->uReplyFreeQueueNextAddressRead];
633
634 pLsiLogic->uReplyFreeQueueNextAddressRead++;
635 pLsiLogic->uReplyFreeQueueNextAddressRead %= pLsiLogic->cReplyQueueEntries;
636
637 PDMCritSectLeave(&pLsiLogic->ReplyFreeQueueCritSect);
638
639 /* Build 64bit physical address. */
640 RTGCPHYS GCPhysReplyMessage = LSILOGIC_RTGCPHYS_FROM_U32(pLsiLogic->u32HostMFAHighAddr, u32ReplyFrameAddressLow);
641 size_t cbReplyCopied = (pLsiLogic->cbReplyFrame < sizeof(MptReplyUnion)) ? pLsiLogic->cbReplyFrame : sizeof(MptReplyUnion);
642
643 /* Write reply to guest memory. */
644 PDMDevHlpPhysWrite(pLsiLogic->CTX_SUFF(pDevIns), GCPhysReplyMessage, pReply, cbReplyCopied);
645
646 /* Write low 32bits of reply frame into post reply queue. */
647 rc = PDMCritSectEnter(&pLsiLogic->ReplyPostQueueCritSect, VINF_SUCCESS);
648 AssertRC(rc);
649
650#if 0
651 /* Check for a entry in the queue. */
652 if (RT_UNLIKELY(pLsiLogic->uReplyPostQueueNextAddressRead != pLsiLogic->uReplyPostQueueNextEntryFreeWrite))
653 {
654 /* Set error code. */
655 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
656 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
657 return;
658 }
659#endif
660
661 /* We have a address reply. Set the 31th bit to indicate that. */
662 ASMAtomicWriteU32(&pLsiLogic->CTX_SUFF(pReplyPostQueueBase)[pLsiLogic->uReplyPostQueueNextEntryFreeWrite],
663 RT_BIT(31) | (u32ReplyFrameAddressLow >> 1));
664 ASMAtomicIncU32(&pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
665 pLsiLogic->uReplyPostQueueNextEntryFreeWrite %= pLsiLogic->cReplyQueueEntries;
666
667 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
668
669 if (fForceReplyFifo)
670 {
671 pLsiLogic->fDoorbellInProgress = false;
672 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
673 }
674
675 /* Set interrupt. */
676 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
677#else
678 AssertMsgFailed(("This is not allowed to happen.\n"));
679#endif
680 }
681}
682
683#ifdef IN_RING3
684/**
685 * Processes a given Request from the guest
686 *
687 * @returns VBox status code.
688 * @param pLsiLogic Pointer to the device instance.
689 * @param pMessageHdr Pointer to the message header of the request.
690 * @param pReply Pointer to the reply.
691 */
692static int lsilogicProcessMessageRequest(PLSILOGICSCSI pLsiLogic, PMptMessageHdr pMessageHdr, PMptReplyUnion pReply)
693{
694 int rc = VINF_SUCCESS;
695 bool fForceReplyPostFifo = false;
696
697#ifdef DEBUG
698 if (pMessageHdr->u8Function < RT_ELEMENTS(g_apszMPTFunctionNames))
699 Log(("Message request function: %s\n", g_apszMPTFunctionNames[pMessageHdr->u8Function]));
700 else
701 Log(("Message request function: <unknown>\n"));
702#endif
703
704 memset(pReply, 0, sizeof(MptReplyUnion));
705
706 switch (pMessageHdr->u8Function)
707 {
708 case MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT:
709 {
710 PMptSCSITaskManagementRequest pTaskMgmtReq = (PMptSCSITaskManagementRequest)pMessageHdr;
711
712 pReply->SCSITaskManagement.u8MessageLength = 6; /* 6 32bit dwords. */
713 pReply->SCSITaskManagement.u8TaskType = pTaskMgmtReq->u8TaskType;
714 pReply->SCSITaskManagement.u32TerminationCount = 0;
715 fForceReplyPostFifo = true;
716 break;
717 }
718 case MPT_MESSAGE_HDR_FUNCTION_IOC_INIT:
719 {
720 /*
721 * This request sets the I/O controller to the
722 * operational state.
723 */
724 PMptIOCInitRequest pIOCInitReq = (PMptIOCInitRequest)pMessageHdr;
725
726 /* Update configuration values. */
727 pLsiLogic->enmWhoInit = (LSILOGICWHOINIT)pIOCInitReq->u8WhoInit;
728 pLsiLogic->cbReplyFrame = pIOCInitReq->u16ReplyFrameSize;
729 pLsiLogic->cMaxBuses = pIOCInitReq->u8MaxBuses;
730 pLsiLogic->cMaxDevices = pIOCInitReq->u8MaxDevices;
731 pLsiLogic->u32HostMFAHighAddr = pIOCInitReq->u32HostMfaHighAddr;
732 pLsiLogic->u32SenseBufferHighAddr = pIOCInitReq->u32SenseBufferHighAddr;
733
734 if (pLsiLogic->enmState == LSILOGICSTATE_READY)
735 {
736 pLsiLogic->enmState = LSILOGICSTATE_OPERATIONAL;
737 }
738
739 /* Return reply. */
740 pReply->IOCInit.u8MessageLength = 5;
741 pReply->IOCInit.u8WhoInit = pLsiLogic->enmWhoInit;
742 pReply->IOCInit.u8MaxDevices = pLsiLogic->cMaxDevices;
743 pReply->IOCInit.u8MaxBuses = pLsiLogic->cMaxBuses;
744 break;
745 }
746 case MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS:
747 {
748 pReply->IOCFacts.u8MessageLength = 15; /* 15 32bit dwords. */
749
750 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
751 {
752 pReply->IOCFacts.u16MessageVersion = 0x0102; /* Version from the specification. */
753 pReply->IOCFacts.u8NumberOfPorts = pLsiLogic->cPorts;
754 }
755 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
756 {
757 pReply->IOCFacts.u16MessageVersion = 0x0105; /* Version from the specification. */
758 pReply->IOCFacts.u8NumberOfPorts = pLsiLogic->cPorts;
759 }
760 else
761 AssertMsgFailed(("Invalid controller type %d\n", pLsiLogic->enmCtrlType));
762
763 pReply->IOCFacts.u8IOCNumber = 0; /* PCI function number. */
764 pReply->IOCFacts.u16IOCExceptions = 0;
765 pReply->IOCFacts.u8MaxChainDepth = LSILOGICSCSI_MAXIMUM_CHAIN_DEPTH;
766 pReply->IOCFacts.u8WhoInit = pLsiLogic->enmWhoInit;
767 pReply->IOCFacts.u8BlockSize = 12; /* Block size in 32bit dwords. This is the largest request we can get (SCSI I/O). */
768 pReply->IOCFacts.u8Flags = 0; /* Bit 0 is set if the guest must upload the FW prior to using the controller. Obviously not needed here. */
769 pReply->IOCFacts.u16ReplyQueueDepth = pLsiLogic->cReplyQueueEntries - 1; /* One entry is always free. */
770 pReply->IOCFacts.u16RequestFrameSize = 128; /* @todo Figure out where it is needed. */
771 pReply->IOCFacts.u16ProductID = 0xcafe; /* Our own product ID :) */
772 pReply->IOCFacts.u32CurrentHostMFAHighAddr = pLsiLogic->u32HostMFAHighAddr;
773 pReply->IOCFacts.u16GlobalCredits = pLsiLogic->cRequestQueueEntries - 1; /* One entry is always free. */
774
775 pReply->IOCFacts.u8EventState = 0; /* Event notifications not enabled. */
776 pReply->IOCFacts.u32CurrentSenseBufferHighAddr = pLsiLogic->u32SenseBufferHighAddr;
777 pReply->IOCFacts.u16CurReplyFrameSize = pLsiLogic->cbReplyFrame;
778 pReply->IOCFacts.u8MaxDevices = pLsiLogic->cMaxDevices;
779 pReply->IOCFacts.u8MaxBuses = pLsiLogic->cMaxBuses;
780 pReply->IOCFacts.u32FwImageSize = 0; /* No image needed. */
781 pReply->IOCFacts.u32FWVersion = 0;
782 break;
783 }
784 case MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS:
785 {
786 PMptPortFactsRequest pPortFactsReq = (PMptPortFactsRequest)pMessageHdr;
787
788 pReply->PortFacts.u8MessageLength = 10;
789 pReply->PortFacts.u8PortNumber = pPortFactsReq->u8PortNumber;
790
791 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
792 {
793 /* This controller only supports one bus with bus number 0. */
794 if (pPortFactsReq->u8PortNumber >= pLsiLogic->cPorts)
795 {
796 pReply->PortFacts.u8PortType = 0; /* Not existant. */
797 }
798 else
799 {
800 pReply->PortFacts.u8PortType = 0x01; /* SCSI Port. */
801 pReply->PortFacts.u16MaxDevices = LSILOGICSCSI_PCI_SPI_DEVICES_PER_BUS_MAX;
802 pReply->PortFacts.u16ProtocolFlags = RT_BIT(3) | RT_BIT(0); /* SCSI initiator and LUN supported. */
803 pReply->PortFacts.u16PortSCSIID = 7; /* Default */
804 pReply->PortFacts.u16MaxPersistentIDs = 0;
805 pReply->PortFacts.u16MaxPostedCmdBuffers = 0; /* Only applies for target mode which we dont support. */
806 pReply->PortFacts.u16MaxLANBuckets = 0; /* Only for the LAN controller. */
807 }
808 }
809 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
810 {
811 if (pPortFactsReq->u8PortNumber >= pLsiLogic->cPorts)
812 {
813 pReply->PortFacts.u8PortType = 0; /* Not existant. */
814 }
815 else
816 {
817 pReply->PortFacts.u8PortType = 0x30; /* SAS Port. */
818 pReply->PortFacts.u16MaxDevices = pLsiLogic->cPorts;
819 pReply->PortFacts.u16ProtocolFlags = RT_BIT(3) | RT_BIT(0); /* SCSI initiator and LUN supported. */
820 pReply->PortFacts.u16PortSCSIID = pLsiLogic->cPorts;
821 pReply->PortFacts.u16MaxPersistentIDs = 0;
822 pReply->PortFacts.u16MaxPostedCmdBuffers = 0; /* Only applies for target mode which we dont support. */
823 pReply->PortFacts.u16MaxLANBuckets = 0; /* Only for the LAN controller. */
824 }
825 }
826 else
827 AssertMsgFailed(("Invalid controller type %d\n", pLsiLogic->enmCtrlType));
828 break;
829 }
830 case MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE:
831 {
832 /*
833 * The port enable request notifies the IOC to make the port available and perform
834 * appropriate discovery on the associated link.
835 */
836 PMptPortEnableRequest pPortEnableReq = (PMptPortEnableRequest)pMessageHdr;
837
838 pReply->PortEnable.u8MessageLength = 5;
839 pReply->PortEnable.u8PortNumber = pPortEnableReq->u8PortNumber;
840 break;
841 }
842 case MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION:
843 {
844 PMptEventNotificationRequest pEventNotificationReq = (PMptEventNotificationRequest)pMessageHdr;
845
846 if (pEventNotificationReq->u8Switch)
847 pLsiLogic->fEventNotificationEnabled = true;
848 else
849 pLsiLogic->fEventNotificationEnabled = false;
850
851 pReply->EventNotification.u16EventDataLength = 1; /* 1 32bit D-Word. */
852 pReply->EventNotification.u8MessageLength = 8;
853 pReply->EventNotification.u8MessageFlags = (1 << 7);
854 pReply->EventNotification.u8AckRequired = 0;
855 pReply->EventNotification.u32Event = MPT_EVENT_EVENT_CHANGE;
856 pReply->EventNotification.u32EventContext = 0;
857 pReply->EventNotification.u32EventData = pLsiLogic->fEventNotificationEnabled ? 1 : 0;
858
859 break;
860 }
861 case MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK:
862 {
863 AssertMsgFailed(("todo"));
864 break;
865 }
866 case MPT_MESSAGE_HDR_FUNCTION_CONFIG:
867 {
868 PMptConfigurationRequest pConfigurationReq = (PMptConfigurationRequest)pMessageHdr;
869
870 rc = lsilogicProcessConfigurationRequest(pLsiLogic, pConfigurationReq, &pReply->Configuration);
871 AssertRC(rc);
872 break;
873 }
874 case MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST: /* Should be handled already. */
875 default:
876 AssertMsgFailed(("Invalid request function %#x\n", pMessageHdr->u8Function));
877 }
878
879 /* Copy common bits from request message frame to reply. */
880 pReply->Header.u8Function = pMessageHdr->u8Function;
881 pReply->Header.u32MessageContext = pMessageHdr->u32MessageContext;
882
883 lsilogicFinishAddressReply(pLsiLogic, pReply, fForceReplyPostFifo);
884 return rc;
885}
886#endif
887
888/**
889 * Writes a value to a register at a given offset.
890 *
891 * @returns VBox status code.
892 * @param pThis Pointer to the LsiLogic SCSI controller instance data.
893 * @param uOffset Offset of the register to write.
894 * @param pv Pointer to the value to write
895 * @param cb Number of bytes to write.
896 */
897static int lsilogicRegisterWrite(PLSILOGICSCSI pThis, uint32_t uOffset, void *pv, unsigned cb)
898{
899 uint32_t u32 = *(uint32_t *)pv;
900
901 LogFlowFunc(("pThis=%#p uOffset=%#x pv=%#p{%.*Rhxs} cb=%u\n", pThis, uOffset, pv, cb, pv, cb));
902
903 switch (uOffset)
904 {
905 case LSILOGIC_REG_REPLY_QUEUE:
906 {
907 /* Add the entry to the reply free queue. */
908 ASMAtomicWriteU32(&pThis->CTX_SUFF(pReplyFreeQueueBase)[pThis->uReplyFreeQueueNextEntryFreeWrite], u32);
909 pThis->uReplyFreeQueueNextEntryFreeWrite++;
910 pThis->uReplyFreeQueueNextEntryFreeWrite %= pThis->cReplyQueueEntries;
911 break;
912 }
913 case LSILOGIC_REG_REQUEST_QUEUE:
914 {
915 ASMAtomicWriteU32(&pThis->CTX_SUFF(pRequestQueueBase)[pThis->uRequestQueueNextEntryFreeWrite], u32);
916 pThis->uRequestQueueNextEntryFreeWrite++;
917 pThis->uRequestQueueNextEntryFreeWrite %= pThis->cRequestQueueEntries;
918
919 /* Send notification to R3 if there is not one send already. */
920 if (!ASMAtomicXchgBool(&pThis->fNotificationSend, true))
921 {
922 PPDMQUEUEITEMCORE pNotificationItem = PDMQueueAlloc(pThis->CTX_SUFF(pNotificationQueue));
923 AssertPtr(pNotificationItem);
924
925 PDMQueueInsert(pThis->CTX_SUFF(pNotificationQueue), pNotificationItem);
926 }
927 break;
928 }
929 case LSILOGIC_REG_DOORBELL:
930 {
931 /*
932 * When the guest writes to this register a real device would set the
933 * doorbell status bit in the interrupt status register to indicate that the IOP
934 * has still to process the message.
935 * The guest needs to wait with posting new messages here until the bit is cleared.
936 * Because the guest is not continuing execution while we are here we can skip this.
937 */
938 if (!pThis->fDoorbellInProgress)
939 {
940 uint32_t uFunction = LSILOGIC_REG_DOORBELL_GET_FUNCTION(u32);
941
942 switch (uFunction)
943 {
944 case LSILOGIC_DOORBELL_FUNCTION_IOC_MSG_UNIT_RESET:
945 {
946 pThis->enmState = LSILOGICSTATE_RESET;
947
948 /* Reset interrupt states. */
949 pThis->uInterruptMask = 0;
950 pThis->uInterruptStatus = 0;
951 lsilogicUpdateInterrupt(pThis);
952
953 /* Reset the queues. */
954 pThis->uReplyFreeQueueNextEntryFreeWrite = 0;
955 pThis->uReplyFreeQueueNextAddressRead = 0;
956 pThis->uReplyPostQueueNextEntryFreeWrite = 0;
957 pThis->uReplyPostQueueNextAddressRead = 0;
958 pThis->uRequestQueueNextEntryFreeWrite = 0;
959 pThis->uRequestQueueNextAddressRead = 0;
960 pThis->enmState = LSILOGICSTATE_READY;
961 break;
962 }
963 case LSILOGIC_DOORBELL_FUNCTION_IO_UNIT_RESET:
964 {
965 AssertMsgFailed(("todo\n"));
966 break;
967 }
968 case LSILOGIC_DOORBELL_FUNCTION_HANDSHAKE:
969 {
970 pThis->cMessage = LSILOGIC_REG_DOORBELL_GET_SIZE(u32);
971 pThis->iMessage = 0;
972 AssertMsg(pThis->cMessage <= RT_ELEMENTS(pThis->aMessage),
973 ("Message doesn't fit into the buffer, cMessage=%u", pThis->cMessage));
974 pThis->fDoorbellInProgress = true;
975 /* Update the interrupt status to notify the guest that a doorbell function was started. */
976 lsilogicSetInterrupt(pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
977 break;
978 }
979 case LSILOGIC_DOORBELL_FUNCTION_REPLY_FRAME_REMOVAL:
980 {
981 AssertMsgFailed(("todo\n"));
982 break;
983 }
984 default:
985 AssertMsgFailed(("Unknown function %u to perform\n", uFunction));
986 }
987 }
988 else
989 {
990 /*
991 * We are already performing a doorbell function.
992 * Get the remaining parameters.
993 */
994 AssertMsg(pThis->iMessage < RT_ELEMENTS(pThis->aMessage), ("Message is too big to fit into the buffer\n"));
995 /*
996 * If the last byte of the message is written, force a switch to R3 because some requests might force
997 * a reply through the FIFO which cannot be handled in GC or R0.
998 */
999#ifndef IN_RING3
1000 if (pThis->iMessage == pThis->cMessage - 1)
1001 return VINF_IOM_HC_MMIO_WRITE;
1002#endif
1003 pThis->aMessage[pThis->iMessage++] = u32;
1004#ifdef IN_RING3
1005 if (pThis->iMessage == pThis->cMessage)
1006 {
1007 int rc = lsilogicProcessMessageRequest(pThis, (PMptMessageHdr)pThis->aMessage, &pThis->ReplyBuffer);
1008 AssertRC(rc);
1009 }
1010#endif
1011 }
1012 break;
1013 }
1014 case LSILOGIC_REG_HOST_INTR_STATUS:
1015 {
1016 /*
1017 * Clear the bits the guest wants except the system doorbell interrupt and the IO controller
1018 * status bit.
1019 * The former bit is always cleared no matter what the guest writes to the register and
1020 * the latter one is read only.
1021 */
1022 pThis->uInterruptStatus = pThis->uInterruptStatus & ~LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL;
1023
1024 /*
1025 * Check if there is still a doorbell function in progress. Set the
1026 * system doorbell interrupt bit again if it is.
1027 * We do not use lsilogicSetInterrupt here because the interrupt status
1028 * is updated afterwards anyway.
1029 */
1030 if ( (pThis->fDoorbellInProgress)
1031 && (pThis->cMessage == pThis->iMessage))
1032 {
1033 if (pThis->uNextReplyEntryRead == pThis->cReplySize)
1034 {
1035 /* Reply finished. Reset doorbell in progress status. */
1036 Log(("%s: Doorbell function finished\n", __FUNCTION__));
1037 pThis->fDoorbellInProgress = false;
1038 }
1039 ASMAtomicOrU32(&pThis->uInterruptStatus, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
1040 }
1041
1042 lsilogicUpdateInterrupt(pThis);
1043 break;
1044 }
1045 case LSILOGIC_REG_HOST_INTR_MASK:
1046 {
1047 pThis->uInterruptMask = (u32 & LSILOGIC_REG_HOST_INTR_MASK_W_MASK);
1048 lsilogicUpdateInterrupt(pThis);
1049 break;
1050 }
1051 case LSILOGIC_REG_WRITE_SEQUENCE:
1052 {
1053 if (pThis->fDiagnosticEnabled)
1054 {
1055 /* Any value will cause a reset and disabling access. */
1056 pThis->fDiagnosticEnabled = false;
1057 pThis->iDiagnosticAccess = 0;
1058 }
1059 else if ((u32 & 0xf) == g_lsilogicDiagnosticAccess[pThis->iDiagnosticAccess])
1060 {
1061 pThis->iDiagnosticAccess++;
1062 if (pThis->iDiagnosticAccess == RT_ELEMENTS(g_lsilogicDiagnosticAccess))
1063 {
1064 /*
1065 * Key sequence successfully written. Enable access to diagnostic
1066 * memory and register.
1067 */
1068 pThis->fDiagnosticEnabled = true;
1069 }
1070 }
1071 else
1072 {
1073 /* Wrong value written - reset to beginning. */
1074 pThis->iDiagnosticAccess = 0;
1075 }
1076 break;
1077 }
1078 case LSILOGIC_REG_HOST_DIAGNOSTIC:
1079 {
1080#ifndef IN_RING3
1081 return VINF_IOM_HC_IOPORT_WRITE;
1082#else
1083 if (u32 & LSILOGIC_REG_HOST_DIAGNOSTIC_RESET_ADAPTER)
1084 {
1085 lsilogicHardReset(pThis);
1086 }
1087 break;
1088#endif
1089 }
1090 default: /* Ignore. */
1091 {
1092 break;
1093 }
1094 }
1095 return VINF_SUCCESS;
1096}
1097
1098/**
1099 * Reads the content of a register at a given offset.
1100 *
1101 * @returns VBox status code.
1102 * @param pThis Pointer to the LsiLogic SCSI controller instance data.
1103 * @param uOffset Offset of the register to read.
1104 * @param pv Where to store the content of the register.
1105 * @param cb Number of bytes to read.
1106 */
1107static int lsilogicRegisterRead(PLSILOGICSCSI pThis, uint32_t uOffset, void *pv, unsigned cb)
1108{
1109 uint32_t u32 = 0;
1110
1111 /* Align to a 4 byte offset. */
1112 switch (uOffset & ~3)
1113 {
1114 case LSILOGIC_REG_REPLY_QUEUE:
1115 {
1116 /*
1117 * Non 4-byte access may cause real strange behavior because the data is part of a physical guest address.
1118 * But some drivers use 1-byte access to scan for SCSI controllers.
1119 */
1120 if (RT_UNLIKELY(cb != 4))
1121 LogFlowFunc((": cb is not 4 (%u)\n", cb));
1122
1123 if (pThis->uReplyPostQueueNextEntryFreeWrite != pThis->uReplyPostQueueNextAddressRead)
1124 {
1125 u32 = pThis->CTX_SUFF(pReplyPostQueueBase)[pThis->uReplyPostQueueNextAddressRead];
1126 pThis->uReplyPostQueueNextAddressRead++;
1127 pThis->uReplyPostQueueNextAddressRead %= pThis->cReplyQueueEntries;
1128 }
1129 else
1130 {
1131 /* The reply post queue is empty. Reset interrupt. */
1132 u32 = UINT32_C(0xffffffff);
1133 lsilogicClearInterrupt(pThis, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
1134 }
1135 Log(("%s: Returning address %#x\n", __FUNCTION__, u32));
1136 break;
1137 }
1138 case LSILOGIC_REG_DOORBELL:
1139 {
1140 u32 = LSILOGIC_REG_DOORBELL_SET_STATE(pThis->enmState);
1141 u32 |= LSILOGIC_REG_DOORBELL_SET_USED(pThis->fDoorbellInProgress);
1142 u32 |= LSILOGIC_REG_DOORBELL_SET_WHOINIT(pThis->enmWhoInit);
1143 /*
1144 * If there is a doorbell function in progress we pass the return value
1145 * instead of the status code. We transfer 16bit of the reply
1146 * during one read.
1147 */
1148 if (pThis->fDoorbellInProgress)
1149 {
1150 /* Return next 16bit value. */
1151 u32 |= pThis->ReplyBuffer.au16Reply[pThis->uNextReplyEntryRead++];
1152 }
1153 else
1154 {
1155 /* We return the status code of the I/O controller. */
1156 u32 |= pThis->u16IOCFaultCode;
1157 }
1158 break;
1159 }
1160 case LSILOGIC_REG_HOST_INTR_STATUS:
1161 {
1162 u32 = pThis->uInterruptStatus;
1163 break;
1164 }
1165 case LSILOGIC_REG_HOST_INTR_MASK:
1166 {
1167 u32 = pThis->uInterruptMask;
1168 break;
1169 }
1170 case LSILOGIC_REG_HOST_DIAGNOSTIC:
1171 {
1172 if (pThis->fDiagnosticEnabled)
1173 u32 = LSILOGIC_REG_HOST_DIAGNOSTIC_DRWE;
1174 else
1175 u32 = 0;
1176 break;
1177 }
1178 case LSILOGIC_REG_TEST_BASE_ADDRESS: /* The spec doesn't say anything about these registers, so we just ignore them */
1179 case LSILOGIC_REG_DIAG_RW_DATA:
1180 case LSILOGIC_REG_DIAG_RW_ADDRESS:
1181 default: /* Ignore. */
1182 {
1183 break;
1184 }
1185 }
1186
1187 /* Clip data according to the read size. */
1188 switch (cb)
1189 {
1190 case 4:
1191 {
1192 *(uint32_t *)pv = u32;
1193 break;
1194 }
1195 case 2:
1196 {
1197 uint8_t uBitsOff = (uOffset - (uOffset & 3))*8;
1198
1199 u32 &= (0xffff << uBitsOff);
1200 *(uint16_t *)pv = (uint16_t)(u32 >> uBitsOff);
1201 break;
1202 }
1203 case 1:
1204 {
1205 uint8_t uBitsOff = (uOffset - (uOffset & 3))*8;
1206
1207 u32 &= (0xff << uBitsOff);
1208 *(uint8_t *)pv = (uint8_t)(u32 >> uBitsOff);
1209 break;
1210 }
1211 default:
1212 AssertMsgFailed(("Invalid access size %u\n", cb));
1213 }
1214
1215 LogFlowFunc(("pThis=%#p uOffset=%#x pv=%#p{%.*Rhxs} cb=%u\n", pThis, uOffset, pv, cb, pv, cb));
1216
1217 return VINF_SUCCESS;
1218}
1219
1220PDMBOTHCBDECL(int) lsilogicIOPortWrite (PPDMDEVINS pDevIns, void *pvUser,
1221 RTIOPORT Port, uint32_t u32, unsigned cb)
1222{
1223 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
1224 uint32_t uOffset = Port - pThis->IOPortBase;
1225
1226 Assert(cb <= 4);
1227
1228 int rc = lsilogicRegisterWrite(pThis, uOffset, &u32, cb);
1229 if (rc == VINF_IOM_HC_MMIO_WRITE)
1230 rc = VINF_IOM_HC_IOPORT_WRITE;
1231
1232 return rc;
1233}
1234
1235PDMBOTHCBDECL(int) lsilogicIOPortRead (PPDMDEVINS pDevIns, void *pvUser,
1236 RTIOPORT Port, uint32_t *pu32, unsigned cb)
1237{
1238 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
1239 uint32_t uOffset = Port - pThis->IOPortBase;
1240
1241 Assert(cb <= 4);
1242
1243 return lsilogicRegisterRead(pThis, uOffset, pu32, cb);
1244}
1245
1246PDMBOTHCBDECL(int) lsilogicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser,
1247 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
1248{
1249 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
1250 uint32_t uOffset = GCPhysAddr - pThis->GCPhysMMIOBase;
1251
1252 return lsilogicRegisterWrite(pThis, uOffset, pv, cb);
1253}
1254
1255PDMBOTHCBDECL(int) lsilogicMMIORead(PPDMDEVINS pDevIns, void *pvUser,
1256 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
1257{
1258 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
1259 uint32_t uOffset = GCPhysAddr - pThis->GCPhysMMIOBase;
1260
1261 return lsilogicRegisterRead(pThis, uOffset, pv, cb);
1262}
1263
1264PDMBOTHCBDECL(int) lsilogicDiagnosticWrite(PPDMDEVINS pDevIns, void *pvUser,
1265 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
1266{
1267 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
1268
1269 LogFlowFunc(("pThis=%#p GCPhysAddr=%RGp pv=%#p{%.*Rhxs} cb=%u\n", pThis, GCPhysAddr, pv, cb, pv, cb));
1270
1271 return VINF_SUCCESS;
1272}
1273
1274PDMBOTHCBDECL(int) lsilogicDiagnosticRead(PPDMDEVINS pDevIns, void *pvUser,
1275 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
1276{
1277 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
1278
1279 LogFlowFunc(("pThis=%#p GCPhysAddr=%RGp pv=%#p{%.*Rhxs} cb=%u\n", pThis, GCPhysAddr, pv, cb, pv, cb));
1280
1281 return VINF_SUCCESS;
1282}
1283
1284#ifdef IN_RING3
1285
1286/**
1287 * Copies a contigous buffer into the scatter gather list provided by the guest.
1288 *
1289 * @returns nothing
1290 * @param pTaskState Pointer to the task state which contains the SGL.
1291 * @param pvBuf Pointer to the buffer to copy.
1292 * @param cbCopy Number of bytes to copy.
1293 */
1294static void lsilogicScatterGatherListCopyFromBuffer(PLSILOGICTASKSTATE pTaskState, void *pvBuf, size_t cbCopy)
1295{
1296 unsigned cSGEntry = 0;
1297 PPDMDATASEG pSGEntry = &pTaskState->pSGListHead[cSGEntry];
1298 uint8_t *pu8Buf = (uint8_t *)pvBuf;
1299
1300 while (cSGEntry < pTaskState->cSGListEntries)
1301 {
1302 size_t cbToCopy = (cbCopy < pSGEntry->cbSeg) ? cbCopy : pSGEntry->cbSeg;
1303
1304 memcpy(pSGEntry->pvSeg, pu8Buf, cbToCopy);
1305
1306 cbCopy -= cbToCopy;
1307 /* We finished. */
1308 if (!cbCopy)
1309 break;
1310
1311 /* Advance the buffer. */
1312 pu8Buf += cbToCopy;
1313
1314 /* Go to the next entry in the list. */
1315 pSGEntry++;
1316 cSGEntry++;
1317 }
1318}
1319
1320/**
1321 * Copy a temporary buffer into a part of the guest scatter gather list
1322 * described by the given descriptor entry.
1323 *
1324 * @returns nothing.
1325 * @param pDevIns Pointer to the device instance data.
1326 * @param pSGInfo Pointer to the segment info structure which describes the guest segments
1327 * to write to which are unaligned.
1328 */
1329static void lsilogicCopyFromBufferIntoSGList(PPDMDEVINS pDevIns, PLSILOGICTASKSTATESGENTRY pSGInfo)
1330{
1331 RTGCPHYS GCPhysBuffer = pSGInfo->u.GCPhysAddrBufferUnaligned;
1332
1333 AssertMsg(!pSGInfo->fGuestMemory, ("This is not possible\n"));
1334
1335 /* Copy into SG entry. */
1336 PDMDevHlpPhysWrite(pDevIns, GCPhysBuffer, pSGInfo->pvBuf, pSGInfo->cbBuf);
1337
1338}
1339
1340/**
1341 * Copy a part of the guest scatter gather list into a temporary buffer.
1342 *
1343 * @returns nothing.
1344 * @param pDevIns Pointer to the device instance data.
1345 * @param pSGInfo Pointer to the segment info structure which describes the guest segments
1346 * to read from which are unaligned.
1347 */
1348static void lsilogicCopyFromSGListIntoBuffer(PPDMDEVINS pDevIns, PLSILOGICTASKSTATESGENTRY pSGInfo)
1349{
1350 RTGCPHYS GCPhysBuffer = pSGInfo->u.GCPhysAddrBufferUnaligned;
1351
1352 AssertMsg(!pSGInfo->fGuestMemory, ("This is not possible\n"));
1353
1354 /* Copy into temporary buffer. */
1355 PDMDevHlpPhysRead(pDevIns, GCPhysBuffer, pSGInfo->pvBuf, pSGInfo->cbBuf);
1356}
1357
1358static int lsilogicScatterGatherListAllocate(PLSILOGICTASKSTATE pTaskState, uint32_t cSGList, uint32_t cSGInfo, uint32_t cbUnaligned)
1359{
1360 if (pTaskState->cSGListSize < cSGList)
1361 {
1362 /* The entries are not allocated yet or the number is too small. */
1363 if (pTaskState->cSGListSize)
1364 RTMemFree(pTaskState->pSGListHead);
1365
1366 /* Allocate R3 scatter gather list. */
1367 pTaskState->pSGListHead = (PPDMDATASEG)RTMemAllocZ(cSGList * sizeof(PDMDATASEG));
1368 if (!pTaskState->pSGListHead)
1369 return VERR_NO_MEMORY;
1370
1371 /* Reset usage statistics. */
1372 pTaskState->cSGListSize = cSGList;
1373 pTaskState->cSGListEntries = cSGList;
1374 pTaskState->cSGListTooBig = 0;
1375 }
1376 else if (pTaskState->cSGListSize > cSGList)
1377 {
1378 /*
1379 * The list is too big. Increment counter.
1380 * So that the destroying function can free
1381 * the list if it is too big too many times
1382 * in a row.
1383 */
1384 pTaskState->cSGListEntries = cSGList;
1385 pTaskState->cSGListTooBig++;
1386 }
1387 else
1388 {
1389 /*
1390 * Needed entries matches current size.
1391 * Reset counter.
1392 */
1393 pTaskState->cSGListEntries = cSGList;
1394 pTaskState->cSGListTooBig = 0;
1395 }
1396
1397 if (pTaskState->cSGInfoSize < cSGInfo)
1398 {
1399 /* The entries are not allocated yet or the number is too small. */
1400 if (pTaskState->cSGInfoSize)
1401 RTMemFree(pTaskState->paSGEntries);
1402
1403 pTaskState->paSGEntries = (PLSILOGICTASKSTATESGENTRY)RTMemAllocZ(cSGInfo * sizeof(LSILOGICTASKSTATESGENTRY));
1404 if (!pTaskState->paSGEntries)
1405 return VERR_NO_MEMORY;
1406
1407 /* Reset usage statistics. */
1408 pTaskState->cSGInfoSize = cSGInfo;
1409 pTaskState->cSGInfoEntries = cSGInfo;
1410 pTaskState->cSGInfoTooBig = 0;
1411 }
1412 else if (pTaskState->cSGInfoSize > cSGInfo)
1413 {
1414 /*
1415 * The list is too big. Increment counter.
1416 * So that the destroying function can free
1417 * the list if it is too big too many times
1418 * in a row.
1419 */
1420 pTaskState->cSGInfoEntries = cSGInfo;
1421 pTaskState->cSGInfoTooBig++;
1422 }
1423 else
1424 {
1425 /*
1426 * Needed entries matches current size.
1427 * Reset counter.
1428 */
1429 pTaskState->cSGInfoEntries = cSGInfo;
1430 pTaskState->cSGInfoTooBig = 0;
1431 }
1432
1433
1434 if (pTaskState->cbBufferUnaligned < cbUnaligned)
1435 {
1436 if (pTaskState->pvBufferUnaligned)
1437 RTMemFree(pTaskState->pvBufferUnaligned);
1438
1439 Log(("%s: Allocating buffer for unaligned segments cbUnaligned=%u\n", __FUNCTION__, cbUnaligned));
1440
1441 pTaskState->pvBufferUnaligned = RTMemAllocZ(cbUnaligned);
1442 if (!pTaskState->pvBufferUnaligned)
1443 return VERR_NO_MEMORY;
1444
1445 pTaskState->cbBufferUnaligned = cbUnaligned;
1446 }
1447
1448 /* Make debugging easier. */
1449#ifdef DEBUG
1450 memset(pTaskState->pSGListHead, 0, pTaskState->cSGListSize * sizeof(PDMDATASEG));
1451 memset(pTaskState->paSGEntries, 0, pTaskState->cSGInfoSize * sizeof(LSILOGICTASKSTATESGENTRY));
1452 if (pTaskState->pvBufferUnaligned)
1453 memset(pTaskState->pvBufferUnaligned, 0, pTaskState->cbBufferUnaligned);
1454#endif
1455 return VINF_SUCCESS;
1456}
1457
1458/**
1459 * Destroy a scatter gather list.
1460 *
1461 * @returns nothing.
1462 * @param pLsiLogic Pointer to the LsiLogic SCSI controller.
1463 * @param pTaskState Pointer to the task state.
1464 */
1465static void lsilogicScatterGatherListDestroy(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
1466{
1467 PPDMDEVINS pDevIns = pLsiLogic->CTX_SUFF(pDevIns);
1468 PLSILOGICTASKSTATESGENTRY pSGInfoCurr = pTaskState->paSGEntries;
1469
1470 for (unsigned i = 0; i < pTaskState->cSGInfoEntries; i++)
1471 {
1472 if (pSGInfoCurr->fGuestMemory)
1473 {
1474 /* Release the lock. */
1475 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &pSGInfoCurr->u.PageLock);
1476 }
1477 else if (!pSGInfoCurr->fBufferContainsData)
1478 {
1479 /* Copy the data into the guest segments now. */
1480 lsilogicCopyFromBufferIntoSGList(pLsiLogic->CTX_SUFF(pDevIns), pSGInfoCurr);
1481 }
1482
1483 pSGInfoCurr++;
1484 }
1485
1486 /* Free allocated memory if the list was too big too many times. */
1487 if (pTaskState->cSGListTooBig >= LSILOGIC_NR_OF_ALLOWED_BIGGER_LISTS)
1488 {
1489 RTMemFree(pTaskState->pSGListHead);
1490 RTMemFree(pTaskState->paSGEntries);
1491 if (pTaskState->pvBufferUnaligned)
1492 RTMemFree(pTaskState->pvBufferUnaligned);
1493 pTaskState->cSGListSize = 0;
1494 pTaskState->cSGInfoSize = 0;
1495 pTaskState->cSGInfoEntries = 0;
1496 pTaskState->cSGListTooBig = 0;
1497 pTaskState->pSGListHead = NULL;
1498 pTaskState->paSGEntries = NULL;
1499 pTaskState->pvBufferUnaligned = NULL;
1500 pTaskState->cbBufferUnaligned = 0;
1501 }
1502}
1503
1504#ifdef DEBUG
1505/**
1506 * Dump an SG entry.
1507 *
1508 * @returns nothing.
1509 * @param pSGEntry Pointer to the SG entry to dump
1510 */
1511static void lsilogicDumpSGEntry(PMptSGEntryUnion pSGEntry)
1512{
1513 switch (pSGEntry->Simple32.u2ElementType)
1514 {
1515 case MPTSGENTRYTYPE_SIMPLE:
1516 {
1517 Log(("%s: Dumping info for SIMPLE SG entry:\n", __FUNCTION__));
1518 Log(("%s: u24Length=%u\n", __FUNCTION__, pSGEntry->Simple32.u24Length));
1519 Log(("%s: fEndOfList=%d\n", __FUNCTION__, pSGEntry->Simple32.fEndOfList));
1520 Log(("%s: f64BitAddress=%d\n", __FUNCTION__, pSGEntry->Simple32.f64BitAddress));
1521 Log(("%s: fBufferContainsData=%d\n", __FUNCTION__, pSGEntry->Simple32.fBufferContainsData));
1522 Log(("%s: fLocalAddress=%d\n", __FUNCTION__, pSGEntry->Simple32.fLocalAddress));
1523 Log(("%s: fEndOfBuffer=%d\n", __FUNCTION__, pSGEntry->Simple32.fEndOfBuffer));
1524 Log(("%s: fLastElement=%d\n", __FUNCTION__, pSGEntry->Simple32.fLastElement));
1525 Log(("%s: u32DataBufferAddressLow=%u\n", __FUNCTION__, pSGEntry->Simple32.u32DataBufferAddressLow));
1526 if (pSGEntry->Simple32.f64BitAddress)
1527 {
1528 Log(("%s: u32DataBufferAddressHigh=%u\n", __FUNCTION__, pSGEntry->Simple64.u32DataBufferAddressHigh));
1529 Log(("%s: GCDataBufferAddress=%RGp\n", __FUNCTION__,
1530 ((uint64_t)pSGEntry->Simple64.u32DataBufferAddressHigh << 32) | pSGEntry->Simple64.u32DataBufferAddressLow));
1531 }
1532 else
1533 Log(("%s: GCDataBufferAddress=%RGp\n", __FUNCTION__, pSGEntry->Simple32.u32DataBufferAddressLow));
1534
1535 break;
1536 }
1537 case MPTSGENTRYTYPE_CHAIN:
1538 {
1539 Log(("%s: Dumping info for CHAIN SG entry:\n", __FUNCTION__));
1540 Log(("%s: u16Length=%u\n", __FUNCTION__, pSGEntry->Chain.u16Length));
1541 Log(("%s: u8NExtChainOffset=%d\n", __FUNCTION__, pSGEntry->Chain.u8NextChainOffset));
1542 Log(("%s: f64BitAddress=%d\n", __FUNCTION__, pSGEntry->Chain.f64BitAddress));
1543 Log(("%s: fLocalAddress=%d\n", __FUNCTION__, pSGEntry->Chain.fLocalAddress));
1544 Log(("%s: u32SegmentAddressLow=%u\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressLow));
1545 Log(("%s: u32SegmentAddressHigh=%u\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressHigh));
1546 if (pSGEntry->Chain.f64BitAddress)
1547 Log(("%s: GCSegmentAddress=%RGp\n", __FUNCTION__,
1548 ((uint64_t)pSGEntry->Chain.u32SegmentAddressHigh << 32) | pSGEntry->Chain.u32SegmentAddressLow));
1549 else
1550 Log(("%s: GCSegmentAddress=%RGp\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressLow));
1551 break;
1552 }
1553 }
1554}
1555#endif
1556
1557/**
1558 * Create scatter gather list descriptors.
1559 *
1560 * @returns VBox status code.
1561 * @param pLsiLogic Pointer to the LsiLogic SCSI controller.
1562 * @param pTaskState Pointer to the task state.
1563 * @param GCPhysSGLStart Guest physical address of the first SG entry.
1564 * @param uChainOffset Offset in bytes from the beginning of the SGL segment to the chain element.
1565 * @thread EMT
1566 */
1567static int lsilogicScatterGatherListCreate(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState,
1568 RTGCPHYS GCPhysSGLStart, uint32_t uChainOffset)
1569{
1570 int rc = VINF_SUCCESS;
1571 PPDMDEVINS pDevIns = pLsiLogic->CTX_SUFF(pDevIns);
1572 PVM pVM = PDMDevHlpGetVM(pDevIns);
1573 bool fUnaligned; /* Flag whether the current buffer is unaligned. */
1574 uint32_t cbUnaligned; /* Size of the unaligned buffers. */
1575 uint32_t cSGEntriesR3 = 0;
1576 uint32_t cSGInfo = 0;
1577 uint32_t cbSegment = 0;
1578 PLSILOGICTASKSTATESGENTRY pSGInfoCurr = NULL;
1579 uint8_t *pu8BufferUnalignedPos = NULL;
1580 uint8_t *pbBufferUnalignedSGInfoPos = NULL;
1581 uint32_t cbUnalignedComplete = 0;
1582 bool fDoMapping = false;
1583 bool fEndOfList;
1584 RTGCPHYS GCPhysSGEntryNext;
1585 RTGCPHYS GCPhysSegmentStart;
1586 uint32_t uChainOffsetNext;
1587
1588 /*
1589 * Two passes - one to count needed scatter gather list entries and needed unaligned
1590 * buffers and one to actually map the SG list into R3.
1591 */
1592 for (int i = 0; i < 2; i++)
1593 {
1594 fUnaligned = false;
1595 cbUnaligned = 0;
1596 fEndOfList = false;
1597
1598 GCPhysSGEntryNext = GCPhysSGLStart;
1599 uChainOffsetNext = uChainOffset;
1600 GCPhysSegmentStart = GCPhysSGLStart;
1601
1602 if (fDoMapping)
1603 {
1604 Log(("%s: cSGInfo=%u\n", __FUNCTION__, cSGInfo));
1605
1606 /* The number of needed SG entries in R3 is known. Allocate needed memory. */
1607 rc = lsilogicScatterGatherListAllocate(pTaskState, cSGInfo, cSGInfo, cbUnalignedComplete);
1608 AssertMsgRC(rc, ("Failed to allocate scatter gather array rc=%Rrc\n", rc));
1609
1610 /* We are now able to map the pages into R3. */
1611 pSGInfoCurr = pTaskState->paSGEntries;
1612 /* Initialize first segment to remove the need for additional if checks later in the code. */
1613 pSGInfoCurr->fGuestMemory= false;
1614 pu8BufferUnalignedPos = (uint8_t *)pTaskState->pvBufferUnaligned;
1615 pbBufferUnalignedSGInfoPos = pu8BufferUnalignedPos;
1616 }
1617
1618 /* Go through the list until we reach the end. */
1619 while (!fEndOfList)
1620 {
1621 bool fEndOfSegment = false;
1622
1623 while (!fEndOfSegment)
1624 {
1625 MptSGEntryUnion SGEntry;
1626
1627 Log(("%s: Reading SG entry from %RGp\n", __FUNCTION__, GCPhysSGEntryNext));
1628
1629 /* Read the entry. */
1630 PDMDevHlpPhysRead(pDevIns, GCPhysSGEntryNext, &SGEntry, sizeof(MptSGEntryUnion));
1631
1632#ifdef DEBUG
1633 lsilogicDumpSGEntry(&SGEntry);
1634#endif
1635
1636 AssertMsg(SGEntry.Simple32.u2ElementType == MPTSGENTRYTYPE_SIMPLE, ("Invalid SG entry type\n"));
1637
1638 /* Check if this is a zero element. */
1639 if ( !SGEntry.Simple32.u24Length
1640 && SGEntry.Simple32.fEndOfList
1641 && SGEntry.Simple32.fEndOfBuffer)
1642 {
1643 pTaskState->cSGListEntries = 0;
1644 pTaskState->cSGInfoEntries = 0;
1645 return VINF_SUCCESS;
1646 }
1647
1648 uint32_t cbDataToTransfer = SGEntry.Simple32.u24Length;
1649 bool fBufferContainsData = !!SGEntry.Simple32.fBufferContainsData;
1650 RTGCPHYS GCPhysAddrDataBuffer = SGEntry.Simple32.u32DataBufferAddressLow;
1651
1652 if (SGEntry.Simple32.f64BitAddress)
1653 {
1654 GCPhysAddrDataBuffer |= ((uint64_t)SGEntry.Simple64.u32DataBufferAddressHigh) << 32;
1655 GCPhysSGEntryNext += sizeof(MptSGEntrySimple64);
1656 }
1657 else
1658 GCPhysSGEntryNext += sizeof(MptSGEntrySimple32);
1659
1660 if (fDoMapping)
1661 {
1662 pSGInfoCurr->fGuestMemory = false;
1663 pSGInfoCurr->fBufferContainsData = fBufferContainsData;
1664 pSGInfoCurr->cbBuf = cbDataToTransfer;
1665 pSGInfoCurr->pvBuf = pbBufferUnalignedSGInfoPos;
1666 pbBufferUnalignedSGInfoPos += cbDataToTransfer;
1667 pSGInfoCurr->u.GCPhysAddrBufferUnaligned = GCPhysAddrDataBuffer;
1668 if (fBufferContainsData)
1669 lsilogicCopyFromSGListIntoBuffer(pDevIns, pSGInfoCurr);
1670 pSGInfoCurr++;
1671 }
1672 else
1673 {
1674 cbUnalignedComplete += cbDataToTransfer;
1675 cSGInfo++;
1676 }
1677
1678 /* Check if we reached the end of the list. */
1679 if (SGEntry.Simple32.fEndOfList)
1680 {
1681 /* We finished. */
1682 fEndOfSegment = true;
1683 fEndOfList = true;
1684 }
1685 else if (SGEntry.Simple32.fLastElement)
1686 {
1687 fEndOfSegment = true;
1688 }
1689 } /* while (!fEndOfSegment) */
1690
1691 /* Get next chain element. */
1692 if (uChainOffsetNext)
1693 {
1694 MptSGEntryChain SGEntryChain;
1695
1696 PDMDevHlpPhysRead(pDevIns, GCPhysSegmentStart + uChainOffsetNext, &SGEntryChain, sizeof(MptSGEntryChain));
1697
1698 AssertMsg(SGEntryChain.u2ElementType == MPTSGENTRYTYPE_CHAIN, ("Invalid SG entry type\n"));
1699
1700 /* Set the next address now. */
1701 GCPhysSGEntryNext = SGEntryChain.u32SegmentAddressLow;
1702 if (SGEntryChain.f64BitAddress)
1703 GCPhysSGEntryNext |= ((uint64_t)SGEntryChain.u32SegmentAddressHigh) << 32;
1704
1705 GCPhysSegmentStart = GCPhysSGEntryNext;
1706 uChainOffsetNext = SGEntryChain.u8NextChainOffset * sizeof(uint32_t);
1707 }
1708
1709 } /* while (!fEndOfList) */
1710
1711 fDoMapping = true;
1712 if (fUnaligned)
1713 cbUnalignedComplete += cbUnaligned;
1714 }
1715
1716 uint32_t cSGEntries;
1717 PPDMDATASEG pSGEntryCurr = pTaskState->pSGListHead;
1718 pSGInfoCurr = pTaskState->paSGEntries;
1719
1720 /* Initialize first entry. */
1721 pSGEntryCurr->pvSeg = pSGInfoCurr->pvBuf;
1722 pSGEntryCurr->cbSeg = pSGInfoCurr->cbBuf;
1723 pSGInfoCurr++;
1724 cSGEntries = 1;
1725
1726 /* Construct the scatter gather list. */
1727 for (unsigned i = 0; i < (pTaskState->cSGInfoEntries-1); i++)
1728 {
1729 if (pSGEntryCurr->cbSeg % 512 != 0)
1730 {
1731 AssertMsg((uint8_t *)pSGEntryCurr->pvSeg + pSGEntryCurr->cbSeg == pSGInfoCurr->pvBuf,
1732 ("Buffer ist not sector aligned but the buffer addresses are not adjacent\n"));
1733
1734 pSGEntryCurr->cbSeg += pSGInfoCurr->cbBuf;
1735 }
1736 else
1737 {
1738 if (((uint8_t *)pSGEntryCurr->pvSeg + pSGEntryCurr->cbSeg) == pSGInfoCurr->pvBuf)
1739 {
1740 pSGEntryCurr->cbSeg += pSGInfoCurr->cbBuf;
1741 }
1742 else
1743 {
1744 pSGEntryCurr++;
1745 cSGEntries++;
1746 pSGEntryCurr->pvSeg = pSGInfoCurr->pvBuf;
1747 pSGEntryCurr->cbSeg = pSGInfoCurr->cbBuf;
1748 }
1749 }
1750
1751 pSGInfoCurr++;
1752 }
1753
1754 pTaskState->cSGListEntries = cSGEntries;
1755
1756 return rc;
1757}
1758
1759/*
1760 * Disabled because the sense buffer provided by the LsiLogic driver for Windows XP
1761 * crosses page boundaries.
1762 */
1763#if 0
1764/**
1765 * Free the sense buffer.
1766 *
1767 * @returns nothing.
1768 * @param pTaskState Pointer to the task state.
1769 */
1770static void lsilogicFreeGCSenseBuffer(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
1771{
1772 PVM pVM = PDMDevHlpGetVM(pLsiLogic->CTX_SUFF(pDevIns));
1773
1774 PGMPhysReleasePageMappingLock(pVM, &pTaskState->PageLockSense);
1775 pTaskState->pbSenseBuffer = NULL;
1776}
1777
1778/**
1779 * Map the sense buffer into R3.
1780 *
1781 * @returns VBox status code.
1782 * @param pTaskState Pointer to the task state.
1783 * @note Current assumption is that the sense buffer is not scattered and does not cross a page boundary.
1784 */
1785static int lsilogicMapGCSenseBufferIntoR3(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
1786{
1787 int rc = VINF_SUCCESS;
1788 PPDMDEVINS pDevIns = pLsiLogic->CTX_SUFF(pDevIns);
1789 RTGCPHYS GCPhysAddrSenseBuffer;
1790
1791 GCPhysAddrSenseBuffer = pTaskState->GuestRequest.SCSIIO.u32SenseBufferLowAddress;
1792 GCPhysAddrSenseBuffer |= ((uint64_t)pLsiLogic->u32SenseBufferHighAddr << 32);
1793
1794#ifdef RT_STRICT
1795 uint32_t cbSenseBuffer = pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength;
1796#endif
1797 RTGCPHYS GCPhysAddrSenseBufferBase = PAGE_ADDRESS(GCPhysAddrSenseBuffer);
1798
1799 AssertMsg(GCPhysAddrSenseBuffer >= GCPhysAddrSenseBufferBase,
1800 ("Impossible GCPhysAddrSenseBuffer < GCPhysAddrSenseBufferBase\n"));
1801
1802 /* Sanity checks for the assumption. */
1803 AssertMsg(((GCPhysAddrSenseBuffer + cbSenseBuffer) <= (GCPhysAddrSenseBufferBase + PAGE_SIZE)),
1804 ("Sense buffer crosses page boundary\n"));
1805
1806 rc = PDMDevHlpPhysGCPhys2CCPtr(pDevIns, GCPhysAddrSenseBufferBase, (void **)&pTaskState->pbSenseBuffer, &pTaskState->PageLockSense);
1807 AssertMsgRC(rc, ("Mapping sense buffer failed rc=%Rrc\n", rc));
1808
1809 /* Correct start address of the sense buffer. */
1810 pTaskState->pbSenseBuffer += (GCPhysAddrSenseBuffer - GCPhysAddrSenseBufferBase);
1811
1812 return rc;
1813}
1814#endif
1815
1816#ifdef DEBUG
1817static void lsilogicDumpSCSIIORequest(PMptSCSIIORequest pSCSIIORequest)
1818{
1819 Log(("%s: u8TargetID=%d\n", __FUNCTION__, pSCSIIORequest->u8TargetID));
1820 Log(("%s: u8Bus=%d\n", __FUNCTION__, pSCSIIORequest->u8Bus));
1821 Log(("%s: u8ChainOffset=%d\n", __FUNCTION__, pSCSIIORequest->u8ChainOffset));
1822 Log(("%s: u8Function=%d\n", __FUNCTION__, pSCSIIORequest->u8Function));
1823 Log(("%s: u8CDBLength=%d\n", __FUNCTION__, pSCSIIORequest->u8CDBLength));
1824 Log(("%s: u8SenseBufferLength=%d\n", __FUNCTION__, pSCSIIORequest->u8SenseBufferLength));
1825 Log(("%s: u8MessageFlags=%d\n", __FUNCTION__, pSCSIIORequest->u8MessageFlags));
1826 Log(("%s: u32MessageContext=%#x\n", __FUNCTION__, pSCSIIORequest->u32MessageContext));
1827 for (unsigned i = 0; i < RT_ELEMENTS(pSCSIIORequest->au8LUN); i++)
1828 Log(("%s: u8LUN[%d]=%d\n", __FUNCTION__, i, pSCSIIORequest->au8LUN[i]));
1829 Log(("%s: u32Control=%#x\n", __FUNCTION__, pSCSIIORequest->u32Control));
1830 for (unsigned i = 0; i < RT_ELEMENTS(pSCSIIORequest->au8CDB); i++)
1831 Log(("%s: u8CDB[%d]=%d\n", __FUNCTION__, i, pSCSIIORequest->au8CDB[i]));
1832 Log(("%s: u32DataLength=%#x\n", __FUNCTION__, pSCSIIORequest->u32DataLength));
1833 Log(("%s: u32SenseBufferLowAddress=%#x\n", __FUNCTION__, pSCSIIORequest->u32SenseBufferLowAddress));
1834}
1835#endif
1836
1837/**
1838 * Processes a SCSI I/O request by setting up the request
1839 * and sending it to the underlying SCSI driver.
1840 * Steps needed to complete request are done in the
1841 * callback called by the driver below upon completion of
1842 * the request.
1843 *
1844 * @returns VBox status code.
1845 * @param pLsiLogic Pointer to the device instance which sends the request.
1846 * @param pTaskState Pointer to the task state data.
1847 */
1848static int lsilogicProcessSCSIIORequest(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
1849{
1850 int rc = VINF_SUCCESS;
1851
1852#ifdef DEBUG
1853 lsilogicDumpSCSIIORequest(&pTaskState->GuestRequest.SCSIIO);
1854#endif
1855
1856 pTaskState->fBIOS = false;
1857
1858 uint32_t uChainOffset = pTaskState->GuestRequest.SCSIIO.u8ChainOffset;
1859
1860 if (uChainOffset)
1861 uChainOffset = uChainOffset * sizeof(uint32_t) - sizeof(MptSCSIIORequest);
1862
1863 /* Create Scatter gather list. */
1864 rc = lsilogicScatterGatherListCreate(pLsiLogic, pTaskState,
1865 pTaskState->GCPhysMessageFrameAddr + sizeof(MptSCSIIORequest),
1866 uChainOffset);
1867 AssertRC(rc);
1868
1869#if 0
1870 /* Map sense buffer. */
1871 rc = lsilogicMapGCSenseBufferIntoR3(pLsiLogic, pTaskState);
1872 AssertRC(rc);
1873#endif
1874
1875 if (RT_LIKELY( (pTaskState->GuestRequest.SCSIIO.u8TargetID < pLsiLogic->cDeviceStates)
1876 && (pTaskState->GuestRequest.SCSIIO.u8Bus == 0)))
1877 {
1878 PLSILOGICDEVICE pTargetDevice;
1879 pTargetDevice = &pLsiLogic->paDeviceStates[pTaskState->GuestRequest.SCSIIO.u8TargetID];
1880
1881 if (pTargetDevice->pDrvBase)
1882 {
1883 /* Setup the SCSI request. */
1884 pTaskState->pTargetDevice = pTargetDevice;
1885 pTaskState->PDMScsiRequest.uLogicalUnit = pTaskState->GuestRequest.SCSIIO.au8LUN[1];
1886
1887 uint8_t uDataDirection = MPT_SCSIIO_REQUEST_CONTROL_TXDIR_GET(pTaskState->GuestRequest.SCSIIO.u32Control);
1888
1889 if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_NONE)
1890 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_NONE;
1891 else if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_WRITE)
1892 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_TO_DEVICE;
1893 else if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_READ)
1894 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_FROM_DEVICE;
1895
1896 pTaskState->PDMScsiRequest.cbCDB = pTaskState->GuestRequest.SCSIIO.u8CDBLength;
1897 pTaskState->PDMScsiRequest.pbCDB = pTaskState->GuestRequest.SCSIIO.au8CDB;
1898 pTaskState->PDMScsiRequest.cbScatterGather = pTaskState->GuestRequest.SCSIIO.u32DataLength;
1899 pTaskState->PDMScsiRequest.cScatterGatherEntries = pTaskState->cSGListEntries;
1900 pTaskState->PDMScsiRequest.paScatterGatherHead = pTaskState->pSGListHead;
1901 pTaskState->PDMScsiRequest.cbSenseBuffer = sizeof(pTaskState->abSenseBuffer);
1902 memset(pTaskState->abSenseBuffer, 0, pTaskState->PDMScsiRequest.cbSenseBuffer);
1903 pTaskState->PDMScsiRequest.pbSenseBuffer = pTaskState->abSenseBuffer;
1904 pTaskState->PDMScsiRequest.pvUser = pTaskState;
1905
1906 ASMAtomicIncU32(&pTargetDevice->cOutstandingRequests);
1907 rc = pTargetDevice->pDrvSCSIConnector->pfnSCSIRequestSend(pTargetDevice->pDrvSCSIConnector, &pTaskState->PDMScsiRequest);
1908 AssertMsgRC(rc, ("Sending request to SCSI layer failed rc=%Rrc\n", rc));
1909 return VINF_SUCCESS;
1910 }
1911 else
1912 {
1913 /* Device is not present report SCSI selection timeout. */
1914 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_DEVICE_NOT_THERE;
1915 }
1916 }
1917 else
1918 {
1919 /* Report out of bounds target ID or bus. */
1920 if (pTaskState->GuestRequest.SCSIIO.u8Bus != 0)
1921 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_BUS;
1922 else
1923 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_TARGETID;
1924 }
1925
1926 /* The rest is equal to both errors. */
1927 pTaskState->IOCReply.SCSIIOError.u8TargetID = pTaskState->GuestRequest.SCSIIO.u8TargetID;
1928 pTaskState->IOCReply.SCSIIOError.u8Bus = pTaskState->GuestRequest.SCSIIO.u8Bus;
1929 pTaskState->IOCReply.SCSIIOError.u8MessageLength = sizeof(MptSCSIIOErrorReply) / 4;
1930 pTaskState->IOCReply.SCSIIOError.u8Function = pTaskState->GuestRequest.SCSIIO.u8Function;
1931 pTaskState->IOCReply.SCSIIOError.u8CDBLength = pTaskState->GuestRequest.SCSIIO.u8CDBLength;
1932 pTaskState->IOCReply.SCSIIOError.u8SenseBufferLength = pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength;
1933 pTaskState->IOCReply.SCSIIOError.u32MessageContext = pTaskState->GuestRequest.SCSIIO.u32MessageContext;
1934 pTaskState->IOCReply.SCSIIOError.u8SCSIStatus = SCSI_STATUS_OK;
1935 pTaskState->IOCReply.SCSIIOError.u8SCSIState = MPT_SCSI_IO_ERROR_SCSI_STATE_TERMINATED;
1936 pTaskState->IOCReply.SCSIIOError.u32IOCLogInfo = 0;
1937 pTaskState->IOCReply.SCSIIOError.u32TransferCount = 0;
1938 pTaskState->IOCReply.SCSIIOError.u32SenseCount = 0;
1939 pTaskState->IOCReply.SCSIIOError.u32ResponseInfo = 0;
1940
1941 lsilogicFinishAddressReply(pLsiLogic, &pTaskState->IOCReply, false);
1942 RTMemCacheFree(pLsiLogic->hTaskCache, pTaskState);
1943
1944 return rc;
1945}
1946
1947/**
1948 * Called upon completion of the request from the SCSI driver below.
1949 * This function frees all allocated ressources and notifies the guest
1950 * that the process finished by asserting an interrupt.
1951 *
1952 * @returns VBox status code.
1953 * @param pInterface Pointer to the interface the called funtion belongs to.
1954 * @param pSCSIRequest Pointer to the SCSI request which finished.
1955 */
1956static DECLCALLBACK(int) lsilogicDeviceSCSIRequestCompleted(PPDMISCSIPORT pInterface, PPDMSCSIREQUEST pSCSIRequest, int rcCompletion)
1957{
1958 PLSILOGICTASKSTATE pTaskState = (PLSILOGICTASKSTATE)pSCSIRequest->pvUser;
1959 PLSILOGICDEVICE pLsiLogicDevice = pTaskState->pTargetDevice;
1960 PLSILOGICSCSI pLsiLogic = pLsiLogicDevice->CTX_SUFF(pLsiLogic);
1961
1962 ASMAtomicDecU32(&pLsiLogicDevice->cOutstandingRequests);
1963
1964 if (RT_UNLIKELY(pTaskState->fBIOS))
1965 {
1966 int rc = vboxscsiRequestFinished(&pLsiLogic->VBoxSCSI, pSCSIRequest);
1967 AssertMsgRC(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc));
1968 }
1969 else
1970 {
1971#if 0
1972 lsilogicFreeGCSenseBuffer(pLsiLogic, pTaskState);
1973#else
1974 RTGCPHYS GCPhysAddrSenseBuffer;
1975
1976 GCPhysAddrSenseBuffer = pTaskState->GuestRequest.SCSIIO.u32SenseBufferLowAddress;
1977 GCPhysAddrSenseBuffer |= ((uint64_t)pLsiLogic->u32SenseBufferHighAddr << 32);
1978
1979 /* Copy the sense buffer over. */
1980 PDMDevHlpPhysWrite(pLsiLogic->CTX_SUFF(pDevIns), GCPhysAddrSenseBuffer, pTaskState->abSenseBuffer,
1981 RT_UNLIKELY(pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength < pTaskState->PDMScsiRequest.cbSenseBuffer)
1982 ? pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength
1983 : pTaskState->PDMScsiRequest.cbSenseBuffer);
1984#endif
1985 lsilogicScatterGatherListDestroy(pLsiLogic, pTaskState);
1986
1987
1988 if (RT_LIKELY(rcCompletion == SCSI_STATUS_OK))
1989 lsilogicFinishContextReply(pLsiLogic, pTaskState->GuestRequest.SCSIIO.u32MessageContext);
1990 else
1991 {
1992 /* The SCSI target encountered an error during processing post a reply. */
1993 memset(&pTaskState->IOCReply, 0, sizeof(MptReplyUnion));
1994 pTaskState->IOCReply.SCSIIOError.u8TargetID = pTaskState->GuestRequest.SCSIIO.u8TargetID;
1995 pTaskState->IOCReply.SCSIIOError.u8Bus = pTaskState->GuestRequest.SCSIIO.u8Bus;
1996 pTaskState->IOCReply.SCSIIOError.u8MessageLength = 8;
1997 pTaskState->IOCReply.SCSIIOError.u8Function = pTaskState->GuestRequest.SCSIIO.u8Function;
1998 pTaskState->IOCReply.SCSIIOError.u8CDBLength = pTaskState->GuestRequest.SCSIIO.u8CDBLength;
1999 pTaskState->IOCReply.SCSIIOError.u8SenseBufferLength = pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength;
2000 pTaskState->IOCReply.SCSIIOError.u8MessageFlags = pTaskState->GuestRequest.SCSIIO.u8MessageFlags;
2001 pTaskState->IOCReply.SCSIIOError.u32MessageContext = pTaskState->GuestRequest.SCSIIO.u32MessageContext;
2002 pTaskState->IOCReply.SCSIIOError.u8SCSIStatus = rcCompletion;
2003 pTaskState->IOCReply.SCSIIOError.u8SCSIState = MPT_SCSI_IO_ERROR_SCSI_STATE_AUTOSENSE_VALID;
2004 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = 0;
2005 pTaskState->IOCReply.SCSIIOError.u32IOCLogInfo = 0;
2006 pTaskState->IOCReply.SCSIIOError.u32TransferCount = 0;
2007 pTaskState->IOCReply.SCSIIOError.u32SenseCount = sizeof(pTaskState->abSenseBuffer);
2008 pTaskState->IOCReply.SCSIIOError.u32ResponseInfo = 0;
2009
2010 lsilogicFinishAddressReply(pLsiLogic, &pTaskState->IOCReply, true);
2011 }
2012 }
2013
2014 RTMemCacheFree(pLsiLogic->hTaskCache, pTaskState);
2015
2016 return VINF_SUCCESS;
2017}
2018
2019/**
2020 * Return the configuration page header and data
2021 * which matches the given page type and number.
2022 *
2023 * @returns VINF_SUCCESS if successful
2024 * VERR_NOT_FOUND if the requested page could be found.
2025 * @param u8PageNumber Number of the page to get.
2026 * @param ppPageHeader Where to store the pointer to the page header.
2027 * @param ppbPageData Where to store the pointer to the page data.
2028 */
2029static int lsilogicConfigurationIOUnitPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2030 PMptConfigurationPagesSupported pPages,
2031 uint8_t u8PageNumber,
2032 PMptConfigurationPageHeader *ppPageHeader,
2033 uint8_t **ppbPageData, size_t *pcbPage)
2034{
2035 int rc = VINF_SUCCESS;
2036
2037 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
2038
2039 switch(u8PageNumber)
2040 {
2041 case 0:
2042 *ppPageHeader = &pPages->IOUnitPage0.u.fields.Header;
2043 *ppbPageData = pPages->IOUnitPage0.u.abPageData;
2044 *pcbPage = sizeof(pPages->IOUnitPage0);
2045 break;
2046 case 1:
2047 *ppPageHeader = &pPages->IOUnitPage1.u.fields.Header;
2048 *ppbPageData = pPages->IOUnitPage1.u.abPageData;
2049 *pcbPage = sizeof(pPages->IOUnitPage1);
2050 break;
2051 case 2:
2052 *ppPageHeader = &pPages->IOUnitPage2.u.fields.Header;
2053 *ppbPageData = pPages->IOUnitPage2.u.abPageData;
2054 *pcbPage = sizeof(pPages->IOUnitPage2);
2055 break;
2056 case 3:
2057 *ppPageHeader = &pPages->IOUnitPage3.u.fields.Header;
2058 *ppbPageData = pPages->IOUnitPage3.u.abPageData;
2059 *pcbPage = sizeof(pPages->IOUnitPage3);
2060 break;
2061 case 4:
2062 *ppPageHeader = &pPages->IOUnitPage4.u.fields.Header;
2063 *ppbPageData = pPages->IOUnitPage4.u.abPageData;
2064 *pcbPage = sizeof(pPages->IOUnitPage4);
2065 break;
2066 default:
2067 rc = VERR_NOT_FOUND;
2068 }
2069
2070 return rc;
2071}
2072
2073/**
2074 * Return the configuration page header and data
2075 * which matches the given page type and number.
2076 *
2077 * @returns VINF_SUCCESS if successful
2078 * VERR_NOT_FOUND if the requested page could be found.
2079 * @param u8PageNumber Number of the page to get.
2080 * @param ppPageHeader Where to store the pointer to the page header.
2081 * @param ppbPageData Where to store the pointer to the page data.
2082 */
2083static int lsilogicConfigurationIOCPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2084 PMptConfigurationPagesSupported pPages,
2085 uint8_t u8PageNumber,
2086 PMptConfigurationPageHeader *ppPageHeader,
2087 uint8_t **ppbPageData, size_t *pcbPage)
2088{
2089 int rc = VINF_SUCCESS;
2090
2091 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
2092
2093 switch(u8PageNumber)
2094 {
2095 case 0:
2096 *ppPageHeader = &pPages->IOCPage0.u.fields.Header;
2097 *ppbPageData = pPages->IOCPage0.u.abPageData;
2098 *pcbPage = sizeof(pPages->IOCPage0);
2099 break;
2100 case 1:
2101 *ppPageHeader = &pPages->IOCPage1.u.fields.Header;
2102 *ppbPageData = pPages->IOCPage1.u.abPageData;
2103 *pcbPage = sizeof(pPages->IOCPage1);
2104 break;
2105 case 2:
2106 *ppPageHeader = &pPages->IOCPage2.u.fields.Header;
2107 *ppbPageData = pPages->IOCPage2.u.abPageData;
2108 *pcbPage = sizeof(pPages->IOCPage2);
2109 break;
2110 case 3:
2111 *ppPageHeader = &pPages->IOCPage3.u.fields.Header;
2112 *ppbPageData = pPages->IOCPage3.u.abPageData;
2113 *pcbPage = sizeof(pPages->IOCPage3);
2114 break;
2115 case 4:
2116 *ppPageHeader = &pPages->IOCPage4.u.fields.Header;
2117 *ppbPageData = pPages->IOCPage4.u.abPageData;
2118 *pcbPage = sizeof(pPages->IOCPage4);
2119 break;
2120 case 6:
2121 *ppPageHeader = &pPages->IOCPage6.u.fields.Header;
2122 *ppbPageData = pPages->IOCPage6.u.abPageData;
2123 *pcbPage = sizeof(pPages->IOCPage6);
2124 break;
2125 default:
2126 rc = VERR_NOT_FOUND;
2127 }
2128
2129 return rc;
2130}
2131
2132/**
2133 * Return the configuration page header and data
2134 * which matches the given page type and number.
2135 *
2136 * @returns VINF_SUCCESS if successful
2137 * VERR_NOT_FOUND if the requested page could be found.
2138 * @param u8PageNumber Number of the page to get.
2139 * @param ppPageHeader Where to store the pointer to the page header.
2140 * @param ppbPageData Where to store the pointer to the page data.
2141 */
2142static int lsilogicConfigurationManufacturingPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2143 PMptConfigurationPagesSupported pPages,
2144 uint8_t u8PageNumber,
2145 PMptConfigurationPageHeader *ppPageHeader,
2146 uint8_t **ppbPageData, size_t *pcbPage)
2147{
2148 int rc = VINF_SUCCESS;
2149
2150 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
2151
2152 switch(u8PageNumber)
2153 {
2154 case 0:
2155 *ppPageHeader = &pPages->ManufacturingPage0.u.fields.Header;
2156 *ppbPageData = pPages->ManufacturingPage0.u.abPageData;
2157 *pcbPage = sizeof(pPages->ManufacturingPage0);
2158 break;
2159 case 1:
2160 *ppPageHeader = &pPages->ManufacturingPage1.u.fields.Header;
2161 *ppbPageData = pPages->ManufacturingPage1.u.abPageData;
2162 *pcbPage = sizeof(pPages->ManufacturingPage1);
2163 break;
2164 case 2:
2165 *ppPageHeader = &pPages->ManufacturingPage2.u.fields.Header;
2166 *ppbPageData = pPages->ManufacturingPage2.u.abPageData;
2167 *pcbPage = sizeof(pPages->ManufacturingPage2);
2168 break;
2169 case 3:
2170 *ppPageHeader = &pPages->ManufacturingPage3.u.fields.Header;
2171 *ppbPageData = pPages->ManufacturingPage3.u.abPageData;
2172 *pcbPage = sizeof(pPages->ManufacturingPage3);
2173 break;
2174 case 4:
2175 *ppPageHeader = &pPages->ManufacturingPage4.u.fields.Header;
2176 *ppbPageData = pPages->ManufacturingPage4.u.abPageData;
2177 *pcbPage = sizeof(pPages->ManufacturingPage4);
2178 break;
2179 case 5:
2180 *ppPageHeader = &pPages->ManufacturingPage5.u.fields.Header;
2181 *ppbPageData = pPages->ManufacturingPage5.u.abPageData;
2182 *pcbPage = sizeof(pPages->ManufacturingPage5);
2183 break;
2184 case 6:
2185 *ppPageHeader = &pPages->ManufacturingPage6.u.fields.Header;
2186 *ppbPageData = pPages->ManufacturingPage6.u.abPageData;
2187 *pcbPage = sizeof(pPages->ManufacturingPage6);
2188 break;
2189 case 7:
2190 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
2191 {
2192 *ppPageHeader = &pPages->u.SasPages.pManufacturingPage7->u.fields.Header;
2193 *ppbPageData = pPages->u.SasPages.pManufacturingPage7->u.abPageData;
2194 *pcbPage = pPages->u.SasPages.cbManufacturingPage7;
2195 }
2196 else
2197 rc = VERR_NOT_FOUND;
2198 break;
2199 case 8:
2200 *ppPageHeader = &pPages->ManufacturingPage8.u.fields.Header;
2201 *ppbPageData = pPages->ManufacturingPage8.u.abPageData;
2202 *pcbPage = sizeof(pPages->ManufacturingPage8);
2203 break;
2204 case 9:
2205 *ppPageHeader = &pPages->ManufacturingPage9.u.fields.Header;
2206 *ppbPageData = pPages->ManufacturingPage9.u.abPageData;
2207 *pcbPage = sizeof(pPages->ManufacturingPage9);
2208 break;
2209 case 10:
2210 *ppPageHeader = &pPages->ManufacturingPage10.u.fields.Header;
2211 *ppbPageData = pPages->ManufacturingPage10.u.abPageData;
2212 *pcbPage = sizeof(pPages->ManufacturingPage10);
2213 break;
2214 default:
2215 rc = VERR_NOT_FOUND;
2216 }
2217
2218 return rc;
2219}
2220
2221/**
2222 * Return the configuration page header and data
2223 * which matches the given page type and number.
2224 *
2225 * @returns VINF_SUCCESS if successful
2226 * VERR_NOT_FOUND if the requested page could be found.
2227 * @param u8PageNumber Number of the page to get.
2228 * @param ppPageHeader Where to store the pointer to the page header.
2229 * @param ppbPageData Where to store the pointer to the page data.
2230 */
2231static int lsilogicConfigurationBiosPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2232 PMptConfigurationPagesSupported pPages,
2233 uint8_t u8PageNumber,
2234 PMptConfigurationPageHeader *ppPageHeader,
2235 uint8_t **ppbPageData, size_t *pcbPage)
2236{
2237 int rc = VINF_SUCCESS;
2238
2239 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
2240
2241 switch(u8PageNumber)
2242 {
2243 case 1:
2244 *ppPageHeader = &pPages->BIOSPage1.u.fields.Header;
2245 *ppbPageData = pPages->BIOSPage1.u.abPageData;
2246 *pcbPage = sizeof(pPages->BIOSPage1);
2247 break;
2248 case 2:
2249 *ppPageHeader = &pPages->BIOSPage2.u.fields.Header;
2250 *ppbPageData = pPages->BIOSPage2.u.abPageData;
2251 *pcbPage = sizeof(pPages->BIOSPage2);
2252 break;
2253 case 4:
2254 *ppPageHeader = &pPages->BIOSPage4.u.fields.Header;
2255 *ppbPageData = pPages->BIOSPage4.u.abPageData;
2256 *pcbPage = sizeof(pPages->BIOSPage4);
2257 break;
2258 default:
2259 rc = VERR_NOT_FOUND;
2260 }
2261
2262 return rc;
2263}
2264
2265/**
2266 * Return the configuration page header and data
2267 * which matches the given page type and number.
2268 *
2269 * @returns VINF_SUCCESS if successful
2270 * VERR_NOT_FOUND if the requested page could be found.
2271 * @param u8PageNumber Number of the page to get.
2272 * @param ppPageHeader Where to store the pointer to the page header.
2273 * @param ppbPageData Where to store the pointer to the page data.
2274 */
2275static int lsilogicConfigurationSCSISPIPortPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2276 PMptConfigurationPagesSupported pPages,
2277 uint8_t u8Port,
2278 uint8_t u8PageNumber,
2279 PMptConfigurationPageHeader *ppPageHeader,
2280 uint8_t **ppbPageData, size_t *pcbPage)
2281{
2282 int rc = VINF_SUCCESS;
2283
2284 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
2285
2286 if (u8Port >= RT_ELEMENTS(pPages->u.SpiPages.aPortPages))
2287 return VERR_NOT_FOUND;
2288
2289 switch(u8PageNumber)
2290 {
2291 case 0:
2292 *ppPageHeader = &pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage0.u.fields.Header;
2293 *ppbPageData = pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage0.u.abPageData;
2294 *pcbPage = sizeof(pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage0);
2295 break;
2296 case 1:
2297 *ppPageHeader = &pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage1.u.fields.Header;
2298 *ppbPageData = pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage1.u.abPageData;
2299 *pcbPage = sizeof(pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage1);
2300 break;
2301 case 2:
2302 *ppPageHeader = &pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage2.u.fields.Header;
2303 *ppbPageData = pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage2.u.abPageData;
2304 *pcbPage = sizeof(pPages->u.SpiPages.aPortPages[u8Port].SCSISPIPortPage2);
2305 break;
2306 default:
2307 rc = VERR_NOT_FOUND;
2308 }
2309
2310 return rc;
2311}
2312
2313/**
2314 * Return the configuration page header and data
2315 * which matches the given page type and number.
2316 *
2317 * @returns VINF_SUCCESS if successful
2318 * VERR_NOT_FOUND if the requested page could be found.
2319 * @param u8PageNumber Number of the page to get.
2320 * @param ppPageHeader Where to store the pointer to the page header.
2321 * @param ppbPageData Where to store the pointer to the page data.
2322 */
2323static int lsilogicConfigurationSCSISPIDevicePageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2324 PMptConfigurationPagesSupported pPages,
2325 uint8_t u8Bus,
2326 uint8_t u8TargetID, uint8_t u8PageNumber,
2327 PMptConfigurationPageHeader *ppPageHeader,
2328 uint8_t **ppbPageData, size_t *pcbPage)
2329{
2330 int rc = VINF_SUCCESS;
2331
2332 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
2333
2334 if (u8Bus >= RT_ELEMENTS(pPages->u.SpiPages.aBuses))
2335 return VERR_NOT_FOUND;
2336
2337 if (u8TargetID >= RT_ELEMENTS(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages))
2338 return VERR_NOT_FOUND;
2339
2340 switch(u8PageNumber)
2341 {
2342 case 0:
2343 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0.u.fields.Header;
2344 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0.u.abPageData;
2345 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0);
2346 break;
2347 case 1:
2348 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1.u.fields.Header;
2349 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1.u.abPageData;
2350 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1);
2351 break;
2352 case 2:
2353 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2.u.fields.Header;
2354 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2.u.abPageData;
2355 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2);
2356 break;
2357 case 3:
2358 *ppPageHeader = &pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3.u.fields.Header;
2359 *ppbPageData = pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3.u.abPageData;
2360 *pcbPage = sizeof(pPages->u.SpiPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3);
2361 break;
2362 default:
2363 rc = VERR_NOT_FOUND;
2364 }
2365
2366 return rc;
2367}
2368
2369static int lsilogicConfigurationSASIOUnitPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2370 PMptConfigurationPagesSupported pPages,
2371 uint8_t u8PageNumber,
2372 PMptExtendedConfigurationPageHeader *ppPageHeader,
2373 uint8_t **ppbPageData, size_t *pcbPage)
2374{
2375 int rc = VINF_SUCCESS;
2376
2377 switch(u8PageNumber)
2378 {
2379 case 0:
2380 *ppPageHeader = &pPages->u.SasPages.pSASIOUnitPage0->u.fields.ExtHeader;
2381 *ppbPageData = pPages->u.SasPages.pSASIOUnitPage0->u.abPageData;
2382 *pcbPage = pPages->u.SasPages.cbSASIOUnitPage0;
2383 break;
2384 case 1:
2385 *ppPageHeader = &pPages->u.SasPages.pSASIOUnitPage1->u.fields.ExtHeader;
2386 *ppbPageData = pPages->u.SasPages.pSASIOUnitPage1->u.abPageData;
2387 *pcbPage = pPages->u.SasPages.cbSASIOUnitPage1;
2388 break;
2389 case 2:
2390 *ppPageHeader = &pPages->u.SasPages.SASIOUnitPage2.u.fields.ExtHeader;
2391 *ppbPageData = pPages->u.SasPages.SASIOUnitPage2.u.abPageData;
2392 *pcbPage = sizeof(pPages->u.SasPages.SASIOUnitPage2);
2393 break;
2394 case 3:
2395 *ppPageHeader = &pPages->u.SasPages.SASIOUnitPage3.u.fields.ExtHeader;
2396 *ppbPageData = pPages->u.SasPages.SASIOUnitPage3.u.abPageData;
2397 *pcbPage = sizeof(pPages->u.SasPages.SASIOUnitPage3);
2398 break;
2399 default:
2400 rc = VERR_NOT_FOUND;
2401 }
2402
2403 return rc;
2404}
2405
2406static int lsilogicConfigurationSASPHYPageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2407 PMptConfigurationPagesSupported pPages,
2408 uint8_t u8PageNumber,
2409 MptConfigurationPageAddress PageAddress,
2410 PMptExtendedConfigurationPageHeader *ppPageHeader,
2411 uint8_t **ppbPageData, size_t *pcbPage)
2412{
2413 int rc = VINF_SUCCESS;
2414 uint8_t uAddressForm = MPT_CONFIGURATION_PAGE_ADDRESS_GET_SAS_FORM(PageAddress);
2415 PMptConfigurationPagesSas pPagesSas = &pPages->u.SasPages;
2416 PMptPHY pPHYPages = NULL;
2417
2418 Log(("Address form %d\n", uAddressForm));
2419
2420 if (uAddressForm == 0) /* PHY number */
2421 {
2422 uint8_t u8PhyNumber = PageAddress.SASPHY.Form0.u8PhyNumber;
2423
2424 Log(("PHY number %d\n", u8PhyNumber));
2425
2426 if (u8PhyNumber >= pPagesSas->cPHYs)
2427 return VERR_NOT_FOUND;
2428
2429 pPHYPages = &pPagesSas->paPHYs[u8PhyNumber];
2430 }
2431 else if (uAddressForm == 1) /* Index form */
2432 {
2433 uint16_t u16Index = PageAddress.SASPHY.Form1.u16Index;
2434
2435 Log(("PHY index %d\n", u16Index));
2436
2437 if (u16Index >= pPagesSas->cPHYs)
2438 return VERR_NOT_FOUND;
2439
2440 pPHYPages = &pPagesSas->paPHYs[u16Index];
2441 }
2442 else
2443 rc = VERR_NOT_FOUND; /* Correct? */
2444
2445 if (pPHYPages)
2446 {
2447 switch(u8PageNumber)
2448 {
2449 case 0:
2450 *ppPageHeader = &pPHYPages->SASPHYPage0.u.fields.ExtHeader;
2451 *ppbPageData = pPHYPages->SASPHYPage0.u.abPageData;
2452 *pcbPage = sizeof(pPHYPages->SASPHYPage0);
2453 break;
2454 case 1:
2455 *ppPageHeader = &pPHYPages->SASPHYPage1.u.fields.ExtHeader;
2456 *ppbPageData = pPHYPages->SASPHYPage1.u.abPageData;
2457 *pcbPage = sizeof(pPHYPages->SASPHYPage1);
2458 break;
2459 default:
2460 rc = VERR_NOT_FOUND;
2461 }
2462 }
2463 else
2464 rc = VERR_NOT_FOUND;
2465
2466 return rc;
2467}
2468
2469static int lsilogicConfigurationSASDevicePageGetFromNumber(PLSILOGICSCSI pLsiLogic,
2470 PMptConfigurationPagesSupported pPages,
2471 uint8_t u8PageNumber,
2472 MptConfigurationPageAddress PageAddress,
2473 PMptExtendedConfigurationPageHeader *ppPageHeader,
2474 uint8_t **ppbPageData, size_t *pcbPage)
2475{
2476 int rc = VINF_SUCCESS;
2477 uint8_t uAddressForm = MPT_CONFIGURATION_PAGE_ADDRESS_GET_SAS_FORM(PageAddress);
2478 PMptConfigurationPagesSas pPagesSas = &pPages->u.SasPages;
2479 PMptSASDevice pSASDevice = NULL;
2480
2481 Log(("Address form %d\n", uAddressForm));
2482
2483 if (uAddressForm == 0)
2484 {
2485 uint16_t u16Handle = PageAddress.SASDevice.Form0And2.u16Handle;
2486
2487 Log(("Get next handle %#x\n", u16Handle));
2488
2489 pSASDevice = pPagesSas->pSASDeviceHead;
2490
2491 /* Get the first device? */
2492 if (u16Handle != 0xffff)
2493 {
2494 /* No, search for the right one. */
2495
2496 while ( pSASDevice
2497 && pSASDevice->SASDevicePage0.u.fields.u16DevHandle != u16Handle)
2498 pSASDevice = pSASDevice->pNext;
2499
2500 if (pSASDevice)
2501 pSASDevice = pSASDevice->pNext;
2502 }
2503 }
2504 else if (uAddressForm == 1)
2505 {
2506 uint8_t u8TargetID = PageAddress.SASDevice.Form1.u8TargetID;
2507 uint8_t u8Bus = PageAddress.SASDevice.Form1.u8Bus;
2508
2509 Log(("u8TargetID=%d u8Bus=%d\n", u8TargetID, u8Bus));
2510
2511 pSASDevice = pPagesSas->pSASDeviceHead;
2512
2513 while ( pSASDevice
2514 && ( pSASDevice->SASDevicePage0.u.fields.u8TargetID != u8TargetID
2515 || pSASDevice->SASDevicePage0.u.fields.u8Bus != u8Bus))
2516 pSASDevice = pSASDevice->pNext;
2517 }
2518 else if (uAddressForm == 2)
2519 {
2520 uint16_t u16Handle = PageAddress.SASDevice.Form0And2.u16Handle;
2521
2522 Log(("Handle %#x\n", u16Handle));
2523
2524 pSASDevice = pPagesSas->pSASDeviceHead;
2525
2526 while ( pSASDevice
2527 && pSASDevice->SASDevicePage0.u.fields.u16DevHandle != u16Handle)
2528 pSASDevice = pSASDevice->pNext;
2529 }
2530
2531 if (pSASDevice)
2532 {
2533 switch(u8PageNumber)
2534 {
2535 case 0:
2536 *ppPageHeader = &pSASDevice->SASDevicePage0.u.fields.ExtHeader;
2537 *ppbPageData = pSASDevice->SASDevicePage0.u.abPageData;
2538 *pcbPage = sizeof(pSASDevice->SASDevicePage0);
2539 break;
2540 case 1:
2541 *ppPageHeader = &pSASDevice->SASDevicePage1.u.fields.ExtHeader;
2542 *ppbPageData = pSASDevice->SASDevicePage1.u.abPageData;
2543 *pcbPage = sizeof(pSASDevice->SASDevicePage1);
2544 break;
2545 case 2:
2546 *ppPageHeader = &pSASDevice->SASDevicePage2.u.fields.ExtHeader;
2547 *ppbPageData = pSASDevice->SASDevicePage2.u.abPageData;
2548 *pcbPage = sizeof(pSASDevice->SASDevicePage2);
2549 break;
2550 default:
2551 rc = VERR_NOT_FOUND;
2552 }
2553 }
2554 else
2555 rc = VERR_NOT_FOUND;
2556
2557 return rc;
2558}
2559
2560/**
2561 * Returns the extended configuration page header and data.
2562 * @returns VINF_SUCCESS if successful
2563 * VERR_NOT_FOUND if the requested page could be found.
2564 * @param pLsiLogic The LsiLogic controller instance.
2565 * @param pConfigurationReq The configuration request.
2566 * @param u8PageNumber Number of the page to get.
2567 * @param ppPageHeader Where to store the pointer to the page header.
2568 * @param ppbPageData Where to store the pointer to the page data.
2569 */
2570static int lsilogicConfigurationPageGetExtended(PLSILOGICSCSI pLsiLogic, PMptConfigurationRequest pConfigurationReq,
2571 PMptExtendedConfigurationPageHeader *ppPageHeader,
2572 uint8_t **ppbPageData, size_t *pcbPage)
2573{
2574 int rc = VINF_SUCCESS;
2575
2576 Log(("Extended page requested:\n"));
2577 Log(("u8ExtPageType=%#x\n", pConfigurationReq->u8ExtPageType));
2578 Log(("u8ExtPageLength=%d\n", pConfigurationReq->u16ExtPageLength));
2579
2580 switch (pConfigurationReq->u8ExtPageType)
2581 {
2582 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT:
2583 {
2584 rc = lsilogicConfigurationSASIOUnitPageGetFromNumber(pLsiLogic,
2585 pLsiLogic->pConfigurationPages,
2586 pConfigurationReq->u8PageNumber,
2587 ppPageHeader, ppbPageData, pcbPage);
2588 break;
2589 }
2590 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASPHYS:
2591 {
2592 rc = lsilogicConfigurationSASPHYPageGetFromNumber(pLsiLogic,
2593 pLsiLogic->pConfigurationPages,
2594 pConfigurationReq->u8PageNumber,
2595 pConfigurationReq->PageAddress,
2596 ppPageHeader, ppbPageData, pcbPage);
2597 break;
2598 }
2599 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE:
2600 {
2601 rc = lsilogicConfigurationSASDevicePageGetFromNumber(pLsiLogic,
2602 pLsiLogic->pConfigurationPages,
2603 pConfigurationReq->u8PageNumber,
2604 pConfigurationReq->PageAddress,
2605 ppPageHeader, ppbPageData, pcbPage);
2606 break;
2607 }
2608 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASEXPANDER: /* No expanders supported */
2609 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_ENCLOSURE: /* No enclosures supported */
2610 default:
2611 rc = VERR_NOT_FOUND;
2612 }
2613
2614 return rc;
2615}
2616
2617/**
2618 * Processes a Configuration request.
2619 *
2620 * @returns VBox status code.
2621 * @param pLsiLogic Pointer to the device instance which sends the request.
2622 * @param pConfigurationReq Pointer to the request structure.
2623 * @param pReply Pointer to the reply message frame
2624 */
2625static int lsilogicProcessConfigurationRequest(PLSILOGICSCSI pLsiLogic, PMptConfigurationRequest pConfigurationReq,
2626 PMptConfigurationReply pReply)
2627{
2628 int rc = VINF_SUCCESS;
2629 uint8_t *pbPageData = NULL;
2630 PMptConfigurationPageHeader pPageHeader = NULL;
2631 PMptExtendedConfigurationPageHeader pExtPageHeader = NULL;
2632 uint8_t u8PageType;
2633 uint8_t u8PageAttribute;
2634 size_t cbPage = 0;
2635
2636 LogFlowFunc(("pLsiLogic=%#p\n", pLsiLogic));
2637
2638 u8PageType = MPT_CONFIGURATION_PAGE_TYPE_GET(pConfigurationReq->u8PageType);
2639 u8PageAttribute = MPT_CONFIGURATION_PAGE_ATTRIBUTE_GET(pConfigurationReq->u8PageType);
2640
2641 Log(("GuestRequest:\n"));
2642 Log(("u8Action=%#x\n", pConfigurationReq->u8Action));
2643 Log(("u8PageType=%#x\n", u8PageType));
2644 Log(("u8PageNumber=%d\n", pConfigurationReq->u8PageNumber));
2645 Log(("u8PageLength=%d\n", pConfigurationReq->u8PageLength));
2646 Log(("u8PageVersion=%d\n", pConfigurationReq->u8PageVersion));
2647
2648 /* Copy common bits from the request into the reply. */
2649 pReply->u8MessageLength = 6; /* 6 32bit D-Words. */
2650 pReply->u8Action = pConfigurationReq->u8Action;
2651 pReply->u8Function = pConfigurationReq->u8Function;
2652 pReply->u32MessageContext = pConfigurationReq->u32MessageContext;
2653
2654 switch (u8PageType)
2655 {
2656 case MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT:
2657 {
2658 /* Get the page data. */
2659 rc = lsilogicConfigurationIOUnitPageGetFromNumber(pLsiLogic,
2660 pLsiLogic->pConfigurationPages,
2661 pConfigurationReq->u8PageNumber,
2662 &pPageHeader, &pbPageData, &cbPage);
2663 break;
2664 }
2665 case MPT_CONFIGURATION_PAGE_TYPE_IOC:
2666 {
2667 /* Get the page data. */
2668 rc = lsilogicConfigurationIOCPageGetFromNumber(pLsiLogic,
2669 pLsiLogic->pConfigurationPages,
2670 pConfigurationReq->u8PageNumber,
2671 &pPageHeader, &pbPageData, &cbPage);
2672 break;
2673 }
2674 case MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING:
2675 {
2676 /* Get the page data. */
2677 rc = lsilogicConfigurationManufacturingPageGetFromNumber(pLsiLogic,
2678 pLsiLogic->pConfigurationPages,
2679 pConfigurationReq->u8PageNumber,
2680 &pPageHeader, &pbPageData, &cbPage);
2681 break;
2682 }
2683 case MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT:
2684 {
2685 /* Get the page data. */
2686 rc = lsilogicConfigurationSCSISPIPortPageGetFromNumber(pLsiLogic,
2687 pLsiLogic->pConfigurationPages,
2688 pConfigurationReq->PageAddress.MPIPortNumber.u8PortNumber,
2689 pConfigurationReq->u8PageNumber,
2690 &pPageHeader, &pbPageData, &cbPage);
2691 break;
2692 }
2693 case MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE:
2694 {
2695 /* Get the page data. */
2696 rc = lsilogicConfigurationSCSISPIDevicePageGetFromNumber(pLsiLogic,
2697 pLsiLogic->pConfigurationPages,
2698 pConfigurationReq->PageAddress.BusAndTargetId.u8Bus,
2699 pConfigurationReq->PageAddress.BusAndTargetId.u8TargetID,
2700 pConfigurationReq->u8PageNumber,
2701 &pPageHeader, &pbPageData, &cbPage);
2702 break;
2703 }
2704 case MPT_CONFIGURATION_PAGE_TYPE_BIOS:
2705 {
2706 rc = lsilogicConfigurationBiosPageGetFromNumber(pLsiLogic,
2707 pLsiLogic->pConfigurationPages,
2708 pConfigurationReq->u8PageNumber,
2709 &pPageHeader, &pbPageData, &cbPage);
2710 break;
2711 }
2712 case MPT_CONFIGURATION_PAGE_TYPE_EXTENDED:
2713 {
2714 rc = lsilogicConfigurationPageGetExtended(pLsiLogic,
2715 pConfigurationReq,
2716 &pExtPageHeader, &pbPageData, &cbPage);
2717 break;
2718 }
2719 default:
2720 rc = VERR_NOT_FOUND;
2721 }
2722
2723 if (rc == VERR_NOT_FOUND)
2724 {
2725 Log(("Page not found\n"));
2726 pReply->u8PageType = pConfigurationReq->u8PageType;
2727 pReply->u8PageNumber = pConfigurationReq->u8PageNumber;
2728 pReply->u8PageLength = pConfigurationReq->u8PageLength;
2729 pReply->u8PageVersion = pConfigurationReq->u8PageVersion;
2730 pReply->u16IOCStatus = MPT_IOCSTATUS_CONFIG_INVALID_PAGE;
2731 return VINF_SUCCESS;
2732 }
2733
2734 if (u8PageType == MPT_CONFIGURATION_PAGE_TYPE_EXTENDED)
2735 {
2736 pReply->u8PageType = pExtPageHeader->u8PageType;
2737 pReply->u8PageNumber = pExtPageHeader->u8PageNumber;
2738 pReply->u8PageVersion = pExtPageHeader->u8PageVersion;
2739 pReply->u8ExtPageType = pExtPageHeader->u8ExtPageType;
2740 pReply->u16ExtPageLength = pExtPageHeader->u16ExtPageLength;
2741
2742 for (int i = 0; i < pExtPageHeader->u16ExtPageLength; i++)
2743 LogFlowFunc(("PageData[%d]=%#x\n", i, ((uint32_t *)pbPageData)[i]));
2744 }
2745 else
2746 {
2747 pReply->u8PageType = pPageHeader->u8PageType;
2748 pReply->u8PageNumber = pPageHeader->u8PageNumber;
2749 pReply->u8PageLength = pPageHeader->u8PageLength;
2750 pReply->u8PageVersion = pPageHeader->u8PageVersion;
2751
2752 for (int i = 0; i < pReply->u8PageLength; i++)
2753 LogFlowFunc(("PageData[%d]=%#x\n", i, ((uint32_t *)pbPageData)[i]));
2754 }
2755
2756 /*
2757 * Don't use the scatter gather handling code as the configuration request always have only one
2758 * simple element.
2759 */
2760 switch (pConfigurationReq->u8Action)
2761 {
2762 case MPT_CONFIGURATION_REQUEST_ACTION_DEFAULT: /* Nothing to do. We are always using the defaults. */
2763 case MPT_CONFIGURATION_REQUEST_ACTION_HEADER:
2764 {
2765 /* Already copied above nothing to do. */
2766 break;
2767 }
2768 case MPT_CONFIGURATION_REQUEST_ACTION_READ_NVRAM:
2769 case MPT_CONFIGURATION_REQUEST_ACTION_READ_CURRENT:
2770 case MPT_CONFIGURATION_REQUEST_ACTION_READ_DEFAULT:
2771 {
2772 uint32_t cbBuffer = pConfigurationReq->SimpleSGElement.u24Length;
2773 if (cbBuffer != 0)
2774 {
2775 RTGCPHYS GCPhysAddrPageBuffer = pConfigurationReq->SimpleSGElement.u32DataBufferAddressLow;
2776 if (pConfigurationReq->SimpleSGElement.f64BitAddress)
2777 GCPhysAddrPageBuffer |= (uint64_t)pConfigurationReq->SimpleSGElement.u32DataBufferAddressHigh << 32;
2778
2779 PDMDevHlpPhysWrite(pLsiLogic->CTX_SUFF(pDevIns), GCPhysAddrPageBuffer, pbPageData,
2780 RT_MIN(cbBuffer, cbPage));
2781 }
2782 break;
2783 }
2784 case MPT_CONFIGURATION_REQUEST_ACTION_WRITE_CURRENT:
2785 case MPT_CONFIGURATION_REQUEST_ACTION_WRITE_NVRAM:
2786 {
2787 uint32_t cbBuffer = pConfigurationReq->SimpleSGElement.u24Length;
2788 if (cbBuffer != 0)
2789 {
2790 RTGCPHYS GCPhysAddrPageBuffer = pConfigurationReq->SimpleSGElement.u32DataBufferAddressLow;
2791 if (pConfigurationReq->SimpleSGElement.f64BitAddress)
2792 GCPhysAddrPageBuffer |= (uint64_t)pConfigurationReq->SimpleSGElement.u32DataBufferAddressHigh << 32;
2793
2794 PDMDevHlpPhysRead(pLsiLogic->CTX_SUFF(pDevIns), GCPhysAddrPageBuffer, pbPageData,
2795 RT_MIN(cbBuffer, cbPage));
2796 }
2797 break;
2798 }
2799 default:
2800 AssertMsgFailed(("todo\n"));
2801 }
2802
2803 return VINF_SUCCESS;
2804}
2805
2806/**
2807 * Initializes the configuration pages for the SPI SCSI controller.
2808 *
2809 * @returns nothing
2810 * @param pLsiLogic Pointer to the Lsilogic SCSI instance.
2811 */
2812static void lsilogicInitializeConfigurationPagesSpi(PLSILOGICSCSI pLsiLogic)
2813{
2814 PMptConfigurationPagesSpi pPages = &pLsiLogic->pConfigurationPages->u.SpiPages;
2815
2816 AssertMsg(pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI, ("Controller is not the SPI SCSI one\n"));
2817
2818 LogFlowFunc(("pLsiLogic=%#p\n", pLsiLogic));
2819
2820 /* Clear everything first. */
2821 memset(pPages, 0, sizeof(PMptConfigurationPagesSpi));
2822
2823 for (unsigned i = 0; i < RT_ELEMENTS(pPages->aPortPages); i++)
2824 {
2825 /* SCSI-SPI port page 0. */
2826 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
2827 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
2828 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageNumber = 0;
2829 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort0) / 4;
2830 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fInformationUnitTransfersCapable = true;
2831 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fDTCapable = true;
2832 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fQASCapable = true;
2833 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u8MinimumSynchronousTransferPeriod = 0;
2834 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u8MaximumSynchronousOffset = 0xff;
2835 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fWide = true;
2836 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fAIPCapable = true;
2837 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u2SignalingType = 0x3; /* Single Ended. */
2838
2839 /* SCSI-SPI port page 1. */
2840 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
2841 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
2842 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageNumber = 1;
2843 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort1) / 4;
2844 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u8SCSIID = 7;
2845 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u16PortResponseIDsBitmask = (1 << 7);
2846 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u32OnBusTimerValue = 0;
2847
2848 /* SCSI-SPI port page 2. */
2849 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
2850 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
2851 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageNumber = 2;
2852 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort2) / 4;
2853 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.u4HostSCSIID = 7;
2854 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.u2InitializeHBA = 0x3;
2855 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.fTerminationDisabled = true;
2856 for (unsigned iDevice = 0; iDevice < RT_ELEMENTS(pPages->aPortPages[i].SCSISPIPortPage2.u.fields.aDeviceSettings); iDevice++)
2857 {
2858 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.aDeviceSettings[iDevice].fBootChoice = true;
2859 }
2860 /* Everything else 0 for now. */
2861 }
2862
2863 for (unsigned uBusCurr = 0; uBusCurr < RT_ELEMENTS(pPages->aBuses); uBusCurr++)
2864 {
2865 for (unsigned uDeviceCurr = 0; uDeviceCurr < RT_ELEMENTS(pPages->aBuses[uBusCurr].aDevicePages); uDeviceCurr++)
2866 {
2867 /* SCSI-SPI device page 0. */
2868 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
2869 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
2870 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageNumber = 0;
2871 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice0) / 4;
2872 /* Everything else 0 for now. */
2873
2874 /* SCSI-SPI device page 1. */
2875 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
2876 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
2877 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageNumber = 1;
2878 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice1) / 4;
2879 /* Everything else 0 for now. */
2880
2881 /* SCSI-SPI device page 2. */
2882 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
2883 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
2884 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageNumber = 2;
2885 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice2) / 4;
2886 /* Everything else 0 for now. */
2887
2888 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
2889 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
2890 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageNumber = 3;
2891 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice3) / 4;
2892 /* Everything else 0 for now. */
2893 }
2894 }
2895}
2896
2897/**
2898 * Generates a handle.
2899 *
2900 * @returns the handle.
2901 * @param pThis The LsiLogic instance.
2902 */
2903DECLINLINE(uint16_t) lsilogicGetHandle(PLSILOGICSCSI pThis)
2904{
2905 uint16_t u16Handle = pThis->u16NextHandle++;
2906 return u16Handle;
2907}
2908
2909/**
2910 * Generates a SAS address (WWID)
2911 *
2912 * @returns nothing.
2913 * @param pSASAddress Pointer to an unitialised SAS address.
2914 * @param iId iId which will go into the address.
2915 *
2916 * @todo Generate better SAS addresses. (Request a block from SUN probably)
2917 */
2918void lsilogicSASAddressGenerate(PSASADDRESS pSASAddress, unsigned iId)
2919{
2920 pSASAddress->u8Address[0] = (0x5 << 5);
2921 pSASAddress->u8Address[1] = 0x01;
2922 pSASAddress->u8Address[2] = 0x02;
2923 pSASAddress->u8Address[3] = 0x03;
2924 pSASAddress->u8Address[4] = 0x04;
2925 pSASAddress->u8Address[5] = 0x05;
2926 pSASAddress->u8Address[6] = 0x06;
2927 pSASAddress->u8Address[7] = iId;
2928}
2929
2930/**
2931 * Initializes the configuration pages for the SAS SCSI controller.
2932 *
2933 * @returns nothing
2934 * @param pThis Pointer to the Lsilogic SCSI instance.
2935 */
2936static void lsilogicInitializeConfigurationPagesSas(PLSILOGICSCSI pThis)
2937{
2938 PMptConfigurationPagesSas pPages = &pThis->pConfigurationPages->u.SasPages;
2939
2940 AssertMsg(pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS, ("Controller is not the SAS SCSI one\n"));
2941
2942 LogFlowFunc(("pThis=%#p\n", pThis));
2943
2944 /* Manufacturing Page 7 - Connector settings. */
2945 pPages->cbManufacturingPage7 = LSILOGICSCSI_MANUFACTURING7_GET_SIZE(pThis->cPorts);
2946 PMptConfigurationPageManufacturing7 pManufacturingPage7 = (PMptConfigurationPageManufacturing7)RTMemAllocZ(pPages->cbManufacturingPage7);
2947 AssertPtr(pManufacturingPage7);
2948 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(pManufacturingPage7,
2949 0, 7,
2950 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
2951 /* Set size manually. */
2952 if (pPages->cbManufacturingPage7 / 4 > 255)
2953 pManufacturingPage7->u.fields.Header.u8PageLength = 255;
2954 else
2955 pManufacturingPage7->u.fields.Header.u8PageLength = pPages->cbManufacturingPage7 / 4;
2956 pManufacturingPage7->u.fields.u8NumPhys = pThis->cPorts;
2957 pPages->pManufacturingPage7 = pManufacturingPage7;
2958
2959 /* SAS I/O unit page 0 - Port specific informations. */
2960 pPages->cbSASIOUnitPage0 = LSILOGICSCSI_SASIOUNIT0_GET_SIZE(pThis->cPorts);
2961 PMptConfigurationPageSASIOUnit0 pSASPage0 = (PMptConfigurationPageSASIOUnit0)RTMemAllocZ(pPages->cbSASIOUnitPage0);
2962 AssertPtr(pSASPage0);
2963
2964 MPT_CONFIG_EXTENDED_PAGE_HEADER_INIT(pSASPage0, pPages->cbSASIOUnitPage0,
2965 0, MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY,
2966 MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT);
2967 pSASPage0->u.fields.u8NumPhys = pThis->cPorts;
2968 pPages->pSASIOUnitPage0 = pSASPage0;
2969
2970 /* SAS I/O unit page 1 - Port specific settings. */
2971 pPages->cbSASIOUnitPage1 = LSILOGICSCSI_SASIOUNIT1_GET_SIZE(pThis->cPorts);
2972 PMptConfigurationPageSASIOUnit1 pSASPage1 = (PMptConfigurationPageSASIOUnit1)RTMemAllocZ(pPages->cbSASIOUnitPage1);
2973 AssertPtr(pSASPage1);
2974
2975 MPT_CONFIG_EXTENDED_PAGE_HEADER_INIT(pSASPage1, pPages->cbSASIOUnitPage1,
2976 1, MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE,
2977 MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT);
2978 pSASPage1->u.fields.u8NumPhys = pSASPage0->u.fields.u8NumPhys;
2979 pSASPage1->u.fields.u16ControlFlags = 0;
2980 pSASPage1->u.fields.u16AdditionalControlFlags = 0;
2981 pPages->pSASIOUnitPage1 = pSASPage1;
2982
2983 /* SAS I/O unit page 2 - Port specific informations. */
2984 pPages->SASIOUnitPage2.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
2985 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
2986 pPages->SASIOUnitPage2.u.fields.ExtHeader.u8PageNumber = 2;
2987 pPages->SASIOUnitPage2.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT;
2988 pPages->SASIOUnitPage2.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASIOUnit2) / 4;
2989
2990 /* SAS I/O unit page 3 - Port specific informations. */
2991 pPages->SASIOUnitPage3.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
2992 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
2993 pPages->SASIOUnitPage3.u.fields.ExtHeader.u8PageNumber = 3;
2994 pPages->SASIOUnitPage3.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASIOUNIT;
2995 pPages->SASIOUnitPage3.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASIOUnit3) / 4;
2996
2997 pPages->cPHYs = pThis->cPorts;
2998 pPages->paPHYs = (PMptPHY)RTMemAllocZ(pPages->cPHYs * sizeof(MptPHY));
2999 AssertPtr(pPages->paPHYs);
3000
3001 /* Initialize the PHY configuration */
3002 for (unsigned i = 0; i < pThis->cPorts; i++)
3003 {
3004 PMptPHY pPHYPages = &pPages->paPHYs[i];
3005 uint16_t u16ControllerHandle = lsilogicGetHandle(pThis);
3006
3007 pManufacturingPage7->u.fields.aPHY[i].u8Location = LSILOGICSCSI_MANUFACTURING7_LOCATION_AUTO;
3008
3009 pSASPage0->u.fields.aPHY[i].u8Port = i;
3010 pSASPage0->u.fields.aPHY[i].u8PortFlags = 0;
3011 pSASPage0->u.fields.aPHY[i].u8PhyFlags = 0;
3012 pSASPage0->u.fields.aPHY[i].u8NegotiatedLinkRate = LSILOGICSCSI_SASIOUNIT0_NEGOTIATED_RATE_FAILED;
3013 pSASPage0->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_NO);
3014 pSASPage0->u.fields.aPHY[i].u16ControllerDevHandle = u16ControllerHandle;
3015 pSASPage0->u.fields.aPHY[i].u16AttachedDevHandle = 0; /* No device attached. */
3016 pSASPage0->u.fields.aPHY[i].u32DiscoveryStatus = 0; /* No errors */
3017
3018 pSASPage1->u.fields.aPHY[i].u8Port = i;
3019 pSASPage1->u.fields.aPHY[i].u8PortFlags = 0;
3020 pSASPage1->u.fields.aPHY[i].u8PhyFlags = 0;
3021 pSASPage1->u.fields.aPHY[i].u8MaxMinLinkRate = LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MIN_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_15GB)
3022 | LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MAX_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_30GB);
3023 pSASPage1->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_NO);
3024
3025 /* SAS PHY page 0. */
3026 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3027 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3028 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u8PageNumber = 0;
3029 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASPHYS;
3030 pPHYPages->SASPHYPage0.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASPHY0) / 4;
3031 pPHYPages->SASPHYPage0.u.fields.u8AttachedPhyIdentifier = i;
3032 pPHYPages->SASPHYPage0.u.fields.u32AttachedDeviceInfo = LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_SET(LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_NO);
3033 pPHYPages->SASPHYPage0.u.fields.u8ProgrammedLinkRate = LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MIN_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_15GB)
3034 | LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MAX_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_30GB);
3035 pPHYPages->SASPHYPage0.u.fields.u8HwLinkRate = LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MIN_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_15GB)
3036 | LSILOGICSCSI_SASIOUNIT1_LINK_RATE_MAX_SET(LSILOGICSCSI_SASIOUNIT1_LINK_RATE_30GB);
3037
3038 /* SAS PHY page 1. */
3039 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3040 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3041 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u8PageNumber = 1;
3042 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASPHYS;
3043 pPHYPages->SASPHYPage1.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASPHY1) / 4;
3044
3045 /* Settings for present devices. */
3046 if (pThis->paDeviceStates[i].pDrvBase)
3047 {
3048 uint16_t u16DeviceHandle = lsilogicGetHandle(pThis);
3049 SASADDRESS SASAddress;
3050 PMptSASDevice pSASDevice = (PMptSASDevice)RTMemAllocZ(sizeof(MptSASDevice));
3051 AssertPtr(pSASDevice);
3052
3053 memset(&SASAddress, 0, sizeof(SASADDRESS));
3054 lsilogicSASAddressGenerate(&SASAddress, i);
3055
3056 pSASPage0->u.fields.aPHY[i].u8NegotiatedLinkRate = LSILOGICSCSI_SASIOUNIT0_NEGOTIATED_RATE_SET(LSILOGICSCSI_SASIOUNIT0_NEGOTIATED_RATE_30GB);
3057 pSASPage0->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_END)
3058 | LSILOGICSCSI_SASIOUNIT0_DEVICE_SSP_TARGET;
3059 pSASPage0->u.fields.aPHY[i].u16AttachedDevHandle = u16DeviceHandle;
3060 pSASPage1->u.fields.aPHY[i].u32ControllerPhyDeviceInfo = LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_SET(LSILOGICSCSI_SASIOUNIT0_DEVICE_TYPE_END)
3061 | LSILOGICSCSI_SASIOUNIT0_DEVICE_SSP_TARGET;
3062 pSASPage0->u.fields.aPHY[i].u16ControllerDevHandle = u16DeviceHandle;
3063
3064 pPHYPages->SASPHYPage0.u.fields.u32AttachedDeviceInfo = LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_SET(LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_END);
3065 pPHYPages->SASPHYPage0.u.fields.SASAddress = SASAddress;
3066 pPHYPages->SASPHYPage0.u.fields.u16OwnerDevHandle = u16DeviceHandle;
3067 pPHYPages->SASPHYPage0.u.fields.u16AttachedDevHandle = u16DeviceHandle;
3068
3069 /* SAS device page 0. */
3070 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3071 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3072 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u8PageNumber = 0;
3073 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE;
3074 pSASDevice->SASDevicePage0.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASDevice0) / 4;
3075 pSASDevice->SASDevicePage0.u.fields.SASAddress = SASAddress;
3076 pSASDevice->SASDevicePage0.u.fields.u16ParentDevHandle = u16ControllerHandle;
3077 pSASDevice->SASDevicePage0.u.fields.u8PhyNum = i;
3078 pSASDevice->SASDevicePage0.u.fields.u8AccessStatus = LSILOGICSCSI_SASDEVICE0_STATUS_NO_ERRORS;
3079 pSASDevice->SASDevicePage0.u.fields.u16DevHandle = u16DeviceHandle;
3080 pSASDevice->SASDevicePage0.u.fields.u8TargetID = i;
3081 pSASDevice->SASDevicePage0.u.fields.u8Bus = 0;
3082 pSASDevice->SASDevicePage0.u.fields.u32DeviceInfo = LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_SET(LSILOGICSCSI_SASPHY0_DEV_INFO_DEVICE_TYPE_END)
3083 | LSILOGICSCSI_SASIOUNIT0_DEVICE_SSP_TARGET;
3084 pSASDevice->SASDevicePage0.u.fields.u16Flags = LSILOGICSCSI_SASDEVICE0_FLAGS_DEVICE_PRESENT
3085 | LSILOGICSCSI_SASDEVICE0_FLAGS_DEVICE_MAPPED_TO_BUS_AND_TARGET_ID
3086 | LSILOGICSCSI_SASDEVICE0_FLAGS_DEVICE_MAPPING_PERSISTENT;
3087 pSASDevice->SASDevicePage0.u.fields.u8PhysicalPort = i;
3088
3089 /* SAS device page 1. */
3090 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3091 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3092 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u8PageNumber = 1;
3093 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE;
3094 pSASDevice->SASDevicePage1.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASDevice1) / 4;
3095 pSASDevice->SASDevicePage1.u.fields.SASAddress = SASAddress;
3096 pSASDevice->SASDevicePage1.u.fields.u16DevHandle = u16DeviceHandle;
3097 pSASDevice->SASDevicePage1.u.fields.u8TargetID = i;
3098 pSASDevice->SASDevicePage1.u.fields.u8Bus = 0;
3099
3100 /* SAS device page 2. */
3101 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
3102 | MPT_CONFIGURATION_PAGE_TYPE_EXTENDED;
3103 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u8PageNumber = 2;
3104 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u8ExtPageType = MPT_CONFIGURATION_PAGE_TYPE_EXTENDED_SASDEVICE;
3105 pSASDevice->SASDevicePage2.u.fields.ExtHeader.u16ExtPageLength = sizeof(MptConfigurationPageSASDevice2) / 4;
3106 pSASDevice->SASDevicePage2.u.fields.SASAddress = SASAddress;
3107
3108 /* Link into device list. */
3109 if (!pPages->cDevices)
3110 {
3111 pPages->pSASDeviceHead = pSASDevice;
3112 pPages->pSASDeviceTail = pSASDevice;
3113 pPages->cDevices = 1;
3114 }
3115 else
3116 {
3117 pSASDevice->pPrev = pPages->pSASDeviceTail;
3118 pPages->pSASDeviceTail->pNext = pSASDevice;
3119 pPages->pSASDeviceTail = pSASDevice;
3120 pPages->cDevices++;
3121 }
3122 }
3123 }
3124}
3125
3126/**
3127 * Initializes the configuration pages.
3128 *
3129 * @returns nothing
3130 * @param pLsiLogic Pointer to the Lsilogic SCSI instance.
3131 */
3132static void lsilogicInitializeConfigurationPages(PLSILOGICSCSI pLsiLogic)
3133{
3134 /* Initialize the common pages. */
3135 PMptConfigurationPagesSupported pPages = (PMptConfigurationPagesSupported)RTMemAllocZ(sizeof(MptConfigurationPagesSupported));
3136
3137 pLsiLogic->pConfigurationPages = pPages;
3138
3139 LogFlowFunc(("pLsiLogic=%#p\n", pLsiLogic));
3140
3141 /* Clear everything first. */
3142 memset(pPages, 0, sizeof(MptConfigurationPagesSupported));
3143
3144 /* Manufacturing Page 0. */
3145 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage0,
3146 MptConfigurationPageManufacturing0, 0,
3147 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3148 strncpy((char *)pPages->ManufacturingPage0.u.fields.abChipName, "VBox MPT Fusion", 16);
3149 strncpy((char *)pPages->ManufacturingPage0.u.fields.abChipRevision, "1.0", 8);
3150 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardName, "VBox MPT Fusion", 16);
3151 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardAssembly, "SUN", 8);
3152 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardTracerNumber, "CAFECAFECAFECAFE", 16);
3153
3154 /* Manufacturing Page 1 - I don't know what this contains so we leave it 0 for now. */
3155 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage1,
3156 MptConfigurationPageManufacturing1, 1,
3157 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3158
3159 /* Manufacturing Page 2. */
3160 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage2,
3161 MptConfigurationPageManufacturing2, 2,
3162 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3163
3164 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3165 {
3166 pPages->ManufacturingPage2.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SPI_DEVICE_ID;
3167 pPages->ManufacturingPage2.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SPI_REVISION_ID;
3168 }
3169 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3170 {
3171 pPages->ManufacturingPage2.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SAS_DEVICE_ID;
3172 pPages->ManufacturingPage2.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SAS_REVISION_ID;
3173 }
3174
3175 /* Manufacturing Page 3. */
3176 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage3,
3177 MptConfigurationPageManufacturing3, 3,
3178 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3179
3180 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3181 {
3182 pPages->ManufacturingPage3.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SPI_DEVICE_ID;
3183 pPages->ManufacturingPage3.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SPI_REVISION_ID;
3184 }
3185 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3186 {
3187 pPages->ManufacturingPage3.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_SAS_DEVICE_ID;
3188 pPages->ManufacturingPage3.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_SAS_REVISION_ID;
3189 }
3190
3191 /* Manufacturing Page 4 - I don't know what this contains so we leave it 0 for now. */
3192 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage4,
3193 MptConfigurationPageManufacturing4, 4,
3194 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3195
3196 /* Manufacturing Page 5 - WWID settings. */
3197 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage5,
3198 MptConfigurationPageManufacturing5, 5,
3199 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY);
3200
3201 /* Manufacturing Page 6 - Product sepcific settings. */
3202 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage6,
3203 MptConfigurationPageManufacturing6, 6,
3204 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3205
3206 /* Manufacturing Page 8 - Product sepcific settings. */
3207 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage8,
3208 MptConfigurationPageManufacturing8, 8,
3209 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3210
3211 /* Manufacturing Page 9 - Product sepcific settings. */
3212 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage9,
3213 MptConfigurationPageManufacturing9, 9,
3214 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3215
3216 /* Manufacturing Page 10 - Product sepcific settings. */
3217 MPT_CONFIG_PAGE_HEADER_INIT_MANUFACTURING(&pPages->ManufacturingPage10,
3218 MptConfigurationPageManufacturing10, 10,
3219 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3220
3221 /* I/O Unit page 0. */
3222 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage0,
3223 MptConfigurationPageIOUnit0, 0,
3224 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3225 pPages->IOUnitPage0.u.fields.u64UniqueIdentifier = 0xcafe;
3226
3227 /* I/O Unit page 1. */
3228 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage1,
3229 MptConfigurationPageIOUnit1, 1,
3230 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3231 pPages->IOUnitPage1.u.fields.fSingleFunction = true;
3232 pPages->IOUnitPage1.u.fields.fAllPathsMapped = false;
3233 pPages->IOUnitPage1.u.fields.fIntegratedRAIDDisabled = true;
3234 pPages->IOUnitPage1.u.fields.f32BitAccessForced = false;
3235
3236 /* I/O Unit page 2. */
3237 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage2,
3238 MptConfigurationPageIOUnit2, 2,
3239 MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT);
3240 pPages->IOUnitPage2.u.fields.fPauseOnError = false;
3241 pPages->IOUnitPage2.u.fields.fVerboseModeEnabled = false;
3242 pPages->IOUnitPage2.u.fields.fDisableColorVideo = false;
3243 pPages->IOUnitPage2.u.fields.fNotHookInt40h = false;
3244 pPages->IOUnitPage2.u.fields.u32BIOSVersion = 0xcafecafe;
3245 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].fAdapterEnabled = true;
3246 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].fAdapterEmbedded = true;
3247 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].u8PCIBusNumber = 0;
3248 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].u8PCIDevFn = pLsiLogic->PciDev.devfn;
3249
3250 /* I/O Unit page 3. */
3251 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage3,
3252 MptConfigurationPageIOUnit3, 3,
3253 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3254 pPages->IOUnitPage3.u.fields.u8GPIOCount = 0;
3255
3256 /* I/O Unit page 4. */
3257 MPT_CONFIG_PAGE_HEADER_INIT_IO_UNIT(&pPages->IOUnitPage4,
3258 MptConfigurationPageIOUnit4, 4,
3259 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3260
3261 /* IOC page 0. */
3262 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage0,
3263 MptConfigurationPageIOC0, 0,
3264 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3265 pPages->IOCPage0.u.fields.u32TotalNVStore = 0;
3266 pPages->IOCPage0.u.fields.u32FreeNVStore = 0;
3267
3268 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3269 {
3270 pPages->IOCPage0.u.fields.u16VendorId = LSILOGICSCSI_PCI_VENDOR_ID;
3271 pPages->IOCPage0.u.fields.u16DeviceId = LSILOGICSCSI_PCI_SPI_DEVICE_ID;
3272 pPages->IOCPage0.u.fields.u8RevisionId = LSILOGICSCSI_PCI_SPI_REVISION_ID;
3273 pPages->IOCPage0.u.fields.u32ClassCode = LSILOGICSCSI_PCI_SPI_CLASS_CODE;
3274 pPages->IOCPage0.u.fields.u16SubsystemVendorId = LSILOGICSCSI_PCI_SPI_SUBSYSTEM_VENDOR_ID;
3275 pPages->IOCPage0.u.fields.u16SubsystemId = LSILOGICSCSI_PCI_SPI_SUBSYSTEM_ID;
3276 }
3277 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3278 {
3279 pPages->IOCPage0.u.fields.u16VendorId = LSILOGICSCSI_PCI_VENDOR_ID;
3280 pPages->IOCPage0.u.fields.u16DeviceId = LSILOGICSCSI_PCI_SAS_DEVICE_ID;
3281 pPages->IOCPage0.u.fields.u8RevisionId = LSILOGICSCSI_PCI_SAS_REVISION_ID;
3282 pPages->IOCPage0.u.fields.u32ClassCode = LSILOGICSCSI_PCI_SAS_CLASS_CODE;
3283 pPages->IOCPage0.u.fields.u16SubsystemVendorId = LSILOGICSCSI_PCI_SAS_SUBSYSTEM_VENDOR_ID;
3284 pPages->IOCPage0.u.fields.u16SubsystemId = LSILOGICSCSI_PCI_SAS_SUBSYSTEM_ID;
3285 }
3286
3287 /* IOC page 1. */
3288 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage1,
3289 MptConfigurationPageIOC1, 1,
3290 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3291 pPages->IOCPage1.u.fields.fReplyCoalescingEnabled = false;
3292 pPages->IOCPage1.u.fields.u32CoalescingTimeout = 0;
3293 pPages->IOCPage1.u.fields.u8CoalescingDepth = 0;
3294
3295 /* IOC page 2. */
3296 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage2,
3297 MptConfigurationPageIOC2, 2,
3298 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3299 /* Everything else here is 0. */
3300
3301 /* IOC page 3. */
3302 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage3,
3303 MptConfigurationPageIOC3, 3,
3304 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3305 /* Everything else here is 0. */
3306
3307 /* IOC page 4. */
3308 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage4,
3309 MptConfigurationPageIOC4, 4,
3310 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3311 /* Everything else here is 0. */
3312
3313 /* IOC page 6. */
3314 MPT_CONFIG_PAGE_HEADER_INIT_IOC(&pPages->IOCPage6,
3315 MptConfigurationPageIOC6, 6,
3316 MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY);
3317 /* Everything else here is 0. */
3318
3319 /* BIOS page 1. */
3320 MPT_CONFIG_PAGE_HEADER_INIT_BIOS(&pPages->BIOSPage1,
3321 MptConfigurationPageBIOS1, 1,
3322 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3323
3324 /* BIOS page 2. */
3325 MPT_CONFIG_PAGE_HEADER_INIT_BIOS(&pPages->BIOSPage2,
3326 MptConfigurationPageBIOS2, 2,
3327 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3328
3329 /* BIOS page 4. */
3330 MPT_CONFIG_PAGE_HEADER_INIT_BIOS(&pPages->BIOSPage4,
3331 MptConfigurationPageBIOS4, 4,
3332 MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE);
3333
3334 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3335 lsilogicInitializeConfigurationPagesSpi(pLsiLogic);
3336 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3337 lsilogicInitializeConfigurationPagesSas(pLsiLogic);
3338 else
3339 AssertMsgFailed(("Invalid controller type %d\n", pLsiLogic->enmCtrlType));
3340}
3341
3342/**
3343 * Transmit queue consumer
3344 * Queue a new async task.
3345 *
3346 * @returns Success indicator.
3347 * If false the item will not be removed and the flushing will stop.
3348 * @param pDevIns The device instance.
3349 * @param pItem The item to consume. Upon return this item will be freed.
3350 */
3351static DECLCALLBACK(bool) lsilogicNotifyQueueConsumer(PPDMDEVINS pDevIns, PPDMQUEUEITEMCORE pItem)
3352{
3353 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3354 int rc = VINF_SUCCESS;
3355
3356 LogFlowFunc(("pDevIns=%#p pItem=%#p\n", pDevIns, pItem));
3357
3358 /* Only process request which arrived before we received the notification. */
3359 uint32_t uRequestQueueNextEntryWrite = ASMAtomicReadU32(&pLsiLogic->uRequestQueueNextEntryFreeWrite);
3360
3361 /* Reset notification event. */
3362 ASMAtomicXchgBool(&pLsiLogic->fNotificationSend, false);
3363
3364 /* Go through the messages now and process them. */
3365 while ( RT_LIKELY(pLsiLogic->enmState == LSILOGICSTATE_OPERATIONAL)
3366 && (pLsiLogic->uRequestQueueNextAddressRead != uRequestQueueNextEntryWrite))
3367 {
3368 uint32_t u32RequestMessageFrameDesc = pLsiLogic->CTX_SUFF(pRequestQueueBase)[pLsiLogic->uRequestQueueNextAddressRead];
3369 RTGCPHYS GCPhysMessageFrameAddr = LSILOGIC_RTGCPHYS_FROM_U32(pLsiLogic->u32HostMFAHighAddr,
3370 (u32RequestMessageFrameDesc & ~0x07));
3371
3372 PLSILOGICTASKSTATE pTaskState;
3373
3374 /* Get new task state. */
3375 rc = RTMemCacheAllocEx(pLsiLogic->hTaskCache, (void **)&pTaskState);
3376 AssertRC(rc);
3377
3378 pTaskState->GCPhysMessageFrameAddr = GCPhysMessageFrameAddr;
3379
3380 /* Read the message header from the guest first. */
3381 PDMDevHlpPhysRead(pDevIns, GCPhysMessageFrameAddr, &pTaskState->GuestRequest, sizeof(MptMessageHdr));
3382
3383 /* Determine the size of the request. */
3384 uint32_t cbRequest = 0;
3385
3386 switch (pTaskState->GuestRequest.Header.u8Function)
3387 {
3388 case MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST:
3389 cbRequest = sizeof(MptSCSIIORequest);
3390 break;
3391 case MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT:
3392 cbRequest = sizeof(MptSCSITaskManagementRequest);
3393 break;
3394 case MPT_MESSAGE_HDR_FUNCTION_IOC_INIT:
3395 cbRequest = sizeof(MptIOCInitRequest);
3396 break;
3397 case MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS:
3398 cbRequest = sizeof(MptIOCFactsRequest);
3399 break;
3400 case MPT_MESSAGE_HDR_FUNCTION_CONFIG:
3401 cbRequest = sizeof(MptConfigurationRequest);
3402 break;
3403 case MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS:
3404 cbRequest = sizeof(MptPortFactsRequest);
3405 break;
3406 case MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE:
3407 cbRequest = sizeof(MptPortEnableRequest);
3408 break;
3409 case MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION:
3410 cbRequest = sizeof(MptEventNotificationRequest);
3411 break;
3412 case MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK:
3413 AssertMsgFailed(("todo\n"));
3414 //cbRequest = sizeof(MptEventAckRequest);
3415 break;
3416 case MPT_MESSAGE_HDR_FUNCTION_FW_DOWNLOAD:
3417 AssertMsgFailed(("todo\n"));
3418 break;
3419 default:
3420 AssertMsgFailed(("Unknown function issued %u\n", pTaskState->GuestRequest.Header.u8Function));
3421 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INVALID_FUNCTION);
3422 }
3423
3424 if (cbRequest != 0)
3425 {
3426 /* Read the complete message frame from guest memory now. */
3427 PDMDevHlpPhysRead(pDevIns, GCPhysMessageFrameAddr, &pTaskState->GuestRequest, cbRequest);
3428
3429 /* Handle SCSI I/O requests now. */
3430 if (pTaskState->GuestRequest.Header.u8Function == MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST)
3431 {
3432 rc = lsilogicProcessSCSIIORequest(pLsiLogic, pTaskState);
3433 AssertRC(rc);
3434 }
3435 else
3436 {
3437 MptReplyUnion Reply;
3438 rc = lsilogicProcessMessageRequest(pLsiLogic, &pTaskState->GuestRequest.Header, &Reply);
3439 AssertRC(rc);
3440 RTMemCacheFree(pLsiLogic->hTaskCache, pTaskState);
3441 }
3442
3443 pLsiLogic->uRequestQueueNextAddressRead++;
3444 pLsiLogic->uRequestQueueNextAddressRead %= pLsiLogic->cRequestQueueEntries;
3445 }
3446 }
3447
3448 return true;
3449}
3450
3451/**
3452 * Sets the emulated controller type from a given string.
3453 *
3454 * @returns VBox status code.
3455 *
3456 * @param pThis The LsiLogic devi state.
3457 * @param pcszCtrlType The string to use.
3458 */
3459static int lsilogicGetCtrlTypeFromString(PLSILOGICSCSI pThis, const char *pcszCtrlType)
3460{
3461 int rc = VERR_INVALID_PARAMETER;
3462
3463 if (!RTStrCmp(pcszCtrlType, LSILOGICSCSI_PCI_SPI_CTRLNAME))
3464 {
3465 pThis->enmCtrlType = LSILOGICCTRLTYPE_SCSI_SPI;
3466 rc = VINF_SUCCESS;
3467 }
3468 else if (!RTStrCmp(pcszCtrlType, LSILOGICSCSI_PCI_SAS_CTRLNAME))
3469 {
3470 pThis->enmCtrlType = LSILOGICCTRLTYPE_SCSI_SAS;
3471 rc = VINF_SUCCESS;
3472 }
3473
3474 return rc;
3475}
3476
3477/**
3478 * Port I/O Handler for IN operations - legacy port.
3479 *
3480 * @returns VBox status code.
3481 *
3482 * @param pDevIns The device instance.
3483 * @param pvUser User argument.
3484 * @param uPort Port number used for the IN operation.
3485 * @param pu32 Where to store the result.
3486 * @param cb Number of bytes read.
3487 */
3488static int lsilogicIsaIOPortRead (PPDMDEVINS pDevIns, void *pvUser,
3489 RTIOPORT Port, uint32_t *pu32, unsigned cb)
3490{
3491 int rc;
3492 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3493
3494 Assert(cb == 1);
3495
3496 rc = vboxscsiReadRegister(&pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT), pu32);
3497
3498 Log2(("%s: pu32=%p:{%.*Rhxs} iRegister=%d rc=%Rrc\n",
3499 __FUNCTION__, pu32, 1, pu32, (Port - LSILOGIC_ISA_IO_PORT), rc));
3500
3501 return rc;
3502}
3503
3504/**
3505 * Prepares a request from the BIOS.
3506 *
3507 * @returns VBox status code.
3508 * @param pLsiLogic Pointer to the LsiLogic device instance.
3509 */
3510static int lsilogicPrepareBIOSSCSIRequest(PLSILOGICSCSI pLsiLogic)
3511{
3512 int rc;
3513 PLSILOGICTASKSTATE pTaskState;
3514 uint32_t uTargetDevice;
3515
3516 rc = RTMemCacheAllocEx(pLsiLogic->hTaskCache, (void **)&pTaskState);
3517 AssertMsgRCReturn(rc, ("Getting task from cache failed rc=%Rrc\n", rc), rc);
3518
3519 pTaskState->fBIOS = true;
3520
3521 rc = vboxscsiSetupRequest(&pLsiLogic->VBoxSCSI, &pTaskState->PDMScsiRequest, &uTargetDevice);
3522 AssertMsgRCReturn(rc, ("Setting up SCSI request failed rc=%Rrc\n", rc), rc);
3523
3524 pTaskState->PDMScsiRequest.pvUser = pTaskState;
3525
3526 if (uTargetDevice < pLsiLogic->cDeviceStates)
3527 {
3528 pTaskState->pTargetDevice = &pLsiLogic->paDeviceStates[uTargetDevice];
3529
3530 if (pTaskState->pTargetDevice->pDrvBase)
3531 {
3532 ASMAtomicIncU32(&pTaskState->pTargetDevice->cOutstandingRequests);
3533
3534 rc = pTaskState->pTargetDevice->pDrvSCSIConnector->pfnSCSIRequestSend(pTaskState->pTargetDevice->pDrvSCSIConnector,
3535 &pTaskState->PDMScsiRequest);
3536 AssertMsgRCReturn(rc, ("Sending request to SCSI layer failed rc=%Rrc\n", rc), rc);
3537 return VINF_SUCCESS;
3538 }
3539 }
3540
3541 /* Device is not present. */
3542 AssertMsg(pTaskState->PDMScsiRequest.pbCDB[0] == SCSI_INQUIRY,
3543 ("Device is not present but command is not inquiry\n"));
3544
3545 SCSIINQUIRYDATA ScsiInquiryData;
3546
3547 memset(&ScsiInquiryData, 0, sizeof(SCSIINQUIRYDATA));
3548 ScsiInquiryData.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_UNKNOWN;
3549 ScsiInquiryData.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_NOT_CONNECTED_NOT_SUPPORTED;
3550
3551 memcpy(pLsiLogic->VBoxSCSI.pBuf, &ScsiInquiryData, 5);
3552
3553 rc = vboxscsiRequestFinished(&pLsiLogic->VBoxSCSI, &pTaskState->PDMScsiRequest);
3554 AssertMsgRCReturn(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc), rc);
3555
3556 RTMemCacheFree(pLsiLogic->hTaskCache, pTaskState);
3557 return rc;
3558}
3559
3560/**
3561 * Port I/O Handler for OUT operations - legacy port.
3562 *
3563 * @returns VBox status code.
3564 *
3565 * @param pDevIns The device instance.
3566 * @param pvUser User argument.
3567 * @param uPort Port number used for the IN operation.
3568 * @param u32 The value to output.
3569 * @param cb The value size in bytes.
3570 */
3571static int lsilogicIsaIOPortWrite (PPDMDEVINS pDevIns, void *pvUser,
3572 RTIOPORT Port, uint32_t u32, unsigned cb)
3573{
3574 int rc;
3575 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3576
3577 Log2(("#%d %s: pvUser=%#p cb=%d u32=%#x Port=%#x\n",
3578 pDevIns->iInstance, __FUNCTION__, pvUser, cb, u32, Port));
3579
3580 Assert(cb == 1);
3581
3582 rc = vboxscsiWriteRegister(&pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT), (uint8_t)u32);
3583 if (rc == VERR_MORE_DATA)
3584 {
3585 rc = lsilogicPrepareBIOSSCSIRequest(pLsiLogic);
3586 AssertRC(rc);
3587 }
3588 else if (RT_FAILURE(rc))
3589 AssertMsgFailed(("Writing BIOS register failed %Rrc\n", rc));
3590
3591 return VINF_SUCCESS;
3592}
3593
3594/**
3595 * Port I/O Handler for primary port range OUT string operations.
3596 * @see FNIOMIOPORTOUTSTRING for details.
3597 */
3598static DECLCALLBACK(int) lsilogicIsaIOPortWriteStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, RTGCPTR *pGCPtrSrc, PRTGCUINTREG pcTransfer, unsigned cb)
3599{
3600 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3601 int rc;
3602
3603 Log2(("#%d %s: pvUser=%#p cb=%d Port=%#x\n",
3604 pDevIns->iInstance, __FUNCTION__, pvUser, cb, Port));
3605
3606 rc = vboxscsiWriteString(pDevIns, &pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT),
3607 pGCPtrSrc, pcTransfer, cb);
3608 if (rc == VERR_MORE_DATA)
3609 {
3610 rc = lsilogicPrepareBIOSSCSIRequest(pLsiLogic);
3611 AssertRC(rc);
3612 }
3613 else if (RT_FAILURE(rc))
3614 AssertMsgFailed(("Writing BIOS register failed %Rrc\n", rc));
3615
3616 return rc;
3617}
3618
3619/**
3620 * Port I/O Handler for primary port range IN string operations.
3621 * @see FNIOMIOPORTINSTRING for details.
3622 */
3623static DECLCALLBACK(int) lsilogicIsaIOPortReadStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, RTGCPTR *pGCPtrDst, PRTGCUINTREG pcTransfer, unsigned cb)
3624{
3625 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3626
3627 LogFlowFunc(("#%d %s: pvUser=%#p cb=%d Port=%#x\n",
3628 pDevIns->iInstance, __FUNCTION__, pvUser, cb, Port));
3629
3630 return vboxscsiReadString(pDevIns, &pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT),
3631 pGCPtrDst, pcTransfer, cb);
3632}
3633
3634static DECLCALLBACK(int) lsilogicMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
3635 RTGCPHYS GCPhysAddress, uint32_t cb,
3636 PCIADDRESSSPACE enmType)
3637{
3638 PPDMDEVINS pDevIns = pPciDev->pDevIns;
3639 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3640 int rc = VINF_SUCCESS;
3641
3642 Log2(("%s: registering area at GCPhysAddr=%RGp cb=%u\n", __FUNCTION__, GCPhysAddress, cb));
3643
3644 AssertMsg( (enmType == PCI_ADDRESS_SPACE_MEM && cb >= LSILOGIC_PCI_SPACE_MEM_SIZE)
3645 || (enmType == PCI_ADDRESS_SPACE_IO && cb >= LSILOGIC_PCI_SPACE_IO_SIZE),
3646 ("PCI region type and size do not match\n"));
3647
3648 if ((enmType == PCI_ADDRESS_SPACE_MEM) && (iRegion == 1))
3649 {
3650 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
3651 rc = PDMDevHlpMMIORegister(pDevIns, GCPhysAddress, cb, NULL,
3652 lsilogicMMIOWrite, lsilogicMMIORead, NULL, "LsiLogic");
3653 if (RT_FAILURE(rc))
3654 return rc;
3655
3656 if (pThis->fR0Enabled)
3657 {
3658 rc = PDMDevHlpMMIORegisterR0(pDevIns, GCPhysAddress, cb, 0,
3659 "lsilogicMMIOWrite", "lsilogicMMIORead", NULL);
3660 if (RT_FAILURE(rc))
3661 return rc;
3662 }
3663
3664 if (pThis->fGCEnabled)
3665 {
3666 rc = PDMDevHlpMMIORegisterRC(pDevIns, GCPhysAddress, cb, 0,
3667 "lsilogicMMIOWrite", "lsilogicMMIORead", NULL);
3668 if (RT_FAILURE(rc))
3669 return rc;
3670 }
3671
3672 pThis->GCPhysMMIOBase = GCPhysAddress;
3673 }
3674 else if ((enmType == PCI_ADDRESS_SPACE_MEM) && (iRegion == 2))
3675 {
3676 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
3677 rc = PDMDevHlpMMIORegister(pDevIns, GCPhysAddress, cb, NULL,
3678 lsilogicDiagnosticWrite, lsilogicDiagnosticRead, NULL, "LsiLogicDiag");
3679 if (RT_FAILURE(rc))
3680 return rc;
3681
3682 if (pThis->fR0Enabled)
3683 {
3684 rc = PDMDevHlpMMIORegisterR0(pDevIns, GCPhysAddress, cb, 0,
3685 "lsilogicDiagnosticWrite", "lsilogicDiagnosticRead", NULL);
3686 if (RT_FAILURE(rc))
3687 return rc;
3688 }
3689
3690 if (pThis->fGCEnabled)
3691 {
3692 rc = PDMDevHlpMMIORegisterRC(pDevIns, GCPhysAddress, cb, 0,
3693 "lsilogicDiagnosticWrite", "lsilogicDiagnosticRead", NULL);
3694 if (RT_FAILURE(rc))
3695 return rc;
3696 }
3697 }
3698 else if (enmType == PCI_ADDRESS_SPACE_IO)
3699 {
3700 rc = PDMDevHlpIOPortRegister(pDevIns, (RTIOPORT)GCPhysAddress, LSILOGIC_PCI_SPACE_IO_SIZE,
3701 NULL, lsilogicIOPortWrite, lsilogicIOPortRead, NULL, NULL, "LsiLogic");
3702 if (RT_FAILURE(rc))
3703 return rc;
3704
3705 if (pThis->fR0Enabled)
3706 {
3707 rc = PDMDevHlpIOPortRegisterR0(pDevIns, (RTIOPORT)GCPhysAddress, LSILOGIC_PCI_SPACE_IO_SIZE,
3708 0, "lsilogicIOPortWrite", "lsilogicIOPortRead", NULL, NULL, "LsiLogic");
3709 if (RT_FAILURE(rc))
3710 return rc;
3711 }
3712
3713 if (pThis->fGCEnabled)
3714 {
3715 rc = PDMDevHlpIOPortRegisterRC(pDevIns, (RTIOPORT)GCPhysAddress, LSILOGIC_PCI_SPACE_IO_SIZE,
3716 0, "lsilogicIOPortWrite", "lsilogicIOPortRead", NULL, NULL, "LsiLogic");
3717 if (RT_FAILURE(rc))
3718 return rc;
3719 }
3720
3721 pThis->IOPortBase = (RTIOPORT)GCPhysAddress;
3722 }
3723 else
3724 AssertMsgFailed(("Invalid enmType=%d iRegion=%d\n", enmType, iRegion));
3725
3726 return rc;
3727}
3728
3729static DECLCALLBACK(int) lsilogicLiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
3730{
3731 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3732
3733 SSMR3PutU32(pSSM, pThis->enmCtrlType);
3734 SSMR3PutU32(pSSM, pThis->cDeviceStates);
3735 SSMR3PutU32(pSSM, pThis->cPorts);
3736
3737 /* Save the device config. */
3738 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
3739 SSMR3PutBool(pSSM, pThis->paDeviceStates[i].pDrvBase != NULL);
3740
3741 return VINF_SSM_DONT_CALL_AGAIN;
3742}
3743
3744static DECLCALLBACK(int) lsilogicSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3745{
3746 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3747
3748 /* Every device first. */
3749 lsilogicLiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
3750 for (unsigned i = 0; i < pLsiLogic->cDeviceStates; i++)
3751 {
3752 PLSILOGICDEVICE pDevice = &pLsiLogic->paDeviceStates[i];
3753
3754 AssertMsg(!pDevice->cOutstandingRequests,
3755 ("There are still outstanding requests on this device\n"));
3756 SSMR3PutU32(pSSM, pDevice->cOutstandingRequests);
3757 }
3758 /* Now the main device state. */
3759 SSMR3PutU32 (pSSM, pLsiLogic->enmState);
3760 SSMR3PutU32 (pSSM, pLsiLogic->enmWhoInit);
3761 SSMR3PutBool (pSSM, pLsiLogic->fDoorbellInProgress);
3762 SSMR3PutBool (pSSM, pLsiLogic->fDiagnosticEnabled);
3763 SSMR3PutBool (pSSM, pLsiLogic->fNotificationSend);
3764 SSMR3PutBool (pSSM, pLsiLogic->fEventNotificationEnabled);
3765 SSMR3PutU32 (pSSM, pLsiLogic->uInterruptMask);
3766 SSMR3PutU32 (pSSM, pLsiLogic->uInterruptStatus);
3767 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aMessage); i++)
3768 SSMR3PutU32 (pSSM, pLsiLogic->aMessage[i]);
3769 SSMR3PutU32 (pSSM, pLsiLogic->iMessage);
3770 SSMR3PutU32 (pSSM, pLsiLogic->cMessage);
3771 SSMR3PutMem (pSSM, &pLsiLogic->ReplyBuffer, sizeof(pLsiLogic->ReplyBuffer));
3772 SSMR3PutU32 (pSSM, pLsiLogic->uNextReplyEntryRead);
3773 SSMR3PutU32 (pSSM, pLsiLogic->cReplySize);
3774 SSMR3PutU16 (pSSM, pLsiLogic->u16IOCFaultCode);
3775 SSMR3PutU32 (pSSM, pLsiLogic->u32HostMFAHighAddr);
3776 SSMR3PutU32 (pSSM, pLsiLogic->u32SenseBufferHighAddr);
3777 SSMR3PutU8 (pSSM, pLsiLogic->cMaxDevices);
3778 SSMR3PutU8 (pSSM, pLsiLogic->cMaxBuses);
3779 SSMR3PutU16 (pSSM, pLsiLogic->cbReplyFrame);
3780 SSMR3PutU32 (pSSM, pLsiLogic->iDiagnosticAccess);
3781 SSMR3PutU32 (pSSM, pLsiLogic->cReplyQueueEntries);
3782 SSMR3PutU32 (pSSM, pLsiLogic->cRequestQueueEntries);
3783 SSMR3PutU32 (pSSM, pLsiLogic->uReplyFreeQueueNextEntryFreeWrite);
3784 SSMR3PutU32 (pSSM, pLsiLogic->uReplyFreeQueueNextAddressRead);
3785 SSMR3PutU32 (pSSM, pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
3786 SSMR3PutU32 (pSSM, pLsiLogic->uReplyPostQueueNextAddressRead);
3787 SSMR3PutU32 (pSSM, pLsiLogic->uRequestQueueNextEntryFreeWrite);
3788 SSMR3PutU32 (pSSM, pLsiLogic->uRequestQueueNextAddressRead);
3789
3790 for (unsigned i = 0; i < pLsiLogic->cReplyQueueEntries; i++)
3791 SSMR3PutU32(pSSM, pLsiLogic->pReplyFreeQueueBaseR3[i]);
3792 for (unsigned i = 0; i < pLsiLogic->cReplyQueueEntries; i++)
3793 SSMR3PutU32(pSSM, pLsiLogic->pReplyPostQueueBaseR3[i]);
3794 for (unsigned i = 0; i < pLsiLogic->cRequestQueueEntries; i++)
3795 SSMR3PutU32(pSSM, pLsiLogic->pRequestQueueBaseR3[i]);
3796
3797 SSMR3PutU16 (pSSM, pLsiLogic->u16NextHandle);
3798
3799 PMptConfigurationPagesSupported pPages = pLsiLogic->pConfigurationPages;
3800
3801 SSMR3PutMem (pSSM, &pPages->ManufacturingPage0, sizeof(MptConfigurationPageManufacturing0));
3802 SSMR3PutMem (pSSM, &pPages->ManufacturingPage1, sizeof(MptConfigurationPageManufacturing1));
3803 SSMR3PutMem (pSSM, &pPages->ManufacturingPage2, sizeof(MptConfigurationPageManufacturing2));
3804 SSMR3PutMem (pSSM, &pPages->ManufacturingPage3, sizeof(MptConfigurationPageManufacturing3));
3805 SSMR3PutMem (pSSM, &pPages->ManufacturingPage4, sizeof(MptConfigurationPageManufacturing4));
3806 SSMR3PutMem (pSSM, &pPages->ManufacturingPage5, sizeof(MptConfigurationPageManufacturing5));
3807 SSMR3PutMem (pSSM, &pPages->ManufacturingPage6, sizeof(MptConfigurationPageManufacturing6));
3808 SSMR3PutMem (pSSM, &pPages->ManufacturingPage8, sizeof(MptConfigurationPageManufacturing8));
3809 SSMR3PutMem (pSSM, &pPages->ManufacturingPage9, sizeof(MptConfigurationPageManufacturing9));
3810 SSMR3PutMem (pSSM, &pPages->ManufacturingPage10, sizeof(MptConfigurationPageManufacturing10));
3811 SSMR3PutMem (pSSM, &pPages->IOUnitPage0, sizeof(MptConfigurationPageIOUnit0));
3812 SSMR3PutMem (pSSM, &pPages->IOUnitPage1, sizeof(MptConfigurationPageIOUnit1));
3813 SSMR3PutMem (pSSM, &pPages->IOUnitPage2, sizeof(MptConfigurationPageIOUnit2));
3814 SSMR3PutMem (pSSM, &pPages->IOUnitPage3, sizeof(MptConfigurationPageIOUnit3));
3815 SSMR3PutMem (pSSM, &pPages->IOUnitPage4, sizeof(MptConfigurationPageIOUnit4));
3816 SSMR3PutMem (pSSM, &pPages->IOCPage0, sizeof(MptConfigurationPageIOC0));
3817 SSMR3PutMem (pSSM, &pPages->IOCPage1, sizeof(MptConfigurationPageIOC1));
3818 SSMR3PutMem (pSSM, &pPages->IOCPage2, sizeof(MptConfigurationPageIOC2));
3819 SSMR3PutMem (pSSM, &pPages->IOCPage3, sizeof(MptConfigurationPageIOC3));
3820 SSMR3PutMem (pSSM, &pPages->IOCPage4, sizeof(MptConfigurationPageIOC4));
3821 SSMR3PutMem (pSSM, &pPages->IOCPage6, sizeof(MptConfigurationPageIOC6));
3822 SSMR3PutMem (pSSM, &pPages->BIOSPage1, sizeof(MptConfigurationPageBIOS1));
3823 SSMR3PutMem (pSSM, &pPages->BIOSPage2, sizeof(MptConfigurationPageBIOS2));
3824 SSMR3PutMem (pSSM, &pPages->BIOSPage4, sizeof(MptConfigurationPageBIOS4));
3825
3826 /* Device dependent pages */
3827 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
3828 {
3829 PMptConfigurationPagesSpi pSpiPages = &pPages->u.SpiPages;
3830
3831 SSMR3PutMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage0, sizeof(MptConfigurationPageSCSISPIPort0));
3832 SSMR3PutMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage1, sizeof(MptConfigurationPageSCSISPIPort1));
3833 SSMR3PutMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage2, sizeof(MptConfigurationPageSCSISPIPort2));
3834
3835 for (unsigned i = 0; i < RT_ELEMENTS(pSpiPages->aBuses[0].aDevicePages); i++)
3836 {
3837 SSMR3PutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage0, sizeof(MptConfigurationPageSCSISPIDevice0));
3838 SSMR3PutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage1, sizeof(MptConfigurationPageSCSISPIDevice1));
3839 SSMR3PutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage2, sizeof(MptConfigurationPageSCSISPIDevice2));
3840 SSMR3PutMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage3, sizeof(MptConfigurationPageSCSISPIDevice3));
3841 }
3842 }
3843 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
3844 {
3845 PMptConfigurationPagesSas pSasPages = &pPages->u.SasPages;
3846
3847 SSMR3PutU32(pSSM, pSasPages->cbManufacturingPage7);
3848 SSMR3PutU32(pSSM, pSasPages->cbSASIOUnitPage0);
3849 SSMR3PutU32(pSSM, pSasPages->cbSASIOUnitPage1);
3850
3851 SSMR3PutMem(pSSM, pSasPages->pManufacturingPage7, pSasPages->cbManufacturingPage7);
3852 SSMR3PutMem(pSSM, pSasPages->pSASIOUnitPage0, pSasPages->cbSASIOUnitPage0);
3853 SSMR3PutMem(pSSM, pSasPages->pSASIOUnitPage1, pSasPages->cbSASIOUnitPage1);
3854
3855 SSMR3PutMem(pSSM, &pSasPages->SASIOUnitPage2, sizeof(MptConfigurationPageSASIOUnit2));
3856 SSMR3PutMem(pSSM, &pSasPages->SASIOUnitPage3, sizeof(MptConfigurationPageSASIOUnit3));
3857
3858 SSMR3PutU32(pSSM, pSasPages->cPHYs);
3859 for (unsigned i = 0; i < pSasPages->cPHYs; i++)
3860 {
3861 SSMR3PutMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage0, sizeof(MptConfigurationPageSASPHY0));
3862 SSMR3PutMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage1, sizeof(MptConfigurationPageSASPHY1));
3863 }
3864
3865 /* The number of devices first. */
3866 SSMR3PutU32(pSSM, pSasPages->cDevices);
3867
3868 PMptSASDevice pCurr = pSasPages->pSASDeviceHead;
3869
3870 while (pCurr)
3871 {
3872 SSMR3PutMem(pSSM, &pCurr->SASDevicePage0, sizeof(MptConfigurationPageSASDevice0));
3873 SSMR3PutMem(pSSM, &pCurr->SASDevicePage1, sizeof(MptConfigurationPageSASDevice1));
3874 SSMR3PutMem(pSSM, &pCurr->SASDevicePage2, sizeof(MptConfigurationPageSASDevice2));
3875
3876 pCurr = pCurr->pNext;
3877 }
3878 }
3879 else
3880 AssertMsgFailed(("Invalid controller type %d\n", pLsiLogic->enmCtrlType));
3881
3882 /* Now the data for the BIOS interface. */
3883 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.regIdentify);
3884 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.uTargetDevice);
3885 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.uTxDir);
3886 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.cbCDB);
3887 SSMR3PutMem (pSSM, pLsiLogic->VBoxSCSI.aCDB, sizeof(pLsiLogic->VBoxSCSI.aCDB));
3888 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.iCDB);
3889 SSMR3PutU32 (pSSM, pLsiLogic->VBoxSCSI.cbBuf);
3890 SSMR3PutU32 (pSSM, pLsiLogic->VBoxSCSI.iBuf);
3891 SSMR3PutBool (pSSM, pLsiLogic->VBoxSCSI.fBusy);
3892 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.enmState);
3893 if (pLsiLogic->VBoxSCSI.cbCDB)
3894 SSMR3PutMem(pSSM, pLsiLogic->VBoxSCSI.pBuf, pLsiLogic->VBoxSCSI.cbBuf);
3895
3896 return SSMR3PutU32(pSSM, ~0);
3897}
3898
3899static DECLCALLBACK(int) lsilogicLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
3900{
3901 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3902 int rc;
3903
3904 if ( uVersion != LSILOGIC_SAVED_STATE_VERSION
3905 && uVersion != LSILOGIC_SAVED_STATE_VERSION_PRE_SAS
3906 && uVersion != LSILOGIC_SAVED_STATE_VERSION_VBOX_30)
3907 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
3908
3909 /* device config */
3910 if (uVersion > LSILOGIC_SAVED_STATE_VERSION_PRE_SAS)
3911 {
3912 LSILOGICCTRLTYPE enmCtrlType;
3913 uint32_t cDeviceStates, cPorts;
3914
3915 rc = SSMR3GetU32(pSSM, (uint32_t *)&enmCtrlType);
3916 AssertRCReturn(rc, rc);
3917 rc = SSMR3GetU32(pSSM, &cDeviceStates);
3918 AssertRCReturn(rc, rc);
3919 rc = SSMR3GetU32(pSSM, &cPorts);
3920 AssertRCReturn(rc, rc);
3921
3922 if (enmCtrlType != pLsiLogic->enmCtrlType)
3923 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Target config mismatch (Controller type): config=%d state=%d"),
3924 pLsiLogic->enmCtrlType, enmCtrlType);
3925 if (cDeviceStates != pLsiLogic->cDeviceStates)
3926 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Target config mismatch (Device states): config=%u state=%u"),
3927 pLsiLogic->cDeviceStates, cDeviceStates);
3928 if (cPorts != pLsiLogic->cPorts)
3929 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Target config mismatch (Ports): config=%u state=%u"),
3930 pLsiLogic->cPorts, cPorts);
3931 }
3932 if (uVersion > LSILOGIC_SAVED_STATE_VERSION_VBOX_30)
3933 {
3934 for (unsigned i = 0; i < pLsiLogic->cDeviceStates; i++)
3935 {
3936 bool fPresent;
3937 rc = SSMR3GetBool(pSSM, &fPresent);
3938 AssertRCReturn(rc, rc);
3939 if (fPresent != (pLsiLogic->paDeviceStates[i].pDrvBase != NULL))
3940 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Target %u config mismatch: config=%RTbool state=%RTbool"),
3941 i, pLsiLogic->paDeviceStates[i].pDrvBase != NULL, fPresent);
3942 }
3943 }
3944 if (uPass != SSM_PASS_FINAL)
3945 return VINF_SUCCESS;
3946
3947 /* Every device first. */
3948 for (unsigned i = 0; i < pLsiLogic->cDeviceStates; i++)
3949 {
3950 PLSILOGICDEVICE pDevice = &pLsiLogic->paDeviceStates[i];
3951
3952 AssertMsg(!pDevice->cOutstandingRequests,
3953 ("There are still outstanding requests on this device\n"));
3954 SSMR3GetU32(pSSM, (uint32_t *)&pDevice->cOutstandingRequests);
3955 }
3956 /* Now the main device state. */
3957 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->enmState);
3958 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->enmWhoInit);
3959 SSMR3GetBool (pSSM, &pLsiLogic->fDoorbellInProgress);
3960 SSMR3GetBool (pSSM, &pLsiLogic->fDiagnosticEnabled);
3961 SSMR3GetBool (pSSM, &pLsiLogic->fNotificationSend);
3962 SSMR3GetBool (pSSM, &pLsiLogic->fEventNotificationEnabled);
3963 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uInterruptMask);
3964 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uInterruptStatus);
3965 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aMessage); i++)
3966 SSMR3GetU32 (pSSM, &pLsiLogic->aMessage[i]);
3967 SSMR3GetU32 (pSSM, &pLsiLogic->iMessage);
3968 SSMR3GetU32 (pSSM, &pLsiLogic->cMessage);
3969 SSMR3GetMem (pSSM, &pLsiLogic->ReplyBuffer, sizeof(pLsiLogic->ReplyBuffer));
3970 SSMR3GetU32 (pSSM, &pLsiLogic->uNextReplyEntryRead);
3971 SSMR3GetU32 (pSSM, &pLsiLogic->cReplySize);
3972 SSMR3GetU16 (pSSM, &pLsiLogic->u16IOCFaultCode);
3973 SSMR3GetU32 (pSSM, &pLsiLogic->u32HostMFAHighAddr);
3974 SSMR3GetU32 (pSSM, &pLsiLogic->u32SenseBufferHighAddr);
3975 SSMR3GetU8 (pSSM, &pLsiLogic->cMaxDevices);
3976 SSMR3GetU8 (pSSM, &pLsiLogic->cMaxBuses);
3977 SSMR3GetU16 (pSSM, &pLsiLogic->cbReplyFrame);
3978 SSMR3GetU32 (pSSM, &pLsiLogic->iDiagnosticAccess);
3979 SSMR3GetU32 (pSSM, &pLsiLogic->cReplyQueueEntries);
3980 SSMR3GetU32 (pSSM, &pLsiLogic->cRequestQueueEntries);
3981 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyFreeQueueNextEntryFreeWrite);
3982 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyFreeQueueNextAddressRead);
3983 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
3984 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyPostQueueNextAddressRead);
3985 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uRequestQueueNextEntryFreeWrite);
3986 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uRequestQueueNextAddressRead);
3987
3988 PMptConfigurationPagesSupported pPages = pLsiLogic->pConfigurationPages;
3989
3990 if (uVersion <= LSILOGIC_SAVED_STATE_VERSION_PRE_SAS)
3991 {
3992 PMptConfigurationPagesSpi pSpiPages = &pPages->u.SpiPages;
3993 MptConfigurationPagesSupported_SSM_V2 ConfigPagesV2;
3994
3995 if (pLsiLogic->enmCtrlType != LSILOGICCTRLTYPE_SCSI_SPI)
3996 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch: Expected SPI SCSI controller"));
3997
3998 SSMR3GetMem(pSSM, &ConfigPagesV2,
3999 sizeof(MptConfigurationPagesSupported_SSM_V2));
4000
4001 pPages->ManufacturingPage0 = ConfigPagesV2.ManufacturingPage0;
4002 pPages->ManufacturingPage1 = ConfigPagesV2.ManufacturingPage1;
4003 pPages->ManufacturingPage2 = ConfigPagesV2.ManufacturingPage2;
4004 pPages->ManufacturingPage3 = ConfigPagesV2.ManufacturingPage3;
4005 pPages->ManufacturingPage4 = ConfigPagesV2.ManufacturingPage4;
4006 pPages->IOUnitPage0 = ConfigPagesV2.IOUnitPage0;
4007 pPages->IOUnitPage1 = ConfigPagesV2.IOUnitPage1;
4008 pPages->IOUnitPage2 = ConfigPagesV2.IOUnitPage2;
4009 pPages->IOUnitPage3 = ConfigPagesV2.IOUnitPage3;
4010 pPages->IOCPage0 = ConfigPagesV2.IOCPage0;
4011 pPages->IOCPage1 = ConfigPagesV2.IOCPage1;
4012 pPages->IOCPage2 = ConfigPagesV2.IOCPage2;
4013 pPages->IOCPage3 = ConfigPagesV2.IOCPage3;
4014 pPages->IOCPage4 = ConfigPagesV2.IOCPage4;
4015 pPages->IOCPage6 = ConfigPagesV2.IOCPage6;
4016
4017 pSpiPages->aPortPages[0].SCSISPIPortPage0 = ConfigPagesV2.aPortPages[0].SCSISPIPortPage0;
4018 pSpiPages->aPortPages[0].SCSISPIPortPage1 = ConfigPagesV2.aPortPages[0].SCSISPIPortPage1;
4019 pSpiPages->aPortPages[0].SCSISPIPortPage2 = ConfigPagesV2.aPortPages[0].SCSISPIPortPage2;
4020
4021 for (unsigned i = 0; i < RT_ELEMENTS(pPages->u.SpiPages.aBuses[0].aDevicePages); i++)
4022 {
4023 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage0 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage0;
4024 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage1 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage1;
4025 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage2 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage2;
4026 pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage3 = ConfigPagesV2.aBuses[0].aDevicePages[i].SCSISPIDevicePage3;
4027 }
4028 }
4029 else
4030 {
4031 /* Queue content */
4032 for (unsigned i = 0; i < pLsiLogic->cReplyQueueEntries; i++)
4033 SSMR3GetU32(pSSM, (uint32_t *)&pLsiLogic->pReplyFreeQueueBaseR3[i]);
4034 for (unsigned i = 0; i < pLsiLogic->cReplyQueueEntries; i++)
4035 SSMR3GetU32(pSSM, (uint32_t *)&pLsiLogic->pReplyPostQueueBaseR3[i]);
4036 for (unsigned i = 0; i < pLsiLogic->cRequestQueueEntries; i++)
4037 SSMR3GetU32(pSSM, (uint32_t *)&pLsiLogic->pRequestQueueBaseR3[i]);
4038
4039 SSMR3GetU16(pSSM, &pLsiLogic->u16NextHandle);
4040
4041 /* Configuration pages */
4042 SSMR3GetMem(pSSM, &pPages->ManufacturingPage0, sizeof(MptConfigurationPageManufacturing0));
4043 SSMR3GetMem(pSSM, &pPages->ManufacturingPage1, sizeof(MptConfigurationPageManufacturing1));
4044 SSMR3GetMem(pSSM, &pPages->ManufacturingPage2, sizeof(MptConfigurationPageManufacturing2));
4045 SSMR3GetMem(pSSM, &pPages->ManufacturingPage3, sizeof(MptConfigurationPageManufacturing3));
4046 SSMR3GetMem(pSSM, &pPages->ManufacturingPage4, sizeof(MptConfigurationPageManufacturing4));
4047 SSMR3GetMem(pSSM, &pPages->ManufacturingPage5, sizeof(MptConfigurationPageManufacturing5));
4048 SSMR3GetMem(pSSM, &pPages->ManufacturingPage6, sizeof(MptConfigurationPageManufacturing6));
4049 SSMR3GetMem(pSSM, &pPages->ManufacturingPage8, sizeof(MptConfigurationPageManufacturing8));
4050 SSMR3GetMem(pSSM, &pPages->ManufacturingPage9, sizeof(MptConfigurationPageManufacturing9));
4051 SSMR3GetMem(pSSM, &pPages->ManufacturingPage10, sizeof(MptConfigurationPageManufacturing10));
4052 SSMR3GetMem(pSSM, &pPages->IOUnitPage0, sizeof(MptConfigurationPageIOUnit0));
4053 SSMR3GetMem(pSSM, &pPages->IOUnitPage1, sizeof(MptConfigurationPageIOUnit1));
4054 SSMR3GetMem(pSSM, &pPages->IOUnitPage2, sizeof(MptConfigurationPageIOUnit2));
4055 SSMR3GetMem(pSSM, &pPages->IOUnitPage3, sizeof(MptConfigurationPageIOUnit3));
4056 SSMR3GetMem(pSSM, &pPages->IOUnitPage4, sizeof(MptConfigurationPageIOUnit4));
4057 SSMR3GetMem(pSSM, &pPages->IOCPage0, sizeof(MptConfigurationPageIOC0));
4058 SSMR3GetMem(pSSM, &pPages->IOCPage1, sizeof(MptConfigurationPageIOC1));
4059 SSMR3GetMem(pSSM, &pPages->IOCPage2, sizeof(MptConfigurationPageIOC2));
4060 SSMR3GetMem(pSSM, &pPages->IOCPage3, sizeof(MptConfigurationPageIOC3));
4061 SSMR3GetMem(pSSM, &pPages->IOCPage4, sizeof(MptConfigurationPageIOC4));
4062 SSMR3GetMem(pSSM, &pPages->IOCPage6, sizeof(MptConfigurationPageIOC6));
4063 SSMR3GetMem(pSSM, &pPages->BIOSPage1, sizeof(MptConfigurationPageBIOS1));
4064 SSMR3GetMem(pSSM, &pPages->BIOSPage2, sizeof(MptConfigurationPageBIOS2));
4065 SSMR3GetMem(pSSM, &pPages->BIOSPage4, sizeof(MptConfigurationPageBIOS4));
4066
4067 /* Device dependent pages */
4068 if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
4069 {
4070 PMptConfigurationPagesSpi pSpiPages = &pPages->u.SpiPages;
4071
4072 SSMR3GetMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage0, sizeof(MptConfigurationPageSCSISPIPort0));
4073 SSMR3GetMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage1, sizeof(MptConfigurationPageSCSISPIPort1));
4074 SSMR3GetMem(pSSM, &pSpiPages->aPortPages[0].SCSISPIPortPage2, sizeof(MptConfigurationPageSCSISPIPort2));
4075
4076 for (unsigned i = 0; i < RT_ELEMENTS(pSpiPages->aBuses[0].aDevicePages); i++)
4077 {
4078 SSMR3GetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage0, sizeof(MptConfigurationPageSCSISPIDevice0));
4079 SSMR3GetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage1, sizeof(MptConfigurationPageSCSISPIDevice1));
4080 SSMR3GetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage2, sizeof(MptConfigurationPageSCSISPIDevice2));
4081 SSMR3GetMem(pSSM, &pSpiPages->aBuses[0].aDevicePages[i].SCSISPIDevicePage3, sizeof(MptConfigurationPageSCSISPIDevice3));
4082 }
4083 }
4084 else if (pLsiLogic->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
4085 {
4086 uint32_t cbPage0, cbPage1, cPHYs, cbManufacturingPage7;
4087 PMptConfigurationPagesSas pSasPages = &pPages->u.SasPages;
4088
4089 SSMR3GetU32(pSSM, &cbManufacturingPage7);
4090 SSMR3GetU32(pSSM, &cbPage0);
4091 SSMR3GetU32(pSSM, &cbPage1);
4092
4093 if ( (cbPage0 != pSasPages->cbSASIOUnitPage0)
4094 || (cbPage1 != pSasPages->cbSASIOUnitPage1)
4095 || (cbManufacturingPage7 != pSasPages->cbManufacturingPage7))
4096 return VERR_SSM_LOAD_CONFIG_MISMATCH;
4097
4098 AssertPtr(pSasPages->pManufacturingPage7);
4099 AssertPtr(pSasPages->pSASIOUnitPage0);
4100 AssertPtr(pSasPages->pSASIOUnitPage1);
4101
4102 SSMR3GetMem(pSSM, pSasPages->pManufacturingPage7, pSasPages->cbManufacturingPage7);
4103 SSMR3GetMem(pSSM, pSasPages->pSASIOUnitPage0, pSasPages->cbSASIOUnitPage0);
4104 SSMR3GetMem(pSSM, pSasPages->pSASIOUnitPage1, pSasPages->cbSASIOUnitPage1);
4105
4106 SSMR3GetMem(pSSM, &pSasPages->SASIOUnitPage2, sizeof(MptConfigurationPageSASIOUnit2));
4107 SSMR3GetMem(pSSM, &pSasPages->SASIOUnitPage3, sizeof(MptConfigurationPageSASIOUnit3));
4108
4109 SSMR3GetU32(pSSM, &cPHYs);
4110 if (cPHYs != pSasPages->cPHYs)
4111 return VERR_SSM_LOAD_CONFIG_MISMATCH;
4112
4113 AssertPtr(pSasPages->paPHYs);
4114 for (unsigned i = 0; i < pSasPages->cPHYs; i++)
4115 {
4116 SSMR3GetMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage0, sizeof(MptConfigurationPageSASPHY0));
4117 SSMR3GetMem(pSSM, &pSasPages->paPHYs[i].SASPHYPage1, sizeof(MptConfigurationPageSASPHY1));
4118 }
4119
4120 /* The number of devices first. */
4121 SSMR3GetU32(pSSM, &pSasPages->cDevices);
4122
4123 PMptSASDevice pCurr = pSasPages->pSASDeviceHead;
4124
4125 for (unsigned i = 0; i < pSasPages->cDevices; i++)
4126 {
4127 SSMR3GetMem(pSSM, &pCurr->SASDevicePage0, sizeof(MptConfigurationPageSASDevice0));
4128 SSMR3GetMem(pSSM, &pCurr->SASDevicePage1, sizeof(MptConfigurationPageSASDevice1));
4129 SSMR3GetMem(pSSM, &pCurr->SASDevicePage2, sizeof(MptConfigurationPageSASDevice2));
4130
4131 pCurr = pCurr->pNext;
4132 }
4133
4134 Assert(!pCurr);
4135 }
4136 else
4137 AssertMsgFailed(("Invalid controller type %d\n", pLsiLogic->enmCtrlType));
4138 }
4139
4140 /* Now the data for the BIOS interface. */
4141 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.regIdentify);
4142 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.uTargetDevice);
4143 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.uTxDir);
4144 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.cbCDB);
4145 SSMR3GetMem (pSSM, pLsiLogic->VBoxSCSI.aCDB, sizeof(pLsiLogic->VBoxSCSI.aCDB));
4146 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.iCDB);
4147 SSMR3GetU32 (pSSM, &pLsiLogic->VBoxSCSI.cbBuf);
4148 SSMR3GetU32 (pSSM, &pLsiLogic->VBoxSCSI.iBuf);
4149 SSMR3GetBool(pSSM, (bool *)&pLsiLogic->VBoxSCSI.fBusy);
4150 SSMR3GetU8 (pSSM, (uint8_t *)&pLsiLogic->VBoxSCSI.enmState);
4151 if (pLsiLogic->VBoxSCSI.cbCDB)
4152 {
4153 pLsiLogic->VBoxSCSI.pBuf = (uint8_t *)RTMemAllocZ(pLsiLogic->VBoxSCSI.cbCDB);
4154 if (!pLsiLogic->VBoxSCSI.pBuf)
4155 {
4156 LogRel(("LsiLogic: Out of memory during restore.\n"));
4157 return PDMDEV_SET_ERROR(pDevIns, VERR_NO_MEMORY,
4158 N_("LsiLogic: Out of memory during restore\n"));
4159 }
4160 SSMR3GetMem(pSSM, pLsiLogic->VBoxSCSI.pBuf, pLsiLogic->VBoxSCSI.cbBuf);
4161 }
4162
4163 uint32_t u32;
4164 rc = SSMR3GetU32(pSSM, &u32);
4165 if (RT_FAILURE(rc))
4166 return rc;
4167 AssertMsgReturn(u32 == ~0U, ("%#x\n", u32), VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
4168
4169 return VINF_SUCCESS;
4170}
4171
4172/**
4173 * Gets the pointer to the status LED of a device - called from the SCSi driver.
4174 *
4175 * @returns VBox status code.
4176 * @param pInterface Pointer to the interface structure containing the called function pointer.
4177 * @param iLUN The unit which status LED we desire. Always 0 here as the driver
4178 * doesn't know about other LUN's.
4179 * @param ppLed Where to store the LED pointer.
4180 */
4181static DECLCALLBACK(int) lsilogicDeviceQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
4182{
4183 PLSILOGICDEVICE pDevice = PDMILEDPORTS_2_PLSILOGICDEVICE(pInterface);
4184 if (iLUN == 0)
4185 {
4186 *ppLed = &pDevice->Led;
4187 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
4188 return VINF_SUCCESS;
4189 }
4190 return VERR_PDM_LUN_NOT_FOUND;
4191}
4192
4193
4194/**
4195 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
4196 */
4197static DECLCALLBACK(void *) lsilogicDeviceQueryInterface(PPDMIBASE pInterface, const char *pszIID)
4198{
4199 PLSILOGICDEVICE pDevice = PDMIBASE_2_PLSILOGICDEVICE(pInterface);
4200
4201 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDevice->IBase);
4202 PDMIBASE_RETURN_INTERFACE(pszIID, PDMISCSIPORT, &pDevice->ISCSIPort);
4203 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pDevice->ILed);
4204 return NULL;
4205}
4206
4207/**
4208 * Gets the pointer to the status LED of a unit.
4209 *
4210 * @returns VBox status code.
4211 * @param pInterface Pointer to the interface structure containing the called function pointer.
4212 * @param iLUN The unit which status LED we desire.
4213 * @param ppLed Where to store the LED pointer.
4214 */
4215static DECLCALLBACK(int) lsilogicStatusQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
4216{
4217 PLSILOGICSCSI pLsiLogic = PDMILEDPORTS_2_PLSILOGICSCSI(pInterface);
4218 if (iLUN < pLsiLogic->cDeviceStates)
4219 {
4220 *ppLed = &pLsiLogic->paDeviceStates[iLUN].Led;
4221 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
4222 return VINF_SUCCESS;
4223 }
4224 return VERR_PDM_LUN_NOT_FOUND;
4225}
4226
4227/**
4228 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
4229 */
4230static DECLCALLBACK(void *) lsilogicStatusQueryInterface(PPDMIBASE pInterface, const char *pszIID)
4231{
4232 PLSILOGICSCSI pThis = PDMIBASE_2_PLSILOGICSCSI(pInterface);
4233 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
4234 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->ILeds);
4235 return NULL;
4236}
4237
4238/**
4239 * Detach notification.
4240 *
4241 * One harddisk at one port has been unplugged.
4242 * The VM is suspended at this point.
4243 *
4244 * @param pDevIns The device instance.
4245 * @param iLUN The logical unit which is being detached.
4246 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
4247 */
4248static DECLCALLBACK(void) lsilogicDetach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
4249{
4250 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4251 PLSILOGICDEVICE pDevice = &pThis->paDeviceStates[iLUN];
4252
4253 if (iLUN >= pThis->cDeviceStates)
4254 return;
4255
4256 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
4257 ("LsiLogic: Device does not support hotplugging\n"));
4258
4259 Log(("%s:\n", __FUNCTION__));
4260
4261 /*
4262 * Zero some important members.
4263 */
4264 pDevice->pDrvBase = NULL;
4265 pDevice->pDrvSCSIConnector = NULL;
4266}
4267
4268/**
4269 * Attach command.
4270 *
4271 * This is called when we change block driver.
4272 *
4273 * @returns VBox status code.
4274 * @param pDevIns The device instance.
4275 * @param iLUN The logical unit which is being detached.
4276 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
4277 */
4278static DECLCALLBACK(int) lsilogicAttach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
4279{
4280 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4281 PLSILOGICDEVICE pDevice = &pThis->paDeviceStates[iLUN];
4282 int rc;
4283
4284 if (iLUN >= pThis->cDeviceStates)
4285 return VERR_PDM_LUN_NOT_FOUND;
4286
4287 AssertMsgReturn(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
4288 ("LsiLogic: Device does not support hotplugging\n"),
4289 VERR_INVALID_PARAMETER);
4290
4291 /* the usual paranoia */
4292 AssertRelease(!pDevice->pDrvBase);
4293 AssertRelease(!pDevice->pDrvSCSIConnector);
4294 Assert(pDevice->iLUN == iLUN);
4295
4296 /*
4297 * Try attach the block device and get the interfaces,
4298 * required as well as optional.
4299 */
4300 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, NULL);
4301 if (RT_SUCCESS(rc))
4302 {
4303 /* Get SCSI connector interface. */
4304 pDevice->pDrvSCSIConnector = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMISCSICONNECTOR);
4305 AssertMsgReturn(pDevice->pDrvSCSIConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
4306 }
4307 else
4308 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", pDevice->iLUN, rc));
4309
4310 if (RT_FAILURE(rc))
4311 {
4312 pDevice->pDrvBase = NULL;
4313 pDevice->pDrvSCSIConnector = NULL;
4314 }
4315 return rc;
4316}
4317
4318/**
4319 * @copydoc FNPDMDEVRESET
4320 */
4321static DECLCALLBACK(void) lsilogicReset(PPDMDEVINS pDevIns)
4322{
4323 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4324 int rc;
4325
4326 rc = lsilogicHardReset(pLsiLogic);
4327 AssertRC(rc);
4328
4329 vboxscsiInitialize(&pLsiLogic->VBoxSCSI);
4330}
4331
4332/**
4333 * @copydoc FNPDMDEVRELOCATE
4334 */
4335static DECLCALLBACK(void) lsilogicRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
4336{
4337 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4338
4339 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
4340 pThis->pNotificationQueueRC = PDMQueueRCPtr(pThis->pNotificationQueueR3);
4341
4342 /* Relocate queues. */
4343 pThis->pReplyFreeQueueBaseRC += offDelta;
4344 pThis->pReplyPostQueueBaseRC += offDelta;
4345 pThis->pRequestQueueBaseRC += offDelta;
4346}
4347
4348/**
4349 * @copydoc FNPDMDEVDESTRUCT
4350 */
4351static DECLCALLBACK(int) lsilogicDestruct(PPDMDEVINS pDevIns)
4352{
4353 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4354 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
4355
4356 PDMR3CritSectDelete(&pThis->ReplyFreeQueueCritSect);
4357 PDMR3CritSectDelete(&pThis->ReplyPostQueueCritSect);
4358
4359 if (pThis->paDeviceStates)
4360 RTMemFree(pThis->paDeviceStates);
4361
4362 /* Destroy task cache. */
4363 int rc = VINF_SUCCESS;
4364 if (pThis->hTaskCache != NIL_RTMEMCACHE)
4365 rc = RTMemCacheDestroy(pThis->hTaskCache);
4366
4367 lsilogicConfigurationPagesFree(pThis);
4368
4369 return rc;
4370}
4371
4372/**
4373 * @copydoc FNPDMDEVCONSTRUCT
4374 */
4375static DECLCALLBACK(int) lsilogicConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
4376{
4377 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4378 int rc = VINF_SUCCESS;
4379 char *pszCtrlType = NULL;
4380 PVM pVM = PDMDevHlpGetVM(pDevIns);
4381 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
4382
4383 /*
4384 * Validate and read configuration.
4385 */
4386 rc = CFGMR3AreValuesValid(pCfg, "GCEnabled\0"
4387 "R0Enabled\0"
4388 "ReplyQueueDepth\0"
4389 "RequestQueueDepth\0"
4390 "ControllerType\0"
4391 "NumPorts\0");
4392 if (RT_FAILURE(rc))
4393 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
4394 N_("LsiLogic configuration error: unknown option specified"));
4395 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &pThis->fGCEnabled, true);
4396 if (RT_FAILURE(rc))
4397 return PDMDEV_SET_ERROR(pDevIns, rc,
4398 N_("LsiLogic configuration error: failed to read GCEnabled as boolean"));
4399 Log(("%s: fGCEnabled=%d\n", __FUNCTION__, pThis->fGCEnabled));
4400
4401 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &pThis->fR0Enabled, true);
4402 if (RT_FAILURE(rc))
4403 return PDMDEV_SET_ERROR(pDevIns, rc,
4404 N_("LsiLogic configuration error: failed to read R0Enabled as boolean"));
4405 Log(("%s: fR0Enabled=%d\n", __FUNCTION__, pThis->fR0Enabled));
4406
4407 rc = CFGMR3QueryU32Def(pCfg, "ReplyQueueDepth",
4408 &pThis->cReplyQueueEntries,
4409 LSILOGICSCSI_REPLY_QUEUE_DEPTH_DEFAULT);
4410 if (RT_FAILURE(rc))
4411 return PDMDEV_SET_ERROR(pDevIns, rc,
4412 N_("LsiLogic configuration error: failed to read ReplyQueue as integer"));
4413 Log(("%s: ReplyQueueDepth=%u\n", __FUNCTION__, pThis->cReplyQueueEntries));
4414
4415 rc = CFGMR3QueryU32Def(pCfg, "RequestQueueDepth",
4416 &pThis->cRequestQueueEntries,
4417 LSILOGICSCSI_REQUEST_QUEUE_DEPTH_DEFAULT);
4418 if (RT_FAILURE(rc))
4419 return PDMDEV_SET_ERROR(pDevIns, rc,
4420 N_("LsiLogic configuration error: failed to read RequestQueue as integer"));
4421 Log(("%s: RequestQueueDepth=%u\n", __FUNCTION__, pThis->cRequestQueueEntries));
4422
4423 rc = CFGMR3QueryStringAllocDef(pCfg, "ControllerType",
4424 &pszCtrlType, LSILOGICSCSI_PCI_SPI_CTRLNAME);
4425 if (RT_FAILURE(rc))
4426 return PDMDEV_SET_ERROR(pDevIns, rc,
4427 N_("LsiLogic configuration error: failed to read ControllerType as string"));
4428 Log(("%s: ControllerType=%s\n", __FUNCTION__, pszCtrlType));
4429
4430 rc = lsilogicGetCtrlTypeFromString(pThis, pszCtrlType);
4431 MMR3HeapFree(pszCtrlType);
4432
4433 if (RT_FAILURE(rc))
4434 return PDMDEV_SET_ERROR(pDevIns, rc,
4435 N_("LsiLogic configuration error: failed to determine controller type from string"));
4436
4437 rc = CFGMR3QueryU8(pCfg, "NumPorts",
4438 &pThis->cPorts);
4439 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
4440 {
4441 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
4442 pThis->cPorts = LSILOGICSCSI_PCI_SPI_PORTS_MAX;
4443 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
4444 pThis->cPorts = LSILOGICSCSI_PCI_SAS_PORTS_DEFAULT;
4445 else
4446 AssertMsgFailed(("Invalid controller type: %d\n", pThis->enmCtrlType));
4447 }
4448 else if (RT_FAILURE(rc))
4449 return PDMDEV_SET_ERROR(pDevIns, rc,
4450 N_("LsiLogic configuration error: failed to read NumPorts as integer"));
4451
4452 /* Init static parts. */
4453 PCIDevSetVendorId(&pThis->PciDev, LSILOGICSCSI_PCI_VENDOR_ID); /* LsiLogic */
4454
4455 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
4456 {
4457 PCIDevSetDeviceId (&pThis->PciDev, LSILOGICSCSI_PCI_SPI_DEVICE_ID); /* LSI53C1030 */
4458 PCIDevSetSubSystemVendorId(&pThis->PciDev, LSILOGICSCSI_PCI_SPI_SUBSYSTEM_VENDOR_ID);
4459 PCIDevSetSubSystemId (&pThis->PciDev, LSILOGICSCSI_PCI_SPI_SUBSYSTEM_ID);
4460 }
4461 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
4462 {
4463 PCIDevSetDeviceId (&pThis->PciDev, LSILOGICSCSI_PCI_SAS_DEVICE_ID); /* SAS1068 */
4464 PCIDevSetSubSystemVendorId(&pThis->PciDev, LSILOGICSCSI_PCI_SAS_SUBSYSTEM_VENDOR_ID);
4465 PCIDevSetSubSystemId (&pThis->PciDev, LSILOGICSCSI_PCI_SAS_SUBSYSTEM_ID);
4466 }
4467 else
4468 AssertMsgFailed(("Invalid controller type: %d\n", pThis->enmCtrlType));
4469
4470 PCIDevSetClassProg (&pThis->PciDev, 0x00); /* SCSI */
4471 PCIDevSetClassSub (&pThis->PciDev, 0x00); /* SCSI */
4472 PCIDevSetClassBase (&pThis->PciDev, 0x01); /* Mass storage */
4473 PCIDevSetInterruptPin(&pThis->PciDev, 0x01); /* Interrupt pin A */
4474
4475 pThis->pDevInsR3 = pDevIns;
4476 pThis->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
4477 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
4478 pThis->IBase.pfnQueryInterface = lsilogicStatusQueryInterface;
4479 pThis->ILeds.pfnQueryStatusLed = lsilogicStatusQueryStatusLed;
4480
4481 /*
4482 * Register the PCI device, it's I/O regions.
4483 */
4484 rc = PDMDevHlpPCIRegister (pDevIns, &pThis->PciDev);
4485 if (RT_FAILURE(rc))
4486 return rc;
4487
4488 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, LSILOGIC_PCI_SPACE_IO_SIZE, PCI_ADDRESS_SPACE_IO, lsilogicMap);
4489 if (RT_FAILURE(rc))
4490 return rc;
4491
4492 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, LSILOGIC_PCI_SPACE_MEM_SIZE, PCI_ADDRESS_SPACE_MEM, lsilogicMap);
4493 if (RT_FAILURE(rc))
4494 return rc;
4495
4496 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 2, LSILOGIC_PCI_SPACE_MEM_SIZE, PCI_ADDRESS_SPACE_MEM, lsilogicMap);
4497 if (RT_FAILURE(rc))
4498 return rc;
4499
4500 /* Intialize task queue. */
4501 rc = PDMDevHlpQueueCreate(pDevIns, sizeof(PDMQUEUEITEMCORE), 2, 0,
4502 lsilogicNotifyQueueConsumer, true, "LsiLogic-Task", &pThis->pNotificationQueueR3);
4503 if (RT_FAILURE(rc))
4504 return rc;
4505 pThis->pNotificationQueueR0 = PDMQueueR0Ptr(pThis->pNotificationQueueR3);
4506 pThis->pNotificationQueueRC = PDMQueueRCPtr(pThis->pNotificationQueueR3);
4507
4508 /*
4509 * We need one entry free in the queue.
4510 */
4511 pThis->cReplyQueueEntries++;
4512 pThis->cRequestQueueEntries++;
4513
4514 /*
4515 * Allocate memory for the queues.
4516 */
4517 uint32_t cbQueues;
4518
4519 cbQueues = 2*pThis->cReplyQueueEntries * sizeof(uint32_t);
4520 cbQueues += pThis->cRequestQueueEntries * sizeof(uint32_t);
4521 rc = MMHyperAlloc(pVM, cbQueues, 1, MM_TAG_PDM_DEVICE_USER,
4522 (void **)&pThis->pReplyFreeQueueBaseR3);
4523 if (RT_FAILURE(rc))
4524 return VERR_NO_MEMORY;
4525 pThis->pReplyFreeQueueBaseR0 = MMHyperR3ToR0(pVM, (void *)pThis->pReplyFreeQueueBaseR3);
4526 pThis->pReplyFreeQueueBaseRC = MMHyperR3ToRC(pVM, (void *)pThis->pReplyFreeQueueBaseR3);
4527
4528 pThis->pReplyPostQueueBaseR3 = pThis->pReplyFreeQueueBaseR3 + pThis->cReplyQueueEntries;
4529 pThis->pReplyPostQueueBaseR0 = MMHyperR3ToR0(pVM, (void *)pThis->pReplyPostQueueBaseR3);
4530 pThis->pReplyPostQueueBaseRC = MMHyperR3ToRC(pVM, (void *)pThis->pReplyPostQueueBaseR3);
4531
4532 pThis->pRequestQueueBaseR3 = pThis->pReplyPostQueueBaseR3 + pThis->cReplyQueueEntries;
4533 pThis->pRequestQueueBaseR0 = MMHyperR3ToR0(pVM, (void *)pThis->pRequestQueueBaseR3);
4534 pThis->pRequestQueueBaseRC = MMHyperR3ToRC(pVM, (void *)pThis->pRequestQueueBaseR3);
4535
4536 /*
4537 * Create critical sections protecting the reply post and free queues.
4538 */
4539 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyFreeQueueCritSect, RT_SRC_POS, "LsiLogicRFQ");
4540 if (RT_FAILURE(rc))
4541 return PDMDEV_SET_ERROR(pDevIns, rc,
4542 N_("LsiLogic: cannot create critical section for reply free queue"));
4543
4544 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyPostQueueCritSect, RT_SRC_POS, "LsiLogicRPQ");
4545 if (RT_FAILURE(rc))
4546 return PDMDEV_SET_ERROR(pDevIns, rc,
4547 N_("LsiLogic: cannot create critical section for reply post queue"));
4548
4549 /*
4550 * Allocate task cache.
4551 */
4552 rc = RTMemCacheCreate(&pThis->hTaskCache, sizeof(LSILOGICTASKSTATE), 0, UINT32_MAX,
4553 NULL, NULL, NULL, 0);
4554 if (RT_FAILURE(rc))
4555 return PDMDEV_SET_ERROR(pDevIns, rc,
4556 N_("Cannot create task cache"));
4557
4558 if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SPI)
4559 pThis->cDeviceStates = pThis->cPorts * LSILOGICSCSI_PCI_SPI_DEVICES_PER_BUS_MAX;
4560 else if (pThis->enmCtrlType == LSILOGICCTRLTYPE_SCSI_SAS)
4561 pThis->cDeviceStates = pThis->cPorts * LSILOGICSCSI_PCI_SAS_DEVICES_PER_PORT_MAX;
4562 else
4563 AssertMsgFailed(("Invalid controller type: %d\n", pThis->enmCtrlType));
4564
4565 /*
4566 * Allocate device states.
4567 */
4568 pThis->paDeviceStates = (PLSILOGICDEVICE)RTMemAllocZ(sizeof(LSILOGICDEVICE) * pThis->cDeviceStates);
4569 if (!pThis->paDeviceStates)
4570 return PDMDEV_SET_ERROR(pDevIns, rc,
4571 N_("Failed to allocate memory for device states"));
4572
4573 for (unsigned i = 0; i < pThis->cDeviceStates; i++)
4574 {
4575 char szName[24];
4576 PLSILOGICDEVICE pDevice = &pThis->paDeviceStates[i];
4577
4578 /* Initialize static parts of the device. */
4579 pDevice->iLUN = i;
4580 pDevice->pLsiLogicR3 = pThis;
4581 pDevice->Led.u32Magic = PDMLED_MAGIC;
4582 pDevice->IBase.pfnQueryInterface = lsilogicDeviceQueryInterface;
4583 pDevice->ISCSIPort.pfnSCSIRequestCompleted = lsilogicDeviceSCSIRequestCompleted;
4584 pDevice->ILed.pfnQueryStatusLed = lsilogicDeviceQueryStatusLed;
4585
4586 RTStrPrintf(szName, sizeof(szName), "Device%d", i);
4587
4588 /* Attach SCSI driver. */
4589 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, szName);
4590 if (RT_SUCCESS(rc))
4591 {
4592 /* Get SCSI connector interface. */
4593 pDevice->pDrvSCSIConnector = PDMIBASE_QUERY_INTERFACE(pDevice->pDrvBase, PDMISCSICONNECTOR);
4594 AssertMsgReturn(pDevice->pDrvSCSIConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
4595 }
4596 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
4597 {
4598 pDevice->pDrvBase = NULL;
4599 rc = VINF_SUCCESS;
4600 Log(("LsiLogic: no driver attached to device %s\n", szName));
4601 }
4602 else
4603 {
4604 AssertLogRelMsgFailed(("LsiLogic: Failed to attach %s\n", szName));
4605 return rc;
4606 }
4607 }
4608
4609 /*
4610 * Attach status driver (optional).
4611 */
4612 PPDMIBASE pBase;
4613 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
4614 if (RT_SUCCESS(rc))
4615 pThis->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
4616 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
4617 {
4618 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
4619 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot attach to status driver"));
4620 }
4621
4622 /* Initialize the SCSI emulation for the BIOS. */
4623 rc = vboxscsiInitialize(&pThis->VBoxSCSI);
4624 AssertRC(rc);
4625
4626 /* Register I/O port space in ISA region for BIOS access. */
4627 rc = PDMDevHlpIOPortRegister(pDevIns, LSILOGIC_ISA_IO_PORT, 3, NULL,
4628 lsilogicIsaIOPortWrite, lsilogicIsaIOPortRead,
4629 lsilogicIsaIOPortWriteStr, lsilogicIsaIOPortReadStr,
4630 "LsiLogic BIOS");
4631 if (RT_FAILURE(rc))
4632 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot register legacy I/O handlers"));
4633
4634 /* Register save state handlers. */
4635 rc = PDMDevHlpSSMRegister3(pDevIns, LSILOGIC_SAVED_STATE_VERSION, sizeof(*pThis),
4636 lsilogicLiveExec, lsilogicSaveExec, lsilogicLoadExec);
4637 if (RT_FAILURE(rc))
4638 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot register save state handlers"));
4639
4640 pThis->enmWhoInit = LSILOGICWHOINIT_SYSTEM_BIOS;
4641
4642 /* Perform hard reset. */
4643 rc = lsilogicHardReset(pThis);
4644 AssertRC(rc);
4645
4646 return rc;
4647}
4648
4649/**
4650 * The device registration structure.
4651 */
4652const PDMDEVREG g_DeviceLsiLogicSCSI =
4653{
4654 /* u32Version */
4655 PDM_DEVREG_VERSION,
4656 /* szName */
4657 "lsilogicscsi",
4658 /* szRCMod */
4659 "VBoxDDGC.gc",
4660 /* szR0Mod */
4661 "VBoxDDR0.r0",
4662 /* pszDescription */
4663 "LSI Logic 53c1030 SCSI controller.\n",
4664 /* fFlags */
4665 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
4666 /* fClass */
4667 PDM_DEVREG_CLASS_STORAGE,
4668 /* cMaxInstances */
4669 ~0,
4670 /* cbInstance */
4671 sizeof(LSILOGICSCSI),
4672 /* pfnConstruct */
4673 lsilogicConstruct,
4674 /* pfnDestruct */
4675 lsilogicDestruct,
4676 /* pfnRelocate */
4677 lsilogicRelocate,
4678 /* pfnIOCtl */
4679 NULL,
4680 /* pfnPowerOn */
4681 NULL,
4682 /* pfnReset */
4683 lsilogicReset,
4684 /* pfnSuspend */
4685 NULL,
4686 /* pfnResume */
4687 NULL,
4688 /* pfnAttach */
4689 lsilogicAttach,
4690 /* pfnDetach */
4691 lsilogicDetach,
4692 /* pfnQueryInterface. */
4693 NULL,
4694 /* pfnInitComplete */
4695 NULL,
4696 /* pfnPowerOff */
4697 NULL,
4698 /* pfnSoftReset */
4699 NULL,
4700 /* u32VersionEnd */
4701 PDM_DEVREG_VERSION
4702};
4703
4704#endif /* IN_RING3 */
4705#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
4706
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