/* 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 enum efc_sm_event; /* State machine context header */ struct efc_sm_ctx { void (*current_state)(struct efc_sm_ctx *ctx, enum efc_sm_event 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; 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, enum efc_sm_event 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__ */