xref: /OK3568_Linux_fs/kernel/drivers/s390/scsi/zfcp_fc.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0 */
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * zfcp device driver
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Fibre Channel related definitions and inline functions for the zfcp
6*4882a593Smuzhiyun  * device driver
7*4882a593Smuzhiyun  *
8*4882a593Smuzhiyun  * Copyright IBM Corp. 2009, 2017
9*4882a593Smuzhiyun  */
10*4882a593Smuzhiyun 
11*4882a593Smuzhiyun #ifndef ZFCP_FC_H
12*4882a593Smuzhiyun #define ZFCP_FC_H
13*4882a593Smuzhiyun 
14*4882a593Smuzhiyun #include <scsi/fc/fc_els.h>
15*4882a593Smuzhiyun #include <scsi/fc/fc_fcp.h>
16*4882a593Smuzhiyun #include <scsi/fc/fc_ns.h>
17*4882a593Smuzhiyun #include <scsi/scsi_cmnd.h>
18*4882a593Smuzhiyun #include <scsi/scsi_tcq.h>
19*4882a593Smuzhiyun #include "zfcp_fsf.h"
20*4882a593Smuzhiyun 
21*4882a593Smuzhiyun #define ZFCP_FC_CT_SIZE_PAGE	  (PAGE_SIZE - sizeof(struct fc_ct_hdr))
22*4882a593Smuzhiyun #define ZFCP_FC_GPN_FT_ENT_PAGE	  (ZFCP_FC_CT_SIZE_PAGE \
23*4882a593Smuzhiyun 					/ sizeof(struct fc_gpn_ft_resp))
24*4882a593Smuzhiyun #define ZFCP_FC_GPN_FT_NUM_BUFS	  4 /* memory pages */
25*4882a593Smuzhiyun 
26*4882a593Smuzhiyun #define ZFCP_FC_GPN_FT_MAX_SIZE	  (ZFCP_FC_GPN_FT_NUM_BUFS * PAGE_SIZE \
27*4882a593Smuzhiyun 					- sizeof(struct fc_ct_hdr))
28*4882a593Smuzhiyun #define ZFCP_FC_GPN_FT_MAX_ENT	  (ZFCP_FC_GPN_FT_NUM_BUFS * \
29*4882a593Smuzhiyun 					(ZFCP_FC_GPN_FT_ENT_PAGE + 1))
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun #define ZFCP_FC_CTELS_TMO	(2 * FC_DEF_R_A_TOV / 1000)
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun /**
34*4882a593Smuzhiyun  * struct zfcp_fc_event - FC HBAAPI event for internal queueing from irq context
35*4882a593Smuzhiyun  * @code: Event code
36*4882a593Smuzhiyun  * @data: Event data
37*4882a593Smuzhiyun  * @list: list_head for zfcp_fc_events list
38*4882a593Smuzhiyun  */
39*4882a593Smuzhiyun struct zfcp_fc_event {
40*4882a593Smuzhiyun 	enum fc_host_event_code code;
41*4882a593Smuzhiyun 	u32 data;
42*4882a593Smuzhiyun 	struct list_head list;
43*4882a593Smuzhiyun };
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun /**
46*4882a593Smuzhiyun  * struct zfcp_fc_events - Infrastructure for posting FC events from irq context
47*4882a593Smuzhiyun  * @list: List for queueing of events from irq context to workqueue
48*4882a593Smuzhiyun  * @list_lock: Lock for event list
49*4882a593Smuzhiyun  * @work: work_struct for forwarding events in workqueue
50*4882a593Smuzhiyun */
51*4882a593Smuzhiyun struct zfcp_fc_events {
52*4882a593Smuzhiyun 	struct list_head list;
53*4882a593Smuzhiyun 	spinlock_t list_lock;
54*4882a593Smuzhiyun 	struct work_struct work;
55*4882a593Smuzhiyun };
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun /**
58*4882a593Smuzhiyun  * struct zfcp_fc_gid_pn_req - container for ct header plus gid_pn request
59*4882a593Smuzhiyun  * @ct_hdr: FC GS common transport header
60*4882a593Smuzhiyun  * @gid_pn: GID_PN request
61*4882a593Smuzhiyun  */
62*4882a593Smuzhiyun struct zfcp_fc_gid_pn_req {
63*4882a593Smuzhiyun 	struct fc_ct_hdr	ct_hdr;
64*4882a593Smuzhiyun 	struct fc_ns_gid_pn	gid_pn;
65*4882a593Smuzhiyun } __packed;
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun /**
68*4882a593Smuzhiyun  * struct zfcp_fc_gid_pn_rsp - container for ct header plus gid_pn response
69*4882a593Smuzhiyun  * @ct_hdr: FC GS common transport header
70*4882a593Smuzhiyun  * @gid_pn: GID_PN response
71*4882a593Smuzhiyun  */
72*4882a593Smuzhiyun struct zfcp_fc_gid_pn_rsp {
73*4882a593Smuzhiyun 	struct fc_ct_hdr	ct_hdr;
74*4882a593Smuzhiyun 	struct fc_gid_pn_resp	gid_pn;
75*4882a593Smuzhiyun } __packed;
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun /**
78*4882a593Smuzhiyun  * struct zfcp_fc_gpn_ft - container for ct header plus gpn_ft request
79*4882a593Smuzhiyun  * @ct_hdr: FC GS common transport header
80*4882a593Smuzhiyun  * @gpn_ft: GPN_FT request
81*4882a593Smuzhiyun  */
82*4882a593Smuzhiyun struct zfcp_fc_gpn_ft_req {
83*4882a593Smuzhiyun 	struct fc_ct_hdr	ct_hdr;
84*4882a593Smuzhiyun 	struct fc_ns_gid_ft	gpn_ft;
85*4882a593Smuzhiyun } __packed;
86*4882a593Smuzhiyun 
87*4882a593Smuzhiyun /**
88*4882a593Smuzhiyun  * struct zfcp_fc_gspn_req - container for ct header plus GSPN_ID request
89*4882a593Smuzhiyun  * @ct_hdr: FC GS common transport header
90*4882a593Smuzhiyun  * @gspn: GSPN_ID request
91*4882a593Smuzhiyun  */
92*4882a593Smuzhiyun struct zfcp_fc_gspn_req {
93*4882a593Smuzhiyun 	struct fc_ct_hdr	ct_hdr;
94*4882a593Smuzhiyun 	struct fc_gid_pn_resp	gspn;
95*4882a593Smuzhiyun } __packed;
96*4882a593Smuzhiyun 
97*4882a593Smuzhiyun /**
98*4882a593Smuzhiyun  * struct zfcp_fc_gspn_rsp - container for ct header plus GSPN_ID response
99*4882a593Smuzhiyun  * @ct_hdr: FC GS common transport header
100*4882a593Smuzhiyun  * @gspn: GSPN_ID response
101*4882a593Smuzhiyun  * @name: The name string of the GSPN_ID response
102*4882a593Smuzhiyun  */
103*4882a593Smuzhiyun struct zfcp_fc_gspn_rsp {
104*4882a593Smuzhiyun 	struct fc_ct_hdr	ct_hdr;
105*4882a593Smuzhiyun 	struct fc_gspn_resp	gspn;
106*4882a593Smuzhiyun 	char			name[FC_SYMBOLIC_NAME_SIZE];
107*4882a593Smuzhiyun } __packed;
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun /**
110*4882a593Smuzhiyun  * struct zfcp_fc_rspn_req - container for ct header plus RSPN_ID request
111*4882a593Smuzhiyun  * @ct_hdr: FC GS common transport header
112*4882a593Smuzhiyun  * @rspn: RSPN_ID request
113*4882a593Smuzhiyun  * @name: The name string of the RSPN_ID request
114*4882a593Smuzhiyun  */
115*4882a593Smuzhiyun struct zfcp_fc_rspn_req {
116*4882a593Smuzhiyun 	struct fc_ct_hdr	ct_hdr;
117*4882a593Smuzhiyun 	struct fc_ns_rspn	rspn;
118*4882a593Smuzhiyun 	char			name[FC_SYMBOLIC_NAME_SIZE];
119*4882a593Smuzhiyun } __packed;
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun /**
122*4882a593Smuzhiyun  * struct zfcp_fc_req - Container for FC ELS and CT requests sent from zfcp
123*4882a593Smuzhiyun  * @ct_els: data required for issuing fsf command
124*4882a593Smuzhiyun  * @sg_req: scatterlist entry for request data, refers to embedded @u submember
125*4882a593Smuzhiyun  * @sg_rsp: scatterlist entry for response data, refers to embedded @u submember
126*4882a593Smuzhiyun  * @u: request and response specific data
127*4882a593Smuzhiyun  * @u.adisc: ADISC specific data
128*4882a593Smuzhiyun  * @u.adisc.req: ADISC request
129*4882a593Smuzhiyun  * @u.adisc.rsp: ADISC response
130*4882a593Smuzhiyun  * @u.gid_pn: GID_PN specific data
131*4882a593Smuzhiyun  * @u.gid_pn.req: GID_PN request
132*4882a593Smuzhiyun  * @u.gid_pn.rsp: GID_PN response
133*4882a593Smuzhiyun  * @u.gpn_ft: GPN_FT specific data
134*4882a593Smuzhiyun  * @u.gpn_ft.sg_rsp2: GPN_FT response, not embedded here, allocated elsewhere
135*4882a593Smuzhiyun  * @u.gpn_ft.req: GPN_FT request
136*4882a593Smuzhiyun  * @u.gspn: GSPN specific data
137*4882a593Smuzhiyun  * @u.gspn.req: GSPN request
138*4882a593Smuzhiyun  * @u.gspn.rsp: GSPN response
139*4882a593Smuzhiyun  * @u.rspn: RSPN specific data
140*4882a593Smuzhiyun  * @u.rspn.req: RSPN request
141*4882a593Smuzhiyun  * @u.rspn.rsp: RSPN response
142*4882a593Smuzhiyun  */
143*4882a593Smuzhiyun struct zfcp_fc_req {
144*4882a593Smuzhiyun 	struct zfcp_fsf_ct_els				ct_els;
145*4882a593Smuzhiyun 	struct scatterlist				sg_req;
146*4882a593Smuzhiyun 	struct scatterlist				sg_rsp;
147*4882a593Smuzhiyun 	union {
148*4882a593Smuzhiyun 		struct {
149*4882a593Smuzhiyun 			struct fc_els_adisc		req;
150*4882a593Smuzhiyun 			struct fc_els_adisc		rsp;
151*4882a593Smuzhiyun 		} adisc;
152*4882a593Smuzhiyun 		struct {
153*4882a593Smuzhiyun 			struct zfcp_fc_gid_pn_req	req;
154*4882a593Smuzhiyun 			struct zfcp_fc_gid_pn_rsp	rsp;
155*4882a593Smuzhiyun 		} gid_pn;
156*4882a593Smuzhiyun 		struct {
157*4882a593Smuzhiyun 			struct scatterlist sg_rsp2[ZFCP_FC_GPN_FT_NUM_BUFS - 1];
158*4882a593Smuzhiyun 			struct zfcp_fc_gpn_ft_req	req;
159*4882a593Smuzhiyun 		} gpn_ft;
160*4882a593Smuzhiyun 		struct {
161*4882a593Smuzhiyun 			struct zfcp_fc_gspn_req		req;
162*4882a593Smuzhiyun 			struct zfcp_fc_gspn_rsp		rsp;
163*4882a593Smuzhiyun 		} gspn;
164*4882a593Smuzhiyun 		struct {
165*4882a593Smuzhiyun 			struct zfcp_fc_rspn_req		req;
166*4882a593Smuzhiyun 			struct fc_ct_hdr		rsp;
167*4882a593Smuzhiyun 		} rspn;
168*4882a593Smuzhiyun 	} u;
169*4882a593Smuzhiyun };
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun /**
172*4882a593Smuzhiyun  * enum zfcp_fc_wka_status - FC WKA port status in zfcp
173*4882a593Smuzhiyun  * @ZFCP_FC_WKA_PORT_OFFLINE: Port is closed and not in use
174*4882a593Smuzhiyun  * @ZFCP_FC_WKA_PORT_CLOSING: The FSF "close port" request is pending
175*4882a593Smuzhiyun  * @ZFCP_FC_WKA_PORT_OPENING: The FSF "open port" request is pending
176*4882a593Smuzhiyun  * @ZFCP_FC_WKA_PORT_ONLINE: The port is open and the port handle is valid
177*4882a593Smuzhiyun  */
178*4882a593Smuzhiyun enum zfcp_fc_wka_status {
179*4882a593Smuzhiyun 	ZFCP_FC_WKA_PORT_OFFLINE,
180*4882a593Smuzhiyun 	ZFCP_FC_WKA_PORT_CLOSING,
181*4882a593Smuzhiyun 	ZFCP_FC_WKA_PORT_OPENING,
182*4882a593Smuzhiyun 	ZFCP_FC_WKA_PORT_ONLINE,
183*4882a593Smuzhiyun };
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun /**
186*4882a593Smuzhiyun  * struct zfcp_fc_wka_port - representation of well-known-address (WKA) FC port
187*4882a593Smuzhiyun  * @adapter: Pointer to adapter structure this WKA port belongs to
188*4882a593Smuzhiyun  * @opened: Wait for completion of open command
189*4882a593Smuzhiyun  * @closed: Wait for completion of close command
190*4882a593Smuzhiyun  * @status: Current status of WKA port
191*4882a593Smuzhiyun  * @refcount: Reference count to keep port open as long as it is in use
192*4882a593Smuzhiyun  * @d_id: FC destination id or well-known-address
193*4882a593Smuzhiyun  * @handle: FSF handle for the open WKA port
194*4882a593Smuzhiyun  * @mutex: Mutex used during opening/closing state changes
195*4882a593Smuzhiyun  * @work: For delaying the closing of the WKA port
196*4882a593Smuzhiyun  */
197*4882a593Smuzhiyun struct zfcp_fc_wka_port {
198*4882a593Smuzhiyun 	struct zfcp_adapter	*adapter;
199*4882a593Smuzhiyun 	wait_queue_head_t	opened;
200*4882a593Smuzhiyun 	wait_queue_head_t	closed;
201*4882a593Smuzhiyun 	enum zfcp_fc_wka_status	status;
202*4882a593Smuzhiyun 	atomic_t		refcount;
203*4882a593Smuzhiyun 	u32			d_id;
204*4882a593Smuzhiyun 	u32			handle;
205*4882a593Smuzhiyun 	struct mutex		mutex;
206*4882a593Smuzhiyun 	struct delayed_work	work;
207*4882a593Smuzhiyun };
208*4882a593Smuzhiyun 
209*4882a593Smuzhiyun /**
210*4882a593Smuzhiyun  * struct zfcp_fc_wka_ports - Data structures for FC generic services
211*4882a593Smuzhiyun  * @ms: FC Management service
212*4882a593Smuzhiyun  * @ts: FC time service
213*4882a593Smuzhiyun  * @ds: FC directory service
214*4882a593Smuzhiyun  * @as: FC alias service
215*4882a593Smuzhiyun  */
216*4882a593Smuzhiyun struct zfcp_fc_wka_ports {
217*4882a593Smuzhiyun 	struct zfcp_fc_wka_port ms;
218*4882a593Smuzhiyun 	struct zfcp_fc_wka_port ts;
219*4882a593Smuzhiyun 	struct zfcp_fc_wka_port ds;
220*4882a593Smuzhiyun 	struct zfcp_fc_wka_port as;
221*4882a593Smuzhiyun };
222*4882a593Smuzhiyun 
223*4882a593Smuzhiyun /**
224*4882a593Smuzhiyun  * zfcp_fc_scsi_to_fcp - setup FCP command with data from scsi_cmnd
225*4882a593Smuzhiyun  * @fcp: fcp_cmnd to setup
226*4882a593Smuzhiyun  * @scsi: scsi_cmnd where to get LUN, task attributes/flags and CDB
227*4882a593Smuzhiyun  */
228*4882a593Smuzhiyun static inline
zfcp_fc_scsi_to_fcp(struct fcp_cmnd * fcp,struct scsi_cmnd * scsi)229*4882a593Smuzhiyun void zfcp_fc_scsi_to_fcp(struct fcp_cmnd *fcp, struct scsi_cmnd *scsi)
230*4882a593Smuzhiyun {
231*4882a593Smuzhiyun 	u32 datalen;
232*4882a593Smuzhiyun 
233*4882a593Smuzhiyun 	int_to_scsilun(scsi->device->lun, (struct scsi_lun *) &fcp->fc_lun);
234*4882a593Smuzhiyun 
235*4882a593Smuzhiyun 	fcp->fc_pri_ta = FCP_PTA_SIMPLE;
236*4882a593Smuzhiyun 
237*4882a593Smuzhiyun 	if (scsi->sc_data_direction == DMA_FROM_DEVICE)
238*4882a593Smuzhiyun 		fcp->fc_flags |= FCP_CFL_RDDATA;
239*4882a593Smuzhiyun 	if (scsi->sc_data_direction == DMA_TO_DEVICE)
240*4882a593Smuzhiyun 		fcp->fc_flags |= FCP_CFL_WRDATA;
241*4882a593Smuzhiyun 
242*4882a593Smuzhiyun 	memcpy(fcp->fc_cdb, scsi->cmnd, scsi->cmd_len);
243*4882a593Smuzhiyun 
244*4882a593Smuzhiyun 	datalen = scsi_bufflen(scsi);
245*4882a593Smuzhiyun 	fcp->fc_dl = cpu_to_be32(datalen);
246*4882a593Smuzhiyun 
247*4882a593Smuzhiyun 	if (scsi_get_prot_type(scsi) == SCSI_PROT_DIF_TYPE1) {
248*4882a593Smuzhiyun 		datalen += datalen / scsi->device->sector_size * 8;
249*4882a593Smuzhiyun 		fcp->fc_dl = cpu_to_be32(datalen);
250*4882a593Smuzhiyun 	}
251*4882a593Smuzhiyun }
252*4882a593Smuzhiyun 
253*4882a593Smuzhiyun /**
254*4882a593Smuzhiyun  * zfcp_fc_fcp_tm() - Setup FCP command as task management command.
255*4882a593Smuzhiyun  * @fcp: Pointer to FCP_CMND IU to set up.
256*4882a593Smuzhiyun  * @dev: Pointer to SCSI_device where to send the task management command.
257*4882a593Smuzhiyun  * @tm_flags: Task management flags to setup tm command.
258*4882a593Smuzhiyun  */
259*4882a593Smuzhiyun static inline
zfcp_fc_fcp_tm(struct fcp_cmnd * fcp,struct scsi_device * dev,u8 tm_flags)260*4882a593Smuzhiyun void zfcp_fc_fcp_tm(struct fcp_cmnd *fcp, struct scsi_device *dev, u8 tm_flags)
261*4882a593Smuzhiyun {
262*4882a593Smuzhiyun 	int_to_scsilun(dev->lun, (struct scsi_lun *) &fcp->fc_lun);
263*4882a593Smuzhiyun 	fcp->fc_tm_flags = tm_flags;
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun 
266*4882a593Smuzhiyun /**
267*4882a593Smuzhiyun  * zfcp_fc_evap_fcp_rsp - evaluate FCP RSP IU and update scsi_cmnd accordingly
268*4882a593Smuzhiyun  * @fcp_rsp: FCP RSP IU to evaluate
269*4882a593Smuzhiyun  * @scsi: SCSI command where to update status and sense buffer
270*4882a593Smuzhiyun  */
271*4882a593Smuzhiyun static inline
zfcp_fc_eval_fcp_rsp(struct fcp_resp_with_ext * fcp_rsp,struct scsi_cmnd * scsi)272*4882a593Smuzhiyun void zfcp_fc_eval_fcp_rsp(struct fcp_resp_with_ext *fcp_rsp,
273*4882a593Smuzhiyun 			  struct scsi_cmnd *scsi)
274*4882a593Smuzhiyun {
275*4882a593Smuzhiyun 	struct fcp_resp_rsp_info *rsp_info;
276*4882a593Smuzhiyun 	char *sense;
277*4882a593Smuzhiyun 	u32 sense_len, resid;
278*4882a593Smuzhiyun 	u8 rsp_flags;
279*4882a593Smuzhiyun 
280*4882a593Smuzhiyun 	set_msg_byte(scsi, COMMAND_COMPLETE);
281*4882a593Smuzhiyun 	scsi->result |= fcp_rsp->resp.fr_status;
282*4882a593Smuzhiyun 
283*4882a593Smuzhiyun 	rsp_flags = fcp_rsp->resp.fr_flags;
284*4882a593Smuzhiyun 
285*4882a593Smuzhiyun 	if (unlikely(rsp_flags & FCP_RSP_LEN_VAL)) {
286*4882a593Smuzhiyun 		rsp_info = (struct fcp_resp_rsp_info *) &fcp_rsp[1];
287*4882a593Smuzhiyun 		if (rsp_info->rsp_code == FCP_TMF_CMPL)
288*4882a593Smuzhiyun 			set_host_byte(scsi, DID_OK);
289*4882a593Smuzhiyun 		else {
290*4882a593Smuzhiyun 			set_host_byte(scsi, DID_ERROR);
291*4882a593Smuzhiyun 			return;
292*4882a593Smuzhiyun 		}
293*4882a593Smuzhiyun 	}
294*4882a593Smuzhiyun 
295*4882a593Smuzhiyun 	if (unlikely(rsp_flags & FCP_SNS_LEN_VAL)) {
296*4882a593Smuzhiyun 		sense = (char *) &fcp_rsp[1];
297*4882a593Smuzhiyun 		if (rsp_flags & FCP_RSP_LEN_VAL)
298*4882a593Smuzhiyun 			sense += be32_to_cpu(fcp_rsp->ext.fr_rsp_len);
299*4882a593Smuzhiyun 		sense_len = min_t(u32, be32_to_cpu(fcp_rsp->ext.fr_sns_len),
300*4882a593Smuzhiyun 				  SCSI_SENSE_BUFFERSIZE);
301*4882a593Smuzhiyun 		memcpy(scsi->sense_buffer, sense, sense_len);
302*4882a593Smuzhiyun 	}
303*4882a593Smuzhiyun 
304*4882a593Smuzhiyun 	if (unlikely(rsp_flags & FCP_RESID_UNDER)) {
305*4882a593Smuzhiyun 		resid = be32_to_cpu(fcp_rsp->ext.fr_resid);
306*4882a593Smuzhiyun 		scsi_set_resid(scsi, resid);
307*4882a593Smuzhiyun 		if (scsi_bufflen(scsi) - resid < scsi->underflow &&
308*4882a593Smuzhiyun 		     !(rsp_flags & FCP_SNS_LEN_VAL) &&
309*4882a593Smuzhiyun 		     fcp_rsp->resp.fr_status == SAM_STAT_GOOD)
310*4882a593Smuzhiyun 			set_host_byte(scsi, DID_ERROR);
311*4882a593Smuzhiyun 	} else if (unlikely(rsp_flags & FCP_RESID_OVER)) {
312*4882a593Smuzhiyun 		/* FCP_DL was not sufficient for SCSI data length */
313*4882a593Smuzhiyun 		if (fcp_rsp->resp.fr_status == SAM_STAT_GOOD)
314*4882a593Smuzhiyun 			set_host_byte(scsi, DID_ERROR);
315*4882a593Smuzhiyun 	}
316*4882a593Smuzhiyun }
317*4882a593Smuzhiyun 
318*4882a593Smuzhiyun #endif
319