VirtualBox

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

Last change on this file since 24340 was 24265, checked in by vboxsync, 15 years ago

Devices,VMM: Replaced all VERR_SSM_LOAD_CONFIG_MISMATCH returns with SSMR3SetCfgError calls.

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