|  | /* 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			attaching; | 
|  | 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__ */ |