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