VirtualBox

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

Last change on this file since 22432 was 22277, checked in by vboxsync, 15 years ago

PDMDRVREG change (big changeset).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 210.5 KB
Line 
1/* $Id: DevLsiLogicSCSI.cpp 22277 2009-08-16 21:12:50Z vboxsync $ */
2/** @file
3 *
4 * VBox storage devices:
5 * LsiLogic LSI53c1030 SCSI controller.
6 */
7
8/*
9 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23//#define DEBUG
24#define LOG_GROUP LOG_GROUP_DEV_LSILOGICSCSI
25#include <VBox/pdmdev.h>
26#include <VBox/pdmqueue.h>
27#include <VBox/pdmcritsect.h>
28#include <VBox/scsi.h>
29#include <iprt/assert.h>
30#include <iprt/asm.h>
31#include <iprt/string.h>
32#ifdef IN_RING3
33# include <iprt/param.h>
34# include <iprt/alloc.h>
35# include <iprt/cache.h>
36#endif
37
38#include "VBoxSCSI.h"
39
40#include "../Builtins.h"
41
42/* I/O port registered in the ISA compatible range to let the BIOS access
43 * the controller.
44 */
45#define LSILOGIC_ISA_IO_PORT 0x340
46
47#define LSILOGIC_PORTS_MAX 1
48#define LSILOGIC_BUSES_MAX 1
49#define LSILOGIC_DEVICES_PER_BUS_MAX 16
50
51#define LSILOGIC_DEVICES_MAX (LSILOGIC_BUSES_MAX*LSILOGIC_DEVICES_PER_BUS_MAX)
52
53#define LSILOGICSCSI_REQUEST_QUEUE_DEPTH_DEFAULT 1024
54#define LSILOGICSCSI_REPLY_QUEUE_DEPTH_DEFAULT 128
55
56#define LSILOGICSCSI_MAXIMUM_CHAIN_DEPTH 3
57
58#define LSILOGIC_NR_OF_ALLOWED_BIGGER_LISTS 100
59
60#define LSILOGICSCSI_PCI_VENDOR_ID (0x1000)
61#define LSILOGICSCSI_PCI_DEVICE_ID (0x0030)
62#define LSILOGICSCSI_PCI_REVISION_ID (0x00)
63#define LSILOGICSCSI_PCI_CLASS_CODE (0x01)
64#define LSILOGICSCSI_PCI_SUBSYSTEM_VENDOR_ID (0x1000)
65#define LSILOGICSCSI_PCI_SUBSYSTEM_ID (0x8000)
66
67#define LSILOGIC_SAVED_STATE_MINOR_VERSION 1
68
69/**
70 * A simple SG element for a 64bit adress.
71 */
72#pragma pack(1)
73typedef struct MptSGEntrySimple64
74{
75 /** Length of the buffer this entry describes. */
76 unsigned u24Length: 24;
77 /** Flag whether this element is the end of the list. */
78 unsigned fEndOfList: 1;
79 /** Flag whether the address is 32bit or 64bits wide. */
80 unsigned f64BitAddress: 1;
81 /** Flag whether this buffer contains data to be transfered or is the destination. */
82 unsigned fBufferContainsData: 1;
83 /** Flag whether this is a local address or a system address. */
84 unsigned fLocalAddress: 1;
85 /** Element type. */
86 unsigned u2ElementType: 2;
87 /** Flag whether this is the last element of the buffer. */
88 unsigned fEndOfBuffer: 1;
89 /** Flag whether this is the last element of the current segment. */
90 unsigned fLastElement: 1;
91 /** Lower 32bits of the address of the data buffer. */
92 unsigned u32DataBufferAddressLow: 32;
93 /** Upper 32bits of the address of the data buffer. */
94 unsigned u32DataBufferAddressHigh: 32;
95} MptSGEntrySimple64, *PMptSGEntrySimple64;
96#pragma pack()
97AssertCompileSize(MptSGEntrySimple64, 12);
98
99/**
100 * A simple SG element for a 32bit adress.
101 */
102#pragma pack(1)
103typedef struct MptSGEntrySimple32
104{
105 /** Length of the buffer this entry describes. */
106 unsigned u24Length: 24;
107 /** Flag whether this element is the end of the list. */
108 unsigned fEndOfList: 1;
109 /** Flag whether the address is 32bit or 64bits wide. */
110 unsigned f64BitAddress: 1;
111 /** Flag whether this buffer contains data to be transfered or is the destination. */
112 unsigned fBufferContainsData: 1;
113 /** Flag whether this is a local address or a system address. */
114 unsigned fLocalAddress: 1;
115 /** Element type. */
116 unsigned u2ElementType: 2;
117 /** Flag whether this is the last element of the buffer. */
118 unsigned fEndOfBuffer: 1;
119 /** Flag whether this is the last element of the current segment. */
120 unsigned fLastElement: 1;
121 /** Lower 32bits of the address of the data buffer. */
122 unsigned u32DataBufferAddressLow: 32;
123} MptSGEntrySimple32, *PMptSGEntrySimple32;
124#pragma pack()
125AssertCompileSize(MptSGEntrySimple32, 8);
126
127/**
128 * A chain SG element.
129 */
130#pragma pack(1)
131typedef struct MptSGEntryChain
132{
133 /** Size of the segment. */
134 unsigned u16Length: 16;
135 /** Offset in 32bit words of the next chain element in the segment
136 * identified by this element. */
137 unsigned u8NextChainOffset: 8;
138 /** Reserved. */
139 unsigned fReserved0: 1;
140 /** Flag whether the address is 32bit or 64bits wide. */
141 unsigned f64BitAddress: 1;
142 /** Reserved. */
143 unsigned fReserved1: 1;
144 /** Flag whether this is a local address or a system address. */
145 unsigned fLocalAddress: 1;
146 /** Element type. */
147 unsigned u2ElementType: 2;
148 /** Flag whether this is the last element of the buffer. */
149 unsigned u2Reserved2: 2;
150 /** Lower 32bits of the address of the data buffer. */
151 unsigned u32SegmentAddressLow: 32;
152 /** Upper 32bits of the address of the data buffer. */
153 unsigned u32SegmentAddressHigh: 32;
154} MptSGEntryChain, *PMptSGEntryChain;
155#pragma pack()
156AssertCompileSize(MptSGEntryChain, 12);
157
158typedef union MptSGEntryUnion
159{
160 MptSGEntrySimple64 Simple64;
161 MptSGEntrySimple32 Simple32;
162 MptSGEntryChain Chain;
163} MptSGEntryUnion, *PMptSGEntryUnion;
164
165/**
166 * MPT Fusion message header - Common for all message frames.
167 * This is filled in by the guest.
168 */
169#pragma pack(1)
170typedef struct MptMessageHdr
171{
172 /** Function dependent data. */
173 uint16_t u16FunctionDependent;
174 /** Chain offset. */
175 uint8_t u8ChainOffset;
176 /** The function code. */
177 uint8_t u8Function;
178 /** Function dependent data. */
179 uint8_t au8FunctionDependent[3];
180 /** Message flags. */
181 uint8_t u8MessageFlags;
182 /** Message context - Unique ID from the guest unmodified by the device. */
183 uint32_t u32MessageContext;
184} MptMessageHdr, *PMptMessageHdr;
185#pragma pack()
186AssertCompileSize(MptMessageHdr, 12);
187
188/** Defined function codes found in the message header. */
189#define MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST (0x00)
190#define MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT (0x01)
191#define MPT_MESSAGE_HDR_FUNCTION_IOC_INIT (0x02)
192#define MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS (0x03)
193#define MPT_MESSAGE_HDR_FUNCTION_CONFIG (0x04)
194#define MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS (0x05)
195#define MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE (0x06)
196#define MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION (0x07)
197#define MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK (0x08)
198#define MPT_MESSAGE_HDR_FUNCTION_FW_DOWNLOAD (0x09)
199#define MPT_MESSAGE_HDR_FUNCTION_TARGET_CMD_BUFFER_POST (0x0A)
200#define MPT_MESSAGE_HDR_FUNCTION_TARGET_ASSIST (0x0B)
201#define MPT_MESSAGE_HDR_FUNCTION_TARGET_STATUS_SEND (0x0C)
202#define MPT_MESSAGE_HDR_FUNCTION_TARGET_MODE_ABORT (0x0D)
203
204#ifdef DEBUG
205/**
206 * Function names
207 */
208static const char * const g_apszMPTFunctionNames[] =
209{
210 "SCSI I/O Request",
211 "SCSI Task Management",
212 "IOC Init",
213 "IOC Facts",
214 "Config",
215 "Port Facts",
216 "Port Enable",
217 "Event Notification",
218 "Event Ack",
219 "Firmware Download"
220};
221#endif
222
223/**
224 * Default reply message.
225 * Send from the device to the guest upon completion of a request.
226 */
227 #pragma pack(1)
228typedef struct MptDefaultReplyMessage
229{
230 /** Function dependent data. */
231 uint16_t u16FunctionDependent;
232 /** Length of the message in 32bit DWords. */
233 uint8_t u8MessageLength;
234 /** Function which completed. */
235 uint8_t u8Function;
236 /** Function dependent. */
237 uint8_t au8FunctionDependent[3];
238 /** Message flags. */
239 uint8_t u8MessageFlags;
240 /** Message context given in the request. */
241 uint32_t u32MessageContext;
242 /** Function dependent status code. */
243 uint16_t u16FunctionDependentStatus;
244 /** Status of the IOC. */
245 uint16_t u16IOCStatus;
246 /** Additional log info. */
247 uint32_t u32IOCLogInfo;
248} MptDefaultReplyMessage, *PMptDefaultReplyMessage;
249#pragma pack()
250AssertCompileSize(MptDefaultReplyMessage, 20);
251
252/**
253 * IO controller init request.
254 */
255#pragma pack(1)
256typedef struct MptIOCInitRequest
257{
258 /** Which system send this init request. */
259 uint8_t u8WhoInit;
260 /** Reserved */
261 uint8_t u8Reserved;
262 /** Chain offset in the SG list. */
263 uint8_t u8ChainOffset;
264 /** Function to execute. */
265 uint8_t u8Function;
266 /** Flags */
267 uint8_t u8Flags;
268 /** Maximum number of devices the driver can handle. */
269 uint8_t u8MaxDevices;
270 /** Maximum number of buses the driver can handle. */
271 uint8_t u8MaxBuses;
272 /** Message flags. */
273 uint8_t u8MessageFlags;
274 /** Message context ID. */
275 uint32_t u32MessageContext;
276 /** Reply frame size. */
277 uint16_t u16ReplyFrameSize;
278 /** Reserved */
279 uint16_t u16Reserved;
280 /** Upper 32bit part of the 64bit address the message frames are in.
281 * That means all frames must be in the same 4GB segment. */
282 uint32_t u32HostMfaHighAddr;
283 /** Upper 32bit of the sense buffer. */
284 uint32_t u32SenseBufferHighAddr;
285} MptIOCInitRequest, *PMptIOCInitRequest;
286#pragma pack()
287AssertCompileSize(MptIOCInitRequest, 24);
288
289/**
290 * IO controller init reply.
291 */
292#pragma pack(1)
293typedef struct MptIOCInitReply
294{
295 /** Which subsystem send this init request. */
296 uint8_t u8WhoInit;
297 /** Reserved */
298 uint8_t u8Reserved;
299 /** Message length */
300 uint8_t u8MessageLength;
301 /** Function. */
302 uint8_t u8Function;
303 /** Flags */
304 uint8_t u8Flags;
305 /** Maximum number of devices the driver can handle. */
306 uint8_t u8MaxDevices;
307 /** Maximum number of busses the driver can handle. */
308 uint8_t u8MaxBuses;
309 /** Message flags. */
310 uint8_t u8MessageFlags;
311 /** Message context ID */
312 uint32_t u32MessageContext;
313 /** Reserved */
314 uint16_t u16Reserved;
315 /** IO controller status. */
316 uint16_t u16IOCStatus;
317 /** IO controller log information. */
318 uint32_t u32IOCLogInfo;
319} MptIOCInitReply, *PMptIOCInitReply;
320#pragma pack()
321AssertCompileSize(MptIOCInitReply, 20);
322
323/**
324 * IO controller facts request.
325 */
326#pragma pack(1)
327typedef struct MptIOCFactsRequest
328{
329 /** Reserved. */
330 uint16_t u16Reserved;
331 /** Chain offset in SG list. */
332 uint8_t u8ChainOffset;
333 /** Function number. */
334 uint8_t u8Function;
335 /** Reserved */
336 uint8_t u8Reserved[3];
337 /** Message flags. */
338 uint8_t u8MessageFlags;
339 /** Message context ID. */
340 uint32_t u32MessageContext;
341} MptIOCFactsRequest, *PMptIOCFactsRequest;
342#pragma pack()
343AssertCompileSize(MptIOCFactsRequest, 12);
344
345/**
346 * IO controller facts reply.
347 */
348#pragma pack(1)
349typedef struct MptIOCFactsReply
350{
351 /** Message version. */
352 uint16_t u16MessageVersion;
353 /** Message length. */
354 uint8_t u8MessageLength;
355 /** Function number. */
356 uint8_t u8Function;
357 /** Reserved */
358 uint16_t u16Reserved1;
359 /** IO controller number */
360 uint8_t u8IOCNumber;
361 /** Message flags. */
362 uint8_t u8MessageFlags;
363 /** Message context ID. */
364 uint32_t u32MessageContext;
365 /** IO controller exceptions */
366 uint16_t u16IOCExceptions;
367 /** IO controller status. */
368 uint16_t u16IOCStatus;
369 /** IO controller log information. */
370 uint32_t u32IOCLogInfo;
371 /** Maximum chain depth. */
372 uint8_t u8MaxChainDepth;
373 /** The current value of the WhoInit field. */
374 uint8_t u8WhoInit;
375 /** Block size. */
376 uint8_t u8BlockSize;
377 /** Flags. */
378 uint8_t u8Flags;
379 /** Depth of the reply queue. */
380 uint16_t u16ReplyQueueDepth;
381 /** Size of a request frame. */
382 uint16_t u16RequestFrameSize;
383 /** Reserved */
384 uint16_t u16Reserved2;
385 /** Product ID. */
386 uint16_t u16ProductID;
387 /** Current value of the high 32bit MFA address. */
388 uint32_t u32CurrentHostMFAHighAddr;
389 /** Global credits - Number of entries allocated to queues */
390 uint16_t u16GlobalCredits;
391 /** Number of ports on the IO controller */
392 uint8_t u8NumberOfPorts;
393 /** Event state. */
394 uint8_t u8EventState;
395 /** Current value of the high 32bit sense buffer address. */
396 uint32_t u32CurrentSenseBufferHighAddr;
397 /** Current reply frame size. */
398 uint16_t u16CurReplyFrameSize;
399 /** Maximum number of devices. */
400 uint8_t u8MaxDevices;
401 /** Maximum number of buses. */
402 uint8_t u8MaxBuses;
403 /** Size of the firmware image. */
404 uint32_t u32FwImageSize;
405 /** Reserved. */
406 uint32_t u32Reserved;
407 /** Firmware version */
408 uint32_t u32FWVersion;
409} MptIOCFactsReply, *PMptIOCFactsReply;
410#pragma pack()
411AssertCompileSize(MptIOCFactsReply, 60);
412
413/**
414 * Port facts request
415 */
416#pragma pack(1)
417typedef struct MptPortFactsRequest
418{
419 /** Reserved */
420 uint16_t u16Reserved1;
421 /** Message length. */
422 uint8_t u8MessageLength;
423 /** Function number. */
424 uint8_t u8Function;
425 /** Reserved */
426 uint16_t u16Reserved2;
427 /** Port number to get facts for. */
428 uint8_t u8PortNumber;
429 /** Message flags. */
430 uint8_t u8MessageFlags;
431 /** Message context ID. */
432 uint32_t u32MessageContext;
433} MptPortFactsRequest, *PMptPortFactsRequest;
434#pragma pack()
435AssertCompileSize(MptPortFactsRequest, 12);
436
437/**
438 * Port facts reply.
439 */
440#pragma pack(1)
441typedef struct MptPortFactsReply
442{
443 /** Reserved. */
444 uint16_t u16Reserved1;
445 /** Message length. */
446 uint8_t u8MessageLength;
447 /** Function number. */
448 uint8_t u8Function;
449 /** Reserved */
450 uint16_t u16Reserved2;
451 /** Port number the facts are for. */
452 uint8_t u8PortNumber;
453 /** Message flags. */
454 uint8_t u8MessageFlags;
455 /** Message context ID. */
456 uint32_t u32MessageContext;
457 /** Reserved. */
458 uint16_t u16Reserved3;
459 /** IO controller status. */
460 uint16_t u16IOCStatus;
461 /** IO controller log information. */
462 uint32_t u32IOCLogInfo;
463 /** Reserved */
464 uint8_t u8Reserved;
465 /** Port type */
466 uint8_t u8PortType;
467 /** Maximum number of devices on this port. */
468 uint16_t u16MaxDevices;
469 /** SCSI ID of this port on the attached bus. */
470 uint16_t u16PortSCSIID;
471 /** Protocol flags. */
472 uint16_t u16ProtocolFlags;
473 /** Maxmimum number of target command buffers which can be posted to this port at a time. */
474 uint16_t u16MaxPostedCmdBuffers;
475 /** Maximum number of target IDs that remain persistent between power/reset cycles. */
476 uint16_t u16MaxPersistentIDs;
477 /** Maximum number of LAN buckets. */
478 uint16_t u16MaxLANBuckets;
479 /** Reserved. */
480 uint16_t u16Reserved4;
481 /** Reserved. */
482 uint32_t u32Reserved;
483} MptPortFactsReply, *PMptPortFactsReply;
484#pragma pack()
485AssertCompileSize(MptPortFactsReply, 40);
486
487/**
488 * Port Enable request.
489 */
490#pragma pack(1)
491typedef struct MptPortEnableRequest
492{
493 /** Reserved. */
494 uint16_t u16Reserved1;
495 /** Message length. */
496 uint8_t u8MessageLength;
497 /** Function number. */
498 uint8_t u8Function;
499 /** Reserved. */
500 uint16_t u16Reserved2;
501 /** Port number to enable. */
502 uint8_t u8PortNumber;
503 /** Message flags. */
504 uint8_t u8MessageFlags;
505 /** Message context ID. */
506 uint32_t u32MessageContext;
507} MptPortEnableRequest, *PMptPortEnableRequest;
508#pragma pack()
509AssertCompileSize(MptPortEnableRequest, 12);
510
511/**
512 * Port enable reply.
513 */
514#pragma pack(1)
515typedef struct MptPortEnableReply
516{
517 /** Reserved. */
518 uint16_t u16Reserved1;
519 /** Message length. */
520 uint8_t u8MessageLength;
521 /** Function number. */
522 uint8_t u8Function;
523 /** Reserved */
524 uint16_t u16Reserved2;
525 /** Port number which was enabled. */
526 uint8_t u8PortNumber;
527 /** Message flags. */
528 uint8_t u8MessageFlags;
529 /** Message context ID. */
530 uint32_t u32MessageContext;
531 /** Reserved. */
532 uint16_t u16Reserved3;
533 /** IO controller status */
534 uint16_t u16IOCStatus;
535 /** IO controller log information. */
536 uint32_t u32IOCLogInfo;
537} MptPortEnableReply, *PMptPortEnableReply;
538#pragma pack()
539AssertCompileSize(MptPortEnableReply, 20);
540
541/**
542 * Event notification request.
543 */
544#pragma pack(1)
545typedef struct MptEventNotificationRequest
546{
547 /** Switch - Turns event notification on and off. */
548 uint8_t u8Switch;
549 /** Reserved. */
550 uint8_t u8Reserved1;
551 /** Chain offset. */
552 uint8_t u8ChainOffset;
553 /** Function number. */
554 uint8_t u8Function;
555 /** Reserved. */
556 uint8_t u8reserved2[3];
557 /** Message flags. */
558 uint8_t u8MessageFlags;
559 /** Message context ID. */
560 uint32_t u32MessageContext;
561} MptEventNotificationRequest, *PMptEventNotificationRequest;
562#pragma pack()
563AssertCompileSize(MptEventNotificationRequest, 12);
564
565/**
566 * Event notification reply.
567 */
568#pragma pack(1)
569typedef struct MptEventNotificationReply
570{
571 /** Event data length. */
572 uint16_t u16EventDataLength;
573 /** Message length. */
574 uint8_t u8MessageLength;
575 /** Function number. */
576 uint8_t u8Function;
577 /** Reserved. */
578 uint16_t u16Reserved1;
579 /** Ack required. */
580 uint8_t u8AckRequired;
581 /** Message flags. */
582 uint8_t u8MessageFlags;
583 /** Message context ID. */
584 uint32_t u32MessageContext;
585 /** Reserved. */
586 uint16_t u16Reserved2;
587 /** IO controller status. */
588 uint16_t u16IOCStatus;
589 /** IO controller log information. */
590 uint32_t u32IOCLogInfo;
591 /** Notification event. */
592 uint32_t u32Event;
593 /** Event context. */
594 uint32_t u32EventContext;
595 /** Event data. */
596 uint32_t u32EventData;
597} MptEventNotificationReply, *PMptEventNotificationReply;
598#pragma pack()
599AssertCompileSize(MptEventNotificationReply, 32);
600
601#define MPT_EVENT_EVENT_CHANGE (0x0000000a)
602
603/**
604 * SCSI IO Request
605 */
606#pragma pack(1)
607typedef struct MptSCSIIORequest
608{
609 /** Target ID */
610 uint8_t u8TargetID;
611 /** Bus number */
612 uint8_t u8Bus;
613 /** Chain offset */
614 uint8_t u8ChainOffset;
615 /** Function number. */
616 uint8_t u8Function;
617 /** CDB length. */
618 uint8_t u8CDBLength;
619 /** Sense buffer length. */
620 uint8_t u8SenseBufferLength;
621 /** Rserved */
622 uint8_t u8Reserved;
623 /** Message flags. */
624 uint8_t u8MessageFlags;
625 /** Message context ID. */
626 uint32_t u32MessageContext;
627 /** LUN */
628 uint8_t au8LUN[8];
629 /** Control values. */
630 uint32_t u32Control;
631 /** The CDB. */
632 uint8_t au8CDB[16];
633 /** Data length. */
634 uint32_t u32DataLength;
635 /** Sense buffer low 32bit address. */
636 uint32_t u32SenseBufferLowAddress;
637} MptSCSIIORequest, *PMptSCSIIORequest;
638#pragma pack()
639AssertCompileSize(MptSCSIIORequest, 48);
640
641#define MPT_SCSIIO_REQUEST_CONTROL_TXDIR_GET(x) (((x) & 0x3000000) >> 24)
642#define MPT_SCSIIO_REQUEST_CONTROL_TXDIR_NONE (0x0)
643#define MPT_SCSIIO_REQUEST_CONTROL_TXDIR_WRITE (0x1)
644#define MPT_SCSIIO_REQUEST_CONTROL_TXDIR_READ (0x2)
645
646/**
647 * SCSI IO error reply.
648 */
649#pragma pack(1)
650typedef struct MptSCSIIOErrorReply
651{
652 /** Target ID */
653 uint8_t u8TargetID;
654 /** Bus number */
655 uint8_t u8Bus;
656 /** Message length. */
657 uint8_t u8MessageLength;
658 /** Function number. */
659 uint8_t u8Function;
660 /** CDB length */
661 uint8_t u8CDBLength;
662 /** Sense buffer length */
663 uint8_t u8SenseBufferLength;
664 /** Reserved */
665 uint8_t u8Reserved;
666 /** Message flags */
667 uint8_t u8MessageFlags;
668 /** Message context ID */
669 uint32_t u32MessageContext;
670 /** SCSI status. */
671 uint8_t u8SCSIStatus;
672 /** SCSI state */
673 uint8_t u8SCSIState;
674 /** IO controller status */
675 uint16_t u16IOCStatus;
676 /** IO controller log information */
677 uint32_t u32IOCLogInfo;
678 /** Transfer count */
679 uint32_t u32TransferCount;
680 /** Sense count */
681 uint32_t u32SenseCount;
682 /** Response information */
683 uint32_t u32ResponseInfo;
684} MptSCSIIOErrorReply, *PMptSCSIIOErrorReply;
685#pragma pack()
686AssertCompileSize(MptSCSIIOErrorReply, 32);
687
688#define MPT_SCSI_IO_ERROR_SCSI_STATE_AUTOSENSE_VALID (0x01)
689#define MPT_SCSI_IO_ERROR_SCSI_STATE_TERMINATED (0x08)
690
691/**
692 * IOC status codes sepcific to the SCSI I/O error reply.
693 */
694#define MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_BUS (0x0041)
695#define MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_TARGETID (0x0042)
696#define MPT_SCSI_IO_ERROR_IOCSTATUS_DEVICE_NOT_THERE (0x0043)
697
698/**
699 * SCSI task management request.
700 */
701#pragma pack(1)
702typedef struct MptSCSITaskManagementRequest
703{
704 /** Target ID */
705 uint8_t u8TargetID;
706 /** Bus number */
707 uint8_t u8Bus;
708 /** Chain offset */
709 uint8_t u8ChainOffset;
710 /** Function number */
711 uint8_t u8Function;
712 /** Reserved */
713 uint8_t u8Reserved1;
714 /** Task type */
715 uint8_t u8TaskType;
716 /** Reserved */
717 uint8_t u8Reserved2;
718 /** Message flags */
719 uint8_t u8MessageFlags;
720 /** Message context ID */
721 uint32_t u32MessageContext;
722 /** LUN */
723 uint8_t au8LUN[8];
724 /** Reserved */
725 uint8_t auReserved[28];
726 /** Task message context ID. */
727 uint32_t u32TaskMessageContext;
728} MptSCSITaskManagementRequest, *PMptSCSITaskManagementRequest;
729#pragma pack()
730AssertCompileSize(MptSCSITaskManagementRequest, 52);
731
732/**
733 * SCSI task management reply.
734 */
735#pragma pack(1)
736typedef struct MptSCSITaskManagementReply
737{
738 /** Target ID */
739 uint8_t u8TargetID;
740 /** Bus number */
741 uint8_t u8Bus;
742 /** Message length */
743 uint8_t u8MessageLength;
744 /** Function number */
745 uint8_t u8Function;
746 /** Reserved */
747 uint8_t u8Reserved1;
748 /** Task type */
749 uint8_t u8TaskType;
750 /** Reserved */
751 uint8_t u8Reserved2;
752 /** Message flags */
753 uint8_t u8MessageFlags;
754 /** Message context ID */
755 uint32_t u32MessageContext;
756 /** Reserved */
757 uint16_t u16Reserved;
758 /** IO controller status */
759 uint16_t u16IOCStatus;
760 /** IO controller log information */
761 uint32_t u32IOCLogInfo;
762 /** Termination count */
763 uint32_t u32TerminationCount;
764} MptSCSITaskManagementReply, *PMptSCSITaskManagementReply;
765#pragma pack()
766AssertCompileSize(MptSCSITaskManagementReply, 24);
767
768/**
769 * Configuration request
770 */
771#pragma pack(1)
772typedef struct MptConfigurationRequest
773{
774 /** Action code. */
775 uint8_t u8Action;
776 /** Reserved. */
777 uint8_t u8Reserved1;
778 /** Chain offset. */
779 uint8_t u8ChainOffset;
780 /** Function number. */
781 uint8_t u8Function;
782 /** Reserved. */
783 uint8_t u8Reserved2[3];
784 /** Message flags. */
785 uint8_t u8MessageFlags;
786 /** Message context ID. */
787 uint32_t u32MessageContext;
788 /** Reserved. */
789 uint8_t u8Reserved3[8];
790 /** Version number of the page. */
791 uint8_t u8PageVersion;
792 /** Length of the page in 32bit Dwords. */
793 uint8_t u8PageLength;
794 /** Page number to access. */
795 uint8_t u8PageNumber;
796 /** Type of the page beeing accessed. */
797 uint8_t u8PageType;
798 /** Page type dependent address. */
799 union
800 {
801 /** 32bit view. */
802 uint32_t u32PageAddress;
803 struct
804 {
805 /** Port number to get the configuration page for. */
806 uint8_t u8PortNumber;
807 /** Reserved. */
808 uint8_t u8Reserved[3];
809 } MPIPortNumber;
810 struct
811 {
812 /** Target ID to get the configuration page for. */
813 uint8_t u8TargetID;
814 /** Bus number to get the configuration page for. */
815 uint8_t u8Bus;
816 /** Reserved. */
817 uint8_t u8Reserved[2];
818 } BusAndTargetId;
819 } u;
820 MptSGEntrySimple64 SimpleSGElement;
821} MptConfigurationRequest, *PMptConfigurationRequest;
822#pragma pack()
823AssertCompileSize(MptConfigurationRequest, 40);
824
825/** Possible action codes. */
826#define MPT_CONFIGURATION_REQUEST_ACTION_HEADER (0x00)
827#define MPT_CONFIGURATION_REQUEST_ACTION_READ_CURRENT (0x01)
828#define MPT_CONFIGURATION_REQUEST_ACTION_WRITE_CURRENT (0x02)
829#define MPT_CONFIGURATION_REQUEST_ACTION_READ_DEFAULT (0x03)
830#define MPT_CONFIGURATION_REQUEST_ACTION_DEFAULT (0x04)
831#define MPT_CONFIGURATION_REQUEST_ACTION_READ_NVRAM (0x05)
832#define MPT_CONFIGURATION_REQUEST_ACTION_WRITE_NVRAM (0x06)
833
834/** Page type codes. */
835#define MPT_CONFIGURATION_REQUEST_PAGE_TYPE_IO_UNIT (0x00)
836#define MPT_CONFIGURATION_REQUEST_PAGE_TYPE_IOC (0x01)
837#define MPT_CONFIGURATION_REQUEST_PAGE_TYPE_BIOS (0x02)
838#define MPT_CONFIGURATION_REQUEST_PAGE_TYPE_SCSI_PORT (0x03)
839
840/**
841 * Configuration reply.
842 */
843#pragma pack(1)
844typedef struct MptConfigurationReply
845{
846 /** Action code. */
847 uint8_t u8Action;
848 /** Reserved. */
849 uint8_t u8Reserved;
850 /** Message length. */
851 uint8_t u8MessageLength;
852 /** Function number. */
853 uint8_t u8Function;
854 /** Reserved. */
855 uint8_t u8Reserved2[3];
856 /** Message flags. */
857 uint8_t u8MessageFlags;
858 /** Message context ID. */
859 uint32_t u32MessageContext;
860 /** Reserved. */
861 uint16_t u16Reserved;
862 /** I/O controller status. */
863 uint16_t u16IOCStatus;
864 /** I/O controller log information. */
865 uint32_t u32IOCLogInfo;
866 /** Version number of the page. */
867 uint8_t u8PageVersion;
868 /** Length of the page in 32bit Dwords. */
869 uint8_t u8PageLength;
870 /** Page number to access. */
871 uint8_t u8PageNumber;
872 /** Type of the page beeing accessed. */
873 uint8_t u8PageType;
874} MptConfigurationReply, *PMptConfigurationReply;
875#pragma pack()
876AssertCompileSize(MptConfigurationReply, 24);
877
878/** Additional I/O controller status codes for the configuration reply. */
879#define MPT_IOCSTATUS_CONFIG_INVALID_ACTION (0x0020)
880#define MPT_IOCSTATUS_CONFIG_INVALID_TYPE (0x0021)
881#define MPT_IOCSTATUS_CONFIG_INVALID_PAGE (0x0022)
882#define MPT_IOCSTATUS_CONFIG_INVALID_DATA (0x0023)
883#define MPT_IOCSTATUS_CONFIG_NO_DEFAULTS (0x0024)
884#define MPT_IOCSTATUS_CONFIG_CANT_COMMIT (0x0025)
885
886/**
887 * Union of all possible request messages.
888 */
889typedef union MptRequestUnion
890{
891 MptMessageHdr Header;
892 MptIOCInitRequest IOCInit;
893 MptIOCFactsRequest IOCFacts;
894 MptPortFactsRequest PortFacts;
895 MptPortEnableRequest PortEnable;
896 MptEventNotificationRequest EventNotification;
897 MptSCSIIORequest SCSIIO;
898 MptSCSITaskManagementRequest SCSITaskManagement;
899 MptConfigurationRequest Configuration;
900} MptRequestUnion, *PMptRequestUnion;
901
902/**
903 * Union of all possible reply messages.
904 */
905typedef union MptReplyUnion
906{
907 /** 16bit view. */
908 uint16_t au16Reply[30];
909 MptDefaultReplyMessage Header;
910 MptIOCInitReply IOCInit;
911 MptIOCFactsReply IOCFacts;
912 MptPortFactsReply PortFacts;
913 MptPortEnableReply PortEnable;
914 MptEventNotificationReply EventNotification;
915 MptSCSIIOErrorReply SCSIIOError;
916 MptSCSITaskManagementReply SCSITaskManagement;
917 MptConfigurationReply Configuration;
918} MptReplyUnion, *PMptReplyUnion;
919
920
921/**
922 * Configuration Page attributes.
923 */
924#define MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY (0x00)
925#define MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE (0x10)
926#define MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT (0x20)
927#define MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY (0x30)
928
929#define MPT_CONFIGURATION_PAGE_ATTRIBUTE_GET(u8PageType) ((u8PageType) & 0xf0)
930
931/**
932 * Configuration Page types.
933 */
934#define MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT (0x00)
935#define MPT_CONFIGURATION_PAGE_TYPE_IOC (0x01)
936#define MPT_CONFIGURATION_PAGE_TYPE_BIOS (0x02)
937#define MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT (0x03)
938#define MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE (0x04)
939#define MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING (0x09)
940
941#define MPT_CONFIGURATION_PAGE_TYPE_GET(u8PageType) ((u8PageType) & 0x0f)
942
943/**
944 * Configuration Page header - Common to all pages.
945 */
946#pragma pack(1)
947typedef struct MptConfigurationPageHeader
948{
949 /** Version of the page. */
950 uint8_t u8PageVersion;
951 /** The length of the page in 32bit D-Words. */
952 uint8_t u8PageLength;
953 /** Number of the page. */
954 uint8_t u8PageNumber;
955 /** Type of the page. */
956 uint8_t u8PageType;
957} MptConfigurationPageHeader, *PMptConfigurationPageHeader;
958#pragma pack()
959AssertCompileSize(MptConfigurationPageHeader, 4);
960
961/**
962 * Manufacturing page 0. - Readonly.
963 */
964#pragma pack(1)
965typedef struct MptConfigurationPageManufacturing0
966{
967 /** Union. */
968 union
969 {
970 /** Byte view. */
971 uint8_t abPageData[76];
972 /** Field view. */
973 struct
974 {
975 /** The omnipresent header. */
976 MptConfigurationPageHeader Header;
977 /** Name of the chip. */
978 uint8_t abChipName[16];
979 /** Chip revision. */
980 uint8_t abChipRevision[8];
981 /** Board name. */
982 uint8_t abBoardName[16];
983 /** Board assembly. */
984 uint8_t abBoardAssembly[16];
985 /** Board tracer number. */
986 uint8_t abBoardTracerNumber[16];
987 } fields;
988 } u;
989} MptConfigurationPageManufacturing0, *PMptConfigurationPageManufacturing0;
990#pragma pack()
991AssertCompileSize(MptConfigurationPageManufacturing0, 76);
992
993/**
994 * Manufacturing page 1. - Readonly Persistent.
995 */
996#pragma pack(1)
997typedef struct MptConfigurationPageManufacturing1
998{
999 /** The omnipresent header. */
1000 MptConfigurationPageHeader Header;
1001 /** VPD info - don't know what belongs here so all zero. */
1002 uint8_t abVPDInfo[256];
1003} MptConfigurationPageManufacturing1, *PMptConfigurationPageManufacturing1;
1004#pragma pack()
1005AssertCompileSize(MptConfigurationPageManufacturing1, 260);
1006
1007/**
1008 * Manufacturing page 2. - Readonly.
1009 */
1010#pragma pack(1)
1011typedef struct MptConfigurationPageManufacturing2
1012{
1013 /** Union. */
1014 union
1015 {
1016 /** Byte view. */
1017 uint8_t abPageData[8];
1018 /** Field view. */
1019 struct
1020 {
1021 /** The omnipresent header. */
1022 MptConfigurationPageHeader Header;
1023 /** PCI Device ID. */
1024 uint16_t u16PCIDeviceID;
1025 /** PCI Revision ID. */
1026 uint8_t u8PCIRevisionID;
1027 /** Reserved. */
1028 uint8_t u8Reserved;
1029 /** Hardware specific settings... */
1030 } fields;
1031 } u;
1032} MptConfigurationPageManufacturing2, *PMptConfigurationPageManufacturing2;
1033#pragma pack()
1034AssertCompileSize(MptConfigurationPageManufacturing2, 8);
1035
1036/**
1037 * Manufacturing page 3. - Readonly.
1038 */
1039#pragma pack(1)
1040typedef struct MptConfigurationPageManufacturing3
1041{
1042 /** Union. */
1043 union
1044 {
1045 /** Byte view. */
1046 uint8_t abPageData[8];
1047 /** Field view. */
1048 struct
1049 {
1050 /** The omnipresent header. */
1051 MptConfigurationPageHeader Header;
1052 /** PCI Device ID. */
1053 uint16_t u16PCIDeviceID;
1054 /** PCI Revision ID. */
1055 uint8_t u8PCIRevisionID;
1056 /** Reserved. */
1057 uint8_t u8Reserved;
1058 /** Chip specific settings... */
1059 } fields;
1060 } u;
1061} MptConfigurationPageManufacturing3, *PMptConfigurationPageManufacturing3;
1062#pragma pack()
1063AssertCompileSize(MptConfigurationPageManufacturing3, 8);
1064
1065/**
1066 * Manufacturing page 4. - Readonly.
1067 */
1068#pragma pack(1)
1069typedef struct MptConfigurationPageManufacturing4
1070{
1071 /** Union. */
1072 union
1073 {
1074 /** Byte view. */
1075 uint8_t abPageData[84];
1076 /** Field view. */
1077 struct
1078 {
1079 /** The omnipresent header. */
1080 MptConfigurationPageHeader Header;
1081 /** Reserved. */
1082 uint32_t u32Reserved;
1083 /** InfoOffset0. */
1084 uint8_t u8InfoOffset0;
1085 /** Info size. */
1086 uint8_t u8InfoSize0;
1087 /** InfoOffset1. */
1088 uint8_t u8InfoOffset1;
1089 /** Info size. */
1090 uint8_t u8InfoSize1;
1091 /** Size of the inquiry data. */
1092 uint8_t u8InquirySize;
1093 /** Reserved. */
1094 uint8_t abReserved[3];
1095 /** Inquiry data. */
1096 uint8_t abInquiryData[56];
1097 /** IS volume settings. */
1098 uint32_t u32ISVolumeSettings;
1099 /** IME volume settings. */
1100 uint32_t u32IMEVolumeSettings;
1101 /** IM volume settings. */
1102 uint32_t u32IMVolumeSettings;
1103 } fields;
1104 } u;
1105} MptConfigurationPageManufacturing4, *PMptConfigurationPageManufacturing4;
1106#pragma pack()
1107AssertCompileSize(MptConfigurationPageManufacturing4, 84);
1108
1109/**
1110 * IO Unit page 0. - Readonly.
1111 */
1112#pragma pack(1)
1113typedef struct MptConfigurationPageIOUnit0
1114{
1115 /** Union. */
1116 union
1117 {
1118 /** Byte view. */
1119 uint8_t abPageData[12];
1120 /** Field view. */
1121 struct
1122 {
1123 /** The omnipresent header. */
1124 MptConfigurationPageHeader Header;
1125 /** A unique identifier. */
1126 uint64_t u64UniqueIdentifier;
1127 } fields;
1128 } u;
1129} MptConfigurationPageIOUnit0, *PMptConfigurationPageIOUnit0;
1130#pragma pack()
1131AssertCompileSize(MptConfigurationPageIOUnit0, 12);
1132
1133/**
1134 * IO Unit page 1. - Read/Write.
1135 */
1136#pragma pack(1)
1137typedef struct MptConfigurationPageIOUnit1
1138{
1139 /** Union. */
1140 union
1141 {
1142 /** Byte view. */
1143 uint8_t abPageData[8];
1144 /** Field view. */
1145 struct
1146 {
1147 /** The omnipresent header. */
1148 MptConfigurationPageHeader Header;
1149 /** Flag whether this is a single function PCI device. */
1150 unsigned fSingleFunction: 1;
1151 /** Flag whether all possible paths to a device are mapped. */
1152 unsigned fAllPathsMapped: 1;
1153 /** Reserved. */
1154 unsigned u4Reserved: 4;
1155 /** Flag whether all RAID functionality is disabled. */
1156 unsigned fIntegratedRAIDDisabled: 1;
1157 /** Flag whether 32bit PCI accesses are forced. */
1158 unsigned f32BitAccessForced: 1;
1159 /** Reserved. */
1160 unsigned abReserved: 24;
1161 } fields;
1162 } u;
1163} MptConfigurationPageIOUnit1, *PMptConfigurationPageIOUnit1;
1164#pragma pack()
1165AssertCompileSize(MptConfigurationPageIOUnit1, 8);
1166
1167/**
1168 * Adapter Ordering.
1169 */
1170#pragma pack(1)
1171typedef struct MptConfigurationPageIOUnit2AdapterOrdering
1172{
1173 /** PCI bus number. */
1174 unsigned u8PCIBusNumber: 8;
1175 /** PCI device and function number. */
1176 unsigned u8PCIDevFn: 8;
1177 /** Flag whether the adapter is embedded. */
1178 unsigned fAdapterEmbedded: 1;
1179 /** Flag whether the adapter is enabled. */
1180 unsigned fAdapterEnabled: 1;
1181 /** Reserved. */
1182 unsigned u6Reserved: 6;
1183 /** Reserved. */
1184 unsigned u8Reserved: 8;
1185} MptConfigurationPageIOUnit2AdapterOrdering, *PMptConfigurationPageIOUnit2AdapterOrdering;
1186#pragma pack()
1187AssertCompileSize(MptConfigurationPageIOUnit2AdapterOrdering, 4);
1188
1189/**
1190 * IO Unit page 2. - Read/Write.
1191 */
1192#pragma pack(1)
1193typedef struct MptConfigurationPageIOUnit2
1194{
1195 /** Union. */
1196 union
1197 {
1198 /** Byte view. */
1199 uint8_t abPageData[28];
1200 /** Field view. */
1201 struct
1202 {
1203 /** The omnipresent header. */
1204 MptConfigurationPageHeader Header;
1205 /** Reserved. */
1206 unsigned fReserved: 1;
1207 /** Flag whether Pause on error is enabled. */
1208 unsigned fPauseOnError: 1;
1209 /** Flag whether verbose mode is enabled. */
1210 unsigned fVerboseModeEnabled: 1;
1211 /** Set to disable color video. */
1212 unsigned fDisableColorVideo: 1;
1213 /** Flag whether int 40h is hooked. */
1214 unsigned fNotHookInt40h: 1;
1215 /** Reserved. */
1216 unsigned u3Reserved: 3;
1217 /** Reserved. */
1218 unsigned abReserved: 24;
1219 /** BIOS version. */
1220 uint32_t u32BIOSVersion;
1221 /** Adapter ordering. */
1222 MptConfigurationPageIOUnit2AdapterOrdering aAdapterOrder[4];
1223 } fields;
1224 } u;
1225} MptConfigurationPageIOUnit2, *PMptConfigurationPageIOUnit2;
1226#pragma pack()
1227AssertCompileSize(MptConfigurationPageIOUnit2, 28);
1228
1229/*
1230 * IO Unit page 3. - Read/Write.
1231 */
1232#pragma pack(1)
1233typedef struct MptConfigurationPageIOUnit3
1234{
1235 /** Union. */
1236 union
1237 {
1238 /** Byte view. */
1239 uint8_t abPageData[8];
1240 /** Field view. */
1241 struct
1242 {
1243 /** The omnipresent header. */
1244 MptConfigurationPageHeader Header;
1245 /** Number of GPIO values. */
1246 uint8_t u8GPIOCount;
1247 /** Reserved. */
1248 uint8_t abReserved[3];
1249 } fields;
1250 } u;
1251} MptConfigurationPageIOUnit3, *PMptConfigurationPageIOUnit3;
1252#pragma pack()
1253AssertCompileSize(MptConfigurationPageIOUnit3, 8);
1254
1255/**
1256 * IOC page 0. - Readonly
1257 */
1258#pragma pack(1)
1259typedef struct MptConfigurationPageIOC0
1260{
1261 /** Union. */
1262 union
1263 {
1264 /** Byte view. */
1265 uint8_t abPageData[28];
1266 /** Field view. */
1267 struct
1268 {
1269 /** The omnipresent header. */
1270 MptConfigurationPageHeader Header;
1271 /** Total ammount of NV memory in bytes. */
1272 uint32_t u32TotalNVStore;
1273 /** Number of free bytes in the NV store. */
1274 uint32_t u32FreeNVStore;
1275 /** PCI vendor ID. */
1276 uint16_t u16VendorId;
1277 /** PCI device ID. */
1278 uint16_t u16DeviceId;
1279 /** PCI revision ID. */
1280 uint8_t u8RevisionId;
1281 /** Reserved. */
1282 uint8_t abReserved[3];
1283 /** PCI class code. */
1284 uint32_t u32ClassCode;
1285 /** Subsystem vendor Id. */
1286 uint16_t u16SubsystemVendorId;
1287 /** Subsystem Id. */
1288 uint16_t u16SubsystemId;
1289 } fields;
1290 } u;
1291} MptConfigurationPageIOC0, *PMptConfigurationPageIOC0;
1292#pragma pack()
1293AssertCompileSize(MptConfigurationPageIOC0, 28);
1294
1295/**
1296 * IOC page 1. - Read/Write
1297 */
1298#pragma pack(1)
1299typedef struct MptConfigurationPageIOC1
1300{
1301 /** Union. */
1302 union
1303 {
1304 /** Byte view. */
1305 uint8_t abPageData[16];
1306 /** Field view. */
1307 struct
1308 {
1309 /** The omnipresent header. */
1310 MptConfigurationPageHeader Header;
1311 /** Flag whether reply coalescing is enabled. */
1312 unsigned fReplyCoalescingEnabled: 1;
1313 /** Reserved. */
1314 unsigned u31Reserved: 31;
1315 /** Coalescing Timeout in microseconds. */
1316 unsigned u32CoalescingTimeout: 32;
1317 /** Coalescing depth. */
1318 unsigned u8CoalescingDepth: 8;
1319 /** Reserved. */
1320 unsigned u8Reserved0: 8;
1321 unsigned u8Reserved1: 8;
1322 unsigned u8Reserved2: 8;
1323 } fields;
1324 } u;
1325} MptConfigurationPageIOC1, *PMptConfigurationPageIOC1;
1326#pragma pack()
1327AssertCompileSize(MptConfigurationPageIOC1, 16);
1328
1329/**
1330 * IOC page 2. - Readonly
1331 */
1332#pragma pack(1)
1333typedef struct MptConfigurationPageIOC2
1334{
1335 /** Union. */
1336 union
1337 {
1338 /** Byte view. */
1339 uint8_t abPageData[12];
1340 /** Field view. */
1341 struct
1342 {
1343 /** The omnipresent header. */
1344 MptConfigurationPageHeader Header;
1345 /** Flag whether striping is supported. */
1346 unsigned fStripingSupported: 1;
1347 /** Flag whether enhanced mirroring is supported. */
1348 unsigned fEnhancedMirroringSupported: 1;
1349 /** Flag whether mirroring is supported. */
1350 unsigned fMirroringSupported: 1;
1351 /** Reserved. */
1352 unsigned u26Reserved: 26;
1353 /** Flag whether SES is supported. */
1354 unsigned fSESSupported: 1;
1355 /** Flag whether SAF-TE is supported. */
1356 unsigned fSAFTESupported: 1;
1357 /** Flag whether cross channel volumes are supported. */
1358 unsigned fCrossChannelVolumesSupported: 1;
1359 /** Number of active integrated RAID volumes. */
1360 unsigned u8NumActiveVolumes: 8;
1361 /** Maximum number of integrated RAID volumes supported. */
1362 unsigned u8MaxVolumes: 8;
1363 /** Number of active integrated RAID physical disks. */
1364 unsigned u8NumActivePhysDisks: 8;
1365 /** Maximum number of integrated RAID physical disks supported. */
1366 unsigned u8MaxPhysDisks: 8;
1367 /** RAID volumes... - not supported. */
1368 } fields;
1369 } u;
1370} MptConfigurationPageIOC2, *PMptConfigurationPageIOC2;
1371#pragma pack()
1372AssertCompileSize(MptConfigurationPageIOC2, 12);
1373
1374/**
1375 * IOC page 3. - Readonly
1376 */
1377#pragma pack(1)
1378typedef struct MptConfigurationPageIOC3
1379{
1380 /** Union. */
1381 union
1382 {
1383 /** Byte view. */
1384 uint8_t abPageData[8];
1385 /** Field view. */
1386 struct
1387 {
1388 /** The omnipresent header. */
1389 MptConfigurationPageHeader Header;
1390 /** Number of active integrated RAID physical disks. */
1391 uint8_t u8NumPhysDisks;
1392 /** Reserved. */
1393 uint8_t abReserved[3];
1394 } fields;
1395 } u;
1396} MptConfigurationPageIOC3, *PMptConfigurationPageIOC3;
1397#pragma pack()
1398AssertCompileSize(MptConfigurationPageIOC3, 8);
1399
1400/**
1401 * IOC page 4. - Read/Write
1402 */
1403#pragma pack(1)
1404typedef struct MptConfigurationPageIOC4
1405{
1406 /** Union. */
1407 union
1408 {
1409 /** Byte view. */
1410 uint8_t abPageData[8];
1411 /** Field view. */
1412 struct
1413 {
1414 /** The omnipresent header. */
1415 MptConfigurationPageHeader Header;
1416 /** Number of SEP entries in this page. */
1417 uint8_t u8ActiveSEP;
1418 /** Maximum number of SEp entries supported. */
1419 uint8_t u8MaxSEP;
1420 /** Reserved. */
1421 uint16_t u16Reserved;
1422 /** SEP entries... - not supported. */
1423 } fields;
1424 } u;
1425} MptConfigurationPageIOC4, *PMptConfigurationPageIOC4;
1426#pragma pack()
1427AssertCompileSize(MptConfigurationPageIOC4, 8);
1428
1429/**
1430 * IOC page 6. - Read/Write
1431 */
1432#pragma pack(1)
1433typedef struct MptConfigurationPageIOC6
1434{
1435 /** Union. */
1436 union
1437 {
1438 /** Byte view. */
1439 uint8_t abPageData[60];
1440 /** Field view. */
1441 struct
1442 {
1443 /** The omnipresent header. */
1444 MptConfigurationPageHeader Header;
1445 uint32_t u32CapabilitiesFlags;
1446 uint8_t u8MaxDrivesIS;
1447 uint8_t u8MaxDrivesIM;
1448 uint8_t u8MaxDrivesIME;
1449 uint8_t u8Reserved1;
1450 uint8_t u8MinDrivesIS;
1451 uint8_t u8MinDrivesIM;
1452 uint8_t u8MinDrivesIME;
1453 uint8_t u8Reserved2;
1454 uint8_t u8MaxGlobalHotSpares;
1455 uint8_t u8Reserved3;
1456 uint16_t u16Reserved4;
1457 uint32_t u32Reserved5;
1458 uint32_t u32SupportedStripeSizeMapIS;
1459 uint32_t u32SupportedStripeSizeMapIME;
1460 uint32_t u32Reserved6;
1461 uint8_t u8MetadataSize;
1462 uint8_t u8Reserved7;
1463 uint16_t u16Reserved8;
1464 uint16_t u16MaxBadBlockTableEntries;
1465 uint16_t u16Reserved9;
1466 uint16_t u16IRNvsramUsage;
1467 uint16_t u16Reserved10;
1468 uint32_t u32IRNvsramVersion;
1469 uint32_t u32Reserved11;
1470 } fields;
1471 } u;
1472} MptConfigurationPageIOC6, *PMptConfigurationPageIOC6;
1473#pragma pack()
1474AssertCompileSize(MptConfigurationPageIOC6, 60);
1475
1476/**
1477 * SCSI-SPI port page 0. - Readonly
1478 */
1479#pragma pack(1)
1480typedef struct MptConfigurationPageSCSISPIPort0
1481{
1482 /** Union. */
1483 union
1484 {
1485 /** Byte view. */
1486 uint8_t abPageData[12];
1487 /** Field view. */
1488 struct
1489 {
1490 /** The omnipresent header. */
1491 MptConfigurationPageHeader Header;
1492 /** Flag whether this port is information unit trnafsers capable. */
1493 unsigned fInformationUnitTransfersCapable: 1;
1494 /** Flag whether the port is DT (Dual Transfer) capable. */
1495 unsigned fDTCapable: 1;
1496 /** Flag whether the port is QAS (Quick Arbitrate and Select) capable. */
1497 unsigned fQASCapable: 1;
1498 /** Reserved. */
1499 unsigned u5Reserved1: 5;
1500 /** Minimum Synchronous transfer period. */
1501 unsigned u8MinimumSynchronousTransferPeriod: 8;
1502 /** Maximum synchronous offset. */
1503 unsigned u8MaximumSynchronousOffset: 8;
1504 /** Reserved. */
1505 unsigned u5Reserved2: 5;
1506 /** Flag whether indicating the width of the bus - 0 narrow and 1 for wide. */
1507 unsigned fWide: 1;
1508 /** Reserved */
1509 unsigned fReserved: 1;
1510 /** Flag whether the port is AIP (Asynchronous Information Protection) capable. */
1511 unsigned fAIPCapable: 1;
1512 /** Signaling Type. */
1513 unsigned u2SignalingType: 2;
1514 /** Reserved. */
1515 unsigned u30Reserved: 30;
1516 } fields;
1517 } u;
1518} MptConfigurationPageSCSISPIPort0, *PMptConfigurationPageSCSISPIPort0;
1519#pragma pack()
1520AssertCompileSize(MptConfigurationPageSCSISPIPort0, 12);
1521
1522/**
1523 * SCSI-SPI port page 1. - Read/Write
1524 */
1525#pragma pack(1)
1526typedef struct MptConfigurationPageSCSISPIPort1
1527{
1528 /** Union. */
1529 union
1530 {
1531 /** Byte view. */
1532 uint8_t abPageData[12];
1533 /** Field view. */
1534 struct
1535 {
1536 /** The omnipresent header. */
1537 MptConfigurationPageHeader Header;
1538 /** The SCSI ID of the port. */
1539 uint8_t u8SCSIID;
1540 /** Reserved. */
1541 uint8_t u8Reserved;
1542 /** Port response IDs Bit mask field. */
1543 uint16_t u16PortResponseIDsBitmask;
1544 /** Value for the on BUS timer. */
1545 uint32_t u32OnBusTimerValue;
1546 } fields;
1547 } u;
1548} MptConfigurationPageSCSISPIPort1, *PMptConfigurationPageSCSISPIPort1;
1549#pragma pack()
1550AssertCompileSize(MptConfigurationPageSCSISPIPort1, 12);
1551
1552/**
1553 * Device settings for one device.
1554 */
1555#pragma pack(1)
1556typedef struct MptDeviceSettings
1557{
1558 /** Timeout for I/O in seconds. */
1559 unsigned u8Timeout: 8;
1560 /** Minimum synchronous factor. */
1561 unsigned u8SyncFactor: 8;
1562 /** Flag whether disconnect is enabled. */
1563 unsigned fDisconnectEnable: 1;
1564 /** Flag whether Scan ID is enabled. */
1565 unsigned fScanIDEnable: 1;
1566 /** Flag whether Scan LUNs is enabled. */
1567 unsigned fScanLUNEnable: 1;
1568 /** Flag whether tagged queuing is enabled. */
1569 unsigned fTaggedQueuingEnabled: 1;
1570 /** Flag whether wide is enabled. */
1571 unsigned fWideDisable: 1;
1572 /** Flag whether this device is bootable. */
1573 unsigned fBootChoice: 1;
1574 /** Reserved. */
1575 unsigned u10Reserved: 10;
1576} MptDeviceSettings, *PMptDeviceSettings;
1577#pragma pack()
1578AssertCompileSize(MptDeviceSettings, 4);
1579
1580/**
1581 * SCSI-SPI port page 2. - Read/Write for the BIOS
1582 */
1583#pragma pack(1)
1584typedef struct MptConfigurationPageSCSISPIPort2
1585{
1586 /** Union. */
1587 union
1588 {
1589 /** Byte view. */
1590 uint8_t abPageData[76];
1591 /** Field view. */
1592 struct
1593 {
1594 /** The omnipresent header. */
1595 MptConfigurationPageHeader Header;
1596 /** Flag indicating the bus scan order. */
1597 unsigned fBusScanOrderHighToLow: 1;
1598 /** Reserved. */
1599 unsigned fReserved: 1;
1600 /** Flag whether SCSI Bus resets are avoided. */
1601 unsigned fAvoidSCSIBusResets: 1;
1602 /** Flag whether alternate CHS is used. */
1603 unsigned fAlternateCHS: 1;
1604 /** Flag whether termination is disabled. */
1605 unsigned fTerminationDisabled: 1;
1606 /** Reserved. */
1607 unsigned u27Reserved: 27;
1608 /** Host SCSI ID. */
1609 unsigned u4HostSCSIID: 4;
1610 /** Initialize HBA. */
1611 unsigned u2InitializeHBA: 2;
1612 /** Removeable media setting. */
1613 unsigned u2RemovableMediaSetting: 2;
1614 /** Spinup delay. */
1615 unsigned u4SpinupDelay: 4;
1616 /** Negotiating settings. */
1617 unsigned u2NegotitatingSettings: 2;
1618 /** Reserved. */
1619 unsigned u18Reserved: 18;
1620 /** Device Settings. */
1621 MptDeviceSettings aDeviceSettings[16];
1622 } fields;
1623 } u;
1624} MptConfigurationPageSCSISPIPort2, *PMptConfigurationPageSCSISPIPort2;
1625#pragma pack()
1626AssertCompileSize(MptConfigurationPageSCSISPIPort2, 76);
1627
1628/**
1629 * SCSI-SPI device page 0. - Readonly
1630 */
1631#pragma pack(1)
1632typedef struct MptConfigurationPageSCSISPIDevice0
1633{
1634 /** Union. */
1635 union
1636 {
1637 /** Byte view. */
1638 uint8_t abPageData[12];
1639 /** Field view. */
1640 struct
1641 {
1642 /** The omnipresent header. */
1643 MptConfigurationPageHeader Header;
1644 /** Negotiated Parameters. */
1645 /** Information Units enabled. */
1646 unsigned fInformationUnitsEnabled: 1;
1647 /** Dual Transfers Enabled. */
1648 unsigned fDTEnabled: 1;
1649 /** QAS enabled. */
1650 unsigned fQASEnabled: 1;
1651 /** Reserved. */
1652 unsigned u5Reserved1: 5;
1653 /** Synchronous Transfer period. */
1654 unsigned u8NegotiatedSynchronousTransferPeriod: 8;
1655 /** Synchronous offset. */
1656 unsigned u8NegotiatedSynchronousOffset: 8;
1657 /** Reserved. */
1658 unsigned u5Reserved2: 5;
1659 /** Width - 0 for narrow and 1 for wide. */
1660 unsigned fWide: 1;
1661 /** Reserved. */
1662 unsigned fReserved: 1;
1663 /** AIP enabled. */
1664 unsigned fAIPEnabled: 1;
1665 /** Flag whether negotiation occurred. */
1666 unsigned fNegotationOccured: 1;
1667 /** Flag whether a SDTR message was rejected. */
1668 unsigned fSDTRRejected: 1;
1669 /** Flag whether a WDTR message was rejected. */
1670 unsigned fWDTRRejected: 1;
1671 /** Flag whether a PPR message was rejected. */
1672 unsigned fPPRRejected: 1;
1673 /** Reserved. */
1674 unsigned u28Reserved: 28;
1675 } fields;
1676 } u;
1677} MptConfigurationPageSCSISPIDevice0, *PMptConfigurationPageSCSISPIDevice0;
1678#pragma pack()
1679AssertCompileSize(MptConfigurationPageSCSISPIDevice0, 12);
1680
1681/**
1682 * SCSI-SPI device page 1. - Read/Write
1683 */
1684#pragma pack(1)
1685typedef struct MptConfigurationPageSCSISPIDevice1
1686{
1687 /** Union. */
1688 union
1689 {
1690 /** Byte view. */
1691 uint8_t abPageData[16];
1692 /** Field view. */
1693 struct
1694 {
1695 /** The omnipresent header. */
1696 MptConfigurationPageHeader Header;
1697 /** Requested Parameters. */
1698 /** Information Units enable. */
1699 bool fInformationUnitsEnable: 1;
1700 /** Dual Transfers Enable. */
1701 bool fDTEnable: 1;
1702 /** QAS enable. */
1703 bool fQASEnable: 1;
1704 /** Reserved. */
1705 unsigned u5Reserved1: 5;
1706 /** Synchronous Transfer period. */
1707 unsigned u8NegotiatedSynchronousTransferPeriod: 8;
1708 /** Synchronous offset. */
1709 unsigned u8NegotiatedSynchronousOffset: 8;
1710 /** Reserved. */
1711 unsigned u5Reserved2: 5;
1712 /** Width - 0 for narrow and 1 for wide. */
1713 bool fWide: 1;
1714 /** Reserved. */
1715 bool fReserved1: 1;
1716 /** AIP enable. */
1717 bool fAIPEnable: 1;
1718 /** Reserved. */
1719 bool fReserved2: 1;
1720 /** WDTR disallowed. */
1721 bool fWDTRDisallowed: 1;
1722 /** SDTR disallowed. */
1723 bool fSDTRDisallowed: 1;
1724 /** Reserved. */
1725 unsigned u29Reserved: 29;
1726 } fields;
1727 } u;
1728} MptConfigurationPageSCSISPIDevice1, *PMptConfigurationPageSCSISPIDevice1;
1729#pragma pack()
1730AssertCompileSize(MptConfigurationPageSCSISPIDevice1, 16);
1731
1732/**
1733 * SCSI-SPI device page 2. - Read/Write
1734 */
1735#pragma pack(1)
1736typedef struct MptConfigurationPageSCSISPIDevice2
1737{
1738 /** Union. */
1739 union
1740 {
1741 /** Byte view. */
1742 uint8_t abPageData[16];
1743 /** Field view. */
1744 struct
1745 {
1746 /** The omnipresent header. */
1747 MptConfigurationPageHeader Header;
1748 /** Reserved. */
1749 unsigned u4Reserved: 4;
1750 /** ISI enable. */
1751 unsigned fISIEnable: 1;
1752 /** Secondary driver enable. */
1753 unsigned fSecondaryDriverEnable: 1;
1754 /** Reserved. */
1755 unsigned fReserved: 1;
1756 /** Slew reate controler. */
1757 unsigned u3SlewRateControler: 3;
1758 /** Primary drive strength controler. */
1759 unsigned u3PrimaryDriveStrengthControl: 3;
1760 /** Secondary drive strength controler. */
1761 unsigned u3SecondaryDriveStrengthControl: 3;
1762 /** Reserved. */
1763 unsigned u12Reserved: 12;
1764 /** XCLKH_ST. */
1765 unsigned fXCLKH_ST: 1;
1766 /** XCLKS_ST. */
1767 unsigned fXCLKS_ST: 1;
1768 /** XCLKH_DT. */
1769 unsigned fXCLKH_DT: 1;
1770 /** XCLKS_DT. */
1771 unsigned fXCLKS_DT: 1;
1772 /** Parity pipe select. */
1773 unsigned u2ParityPipeSelect: 2;
1774 /** Reserved. */
1775 unsigned u30Reserved: 30;
1776 /** Data bit pipeline select. */
1777 unsigned u32DataPipelineSelect: 32;
1778 } fields;
1779 } u;
1780} MptConfigurationPageSCSISPIDevice2, *PMptConfigurationPageSCSISPIDevice2;
1781#pragma pack()
1782AssertCompileSize(MptConfigurationPageSCSISPIDevice2, 16);
1783
1784/**
1785 * SCSI-SPI device page 3 (Revision G). - Readonly
1786 */
1787#pragma pack(1)
1788typedef struct MptConfigurationPageSCSISPIDevice3
1789{
1790 /** Union. */
1791 union
1792 {
1793 /** Byte view. */
1794 uint8_t abPageData[12];
1795 /** Field view. */
1796 struct
1797 {
1798 /** The omnipresent header. */
1799 MptConfigurationPageHeader Header;
1800 /** Number of times the IOC rejected a message because it doesn't support the operation. */
1801 uint16_t u16MsgRejectCount;
1802 /** Number of times the SCSI bus entered an invalid operation state. */
1803 uint16_t u16PhaseErrorCount;
1804 /** Number of parity errors. */
1805 uint16_t u16ParityCount;
1806 /** Reserved. */
1807 uint16_t u16Reserved;
1808 } fields;
1809 } u;
1810} MptConfigurationPageSCSISPIDevice3, *PMptConfigurationPageSCSISPIDevice3;
1811#pragma pack()
1812AssertCompileSize(MptConfigurationPageSCSISPIDevice3, 12);
1813
1814/**
1815 * Structure of all supported pages.
1816 */
1817typedef struct MptConfigurationPagesSupported
1818{
1819 MptConfigurationPageManufacturing0 ManufacturingPage0;
1820 MptConfigurationPageManufacturing1 ManufacturingPage1;
1821 MptConfigurationPageManufacturing2 ManufacturingPage2;
1822 MptConfigurationPageManufacturing3 ManufacturingPage3;
1823 MptConfigurationPageManufacturing4 ManufacturingPage4;
1824 MptConfigurationPageIOUnit0 IOUnitPage0;
1825 MptConfigurationPageIOUnit1 IOUnitPage1;
1826 MptConfigurationPageIOUnit2 IOUnitPage2;
1827 MptConfigurationPageIOUnit3 IOUnitPage3;
1828 MptConfigurationPageIOC0 IOCPage0;
1829 MptConfigurationPageIOC1 IOCPage1;
1830 MptConfigurationPageIOC2 IOCPage2;
1831 MptConfigurationPageIOC3 IOCPage3;
1832 MptConfigurationPageIOC4 IOCPage4;
1833 MptConfigurationPageIOC6 IOCPage6;
1834 struct
1835 {
1836 MptConfigurationPageSCSISPIPort0 SCSISPIPortPage0;
1837 MptConfigurationPageSCSISPIPort1 SCSISPIPortPage1;
1838 MptConfigurationPageSCSISPIPort2 SCSISPIPortPage2;
1839 } aPortPages[1]; /* Currently only one port supported. */
1840 struct
1841 {
1842 struct
1843 {
1844 MptConfigurationPageSCSISPIDevice0 SCSISPIDevicePage0;
1845 MptConfigurationPageSCSISPIDevice1 SCSISPIDevicePage1;
1846 MptConfigurationPageSCSISPIDevice2 SCSISPIDevicePage2;
1847 MptConfigurationPageSCSISPIDevice3 SCSISPIDevicePage3;
1848 } aDevicePages[LSILOGIC_DEVICES_MAX];
1849 } aBuses[1]; /* Only one bus at the moment. */
1850} MptConfigurationPagesSupported, *PMptConfigurationPagesSupported;
1851
1852/**
1853 * Possible SG element types.
1854 */
1855enum MPTSGENTRYTYPE
1856{
1857 MPTSGENTRYTYPE_TRANSACTION_CONTEXT = 0x00,
1858 MPTSGENTRYTYPE_SIMPLE = 0x01,
1859 MPTSGENTRYTYPE_CHAIN = 0x03
1860};
1861
1862/**
1863 * Reply data.
1864 */
1865typedef struct LSILOGICSCSIREPLY
1866{
1867 /** Lower 32 bits of the reply address in memory. */
1868 uint32_t u32HostMFALowAddress;
1869 /** Full address of the reply in guest memory. */
1870 RTGCPHYS GCPhysReplyAddress;
1871 /** Size of the reply. */
1872 uint32_t cbReply;
1873 /** Different views to the reply depending on the request type. */
1874 MptReplyUnion Reply;
1875} LSILOGICSCSIREPLY, *PLSILOGICSCSIREPLY;
1876
1877/*
1878 * State of a device attached to the buslogic host adapter.
1879 */
1880typedef struct LSILOGICDEVICE
1881{
1882 /** Pointer to the owning lsilogic device instance. - R3 pointer */
1883 R3PTRTYPE(struct LSILOGICSCSI *) pLsiLogicR3;
1884 /** Pointer to the owning lsilogic device instance. - R0 pointer */
1885 R0PTRTYPE(struct LSILOGICSCSI *) pLsiLogicR0;
1886 /** Pointer to the owning lsilogic device instance. - RC pointer */
1887 RCPTRTYPE(struct LSILOGICSCSI *) pLsiLogicRC;
1888
1889 /** LUN of the device. */
1890 RTUINT iLUN;
1891 /** Number of outstanding tasks on the port. */
1892 volatile uint32_t cOutstandingRequests;
1893
1894#if HC_ARCH_BITS == 64
1895 uint32_t Alignment0;
1896#endif
1897
1898 /** Our base interace. */
1899 PDMIBASE IBase;
1900 /** SCSI port interface. */
1901 PDMISCSIPORT ISCSIPort;
1902 /** Led interface. */
1903 PDMILEDPORTS ILed;
1904 /** Pointer to the attached driver's base interface. */
1905 R3PTRTYPE(PPDMIBASE) pDrvBase;
1906 /** Pointer to the underlying SCSI connector interface. */
1907 R3PTRTYPE(PPDMISCSICONNECTOR) pDrvSCSIConnector;
1908 /** The status LED state for this device. */
1909 PDMLED Led;
1910
1911} LSILOGICDEVICE, *PLSILOGICDEVICE;
1912
1913/**
1914 * Defined states that the SCSI controller can have.
1915 */
1916typedef enum LSILOGICSTATE
1917{
1918 /** Reset state. */
1919 LSILOGICSTATE_RESET = 0x00,
1920 /** Ready state. */
1921 LSILOGICSTATE_READY = 0x01,
1922 /** Operational state. */
1923 LSILOGICSTATE_OPERATIONAL = 0x02,
1924 /** Fault state. */
1925 LSILOGICSTATE_FAULT = 0x04,
1926 /** 32bit size hack */
1927 LSILOGICSTATE_32BIT_HACK = 0x7fffffff
1928} LSILOGICSTATE;
1929
1930/**
1931 * Which entity needs to initialize the controller
1932 * to get into the operational state.
1933 */
1934typedef enum LSILOGICWHOINIT
1935{
1936 /** Not initialized. */
1937 LSILOGICWHOINIT_NOT_INITIALIZED = 0x00,
1938 /** System BIOS. */
1939 LSILOGICWHOINIT_SYSTEM_BIOS = 0x01,
1940 /** ROM Bios. */
1941 LSILOGICWHOINIT_ROM_BIOS = 0x02,
1942 /** PCI Peer. */
1943 LSILOGICWHOINIT_PCI_PEER = 0x03,
1944 /** Host driver. */
1945 LSILOGICWHOINIT_HOST_DRIVER = 0x04,
1946 /** Manufacturing. */
1947 LSILOGICWHOINIT_MANUFACTURING = 0x05,
1948 /** 32bit size hack. */
1949 LSILOGICWHOINIT_32BIT_HACK = 0x7fffffff
1950} LSILOGICWHOINIT;
1951
1952
1953/**
1954 * IOC status codes.
1955 */
1956#define LSILOGIC_IOCSTATUS_SUCCESS 0x0000
1957#define LSILOGIC_IOCSTATUS_INVALID_FUNCTION 0x0001
1958#define LSILOGIC_IOCSTATUS_BUSY 0x0002
1959#define LSILOGIC_IOCSTATUS_INVALID_SGL 0x0003
1960#define LSILOGIC_IOCSTATUS_INTERNAL_ERROR 0x0004
1961#define LSILOGIC_IOCSTATUS_RESERVED 0x0005
1962#define LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES 0x0006
1963#define LSILOGIC_IOCSTATUS_INVALID_FIELD 0x0007
1964#define LSILOGIC_IOCSTATUS_INVALID_STATE 0x0008
1965#define LSILOGIC_IOCSTATUS_OP_STATE_NOT_SUPPOTED 0x0009
1966
1967/**
1968 * Device instance data for the emulated
1969 * SCSI controller.
1970 */
1971typedef struct LSILOGICSCSI
1972{
1973 /** PCI device structure. */
1974 PCIDEVICE PciDev;
1975 /** Pointer to the device instance. - R3 ptr. */
1976 PPDMDEVINSR3 pDevInsR3;
1977 /** Pointer to the device instance. - R0 ptr. */
1978 PPDMDEVINSR0 pDevInsR0;
1979 /** Pointer to the device instance. - RC ptr. */
1980 PPDMDEVINSRC pDevInsRC;
1981
1982 /** Flag whether the GC part of the device is enabled. */
1983 bool fGCEnabled;
1984 /** Flag whether the R0 part of the device is enabled. */
1985 bool fR0Enabled;
1986
1987 /** The state the controller is currently in. */
1988 LSILOGICSTATE enmState;
1989 /** Who needs to init the driver to get into operational state. */
1990 LSILOGICWHOINIT enmWhoInit;
1991 /** Flag whether we are in doorbell function. */
1992 bool fDoorbellInProgress;
1993 /** Flag whether diagnostic access is enabled. */
1994 bool fDiagnosticEnabled;
1995
1996 /** Flag whether a notification was send to R3. */
1997 bool fNotificationSend;
1998
1999 /** Flag whether the guest enabled event notification from the IOC. */
2000 bool fEventNotificationEnabled;
2001
2002#if HC_ARCH_BITS == 64
2003 uint32_t Alignment0;
2004#endif
2005
2006 /** Queue to send tasks to R3. - R3 ptr */
2007 R3PTRTYPE(PPDMQUEUE) pNotificationQueueR3;
2008 /** Queue to send tasks to R3. - R0 ptr */
2009 R0PTRTYPE(PPDMQUEUE) pNotificationQueueR0;
2010 /** Queue to send tasks to R3. - RC ptr */
2011 RCPTRTYPE(PPDMQUEUE) pNotificationQueueRC;
2012
2013#if HC_ARCH_BITS == 64
2014 uint32_t Alignment1;
2015#endif
2016
2017 /** States for attached devices. */
2018 LSILOGICDEVICE aDeviceStates[LSILOGIC_DEVICES_MAX];
2019
2020 /** MMIO address the device is mapped to. */
2021 RTGCPHYS GCPhysMMIOBase;
2022 /** I/O port address the device is mapped to. */
2023 RTIOPORT IOPortBase;
2024
2025 /** Interrupt mask. */
2026 volatile uint32_t uInterruptMask;
2027 /** Interrupt status register. */
2028 volatile uint32_t uInterruptStatus;
2029
2030 /** Buffer for messages which are passed
2031 * through the doorbell using the
2032 * handshake method. */
2033 uint32_t aMessage[sizeof(MptConfigurationRequest)];
2034 /** Actual position in the buffer. */
2035 uint32_t iMessage;
2036 /** Size of the message which is given in the doorbell message in dwords. */
2037 uint32_t cMessage;
2038
2039 /** Reply buffer. */
2040 MptReplyUnion ReplyBuffer;
2041 /** Next entry to read. */
2042 uint32_t uNextReplyEntryRead;
2043 /** Size of the reply in the buffer in 16bit words. */
2044 uint32_t cReplySize;
2045
2046 /** The fault code of the I/O controller if we are in the fault state. */
2047 uint16_t u16IOCFaultCode;
2048
2049 /** Upper 32 bits of the moessage frame address to locate requests in guest memory. */
2050 uint32_t u32HostMFAHighAddr;
2051 /** Upper 32 bits of the sense buffer address. */
2052 uint32_t u32SenseBufferHighAddr;
2053 /** Maximum number of devices the driver reported he can handle. */
2054 uint8_t cMaxDevices;
2055 /** Maximum number of buses the driver reported he can handle. */
2056 uint8_t cMaxBuses;
2057 /** Current size of reply message frames in the guest. */
2058 uint16_t cbReplyFrame;
2059
2060 /** Next key to write in the sequence to get access
2061 * to diagnostic memory. */
2062 uint32_t iDiagnosticAccess;
2063
2064 /** Number entries allocated for the reply queue. */
2065 uint32_t cReplyQueueEntries;
2066 /** Number entries allocated for the outstanding request queue. */
2067 uint32_t cRequestQueueEntries;
2068
2069 /** Critical section protecting the reply post queue. */
2070 PDMCRITSECT ReplyPostQueueCritSect;
2071 /** Critical section protecting the reply free queue. */
2072 PDMCRITSECT ReplyFreeQueueCritSect;
2073
2074#if HC_ARCH_BITS == 64
2075 uint32_t Alignment2;
2076#endif
2077
2078 /** Pointer to the start of the reply free queue - R3. */
2079 R3PTRTYPE(volatile uint32_t *) pReplyFreeQueueBaseR3;
2080 /** Pointer to the start of the reply post queue - R3. */
2081 R3PTRTYPE(volatile uint32_t *) pReplyPostQueueBaseR3;
2082 /** Pointer to the start of the request queue - R3. */
2083 R3PTRTYPE(volatile uint32_t *) pRequestQueueBaseR3;
2084
2085 /** Pointer to the start of the reply queue - R0. */
2086 R0PTRTYPE(volatile uint32_t *) pReplyFreeQueueBaseR0;
2087 /** Pointer to the start of the reply queue - R0. */
2088 R0PTRTYPE(volatile uint32_t *) pReplyPostQueueBaseR0;
2089 /** Pointer to the start of the request queue - R0. */
2090 R0PTRTYPE(volatile uint32_t *) pRequestQueueBaseR0;
2091
2092 /** Pointer to the start of the reply queue - RC. */
2093 RCPTRTYPE(volatile uint32_t *) pReplyFreeQueueBaseRC;
2094 /** Pointer to the start of the reply queue - RC. */
2095 RCPTRTYPE(volatile uint32_t *) pReplyPostQueueBaseRC;
2096 /** Pointer to the start of the request queue - RC. */
2097 RCPTRTYPE(volatile uint32_t *) pRequestQueueBaseRC;
2098
2099 /** Next free entry in the reply queue the guest can write a address to. */
2100 volatile uint32_t uReplyFreeQueueNextEntryFreeWrite;
2101 /** Next valid entry the controller can read a valid address for reply frames from. */
2102 volatile uint32_t uReplyFreeQueueNextAddressRead;
2103
2104 /** Next free entry in the reply queue the guest can write a address to. */
2105 volatile uint32_t uReplyPostQueueNextEntryFreeWrite;
2106 /** Next valid entry the controller can read a valid address for reply frames from. */
2107 volatile uint32_t uReplyPostQueueNextAddressRead;
2108
2109 /** Next free entry the guest can write a address to a request frame to. */
2110 volatile uint32_t uRequestQueueNextEntryFreeWrite;
2111 /** Next valid entry the controller can read a valid address for request frames from. */
2112 volatile uint32_t uRequestQueueNextAddressRead;
2113
2114 /** Configuration pages. */
2115 MptConfigurationPagesSupported ConfigurationPages;
2116
2117 /** BIOS emulation. */
2118 VBOXSCSI VBoxSCSI;
2119
2120 /** Cache for allocated tasks. */
2121 R3PTRTYPE(PRTOBJCACHE) pTaskCache;
2122
2123 /** The base interface */
2124 PDMIBASE IBase;
2125 /** Status Port - Leds interface. */
2126 PDMILEDPORTS ILeds;
2127 /** Partner of ILeds. */
2128 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
2129} LSILOGISCSI, *PLSILOGICSCSI;
2130
2131#define LSILOGIC_PCI_SPACE_IO_SIZE 256
2132#define LSILOGIC_PCI_SPACE_MEM_SIZE 128 * _1K
2133
2134#define LSILOGIC_REG_DOORBELL 0x00
2135# define LSILOGIC_REG_DOORBELL_SET_STATE(enmState) (((enmState) & 0x0f) << 28)
2136# define LSILOGIC_REG_DOORBELL_SET_USED(fUsed) (((fUsed) ? 1 : 0) << 27)
2137# define LSILOGIC_REG_DOORBELL_SET_WHOINIT(enmWhoInit) (((enmWhoInit) & 0x07) << 24)
2138# define LSILOGIC_REG_DOORBELL_SET_FAULT_CODE(u16Code) (u16Code)
2139# define LSILOGIC_REG_DOORBELL_GET_FUNCTION(x) (((x) & 0xff000000) >> 24)
2140# define LSILOGIC_REG_DOORBELL_GET_SIZE(x) (((x) & 0x00ff0000) >> 16)
2141
2142#define LSILOGIC_REG_WRITE_SEQUENCE 0x04
2143#define LSILOGIC_REG_HOST_DIAGNOSTIC 0x08
2144#define LSILOGIC_REG_TEST_BASE_ADDRESS 0x0c
2145#define LSILOGIC_REG_DIAG_RW_DATA 0x10
2146#define LSILOGIC_REG_DIAG_RW_ADDRESS 0x14
2147
2148#define LSILOGIC_REG_HOST_INTR_STATUS 0x30
2149# define LSILOGIC_REG_HOST_INTR_STATUS_W_MASK (RT_BIT(3))
2150# define LSILOGIC_REG_HOST_INTR_STATUS_DOORBELL_STS (RT_BIT(31))
2151# define LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR (RT_BIT(3))
2152# define LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL (RT_BIT(0))
2153
2154#define LSILOGIC_REG_HOST_INTR_MASK 0x34
2155# define LSILOGIC_REG_HOST_INTR_MASK_W_MASK (RT_BIT(0) | RT_BIT(3) | RT_BIT(8) | RT_BIT(9))
2156# define LSILOGIC_REG_HOST_INTR_MASK_IRQ_ROUTING (RT_BIT(8) | RT_BIT(9))
2157# define LSILOGIC_REG_HOST_INTR_MASK_DOORBELL RT_BIT(0)
2158# define LSILOGIC_REG_HOST_INTR_MASK_REPLY RT_BIT(3)
2159
2160#define LSILOGIC_REG_REQUEST_QUEUE 0x40
2161#define LSILOGIC_REG_REPLY_QUEUE 0x44
2162
2163/* Functions which can be passed through the system doorbell. */
2164#define LSILOGIC_DOORBELL_FUNCTION_IOC_MSG_UNIT_RESET 0x40
2165#define LSILOGIC_DOORBELL_FUNCTION_IO_UNIT_RESET 0x41
2166#define LSILOGIC_DOORBELL_FUNCTION_HANDSHAKE 0x42
2167#define LSILOGIC_DOORBELL_FUNCTION_REPLY_FRAME_REMOVAL 0x43
2168
2169/**
2170 * Scatter gather list entry data.
2171 */
2172typedef struct LSILOGICTASKSTATESGENTRY
2173{
2174 /** Flag whether the buffer in the list is from the guest or an
2175 * allocated temporary buffer because the segments in the guest
2176 * are not sector aligned.
2177 */
2178 bool fGuestMemory;
2179 /** Flag whether the buffer contains data or is the destination for the transfer. */
2180 bool fBufferContainsData;
2181 /** Pointer to the start of the buffer. */
2182 void *pvBuf;
2183 /** Size of the buffer. */
2184 uint32_t cbBuf;
2185 /** Flag dependent data. */
2186 union
2187 {
2188 /** Data to handle direct mappings of guest buffers. */
2189 PGMPAGEMAPLOCK PageLock;
2190 /** The segment in the guest which is not sector aligned. */
2191 RTGCPHYS GCPhysAddrBufferUnaligned;
2192 } u;
2193} LSILOGICTASKSTATESGENTRY, *PLSILOGICTASKSTATESGENTRY;
2194
2195/**
2196 * Task state object which holds all neccessary data while
2197 * processing the request from the guest.
2198 */
2199typedef struct LSILOGICTASKSTATE
2200{
2201 /** Target device. */
2202 PLSILOGICDEVICE pTargetDevice;
2203 /** The message request from the guest. */
2204 MptRequestUnion GuestRequest;
2205 /** Reply message if the request produces one. */
2206 MptReplyUnion IOCReply;
2207 /** SCSI request structure for the SCSI driver. */
2208 PDMSCSIREQUEST PDMScsiRequest;
2209 /** Address of the message request frame in guests memory.
2210 * Used to read the S/G entries in the second step. */
2211 RTGCPHYS GCPhysMessageFrameAddr;
2212 /** Number of scatter gather list entries. */
2213 uint32_t cSGListEntries;
2214 /** How many entries would fit into the sg list. */
2215 uint32_t cSGListSize;
2216 /** How many times the list was too big. */
2217 uint32_t cSGListTooBig;
2218 /** Pointer to the first entry of the scatter gather list. */
2219 PPDMDATASEG pSGListHead;
2220 /** How many entries would fit into the sg info list. */
2221 uint32_t cSGInfoSize;
2222 /** Number of entries for the information entries. */
2223 uint32_t cSGInfoEntries;
2224 /** How many times the list was too big. */
2225 uint32_t cSGInfoTooBig;
2226 /** Pointer to the first mapping information entry. */
2227 PLSILOGICTASKSTATESGENTRY paSGEntries;
2228 /** Size of the temporary buffer for unaligned guest segments. */
2229 uint32_t cbBufferUnaligned;
2230 /** Pointer to the temporary buffer. */
2231 void *pvBufferUnaligned;
2232 /** Pointer to the sense buffer. */
2233 uint8_t abSenseBuffer[18];
2234 /** Flag whether the request was issued from the BIOS. */
2235 bool fBIOS;
2236} LSILOGICTASKSTATE, *PLSILOGICTASKSTATE;
2237
2238#ifndef VBOX_DEVICE_STRUCT_TESTCASE
2239
2240RT_C_DECLS_BEGIN
2241PDMBOTHCBDECL(int) lsilogicIOPortWrite (PPDMDEVINS pDevIns, void *pvUser,
2242 RTIOPORT Port, uint32_t u32, unsigned cb);
2243PDMBOTHCBDECL(int) lsilogicIOPortRead (PPDMDEVINS pDevIns, void *pvUser,
2244 RTIOPORT Port, uint32_t *pu32, unsigned cb);
2245PDMBOTHCBDECL(int) lsilogicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser,
2246 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
2247PDMBOTHCBDECL(int) lsilogicMMIORead(PPDMDEVINS pDevIns, void *pvUser,
2248 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
2249PDMBOTHCBDECL(int) lsilogicDiagnosticWrite(PPDMDEVINS pDevIns, void *pvUser,
2250 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
2251PDMBOTHCBDECL(int) lsilogicDiagnosticRead(PPDMDEVINS pDevIns, void *pvUser,
2252 RTGCPHYS GCPhysAddr, void *pv, unsigned cb);
2253#ifdef IN_RING3
2254static void lsilogicInitializeConfigurationPages(PLSILOGICSCSI pLsiLogic);
2255static int lsilogicProcessConfigurationRequest(PLSILOGICSCSI pLsiLogic, PMptConfigurationRequest pConfigurationReq,
2256 PMptConfigurationReply pReply);
2257#endif
2258RT_C_DECLS_END
2259
2260#define PDMIBASE_2_PLSILOGICDEVICE(pInterface) ( (PLSILOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICDEVICE, IBase)) )
2261#define PDMISCSIPORT_2_PLSILOGICDEVICE(pInterface) ( (PLSILOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICDEVICE, ISCSIPort)) )
2262#define PDMILEDPORTS_2_PLSILOGICDEVICE(pInterface) ( (PLSILOGICDEVICE)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICDEVICE, ILed)) )
2263#define LSILOGIC_RTGCPHYS_FROM_U32(Hi, Lo) ( (RTGCPHYS)RT_MAKE_U64(Lo, Hi) )
2264#define PDMIBASE_2_PLSILOGICSCSI(pInterface) ( (PLSILOGICSCSI)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICSCSI, IBase)) )
2265#define PDMILEDPORTS_2_PLSILOGICSCSI(pInterface) ( (PLSILOGICSCSI)((uintptr_t)(pInterface) - RT_OFFSETOF(LSILOGICSCSI, ILeds)) )
2266
2267/** Key sequence the guest has to write to enable access
2268 * to diagnostic memory. */
2269static const uint8_t g_lsilogicDiagnosticAccess[] = {0x04, 0x0b, 0x02, 0x07, 0x0d};
2270
2271/**
2272 * Updates the status of the interrupt pin of the device.
2273 *
2274 * @returns nothing.
2275 * @param pThis Pointer to the device instance data.
2276 */
2277static void lsilogicUpdateInterrupt(PLSILOGICSCSI pThis)
2278{
2279 uint32_t uIntSts;
2280
2281 LogFlowFunc(("Updating interrupts\n"));
2282
2283 /* Mask out doorbell status so that it does not affect interrupt updating. */
2284 uIntSts = (ASMAtomicReadU32(&pThis->uInterruptStatus) & ~LSILOGIC_REG_HOST_INTR_STATUS_DOORBELL_STS);
2285 /* Check maskable interrupts. */
2286 uIntSts &= ~(pThis->uInterruptMask & ~LSILOGIC_REG_HOST_INTR_MASK_IRQ_ROUTING);
2287
2288 if (uIntSts)
2289 {
2290 LogFlowFunc(("Setting interrupt\n"));
2291 PDMDevHlpPCISetIrq(pThis->CTX_SUFF(pDevIns), 0, 1);
2292 }
2293 else
2294 {
2295 LogFlowFunc(("Clearing interrupt\n"));
2296 PDMDevHlpPCISetIrq(pThis->CTX_SUFF(pDevIns), 0, 0);
2297 }
2298}
2299
2300/**
2301 * Sets a given interrupt status bit in the status register and
2302 * updates the interupt status.
2303 *
2304 * @returns nothing.
2305 * @param pLsiLogic Pointer to the device instance.
2306 * @param uStatus The status bit to set.
2307 */
2308DECLINLINE(void) lsilogicSetInterrupt(PLSILOGICSCSI pLsiLogic, uint32_t uStatus)
2309{
2310 ASMAtomicOrU32(&pLsiLogic->uInterruptStatus, uStatus);
2311 lsilogicUpdateInterrupt(pLsiLogic);
2312}
2313
2314/**
2315 * Clears a given interrupt status bit in the status register and
2316 * updates the interupt status.
2317 *
2318 * @returns nothing.
2319 * @param pLsiLogic Pointer to the device instance.
2320 * @param uStatus The status bit to set.
2321 */
2322DECLINLINE(void) lsilogicClearInterrupt(PLSILOGICSCSI pLsiLogic, uint32_t uStatus)
2323{
2324 ASMAtomicAndU32(&pLsiLogic->uInterruptStatus, ~uStatus);
2325 lsilogicUpdateInterrupt(pLsiLogic);
2326}
2327
2328/**
2329 * Sets the I/O controller into fault state and sets the fault code.
2330 *
2331 * @returns nothing
2332 * @param pLsiLogic Pointer to the controller device instance.
2333 * @param uIOCFaultCode Fault code to set.
2334 */
2335DECLINLINE(void) lsilogicSetIOCFaultCode(PLSILOGICSCSI pLsiLogic, uint16_t uIOCFaultCode)
2336{
2337 if (pLsiLogic->enmState != LSILOGICSTATE_FAULT)
2338 {
2339 Log(("%s: Setting I/O controller into FAULT state: uIOCFaultCode=%u\n", __FUNCTION__, uIOCFaultCode));
2340 pLsiLogic->enmState = LSILOGICSTATE_FAULT;
2341 pLsiLogic->u16IOCFaultCode = uIOCFaultCode;
2342 }
2343 else
2344 {
2345 Log(("%s: We are already in FAULT state\n"));
2346 }
2347}
2348
2349#ifdef IN_RING3
2350/**
2351 * Performs a hard reset on the controller.
2352 *
2353 * @returns VBox status code.
2354 * @param pThis Pointer to the device instance to initialize.
2355 */
2356static int lsilogicHardReset(PLSILOGICSCSI pThis)
2357{
2358 pThis->enmState = LSILOGICSTATE_RESET;
2359
2360 /* The interrupts are masked out. */
2361 pThis->uInterruptMask |= LSILOGIC_REG_HOST_INTR_MASK_DOORBELL |
2362 LSILOGIC_REG_HOST_INTR_MASK_REPLY;
2363 /* Reset interrupt states. */
2364 pThis->uInterruptStatus = 0;
2365 lsilogicUpdateInterrupt(pThis);
2366
2367 /* Reset the queues. */
2368 pThis->uReplyFreeQueueNextEntryFreeWrite = 0;
2369 pThis->uReplyFreeQueueNextAddressRead = 0;
2370 pThis->uReplyPostQueueNextEntryFreeWrite = 0;
2371 pThis->uReplyPostQueueNextAddressRead = 0;
2372 pThis->uRequestQueueNextEntryFreeWrite = 0;
2373 pThis->uRequestQueueNextAddressRead = 0;
2374
2375 /* Disable diagnostic access. */
2376 pThis->iDiagnosticAccess = 0;
2377
2378 /* Set default values. */
2379 pThis->cMaxDevices = LSILOGIC_DEVICES_MAX;
2380 pThis->cMaxBuses = 1;
2381 pThis->cbReplyFrame = 128; /* @todo Figure out where it is needed. */
2382 /* @todo: Put stuff to reset here. */
2383
2384 lsilogicInitializeConfigurationPages(pThis);
2385
2386 /* Mark that we finished performing the reset. */
2387 pThis->enmState = LSILOGICSTATE_READY;
2388 return VINF_SUCCESS;
2389}
2390
2391/**
2392 * Finishes a context reply.
2393 *
2394 * @returns nothing
2395 * @param pLsiLogic Pointer to the device instance
2396 * @param u32MessageContext The message context ID to post.
2397 */
2398static void lsilogicFinishContextReply(PLSILOGICSCSI pLsiLogic, uint32_t u32MessageContext)
2399{
2400 int rc;
2401 AssertMsg(!pLsiLogic->fDoorbellInProgress, ("We are in a doorbell function\n"));
2402
2403 /* Write message context ID into reply post queue. */
2404 rc = PDMCritSectEnter(&pLsiLogic->ReplyPostQueueCritSect, VINF_SUCCESS);
2405 AssertRC(rc);
2406
2407#if 0
2408 /* Check for a entry in the queue. */
2409 if (RT_UNLIKELY(pLsiLogic->uReplyPostQueueNextAddressRead != pLsiLogic->uReplyPostQueueNextEntryFreeWrite))
2410 {
2411 /* Set error code. */
2412 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
2413 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
2414 return;
2415 }
2416#endif
2417
2418 /* We have a context reply. */
2419 ASMAtomicWriteU32(&pLsiLogic->CTX_SUFF(pReplyPostQueueBase)[pLsiLogic->uReplyPostQueueNextEntryFreeWrite], u32MessageContext);
2420 ASMAtomicIncU32(&pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
2421 pLsiLogic->uReplyPostQueueNextEntryFreeWrite %= pLsiLogic->cReplyQueueEntries;
2422
2423 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
2424
2425 /* Set interrupt. */
2426 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
2427}
2428#endif /* IN_RING3 */
2429
2430/**
2431 * Takes neccessary steps to finish a reply frame.
2432 *
2433 * @returns nothing
2434 * @param pLsiLogic Pointer to the device instance
2435 * @param pReply Pointer to the reply message.
2436 * @param fForceReplyFifo Flag whether the use of the reply post fifo is forced.
2437 */
2438static void lsilogicFinishAddressReply(PLSILOGICSCSI pLsiLogic, PMptReplyUnion pReply, bool fForceReplyFifo)
2439{
2440 /*
2441 * If we are in a doorbell function we set the reply size now and
2442 * set the system doorbell status interrupt to notify the guest that
2443 * we are ready to send the reply.
2444 */
2445 if (pLsiLogic->fDoorbellInProgress && !fForceReplyFifo)
2446 {
2447 /* Set size of the reply in 16bit words. The size in the reply is in 32bit dwords. */
2448 pLsiLogic->cReplySize = pReply->Header.u8MessageLength * 2;
2449 Log(("%s: cReplySize=%u\n", __FUNCTION__, pLsiLogic->cReplySize));
2450 pLsiLogic->uNextReplyEntryRead = 0;
2451 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
2452 }
2453 else
2454 {
2455 /*
2456 * The reply queues are only used if the request was fetched from the request queue.
2457 * Requests from the request queue are always transferred to R3. So it is not possible
2458 * that this case happens in R0 or GC.
2459 */
2460#ifdef IN_RING3
2461 int rc;
2462 /* Grab a free reply message from the queue. */
2463 rc = PDMCritSectEnter(&pLsiLogic->ReplyFreeQueueCritSect, VINF_SUCCESS);
2464 AssertRC(rc);
2465
2466#if 0
2467 /* Check for a free reply frame. */
2468 if (RT_UNLIKELY(pLsiLogic->uReplyFreeQueueNextAddressRead != pLsiLogic->uReplyFreeQueueNextEntryFreeWrite))
2469 {
2470 /* Set error code. */
2471 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
2472 PDMCritSectLeave(&pLsiLogic->ReplyFreeQueueCritSect);
2473 return;
2474 }
2475#endif
2476
2477 uint32_t u32ReplyFrameAddressLow = pLsiLogic->CTX_SUFF(pReplyFreeQueueBase)[pLsiLogic->uReplyFreeQueueNextAddressRead];
2478
2479 pLsiLogic->uReplyFreeQueueNextAddressRead++;
2480 pLsiLogic->uReplyFreeQueueNextAddressRead %= pLsiLogic->cReplyQueueEntries;
2481
2482 PDMCritSectLeave(&pLsiLogic->ReplyFreeQueueCritSect);
2483
2484 /* Build 64bit physical address. */
2485 RTGCPHYS GCPhysReplyMessage = LSILOGIC_RTGCPHYS_FROM_U32(pLsiLogic->u32HostMFAHighAddr, u32ReplyFrameAddressLow);
2486 size_t cbReplyCopied = (pLsiLogic->cbReplyFrame < sizeof(MptReplyUnion)) ? pLsiLogic->cbReplyFrame : sizeof(MptReplyUnion);
2487
2488 /* Write reply to guest memory. */
2489 PDMDevHlpPhysWrite(pLsiLogic->CTX_SUFF(pDevIns), GCPhysReplyMessage, pReply, cbReplyCopied);
2490
2491 /* Write low 32bits of reply frame into post reply queue. */
2492 rc = PDMCritSectEnter(&pLsiLogic->ReplyPostQueueCritSect, VINF_SUCCESS);
2493 AssertRC(rc);
2494
2495#if 0
2496 /* Check for a entry in the queue. */
2497 if (RT_UNLIKELY(pLsiLogic->uReplyPostQueueNextAddressRead != pLsiLogic->uReplyPostQueueNextEntryFreeWrite))
2498 {
2499 /* Set error code. */
2500 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INSUFFICIENT_RESOURCES);
2501 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
2502 return;
2503 }
2504#endif
2505
2506 /* We have a address reply. Set the 31th bit to indicate that. */
2507 ASMAtomicWriteU32(&pLsiLogic->CTX_SUFF(pReplyPostQueueBase)[pLsiLogic->uReplyPostQueueNextEntryFreeWrite],
2508 RT_BIT(31) | (u32ReplyFrameAddressLow >> 1));
2509 ASMAtomicIncU32(&pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
2510 pLsiLogic->uReplyPostQueueNextEntryFreeWrite %= pLsiLogic->cReplyQueueEntries;
2511
2512 PDMCritSectLeave(&pLsiLogic->ReplyPostQueueCritSect);
2513
2514 if (fForceReplyFifo)
2515 {
2516 pLsiLogic->fDoorbellInProgress = false;
2517 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
2518 }
2519
2520 /* Set interrupt. */
2521 lsilogicSetInterrupt(pLsiLogic, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
2522#else
2523 AssertMsgFailed(("This is not allowed to happen.\n"));
2524#endif
2525 }
2526}
2527
2528#ifdef IN_RING3
2529/**
2530 * Processes a given Request from the guest
2531 *
2532 * @returns VBox status code.
2533 * @param pLsiLogic Pointer to the device instance.
2534 * @param pMessageHdr Pointer to the message header of the request.
2535 * @param pReply Pointer to the reply.
2536 */
2537static int lsilogicProcessMessageRequest(PLSILOGICSCSI pLsiLogic, PMptMessageHdr pMessageHdr, PMptReplyUnion pReply)
2538{
2539 int rc = VINF_SUCCESS;
2540 bool fForceReplyPostFifo = false;
2541
2542#ifdef DEBUG
2543 if (pMessageHdr->u8Function < RT_ELEMENTS(g_apszMPTFunctionNames))
2544 Log(("Message request function: %s\n", g_apszMPTFunctionNames[pMessageHdr->u8Function]));
2545 else
2546 Log(("Message request function: <unknown>\n"));
2547#endif
2548
2549 memset(pReply, 0, sizeof(MptReplyUnion));
2550
2551 switch (pMessageHdr->u8Function)
2552 {
2553 case MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT:
2554 {
2555 PMptSCSITaskManagementRequest pTaskMgmtReq = (PMptSCSITaskManagementRequest)pMessageHdr;
2556
2557 pReply->SCSITaskManagement.u8MessageLength = 6; /* 6 32bit dwords. */
2558 pReply->SCSITaskManagement.u8TaskType = pTaskMgmtReq->u8TaskType;
2559 pReply->SCSITaskManagement.u32TerminationCount = 0;
2560 //AssertMsgFailed(("todo\n"));
2561 fForceReplyPostFifo = true;
2562 break;
2563 }
2564 case MPT_MESSAGE_HDR_FUNCTION_IOC_INIT:
2565 {
2566 /*
2567 * This request sets the I/O controller to the
2568 * operational state.
2569 */
2570 PMptIOCInitRequest pIOCInitReq = (PMptIOCInitRequest)pMessageHdr;
2571
2572 /* Update configuration values. */
2573 pLsiLogic->enmWhoInit = (LSILOGICWHOINIT)pIOCInitReq->u8WhoInit;
2574 pLsiLogic->cbReplyFrame = pIOCInitReq->u16ReplyFrameSize;
2575 pLsiLogic->cMaxBuses = pIOCInitReq->u8MaxBuses;
2576 pLsiLogic->cMaxDevices = pIOCInitReq->u8MaxDevices;
2577 pLsiLogic->u32HostMFAHighAddr = pIOCInitReq->u32HostMfaHighAddr;
2578 pLsiLogic->u32SenseBufferHighAddr = pIOCInitReq->u32SenseBufferHighAddr;
2579
2580 if (pLsiLogic->enmState == LSILOGICSTATE_READY)
2581 {
2582 pLsiLogic->enmState = LSILOGICSTATE_OPERATIONAL;
2583 }
2584
2585 /* Return reply. */
2586 pReply->IOCInit.u8MessageLength = 5;
2587 pReply->IOCInit.u8WhoInit = pLsiLogic->enmWhoInit;
2588 pReply->IOCInit.u8MaxDevices = pLsiLogic->cMaxDevices;
2589 pReply->IOCInit.u8MaxBuses = pLsiLogic->cMaxBuses;
2590 break;
2591 }
2592 case MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS:
2593 {
2594 pReply->IOCFacts.u8MessageLength = 15; /* 15 32bit dwords. */
2595 pReply->IOCFacts.u16MessageVersion = 0x0102; /* Version from the specification. */
2596 pReply->IOCFacts.u8IOCNumber = 0; /* PCI function number. */
2597 pReply->IOCFacts.u16IOCExceptions = 0;
2598 pReply->IOCFacts.u8MaxChainDepth = LSILOGICSCSI_MAXIMUM_CHAIN_DEPTH;
2599 pReply->IOCFacts.u8WhoInit = pLsiLogic->enmWhoInit;
2600 pReply->IOCFacts.u8BlockSize = 12; /* Block size in 32bit dwords. This is the largest request we can get (SCSI I/O). */
2601 pReply->IOCFacts.u8Flags = 0; /* Bit 0 is set if the guest must upload the FW prior to using the controller. Obviously not needed here. */
2602 pReply->IOCFacts.u16ReplyQueueDepth = pLsiLogic->cReplyQueueEntries - 1; /* One entry is always free. */
2603 pReply->IOCFacts.u16RequestFrameSize = 128; /* @todo Figure out where it is needed. */
2604 pReply->IOCFacts.u16ProductID = 0xcafe; /* Our own product ID :) */
2605 pReply->IOCFacts.u32CurrentHostMFAHighAddr = pLsiLogic->u32HostMFAHighAddr;
2606 pReply->IOCFacts.u16GlobalCredits = pLsiLogic->cRequestQueueEntries - 1; /* One entry is always free. */
2607 pReply->IOCFacts.u8NumberOfPorts = 1;
2608 pReply->IOCFacts.u8EventState = 0; /* Event notifications not enabled. */
2609 pReply->IOCFacts.u32CurrentSenseBufferHighAddr = pLsiLogic->u32SenseBufferHighAddr;
2610 pReply->IOCFacts.u16CurReplyFrameSize = pLsiLogic->cbReplyFrame;
2611 pReply->IOCFacts.u8MaxDevices = pLsiLogic->cMaxDevices;
2612 pReply->IOCFacts.u8MaxBuses = pLsiLogic->cMaxBuses;
2613 pReply->IOCFacts.u32FwImageSize = 0; /* No image needed. */
2614 pReply->IOCFacts.u32FWVersion = 0;
2615 break;
2616 }
2617 case MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS:
2618 {
2619 PMptPortFactsRequest pPortFactsReq = (PMptPortFactsRequest)pMessageHdr;
2620
2621 pReply->PortFacts.u8MessageLength = 10;
2622 pReply->PortFacts.u8PortNumber = pPortFactsReq->u8PortNumber;
2623
2624 /* This controller only supports one bus with bus number 0. */
2625 if (pPortFactsReq->u8PortNumber != 0)
2626 {
2627 pReply->PortFacts.u8PortType = 0; /* Not existant. */
2628 }
2629 else
2630 {
2631 pReply->PortFacts.u8PortType = 0x01; /* SCSI Port. */
2632 pReply->PortFacts.u16MaxDevices = LSILOGIC_DEVICES_MAX;
2633 pReply->PortFacts.u16ProtocolFlags = RT_BIT(3) | RT_BIT(0); /* SCSI initiator and LUN supported. */
2634 pReply->PortFacts.u16PortSCSIID = 7; /* Default */
2635 pReply->PortFacts.u16MaxPersistentIDs = 0;
2636 pReply->PortFacts.u16MaxPostedCmdBuffers = 0; /* Only applies for target mode which we dont support. */
2637 pReply->PortFacts.u16MaxLANBuckets = 0; /* Only for the LAN controller. */
2638 }
2639 break;
2640 }
2641 case MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE:
2642 {
2643 /*
2644 * The port enable request notifies the IOC to make the port available and perform
2645 * appropriate discovery on the associated link.
2646 */
2647 PMptPortEnableRequest pPortEnableReq = (PMptPortEnableRequest)pMessageHdr;
2648
2649 pReply->PortEnable.u8MessageLength = 5;
2650 pReply->PortEnable.u8PortNumber = pPortEnableReq->u8PortNumber;
2651 break;
2652 }
2653 case MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION:
2654 {
2655 PMptEventNotificationRequest pEventNotificationReq = (PMptEventNotificationRequest)pMessageHdr;
2656
2657 if (pEventNotificationReq->u8Switch)
2658 pLsiLogic->fEventNotificationEnabled = true;
2659 else
2660 pLsiLogic->fEventNotificationEnabled = false;
2661
2662 pReply->EventNotification.u16EventDataLength = 1; /* 1 32bit D-Word. */
2663 pReply->EventNotification.u8MessageLength = 8;
2664 pReply->EventNotification.u8MessageFlags = (1 << 7);
2665 pReply->EventNotification.u8AckRequired = 0;
2666 pReply->EventNotification.u32Event = MPT_EVENT_EVENT_CHANGE;
2667 pReply->EventNotification.u32EventContext = 0;
2668 pReply->EventNotification.u32EventData = pLsiLogic->fEventNotificationEnabled ? 1 : 0;
2669
2670 break;
2671 }
2672 case MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK:
2673 {
2674 AssertMsgFailed(("todo"));
2675 break;
2676 }
2677 case MPT_MESSAGE_HDR_FUNCTION_CONFIG:
2678 {
2679 PMptConfigurationRequest pConfigurationReq = (PMptConfigurationRequest)pMessageHdr;
2680
2681 rc = lsilogicProcessConfigurationRequest(pLsiLogic, pConfigurationReq, &pReply->Configuration);
2682 AssertRC(rc);
2683 break;
2684 }
2685 case MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST: /* Should be handled already. */
2686 default:
2687 AssertMsgFailed(("Invalid request function %#x\n", pMessageHdr->u8Function));
2688 }
2689
2690 /* Copy common bits from request message frame to reply. */
2691 pReply->Header.u8Function = pMessageHdr->u8Function;
2692 pReply->Header.u32MessageContext = pMessageHdr->u32MessageContext;
2693
2694 lsilogicFinishAddressReply(pLsiLogic, pReply, fForceReplyPostFifo);
2695 return rc;
2696}
2697#endif
2698
2699/**
2700 * Writes a value to a register at a given offset.
2701 *
2702 * @returns VBox status code.
2703 * @param pThis Pointer to the LsiLogic SCSI controller instance data.
2704 * @param uOffset Offset of the register to write.
2705 * @param pv Pointer to the value to write
2706 * @param cb Number of bytes to write.
2707 */
2708static int lsilogicRegisterWrite(PLSILOGICSCSI pThis, uint32_t uOffset, void *pv, unsigned cb)
2709{
2710 uint32_t u32 = *(uint32_t *)pv;
2711 AssertMsg(cb == 4, ("cb != 4 is %d\n", cb));
2712
2713 LogFlowFunc(("pThis=%#p uOffset=%#x pv=%#p{%.*Rhxs} cb=%u\n", pThis, uOffset, pv, cb, pv, cb));
2714
2715 switch (uOffset)
2716 {
2717 case LSILOGIC_REG_REPLY_QUEUE:
2718 {
2719 /* Add the entry to the reply free queue. */
2720 ASMAtomicWriteU32(&pThis->CTX_SUFF(pReplyFreeQueueBase)[pThis->uReplyFreeQueueNextEntryFreeWrite], u32);
2721 pThis->uReplyFreeQueueNextEntryFreeWrite++;
2722 pThis->uReplyFreeQueueNextEntryFreeWrite %= pThis->cReplyQueueEntries;
2723 break;
2724 }
2725 case LSILOGIC_REG_REQUEST_QUEUE:
2726 {
2727 ASMAtomicWriteU32(&pThis->CTX_SUFF(pRequestQueueBase)[pThis->uRequestQueueNextEntryFreeWrite], u32);
2728 pThis->uRequestQueueNextEntryFreeWrite++;
2729 pThis->uRequestQueueNextEntryFreeWrite %= pThis->cRequestQueueEntries;
2730
2731 /* Send notification to R3 if there is not one send already. */
2732 if (!ASMAtomicXchgBool(&pThis->fNotificationSend, true))
2733 {
2734 PPDMQUEUEITEMCORE pNotificationItem = PDMQueueAlloc(pThis->CTX_SUFF(pNotificationQueue));
2735 AssertPtr(pNotificationItem);
2736
2737 PDMQueueInsert(pThis->CTX_SUFF(pNotificationQueue), pNotificationItem);
2738 }
2739 break;
2740 }
2741 case LSILOGIC_REG_DOORBELL:
2742 {
2743 /*
2744 * When the guest writes to this register a real device would set the
2745 * doorbell status bit in the interrupt status register to indicate that the IOP
2746 * has still to process the message.
2747 * The guest needs to wait with posting new messages here until the bit is cleared.
2748 * Because the guest is not continuing execution while we are here we can skip this.
2749 */
2750 if (!pThis->fDoorbellInProgress)
2751 {
2752 uint32_t uFunction = LSILOGIC_REG_DOORBELL_GET_FUNCTION(u32);
2753
2754 switch (uFunction)
2755 {
2756 case LSILOGIC_DOORBELL_FUNCTION_IOC_MSG_UNIT_RESET:
2757 {
2758 pThis->enmState = LSILOGICSTATE_RESET;
2759
2760 /* Reset interrupt states. */
2761 pThis->uInterruptMask = 0;
2762 pThis->uInterruptStatus = 0;
2763 lsilogicUpdateInterrupt(pThis);
2764
2765 /* Reset the queues. */
2766 pThis->uReplyFreeQueueNextEntryFreeWrite = 0;
2767 pThis->uReplyFreeQueueNextAddressRead = 0;
2768 pThis->uReplyPostQueueNextEntryFreeWrite = 0;
2769 pThis->uReplyPostQueueNextAddressRead = 0;
2770 pThis->uRequestQueueNextEntryFreeWrite = 0;
2771 pThis->uRequestQueueNextAddressRead = 0;
2772 pThis->enmState = LSILOGICSTATE_READY;
2773 break;
2774 }
2775 case LSILOGIC_DOORBELL_FUNCTION_IO_UNIT_RESET:
2776 {
2777 AssertMsgFailed(("todo\n"));
2778 break;
2779 }
2780 case LSILOGIC_DOORBELL_FUNCTION_HANDSHAKE:
2781 {
2782 pThis->cMessage = LSILOGIC_REG_DOORBELL_GET_SIZE(u32);
2783 pThis->iMessage = 0;
2784 AssertMsg(pThis->cMessage <= RT_ELEMENTS(pThis->aMessage),
2785 ("Message doesn't fit into the buffer, cMessage=%u", pThis->cMessage));
2786 pThis->fDoorbellInProgress = true;
2787 /* Update the interrupt status to notify the guest that a doorbell function was started. */
2788 lsilogicSetInterrupt(pThis, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
2789 break;
2790 }
2791 case LSILOGIC_DOORBELL_FUNCTION_REPLY_FRAME_REMOVAL:
2792 {
2793 AssertMsgFailed(("todo\n"));
2794 break;
2795 }
2796 default:
2797 AssertMsgFailed(("Unknown function %u to perform\n", uFunction));
2798 }
2799 }
2800 else
2801 {
2802 /*
2803 * We are already performing a doorbell function.
2804 * Get the remaining parameters.
2805 */
2806 AssertMsg(pThis->iMessage < RT_ELEMENTS(pThis->aMessage), ("Message is too big to fit into the buffer\n"));
2807 /*
2808 * If the last byte of the message is written, force a switch to R3 because some requests might force
2809 * a reply through the FIFO which cannot be handled in GC or R0.
2810 */
2811#ifndef IN_RING3
2812 if (pThis->iMessage == pThis->cMessage - 1)
2813 return VINF_IOM_HC_MMIO_WRITE;
2814#endif
2815 pThis->aMessage[pThis->iMessage++] = u32;
2816#ifdef IN_RING3
2817 if (pThis->iMessage == pThis->cMessage)
2818 {
2819 int rc = lsilogicProcessMessageRequest(pThis, (PMptMessageHdr)pThis->aMessage, &pThis->ReplyBuffer);
2820 AssertRC(rc);
2821 }
2822#endif
2823 }
2824 break;
2825 }
2826 case LSILOGIC_REG_HOST_INTR_STATUS:
2827 {
2828 /*
2829 * Clear the bits the guest wants except the system doorbell interrupt and the IO controller
2830 * status bit.
2831 * The former bit is always cleared no matter what the guest writes to the register and
2832 * the latter one is read only.
2833 */
2834 pThis->uInterruptStatus = pThis->uInterruptStatus & ~LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL;
2835
2836 /*
2837 * Check if there is still a doorbell function in progress. Set the
2838 * system doorbell interrupt bit again if it is.
2839 * We do not use lsilogicSetInterrupt here because the interrupt status
2840 * is updated afterwards anyway.
2841 */
2842 if ( (pThis->fDoorbellInProgress)
2843 && (pThis->cMessage == pThis->iMessage))
2844 {
2845 if (pThis->uNextReplyEntryRead == pThis->cReplySize)
2846 {
2847 /* Reply finished. Reset doorbell in progress status. */
2848 Log(("%s: Doorbell function finished\n", __FUNCTION__));
2849 pThis->fDoorbellInProgress = false;
2850 }
2851 ASMAtomicOrU32(&pThis->uInterruptStatus, LSILOGIC_REG_HOST_INTR_STATUS_SYSTEM_DOORBELL);
2852 }
2853
2854 lsilogicUpdateInterrupt(pThis);
2855 break;
2856 }
2857 case LSILOGIC_REG_HOST_INTR_MASK:
2858 {
2859 pThis->uInterruptMask = (u32 & LSILOGIC_REG_HOST_INTR_MASK_W_MASK);
2860 lsilogicUpdateInterrupt(pThis);
2861 break;
2862 }
2863 case LSILOGIC_REG_WRITE_SEQUENCE:
2864 {
2865 if (pThis->fDiagnosticEnabled)
2866 {
2867 /* Any value will cause a reset and disabling access. */
2868 pThis->fDiagnosticEnabled = false;
2869 pThis->iDiagnosticAccess = 0;
2870 }
2871 else if ((u32 & 0xf) == g_lsilogicDiagnosticAccess[pThis->iDiagnosticAccess])
2872 {
2873 pThis->iDiagnosticAccess++;
2874 if (pThis->iDiagnosticAccess == RT_ELEMENTS(g_lsilogicDiagnosticAccess))
2875 {
2876 /*
2877 * Key sequence successfully written. Enable access to diagnostic
2878 * memory and register.
2879 */
2880 pThis->fDiagnosticEnabled = true;
2881 }
2882 }
2883 else
2884 {
2885 /* Wrong value written - reset to beginning. */
2886 pThis->iDiagnosticAccess = 0;
2887 }
2888 break;
2889 }
2890 default: /* Ignore. */
2891 {
2892 break;
2893 }
2894 }
2895 return VINF_SUCCESS;
2896}
2897
2898/**
2899 * Reads the content of a register at a given offset.
2900 *
2901 * @returns VBox status code.
2902 * @param pThis Pointer to the LsiLogic SCSI controller instance data.
2903 * @param uOffset Offset of the register to read.
2904 * @param pv Where to store the content of the register.
2905 * @param cb Number of bytes to read.
2906 */
2907static int lsilogicRegisterRead(PLSILOGICSCSI pThis, uint32_t uOffset, void *pv, unsigned cb)
2908{
2909 uint32_t *pu32 = (uint32_t *)pv;
2910 AssertMsg(cb == 4, ("cb != 4 is %d\n", cb));
2911
2912 switch (uOffset)
2913 {
2914 case LSILOGIC_REG_REPLY_QUEUE:
2915 {
2916 if (pThis->uReplyPostQueueNextEntryFreeWrite != pThis->uReplyPostQueueNextAddressRead)
2917 {
2918 *pu32 = pThis->CTX_SUFF(pReplyPostQueueBase)[pThis->uReplyPostQueueNextAddressRead];
2919 pThis->uReplyPostQueueNextAddressRead++;
2920 pThis->uReplyPostQueueNextAddressRead %= pThis->cReplyQueueEntries;
2921 }
2922 else
2923 {
2924 /* The reply post queue is empty. Reset interrupt. */
2925 *pu32 = UINT32_C(0xffffffff);
2926 lsilogicClearInterrupt(pThis, LSILOGIC_REG_HOST_INTR_STATUS_REPLY_INTR);
2927 }
2928 Log(("%s: Returning address %#x\n", __FUNCTION__, *pu32));
2929 break;
2930 }
2931 case LSILOGIC_REG_DOORBELL:
2932 {
2933 *pu32 = LSILOGIC_REG_DOORBELL_SET_STATE(pThis->enmState);
2934 *pu32 |= LSILOGIC_REG_DOORBELL_SET_USED(pThis->fDoorbellInProgress);
2935 *pu32 |= LSILOGIC_REG_DOORBELL_SET_WHOINIT(pThis->enmWhoInit);
2936 /*
2937 * If there is a doorbell function in progress we pass the return value
2938 * instead of the status code. We transfer 16bit of the reply
2939 * during one read.
2940 */
2941 if (pThis->fDoorbellInProgress)
2942 {
2943 /* Return next 16bit value. */
2944 *pu32 |= pThis->ReplyBuffer.au16Reply[pThis->uNextReplyEntryRead++];
2945 }
2946 else
2947 {
2948 /* We return the status code of the I/O controller. */
2949 *pu32 |= pThis->u16IOCFaultCode;
2950 }
2951 break;
2952 }
2953 case LSILOGIC_REG_HOST_INTR_STATUS:
2954 {
2955 *pu32 = pThis->uInterruptStatus;
2956 break;
2957 }
2958 case LSILOGIC_REG_HOST_INTR_MASK:
2959 {
2960 *pu32 = pThis->uInterruptMask;
2961 break;
2962 }
2963 case LSILOGIC_REG_HOST_DIAGNOSTIC:
2964 {
2965 //AssertMsgFailed(("todo\n"));
2966 break;
2967 }
2968 case LSILOGIC_REG_TEST_BASE_ADDRESS:
2969 {
2970 AssertMsgFailed(("todo\n"));
2971 break;
2972 }
2973 case LSILOGIC_REG_DIAG_RW_DATA:
2974 {
2975 AssertMsgFailed(("todo\n"));
2976 break;
2977 }
2978 case LSILOGIC_REG_DIAG_RW_ADDRESS:
2979 {
2980 AssertMsgFailed(("todo\n"));
2981 break;
2982 }
2983 default: /* Ignore. */
2984 {
2985 break;
2986 }
2987 }
2988
2989 LogFlowFunc(("pThis=%#p uOffset=%#x pv=%#p{%.*Rhxs} cb=%u\n", pThis, uOffset, pv, cb, pv, cb));
2990
2991 return VINF_SUCCESS;
2992}
2993
2994PDMBOTHCBDECL(int) lsilogicIOPortWrite (PPDMDEVINS pDevIns, void *pvUser,
2995 RTIOPORT Port, uint32_t u32, unsigned cb)
2996{
2997 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
2998 uint32_t uOffset = Port - pThis->IOPortBase;
2999
3000 Assert(cb <= 4);
3001
3002 int rc = lsilogicRegisterWrite(pThis, uOffset, &u32, cb);
3003 if (rc == VINF_IOM_HC_MMIO_WRITE)
3004 rc = VINF_IOM_HC_IOPORT_WRITE;
3005
3006 return rc;
3007}
3008
3009PDMBOTHCBDECL(int) lsilogicIOPortRead (PPDMDEVINS pDevIns, void *pvUser,
3010 RTIOPORT Port, uint32_t *pu32, unsigned cb)
3011{
3012 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3013 uint32_t uOffset = Port - pThis->IOPortBase;
3014
3015 Assert(cb <= 4);
3016
3017 return lsilogicRegisterRead(pThis, uOffset, pu32, cb);
3018}
3019
3020PDMBOTHCBDECL(int) lsilogicMMIOWrite(PPDMDEVINS pDevIns, void *pvUser,
3021 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3022{
3023 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3024 uint32_t uOffset = GCPhysAddr - pThis->GCPhysMMIOBase;
3025
3026 return lsilogicRegisterWrite(pThis, uOffset, pv, cb);
3027}
3028
3029PDMBOTHCBDECL(int) lsilogicMMIORead(PPDMDEVINS pDevIns, void *pvUser,
3030 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3031{
3032 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3033 uint32_t uOffset = GCPhysAddr - pThis->GCPhysMMIOBase;
3034
3035 return lsilogicRegisterRead(pThis, uOffset, pv, cb);
3036}
3037
3038PDMBOTHCBDECL(int) lsilogicDiagnosticWrite(PPDMDEVINS pDevIns, void *pvUser,
3039 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3040{
3041 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3042
3043 LogFlowFunc(("pThis=%#p GCPhysAddr=%RGp pv=%#p{%.*Rhxs} cb=%u\n", pThis, GCPhysAddr, pv, cb, pv, cb));
3044
3045 return VINF_SUCCESS;
3046}
3047
3048PDMBOTHCBDECL(int) lsilogicDiagnosticRead(PPDMDEVINS pDevIns, void *pvUser,
3049 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3050{
3051 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
3052
3053 LogFlowFunc(("pThis=%#p GCPhysAddr=%RGp pv=%#p{%.*Rhxs} cb=%u\n", pThis, GCPhysAddr, pv, cb, pv, cb));
3054
3055 return VINF_SUCCESS;
3056}
3057
3058#ifdef IN_RING3
3059
3060/**
3061 * Copies a contigous buffer into the scatter gather list provided by the guest.
3062 *
3063 * @returns nothing
3064 * @param pTaskState Pointer to the task state which contains the SGL.
3065 * @param pvBuf Pointer to the buffer to copy.
3066 * @param cbCopy Number of bytes to copy.
3067 */
3068static void lsilogicScatterGatherListCopyFromBuffer(PLSILOGICTASKSTATE pTaskState, void *pvBuf, size_t cbCopy)
3069{
3070 unsigned cSGEntry = 0;
3071 PPDMDATASEG pSGEntry = &pTaskState->pSGListHead[cSGEntry];
3072 uint8_t *pu8Buf = (uint8_t *)pvBuf;
3073
3074 while (cSGEntry < pTaskState->cSGListEntries)
3075 {
3076 size_t cbToCopy = (cbCopy < pSGEntry->cbSeg) ? cbCopy : pSGEntry->cbSeg;
3077
3078 memcpy(pSGEntry->pvSeg, pu8Buf, cbToCopy);
3079
3080 cbCopy -= cbToCopy;
3081 /* We finished. */
3082 if (!cbCopy)
3083 break;
3084
3085 /* Advance the buffer. */
3086 pu8Buf += cbToCopy;
3087
3088 /* Go to the next entry in the list. */
3089 pSGEntry++;
3090 cSGEntry++;
3091 }
3092}
3093
3094/**
3095 * Copy a temporary buffer into a part of the guest scatter gather list
3096 * described by the given descriptor entry.
3097 *
3098 * @returns nothing.
3099 * @param pDevIns Pointer to the device instance data.
3100 * @param pSGInfo Pointer to the segment info structure which describes the guest segments
3101 * to write to which are unaligned.
3102 */
3103static void lsilogicCopyFromBufferIntoSGList(PPDMDEVINS pDevIns, PLSILOGICTASKSTATESGENTRY pSGInfo)
3104{
3105 RTGCPHYS GCPhysBuffer = pSGInfo->u.GCPhysAddrBufferUnaligned;
3106
3107 AssertMsg(!pSGInfo->fGuestMemory, ("This is not possible\n"));
3108
3109 /* Copy into SG entry. */
3110 PDMDevHlpPhysWrite(pDevIns, GCPhysBuffer, pSGInfo->pvBuf, pSGInfo->cbBuf);
3111
3112}
3113
3114/**
3115 * Copy a part of the guest scatter gather list into a temporary buffer.
3116 *
3117 * @returns nothing.
3118 * @param pDevIns Pointer to the device instance data.
3119 * @param pSGInfo Pointer to the segment info structure which describes the guest segments
3120 * to read from which are unaligned.
3121 */
3122static void lsilogicCopyFromSGListIntoBuffer(PPDMDEVINS pDevIns, PLSILOGICTASKSTATESGENTRY pSGInfo)
3123{
3124 RTGCPHYS GCPhysBuffer = pSGInfo->u.GCPhysAddrBufferUnaligned;
3125
3126 AssertMsg(!pSGInfo->fGuestMemory, ("This is not possible\n"));
3127
3128 /* Copy into temporary buffer. */
3129 PDMDevHlpPhysRead(pDevIns, GCPhysBuffer, pSGInfo->pvBuf, pSGInfo->cbBuf);
3130}
3131
3132static int lsilogicScatterGatherListAllocate(PLSILOGICTASKSTATE pTaskState, uint32_t cSGList, uint32_t cSGInfo, uint32_t cbUnaligned)
3133{
3134 if (pTaskState->cSGListSize < cSGList)
3135 {
3136 /* The entries are not allocated yet or the number is too small. */
3137 if (pTaskState->cSGListSize)
3138 RTMemFree(pTaskState->pSGListHead);
3139
3140 /* Allocate R3 scatter gather list. */
3141 pTaskState->pSGListHead = (PPDMDATASEG)RTMemAllocZ(cSGList * sizeof(PDMDATASEG));
3142 if (!pTaskState->pSGListHead)
3143 return VERR_NO_MEMORY;
3144
3145 /* Reset usage statistics. */
3146 pTaskState->cSGListSize = cSGList;
3147 pTaskState->cSGListEntries = cSGList;
3148 pTaskState->cSGListTooBig = 0;
3149 }
3150 else if (pTaskState->cSGListSize > cSGList)
3151 {
3152 /*
3153 * The list is too big. Increment counter.
3154 * So that the destroying function can free
3155 * the list if it is too big too many times
3156 * in a row.
3157 */
3158 pTaskState->cSGListEntries = cSGList;
3159 pTaskState->cSGListTooBig++;
3160 }
3161 else
3162 {
3163 /*
3164 * Needed entries matches current size.
3165 * Reset counter.
3166 */
3167 pTaskState->cSGListEntries = cSGList;
3168 pTaskState->cSGListTooBig = 0;
3169 }
3170
3171 if (pTaskState->cSGInfoSize < cSGInfo)
3172 {
3173 /* The entries are not allocated yet or the number is too small. */
3174 if (pTaskState->cSGInfoSize)
3175 RTMemFree(pTaskState->paSGEntries);
3176
3177 pTaskState->paSGEntries = (PLSILOGICTASKSTATESGENTRY)RTMemAllocZ(cSGInfo * sizeof(LSILOGICTASKSTATESGENTRY));
3178 if (!pTaskState->paSGEntries)
3179 return VERR_NO_MEMORY;
3180
3181 /* Reset usage statistics. */
3182 pTaskState->cSGInfoSize = cSGInfo;
3183 pTaskState->cSGInfoEntries = cSGInfo;
3184 pTaskState->cSGInfoTooBig = 0;
3185 }
3186 else if (pTaskState->cSGInfoSize > cSGInfo)
3187 {
3188 /*
3189 * The list is too big. Increment counter.
3190 * So that the destroying function can free
3191 * the list if it is too big too many times
3192 * in a row.
3193 */
3194 pTaskState->cSGInfoEntries = cSGInfo;
3195 pTaskState->cSGInfoTooBig++;
3196 }
3197 else
3198 {
3199 /*
3200 * Needed entries matches current size.
3201 * Reset counter.
3202 */
3203 pTaskState->cSGInfoEntries = cSGInfo;
3204 pTaskState->cSGInfoTooBig = 0;
3205 }
3206
3207
3208 if (pTaskState->cbBufferUnaligned < cbUnaligned)
3209 {
3210 if (pTaskState->pvBufferUnaligned)
3211 RTMemFree(pTaskState->pvBufferUnaligned);
3212
3213 Log(("%s: Allocating buffer for unaligned segments cbUnaligned=%u\n", __FUNCTION__, cbUnaligned));
3214
3215 pTaskState->pvBufferUnaligned = RTMemAllocZ(cbUnaligned);
3216 if (!pTaskState->pvBufferUnaligned)
3217 return VERR_NO_MEMORY;
3218
3219 pTaskState->cbBufferUnaligned = cbUnaligned;
3220 }
3221
3222 /* Make debugging easier. */
3223#ifdef DEBUG
3224 memset(pTaskState->pSGListHead, 0, pTaskState->cSGListSize * sizeof(PDMDATASEG));
3225 memset(pTaskState->paSGEntries, 0, pTaskState->cSGInfoSize * sizeof(LSILOGICTASKSTATESGENTRY));
3226 if (pTaskState->pvBufferUnaligned)
3227 memset(pTaskState->pvBufferUnaligned, 0, pTaskState->cbBufferUnaligned);
3228#endif
3229 return VINF_SUCCESS;
3230}
3231
3232/**
3233 * Destroy a scatter gather list.
3234 *
3235 * @returns nothing.
3236 * @param pLsiLogic Pointer to the LsiLogic SCSI controller.
3237 * @param pTaskState Pointer to the task state.
3238 */
3239static void lsilogicScatterGatherListDestroy(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
3240{
3241 PPDMDEVINS pDevIns = pLsiLogic->CTX_SUFF(pDevIns);
3242 PLSILOGICTASKSTATESGENTRY pSGInfoCurr = pTaskState->paSGEntries;
3243
3244 for (unsigned i = 0; i < pTaskState->cSGInfoEntries; i++)
3245 {
3246 if (pSGInfoCurr->fGuestMemory)
3247 {
3248 /* Release the lock. */
3249 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &pSGInfoCurr->u.PageLock);
3250 }
3251 else if (!pSGInfoCurr->fBufferContainsData)
3252 {
3253 /* Copy the data into the guest segments now. */
3254 lsilogicCopyFromBufferIntoSGList(pLsiLogic->CTX_SUFF(pDevIns), pSGInfoCurr);
3255 }
3256
3257 pSGInfoCurr++;
3258 }
3259
3260 /* Free allocated memory if the list was too big too many times. */
3261 if (pTaskState->cSGListTooBig >= LSILOGIC_NR_OF_ALLOWED_BIGGER_LISTS)
3262 {
3263 RTMemFree(pTaskState->pSGListHead);
3264 RTMemFree(pTaskState->paSGEntries);
3265 if (pTaskState->pvBufferUnaligned)
3266 RTMemFree(pTaskState->pvBufferUnaligned);
3267 pTaskState->cSGListSize = 0;
3268 pTaskState->cSGInfoSize = 0;
3269 pTaskState->cSGInfoEntries = 0;
3270 pTaskState->cSGListTooBig = 0;
3271 pTaskState->pSGListHead = NULL;
3272 pTaskState->paSGEntries = NULL;
3273 pTaskState->pvBufferUnaligned = NULL;
3274 pTaskState->cbBufferUnaligned = 0;
3275 }
3276}
3277
3278#ifdef DEBUG
3279/**
3280 * Dump an SG entry.
3281 *
3282 * @returns nothing.
3283 * @param pSGEntry Pointer to the SG entry to dump
3284 */
3285static void lsilogicDumpSGEntry(PMptSGEntryUnion pSGEntry)
3286{
3287 switch (pSGEntry->Simple32.u2ElementType)
3288 {
3289 case MPTSGENTRYTYPE_SIMPLE:
3290 {
3291 Log(("%s: Dumping info for SIMPLE SG entry:\n", __FUNCTION__));
3292 Log(("%s: u24Length=%u\n", __FUNCTION__, pSGEntry->Simple32.u24Length));
3293 Log(("%s: fEndOfList=%d\n", __FUNCTION__, pSGEntry->Simple32.fEndOfList));
3294 Log(("%s: f64BitAddress=%d\n", __FUNCTION__, pSGEntry->Simple32.f64BitAddress));
3295 Log(("%s: fBufferContainsData=%d\n", __FUNCTION__, pSGEntry->Simple32.fBufferContainsData));
3296 Log(("%s: fLocalAddress=%d\n", __FUNCTION__, pSGEntry->Simple32.fLocalAddress));
3297 Log(("%s: fEndOfBuffer=%d\n", __FUNCTION__, pSGEntry->Simple32.fEndOfBuffer));
3298 Log(("%s: fLastElement=%d\n", __FUNCTION__, pSGEntry->Simple32.fLastElement));
3299 Log(("%s: u32DataBufferAddressLow=%u\n", __FUNCTION__, pSGEntry->Simple32.u32DataBufferAddressLow));
3300 if (pSGEntry->Simple32.f64BitAddress)
3301 {
3302 Log(("%s: u32DataBufferAddressHigh=%u\n", __FUNCTION__, pSGEntry->Simple64.u32DataBufferAddressHigh));
3303 Log(("%s: GCDataBufferAddress=%RGp\n", __FUNCTION__,
3304 ((uint64_t)pSGEntry->Simple64.u32DataBufferAddressHigh << 32) | pSGEntry->Simple64.u32DataBufferAddressLow));
3305 }
3306 else
3307 Log(("%s: GCDataBufferAddress=%RGp\n", __FUNCTION__, pSGEntry->Simple32.u32DataBufferAddressLow));
3308
3309 break;
3310 }
3311 case MPTSGENTRYTYPE_CHAIN:
3312 {
3313 Log(("%s: Dumping info for CHAIN SG entry:\n", __FUNCTION__));
3314 Log(("%s: u16Length=%u\n", __FUNCTION__, pSGEntry->Chain.u16Length));
3315 Log(("%s: u8NExtChainOffset=%d\n", __FUNCTION__, pSGEntry->Chain.u8NextChainOffset));
3316 Log(("%s: f64BitAddress=%d\n", __FUNCTION__, pSGEntry->Chain.f64BitAddress));
3317 Log(("%s: fLocalAddress=%d\n", __FUNCTION__, pSGEntry->Chain.fLocalAddress));
3318 Log(("%s: u32SegmentAddressLow=%u\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressLow));
3319 Log(("%s: u32SegmentAddressHigh=%u\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressHigh));
3320 if (pSGEntry->Chain.f64BitAddress)
3321 Log(("%s: GCSegmentAddress=%RGp\n", __FUNCTION__,
3322 ((uint64_t)pSGEntry->Chain.u32SegmentAddressHigh << 32) | pSGEntry->Chain.u32SegmentAddressLow));
3323 else
3324 Log(("%s: GCSegmentAddress=%RGp\n", __FUNCTION__, pSGEntry->Chain.u32SegmentAddressLow));
3325 break;
3326 }
3327 }
3328}
3329#endif
3330
3331/**
3332 * Create scatter gather list descriptors.
3333 *
3334 * @returns VBox status code.
3335 * @param pLsiLogic Pointer to the LsiLogic SCSI controller.
3336 * @param pTaskState Pointer to the task state.
3337 * @param GCPhysSGLStart Guest physical address of the first SG entry.
3338 * @param uChainOffset Offset in bytes from the beginning of the SGL segment to the chain element.
3339 * @thread EMT
3340 */
3341static int lsilogicScatterGatherListCreate(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState,
3342 RTGCPHYS GCPhysSGLStart, uint32_t uChainOffset)
3343{
3344 int rc = VINF_SUCCESS;
3345 PPDMDEVINS pDevIns = pLsiLogic->CTX_SUFF(pDevIns);
3346 PVM pVM = PDMDevHlpGetVM(pDevIns);
3347 bool fUnaligned; /* Flag whether the current buffer is unaligned. */
3348 uint32_t cbUnaligned; /* Size of the unaligned buffers. */
3349 uint32_t cSGEntriesR3 = 0;
3350 uint32_t cSGInfo = 0;
3351 uint32_t cbSegment = 0;
3352 PLSILOGICTASKSTATESGENTRY pSGInfoCurr = NULL;
3353 uint8_t *pu8BufferUnalignedPos = NULL;
3354 uint8_t *pbBufferUnalignedSGInfoPos = NULL;
3355 uint32_t cbUnalignedComplete = 0;
3356 bool fDoMapping = false;
3357 bool fEndOfList;
3358 RTGCPHYS GCPhysSGEntryNext;
3359 RTGCPHYS GCPhysSegmentStart;
3360 uint32_t uChainOffsetNext;
3361
3362 /*
3363 * Two passes - one to count needed scatter gather list entries and needed unaligned
3364 * buffers and one to actually map the SG list into R3.
3365 */
3366 for (int i = 0; i < 2; i++)
3367 {
3368 fUnaligned = false;
3369 cbUnaligned = 0;
3370 fEndOfList = false;
3371
3372 GCPhysSGEntryNext = GCPhysSGLStart;
3373 uChainOffsetNext = uChainOffset;
3374 GCPhysSegmentStart = GCPhysSGLStart;
3375
3376 if (fDoMapping)
3377 {
3378 Log(("%s: cSGInfo=%u\n", __FUNCTION__, cSGInfo));
3379
3380 /* The number of needed SG entries in R3 is known. Allocate needed memory. */
3381 rc = lsilogicScatterGatherListAllocate(pTaskState, cSGInfo, cSGInfo, cbUnalignedComplete);
3382 AssertMsgRC(rc, ("Failed to allocate scatter gather array rc=%Rrc\n", rc));
3383
3384 /* We are now able to map the pages into R3. */
3385 pSGInfoCurr = pTaskState->paSGEntries;
3386 /* Initialize first segment to remove the need for additional if checks later in the code. */
3387 pSGInfoCurr->fGuestMemory= false;
3388 pu8BufferUnalignedPos = (uint8_t *)pTaskState->pvBufferUnaligned;
3389 pbBufferUnalignedSGInfoPos = pu8BufferUnalignedPos;
3390 }
3391
3392 /* Go through the list until we reach the end. */
3393 while (!fEndOfList)
3394 {
3395 bool fEndOfSegment = false;
3396
3397 while (!fEndOfSegment)
3398 {
3399 MptSGEntryUnion SGEntry;
3400
3401 Log(("%s: Reading SG entry from %RGp\n", __FUNCTION__, GCPhysSGEntryNext));
3402
3403 /* Read the entry. */
3404 PDMDevHlpPhysRead(pDevIns, GCPhysSGEntryNext, &SGEntry, sizeof(MptSGEntryUnion));
3405
3406#ifdef DEBUG
3407 lsilogicDumpSGEntry(&SGEntry);
3408#endif
3409
3410 AssertMsg(SGEntry.Simple32.u2ElementType == MPTSGENTRYTYPE_SIMPLE, ("Invalid SG entry type\n"));
3411
3412 /* Check if this is a zero element. */
3413 if ( !SGEntry.Simple32.u24Length
3414 && SGEntry.Simple32.fEndOfList
3415 && SGEntry.Simple32.fEndOfBuffer)
3416 {
3417 pTaskState->cSGListEntries = 0;
3418 pTaskState->cSGInfoEntries = 0;
3419 return VINF_SUCCESS;
3420 }
3421
3422 uint32_t cbDataToTransfer = SGEntry.Simple32.u24Length;
3423 bool fBufferContainsData = !!SGEntry.Simple32.fBufferContainsData;
3424 RTGCPHYS GCPhysAddrDataBuffer = SGEntry.Simple32.u32DataBufferAddressLow;
3425
3426 if (SGEntry.Simple32.f64BitAddress)
3427 {
3428 GCPhysAddrDataBuffer |= ((uint64_t)SGEntry.Simple64.u32DataBufferAddressHigh) << 32;
3429 GCPhysSGEntryNext += sizeof(MptSGEntrySimple64);
3430 }
3431 else
3432 GCPhysSGEntryNext += sizeof(MptSGEntrySimple32);
3433
3434 if (fDoMapping)
3435 {
3436 pSGInfoCurr->fGuestMemory = false;
3437 pSGInfoCurr->fBufferContainsData = fBufferContainsData;
3438 pSGInfoCurr->cbBuf = cbDataToTransfer;
3439 pSGInfoCurr->pvBuf = pbBufferUnalignedSGInfoPos;
3440 pbBufferUnalignedSGInfoPos += cbDataToTransfer;
3441 pSGInfoCurr->u.GCPhysAddrBufferUnaligned = GCPhysAddrDataBuffer;
3442 if (fBufferContainsData)
3443 lsilogicCopyFromSGListIntoBuffer(pDevIns, pSGInfoCurr);
3444 pSGInfoCurr++;
3445 }
3446 else
3447 {
3448 cbUnalignedComplete += cbDataToTransfer;
3449 cSGInfo++;
3450 }
3451
3452 /* Check if we reached the end of the list. */
3453 if (SGEntry.Simple32.fEndOfList)
3454 {
3455 /* We finished. */
3456 fEndOfSegment = true;
3457 fEndOfList = true;
3458 }
3459 else if (SGEntry.Simple32.fLastElement)
3460 {
3461 fEndOfSegment = true;
3462 }
3463 } /* while (!fEndOfSegment) */
3464
3465 /* Get next chain element. */
3466 if (uChainOffsetNext)
3467 {
3468 MptSGEntryChain SGEntryChain;
3469
3470 PDMDevHlpPhysRead(pDevIns, GCPhysSegmentStart + uChainOffsetNext, &SGEntryChain, sizeof(MptSGEntryChain));
3471
3472 AssertMsg(SGEntryChain.u2ElementType == MPTSGENTRYTYPE_CHAIN, ("Invalid SG entry type\n"));
3473
3474 /* Set the next address now. */
3475 GCPhysSGEntryNext = SGEntryChain.u32SegmentAddressLow;
3476 if (SGEntryChain.f64BitAddress)
3477 GCPhysSGEntryNext |= ((uint64_t)SGEntryChain.u32SegmentAddressHigh) << 32;
3478
3479 GCPhysSegmentStart = GCPhysSGEntryNext;
3480 uChainOffsetNext = SGEntryChain.u8NextChainOffset * sizeof(uint32_t);
3481 }
3482
3483 } /* while (!fEndOfList) */
3484
3485 fDoMapping = true;
3486 if (fUnaligned)
3487 cbUnalignedComplete += cbUnaligned;
3488 }
3489
3490 uint32_t cSGEntries;
3491 PPDMDATASEG pSGEntryCurr = pTaskState->pSGListHead;
3492 pSGInfoCurr = pTaskState->paSGEntries;
3493
3494 /* Initialize first entry. */
3495 pSGEntryCurr->pvSeg = pSGInfoCurr->pvBuf;
3496 pSGEntryCurr->cbSeg = pSGInfoCurr->cbBuf;
3497 pSGInfoCurr++;
3498 cSGEntries = 1;
3499
3500 /* Construct the scatter gather list. */
3501 for (unsigned i = 0; i < (pTaskState->cSGInfoEntries-1); i++)
3502 {
3503 if (pSGEntryCurr->cbSeg % 512 != 0)
3504 {
3505 AssertMsg((uint8_t *)pSGEntryCurr->pvSeg + pSGEntryCurr->cbSeg == pSGInfoCurr->pvBuf,
3506 ("Buffer ist not sector aligned but the buffer addresses are not adjacent\n"));
3507
3508 pSGEntryCurr->cbSeg += pSGInfoCurr->cbBuf;
3509 }
3510 else
3511 {
3512 if (((uint8_t *)pSGEntryCurr->pvSeg + pSGEntryCurr->cbSeg) == pSGInfoCurr->pvBuf)
3513 {
3514 pSGEntryCurr->cbSeg += pSGInfoCurr->cbBuf;
3515 }
3516 else
3517 {
3518 pSGEntryCurr++;
3519 cSGEntries++;
3520 pSGEntryCurr->pvSeg = pSGInfoCurr->pvBuf;
3521 pSGEntryCurr->cbSeg = pSGInfoCurr->cbBuf;
3522 }
3523 }
3524
3525 pSGInfoCurr++;
3526 }
3527
3528 pTaskState->cSGListEntries = cSGEntries;
3529
3530 return rc;
3531}
3532
3533/*
3534 * Disabled because the sense buffer provided by the LsiLogic driver for Windows XP
3535 * crosses page boundaries.
3536 */
3537#if 0
3538/**
3539 * Free the sense buffer.
3540 *
3541 * @returns nothing.
3542 * @param pTaskState Pointer to the task state.
3543 */
3544static void lsilogicFreeGCSenseBuffer(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
3545{
3546 PVM pVM = PDMDevHlpGetVM(pLsiLogic->CTX_SUFF(pDevIns));
3547
3548 PGMPhysReleasePageMappingLock(pVM, &pTaskState->PageLockSense);
3549 pTaskState->pbSenseBuffer = NULL;
3550}
3551
3552/**
3553 * Map the sense buffer into R3.
3554 *
3555 * @returns VBox status code.
3556 * @param pTaskState Pointer to the task state.
3557 * @note Current assumption is that the sense buffer is not scattered and does not cross a page boundary.
3558 */
3559static int lsilogicMapGCSenseBufferIntoR3(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
3560{
3561 int rc = VINF_SUCCESS;
3562 PPDMDEVINS pDevIns = pLsiLogic->CTX_SUFF(pDevIns);
3563 RTGCPHYS GCPhysAddrSenseBuffer;
3564
3565 GCPhysAddrSenseBuffer = pTaskState->GuestRequest.SCSIIO.u32SenseBufferLowAddress;
3566 GCPhysAddrSenseBuffer |= ((uint64_t)pLsiLogic->u32SenseBufferHighAddr << 32);
3567
3568#ifdef RT_STRICT
3569 uint32_t cbSenseBuffer = pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength;
3570#endif
3571 RTGCPHYS GCPhysAddrSenseBufferBase = PAGE_ADDRESS(GCPhysAddrSenseBuffer);
3572
3573 AssertMsg(GCPhysAddrSenseBuffer >= GCPhysAddrSenseBufferBase,
3574 ("Impossible GCPhysAddrSenseBuffer < GCPhysAddrSenseBufferBase\n"));
3575
3576 /* Sanity checks for the assumption. */
3577 AssertMsg(((GCPhysAddrSenseBuffer + cbSenseBuffer) <= (GCPhysAddrSenseBufferBase + PAGE_SIZE)),
3578 ("Sense buffer crosses page boundary\n"));
3579
3580 rc = PDMDevHlpPhysGCPhys2CCPtr(pDevIns, GCPhysAddrSenseBufferBase, (void **)&pTaskState->pbSenseBuffer, &pTaskState->PageLockSense);
3581 AssertMsgRC(rc, ("Mapping sense buffer failed rc=%Rrc\n", rc));
3582
3583 /* Correct start address of the sense buffer. */
3584 pTaskState->pbSenseBuffer += (GCPhysAddrSenseBuffer - GCPhysAddrSenseBufferBase);
3585
3586 return rc;
3587}
3588#endif
3589
3590#ifdef DEBUG
3591static void lsilogicDumpSCSIIORequest(PMptSCSIIORequest pSCSIIORequest)
3592{
3593 Log(("%s: u8TargetID=%d\n", __FUNCTION__, pSCSIIORequest->u8TargetID));
3594 Log(("%s: u8Bus=%d\n", __FUNCTION__, pSCSIIORequest->u8Bus));
3595 Log(("%s: u8ChainOffset=%d\n", __FUNCTION__, pSCSIIORequest->u8ChainOffset));
3596 Log(("%s: u8Function=%d\n", __FUNCTION__, pSCSIIORequest->u8Function));
3597 Log(("%s: u8CDBLength=%d\n", __FUNCTION__, pSCSIIORequest->u8CDBLength));
3598 Log(("%s: u8SenseBufferLength=%d\n", __FUNCTION__, pSCSIIORequest->u8SenseBufferLength));
3599 Log(("%s: u8MessageFlags=%d\n", __FUNCTION__, pSCSIIORequest->u8MessageFlags));
3600 Log(("%s: u32MessageContext=%#x\n", __FUNCTION__, pSCSIIORequest->u32MessageContext));
3601 for (unsigned i = 0; i < RT_ELEMENTS(pSCSIIORequest->au8LUN); i++)
3602 Log(("%s: u8LUN[%d]=%d\n", __FUNCTION__, i, pSCSIIORequest->au8LUN[i]));
3603 Log(("%s: u32Control=%#x\n", __FUNCTION__, pSCSIIORequest->u32Control));
3604 for (unsigned i = 0; i < RT_ELEMENTS(pSCSIIORequest->au8CDB); i++)
3605 Log(("%s: u8CDB[%d]=%d\n", __FUNCTION__, i, pSCSIIORequest->au8CDB[i]));
3606 Log(("%s: u32DataLength=%#x\n", __FUNCTION__, pSCSIIORequest->u32DataLength));
3607 Log(("%s: u32SenseBufferLowAddress=%#x\n", __FUNCTION__, pSCSIIORequest->u32SenseBufferLowAddress));
3608}
3609#endif
3610
3611/**
3612 * Processes a SCSI I/O request by setting up the request
3613 * and sending it to the underlying SCSI driver.
3614 * Steps needed to complete request are done in the
3615 * callback called by the driver below upon completion of
3616 * the request.
3617 *
3618 * @returns VBox status code.
3619 * @param pLsiLogic Pointer to the device instance which sends the request.
3620 * @param pTaskState Pointer to the task state data.
3621 */
3622static int lsilogicProcessSCSIIORequest(PLSILOGICSCSI pLsiLogic, PLSILOGICTASKSTATE pTaskState)
3623{
3624 int rc = VINF_SUCCESS;
3625
3626#ifdef DEBUG
3627 lsilogicDumpSCSIIORequest(&pTaskState->GuestRequest.SCSIIO);
3628#endif
3629
3630 pTaskState->fBIOS = false;
3631
3632 uint32_t uChainOffset = pTaskState->GuestRequest.SCSIIO.u8ChainOffset;
3633
3634 if (uChainOffset)
3635 uChainOffset = uChainOffset * sizeof(uint32_t) - sizeof(MptSCSIIORequest);
3636
3637 /* Create Scatter gather list. */
3638 rc = lsilogicScatterGatherListCreate(pLsiLogic, pTaskState,
3639 pTaskState->GCPhysMessageFrameAddr + sizeof(MptSCSIIORequest),
3640 uChainOffset);
3641 AssertRC(rc);
3642
3643#if 0
3644 /* Map sense buffer. */
3645 rc = lsilogicMapGCSenseBufferIntoR3(pLsiLogic, pTaskState);
3646 AssertRC(rc);
3647#endif
3648
3649 if (RT_LIKELY( (pTaskState->GuestRequest.SCSIIO.u8TargetID < LSILOGIC_DEVICES_MAX)
3650 && (pTaskState->GuestRequest.SCSIIO.u8Bus == 0)))
3651 {
3652 PLSILOGICDEVICE pTargetDevice;
3653 pTargetDevice = &pLsiLogic->aDeviceStates[pTaskState->GuestRequest.SCSIIO.u8TargetID];
3654
3655 if (pTargetDevice->pDrvBase)
3656 {
3657 /* Setup the SCSI request. */
3658 pTaskState->pTargetDevice = pTargetDevice;
3659 pTaskState->PDMScsiRequest.uLogicalUnit = pTaskState->GuestRequest.SCSIIO.au8LUN[1];
3660
3661 uint8_t uDataDirection = MPT_SCSIIO_REQUEST_CONTROL_TXDIR_GET(pTaskState->GuestRequest.SCSIIO.u32Control);
3662
3663 if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_NONE)
3664 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_NONE;
3665 else if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_WRITE)
3666 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_TO_DEVICE;
3667 else if (uDataDirection == MPT_SCSIIO_REQUEST_CONTROL_TXDIR_READ)
3668 pTaskState->PDMScsiRequest.uDataDirection = PDMSCSIREQUESTTXDIR_FROM_DEVICE;
3669
3670 pTaskState->PDMScsiRequest.cbCDB = pTaskState->GuestRequest.SCSIIO.u8CDBLength;
3671 pTaskState->PDMScsiRequest.pbCDB = pTaskState->GuestRequest.SCSIIO.au8CDB;
3672 pTaskState->PDMScsiRequest.cbScatterGather = pTaskState->GuestRequest.SCSIIO.u32DataLength;
3673 pTaskState->PDMScsiRequest.cScatterGatherEntries = pTaskState->cSGListEntries;
3674 pTaskState->PDMScsiRequest.paScatterGatherHead = pTaskState->pSGListHead;
3675 pTaskState->PDMScsiRequest.cbSenseBuffer = sizeof(pTaskState->abSenseBuffer);
3676 memset(pTaskState->abSenseBuffer, 0, pTaskState->PDMScsiRequest.cbSenseBuffer);
3677 pTaskState->PDMScsiRequest.pbSenseBuffer = pTaskState->abSenseBuffer;
3678 pTaskState->PDMScsiRequest.pvUser = pTaskState;
3679
3680 ASMAtomicIncU32(&pTargetDevice->cOutstandingRequests);
3681 rc = pTargetDevice->pDrvSCSIConnector->pfnSCSIRequestSend(pTargetDevice->pDrvSCSIConnector, &pTaskState->PDMScsiRequest);
3682 AssertMsgRC(rc, ("Sending request to SCSI layer failed rc=%Rrc\n", rc));
3683 return VINF_SUCCESS;
3684 }
3685 else
3686 {
3687 /* Device is not present report SCSI selection timeout. */
3688 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_DEVICE_NOT_THERE;
3689 }
3690 }
3691 else
3692 {
3693 /* Report out of bounds target ID or bus. */
3694 if (pTaskState->GuestRequest.SCSIIO.u8Bus != 0)
3695 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_BUS;
3696 else
3697 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = MPT_SCSI_IO_ERROR_IOCSTATUS_INVALID_TARGETID;
3698 }
3699
3700 /* The rest is equal to both errors. */
3701 pTaskState->IOCReply.SCSIIOError.u8TargetID = pTaskState->GuestRequest.SCSIIO.u8TargetID;
3702 pTaskState->IOCReply.SCSIIOError.u8Bus = pTaskState->GuestRequest.SCSIIO.u8Bus;
3703 pTaskState->IOCReply.SCSIIOError.u8MessageLength = sizeof(MptSCSIIOErrorReply) / 4;
3704 pTaskState->IOCReply.SCSIIOError.u8Function = pTaskState->GuestRequest.SCSIIO.u8Function;
3705 pTaskState->IOCReply.SCSIIOError.u8CDBLength = pTaskState->GuestRequest.SCSIIO.u8CDBLength;
3706 pTaskState->IOCReply.SCSIIOError.u8SenseBufferLength = pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength;
3707 pTaskState->IOCReply.SCSIIOError.u32MessageContext = pTaskState->GuestRequest.SCSIIO.u32MessageContext;
3708 pTaskState->IOCReply.SCSIIOError.u8SCSIStatus = SCSI_STATUS_OK;
3709 pTaskState->IOCReply.SCSIIOError.u8SCSIState = MPT_SCSI_IO_ERROR_SCSI_STATE_TERMINATED;
3710 pTaskState->IOCReply.SCSIIOError.u32IOCLogInfo = 0;
3711 pTaskState->IOCReply.SCSIIOError.u32TransferCount = 0;
3712 pTaskState->IOCReply.SCSIIOError.u32SenseCount = 0;
3713 pTaskState->IOCReply.SCSIIOError.u32ResponseInfo = 0;
3714
3715 lsilogicFinishAddressReply(pLsiLogic, &pTaskState->IOCReply, false);
3716 RTCacheInsert(pLsiLogic->pTaskCache, pTaskState);
3717
3718 return rc;
3719}
3720
3721/**
3722 * Called upon completion of the request from the SCSI driver below.
3723 * This function frees all allocated ressources and notifies the guest
3724 * that the process finished by asserting an interrupt.
3725 *
3726 * @returns VBox status code.
3727 * @param pInterface Pointer to the interface the called funtion belongs to.
3728 * @param pSCSIRequest Pointer to the SCSI request which finished.
3729 */
3730static DECLCALLBACK(int) lsilogicDeviceSCSIRequestCompleted(PPDMISCSIPORT pInterface, PPDMSCSIREQUEST pSCSIRequest, int rcCompletion)
3731{
3732 PLSILOGICTASKSTATE pTaskState = (PLSILOGICTASKSTATE)pSCSIRequest->pvUser;
3733 PLSILOGICDEVICE pLsiLogicDevice = pTaskState->pTargetDevice;
3734 PLSILOGICSCSI pLsiLogic = pLsiLogicDevice->CTX_SUFF(pLsiLogic);
3735
3736 ASMAtomicDecU32(&pLsiLogicDevice->cOutstandingRequests);
3737
3738 if (RT_UNLIKELY(pTaskState->fBIOS))
3739 {
3740 int rc = vboxscsiRequestFinished(&pLsiLogic->VBoxSCSI, pSCSIRequest);
3741 AssertMsgRC(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc));
3742 }
3743 else
3744 {
3745#if 0
3746 lsilogicFreeGCSenseBuffer(pLsiLogic, pTaskState);
3747#else
3748 RTGCPHYS GCPhysAddrSenseBuffer;
3749
3750 GCPhysAddrSenseBuffer = pTaskState->GuestRequest.SCSIIO.u32SenseBufferLowAddress;
3751 GCPhysAddrSenseBuffer |= ((uint64_t)pLsiLogic->u32SenseBufferHighAddr << 32);
3752
3753 /* Copy the sense buffer over. */
3754 PDMDevHlpPhysWrite(pLsiLogic->CTX_SUFF(pDevIns), GCPhysAddrSenseBuffer, pTaskState->abSenseBuffer,
3755 RT_UNLIKELY(pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength < pTaskState->PDMScsiRequest.cbSenseBuffer)
3756 ? pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength
3757 : pTaskState->PDMScsiRequest.cbSenseBuffer);
3758#endif
3759 lsilogicScatterGatherListDestroy(pLsiLogic, pTaskState);
3760
3761
3762 if (RT_LIKELY(rcCompletion == SCSI_STATUS_OK))
3763 lsilogicFinishContextReply(pLsiLogic, pTaskState->GuestRequest.SCSIIO.u32MessageContext);
3764 else
3765 {
3766 /* The SCSI target encountered an error during processing post a reply. */
3767 memset(&pTaskState->IOCReply, 0, sizeof(MptReplyUnion));
3768 pTaskState->IOCReply.SCSIIOError.u8TargetID = pTaskState->GuestRequest.SCSIIO.u8TargetID;
3769 pTaskState->IOCReply.SCSIIOError.u8Bus = pTaskState->GuestRequest.SCSIIO.u8Bus;
3770 pTaskState->IOCReply.SCSIIOError.u8MessageLength = 8;
3771 pTaskState->IOCReply.SCSIIOError.u8Function = pTaskState->GuestRequest.SCSIIO.u8Function;
3772 pTaskState->IOCReply.SCSIIOError.u8CDBLength = pTaskState->GuestRequest.SCSIIO.u8CDBLength;
3773 pTaskState->IOCReply.SCSIIOError.u8SenseBufferLength = pTaskState->GuestRequest.SCSIIO.u8SenseBufferLength;
3774 pTaskState->IOCReply.SCSIIOError.u8MessageFlags = pTaskState->GuestRequest.SCSIIO.u8MessageFlags;
3775 pTaskState->IOCReply.SCSIIOError.u32MessageContext = pTaskState->GuestRequest.SCSIIO.u32MessageContext;
3776 pTaskState->IOCReply.SCSIIOError.u8SCSIStatus = rcCompletion;
3777 pTaskState->IOCReply.SCSIIOError.u8SCSIState = MPT_SCSI_IO_ERROR_SCSI_STATE_AUTOSENSE_VALID;
3778 pTaskState->IOCReply.SCSIIOError.u16IOCStatus = 0;
3779 pTaskState->IOCReply.SCSIIOError.u32IOCLogInfo = 0;
3780 pTaskState->IOCReply.SCSIIOError.u32TransferCount = 0;
3781 pTaskState->IOCReply.SCSIIOError.u32SenseCount = sizeof(pTaskState->abSenseBuffer);
3782 pTaskState->IOCReply.SCSIIOError.u32ResponseInfo = 0;
3783
3784 lsilogicFinishAddressReply(pLsiLogic, &pTaskState->IOCReply, true);
3785 }
3786 }
3787
3788 RTCacheInsert(pLsiLogic->pTaskCache, pTaskState);
3789
3790 return VINF_SUCCESS;
3791}
3792
3793/**
3794 * Return the configuration page header and data
3795 * which matches the given page type and number.
3796 *
3797 * @returns VINF_SUCCESS if successful
3798 * VERR_NOT_FOUND if the requested page could be found.
3799 * @param u8PageNumber Number of the page to get.
3800 * @param ppPageHeader Where to store the pointer to the page header.
3801 * @param ppbPageData Where to store the pointer to the page data.
3802 */
3803static int lsilogicConfigurationIOUnitPageGetFromNumber(PLSILOGICSCSI pLsiLogic, uint8_t u8PageNumber,
3804 PMptConfigurationPageHeader *ppPageHeader,
3805 uint8_t **ppbPageData, size_t *pcbPage)
3806{
3807 int rc = VINF_SUCCESS;
3808
3809 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
3810
3811 switch(u8PageNumber)
3812 {
3813 case 0:
3814 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOUnitPage0.u.fields.Header;
3815 *ppbPageData = pLsiLogic->ConfigurationPages.IOUnitPage0.u.abPageData;
3816 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOUnitPage0);
3817 break;
3818 case 1:
3819 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOUnitPage1.u.fields.Header;
3820 *ppbPageData = pLsiLogic->ConfigurationPages.IOUnitPage1.u.abPageData;
3821 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOUnitPage1);
3822 break;
3823 case 2:
3824 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOUnitPage2.u.fields.Header;
3825 *ppbPageData = pLsiLogic->ConfigurationPages.IOUnitPage2.u.abPageData;
3826 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOUnitPage2);
3827 break;
3828 case 3:
3829 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOUnitPage3.u.fields.Header;
3830 *ppbPageData = pLsiLogic->ConfigurationPages.IOUnitPage3.u.abPageData;
3831 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOUnitPage3);
3832 break;
3833 default:
3834 rc = VERR_NOT_FOUND;
3835 }
3836
3837 return rc;
3838}
3839
3840/**
3841 * Return the configuration page header and data
3842 * which matches the given page type and number.
3843 *
3844 * @returns VINF_SUCCESS if successful
3845 * VERR_NOT_FOUND if the requested page could be found.
3846 * @param u8PageNumber Number of the page to get.
3847 * @param ppPageHeader Where to store the pointer to the page header.
3848 * @param ppbPageData Where to store the pointer to the page data.
3849 */
3850static int lsilogicConfigurationIOCPageGetFromNumber(PLSILOGICSCSI pLsiLogic, uint8_t u8PageNumber,
3851 PMptConfigurationPageHeader *ppPageHeader,
3852 uint8_t **ppbPageData, size_t *pcbPage)
3853{
3854 int rc = VINF_SUCCESS;
3855
3856 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
3857
3858 switch(u8PageNumber)
3859 {
3860 case 0:
3861 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOCPage0.u.fields.Header;
3862 *ppbPageData = pLsiLogic->ConfigurationPages.IOCPage0.u.abPageData;
3863 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOCPage0);
3864 break;
3865 case 1:
3866 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOCPage1.u.fields.Header;
3867 *ppbPageData = pLsiLogic->ConfigurationPages.IOCPage1.u.abPageData;
3868 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOCPage1);
3869 break;
3870 case 2:
3871 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOCPage2.u.fields.Header;
3872 *ppbPageData = pLsiLogic->ConfigurationPages.IOCPage2.u.abPageData;
3873 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOCPage2);
3874 break;
3875 case 3:
3876 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOCPage3.u.fields.Header;
3877 *ppbPageData = pLsiLogic->ConfigurationPages.IOCPage3.u.abPageData;
3878 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOCPage3);
3879 break;
3880 case 4:
3881 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOCPage4.u.fields.Header;
3882 *ppbPageData = pLsiLogic->ConfigurationPages.IOCPage4.u.abPageData;
3883 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOCPage4);
3884 break;
3885 case 6:
3886 *ppPageHeader = &pLsiLogic->ConfigurationPages.IOCPage6.u.fields.Header;
3887 *ppbPageData = pLsiLogic->ConfigurationPages.IOCPage6.u.abPageData;
3888 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.IOCPage6);
3889 break;
3890 default:
3891 rc = VERR_NOT_FOUND;
3892 }
3893
3894 return rc;
3895}
3896
3897/**
3898 * Return the configuration page header and data
3899 * which matches the given page type and number.
3900 *
3901 * @returns VINF_SUCCESS if successful
3902 * VERR_NOT_FOUND if the requested page could be found.
3903 * @param u8PageNumber Number of the page to get.
3904 * @param ppPageHeader Where to store the pointer to the page header.
3905 * @param ppbPageData Where to store the pointer to the page data.
3906 */
3907static int lsilogicConfigurationManufacturingPageGetFromNumber(PLSILOGICSCSI pLsiLogic, uint8_t u8PageNumber,
3908 PMptConfigurationPageHeader *ppPageHeader,
3909 uint8_t **ppbPageData, size_t *pcbPage)
3910{
3911 int rc = VINF_SUCCESS;
3912
3913 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
3914
3915 switch(u8PageNumber)
3916 {
3917 case 0:
3918 *ppPageHeader = &pLsiLogic->ConfigurationPages.ManufacturingPage0.u.fields.Header;
3919 *ppbPageData = pLsiLogic->ConfigurationPages.ManufacturingPage0.u.abPageData;
3920 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.ManufacturingPage0);
3921 break;
3922 case 1:
3923 *ppPageHeader = &pLsiLogic->ConfigurationPages.ManufacturingPage1.Header;
3924 *ppbPageData = pLsiLogic->ConfigurationPages.ManufacturingPage1.abVPDInfo;
3925 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.ManufacturingPage1);
3926 break;
3927 case 2:
3928 *ppPageHeader = &pLsiLogic->ConfigurationPages.ManufacturingPage2.u.fields.Header;
3929 *ppbPageData = pLsiLogic->ConfigurationPages.ManufacturingPage2.u.abPageData;
3930 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.ManufacturingPage2);
3931 break;
3932 case 3:
3933 *ppPageHeader = &pLsiLogic->ConfigurationPages.ManufacturingPage3.u.fields.Header;
3934 *ppbPageData = pLsiLogic->ConfigurationPages.ManufacturingPage3.u.abPageData;
3935 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.ManufacturingPage3);
3936 break;
3937 case 4:
3938 *ppPageHeader = &pLsiLogic->ConfigurationPages.ManufacturingPage4.u.fields.Header;
3939 *ppbPageData = pLsiLogic->ConfigurationPages.ManufacturingPage4.u.abPageData;
3940 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.ManufacturingPage4);
3941 break;
3942 default:
3943 rc = VERR_NOT_FOUND;
3944 }
3945
3946 return rc;
3947}
3948
3949/**
3950 * Return the configuration page header and data
3951 * which matches the given page type and number.
3952 *
3953 * @returns VINF_SUCCESS if successful
3954 * VERR_NOT_FOUND if the requested page could be found.
3955 * @param u8PageNumber Number of the page to get.
3956 * @param ppPageHeader Where to store the pointer to the page header.
3957 * @param ppbPageData Where to store the pointer to the page data.
3958 */
3959static int lsilogicConfigurationSCSISPIPortPageGetFromNumber(PLSILOGICSCSI pLsiLogic, uint8_t u8Port,
3960 uint8_t u8PageNumber,
3961 PMptConfigurationPageHeader *ppPageHeader,
3962 uint8_t **ppbPageData, size_t *pcbPage)
3963{
3964 int rc = VINF_SUCCESS;
3965
3966 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
3967
3968 if (u8Port >= RT_ELEMENTS(pLsiLogic->ConfigurationPages.aPortPages))
3969 return VERR_NOT_FOUND;
3970
3971 switch(u8PageNumber)
3972 {
3973 case 0:
3974 *ppPageHeader = &pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage0.u.fields.Header;
3975 *ppbPageData = pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage0.u.abPageData;
3976 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage0);
3977 break;
3978 case 1:
3979 *ppPageHeader = &pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage1.u.fields.Header;
3980 *ppbPageData = pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage1.u.abPageData;
3981 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage1);
3982 break;
3983 case 2:
3984 *ppPageHeader = &pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage2.u.fields.Header;
3985 *ppbPageData = pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage2.u.abPageData;
3986 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aPortPages[u8Port].SCSISPIPortPage2);
3987 break;
3988 default:
3989 rc = VERR_NOT_FOUND;
3990 }
3991
3992 return rc;
3993}
3994
3995/**
3996 * Return the configuration page header and data
3997 * which matches the given page type and number.
3998 *
3999 * @returns VINF_SUCCESS if successful
4000 * VERR_NOT_FOUND if the requested page could be found.
4001 * @param u8PageNumber Number of the page to get.
4002 * @param ppPageHeader Where to store the pointer to the page header.
4003 * @param ppbPageData Where to store the pointer to the page data.
4004 */
4005static int lsilogicConfigurationSCSISPIDevicePageGetFromNumber(PLSILOGICSCSI pLsiLogic, uint8_t u8Bus,
4006 uint8_t u8TargetID, uint8_t u8PageNumber,
4007 PMptConfigurationPageHeader *ppPageHeader,
4008 uint8_t **ppbPageData, size_t *pcbPage)
4009{
4010 int rc = VINF_SUCCESS;
4011
4012 AssertMsg(VALID_PTR(ppPageHeader) && VALID_PTR(ppbPageData), ("Invalid parameters\n"));
4013
4014 if (u8Bus >= RT_ELEMENTS(pLsiLogic->ConfigurationPages.aBuses))
4015 return VERR_NOT_FOUND;
4016
4017 if (u8TargetID >= RT_ELEMENTS(pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages))
4018 return VERR_NOT_FOUND;
4019
4020 switch(u8PageNumber)
4021 {
4022 case 0:
4023 *ppPageHeader = &pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0.u.fields.Header;
4024 *ppbPageData = pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0.u.abPageData;
4025 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage0);
4026 break;
4027 case 1:
4028 *ppPageHeader = &pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1.u.fields.Header;
4029 *ppbPageData = pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1.u.abPageData;
4030 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage1);
4031 break;
4032 case 2:
4033 *ppPageHeader = &pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2.u.fields.Header;
4034 *ppbPageData = pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2.u.abPageData;
4035 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage2);
4036 break;
4037 case 3:
4038 *ppPageHeader = &pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3.u.fields.Header;
4039 *ppbPageData = pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3.u.abPageData;
4040 *pcbPage = sizeof(pLsiLogic->ConfigurationPages.aBuses[u8Bus].aDevicePages[u8TargetID].SCSISPIDevicePage3);
4041 break;
4042 default:
4043 rc = VERR_NOT_FOUND;
4044 }
4045
4046 return rc;
4047}
4048
4049/**
4050 * Processes a Configuration request.
4051 *
4052 * @returns VBox status code.
4053 * @param pLsiLogic Pointer to the device instance which sends the request.
4054 * @param pConfigurationReq Pointer to the request structure.
4055 * @param pReply Pointer to the reply message frame
4056 */
4057static int lsilogicProcessConfigurationRequest(PLSILOGICSCSI pLsiLogic, PMptConfigurationRequest pConfigurationReq,
4058 PMptConfigurationReply pReply)
4059{
4060 int rc = VINF_SUCCESS;
4061 uint8_t *pbPageData;
4062 PMptConfigurationPageHeader pPageHeader;
4063 uint8_t u8PageType, u8PageAttribute;
4064 size_t cbPage;
4065
4066 LogFlowFunc(("pLsiLogic=%#p\n", pLsiLogic));
4067
4068 u8PageType = MPT_CONFIGURATION_PAGE_TYPE_GET(pConfigurationReq->u8PageType);
4069 u8PageAttribute = MPT_CONFIGURATION_PAGE_ATTRIBUTE_GET(pConfigurationReq->u8PageType);
4070
4071 /* Copy common bits from the request into the reply. */
4072 pReply->u8MessageLength = 6; /* 6 32bit D-Words. */
4073 pReply->u8Action = pConfigurationReq->u8Action;
4074 pReply->u8Function = pConfigurationReq->u8Function;
4075 pReply->u32MessageContext = pConfigurationReq->u32MessageContext;
4076
4077 switch (u8PageType)
4078 {
4079 case MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT:
4080 {
4081 /* Get the page data. */
4082 rc = lsilogicConfigurationIOUnitPageGetFromNumber(pLsiLogic,
4083 pConfigurationReq->u8PageNumber,
4084 &pPageHeader, &pbPageData, &cbPage);
4085 break;
4086 }
4087 case MPT_CONFIGURATION_PAGE_TYPE_IOC:
4088 {
4089 /* Get the page data. */
4090 rc = lsilogicConfigurationIOCPageGetFromNumber(pLsiLogic,
4091 pConfigurationReq->u8PageNumber,
4092 &pPageHeader, &pbPageData, &cbPage);
4093 break;
4094 }
4095 case MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING:
4096 {
4097 /* Get the page data. */
4098 rc = lsilogicConfigurationManufacturingPageGetFromNumber(pLsiLogic,
4099 pConfigurationReq->u8PageNumber,
4100 &pPageHeader, &pbPageData, &cbPage);
4101 break;
4102 }
4103 case MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT:
4104 {
4105 /* Get the page data. */
4106 rc = lsilogicConfigurationSCSISPIPortPageGetFromNumber(pLsiLogic,
4107 pConfigurationReq->u.MPIPortNumber.u8PortNumber,
4108 pConfigurationReq->u8PageNumber,
4109 &pPageHeader, &pbPageData, &cbPage);
4110 break;
4111 }
4112 case MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE:
4113 {
4114 /* Get the page data. */
4115 rc = lsilogicConfigurationSCSISPIDevicePageGetFromNumber(pLsiLogic,
4116 pConfigurationReq->u.BusAndTargetId.u8Bus,
4117 pConfigurationReq->u.BusAndTargetId.u8TargetID,
4118 pConfigurationReq->u8PageNumber,
4119 &pPageHeader, &pbPageData, &cbPage);
4120 break;
4121 }
4122 default:
4123 rc = VERR_NOT_FOUND;
4124 }
4125
4126 if (rc == VERR_NOT_FOUND)
4127 {
4128 //AssertMsgFailed(("todo\n"));
4129 pReply->u8PageType = pConfigurationReq->u8PageType;
4130 pReply->u8PageNumber = pConfigurationReq->u8PageNumber;
4131 pReply->u8PageLength = pConfigurationReq->u8PageLength;
4132 pReply->u8PageVersion = pConfigurationReq->u8PageVersion;
4133 return VINF_SUCCESS;
4134 }
4135
4136 pReply->u8PageType = pPageHeader->u8PageType;
4137 pReply->u8PageNumber = pPageHeader->u8PageNumber;
4138 pReply->u8PageLength = pPageHeader->u8PageLength;
4139 pReply->u8PageVersion = pPageHeader->u8PageVersion;
4140
4141 Log(("GuestRequest u8Action=%d\n", pConfigurationReq->u8Action));
4142 Log(("u8PageType=%d\n", pPageHeader->u8PageType));
4143 Log(("u8PageNumber=%d\n", pPageHeader->u8PageNumber));
4144 Log(("u8PageLength=%d\n", pPageHeader->u8PageLength));
4145 Log(("u8PageVersion=%d\n", pPageHeader->u8PageVersion));
4146
4147 for (int i = 0; i < pReply->u8PageLength; i++)
4148 LogFlowFunc(("PageData[%d]=%#x\n", i, ((uint32_t *)pbPageData)[i]));
4149
4150 /*
4151 * Don't use the scatter gather handling code as the configuration request always have only one
4152 * simple element.
4153 */
4154 switch (pConfigurationReq->u8Action)
4155 {
4156 case MPT_CONFIGURATION_REQUEST_ACTION_DEFAULT: /* Nothing to do. We are always using the defaults. */
4157 case MPT_CONFIGURATION_REQUEST_ACTION_HEADER:
4158 {
4159 /* Already copied above nothing to do. */
4160 break;
4161 }
4162 case MPT_CONFIGURATION_REQUEST_ACTION_READ_NVRAM:
4163 case MPT_CONFIGURATION_REQUEST_ACTION_READ_CURRENT:
4164 case MPT_CONFIGURATION_REQUEST_ACTION_READ_DEFAULT:
4165 {
4166 uint32_t cbBuffer = pConfigurationReq->SimpleSGElement.u24Length;
4167 if (cbBuffer != 0)
4168 {
4169 RTGCPHYS GCPhysAddrPageBuffer = pConfigurationReq->SimpleSGElement.u32DataBufferAddressLow;
4170 if (pConfigurationReq->SimpleSGElement.f64BitAddress)
4171 GCPhysAddrPageBuffer |= (uint64_t)pConfigurationReq->SimpleSGElement.u32DataBufferAddressHigh << 32;
4172
4173 PDMDevHlpPhysWrite(pLsiLogic->CTX_SUFF(pDevIns), GCPhysAddrPageBuffer, pbPageData,
4174 cbBuffer < cbPage ? cbBuffer : cbPage);
4175 }
4176 break;
4177 }
4178 case MPT_CONFIGURATION_REQUEST_ACTION_WRITE_CURRENT:
4179 case MPT_CONFIGURATION_REQUEST_ACTION_WRITE_NVRAM:
4180 {
4181 uint32_t cbBuffer = pConfigurationReq->SimpleSGElement.u24Length;
4182 if (cbBuffer != 0)
4183 {
4184 RTGCPHYS GCPhysAddrPageBuffer = pConfigurationReq->SimpleSGElement.u32DataBufferAddressLow;
4185 if (pConfigurationReq->SimpleSGElement.f64BitAddress)
4186 GCPhysAddrPageBuffer |= (uint64_t)pConfigurationReq->SimpleSGElement.u32DataBufferAddressHigh << 32;
4187
4188 PDMDevHlpPhysRead(pLsiLogic->CTX_SUFF(pDevIns), GCPhysAddrPageBuffer, pbPageData,
4189 cbBuffer < cbPage ? cbBuffer : cbPage);
4190 }
4191 break;
4192 }
4193 default:
4194 AssertMsgFailed(("todo\n"));
4195 }
4196
4197 return VINF_SUCCESS;
4198}
4199
4200/**
4201 * Initializes the configuration pages.
4202 *
4203 * @returns nothing
4204 * @param pLsiLogic Pointer to the Lsilogic SCSI instance.
4205 */
4206static void lsilogicInitializeConfigurationPages(PLSILOGICSCSI pLsiLogic)
4207{
4208 PMptConfigurationPagesSupported pPages = &pLsiLogic->ConfigurationPages;
4209
4210 LogFlowFunc(("pLsiLogic=%#p\n", pLsiLogic));
4211
4212 /* Clear everything first. */
4213 memset(pPages, 0, sizeof(MptConfigurationPagesSupported));
4214
4215 /* Manufacturing Page 0. */
4216 pPages->ManufacturingPage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY
4217 | MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING;
4218 pPages->ManufacturingPage0.u.fields.Header.u8PageNumber = 0;
4219 pPages->ManufacturingPage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageManufacturing0) / 4;
4220 strncpy((char *)pPages->ManufacturingPage0.u.fields.abChipName, "VBox MPT Fusion", 16);
4221 strncpy((char *)pPages->ManufacturingPage0.u.fields.abChipRevision, "1.0", 8);
4222 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardName, "VBox MPT Fusion", 16);
4223 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardAssembly, "SUN", 8);
4224 strncpy((char *)pPages->ManufacturingPage0.u.fields.abBoardTracerNumber, "CAFECAFECAFECAFE", 16);
4225
4226 /* Manufacturing Page 1 - I don't know what this contains so we leave it 0 for now. */
4227 pPages->ManufacturingPage1.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY
4228 | MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING;
4229 pPages->ManufacturingPage1.Header.u8PageNumber = 1;
4230 pPages->ManufacturingPage1.Header.u8PageLength = sizeof(MptConfigurationPageManufacturing1) / 4;
4231
4232 /* Manufacturing Page 2. */
4233 pPages->ManufacturingPage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4234 | MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING;
4235 pPages->ManufacturingPage2.u.fields.Header.u8PageNumber = 2;
4236 pPages->ManufacturingPage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageManufacturing2) / 4;
4237 pPages->ManufacturingPage2.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_DEVICE_ID;
4238 pPages->ManufacturingPage2.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_REVISION_ID;
4239 /* Hardware specific settings - everything 0 for now. */
4240
4241 /* Manufacturing Page 3. */
4242 pPages->ManufacturingPage3.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4243 | MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING;
4244 pPages->ManufacturingPage3.u.fields.Header.u8PageNumber = 3;
4245 pPages->ManufacturingPage3.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageManufacturing3) / 4;
4246 pPages->ManufacturingPage3.u.fields.u16PCIDeviceID = LSILOGICSCSI_PCI_DEVICE_ID;
4247 pPages->ManufacturingPage3.u.fields.u8PCIRevisionID = LSILOGICSCSI_PCI_REVISION_ID;
4248 /* Chip specific settings - everything 0 for now. */
4249
4250 /* Manufacturing Page 4 - I don't know what this contains so we leave it 0 for now. */
4251 pPages->ManufacturingPage4.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT_READONLY
4252 | MPT_CONFIGURATION_PAGE_TYPE_MANUFACTURING;
4253 pPages->ManufacturingPage4.u.fields.Header.u8PageNumber = 4;
4254 pPages->ManufacturingPage4.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageManufacturing4) / 4;
4255
4256
4257 /* I/O Unit page 0. */
4258 pPages->IOUnitPage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4259 | MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT;
4260 pPages->IOUnitPage0.u.fields.Header.u8PageNumber = 0;
4261 pPages->IOUnitPage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOUnit0) / 4;
4262 pPages->IOUnitPage0.u.fields.u64UniqueIdentifier = 0xcafe;
4263
4264 /* I/O Unit page 1. */
4265 pPages->IOUnitPage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4266 | MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT;
4267 pPages->IOUnitPage1.u.fields.Header.u8PageNumber = 1;
4268 pPages->IOUnitPage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOUnit1) / 4;
4269 pPages->IOUnitPage1.u.fields.fSingleFunction = true;
4270 pPages->IOUnitPage1.u.fields.fAllPathsMapped = false;
4271 pPages->IOUnitPage1.u.fields.fIntegratedRAIDDisabled = true;
4272 pPages->IOUnitPage1.u.fields.f32BitAccessForced = false;
4273
4274 /* I/O Unit page 2. */
4275 pPages->IOUnitPage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_PERSISTENT
4276 | MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT;
4277 pPages->IOUnitPage2.u.fields.Header.u8PageNumber = 2;
4278 pPages->IOUnitPage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOUnit2) / 4;
4279 pPages->IOUnitPage2.u.fields.fPauseOnError = false;
4280 pPages->IOUnitPage2.u.fields.fVerboseModeEnabled = false;
4281 pPages->IOUnitPage2.u.fields.fDisableColorVideo = false;
4282 pPages->IOUnitPage2.u.fields.fNotHookInt40h = false;
4283 pPages->IOUnitPage2.u.fields.u32BIOSVersion = 0xcafecafe;
4284 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].fAdapterEnabled = true;
4285 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].fAdapterEmbedded = true;
4286 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].u8PCIBusNumber = 0;
4287 pPages->IOUnitPage2.u.fields.aAdapterOrder[0].u8PCIDevFn = pLsiLogic->PciDev.devfn;
4288
4289 /* I/O Unit page 3. */
4290 pPages->IOUnitPage3.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
4291 | MPT_CONFIGURATION_PAGE_TYPE_IO_UNIT;
4292 pPages->IOUnitPage3.u.fields.Header.u8PageNumber = 3;
4293 pPages->IOUnitPage3.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOUnit3) / 4;
4294 pPages->IOUnitPage3.u.fields.u8GPIOCount = 0;
4295
4296 /* IOC page 0. */
4297 pPages->IOCPage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4298 | MPT_CONFIGURATION_PAGE_TYPE_IOC;
4299 pPages->IOCPage0.u.fields.Header.u8PageNumber = 0;
4300 pPages->IOCPage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOC0) / 4;
4301 pPages->IOCPage0.u.fields.u32TotalNVStore = 0;
4302 pPages->IOCPage0.u.fields.u32FreeNVStore = 0;
4303 pPages->IOCPage0.u.fields.u16VendorId = LSILOGICSCSI_PCI_VENDOR_ID;
4304 pPages->IOCPage0.u.fields.u16DeviceId = LSILOGICSCSI_PCI_DEVICE_ID;
4305 pPages->IOCPage0.u.fields.u8RevisionId = LSILOGICSCSI_PCI_REVISION_ID;
4306 pPages->IOCPage0.u.fields.u32ClassCode = LSILOGICSCSI_PCI_CLASS_CODE;
4307 pPages->IOCPage0.u.fields.u16SubsystemVendorId = LSILOGICSCSI_PCI_SUBSYSTEM_VENDOR_ID;
4308 pPages->IOCPage0.u.fields.u16SubsystemId = LSILOGICSCSI_PCI_SUBSYSTEM_ID;
4309
4310 /* IOC page 1. */
4311 pPages->IOCPage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
4312 | MPT_CONFIGURATION_PAGE_TYPE_IOC;
4313 pPages->IOCPage1.u.fields.Header.u8PageNumber = 1;
4314 pPages->IOCPage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOC1) / 4;
4315 pPages->IOCPage1.u.fields.fReplyCoalescingEnabled = false;
4316 pPages->IOCPage1.u.fields.u32CoalescingTimeout = 0;
4317 pPages->IOCPage1.u.fields.u8CoalescingDepth = 0;
4318
4319 /* IOC page 2. */
4320 pPages->IOCPage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4321 | MPT_CONFIGURATION_PAGE_TYPE_IOC;
4322 pPages->IOCPage2.u.fields.Header.u8PageNumber = 2;
4323 pPages->IOCPage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOC2) / 4;
4324 /* Everything else here is 0. */
4325
4326 /* IOC page 3. */
4327 pPages->IOCPage3.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4328 | MPT_CONFIGURATION_PAGE_TYPE_IOC;
4329 pPages->IOCPage3.u.fields.Header.u8PageNumber = 3;
4330 pPages->IOCPage3.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOC3) / 4;
4331 /* Everything else here is 0. */
4332
4333 /* IOC page 4. */
4334 pPages->IOCPage4.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4335 | MPT_CONFIGURATION_PAGE_TYPE_IOC;
4336 pPages->IOCPage4.u.fields.Header.u8PageNumber = 4;
4337 pPages->IOCPage4.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOC4) / 4;
4338 /* Everything else here is 0. */
4339
4340 /* IOC page 6. */
4341 pPages->IOCPage6.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4342 | MPT_CONFIGURATION_PAGE_TYPE_IOC;
4343 pPages->IOCPage6.u.fields.Header.u8PageNumber = 6;
4344 pPages->IOCPage6.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageIOC6) / 4;
4345 /* Everything else here is 0. */
4346
4347 for (unsigned i = 0; i < RT_ELEMENTS(pPages->aPortPages); i++)
4348 {
4349 /* SCSI-SPI port page 0. */
4350 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4351 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
4352 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageNumber = 0;
4353 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort0) / 4;
4354 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fInformationUnitTransfersCapable = true;
4355 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fDTCapable = true;
4356 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fQASCapable = true;
4357 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u8MinimumSynchronousTransferPeriod = 0;
4358 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u8MaximumSynchronousOffset = 0xff;
4359 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fWide = true;
4360 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.fAIPCapable = true;
4361 pPages->aPortPages[i].SCSISPIPortPage0.u.fields.u2SignalingType = 0x3; /* Single Ended. */
4362
4363 /* SCSI-SPI port page 1. */
4364 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
4365 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
4366 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageNumber = 1;
4367 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort1) / 4;
4368 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u8SCSIID = 7;
4369 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u16PortResponseIDsBitmask = (1 << 7);
4370 pPages->aPortPages[i].SCSISPIPortPage1.u.fields.u32OnBusTimerValue = 0;
4371
4372 /* SCSI-SPI port page 2. */
4373 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
4374 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_PORT;
4375 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageNumber = 2;
4376 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIPort2) / 4;
4377 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.u4HostSCSIID = 7;
4378 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.u2InitializeHBA = 0x3;
4379 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.fTerminationDisabled = true;
4380 for (unsigned iDevice = 0; iDevice < RT_ELEMENTS(pPages->aPortPages[i].SCSISPIPortPage2.u.fields.aDeviceSettings); iDevice++)
4381 {
4382 pPages->aPortPages[i].SCSISPIPortPage2.u.fields.aDeviceSettings[iDevice].fBootChoice = true;
4383 }
4384 /* Everything else 0 for now. */
4385 }
4386
4387 for (unsigned uBusCurr = 0; uBusCurr < RT_ELEMENTS(pPages->aBuses); uBusCurr++)
4388 {
4389 for (unsigned uDeviceCurr = 0; uDeviceCurr < RT_ELEMENTS(pPages->aBuses[uBusCurr].aDevicePages); uDeviceCurr++)
4390 {
4391 /* SCSI-SPI device page 0. */
4392 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4393 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
4394 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageNumber = 0;
4395 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage0.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice0) / 4;
4396 /* Everything else 0 for now. */
4397
4398 /* SCSI-SPI device page 1. */
4399 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
4400 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
4401 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageNumber = 1;
4402 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage1.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice1) / 4;
4403 /* Everything else 0 for now. */
4404
4405 /* SCSI-SPI device page 2. */
4406 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_CHANGEABLE
4407 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
4408 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageNumber = 2;
4409 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage2.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice2) / 4;
4410 /* Everything else 0 for now. */
4411
4412 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageType = MPT_CONFIGURATION_PAGE_ATTRIBUTE_READONLY
4413 | MPT_CONFIGURATION_PAGE_TYPE_SCSI_SPI_DEVICE;
4414 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageNumber = 3;
4415 pPages->aBuses[uBusCurr].aDevicePages[uDeviceCurr].SCSISPIDevicePage3.u.fields.Header.u8PageLength = sizeof(MptConfigurationPageSCSISPIDevice3) / 4;
4416 /* Everything else 0 for now. */
4417 }
4418 }
4419}
4420
4421/**
4422 * Transmit queue consumer
4423 * Queue a new async task.
4424 *
4425 * @returns Success indicator.
4426 * If false the item will not be removed and the flushing will stop.
4427 * @param pDevIns The device instance.
4428 * @param pItem The item to consume. Upon return this item will be freed.
4429 */
4430static DECLCALLBACK(bool) lsilogicNotifyQueueConsumer(PPDMDEVINS pDevIns, PPDMQUEUEITEMCORE pItem)
4431{
4432 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4433 int rc = VINF_SUCCESS;
4434
4435 LogFlowFunc(("pDevIns=%#p pItem=%#p\n", pDevIns, pItem));
4436
4437 /* Only process request which arrived before we received the notification. */
4438 uint32_t uRequestQueueNextEntryWrite = ASMAtomicReadU32(&pLsiLogic->uRequestQueueNextEntryFreeWrite);
4439
4440 /* Reset notification event. */
4441 ASMAtomicXchgBool(&pLsiLogic->fNotificationSend, false);
4442
4443 /* Go through the messages now and process them. */
4444 while ( RT_LIKELY(pLsiLogic->enmState == LSILOGICSTATE_OPERATIONAL)
4445 && (pLsiLogic->uRequestQueueNextAddressRead != uRequestQueueNextEntryWrite))
4446 {
4447 uint32_t u32RequestMessageFrameDesc = pLsiLogic->CTX_SUFF(pRequestQueueBase)[pLsiLogic->uRequestQueueNextAddressRead];
4448 RTGCPHYS GCPhysMessageFrameAddr = LSILOGIC_RTGCPHYS_FROM_U32(pLsiLogic->u32HostMFAHighAddr,
4449 (u32RequestMessageFrameDesc & ~0x07));
4450
4451 PLSILOGICTASKSTATE pTaskState;
4452
4453 /* Get new task state. */
4454 rc = RTCacheRequest(pLsiLogic->pTaskCache, (void **)&pTaskState);
4455 AssertRC(rc);
4456
4457 pTaskState->GCPhysMessageFrameAddr = GCPhysMessageFrameAddr;
4458
4459 /* Read the message header from the guest first. */
4460 PDMDevHlpPhysRead(pDevIns, GCPhysMessageFrameAddr, &pTaskState->GuestRequest, sizeof(MptMessageHdr));
4461
4462 /* Determine the size of the request. */
4463 uint32_t cbRequest = 0;
4464
4465 switch (pTaskState->GuestRequest.Header.u8Function)
4466 {
4467 case MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST:
4468 cbRequest = sizeof(MptSCSIIORequest);
4469 break;
4470 case MPT_MESSAGE_HDR_FUNCTION_SCSI_TASK_MGMT:
4471 cbRequest = sizeof(MptSCSITaskManagementRequest);
4472 break;
4473 case MPT_MESSAGE_HDR_FUNCTION_IOC_INIT:
4474 cbRequest = sizeof(MptIOCInitRequest);
4475 break;
4476 case MPT_MESSAGE_HDR_FUNCTION_IOC_FACTS:
4477 cbRequest = sizeof(MptIOCFactsRequest);
4478 break;
4479 case MPT_MESSAGE_HDR_FUNCTION_CONFIG:
4480 cbRequest = sizeof(MptConfigurationRequest);
4481 break;
4482 case MPT_MESSAGE_HDR_FUNCTION_PORT_FACTS:
4483 cbRequest = sizeof(MptPortFactsRequest);
4484 break;
4485 case MPT_MESSAGE_HDR_FUNCTION_PORT_ENABLE:
4486 cbRequest = sizeof(MptPortEnableRequest);
4487 break;
4488 case MPT_MESSAGE_HDR_FUNCTION_EVENT_NOTIFICATION:
4489 cbRequest = sizeof(MptEventNotificationRequest);
4490 break;
4491 case MPT_MESSAGE_HDR_FUNCTION_EVENT_ACK:
4492 AssertMsgFailed(("todo\n"));
4493 //cbRequest = sizeof(MptEventAckRequest);
4494 break;
4495 case MPT_MESSAGE_HDR_FUNCTION_FW_DOWNLOAD:
4496 AssertMsgFailed(("todo\n"));
4497 break;
4498 default:
4499 AssertMsgFailed(("Unknown function issued %u\n", pTaskState->GuestRequest.Header.u8Function));
4500 lsilogicSetIOCFaultCode(pLsiLogic, LSILOGIC_IOCSTATUS_INVALID_FUNCTION);
4501 }
4502
4503 if (cbRequest != 0)
4504 {
4505 /* Read the complete message frame from guest memory now. */
4506 PDMDevHlpPhysRead(pDevIns, GCPhysMessageFrameAddr, &pTaskState->GuestRequest, cbRequest);
4507
4508 /* Handle SCSI I/O requests now. */
4509 if (pTaskState->GuestRequest.Header.u8Function == MPT_MESSAGE_HDR_FUNCTION_SCSI_IO_REQUEST)
4510 {
4511 rc = lsilogicProcessSCSIIORequest(pLsiLogic, pTaskState);
4512 AssertRC(rc);
4513 }
4514 else
4515 {
4516 MptReplyUnion Reply;
4517 rc = lsilogicProcessMessageRequest(pLsiLogic, &pTaskState->GuestRequest.Header, &Reply);
4518 AssertRC(rc);
4519 RTCacheInsert(pLsiLogic->pTaskCache, pTaskState);
4520 }
4521
4522 pLsiLogic->uRequestQueueNextAddressRead++;
4523 pLsiLogic->uRequestQueueNextAddressRead %= pLsiLogic->cRequestQueueEntries;
4524 }
4525 }
4526
4527 return true;
4528}
4529
4530/**
4531 * Port I/O Handler for IN operations - legacy port.
4532 *
4533 * @returns VBox status code.
4534 *
4535 * @param pDevIns The device instance.
4536 * @param pvUser User argument.
4537 * @param uPort Port number used for the IN operation.
4538 * @param pu32 Where to store the result.
4539 * @param cb Number of bytes read.
4540 */
4541static int lsilogicIsaIOPortRead (PPDMDEVINS pDevIns, void *pvUser,
4542 RTIOPORT Port, uint32_t *pu32, unsigned cb)
4543{
4544 int rc;
4545 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4546
4547 Assert(cb == 1);
4548
4549 rc = vboxscsiReadRegister(&pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT), pu32);
4550
4551 Log2(("%s: pu32=%p:{%.*Rhxs} iRegister=%d rc=%Rrc\n",
4552 __FUNCTION__, pu32, 1, pu32, (Port - LSILOGIC_ISA_IO_PORT), rc));
4553
4554 return rc;
4555}
4556
4557/**
4558 * Prepares a request from the BIOS.
4559 *
4560 * @returns VBox status code.
4561 * @param pLsiLogic Pointer to the LsiLogic device instance.
4562 */
4563static int lsilogicPrepareBIOSSCSIRequest(PLSILOGICSCSI pLsiLogic)
4564{
4565 int rc;
4566 PLSILOGICTASKSTATE pTaskState;
4567 uint32_t uTargetDevice;
4568
4569 rc = RTCacheRequest(pLsiLogic->pTaskCache, (void **)&pTaskState);
4570 AssertMsgRCReturn(rc, ("Getting task from cache failed rc=%Rrc\n", rc), rc);
4571
4572 pTaskState->fBIOS = true;
4573
4574 rc = vboxscsiSetupRequest(&pLsiLogic->VBoxSCSI, &pTaskState->PDMScsiRequest, &uTargetDevice);
4575 AssertMsgRCReturn(rc, ("Setting up SCSI request failed rc=%Rrc\n", rc), rc);
4576
4577 pTaskState->PDMScsiRequest.pvUser = pTaskState;
4578
4579 pTaskState->pTargetDevice = &pLsiLogic->aDeviceStates[uTargetDevice];
4580
4581 if (!pTaskState->pTargetDevice->pDrvBase)
4582 {
4583 /* Device is not present. */
4584 AssertMsg(pTaskState->PDMScsiRequest.pbCDB[0] == SCSI_INQUIRY,
4585 ("Device is not present but command is not inquiry\n"));
4586
4587 SCSIINQUIRYDATA ScsiInquiryData;
4588
4589 memset(&ScsiInquiryData, 0, sizeof(SCSIINQUIRYDATA));
4590 ScsiInquiryData.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_UNKNOWN;
4591 ScsiInquiryData.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_NOT_CONNECTED_NOT_SUPPORTED;
4592
4593 memcpy(pLsiLogic->VBoxSCSI.pBuf, &ScsiInquiryData, 5);
4594
4595 rc = vboxscsiRequestFinished(&pLsiLogic->VBoxSCSI, &pTaskState->PDMScsiRequest);
4596 AssertMsgRCReturn(rc, ("Finishing BIOS SCSI request failed rc=%Rrc\n", rc), rc);
4597
4598 rc = RTCacheInsert(pLsiLogic->pTaskCache, pTaskState);
4599 AssertMsgRCReturn(rc, ("Getting task from cache failed rc=%Rrc\n", rc), rc);
4600 }
4601 else
4602 {
4603 ASMAtomicIncU32(&pTaskState->pTargetDevice->cOutstandingRequests);
4604
4605 rc = pTaskState->pTargetDevice->pDrvSCSIConnector->pfnSCSIRequestSend(pTaskState->pTargetDevice->pDrvSCSIConnector,
4606 &pTaskState->PDMScsiRequest);
4607 AssertMsgRCReturn(rc, ("Sending request to SCSI layer failed rc=%Rrc\n", rc), rc);
4608 }
4609
4610 return rc;
4611}
4612
4613/**
4614 * Port I/O Handler for OUT operations - legacy port.
4615 *
4616 * @returns VBox status code.
4617 *
4618 * @param pDevIns The device instance.
4619 * @param pvUser User argument.
4620 * @param uPort Port number used for the IN operation.
4621 * @param u32 The value to output.
4622 * @param cb The value size in bytes.
4623 */
4624static int lsilogicIsaIOPortWrite (PPDMDEVINS pDevIns, void *pvUser,
4625 RTIOPORT Port, uint32_t u32, unsigned cb)
4626{
4627 int rc;
4628 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4629
4630 Log2(("#%d %s: pvUser=%#p cb=%d u32=%#x Port=%#x\n",
4631 pDevIns->iInstance, __FUNCTION__, pvUser, cb, u32, Port));
4632
4633 Assert(cb == 1);
4634
4635 rc = vboxscsiWriteRegister(&pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT), (uint8_t)u32);
4636 if (rc == VERR_MORE_DATA)
4637 {
4638 rc = lsilogicPrepareBIOSSCSIRequest(pLsiLogic);
4639 AssertRC(rc);
4640 }
4641 else if (RT_FAILURE(rc))
4642 AssertMsgFailed(("Writing BIOS register failed %Rrc\n", rc));
4643
4644 return VINF_SUCCESS;
4645}
4646
4647/**
4648 * Port I/O Handler for primary port range OUT string operations.
4649 * @see FNIOMIOPORTOUTSTRING for details.
4650 */
4651static DECLCALLBACK(int) lsilogicIsaIOPortWriteStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, RTGCPTR *pGCPtrSrc, PRTGCUINTREG pcTransfer, unsigned cb)
4652{
4653 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4654 int rc;
4655
4656 Log2(("#%d %s: pvUser=%#p cb=%d Port=%#x\n",
4657 pDevIns->iInstance, __FUNCTION__, pvUser, cb, Port));
4658
4659 rc = vboxscsiWriteString(pDevIns, &pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT),
4660 pGCPtrSrc, pcTransfer, cb);
4661 if (rc == VERR_MORE_DATA)
4662 {
4663 rc = lsilogicPrepareBIOSSCSIRequest(pLsiLogic);
4664 AssertRC(rc);
4665 }
4666 else if (RT_FAILURE(rc))
4667 AssertMsgFailed(("Writing BIOS register failed %Rrc\n", rc));
4668
4669 return rc;
4670}
4671
4672/**
4673 * Port I/O Handler for primary port range IN string operations.
4674 * @see FNIOMIOPORTINSTRING for details.
4675 */
4676static DECLCALLBACK(int) lsilogicIsaIOPortReadStr(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, RTGCPTR *pGCPtrDst, PRTGCUINTREG pcTransfer, unsigned cb)
4677{
4678 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4679
4680 LogFlowFunc(("#%d %s: pvUser=%#p cb=%d Port=%#x\n",
4681 pDevIns->iInstance, __FUNCTION__, pvUser, cb, Port));
4682
4683 return vboxscsiReadString(pDevIns, &pLsiLogic->VBoxSCSI, (Port - LSILOGIC_ISA_IO_PORT),
4684 pGCPtrDst, pcTransfer, cb);
4685}
4686
4687static DECLCALLBACK(int) lsilogicMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
4688 RTGCPHYS GCPhysAddress, uint32_t cb,
4689 PCIADDRESSSPACE enmType)
4690{
4691 PPDMDEVINS pDevIns = pPciDev->pDevIns;
4692 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4693 int rc = VINF_SUCCESS;
4694
4695 Log2(("%s: registering area at GCPhysAddr=%RGp cb=%u\n", __FUNCTION__, GCPhysAddress, cb));
4696
4697 AssertMsg( (enmType == PCI_ADDRESS_SPACE_MEM && cb >= LSILOGIC_PCI_SPACE_MEM_SIZE)
4698 || (enmType == PCI_ADDRESS_SPACE_IO && cb >= LSILOGIC_PCI_SPACE_IO_SIZE),
4699 ("PCI region type and size do not match\n"));
4700
4701 if ((enmType == PCI_ADDRESS_SPACE_MEM) && (iRegion == 1))
4702 {
4703 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
4704 rc = PDMDevHlpMMIORegister(pDevIns, GCPhysAddress, cb, NULL,
4705 lsilogicMMIOWrite, lsilogicMMIORead, NULL, "LsiLogic");
4706 if (RT_FAILURE(rc))
4707 return rc;
4708
4709 if (pThis->fR0Enabled)
4710 {
4711 rc = PDMDevHlpMMIORegisterR0(pDevIns, GCPhysAddress, cb, 0,
4712 "lsilogicMMIOWrite", "lsilogicMMIORead", NULL);
4713 if (RT_FAILURE(rc))
4714 return rc;
4715 }
4716
4717 if (pThis->fGCEnabled)
4718 {
4719 rc = PDMDevHlpMMIORegisterGC(pDevIns, GCPhysAddress, cb, 0,
4720 "lsilogicMMIOWrite", "lsilogicMMIORead", NULL);
4721 if (RT_FAILURE(rc))
4722 return rc;
4723 }
4724
4725 pThis->GCPhysMMIOBase = GCPhysAddress;
4726 }
4727 else if ((enmType == PCI_ADDRESS_SPACE_MEM) && (iRegion == 2))
4728 {
4729 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
4730 rc = PDMDevHlpMMIORegister(pDevIns, GCPhysAddress, cb, NULL,
4731 lsilogicDiagnosticWrite, lsilogicDiagnosticRead, NULL, "LsiLogicDiag");
4732 if (RT_FAILURE(rc))
4733 return rc;
4734
4735 if (pThis->fR0Enabled)
4736 {
4737 rc = PDMDevHlpMMIORegisterR0(pDevIns, GCPhysAddress, cb, 0,
4738 "lsilogicDiagnosticWrite", "lsilogicDiagnosticRead", NULL);
4739 if (RT_FAILURE(rc))
4740 return rc;
4741 }
4742
4743 if (pThis->fGCEnabled)
4744 {
4745 rc = PDMDevHlpMMIORegisterGC(pDevIns, GCPhysAddress, cb, 0,
4746 "lsilogicDiagnosticWrite", "lsilogicDiagnosticRead", NULL);
4747 if (RT_FAILURE(rc))
4748 return rc;
4749 }
4750 }
4751 else if (enmType == PCI_ADDRESS_SPACE_IO)
4752 {
4753 rc = PDMDevHlpIOPortRegister(pDevIns, (RTIOPORT)GCPhysAddress, LSILOGIC_PCI_SPACE_IO_SIZE,
4754 NULL, lsilogicIOPortWrite, lsilogicIOPortRead, NULL, NULL, "LsiLogic");
4755 if (RT_FAILURE(rc))
4756 return rc;
4757
4758 if (pThis->fR0Enabled)
4759 {
4760 rc = PDMDevHlpIOPortRegisterR0(pDevIns, (RTIOPORT)GCPhysAddress, LSILOGIC_PCI_SPACE_IO_SIZE,
4761 0, "lsilogicIOPortWrite", "lsilogicIOPortRead", NULL, NULL, "LsiLogic");
4762 if (RT_FAILURE(rc))
4763 return rc;
4764 }
4765
4766 if (pThis->fGCEnabled)
4767 {
4768 rc = PDMDevHlpIOPortRegisterGC(pDevIns, (RTIOPORT)GCPhysAddress, LSILOGIC_PCI_SPACE_IO_SIZE,
4769 0, "lsilogicIOPortWrite", "lsilogicIOPortRead", NULL, NULL, "LsiLogic");
4770 if (RT_FAILURE(rc))
4771 return rc;
4772 }
4773
4774 pThis->IOPortBase = (RTIOPORT)GCPhysAddress;
4775 }
4776 else
4777 AssertMsgFailed(("Invalid enmType=%d iRegion=%d\n", enmType, iRegion));
4778
4779 return rc;
4780}
4781
4782/**
4783 * Waits until all I/O operations on all devices are complete.
4784 *
4785 * @retruns Flag which indicates if all I/O completed in the given timeout.
4786 * @param pLsiLogic Pointer to the dveice instance to check.
4787 * @param cMillis Timeout in milliseconds to wait.
4788 */
4789static bool lsilogicWaitForAsyncIOFinished(PLSILOGICSCSI pLsiLogic, unsigned cMillies)
4790{
4791 uint64_t u64Start;
4792 bool fIdle;
4793
4794 /*
4795 * Wait for any pending async operation to finish
4796 */
4797 u64Start = RTTimeMilliTS();
4798 do
4799 {
4800 fIdle = true;
4801
4802 /* Check every port. */
4803 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aDeviceStates); i++)
4804 {
4805 PLSILOGICDEVICE pLsiLogicDevice = &pLsiLogic->aDeviceStates[i];
4806 if (ASMAtomicReadU32(&pLsiLogicDevice->cOutstandingRequests))
4807 {
4808 fIdle = false;
4809 break;
4810 }
4811 }
4812 if (RTTimeMilliTS() - u64Start >= cMillies)
4813 break;
4814
4815 /* Sleep for a bit. */
4816 RTThreadSleep(100);
4817 } while (!fIdle);
4818
4819 return fIdle;
4820}
4821
4822static DECLCALLBACK(int) lsilogicSaveLoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4823{
4824 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4825
4826 /* Wait that no task is pending on any device. */
4827 if (!lsilogicWaitForAsyncIOFinished(pLsiLogic, 20000))
4828 {
4829 AssertLogRelMsgFailed(("LsiLogic: There are still tasks outstanding\n"));
4830 return VERR_TIMEOUT;
4831 }
4832
4833 return VINF_SUCCESS;
4834}
4835
4836static DECLCALLBACK(int) lsilogicSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4837{
4838 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4839
4840 /* Every device first. */
4841 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aDeviceStates); i++)
4842 {
4843 PLSILOGICDEVICE pDevice = &pLsiLogic->aDeviceStates[i];
4844
4845 AssertMsg(!pDevice->cOutstandingRequests,
4846 ("There are still outstanding requests on this device\n"));
4847 SSMR3PutU32(pSSM, pDevice->cOutstandingRequests);
4848 }
4849 /* Now the main device state. */
4850 SSMR3PutU32 (pSSM, pLsiLogic->enmState);
4851 SSMR3PutU32 (pSSM, pLsiLogic->enmWhoInit);
4852 SSMR3PutBool (pSSM, pLsiLogic->fDoorbellInProgress);
4853 SSMR3PutBool (pSSM, pLsiLogic->fDiagnosticEnabled);
4854 SSMR3PutBool (pSSM, pLsiLogic->fNotificationSend);
4855 SSMR3PutBool (pSSM, pLsiLogic->fEventNotificationEnabled);
4856 SSMR3PutU32 (pSSM, pLsiLogic->uInterruptMask);
4857 SSMR3PutU32 (pSSM, pLsiLogic->uInterruptStatus);
4858 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aMessage); i++)
4859 SSMR3PutU32 (pSSM, pLsiLogic->aMessage[i]);
4860 SSMR3PutU32 (pSSM, pLsiLogic->iMessage);
4861 SSMR3PutU32 (pSSM, pLsiLogic->cMessage);
4862 SSMR3PutMem (pSSM, &pLsiLogic->ReplyBuffer, sizeof(pLsiLogic->ReplyBuffer));
4863 SSMR3PutU32 (pSSM, pLsiLogic->uNextReplyEntryRead);
4864 SSMR3PutU32 (pSSM, pLsiLogic->cReplySize);
4865 SSMR3PutU16 (pSSM, pLsiLogic->u16IOCFaultCode);
4866 SSMR3PutU32 (pSSM, pLsiLogic->u32HostMFAHighAddr);
4867 SSMR3PutU32 (pSSM, pLsiLogic->u32SenseBufferHighAddr);
4868 SSMR3PutU8 (pSSM, pLsiLogic->cMaxDevices);
4869 SSMR3PutU8 (pSSM, pLsiLogic->cMaxBuses);
4870 SSMR3PutU16 (pSSM, pLsiLogic->cbReplyFrame);
4871 SSMR3PutU32 (pSSM, pLsiLogic->iDiagnosticAccess);
4872 SSMR3PutU32 (pSSM, pLsiLogic->cReplyQueueEntries);
4873 SSMR3PutU32 (pSSM, pLsiLogic->cRequestQueueEntries);
4874 SSMR3PutU32 (pSSM, pLsiLogic->uReplyFreeQueueNextEntryFreeWrite);
4875 SSMR3PutU32 (pSSM, pLsiLogic->uReplyFreeQueueNextAddressRead);
4876 SSMR3PutU32 (pSSM, pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
4877 SSMR3PutU32 (pSSM, pLsiLogic->uReplyPostQueueNextAddressRead);
4878 SSMR3PutU32 (pSSM, pLsiLogic->uRequestQueueNextEntryFreeWrite);
4879 SSMR3PutU32 (pSSM, pLsiLogic->uRequestQueueNextAddressRead);
4880 SSMR3PutMem (pSSM, &pLsiLogic->ConfigurationPages, sizeof(pLsiLogic->ConfigurationPages));
4881 /* Now the data for the BIOS interface. */
4882 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.regIdentify);
4883 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.uTargetDevice);
4884 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.uTxDir);
4885 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.cbCDB);
4886 SSMR3PutMem (pSSM, pLsiLogic->VBoxSCSI.aCDB, sizeof(pLsiLogic->VBoxSCSI.aCDB));
4887 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.iCDB);
4888 SSMR3PutU32 (pSSM, pLsiLogic->VBoxSCSI.cbBuf);
4889 SSMR3PutU32 (pSSM, pLsiLogic->VBoxSCSI.iBuf);
4890 SSMR3PutBool (pSSM, pLsiLogic->VBoxSCSI.fBusy);
4891 SSMR3PutU8 (pSSM, pLsiLogic->VBoxSCSI.enmState);
4892 if (pLsiLogic->VBoxSCSI.cbCDB)
4893 SSMR3PutMem(pSSM, pLsiLogic->VBoxSCSI.pBuf, pLsiLogic->VBoxSCSI.cbBuf);
4894
4895 return SSMR3PutU32(pSSM, ~0);
4896}
4897
4898static DECLCALLBACK(int) lsilogicLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t u32Version)
4899{
4900 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
4901 int rc = VINF_SUCCESS;
4902
4903 /* We support saved states only from this and older versions. */
4904 if (u32Version > LSILOGIC_SAVED_STATE_MINOR_VERSION)
4905 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
4906
4907 /* Every device first. */
4908 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aDeviceStates); i++)
4909 {
4910 PLSILOGICDEVICE pDevice = &pLsiLogic->aDeviceStates[i];
4911
4912 AssertMsg(!pDevice->cOutstandingRequests,
4913 ("There are still outstanding requests on this device\n"));
4914 SSMR3GetU32(pSSM, (uint32_t *)&pDevice->cOutstandingRequests);
4915 }
4916 /* Now the main device state. */
4917 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->enmState);
4918 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->enmWhoInit);
4919 SSMR3GetBool (pSSM, &pLsiLogic->fDoorbellInProgress);
4920 SSMR3GetBool (pSSM, &pLsiLogic->fDiagnosticEnabled);
4921 SSMR3GetBool (pSSM, &pLsiLogic->fNotificationSend);
4922 SSMR3GetBool (pSSM, &pLsiLogic->fEventNotificationEnabled);
4923 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uInterruptMask);
4924 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uInterruptStatus);
4925 for (unsigned i = 0; i < RT_ELEMENTS(pLsiLogic->aMessage); i++)
4926 SSMR3GetU32 (pSSM, &pLsiLogic->aMessage[i]);
4927 SSMR3GetU32 (pSSM, &pLsiLogic->iMessage);
4928 SSMR3GetU32 (pSSM, &pLsiLogic->cMessage);
4929 SSMR3GetMem (pSSM, &pLsiLogic->ReplyBuffer, sizeof(pLsiLogic->ReplyBuffer));
4930 SSMR3GetU32 (pSSM, &pLsiLogic->uNextReplyEntryRead);
4931 SSMR3GetU32 (pSSM, &pLsiLogic->cReplySize);
4932 SSMR3GetU16 (pSSM, &pLsiLogic->u16IOCFaultCode);
4933 SSMR3GetU32 (pSSM, &pLsiLogic->u32HostMFAHighAddr);
4934 SSMR3GetU32 (pSSM, &pLsiLogic->u32SenseBufferHighAddr);
4935 SSMR3GetU8 (pSSM, &pLsiLogic->cMaxDevices);
4936 SSMR3GetU8 (pSSM, &pLsiLogic->cMaxBuses);
4937 SSMR3GetU16 (pSSM, &pLsiLogic->cbReplyFrame);
4938 SSMR3GetU32 (pSSM, &pLsiLogic->iDiagnosticAccess);
4939 SSMR3GetU32 (pSSM, &pLsiLogic->cReplyQueueEntries);
4940 SSMR3GetU32 (pSSM, &pLsiLogic->cRequestQueueEntries);
4941 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyFreeQueueNextEntryFreeWrite);
4942 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyFreeQueueNextAddressRead);
4943 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyPostQueueNextEntryFreeWrite);
4944 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uReplyPostQueueNextAddressRead);
4945 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uRequestQueueNextEntryFreeWrite);
4946 SSMR3GetU32 (pSSM, (uint32_t *)&pLsiLogic->uRequestQueueNextAddressRead);
4947 SSMR3GetMem (pSSM, &pLsiLogic->ConfigurationPages, sizeof(pLsiLogic->ConfigurationPages));
4948 /* Now the data for the BIOS interface. */
4949 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.regIdentify);
4950 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.uTargetDevice);
4951 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.uTxDir);
4952 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.cbCDB);
4953 SSMR3GetMem (pSSM, pLsiLogic->VBoxSCSI.aCDB, sizeof(pLsiLogic->VBoxSCSI.aCDB));
4954 SSMR3GetU8 (pSSM, &pLsiLogic->VBoxSCSI.iCDB);
4955 SSMR3GetU32 (pSSM, &pLsiLogic->VBoxSCSI.cbBuf);
4956 SSMR3GetU32 (pSSM, &pLsiLogic->VBoxSCSI.iBuf);
4957 SSMR3GetBool(pSSM, (bool *)&pLsiLogic->VBoxSCSI.fBusy);
4958 SSMR3GetU8 (pSSM, (uint8_t *)&pLsiLogic->VBoxSCSI.enmState);
4959 if (pLsiLogic->VBoxSCSI.cbCDB)
4960 {
4961 pLsiLogic->VBoxSCSI.pBuf = (uint8_t *)RTMemAllocZ(pLsiLogic->VBoxSCSI.cbCDB);
4962 if (!pLsiLogic->VBoxSCSI.pBuf)
4963 {
4964 LogRel(("LsiLogic: Out of memory during restore.\n"));
4965 return PDMDEV_SET_ERROR(pDevIns, VERR_NO_MEMORY,
4966 N_("LsiLogic: Out of memory during restore\n"));
4967 }
4968 SSMR3GetMem(pSSM, pLsiLogic->VBoxSCSI.pBuf, pLsiLogic->VBoxSCSI.cbBuf);
4969 }
4970
4971 uint32_t u32;
4972 SSMR3GetU32(pSSM, &u32);
4973 if (RT_FAILURE(rc))
4974 return rc;
4975 if (u32 != ~0U)
4976 {
4977 AssertMsgFailed(("u32=%#x expected ~0\n", u32));
4978 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
4979 }
4980
4981 return VINF_SUCCESS;
4982}
4983
4984/**
4985 * Gets the pointer to the status LED of a device - called from the SCSi driver.
4986 *
4987 * @returns VBox status code.
4988 * @param pInterface Pointer to the interface structure containing the called function pointer.
4989 * @param iLUN The unit which status LED we desire. Always 0 here as the driver
4990 * doesn't know about other LUN's.
4991 * @param ppLed Where to store the LED pointer.
4992 */
4993static DECLCALLBACK(int) lsilogicDeviceQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
4994{
4995 PLSILOGICDEVICE pDevice = PDMILEDPORTS_2_PLSILOGICDEVICE(pInterface);
4996 if (iLUN == 0)
4997 {
4998 *ppLed = &pDevice->Led;
4999 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
5000 return VINF_SUCCESS;
5001 }
5002 return VERR_PDM_LUN_NOT_FOUND;
5003}
5004
5005/**
5006 * Queries an interface to the driver.
5007 *
5008 * @returns Pointer to interface.
5009 * @returns NULL if the interface was not supported by the device.
5010 * @param pInterface Pointer to LSILOGICDEVICE::IBase.
5011 * @param enmInterface The requested interface identification.
5012 */
5013static DECLCALLBACK(void *) lsilogicDeviceQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
5014{
5015 PLSILOGICDEVICE pDevice = PDMIBASE_2_PLSILOGICDEVICE(pInterface);
5016
5017 switch (enmInterface)
5018 {
5019 case PDMINTERFACE_SCSI_PORT:
5020 return &pDevice->ISCSIPort;
5021 case PDMINTERFACE_LED_PORTS:
5022 return &pDevice->ILed;
5023 default:
5024 return NULL;
5025 }
5026}
5027
5028/**
5029 * Gets the pointer to the status LED of a unit.
5030 *
5031 * @returns VBox status code.
5032 * @param pInterface Pointer to the interface structure containing the called function pointer.
5033 * @param iLUN The unit which status LED we desire.
5034 * @param ppLed Where to store the LED pointer.
5035 */
5036static DECLCALLBACK(int) lsilogicStatusQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
5037{
5038 PLSILOGICSCSI pLsiLogic = PDMILEDPORTS_2_PLSILOGICSCSI(pInterface);
5039 if (iLUN < LSILOGIC_DEVICES_MAX)
5040 {
5041 *ppLed = &pLsiLogic->aDeviceStates[iLUN].Led;
5042 Assert((*ppLed)->u32Magic == PDMLED_MAGIC);
5043 return VINF_SUCCESS;
5044 }
5045 return VERR_PDM_LUN_NOT_FOUND;
5046}
5047
5048/**
5049 * Queries an interface to the driver.
5050 *
5051 * @returns Pointer to interface.
5052 * @returns NULL if the interface was not supported by the device.
5053 * @param pInterface Pointer to ATADevState::IBase.
5054 * @param enmInterface The requested interface identification.
5055 */
5056static DECLCALLBACK(void *) lsilogicStatusQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
5057{
5058 PLSILOGICSCSI pLsiLogic = PDMIBASE_2_PLSILOGICSCSI(pInterface);
5059 switch (enmInterface)
5060 {
5061 case PDMINTERFACE_BASE:
5062 return &pLsiLogic->IBase;
5063 case PDMINTERFACE_LED_PORTS:
5064 return &pLsiLogic->ILeds;
5065 default:
5066 return NULL;
5067 }
5068}
5069
5070/**
5071 * Detach notification.
5072 *
5073 * One harddisk at one port has been unplugged.
5074 * The VM is suspended at this point.
5075 *
5076 * @param pDevIns The device instance.
5077 * @param iLUN The logical unit which is being detached.
5078 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
5079 */
5080static DECLCALLBACK(void) lsilogicDetach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
5081{
5082 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5083 PLSILOGICDEVICE pDevice = &pThis->aDeviceStates[iLUN];
5084
5085 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
5086 ("LsiLogic: Device does not support hotplugging\n"));
5087
5088 Log(("%s:\n", __FUNCTION__));
5089
5090 /*
5091 * Zero some important members.
5092 */
5093 pDevice->pDrvBase = NULL;
5094 pDevice->pDrvSCSIConnector = NULL;
5095}
5096
5097/**
5098 * Attach command.
5099 *
5100 * This is called when we change block driver.
5101 *
5102 * @returns VBox status code.
5103 * @param pDevIns The device instance.
5104 * @param iLUN The logical unit which is being detached.
5105 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
5106 */
5107static DECLCALLBACK(int) lsilogicAttach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
5108{
5109 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5110 PLSILOGICDEVICE pDevice = &pThis->aDeviceStates[iLUN];
5111 int rc;
5112
5113 AssertMsgReturn(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
5114 ("LsiLogic: Device does not support hotplugging\n"),
5115 VERR_INVALID_PARAMETER);
5116
5117 /* the usual paranoia */
5118 AssertRelease(!pDevice->pDrvBase);
5119 AssertRelease(!pDevice->pDrvSCSIConnector);
5120 Assert(pDevice->iLUN == iLUN);
5121
5122 /*
5123 * Try attach the block device and get the interfaces,
5124 * required as well as optional.
5125 */
5126 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, NULL);
5127 if (RT_SUCCESS(rc))
5128 {
5129 /* Get SCSI connector interface. */
5130 pDevice->pDrvSCSIConnector = (PPDMISCSICONNECTOR)pDevice->pDrvBase->pfnQueryInterface(pDevice->pDrvBase, PDMINTERFACE_SCSI_CONNECTOR);
5131 AssertMsgReturn(pDevice->pDrvSCSIConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
5132 }
5133 else
5134 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", pDevice->iLUN, rc));
5135
5136 if (RT_FAILURE(rc))
5137 {
5138 pDevice->pDrvBase = NULL;
5139 pDevice->pDrvSCSIConnector = NULL;
5140 }
5141 return rc;
5142}
5143
5144/**
5145 * @copydoc FNPDMDEVPOWEROFF
5146 */
5147static DECLCALLBACK(void) lsilogicPowerOff(PPDMDEVINS pDevIns)
5148{
5149 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5150
5151 bool fIdle = lsilogicWaitForAsyncIOFinished(pLsiLogic, 20000);
5152 Assert(fIdle);
5153}
5154
5155/**
5156 * @copydoc FNPDMDEVSUSPEND
5157 */
5158static DECLCALLBACK(void) lsilogicSuspend(PPDMDEVINS pDevIns)
5159{
5160 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5161
5162 bool fIdle = lsilogicWaitForAsyncIOFinished(pLsiLogic, 20000);
5163 Assert(fIdle);
5164}
5165
5166/**
5167 * @copydoc FNPDMDEVRESET
5168 */
5169static DECLCALLBACK(void) lsilogicReset(PPDMDEVINS pDevIns)
5170{
5171 PLSILOGICSCSI pLsiLogic = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5172 int rc;
5173
5174 bool fIdle = lsilogicWaitForAsyncIOFinished(pLsiLogic, 20000);
5175 Assert(fIdle);
5176
5177 rc = lsilogicHardReset(pLsiLogic);
5178 AssertRC(rc);
5179
5180 vboxscsiInitialize(&pLsiLogic->VBoxSCSI);
5181}
5182
5183/**
5184 * @copydoc FNPDMDEVRELOCATE
5185 */
5186static DECLCALLBACK(void) lsilogicRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
5187{
5188 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5189
5190 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5191 pThis->pNotificationQueueRC = PDMQueueRCPtr(pThis->pNotificationQueueR3);
5192
5193 /* Relocate queues. */
5194 pThis->pReplyFreeQueueBaseRC += offDelta;
5195 pThis->pReplyPostQueueBaseRC += offDelta;
5196 pThis->pRequestQueueBaseRC += offDelta;
5197}
5198
5199/**
5200 * @copydoc FNPDMDEVDESTRUCT
5201 */
5202static DECLCALLBACK(int) lsilogicDestruct(PPDMDEVINS pDevIns)
5203{
5204 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5205 int rc = VINF_SUCCESS;
5206
5207 PDMR3CritSectDelete(&pThis->ReplyFreeQueueCritSect);
5208 PDMR3CritSectDelete(&pThis->ReplyPostQueueCritSect);
5209
5210 /* Destroy task cache. */
5211 if (pThis->pTaskCache)
5212 rc = RTCacheDestroy(pThis->pTaskCache);
5213
5214 return rc;
5215}
5216
5217/**
5218 * @copydoc FNPDMDEVCONSTRUCT
5219 */
5220static DECLCALLBACK(int) lsilogicConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfgHandle)
5221{
5222 PLSILOGICSCSI pThis = PDMINS_2_DATA(pDevIns, PLSILOGICSCSI);
5223 int rc = VINF_SUCCESS;
5224 PVM pVM = PDMDevHlpGetVM(pDevIns);
5225
5226 /*
5227 * Validate and read configuration.
5228 */
5229 rc = CFGMR3AreValuesValid(pCfgHandle, "GCEnabled\0"
5230 "R0Enabled\0"
5231 "ReplyQueueDepth\0"
5232 "RequestQueueDepth\0");
5233 if (RT_FAILURE(rc))
5234 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
5235 N_("LsiLogic configuration error: unknown option specified"));
5236 rc = CFGMR3QueryBoolDef(pCfgHandle, "GCEnabled", &pThis->fGCEnabled, true);
5237 if (RT_FAILURE(rc))
5238 return PDMDEV_SET_ERROR(pDevIns, rc,
5239 N_("LsiLogic configuration error: failed to read GCEnabled as boolean"));
5240 Log(("%s: fGCEnabled=%d\n", __FUNCTION__, pThis->fGCEnabled));
5241
5242 rc = CFGMR3QueryBoolDef(pCfgHandle, "R0Enabled", &pThis->fR0Enabled, true);
5243 if (RT_FAILURE(rc))
5244 return PDMDEV_SET_ERROR(pDevIns, rc,
5245 N_("LsiLogic configuration error: failed to read R0Enabled as boolean"));
5246 Log(("%s: fR0Enabled=%d\n", __FUNCTION__, pThis->fR0Enabled));
5247
5248 rc = CFGMR3QueryU32Def(pCfgHandle, "ReplyQueueDepth",
5249 &pThis->cReplyQueueEntries,
5250 LSILOGICSCSI_REPLY_QUEUE_DEPTH_DEFAULT);
5251 if (RT_FAILURE(rc))
5252 return PDMDEV_SET_ERROR(pDevIns, rc,
5253 N_("LsiLogic configuration error: failed to read ReplyQueue as integer"));
5254 Log(("%s: ReplyQueueDepth=%u\n", __FUNCTION__, pThis->cReplyQueueEntries));
5255
5256 rc = CFGMR3QueryU32Def(pCfgHandle, "RequestQueueDepth",
5257 &pThis->cRequestQueueEntries,
5258 LSILOGICSCSI_REQUEST_QUEUE_DEPTH_DEFAULT);
5259 if (RT_FAILURE(rc))
5260 return PDMDEV_SET_ERROR(pDevIns, rc,
5261 N_("LsiLogic configuration error: failed to read RequestQueue as integer"));
5262 Log(("%s: RequestQueueDepth=%u\n", __FUNCTION__, pThis->cRequestQueueEntries));
5263
5264
5265 /* Init static parts. */
5266 PCIDevSetVendorId (&pThis->PciDev, LSILOGICSCSI_PCI_VENDOR_ID); /* LsiLogic */
5267 PCIDevSetDeviceId (&pThis->PciDev, LSILOGICSCSI_PCI_DEVICE_ID); /* LSI53C1030 */
5268 PCIDevSetClassProg (&pThis->PciDev, 0x00); /* SCSI */
5269 PCIDevSetClassSub (&pThis->PciDev, 0x00); /* SCSI */
5270 PCIDevSetClassBase (&pThis->PciDev, 0x01); /* Mass storage */
5271 PCIDevSetSubSystemVendorId(&pThis->PciDev, LSILOGICSCSI_PCI_SUBSYSTEM_VENDOR_ID);
5272 PCIDevSetSubSystemId (&pThis->PciDev, LSILOGICSCSI_PCI_SUBSYSTEM_ID);
5273 PCIDevSetInterruptPin (&pThis->PciDev, 0x01); /* Interrupt pin A */
5274
5275 pThis->pDevInsR3 = pDevIns;
5276 pThis->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
5277 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5278 pThis->IBase.pfnQueryInterface = lsilogicStatusQueryInterface;
5279 pThis->ILeds.pfnQueryStatusLed = lsilogicStatusQueryStatusLed;
5280
5281 /*
5282 * Register the PCI device, it's I/O regions.
5283 */
5284 rc = PDMDevHlpPCIRegister (pDevIns, &pThis->PciDev);
5285 if (RT_FAILURE(rc))
5286 return rc;
5287
5288 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, LSILOGIC_PCI_SPACE_IO_SIZE, PCI_ADDRESS_SPACE_IO, lsilogicMap);
5289 if (RT_FAILURE(rc))
5290 return rc;
5291
5292 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, LSILOGIC_PCI_SPACE_MEM_SIZE, PCI_ADDRESS_SPACE_MEM, lsilogicMap);
5293 if (RT_FAILURE(rc))
5294 return rc;
5295
5296 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 2, LSILOGIC_PCI_SPACE_MEM_SIZE, PCI_ADDRESS_SPACE_MEM, lsilogicMap);
5297 if (RT_FAILURE(rc))
5298 return rc;
5299
5300 /* Intialize task queue. */
5301 rc = PDMDevHlpPDMQueueCreate(pDevIns, sizeof(PDMQUEUEITEMCORE), 2, 0,
5302 lsilogicNotifyQueueConsumer, true, "LsiLogic-Task", &pThis->pNotificationQueueR3);
5303 if (RT_FAILURE(rc))
5304 return rc;
5305 pThis->pNotificationQueueR0 = PDMQueueR0Ptr(pThis->pNotificationQueueR3);
5306 pThis->pNotificationQueueRC = PDMQueueRCPtr(pThis->pNotificationQueueR3);
5307
5308 /*
5309 * We need one entry free in the queue.
5310 */
5311 pThis->cReplyQueueEntries++;
5312 pThis->cRequestQueueEntries++;
5313
5314 /*
5315 * Allocate memory for the queues.
5316 */
5317 uint32_t cbQueues;
5318
5319 cbQueues = 2*pThis->cReplyQueueEntries * sizeof(uint32_t);
5320 cbQueues += pThis->cRequestQueueEntries * sizeof(uint32_t);
5321 rc = MMHyperAlloc(pVM, cbQueues, 1, MM_TAG_PDM_DEVICE_USER,
5322 (void **)&pThis->pReplyFreeQueueBaseR3);
5323 if (RT_FAILURE(rc))
5324 return VERR_NO_MEMORY;
5325 pThis->pReplyFreeQueueBaseR0 = MMHyperR3ToR0(pVM, (void *)pThis->pReplyFreeQueueBaseR3);
5326 pThis->pReplyFreeQueueBaseRC = MMHyperR3ToRC(pVM, (void *)pThis->pReplyFreeQueueBaseR3);
5327
5328 pThis->pReplyPostQueueBaseR3 = pThis->pReplyFreeQueueBaseR3 + pThis->cReplyQueueEntries;
5329 pThis->pReplyPostQueueBaseR0 = MMHyperR3ToR0(pVM, (void *)pThis->pReplyPostQueueBaseR3);
5330 pThis->pReplyPostQueueBaseRC = MMHyperR3ToRC(pVM, (void *)pThis->pReplyPostQueueBaseR3);
5331
5332 pThis->pRequestQueueBaseR3 = pThis->pReplyPostQueueBaseR3 + pThis->cReplyQueueEntries;
5333 pThis->pRequestQueueBaseR0 = MMHyperR3ToR0(pVM, (void *)pThis->pRequestQueueBaseR3);
5334 pThis->pRequestQueueBaseRC = MMHyperR3ToRC(pVM, (void *)pThis->pRequestQueueBaseR3);
5335
5336 /*
5337 * Create critical sections protecting the reply post and free queues.
5338 */
5339 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyFreeQueueCritSect, "LsiLogicRFQ");
5340 if (RT_FAILURE(rc))
5341 return PDMDEV_SET_ERROR(pDevIns, rc,
5342 N_("LsiLogic: cannot create critical section for reply free queue"));
5343
5344 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->ReplyPostQueueCritSect, "LsiLogicRPQ");
5345 if (RT_FAILURE(rc))
5346 return PDMDEV_SET_ERROR(pDevIns, rc,
5347 N_("LsiLogic: cannot create critical section for reply post queue"));
5348
5349 /*
5350 * Allocate task cache.
5351 */
5352 rc = RTCacheCreate(&pThis->pTaskCache, 0, sizeof(LSILOGICTASKSTATE), RTOBJCACHE_PROTECT_INSERT);
5353 if (RT_FAILURE(rc))
5354 return PDMDEV_SET_ERROR(pDevIns, rc,
5355 N_("Cannot create task cache"));
5356
5357 /* Initialize per device state. */
5358 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aDeviceStates); i++)
5359 {
5360 char szName[24];
5361 PLSILOGICDEVICE pDevice = &pThis->aDeviceStates[i];
5362
5363 RTStrPrintf(szName, sizeof(szName), "Device%d", i);
5364
5365 /* Initialize static parts of the device. */
5366 pDevice->iLUN = i;
5367 pDevice->pLsiLogicR3 = pThis;
5368 pDevice->pLsiLogicR0 = PDMINS_2_DATA_R0PTR(pDevIns);
5369 pDevice->pLsiLogicRC = PDMINS_2_DATA_RCPTR(pDevIns);
5370 pDevice->Led.u32Magic = PDMLED_MAGIC;
5371 pDevice->IBase.pfnQueryInterface = lsilogicDeviceQueryInterface;
5372 pDevice->ISCSIPort.pfnSCSIRequestCompleted = lsilogicDeviceSCSIRequestCompleted;
5373 pDevice->ILed.pfnQueryStatusLed = lsilogicDeviceQueryStatusLed;
5374
5375 /* Attach SCSI driver. */
5376 rc = PDMDevHlpDriverAttach(pDevIns, pDevice->iLUN, &pDevice->IBase, &pDevice->pDrvBase, szName);
5377 if (RT_SUCCESS(rc))
5378 {
5379 /* Get SCSI connector interface. */
5380 pDevice->pDrvSCSIConnector = (PPDMISCSICONNECTOR)pDevice->pDrvBase->pfnQueryInterface(pDevice->pDrvBase, PDMINTERFACE_SCSI_CONNECTOR);
5381 AssertMsgReturn(pDevice->pDrvSCSIConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
5382 }
5383 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
5384 {
5385 pDevice->pDrvBase = NULL;
5386 rc = VINF_SUCCESS;
5387 Log(("LsiLogic: no driver attached to device %s\n", szName));
5388 }
5389 else
5390 {
5391 AssertLogRelMsgFailed(("LsiLogic: Failed to attach %s\n", szName));
5392 return rc;
5393 }
5394 }
5395
5396 /*
5397 * Attach status driver (optional).
5398 */
5399 PPDMIBASE pBase;
5400 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
5401 if (RT_SUCCESS(rc))
5402 pThis->pLedsConnector = (PDMILEDCONNECTORS *)pBase->pfnQueryInterface(pBase, PDMINTERFACE_LED_CONNECTORS);
5403 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
5404 {
5405 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
5406 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot attach to status driver"));
5407 }
5408
5409 /* Initialize the SCSI emulation for the BIOS. */
5410 rc = vboxscsiInitialize(&pThis->VBoxSCSI);
5411 AssertRC(rc);
5412
5413 /* Register I/O port space in ISA region for BIOS access. */
5414 rc = PDMDevHlpIOPortRegister(pDevIns, LSILOGIC_ISA_IO_PORT, 3, NULL,
5415 lsilogicIsaIOPortWrite, lsilogicIsaIOPortRead,
5416 lsilogicIsaIOPortWriteStr, lsilogicIsaIOPortReadStr,
5417 "LsiLogic BIOS");
5418 if (RT_FAILURE(rc))
5419 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot register legacy I/O handlers"));
5420
5421 /* Register save state handlers. */
5422 rc = PDMDevHlpSSMRegister(pDevIns, pDevIns->pDevReg->szDeviceName, iInstance,
5423 LSILOGIC_SAVED_STATE_MINOR_VERSION, sizeof(*pThis),
5424 lsilogicSaveLoadPrep, lsilogicSaveExec, NULL,
5425 lsilogicSaveLoadPrep, lsilogicLoadExec, NULL);
5426 if (RT_FAILURE(rc))
5427 return PDMDEV_SET_ERROR(pDevIns, rc, N_("LsiLogic cannot register save state handlers"));
5428
5429 pThis->enmWhoInit = LSILOGICWHOINIT_SYSTEM_BIOS;
5430
5431 /* Perform hard reset. */
5432 rc = lsilogicHardReset(pThis);
5433 AssertRC(rc);
5434
5435 return rc;
5436}
5437
5438/**
5439 * The device registration structure.
5440 */
5441const PDMDEVREG g_DeviceLsiLogicSCSI =
5442{
5443 /* u32Version */
5444 PDM_DEVREG_VERSION,
5445 /* szDeviceName */
5446 "lsilogicscsi",
5447 /* szRCMod */
5448 "VBoxDDGC.gc",
5449 /* szR0Mod */
5450 "VBoxDDR0.r0",
5451 /* pszDescription */
5452 "LSI Logic 53c1030 SCSI controller.\n",
5453 /* fFlags */
5454 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0
5455 | PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION
5456 | PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION,
5457 /* fClass */
5458 PDM_DEVREG_CLASS_STORAGE,
5459 /* cMaxInstances */
5460 ~0,
5461 /* cbInstance */
5462 sizeof(LSILOGICSCSI),
5463 /* pfnConstruct */
5464 lsilogicConstruct,
5465 /* pfnDestruct */
5466 lsilogicDestruct,
5467 /* pfnRelocate */
5468 lsilogicRelocate,
5469 /* pfnIOCtl */
5470 NULL,
5471 /* pfnPowerOn */
5472 NULL,
5473 /* pfnReset */
5474 lsilogicReset,
5475 /* pfnSuspend */
5476 lsilogicSuspend,
5477 /* pfnResume */
5478 NULL,
5479 /* pfnAttach */
5480 lsilogicAttach,
5481 /* pfnDetach */
5482 lsilogicDetach,
5483 /* pfnQueryInterface. */
5484 NULL,
5485 /* pfnInitComplete */
5486 NULL,
5487 /* pfnPowerOff */
5488 lsilogicPowerOff,
5489 /* pfnSoftReset */
5490 NULL,
5491 /* u32VersionEnd */
5492 PDM_DEVREG_VERSION
5493};
5494
5495#endif /* IN_RING3 */
5496#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette