/** @file

  VMware PVSCSI Device specific type and macro definitions.

  Copyright (C) 2020, Oracle and/or its affiliates.

  SPDX-License-Identifier: BSD-2-Clause-Patent

**/

#ifndef __PVSCSI_H_
#define __PVSCSI_H_

//
// Device offsets and constants
//

#define PCI_VENDOR_ID_VMWARE         (0x15ad)
#define PCI_DEVICE_ID_VMWARE_PVSCSI  (0x07c0)

//
// CDB (Command Descriptor Block) with size above this constant
// should be considered out-of-band
//
#define PVSCSI_CDB_MAX_SIZE  (16)

typedef enum {
  PvScsiRegOffsetCommand       =    0x0,
  PvScsiRegOffsetCommandData   =    0x4,
  PvScsiRegOffsetCommandStatus =    0x8,
  PvScsiRegOffsetLastSts0      =  0x100,
  PvScsiRegOffsetLastSts1      =  0x104,
  PvScsiRegOffsetLastSts2      =  0x108,
  PvScsiRegOffsetLastSts3      =  0x10c,
  PvScsiRegOffsetIntrStatus    = 0x100c,
  PvScsiRegOffsetIntrMask      = 0x2010,
  PvScsiRegOffsetKickNonRwIo   = 0x3014,
  PvScsiRegOffsetDebug         = 0x3018,
  PvScsiRegOffsetKickRwIo      = 0x4018,
} PVSCSI_BAR0_OFFSETS;

//
// Define Interrupt-Status register flags
//
#define PVSCSI_INTR_CMPL_0     BIT0
#define PVSCSI_INTR_CMPL_1     BIT1
#define PVSCSI_INTR_CMPL_MASK  (PVSCSI_INTR_CMPL_0 | PVSCSI_INTR_CMPL_1)

typedef enum {
  PvScsiCmdFirst        = 0,
  PvScsiCmdAdapterReset = 1,
  PvScsiCmdIssueScsi    = 2,
  PvScsiCmdSetupRings   = 3,
  PvScsiCmdResetBus     = 4,
  PvScsiCmdResetDevice  = 5,
  PvScsiCmdAbortCmd     = 6,
  PvScsiCmdConfig       = 7,
  PvScsiCmdSetupMsgRing = 8,
  PvScsiCmdDeviceUnplug = 9,
  PvScsiCmdLast         = 10
} PVSCSI_COMMANDS;

#define PVSCSI_SETUP_RINGS_MAX_NUM_PAGES  (32)

#pragma pack (1)
typedef struct {
  UINT32    ReqRingNumPages;
  UINT32    CmpRingNumPages;
  UINT64    RingsStatePPN;
  UINT64    ReqRingPPNs[PVSCSI_SETUP_RINGS_MAX_NUM_PAGES];
  UINT64    CmpRingPPNs[PVSCSI_SETUP_RINGS_MAX_NUM_PAGES];
} PVSCSI_CMD_DESC_SETUP_RINGS;
#pragma pack ()

#define PVSCSI_MAX_CMD_DATA_WORDS   \
  (sizeof (PVSCSI_CMD_DESC_SETUP_RINGS) / sizeof (UINT32))

#pragma pack (1)
typedef struct {
  UINT32    ReqProdIdx;
  UINT32    ReqConsIdx;
  UINT32    ReqNumEntriesLog2;

  UINT32    CmpProdIdx;
  UINT32    CmpConsIdx;
  UINT32    CmpNumEntriesLog2;

  UINT8     Pad[104];

  UINT32    MsgProdIdx;
  UINT32    MsgConsIdx;
  UINT32    MsgNumEntriesLog2;
} PVSCSI_RINGS_STATE;
#pragma pack ()

//
// Define PVSCSI request descriptor tags
//
#define PVSCSI_SIMPLE_QUEUE_TAG  (0x20)

//
// Define PVSCSI request descriptor flags
//
#define PVSCSI_FLAG_CMD_WITH_SG_LIST     BIT0
#define PVSCSI_FLAG_CMD_OUT_OF_BAND_CDB  BIT1
#define PVSCSI_FLAG_CMD_DIR_NONE         BIT2
#define PVSCSI_FLAG_CMD_DIR_TOHOST       BIT3
#define PVSCSI_FLAG_CMD_DIR_TODEVICE     BIT4

#pragma pack (1)
typedef struct {
  UINT64    Context;
  UINT64    DataAddr;
  UINT64    DataLen;
  UINT64    SenseAddr;
  UINT32    SenseLen;
  UINT32    Flags;
  UINT8     Cdb[16];
  UINT8     CdbLen;
  UINT8     Lun[8];
  UINT8     Tag;
  UINT8     Bus;
  UINT8     Target;
  UINT8     VcpuHint;
  UINT8     Unused[59];
} PVSCSI_RING_REQ_DESC;
#pragma pack ()

//
// Host adapter status/error codes
//
typedef enum {
  PvScsiBtStatSuccess                        = 0x00, // CCB complete normally with no errors
  PvScsiBtStatLinkedCommandCompleted         = 0x0a,
  PvScsiBtStatLinkedCommandCompletedWithFlag = 0x0b,
  PvScsiBtStatDataUnderrun                   = 0x0c,
  PvScsiBtStatSelTimeout                     = 0x11, // SCSI selection timeout
  PvScsiBtStatDatarun                        = 0x12, // Data overrun/underrun
  PvScsiBtStatBusFree                        = 0x13, // Unexpected bus free
  PvScsiBtStatInvPhase                       = 0x14, //
                                                     // Invalid bus phase or sequence requested
                                                     // by target
                                                     //
  PvScsiBtStatLunMismatch = 0x17,                    //
                                                     // Linked CCB has different LUN from first
                                                     // CCB
                                                     //
  PvScsiBtStatSensFailed = 0x1b,                     // Auto request sense failed
  PvScsiBtStatTagReject  = 0x1c,                     //
                                                     // SCSI II tagged queueing message rejected
                                                     // by target
                                                     //
  PvScsiBtStatBadMsg = 0x1d,                         //
                                                     // Unsupported message received by the host
                                                     // adapter
                                                     //
  PvScsiBtStatHaHardware = 0x20,                     // Host adapter hardware failed
  PvScsiBtStatNoResponse = 0x21,                     //
                                                     // Target did not respond to SCSI ATN sent
                                                     // a SCSI RST
                                                     //
  PvScsiBtStatSentRst    = 0x22,                     // Host adapter asserted a SCSI RST
  PvScsiBtStatRecvRst    = 0x23,                     // Other SCSI devices asserted a SCSI RST
  PvScsiBtStatDisconnect = 0x24,                     //
                                                     // Target device reconnected improperly
                                                     // (w/o tag)
                                                     //
  PvScsiBtStatBusReset   = 0x25,                     // Host adapter issued BUS device reset
  PvScsiBtStatAbortQueue = 0x26,                     // Abort queue generated
  PvScsiBtStatHaSoftware = 0x27,                     // Host adapter software error
  PvScsiBtStatHaTimeout  = 0x30,                     // Host adapter hardware timeout error
  PvScsiBtStatScsiParity = 0x34,                     // SCSI parity error detected
} PVSCSI_HOST_BUS_ADAPTER_STATUS;

#pragma pack (1)
typedef struct {
  UINT64    Context;
  UINT64    DataLen;
  UINT32    SenseLen;
  UINT16    HostStatus;
  UINT16    ScsiStatus;
  UINT32    Pad[2];
} PVSCSI_RING_CMP_DESC;
#pragma pack ()

#endif // __PVSCSI_H_