162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0 */ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2021 Broadcom. All Rights Reserved. The term 462306a36Sopenharmony_ci * “Broadcom” refers to Broadcom Inc. and/or its subsidiaries. 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#ifndef __EFCLIB_H__ 862306a36Sopenharmony_ci#define __EFCLIB_H__ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include "scsi/fc/fc_els.h" 1162306a36Sopenharmony_ci#include "scsi/fc/fc_fs.h" 1262306a36Sopenharmony_ci#include "scsi/fc/fc_ns.h" 1362306a36Sopenharmony_ci#include "scsi/fc/fc_gs.h" 1462306a36Sopenharmony_ci#include "scsi/fc_frame.h" 1562306a36Sopenharmony_ci#include "../include/efc_common.h" 1662306a36Sopenharmony_ci#include "../libefc_sli/sli4.h" 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#define EFC_SERVICE_PARMS_LENGTH 120 1962306a36Sopenharmony_ci#define EFC_NAME_LENGTH 32 2062306a36Sopenharmony_ci#define EFC_SM_NAME_LENGTH 64 2162306a36Sopenharmony_ci#define EFC_DISPLAY_BUS_INFO_LENGTH 16 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci#define EFC_WWN_LENGTH 32 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci#define EFC_FC_ELS_DEFAULT_RETRIES 3 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci/* Timeouts */ 2862306a36Sopenharmony_ci#define EFC_FC_ELS_SEND_DEFAULT_TIMEOUT 0 2962306a36Sopenharmony_ci#define EFC_FC_FLOGI_TIMEOUT_SEC 5 3062306a36Sopenharmony_ci#define EFC_SHUTDOWN_TIMEOUT_USEC 30000000 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ci/* Return values for calls from base driver to libefc */ 3362306a36Sopenharmony_ci#define EFC_SCSI_CALL_COMPLETE 0 3462306a36Sopenharmony_ci#define EFC_SCSI_CALL_ASYNC 1 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci/* Local port topology */ 3762306a36Sopenharmony_cienum efc_nport_topology { 3862306a36Sopenharmony_ci EFC_NPORT_TOPO_UNKNOWN = 0, 3962306a36Sopenharmony_ci EFC_NPORT_TOPO_FABRIC, 4062306a36Sopenharmony_ci EFC_NPORT_TOPO_P2P, 4162306a36Sopenharmony_ci EFC_NPORT_TOPO_FC_AL, 4262306a36Sopenharmony_ci}; 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci#define enable_target_rscn(efc) 1 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_cienum efc_node_shutd_rsn { 4762306a36Sopenharmony_ci EFC_NODE_SHUTDOWN_DEFAULT = 0, 4862306a36Sopenharmony_ci EFC_NODE_SHUTDOWN_EXPLICIT_LOGO, 4962306a36Sopenharmony_ci EFC_NODE_SHUTDOWN_IMPLICIT_LOGO, 5062306a36Sopenharmony_ci}; 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_cienum efc_node_send_ls_acc { 5362306a36Sopenharmony_ci EFC_NODE_SEND_LS_ACC_NONE = 0, 5462306a36Sopenharmony_ci EFC_NODE_SEND_LS_ACC_PLOGI, 5562306a36Sopenharmony_ci EFC_NODE_SEND_LS_ACC_PRLI, 5662306a36Sopenharmony_ci}; 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci#define EFC_LINK_STATUS_UP 0 5962306a36Sopenharmony_ci#define EFC_LINK_STATUS_DOWN 1 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_cienum efc_sm_event; 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci/* State machine context header */ 6462306a36Sopenharmony_cistruct efc_sm_ctx { 6562306a36Sopenharmony_ci void (*current_state)(struct efc_sm_ctx *ctx, 6662306a36Sopenharmony_ci enum efc_sm_event evt, void *arg); 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci const char *description; 6962306a36Sopenharmony_ci void *app; 7062306a36Sopenharmony_ci}; 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci/* Description of discovered Fabric Domain */ 7362306a36Sopenharmony_cistruct efc_domain_record { 7462306a36Sopenharmony_ci u32 index; 7562306a36Sopenharmony_ci u32 priority; 7662306a36Sopenharmony_ci u8 address[6]; 7762306a36Sopenharmony_ci u8 wwn[8]; 7862306a36Sopenharmony_ci union { 7962306a36Sopenharmony_ci u8 vlan[512]; 8062306a36Sopenharmony_ci u8 loop[128]; 8162306a36Sopenharmony_ci } map; 8262306a36Sopenharmony_ci u32 speed; 8362306a36Sopenharmony_ci u32 fc_id; 8462306a36Sopenharmony_ci bool is_loop; 8562306a36Sopenharmony_ci bool is_nport; 8662306a36Sopenharmony_ci}; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci/* Domain events */ 8962306a36Sopenharmony_cienum efc_hw_domain_event { 9062306a36Sopenharmony_ci EFC_HW_DOMAIN_ALLOC_OK, 9162306a36Sopenharmony_ci EFC_HW_DOMAIN_ALLOC_FAIL, 9262306a36Sopenharmony_ci EFC_HW_DOMAIN_ATTACH_OK, 9362306a36Sopenharmony_ci EFC_HW_DOMAIN_ATTACH_FAIL, 9462306a36Sopenharmony_ci EFC_HW_DOMAIN_FREE_OK, 9562306a36Sopenharmony_ci EFC_HW_DOMAIN_FREE_FAIL, 9662306a36Sopenharmony_ci EFC_HW_DOMAIN_LOST, 9762306a36Sopenharmony_ci EFC_HW_DOMAIN_FOUND, 9862306a36Sopenharmony_ci EFC_HW_DOMAIN_CHANGED, 9962306a36Sopenharmony_ci}; 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_ci/** 10262306a36Sopenharmony_ci * Fibre Channel port object 10362306a36Sopenharmony_ci * 10462306a36Sopenharmony_ci * @list_entry: nport list entry 10562306a36Sopenharmony_ci * @ref: reference count, each node takes a reference 10662306a36Sopenharmony_ci * @release: function to free nport object 10762306a36Sopenharmony_ci * @efc: pointer back to efc 10862306a36Sopenharmony_ci * @instance_index: unique instance index value 10962306a36Sopenharmony_ci * @display_name: port display name 11062306a36Sopenharmony_ci * @is_vport: Is NPIV port 11162306a36Sopenharmony_ci * @free_req_pending: pending request to free resources 11262306a36Sopenharmony_ci * @attached: mark attached if reg VPI succeeds 11362306a36Sopenharmony_ci * @p2p_winner: TRUE if we're the point-to-point winner 11462306a36Sopenharmony_ci * @domain: pointer back to domain 11562306a36Sopenharmony_ci * @wwpn: port wwpn 11662306a36Sopenharmony_ci * @wwnn: port wwnn 11762306a36Sopenharmony_ci * @tgt_data: target backend private port data 11862306a36Sopenharmony_ci * @ini_data: initiator backend private port data 11962306a36Sopenharmony_ci * @indicator: VPI 12062306a36Sopenharmony_ci * @fc_id: port FC address 12162306a36Sopenharmony_ci * @dma: memory for Service Parameters 12262306a36Sopenharmony_ci * @wwnn_str: wwpn string 12362306a36Sopenharmony_ci * @sli_wwpn: SLI provided wwpn 12462306a36Sopenharmony_ci * @sli_wwnn: SLI provided wwnn 12562306a36Sopenharmony_ci * @sm: nport state machine context 12662306a36Sopenharmony_ci * @lookup: fc_id to node lookup object 12762306a36Sopenharmony_ci * @enable_ini: SCSI initiator enabled for this port 12862306a36Sopenharmony_ci * @enable_tgt: SCSI target enabled for this port 12962306a36Sopenharmony_ci * @enable_rscn: port will be expecting RSCN 13062306a36Sopenharmony_ci * @shutting_down: nport in process of shutting down 13162306a36Sopenharmony_ci * @p2p_port_id: our port id for point-to-point 13262306a36Sopenharmony_ci * @topology: topology: fabric/p2p/unknown 13362306a36Sopenharmony_ci * @service_params: login parameters 13462306a36Sopenharmony_ci * @p2p_remote_port_id: remote node's port id for point-to-point 13562306a36Sopenharmony_ci */ 13662306a36Sopenharmony_ci 13762306a36Sopenharmony_cistruct efc_nport { 13862306a36Sopenharmony_ci struct list_head list_entry; 13962306a36Sopenharmony_ci struct kref ref; 14062306a36Sopenharmony_ci void (*release)(struct kref *arg); 14162306a36Sopenharmony_ci struct efc *efc; 14262306a36Sopenharmony_ci u32 instance_index; 14362306a36Sopenharmony_ci char display_name[EFC_NAME_LENGTH]; 14462306a36Sopenharmony_ci bool is_vport; 14562306a36Sopenharmony_ci bool free_req_pending; 14662306a36Sopenharmony_ci bool attached; 14762306a36Sopenharmony_ci bool attaching; 14862306a36Sopenharmony_ci bool p2p_winner; 14962306a36Sopenharmony_ci struct efc_domain *domain; 15062306a36Sopenharmony_ci u64 wwpn; 15162306a36Sopenharmony_ci u64 wwnn; 15262306a36Sopenharmony_ci void *tgt_data; 15362306a36Sopenharmony_ci void *ini_data; 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci u32 indicator; 15662306a36Sopenharmony_ci u32 fc_id; 15762306a36Sopenharmony_ci struct efc_dma dma; 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci u8 wwnn_str[EFC_WWN_LENGTH]; 16062306a36Sopenharmony_ci __be64 sli_wwpn; 16162306a36Sopenharmony_ci __be64 sli_wwnn; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci struct efc_sm_ctx sm; 16462306a36Sopenharmony_ci struct xarray lookup; 16562306a36Sopenharmony_ci bool enable_ini; 16662306a36Sopenharmony_ci bool enable_tgt; 16762306a36Sopenharmony_ci bool enable_rscn; 16862306a36Sopenharmony_ci bool shutting_down; 16962306a36Sopenharmony_ci u32 p2p_port_id; 17062306a36Sopenharmony_ci enum efc_nport_topology topology; 17162306a36Sopenharmony_ci u8 service_params[EFC_SERVICE_PARMS_LENGTH]; 17262306a36Sopenharmony_ci u32 p2p_remote_port_id; 17362306a36Sopenharmony_ci}; 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci/** 17662306a36Sopenharmony_ci * Fibre Channel domain object 17762306a36Sopenharmony_ci * 17862306a36Sopenharmony_ci * This object is a container for the various SLI components needed 17962306a36Sopenharmony_ci * to connect to the domain of a FC or FCoE switch 18062306a36Sopenharmony_ci * @efc: pointer back to efc 18162306a36Sopenharmony_ci * @instance_index: unique instance index value 18262306a36Sopenharmony_ci * @display_name: Node display name 18362306a36Sopenharmony_ci * @nport_list: linked list of nports associated with this domain 18462306a36Sopenharmony_ci * @ref: Reference count, each nport takes a reference 18562306a36Sopenharmony_ci * @release: Function to free domain object 18662306a36Sopenharmony_ci * @ini_domain: initiator backend private domain data 18762306a36Sopenharmony_ci * @tgt_domain: target backend private domain data 18862306a36Sopenharmony_ci * @sm: state machine context 18962306a36Sopenharmony_ci * @fcf: FC Forwarder table index 19062306a36Sopenharmony_ci * @fcf_indicator: FCFI 19162306a36Sopenharmony_ci * @indicator: VFI 19262306a36Sopenharmony_ci * @nport_count: Number of nports allocated 19362306a36Sopenharmony_ci * @dma: memory for Service Parameters 19462306a36Sopenharmony_ci * @fcf_wwn: WWN for FCF/switch 19562306a36Sopenharmony_ci * @drvsm: driver domain sm context 19662306a36Sopenharmony_ci * @attached: set true after attach completes 19762306a36Sopenharmony_ci * @is_fc: is FC 19862306a36Sopenharmony_ci * @is_loop: is loop topology 19962306a36Sopenharmony_ci * @is_nlport: is public loop 20062306a36Sopenharmony_ci * @domain_found_pending:A domain found is pending, drec is updated 20162306a36Sopenharmony_ci * @req_domain_free: True if domain object should be free'd 20262306a36Sopenharmony_ci * @req_accept_frames: set in domain state machine to enable frames 20362306a36Sopenharmony_ci * @domain_notify_pend: Set in domain SM to avoid duplicate node event post 20462306a36Sopenharmony_ci * @pending_drec: Pending drec if a domain found is pending 20562306a36Sopenharmony_ci * @service_params: any nports service parameters 20662306a36Sopenharmony_ci * @flogi_service_params:Fabric/P2p service parameters from FLOGI 20762306a36Sopenharmony_ci * @lookup: d_id to node lookup object 20862306a36Sopenharmony_ci * @nport: Pointer to first (physical) SLI port 20962306a36Sopenharmony_ci */ 21062306a36Sopenharmony_cistruct efc_domain { 21162306a36Sopenharmony_ci struct efc *efc; 21262306a36Sopenharmony_ci char display_name[EFC_NAME_LENGTH]; 21362306a36Sopenharmony_ci struct list_head nport_list; 21462306a36Sopenharmony_ci struct kref ref; 21562306a36Sopenharmony_ci void (*release)(struct kref *arg); 21662306a36Sopenharmony_ci void *ini_domain; 21762306a36Sopenharmony_ci void *tgt_domain; 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_ci /* Declarations private to HW/SLI */ 22062306a36Sopenharmony_ci u32 fcf; 22162306a36Sopenharmony_ci u32 fcf_indicator; 22262306a36Sopenharmony_ci u32 indicator; 22362306a36Sopenharmony_ci u32 nport_count; 22462306a36Sopenharmony_ci struct efc_dma dma; 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci /* Declarations private to FC trannport */ 22762306a36Sopenharmony_ci u64 fcf_wwn; 22862306a36Sopenharmony_ci struct efc_sm_ctx drvsm; 22962306a36Sopenharmony_ci bool attached; 23062306a36Sopenharmony_ci bool is_fc; 23162306a36Sopenharmony_ci bool is_loop; 23262306a36Sopenharmony_ci bool is_nlport; 23362306a36Sopenharmony_ci bool domain_found_pending; 23462306a36Sopenharmony_ci bool req_domain_free; 23562306a36Sopenharmony_ci bool req_accept_frames; 23662306a36Sopenharmony_ci bool domain_notify_pend; 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_ci struct efc_domain_record pending_drec; 23962306a36Sopenharmony_ci u8 service_params[EFC_SERVICE_PARMS_LENGTH]; 24062306a36Sopenharmony_ci u8 flogi_service_params[EFC_SERVICE_PARMS_LENGTH]; 24162306a36Sopenharmony_ci 24262306a36Sopenharmony_ci struct xarray lookup; 24362306a36Sopenharmony_ci 24462306a36Sopenharmony_ci struct efc_nport *nport; 24562306a36Sopenharmony_ci}; 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_ci/** 24862306a36Sopenharmony_ci * Remote Node object 24962306a36Sopenharmony_ci * 25062306a36Sopenharmony_ci * This object represents a connection between the SLI port and another 25162306a36Sopenharmony_ci * Nx_Port on the fabric. Note this can be either a well known port such 25262306a36Sopenharmony_ci * as a F_Port (i.e. ff:ff:fe) or another N_Port. 25362306a36Sopenharmony_ci * @indicator: RPI 25462306a36Sopenharmony_ci * @fc_id: FC address 25562306a36Sopenharmony_ci * @attached: true if attached 25662306a36Sopenharmony_ci * @nport: associated SLI port 25762306a36Sopenharmony_ci * @node: associated node 25862306a36Sopenharmony_ci */ 25962306a36Sopenharmony_cistruct efc_remote_node { 26062306a36Sopenharmony_ci u32 indicator; 26162306a36Sopenharmony_ci u32 index; 26262306a36Sopenharmony_ci u32 fc_id; 26362306a36Sopenharmony_ci 26462306a36Sopenharmony_ci bool attached; 26562306a36Sopenharmony_ci 26662306a36Sopenharmony_ci struct efc_nport *nport; 26762306a36Sopenharmony_ci void *node; 26862306a36Sopenharmony_ci}; 26962306a36Sopenharmony_ci 27062306a36Sopenharmony_ci/** 27162306a36Sopenharmony_ci * FC Node object 27262306a36Sopenharmony_ci * @efc: pointer back to efc structure 27362306a36Sopenharmony_ci * @display_name: Node display name 27462306a36Sopenharmony_ci * @nort: Assosiated nport pointer. 27562306a36Sopenharmony_ci * @hold_frames: hold incoming frames if true 27662306a36Sopenharmony_ci * @els_io_enabled: Enable allocating els ios for this node 27762306a36Sopenharmony_ci * @els_ios_lock: lock to protect the els ios list 27862306a36Sopenharmony_ci * @els_ios_list: ELS I/O's for this node 27962306a36Sopenharmony_ci * @ini_node: backend initiator private node data 28062306a36Sopenharmony_ci * @tgt_node: backend target private node data 28162306a36Sopenharmony_ci * @rnode: Remote node 28262306a36Sopenharmony_ci * @sm: state machine context 28362306a36Sopenharmony_ci * @evtdepth: current event posting nesting depth 28462306a36Sopenharmony_ci * @req_free: this node is to be free'd 28562306a36Sopenharmony_ci * @attached: node is attached (REGLOGIN complete) 28662306a36Sopenharmony_ci * @fcp_enabled: node is enabled to handle FCP 28762306a36Sopenharmony_ci * @rscn_pending: for name server node RSCN is pending 28862306a36Sopenharmony_ci * @send_plogi: send PLOGI accept, upon completion of node attach 28962306a36Sopenharmony_ci * @send_plogi_acc: TRUE if io_alloc() is enabled. 29062306a36Sopenharmony_ci * @send_ls_acc: type of LS acc to send 29162306a36Sopenharmony_ci * @ls_acc_io: SCSI IO for LS acc 29262306a36Sopenharmony_ci * @ls_acc_oxid: OX_ID for pending accept 29362306a36Sopenharmony_ci * @ls_acc_did: D_ID for pending accept 29462306a36Sopenharmony_ci * @shutdown_reason: reason for node shutdown 29562306a36Sopenharmony_ci * @sparm_dma_buf: service parameters buffer 29662306a36Sopenharmony_ci * @service_params: plogi/acc frame from remote device 29762306a36Sopenharmony_ci * @pend_frames_lock: lock for inbound pending frames list 29862306a36Sopenharmony_ci * @pend_frames: inbound pending frames list 29962306a36Sopenharmony_ci * @pend_frames_processed:count of frames processed in hold frames interval 30062306a36Sopenharmony_ci * @ox_id_in_use: used to verify one at a time us of ox_id 30162306a36Sopenharmony_ci * @els_retries_remaining:for ELS, number of retries remaining 30262306a36Sopenharmony_ci * @els_req_cnt: number of outstanding ELS requests 30362306a36Sopenharmony_ci * @els_cmpl_cnt: number of outstanding ELS completions 30462306a36Sopenharmony_ci * @abort_cnt: Abort counter for debugging purpos 30562306a36Sopenharmony_ci * @current_state_name: current node state 30662306a36Sopenharmony_ci * @prev_state_name: previous node state 30762306a36Sopenharmony_ci * @current_evt: current event 30862306a36Sopenharmony_ci * @prev_evt: previous event 30962306a36Sopenharmony_ci * @targ: node is target capable 31062306a36Sopenharmony_ci * @init: node is init capable 31162306a36Sopenharmony_ci * @refound: Handle node refound case when node is being deleted 31262306a36Sopenharmony_ci * @els_io_pend_list: list of pending (not yet processed) ELS IOs 31362306a36Sopenharmony_ci * @els_io_active_list: list of active (processed) ELS IOs 31462306a36Sopenharmony_ci * @nodedb_state: Node debugging, saved state 31562306a36Sopenharmony_ci * @gidpt_delay_timer: GIDPT delay timer 31662306a36Sopenharmony_ci * @time_last_gidpt_msec:Start time of last target RSCN GIDPT 31762306a36Sopenharmony_ci * @wwnn: remote port WWNN 31862306a36Sopenharmony_ci * @wwpn: remote port WWPN 31962306a36Sopenharmony_ci */ 32062306a36Sopenharmony_cistruct efc_node { 32162306a36Sopenharmony_ci struct efc *efc; 32262306a36Sopenharmony_ci char display_name[EFC_NAME_LENGTH]; 32362306a36Sopenharmony_ci struct efc_nport *nport; 32462306a36Sopenharmony_ci struct kref ref; 32562306a36Sopenharmony_ci void (*release)(struct kref *arg); 32662306a36Sopenharmony_ci bool hold_frames; 32762306a36Sopenharmony_ci bool els_io_enabled; 32862306a36Sopenharmony_ci bool send_plogi_acc; 32962306a36Sopenharmony_ci bool send_plogi; 33062306a36Sopenharmony_ci bool rscn_pending; 33162306a36Sopenharmony_ci bool fcp_enabled; 33262306a36Sopenharmony_ci bool attached; 33362306a36Sopenharmony_ci bool req_free; 33462306a36Sopenharmony_ci 33562306a36Sopenharmony_ci spinlock_t els_ios_lock; 33662306a36Sopenharmony_ci struct list_head els_ios_list; 33762306a36Sopenharmony_ci void *ini_node; 33862306a36Sopenharmony_ci void *tgt_node; 33962306a36Sopenharmony_ci 34062306a36Sopenharmony_ci struct efc_remote_node rnode; 34162306a36Sopenharmony_ci /* Declarations private to FC trannport */ 34262306a36Sopenharmony_ci struct efc_sm_ctx sm; 34362306a36Sopenharmony_ci u32 evtdepth; 34462306a36Sopenharmony_ci 34562306a36Sopenharmony_ci enum efc_node_send_ls_acc send_ls_acc; 34662306a36Sopenharmony_ci void *ls_acc_io; 34762306a36Sopenharmony_ci u32 ls_acc_oxid; 34862306a36Sopenharmony_ci u32 ls_acc_did; 34962306a36Sopenharmony_ci enum efc_node_shutd_rsn shutdown_reason; 35062306a36Sopenharmony_ci bool targ; 35162306a36Sopenharmony_ci bool init; 35262306a36Sopenharmony_ci bool refound; 35362306a36Sopenharmony_ci struct efc_dma sparm_dma_buf; 35462306a36Sopenharmony_ci u8 service_params[EFC_SERVICE_PARMS_LENGTH]; 35562306a36Sopenharmony_ci spinlock_t pend_frames_lock; 35662306a36Sopenharmony_ci struct list_head pend_frames; 35762306a36Sopenharmony_ci u32 pend_frames_processed; 35862306a36Sopenharmony_ci u32 ox_id_in_use; 35962306a36Sopenharmony_ci u32 els_retries_remaining; 36062306a36Sopenharmony_ci u32 els_req_cnt; 36162306a36Sopenharmony_ci u32 els_cmpl_cnt; 36262306a36Sopenharmony_ci u32 abort_cnt; 36362306a36Sopenharmony_ci 36462306a36Sopenharmony_ci char current_state_name[EFC_SM_NAME_LENGTH]; 36562306a36Sopenharmony_ci char prev_state_name[EFC_SM_NAME_LENGTH]; 36662306a36Sopenharmony_ci int current_evt; 36762306a36Sopenharmony_ci int prev_evt; 36862306a36Sopenharmony_ci 36962306a36Sopenharmony_ci void (*nodedb_state)(struct efc_sm_ctx *ctx, 37062306a36Sopenharmony_ci enum efc_sm_event evt, void *arg); 37162306a36Sopenharmony_ci struct timer_list gidpt_delay_timer; 37262306a36Sopenharmony_ci u64 time_last_gidpt_msec; 37362306a36Sopenharmony_ci 37462306a36Sopenharmony_ci char wwnn[EFC_WWN_LENGTH]; 37562306a36Sopenharmony_ci char wwpn[EFC_WWN_LENGTH]; 37662306a36Sopenharmony_ci}; 37762306a36Sopenharmony_ci 37862306a36Sopenharmony_ci/** 37962306a36Sopenharmony_ci * NPIV port 38062306a36Sopenharmony_ci * 38162306a36Sopenharmony_ci * Collection of the information required to restore a virtual port across 38262306a36Sopenharmony_ci * link events 38362306a36Sopenharmony_ci * @wwnn: node name 38462306a36Sopenharmony_ci * @wwpn: port name 38562306a36Sopenharmony_ci * @fc_id: port id 38662306a36Sopenharmony_ci * @tgt_data: target backend pointer 38762306a36Sopenharmony_ci * @ini_data: initiator backend pointe 38862306a36Sopenharmony_ci * @nport: Used to match record after attaching for update 38962306a36Sopenharmony_ci * 39062306a36Sopenharmony_ci */ 39162306a36Sopenharmony_ci 39262306a36Sopenharmony_cistruct efc_vport { 39362306a36Sopenharmony_ci struct list_head list_entry; 39462306a36Sopenharmony_ci u64 wwnn; 39562306a36Sopenharmony_ci u64 wwpn; 39662306a36Sopenharmony_ci u32 fc_id; 39762306a36Sopenharmony_ci bool enable_tgt; 39862306a36Sopenharmony_ci bool enable_ini; 39962306a36Sopenharmony_ci void *tgt_data; 40062306a36Sopenharmony_ci void *ini_data; 40162306a36Sopenharmony_ci struct efc_nport *nport; 40262306a36Sopenharmony_ci}; 40362306a36Sopenharmony_ci 40462306a36Sopenharmony_ci#define node_printf(node, fmt, args...) \ 40562306a36Sopenharmony_ci efc_log_info(node->efc, "[%s] " fmt, node->display_name, ##args) 40662306a36Sopenharmony_ci 40762306a36Sopenharmony_ci/* Node SM IO Context Callback structure */ 40862306a36Sopenharmony_cistruct efc_node_cb { 40962306a36Sopenharmony_ci int status; 41062306a36Sopenharmony_ci int ext_status; 41162306a36Sopenharmony_ci struct efc_hw_rq_buffer *header; 41262306a36Sopenharmony_ci struct efc_hw_rq_buffer *payload; 41362306a36Sopenharmony_ci struct efc_dma els_rsp; 41462306a36Sopenharmony_ci 41562306a36Sopenharmony_ci /* Actual length of data received */ 41662306a36Sopenharmony_ci int rsp_len; 41762306a36Sopenharmony_ci}; 41862306a36Sopenharmony_ci 41962306a36Sopenharmony_cistruct efc_hw_rq_buffer { 42062306a36Sopenharmony_ci u16 rqindex; 42162306a36Sopenharmony_ci struct efc_dma dma; 42262306a36Sopenharmony_ci}; 42362306a36Sopenharmony_ci 42462306a36Sopenharmony_ci/** 42562306a36Sopenharmony_ci * FC sequence object 42662306a36Sopenharmony_ci * 42762306a36Sopenharmony_ci * Defines a general FC sequence object 42862306a36Sopenharmony_ci * @hw: HW that owns this sequence 42962306a36Sopenharmony_ci * @fcfi: FCFI associated with sequence 43062306a36Sopenharmony_ci * @header: Received frame header 43162306a36Sopenharmony_ci * @payload: Received frame header 43262306a36Sopenharmony_ci * @hw_priv: HW private context 43362306a36Sopenharmony_ci */ 43462306a36Sopenharmony_cistruct efc_hw_sequence { 43562306a36Sopenharmony_ci struct list_head list_entry; 43662306a36Sopenharmony_ci void *hw; 43762306a36Sopenharmony_ci u8 fcfi; 43862306a36Sopenharmony_ci struct efc_hw_rq_buffer *header; 43962306a36Sopenharmony_ci struct efc_hw_rq_buffer *payload; 44062306a36Sopenharmony_ci void *hw_priv; 44162306a36Sopenharmony_ci}; 44262306a36Sopenharmony_ci 44362306a36Sopenharmony_cienum efc_disc_io_type { 44462306a36Sopenharmony_ci EFC_DISC_IO_ELS_REQ, 44562306a36Sopenharmony_ci EFC_DISC_IO_ELS_RESP, 44662306a36Sopenharmony_ci EFC_DISC_IO_CT_REQ, 44762306a36Sopenharmony_ci EFC_DISC_IO_CT_RESP 44862306a36Sopenharmony_ci}; 44962306a36Sopenharmony_ci 45062306a36Sopenharmony_cistruct efc_io_els_params { 45162306a36Sopenharmony_ci u32 s_id; 45262306a36Sopenharmony_ci u16 ox_id; 45362306a36Sopenharmony_ci u8 timeout; 45462306a36Sopenharmony_ci}; 45562306a36Sopenharmony_ci 45662306a36Sopenharmony_cistruct efc_io_ct_params { 45762306a36Sopenharmony_ci u8 r_ctl; 45862306a36Sopenharmony_ci u8 type; 45962306a36Sopenharmony_ci u8 df_ctl; 46062306a36Sopenharmony_ci u8 timeout; 46162306a36Sopenharmony_ci u16 ox_id; 46262306a36Sopenharmony_ci}; 46362306a36Sopenharmony_ci 46462306a36Sopenharmony_ciunion efc_disc_io_param { 46562306a36Sopenharmony_ci struct efc_io_els_params els; 46662306a36Sopenharmony_ci struct efc_io_ct_params ct; 46762306a36Sopenharmony_ci}; 46862306a36Sopenharmony_ci 46962306a36Sopenharmony_cistruct efc_disc_io { 47062306a36Sopenharmony_ci struct efc_dma req; /* send buffer */ 47162306a36Sopenharmony_ci struct efc_dma rsp; /* receive buffer */ 47262306a36Sopenharmony_ci enum efc_disc_io_type io_type; /* EFC_DISC_IO_TYPE enum*/ 47362306a36Sopenharmony_ci u16 xmit_len; /* Length of els request*/ 47462306a36Sopenharmony_ci u16 rsp_len; /* Max length of rsps to be rcvd */ 47562306a36Sopenharmony_ci u32 rpi; /* Registered RPI */ 47662306a36Sopenharmony_ci u32 vpi; /* VPI for this nport */ 47762306a36Sopenharmony_ci u32 s_id; 47862306a36Sopenharmony_ci u32 d_id; 47962306a36Sopenharmony_ci bool rpi_registered; /* if false, use tmp RPI */ 48062306a36Sopenharmony_ci union efc_disc_io_param iparam; 48162306a36Sopenharmony_ci}; 48262306a36Sopenharmony_ci 48362306a36Sopenharmony_ci/* Return value indiacating the sequence can not be freed */ 48462306a36Sopenharmony_ci#define EFC_HW_SEQ_HOLD 0 48562306a36Sopenharmony_ci/* Return value indiacating the sequence can be freed */ 48662306a36Sopenharmony_ci#define EFC_HW_SEQ_FREE 1 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_cistruct libefc_function_template { 48962306a36Sopenharmony_ci /*Sport*/ 49062306a36Sopenharmony_ci int (*new_nport)(struct efc *efc, struct efc_nport *sp); 49162306a36Sopenharmony_ci void (*del_nport)(struct efc *efc, struct efc_nport *sp); 49262306a36Sopenharmony_ci 49362306a36Sopenharmony_ci /*Scsi Node*/ 49462306a36Sopenharmony_ci int (*scsi_new_node)(struct efc *efc, struct efc_node *n); 49562306a36Sopenharmony_ci int (*scsi_del_node)(struct efc *efc, struct efc_node *n, int reason); 49662306a36Sopenharmony_ci 49762306a36Sopenharmony_ci int (*issue_mbox_rqst)(void *efct, void *buf, void *cb, void *arg); 49862306a36Sopenharmony_ci /*Send ELS IO*/ 49962306a36Sopenharmony_ci int (*send_els)(struct efc *efc, struct efc_disc_io *io); 50062306a36Sopenharmony_ci /*Send BLS IO*/ 50162306a36Sopenharmony_ci int (*send_bls)(struct efc *efc, u32 type, struct sli_bls_params *bls); 50262306a36Sopenharmony_ci /*Free HW frame*/ 50362306a36Sopenharmony_ci int (*hw_seq_free)(struct efc *efc, struct efc_hw_sequence *seq); 50462306a36Sopenharmony_ci}; 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_ci#define EFC_LOG_LIB 0x01 50762306a36Sopenharmony_ci#define EFC_LOG_NODE 0x02 50862306a36Sopenharmony_ci#define EFC_LOG_PORT 0x04 50962306a36Sopenharmony_ci#define EFC_LOG_DOMAIN 0x08 51062306a36Sopenharmony_ci#define EFC_LOG_ELS 0x10 51162306a36Sopenharmony_ci#define EFC_LOG_DOMAIN_SM 0x20 51262306a36Sopenharmony_ci#define EFC_LOG_SM 0x40 51362306a36Sopenharmony_ci 51462306a36Sopenharmony_ci/* efc library port structure */ 51562306a36Sopenharmony_cistruct efc { 51662306a36Sopenharmony_ci void *base; 51762306a36Sopenharmony_ci struct pci_dev *pci; 51862306a36Sopenharmony_ci struct sli4 *sli; 51962306a36Sopenharmony_ci u32 fcfi; 52062306a36Sopenharmony_ci u64 req_wwpn; 52162306a36Sopenharmony_ci u64 req_wwnn; 52262306a36Sopenharmony_ci 52362306a36Sopenharmony_ci u64 def_wwpn; 52462306a36Sopenharmony_ci u64 def_wwnn; 52562306a36Sopenharmony_ci u64 max_xfer_size; 52662306a36Sopenharmony_ci mempool_t *node_pool; 52762306a36Sopenharmony_ci struct dma_pool *node_dma_pool; 52862306a36Sopenharmony_ci u32 nodes_count; 52962306a36Sopenharmony_ci 53062306a36Sopenharmony_ci u32 link_status; 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_ci struct list_head vport_list; 53362306a36Sopenharmony_ci /* lock to protect the vport list */ 53462306a36Sopenharmony_ci spinlock_t vport_lock; 53562306a36Sopenharmony_ci 53662306a36Sopenharmony_ci struct libefc_function_template tt; 53762306a36Sopenharmony_ci /* lock to protect the discovery library. 53862306a36Sopenharmony_ci * Refer to efclib.c for more details. 53962306a36Sopenharmony_ci */ 54062306a36Sopenharmony_ci spinlock_t lock; 54162306a36Sopenharmony_ci 54262306a36Sopenharmony_ci bool enable_ini; 54362306a36Sopenharmony_ci bool enable_tgt; 54462306a36Sopenharmony_ci 54562306a36Sopenharmony_ci u32 log_level; 54662306a36Sopenharmony_ci 54762306a36Sopenharmony_ci struct efc_domain *domain; 54862306a36Sopenharmony_ci void (*domain_free_cb)(struct efc *efc, void *arg); 54962306a36Sopenharmony_ci void *domain_free_cb_arg; 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_ci u64 tgt_rscn_delay_msec; 55262306a36Sopenharmony_ci u64 tgt_rscn_period_msec; 55362306a36Sopenharmony_ci 55462306a36Sopenharmony_ci bool external_loopback; 55562306a36Sopenharmony_ci u32 nodedb_mask; 55662306a36Sopenharmony_ci u32 logmask; 55762306a36Sopenharmony_ci mempool_t *els_io_pool; 55862306a36Sopenharmony_ci atomic_t els_io_alloc_failed_count; 55962306a36Sopenharmony_ci 56062306a36Sopenharmony_ci /* hold pending frames */ 56162306a36Sopenharmony_ci bool hold_frames; 56262306a36Sopenharmony_ci /* lock to protect pending frames list access */ 56362306a36Sopenharmony_ci spinlock_t pend_frames_lock; 56462306a36Sopenharmony_ci struct list_head pend_frames; 56562306a36Sopenharmony_ci /* count of pending frames that were processed */ 56662306a36Sopenharmony_ci u32 pend_frames_processed; 56762306a36Sopenharmony_ci 56862306a36Sopenharmony_ci}; 56962306a36Sopenharmony_ci 57062306a36Sopenharmony_ci/* 57162306a36Sopenharmony_ci * EFC library registration 57262306a36Sopenharmony_ci * **********************************/ 57362306a36Sopenharmony_ciint efcport_init(struct efc *efc); 57462306a36Sopenharmony_civoid efcport_destroy(struct efc *efc); 57562306a36Sopenharmony_ci/* 57662306a36Sopenharmony_ci * EFC Domain 57762306a36Sopenharmony_ci * **********************************/ 57862306a36Sopenharmony_ciint efc_domain_cb(void *arg, int event, void *data); 57962306a36Sopenharmony_civoid 58062306a36Sopenharmony_ciefc_register_domain_free_cb(struct efc *efc, 58162306a36Sopenharmony_ci void (*callback)(struct efc *efc, void *arg), 58262306a36Sopenharmony_ci void *arg); 58362306a36Sopenharmony_ci 58462306a36Sopenharmony_ci/* 58562306a36Sopenharmony_ci * EFC nport 58662306a36Sopenharmony_ci * **********************************/ 58762306a36Sopenharmony_civoid efc_nport_cb(void *arg, int event, void *data); 58862306a36Sopenharmony_cistruct efc_vport * 58962306a36Sopenharmony_ciefc_vport_create_spec(struct efc *efc, u64 wwnn, u64 wwpn, u32 fc_id, 59062306a36Sopenharmony_ci bool enable_ini, bool enable_tgt, 59162306a36Sopenharmony_ci void *tgt_data, void *ini_data); 59262306a36Sopenharmony_ciint efc_nport_vport_new(struct efc_domain *domain, u64 wwpn, 59362306a36Sopenharmony_ci u64 wwnn, u32 fc_id, bool ini, bool tgt, 59462306a36Sopenharmony_ci void *tgt_data, void *ini_data); 59562306a36Sopenharmony_ciint efc_nport_vport_del(struct efc *efc, struct efc_domain *domain, 59662306a36Sopenharmony_ci u64 wwpn, u64 wwnn); 59762306a36Sopenharmony_ci 59862306a36Sopenharmony_civoid efc_vport_del_all(struct efc *efc); 59962306a36Sopenharmony_ci 60062306a36Sopenharmony_ci/* 60162306a36Sopenharmony_ci * EFC Node 60262306a36Sopenharmony_ci * **********************************/ 60362306a36Sopenharmony_ciint efc_remote_node_cb(void *arg, int event, void *data); 60462306a36Sopenharmony_civoid efc_node_fcid_display(u32 fc_id, char *buffer, u32 buf_len); 60562306a36Sopenharmony_civoid efc_node_post_shutdown(struct efc_node *node, void *arg); 60662306a36Sopenharmony_ciu64 efc_node_get_wwpn(struct efc_node *node); 60762306a36Sopenharmony_ci 60862306a36Sopenharmony_ci/* 60962306a36Sopenharmony_ci * EFC FCP/ELS/CT interface 61062306a36Sopenharmony_ci * **********************************/ 61162306a36Sopenharmony_civoid efc_dispatch_frame(struct efc *efc, struct efc_hw_sequence *seq); 61262306a36Sopenharmony_civoid efc_disc_io_complete(struct efc_disc_io *io, u32 len, u32 status, 61362306a36Sopenharmony_ci u32 ext_status); 61462306a36Sopenharmony_ci 61562306a36Sopenharmony_ci/* 61662306a36Sopenharmony_ci * EFC SCSI INTERACTION LAYER 61762306a36Sopenharmony_ci * **********************************/ 61862306a36Sopenharmony_civoid efc_scsi_sess_reg_complete(struct efc_node *node, u32 status); 61962306a36Sopenharmony_civoid efc_scsi_del_initiator_complete(struct efc *efc, struct efc_node *node); 62062306a36Sopenharmony_civoid efc_scsi_del_target_complete(struct efc *efc, struct efc_node *node); 62162306a36Sopenharmony_civoid efc_scsi_io_list_empty(struct efc *efc, struct efc_node *node); 62262306a36Sopenharmony_ci 62362306a36Sopenharmony_ci#endif /* __EFCLIB_H__ */ 624