2021-06-02 07:54:50 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0 */
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 Broadcom. All Rights Reserved. The term
|
|
|
|
* “Broadcom” refers to Broadcom Inc. and/or its subsidiaries.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __EFCLIB_H__
|
|
|
|
#define __EFCLIB_H__
|
|
|
|
|
|
|
|
#include "scsi/fc/fc_els.h"
|
|
|
|
#include "scsi/fc/fc_fs.h"
|
|
|
|
#include "scsi/fc/fc_ns.h"
|
|
|
|
#include "scsi/fc/fc_gs.h"
|
|
|
|
#include "scsi/fc_frame.h"
|
|
|
|
#include "../include/efc_common.h"
|
|
|
|
#include "../libefc_sli/sli4.h"
|
|
|
|
|
|
|
|
#define EFC_SERVICE_PARMS_LENGTH 120
|
|
|
|
#define EFC_NAME_LENGTH 32
|
|
|
|
#define EFC_SM_NAME_LENGTH 64
|
|
|
|
#define EFC_DISPLAY_BUS_INFO_LENGTH 16
|
|
|
|
|
|
|
|
#define EFC_WWN_LENGTH 32
|
|
|
|
|
|
|
|
#define EFC_FC_ELS_DEFAULT_RETRIES 3
|
|
|
|
|
|
|
|
/* Timeouts */
|
|
|
|
#define EFC_FC_ELS_SEND_DEFAULT_TIMEOUT 0
|
|
|
|
#define EFC_FC_FLOGI_TIMEOUT_SEC 5
|
|
|
|
#define EFC_SHUTDOWN_TIMEOUT_USEC 30000000
|
|
|
|
|
|
|
|
/* Return values for calls from base driver to libefc */
|
|
|
|
#define EFC_SCSI_CALL_COMPLETE 0
|
|
|
|
#define EFC_SCSI_CALL_ASYNC 1
|
|
|
|
|
|
|
|
/* Local port topology */
|
|
|
|
enum efc_nport_topology {
|
|
|
|
EFC_NPORT_TOPO_UNKNOWN = 0,
|
|
|
|
EFC_NPORT_TOPO_FABRIC,
|
|
|
|
EFC_NPORT_TOPO_P2P,
|
|
|
|
EFC_NPORT_TOPO_FC_AL,
|
|
|
|
};
|
|
|
|
|
|
|
|
#define enable_target_rscn(efc) 1
|
|
|
|
|
|
|
|
enum efc_node_shutd_rsn {
|
|
|
|
EFC_NODE_SHUTDOWN_DEFAULT = 0,
|
|
|
|
EFC_NODE_SHUTDOWN_EXPLICIT_LOGO,
|
|
|
|
EFC_NODE_SHUTDOWN_IMPLICIT_LOGO,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum efc_node_send_ls_acc {
|
|
|
|
EFC_NODE_SEND_LS_ACC_NONE = 0,
|
|
|
|
EFC_NODE_SEND_LS_ACC_PLOGI,
|
|
|
|
EFC_NODE_SEND_LS_ACC_PRLI,
|
|
|
|
};
|
|
|
|
|
|
|
|
#define EFC_LINK_STATUS_UP 0
|
|
|
|
#define EFC_LINK_STATUS_DOWN 1
|
|
|
|
|
|
|
|
/* State machine context header */
|
|
|
|
struct efc_sm_ctx {
|
|
|
|
void (*current_state)(struct efc_sm_ctx *ctx,
|
|
|
|
u32 evt, void *arg);
|
|
|
|
|
|
|
|
const char *description;
|
|
|
|
void *app;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Description of discovered Fabric Domain */
|
|
|
|
struct efc_domain_record {
|
|
|
|
u32 index;
|
|
|
|
u32 priority;
|
|
|
|
u8 address[6];
|
|
|
|
u8 wwn[8];
|
|
|
|
union {
|
|
|
|
u8 vlan[512];
|
|
|
|
u8 loop[128];
|
|
|
|
} map;
|
|
|
|
u32 speed;
|
|
|
|
u32 fc_id;
|
|
|
|
bool is_loop;
|
|
|
|
bool is_nport;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Domain events */
|
|
|
|
enum efc_hw_domain_event {
|
|
|
|
EFC_HW_DOMAIN_ALLOC_OK,
|
|
|
|
EFC_HW_DOMAIN_ALLOC_FAIL,
|
|
|
|
EFC_HW_DOMAIN_ATTACH_OK,
|
|
|
|
EFC_HW_DOMAIN_ATTACH_FAIL,
|
|
|
|
EFC_HW_DOMAIN_FREE_OK,
|
|
|
|
EFC_HW_DOMAIN_FREE_FAIL,
|
|
|
|
EFC_HW_DOMAIN_LOST,
|
|
|
|
EFC_HW_DOMAIN_FOUND,
|
|
|
|
EFC_HW_DOMAIN_CHANGED,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fibre Channel port object
|
|
|
|
*
|
|
|
|
* @list_entry: nport list entry
|
|
|
|
* @ref: reference count, each node takes a reference
|
|
|
|
* @release: function to free nport object
|
|
|
|
* @efc: pointer back to efc
|
|
|
|
* @instance_index: unique instance index value
|
|
|
|
* @display_name: port display name
|
|
|
|
* @is_vport: Is NPIV port
|
|
|
|
* @free_req_pending: pending request to free resources
|
|
|
|
* @attached: mark attached if reg VPI succeeds
|
|
|
|
* @p2p_winner: TRUE if we're the point-to-point winner
|
|
|
|
* @domain: pointer back to domain
|
|
|
|
* @wwpn: port wwpn
|
|
|
|
* @wwnn: port wwnn
|
|
|
|
* @tgt_data: target backend private port data
|
|
|
|
* @ini_data: initiator backend private port data
|
|
|
|
* @indicator: VPI
|
|
|
|
* @fc_id: port FC address
|
|
|
|
* @dma: memory for Service Parameters
|
|
|
|
* @wwnn_str: wwpn string
|
|
|
|
* @sli_wwpn: SLI provided wwpn
|
|
|
|
* @sli_wwnn: SLI provided wwnn
|
|
|
|
* @sm: nport state machine context
|
|
|
|
* @lookup: fc_id to node lookup object
|
|
|
|
* @enable_ini: SCSI initiator enabled for this port
|
|
|
|
* @enable_tgt: SCSI target enabled for this port
|
|
|
|
* @enable_rscn: port will be expecting RSCN
|
|
|
|
* @shutting_down: nport in process of shutting down
|
|
|
|
* @p2p_port_id: our port id for point-to-point
|
|
|
|
* @topology: topology: fabric/p2p/unknown
|
|
|
|
* @service_params: login parameters
|
|
|
|
* @p2p_remote_port_id: remote node's port id for point-to-point
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct efc_nport {
|
|
|
|
struct list_head list_entry;
|
|
|
|
struct kref ref;
|
|
|
|
void (*release)(struct kref *arg);
|
|
|
|
struct efc *efc;
|
|
|
|
u32 instance_index;
|
|
|
|
char display_name[EFC_NAME_LENGTH];
|
|
|
|
bool is_vport;
|
|
|
|
bool free_req_pending;
|
|
|
|
bool attached;
|
2021-09-14 18:55:38 +08:00
|
|
|
bool attaching;
|
2021-06-02 07:54:50 +08:00
|
|
|
bool p2p_winner;
|
|
|
|
struct efc_domain *domain;
|
|
|
|
u64 wwpn;
|
|
|
|
u64 wwnn;
|
|
|
|
void *tgt_data;
|
|
|
|
void *ini_data;
|
|
|
|
|
|
|
|
u32 indicator;
|
|
|
|
u32 fc_id;
|
|
|
|
struct efc_dma dma;
|
|
|
|
|
|
|
|
u8 wwnn_str[EFC_WWN_LENGTH];
|
|
|
|
__be64 sli_wwpn;
|
|
|
|
__be64 sli_wwnn;
|
|
|
|
|
|
|
|
struct efc_sm_ctx sm;
|
|
|
|
struct xarray lookup;
|
|
|
|
bool enable_ini;
|
|
|
|
bool enable_tgt;
|
|
|
|
bool enable_rscn;
|
|
|
|
bool shutting_down;
|
|
|
|
u32 p2p_port_id;
|
|
|
|
enum efc_nport_topology topology;
|
|
|
|
u8 service_params[EFC_SERVICE_PARMS_LENGTH];
|
|
|
|
u32 p2p_remote_port_id;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fibre Channel domain object
|
|
|
|
*
|
|
|
|
* This object is a container for the various SLI components needed
|
|
|
|
* to connect to the domain of a FC or FCoE switch
|
|
|
|
* @efc: pointer back to efc
|
|
|
|
* @instance_index: unique instance index value
|
|
|
|
* @display_name: Node display name
|
|
|
|
* @nport_list: linked list of nports associated with this domain
|
|
|
|
* @ref: Reference count, each nport takes a reference
|
|
|
|
* @release: Function to free domain object
|
|
|
|
* @ini_domain: initiator backend private domain data
|
|
|
|
* @tgt_domain: target backend private domain data
|
|
|
|
* @sm: state machine context
|
|
|
|
* @fcf: FC Forwarder table index
|
|
|
|
* @fcf_indicator: FCFI
|
|
|
|
* @indicator: VFI
|
|
|
|
* @nport_count: Number of nports allocated
|
|
|
|
* @dma: memory for Service Parameters
|
|
|
|
* @fcf_wwn: WWN for FCF/switch
|
|
|
|
* @drvsm: driver domain sm context
|
|
|
|
* @attached: set true after attach completes
|
|
|
|
* @is_fc: is FC
|
|
|
|
* @is_loop: is loop topology
|
|
|
|
* @is_nlport: is public loop
|
|
|
|
* @domain_found_pending:A domain found is pending, drec is updated
|
|
|
|
* @req_domain_free: True if domain object should be free'd
|
|
|
|
* @req_accept_frames: set in domain state machine to enable frames
|
|
|
|
* @domain_notify_pend: Set in domain SM to avoid duplicate node event post
|
|
|
|
* @pending_drec: Pending drec if a domain found is pending
|
|
|
|
* @service_params: any nports service parameters
|
|
|
|
* @flogi_service_params:Fabric/P2p service parameters from FLOGI
|
|
|
|
* @lookup: d_id to node lookup object
|
|
|
|
* @nport: Pointer to first (physical) SLI port
|
|
|
|
*/
|
|
|
|
struct efc_domain {
|
|
|
|
struct efc *efc;
|
|
|
|
char display_name[EFC_NAME_LENGTH];
|
|
|
|
struct list_head nport_list;
|
|
|
|
struct kref ref;
|
|
|
|
void (*release)(struct kref *arg);
|
|
|
|
void *ini_domain;
|
|
|
|
void *tgt_domain;
|
|
|
|
|
|
|
|
/* Declarations private to HW/SLI */
|
|
|
|
u32 fcf;
|
|
|
|
u32 fcf_indicator;
|
|
|
|
u32 indicator;
|
|
|
|
u32 nport_count;
|
|
|
|
struct efc_dma dma;
|
|
|
|
|
|
|
|
/* Declarations private to FC trannport */
|
|
|
|
u64 fcf_wwn;
|
|
|
|
struct efc_sm_ctx drvsm;
|
|
|
|
bool attached;
|
|
|
|
bool is_fc;
|
|
|
|
bool is_loop;
|
|
|
|
bool is_nlport;
|
|
|
|
bool domain_found_pending;
|
|
|
|
bool req_domain_free;
|
|
|
|
bool req_accept_frames;
|
|
|
|
bool domain_notify_pend;
|
|
|
|
|
|
|
|
struct efc_domain_record pending_drec;
|
|
|
|
u8 service_params[EFC_SERVICE_PARMS_LENGTH];
|
|
|
|
u8 flogi_service_params[EFC_SERVICE_PARMS_LENGTH];
|
|
|
|
|
|
|
|
struct xarray lookup;
|
|
|
|
|
|
|
|
struct efc_nport *nport;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remote Node object
|
|
|
|
*
|
|
|
|
* This object represents a connection between the SLI port and another
|
|
|
|
* Nx_Port on the fabric. Note this can be either a well known port such
|
|
|
|
* as a F_Port (i.e. ff:ff:fe) or another N_Port.
|
|
|
|
* @indicator: RPI
|
|
|
|
* @fc_id: FC address
|
|
|
|
* @attached: true if attached
|
|
|
|
* @nport: associated SLI port
|
|
|
|
* @node: associated node
|
|
|
|
*/
|
|
|
|
struct efc_remote_node {
|
|
|
|
u32 indicator;
|
|
|
|
u32 index;
|
|
|
|
u32 fc_id;
|
|
|
|
|
|
|
|
bool attached;
|
|
|
|
|
|
|
|
struct efc_nport *nport;
|
|
|
|
void *node;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* FC Node object
|
|
|
|
* @efc: pointer back to efc structure
|
|
|
|
* @display_name: Node display name
|
|
|
|
* @nort: Assosiated nport pointer.
|
|
|
|
* @hold_frames: hold incoming frames if true
|
|
|
|
* @els_io_enabled: Enable allocating els ios for this node
|
|
|
|
* @els_ios_lock: lock to protect the els ios list
|
|
|
|
* @els_ios_list: ELS I/O's for this node
|
|
|
|
* @ini_node: backend initiator private node data
|
|
|
|
* @tgt_node: backend target private node data
|
|
|
|
* @rnode: Remote node
|
|
|
|
* @sm: state machine context
|
|
|
|
* @evtdepth: current event posting nesting depth
|
|
|
|
* @req_free: this node is to be free'd
|
|
|
|
* @attached: node is attached (REGLOGIN complete)
|
|
|
|
* @fcp_enabled: node is enabled to handle FCP
|
|
|
|
* @rscn_pending: for name server node RSCN is pending
|
|
|
|
* @send_plogi: send PLOGI accept, upon completion of node attach
|
|
|
|
* @send_plogi_acc: TRUE if io_alloc() is enabled.
|
|
|
|
* @send_ls_acc: type of LS acc to send
|
|
|
|
* @ls_acc_io: SCSI IO for LS acc
|
|
|
|
* @ls_acc_oxid: OX_ID for pending accept
|
|
|
|
* @ls_acc_did: D_ID for pending accept
|
|
|
|
* @shutdown_reason: reason for node shutdown
|
|
|
|
* @sparm_dma_buf: service parameters buffer
|
|
|
|
* @service_params: plogi/acc frame from remote device
|
|
|
|
* @pend_frames_lock: lock for inbound pending frames list
|
|
|
|
* @pend_frames: inbound pending frames list
|
|
|
|
* @pend_frames_processed:count of frames processed in hold frames interval
|
|
|
|
* @ox_id_in_use: used to verify one at a time us of ox_id
|
|
|
|
* @els_retries_remaining:for ELS, number of retries remaining
|
|
|
|
* @els_req_cnt: number of outstanding ELS requests
|
|
|
|
* @els_cmpl_cnt: number of outstanding ELS completions
|
|
|
|
* @abort_cnt: Abort counter for debugging purpos
|
|
|
|
* @current_state_name: current node state
|
|
|
|
* @prev_state_name: previous node state
|
|
|
|
* @current_evt: current event
|
|
|
|
* @prev_evt: previous event
|
|
|
|
* @targ: node is target capable
|
|
|
|
* @init: node is init capable
|
|
|
|
* @refound: Handle node refound case when node is being deleted
|
|
|
|
* @els_io_pend_list: list of pending (not yet processed) ELS IOs
|
|
|
|
* @els_io_active_list: list of active (processed) ELS IOs
|
|
|
|
* @nodedb_state: Node debugging, saved state
|
|
|
|
* @gidpt_delay_timer: GIDPT delay timer
|
|
|
|
* @time_last_gidpt_msec:Start time of last target RSCN GIDPT
|
|
|
|
* @wwnn: remote port WWNN
|
|
|
|
* @wwpn: remote port WWPN
|
|
|
|
*/
|
|
|
|
struct efc_node {
|
|
|
|
struct efc *efc;
|
|
|
|
char display_name[EFC_NAME_LENGTH];
|
|
|
|
struct efc_nport *nport;
|
|
|
|
struct kref ref;
|
|
|
|
void (*release)(struct kref *arg);
|
|
|
|
bool hold_frames;
|
|
|
|
bool els_io_enabled;
|
|
|
|
bool send_plogi_acc;
|
|
|
|
bool send_plogi;
|
|
|
|
bool rscn_pending;
|
|
|
|
bool fcp_enabled;
|
|
|
|
bool attached;
|
|
|
|
bool req_free;
|
|
|
|
|
|
|
|
spinlock_t els_ios_lock;
|
|
|
|
struct list_head els_ios_list;
|
|
|
|
void *ini_node;
|
|
|
|
void *tgt_node;
|
|
|
|
|
|
|
|
struct efc_remote_node rnode;
|
|
|
|
/* Declarations private to FC trannport */
|
|
|
|
struct efc_sm_ctx sm;
|
|
|
|
u32 evtdepth;
|
|
|
|
|
|
|
|
enum efc_node_send_ls_acc send_ls_acc;
|
|
|
|
void *ls_acc_io;
|
|
|
|
u32 ls_acc_oxid;
|
|
|
|
u32 ls_acc_did;
|
|
|
|
enum efc_node_shutd_rsn shutdown_reason;
|
|
|
|
bool targ;
|
|
|
|
bool init;
|
|
|
|
bool refound;
|
|
|
|
struct efc_dma sparm_dma_buf;
|
|
|
|
u8 service_params[EFC_SERVICE_PARMS_LENGTH];
|
|
|
|
spinlock_t pend_frames_lock;
|
|
|
|
struct list_head pend_frames;
|
|
|
|
u32 pend_frames_processed;
|
|
|
|
u32 ox_id_in_use;
|
|
|
|
u32 els_retries_remaining;
|
|
|
|
u32 els_req_cnt;
|
|
|
|
u32 els_cmpl_cnt;
|
|
|
|
u32 abort_cnt;
|
|
|
|
|
|
|
|
char current_state_name[EFC_SM_NAME_LENGTH];
|
|
|
|
char prev_state_name[EFC_SM_NAME_LENGTH];
|
|
|
|
int current_evt;
|
|
|
|
int prev_evt;
|
|
|
|
|
|
|
|
void (*nodedb_state)(struct efc_sm_ctx *ctx,
|
|
|
|
u32 evt, void *arg);
|
|
|
|
struct timer_list gidpt_delay_timer;
|
|
|
|
u64 time_last_gidpt_msec;
|
|
|
|
|
|
|
|
char wwnn[EFC_WWN_LENGTH];
|
|
|
|
char wwpn[EFC_WWN_LENGTH];
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* NPIV port
|
|
|
|
*
|
|
|
|
* Collection of the information required to restore a virtual port across
|
|
|
|
* link events
|
|
|
|
* @wwnn: node name
|
|
|
|
* @wwpn: port name
|
|
|
|
* @fc_id: port id
|
|
|
|
* @tgt_data: target backend pointer
|
|
|
|
* @ini_data: initiator backend pointe
|
|
|
|
* @nport: Used to match record after attaching for update
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct efc_vport {
|
|
|
|
struct list_head list_entry;
|
|
|
|
u64 wwnn;
|
|
|
|
u64 wwpn;
|
|
|
|
u32 fc_id;
|
|
|
|
bool enable_tgt;
|
|
|
|
bool enable_ini;
|
|
|
|
void *tgt_data;
|
|
|
|
void *ini_data;
|
|
|
|
struct efc_nport *nport;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define node_printf(node, fmt, args...) \
|
|
|
|
efc_log_info(node->efc, "[%s] " fmt, node->display_name, ##args)
|
|
|
|
|
|
|
|
/* Node SM IO Context Callback structure */
|
|
|
|
struct efc_node_cb {
|
|
|
|
int status;
|
|
|
|
int ext_status;
|
|
|
|
struct efc_hw_rq_buffer *header;
|
|
|
|
struct efc_hw_rq_buffer *payload;
|
|
|
|
struct efc_dma els_rsp;
|
|
|
|
|
|
|
|
/* Actual length of data received */
|
|
|
|
int rsp_len;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct efc_hw_rq_buffer {
|
|
|
|
u16 rqindex;
|
|
|
|
struct efc_dma dma;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* FC sequence object
|
|
|
|
*
|
|
|
|
* Defines a general FC sequence object
|
|
|
|
* @hw: HW that owns this sequence
|
|
|
|
* @fcfi: FCFI associated with sequence
|
|
|
|
* @header: Received frame header
|
|
|
|
* @payload: Received frame header
|
|
|
|
* @hw_priv: HW private context
|
|
|
|
*/
|
|
|
|
struct efc_hw_sequence {
|
|
|
|
struct list_head list_entry;
|
|
|
|
void *hw;
|
|
|
|
u8 fcfi;
|
|
|
|
struct efc_hw_rq_buffer *header;
|
|
|
|
struct efc_hw_rq_buffer *payload;
|
|
|
|
void *hw_priv;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum efc_disc_io_type {
|
|
|
|
EFC_DISC_IO_ELS_REQ,
|
|
|
|
EFC_DISC_IO_ELS_RESP,
|
|
|
|
EFC_DISC_IO_CT_REQ,
|
|
|
|
EFC_DISC_IO_CT_RESP
|
|
|
|
};
|
|
|
|
|
|
|
|
struct efc_io_els_params {
|
|
|
|
u32 s_id;
|
|
|
|
u16 ox_id;
|
|
|
|
u8 timeout;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct efc_io_ct_params {
|
|
|
|
u8 r_ctl;
|
|
|
|
u8 type;
|
|
|
|
u8 df_ctl;
|
|
|
|
u8 timeout;
|
|
|
|
u16 ox_id;
|
|
|
|
};
|
|
|
|
|
|
|
|
union efc_disc_io_param {
|
|
|
|
struct efc_io_els_params els;
|
|
|
|
struct efc_io_ct_params ct;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct efc_disc_io {
|
|
|
|
struct efc_dma req; /* send buffer */
|
|
|
|
struct efc_dma rsp; /* receive buffer */
|
|
|
|
enum efc_disc_io_type io_type; /* EFC_DISC_IO_TYPE enum*/
|
|
|
|
u16 xmit_len; /* Length of els request*/
|
|
|
|
u16 rsp_len; /* Max length of rsps to be rcvd */
|
|
|
|
u32 rpi; /* Registered RPI */
|
|
|
|
u32 vpi; /* VPI for this nport */
|
|
|
|
u32 s_id;
|
|
|
|
u32 d_id;
|
|
|
|
bool rpi_registered; /* if false, use tmp RPI */
|
|
|
|
union efc_disc_io_param iparam;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Return value indiacating the sequence can not be freed */
|
|
|
|
#define EFC_HW_SEQ_HOLD 0
|
|
|
|
/* Return value indiacating the sequence can be freed */
|
|
|
|
#define EFC_HW_SEQ_FREE 1
|
|
|
|
|
|
|
|
struct libefc_function_template {
|
|
|
|
/*Sport*/
|
|
|
|
int (*new_nport)(struct efc *efc, struct efc_nport *sp);
|
|
|
|
void (*del_nport)(struct efc *efc, struct efc_nport *sp);
|
|
|
|
|
|
|
|
/*Scsi Node*/
|
|
|
|
int (*scsi_new_node)(struct efc *efc, struct efc_node *n);
|
|
|
|
int (*scsi_del_node)(struct efc *efc, struct efc_node *n, int reason);
|
|
|
|
|
|
|
|
int (*issue_mbox_rqst)(void *efct, void *buf, void *cb, void *arg);
|
|
|
|
/*Send ELS IO*/
|
|
|
|
int (*send_els)(struct efc *efc, struct efc_disc_io *io);
|
|
|
|
/*Send BLS IO*/
|
|
|
|
int (*send_bls)(struct efc *efc, u32 type, struct sli_bls_params *bls);
|
|
|
|
/*Free HW frame*/
|
|
|
|
int (*hw_seq_free)(struct efc *efc, struct efc_hw_sequence *seq);
|
|
|
|
};
|
|
|
|
|
|
|
|
#define EFC_LOG_LIB 0x01
|
|
|
|
#define EFC_LOG_NODE 0x02
|
|
|
|
#define EFC_LOG_PORT 0x04
|
|
|
|
#define EFC_LOG_DOMAIN 0x08
|
|
|
|
#define EFC_LOG_ELS 0x10
|
|
|
|
#define EFC_LOG_DOMAIN_SM 0x20
|
|
|
|
#define EFC_LOG_SM 0x40
|
|
|
|
|
|
|
|
/* efc library port structure */
|
|
|
|
struct efc {
|
|
|
|
void *base;
|
|
|
|
struct pci_dev *pci;
|
|
|
|
struct sli4 *sli;
|
|
|
|
u32 fcfi;
|
|
|
|
u64 req_wwpn;
|
|
|
|
u64 req_wwnn;
|
|
|
|
|
|
|
|
u64 def_wwpn;
|
|
|
|
u64 def_wwnn;
|
|
|
|
u64 max_xfer_size;
|
|
|
|
mempool_t *node_pool;
|
|
|
|
struct dma_pool *node_dma_pool;
|
|
|
|
u32 nodes_count;
|
|
|
|
|
|
|
|
u32 link_status;
|
|
|
|
|
|
|
|
struct list_head vport_list;
|
|
|
|
/* lock to protect the vport list */
|
|
|
|
spinlock_t vport_lock;
|
|
|
|
|
|
|
|
struct libefc_function_template tt;
|
|
|
|
/* lock to protect the discovery library.
|
|
|
|
* Refer to efclib.c for more details.
|
|
|
|
*/
|
|
|
|
spinlock_t lock;
|
|
|
|
|
|
|
|
bool enable_ini;
|
|
|
|
bool enable_tgt;
|
|
|
|
|
|
|
|
u32 log_level;
|
|
|
|
|
|
|
|
struct efc_domain *domain;
|
|
|
|
void (*domain_free_cb)(struct efc *efc, void *arg);
|
|
|
|
void *domain_free_cb_arg;
|
|
|
|
|
|
|
|
u64 tgt_rscn_delay_msec;
|
|
|
|
u64 tgt_rscn_period_msec;
|
|
|
|
|
|
|
|
bool external_loopback;
|
|
|
|
u32 nodedb_mask;
|
|
|
|
u32 logmask;
|
|
|
|
mempool_t *els_io_pool;
|
|
|
|
atomic_t els_io_alloc_failed_count;
|
|
|
|
|
|
|
|
/* hold pending frames */
|
|
|
|
bool hold_frames;
|
|
|
|
/* lock to protect pending frames list access */
|
|
|
|
spinlock_t pend_frames_lock;
|
|
|
|
struct list_head pend_frames;
|
|
|
|
/* count of pending frames that were processed */
|
|
|
|
u32 pend_frames_processed;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EFC library registration
|
|
|
|
* **********************************/
|
|
|
|
int efcport_init(struct efc *efc);
|
|
|
|
void efcport_destroy(struct efc *efc);
|
|
|
|
/*
|
|
|
|
* EFC Domain
|
|
|
|
* **********************************/
|
|
|
|
int efc_domain_cb(void *arg, int event, void *data);
|
|
|
|
void
|
|
|
|
efc_register_domain_free_cb(struct efc *efc,
|
|
|
|
void (*callback)(struct efc *efc, void *arg),
|
|
|
|
void *arg);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EFC nport
|
|
|
|
* **********************************/
|
|
|
|
void efc_nport_cb(void *arg, int event, void *data);
|
|
|
|
struct efc_vport *
|
|
|
|
efc_vport_create_spec(struct efc *efc, u64 wwnn, u64 wwpn, u32 fc_id,
|
|
|
|
bool enable_ini, bool enable_tgt,
|
|
|
|
void *tgt_data, void *ini_data);
|
|
|
|
int efc_nport_vport_new(struct efc_domain *domain, u64 wwpn,
|
|
|
|
u64 wwnn, u32 fc_id, bool ini, bool tgt,
|
|
|
|
void *tgt_data, void *ini_data);
|
|
|
|
int efc_nport_vport_del(struct efc *efc, struct efc_domain *domain,
|
|
|
|
u64 wwpn, u64 wwnn);
|
|
|
|
|
|
|
|
void efc_vport_del_all(struct efc *efc);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EFC Node
|
|
|
|
* **********************************/
|
|
|
|
int efc_remote_node_cb(void *arg, int event, void *data);
|
|
|
|
void efc_node_fcid_display(u32 fc_id, char *buffer, u32 buf_len);
|
|
|
|
void efc_node_post_shutdown(struct efc_node *node, void *arg);
|
|
|
|
u64 efc_node_get_wwpn(struct efc_node *node);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EFC FCP/ELS/CT interface
|
|
|
|
* **********************************/
|
|
|
|
void efc_dispatch_frame(struct efc *efc, struct efc_hw_sequence *seq);
|
|
|
|
void efc_disc_io_complete(struct efc_disc_io *io, u32 len, u32 status,
|
|
|
|
u32 ext_status);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EFC SCSI INTERACTION LAYER
|
|
|
|
* **********************************/
|
|
|
|
void efc_scsi_sess_reg_complete(struct efc_node *node, u32 status);
|
|
|
|
void efc_scsi_del_initiator_complete(struct efc *efc, struct efc_node *node);
|
|
|
|
void efc_scsi_del_target_complete(struct efc *efc, struct efc_node *node);
|
|
|
|
void efc_scsi_io_list_empty(struct efc *efc, struct efc_node *node);
|
|
|
|
|
|
|
|
#endif /* __EFCLIB_H__ */
|