VirtualBox

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

Last change on this file since 25963 was 25732, checked in by vboxsync, 15 years ago

PDMCritSect: Deployed lock ordering. (ring-3 only, only DEBUG_bird atm)

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