xref: /freebsd/sys/dev/ocs_fc/ocs_io.h (revision 95ee2897e98f5d444f26ed2334cc7c439f9c16c6)
1*ef270ab1SKenneth D. Merry /*-
2*ef270ab1SKenneth D. Merry  * Copyright (c) 2017 Broadcom. All rights reserved.
3*ef270ab1SKenneth D. Merry  * The term "Broadcom" refers to Broadcom Limited and/or its subsidiaries.
4*ef270ab1SKenneth D. Merry  *
5*ef270ab1SKenneth D. Merry  * Redistribution and use in source and binary forms, with or without
6*ef270ab1SKenneth D. Merry  * modification, are permitted provided that the following conditions are met:
7*ef270ab1SKenneth D. Merry  *
8*ef270ab1SKenneth D. Merry  * 1. Redistributions of source code must retain the above copyright notice,
9*ef270ab1SKenneth D. Merry  *    this list of conditions and the following disclaimer.
10*ef270ab1SKenneth D. Merry  *
11*ef270ab1SKenneth D. Merry  * 2. Redistributions in binary form must reproduce the above copyright notice,
12*ef270ab1SKenneth D. Merry  *    this list of conditions and the following disclaimer in the documentation
13*ef270ab1SKenneth D. Merry  *    and/or other materials provided with the distribution.
14*ef270ab1SKenneth D. Merry  *
15*ef270ab1SKenneth D. Merry  * 3. Neither the name of the copyright holder nor the names of its contributors
16*ef270ab1SKenneth D. Merry  *    may be used to endorse or promote products derived from this software
17*ef270ab1SKenneth D. Merry  *    without specific prior written permission.
18*ef270ab1SKenneth D. Merry  *
19*ef270ab1SKenneth D. Merry  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
20*ef270ab1SKenneth D. Merry  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21*ef270ab1SKenneth D. Merry  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22*ef270ab1SKenneth D. Merry  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
23*ef270ab1SKenneth D. Merry  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24*ef270ab1SKenneth D. Merry  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25*ef270ab1SKenneth D. Merry  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26*ef270ab1SKenneth D. Merry  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27*ef270ab1SKenneth D. Merry  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28*ef270ab1SKenneth D. Merry  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29*ef270ab1SKenneth D. Merry  * POSSIBILITY OF SUCH DAMAGE.
30*ef270ab1SKenneth D. Merry  */
31*ef270ab1SKenneth D. Merry 
32*ef270ab1SKenneth D. Merry /**
33*ef270ab1SKenneth D. Merry  * @file
34*ef270ab1SKenneth D. Merry  * OCS linux driver IO declarations
35*ef270ab1SKenneth D. Merry  */
36*ef270ab1SKenneth D. Merry 
37*ef270ab1SKenneth D. Merry #if !defined(__OCS_IO_H__)
38*ef270ab1SKenneth D. Merry #define __OCS_IO_H__
39*ef270ab1SKenneth D. Merry 
40*ef270ab1SKenneth D. Merry #define io_error_log(io, fmt, ...)  \
41*ef270ab1SKenneth D. Merry 	do { \
42*ef270ab1SKenneth D. Merry 		if (OCS_LOG_ENABLE_IO_ERRORS(io->ocs)) \
43*ef270ab1SKenneth D. Merry 			ocs_log_warn(io->ocs, fmt, ##__VA_ARGS__); \
44*ef270ab1SKenneth D. Merry 	} while (0)
45*ef270ab1SKenneth D. Merry 
46*ef270ab1SKenneth D. Merry /**
47*ef270ab1SKenneth D. Merry  * @brief FCP IO context
48*ef270ab1SKenneth D. Merry  *
49*ef270ab1SKenneth D. Merry  * This structure is used for transport and backend IO requests and responses.
50*ef270ab1SKenneth D. Merry  */
51*ef270ab1SKenneth D. Merry 
52*ef270ab1SKenneth D. Merry #define SCSI_CMD_BUF_LENGTH		48
53*ef270ab1SKenneth D. Merry #define SCSI_RSP_BUF_LENGTH		sizeof(fcp_rsp_iu_t)
54*ef270ab1SKenneth D. Merry 
55*ef270ab1SKenneth D. Merry /**
56*ef270ab1SKenneth D. Merry  * @brief OCS IO types
57*ef270ab1SKenneth D. Merry  */
58*ef270ab1SKenneth D. Merry typedef enum {
59*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_IO = 0,
60*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_ELS,
61*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_CT,
62*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_CT_RESP,
63*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_BLS_RESP,
64*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_ABORT,
65*ef270ab1SKenneth D. Merry 
66*ef270ab1SKenneth D. Merry 	OCS_IO_TYPE_MAX,		/**< must be last */
67*ef270ab1SKenneth D. Merry } ocs_io_type_e;
68*ef270ab1SKenneth D. Merry 
69*ef270ab1SKenneth D. Merry struct ocs_io_s {
70*ef270ab1SKenneth D. Merry 	ocs_t *ocs;			/**< pointer back to ocs */
71*ef270ab1SKenneth D. Merry 	uint32_t instance_index;	/**< unique instance index value */
72*ef270ab1SKenneth D. Merry 	const char *display_name;	/**< display name */
73*ef270ab1SKenneth D. Merry 	ocs_node_t *node;		/**< pointer to node */
74*ef270ab1SKenneth D. Merry 	ocs_list_link_t io_alloc_link;	/**< (io_pool->io_free_list) free list link */
75*ef270ab1SKenneth D. Merry 	uint32_t init_task_tag;		/**< initiator task tag (OX_ID) for back-end and SCSI logging */
76*ef270ab1SKenneth D. Merry 	uint32_t tgt_task_tag;		/**< target task tag (RX_ID) - for back-end and SCSI logging */
77*ef270ab1SKenneth D. Merry 	uint32_t hw_tag;		/**< HW layer unique IO id - for back-end and SCSI logging */
78*ef270ab1SKenneth D. Merry 	uint32_t tag;			/**< unique IO identifier */
79*ef270ab1SKenneth D. Merry 	ocs_scsi_sgl_t *sgl;		/**< SGL */
80*ef270ab1SKenneth D. Merry 	uint32_t sgl_allocated;		/**< Number of allocated SGEs */
81*ef270ab1SKenneth D. Merry 	uint32_t sgl_count;		/**< Number of SGEs in this SGL */
82*ef270ab1SKenneth D. Merry 	ocs_scsi_ini_io_t ini_io;	/**< backend initiator private IO data */
83*ef270ab1SKenneth D. Merry 	ocs_scsi_tgt_io_t tgt_io;	/**< backend target private IO data */
84*ef270ab1SKenneth D. Merry 	uint32_t exp_xfer_len;		/**< expected data transfer length, based on FC or iSCSI header */
85*ef270ab1SKenneth D. Merry 	ocs_mgmt_functions_t *mgmt_functions;
86*ef270ab1SKenneth D. Merry 
87*ef270ab1SKenneth D. Merry 	/* Declarations private to HW/SLI */
88*ef270ab1SKenneth D. Merry 	void *hw_priv;			/**< HW private context */
89*ef270ab1SKenneth D. Merry 
90*ef270ab1SKenneth D. Merry 	/* Declarations private to FC Transport */
91*ef270ab1SKenneth D. Merry 	ocs_io_type_e io_type;		/**< indicates what this ocs_io_t structure is used for */
92*ef270ab1SKenneth D. Merry 	ocs_ref_t ref;			/**< refcount object */
93*ef270ab1SKenneth D. Merry 	void *dslab_item;		/**< pointer back to dslab allocation object */
94*ef270ab1SKenneth D. Merry 	ocs_hw_io_t *hio;		/**< HW IO context */
95*ef270ab1SKenneth D. Merry 	size_t transferred;		/**< Number of bytes transferred so far */
96*ef270ab1SKenneth D. Merry 	uint32_t auto_resp:1,		/**< set if auto_trsp was set */
97*ef270ab1SKenneth D. Merry 		 low_latency:1,		/**< set if low latency request */
98*ef270ab1SKenneth D. Merry 		 wq_steering:4,		/**< selected WQ steering request */
99*ef270ab1SKenneth D. Merry 		 wq_class:4;		/**< selected WQ class if steering is class */
100*ef270ab1SKenneth D. Merry 	uint32_t xfer_req;		/**< transfer size for current request */
101*ef270ab1SKenneth D. Merry 	ocs_scsi_rsp_io_cb_t scsi_ini_cb; /**< initiator callback function */
102*ef270ab1SKenneth D. Merry 	void *scsi_ini_cb_arg;		/**< initiator callback function argument */
103*ef270ab1SKenneth D. Merry 	ocs_scsi_io_cb_t scsi_tgt_cb;	/**< target callback function */
104*ef270ab1SKenneth D. Merry 	void *scsi_tgt_cb_arg;		/**< target callback function argument */
105*ef270ab1SKenneth D. Merry 	ocs_scsi_io_cb_t abort_cb;	/**< abort callback function */
106*ef270ab1SKenneth D. Merry 	void *abort_cb_arg;		/**< abort callback function argument */
107*ef270ab1SKenneth D. Merry 	ocs_scsi_io_cb_t bls_cb;	/**< BLS callback function */
108*ef270ab1SKenneth D. Merry 	void *bls_cb_arg;		/**< BLS callback function argument */
109*ef270ab1SKenneth D. Merry 	ocs_scsi_tmf_cmd_e tmf_cmd;	/**< TMF command being processed */
110*ef270ab1SKenneth D. Merry 	uint16_t abort_rx_id;		/**< rx_id from the ABTS that initiated the command abort */
111*ef270ab1SKenneth D. Merry 
112*ef270ab1SKenneth D. Merry 	uint32_t cmd_tgt:1,		/**< True if this is a Target command */
113*ef270ab1SKenneth D. Merry 		 send_abts:1,		/**< when aborting, indicates ABTS is to be sent */
114*ef270ab1SKenneth D. Merry 		 cmd_ini:1,		/**< True if this is an Initiator command */
115*ef270ab1SKenneth D. Merry 		 seq_init:1;		/**< True if local node has sequence initiative */
116*ef270ab1SKenneth D. Merry 	ocs_hw_io_param_t iparam;	/**< iparams for hw io send call */
117*ef270ab1SKenneth D. Merry 	ocs_hw_dif_info_t hw_dif;	/**< HW formatted DIF parameters */
118*ef270ab1SKenneth D. Merry 	ocs_scsi_dif_info_t scsi_dif_info;	/**< DIF info saved for DIF error recovery */
119*ef270ab1SKenneth D. Merry 	ocs_hw_io_type_e hio_type;	/**< HW IO type */
120*ef270ab1SKenneth D. Merry 	uint32_t wire_len;		/**< wire length */
121*ef270ab1SKenneth D. Merry 	void *hw_cb;			/**< saved HW callback */
122*ef270ab1SKenneth D. Merry 	ocs_list_link_t io_pending_link;/**< link list link pending */
123*ef270ab1SKenneth D. Merry 
124*ef270ab1SKenneth D. Merry 	ocs_dma_t ovfl_sgl;		/**< Overflow SGL */
125*ef270ab1SKenneth D. Merry 
126*ef270ab1SKenneth D. Merry 	/* for ELS requests/responses */
127*ef270ab1SKenneth D. Merry 	uint32_t els_pend:1,		/**< True if ELS is pending */
128*ef270ab1SKenneth D. Merry 		els_active:1;		/**< True if ELS is active */
129*ef270ab1SKenneth D. Merry 	ocs_dma_t els_req;		/**< ELS request payload buffer */
130*ef270ab1SKenneth D. Merry 	ocs_dma_t els_rsp;		/**< ELS response payload buffer */
131*ef270ab1SKenneth D. Merry 	ocs_sm_ctx_t els_sm;		/**< EIO IO state machine context */
132*ef270ab1SKenneth D. Merry 	uint32_t els_evtdepth;		/**< current event posting nesting depth */
133*ef270ab1SKenneth D. Merry 	uint32_t els_req_free:1;	/**< this els is to be free'd */
134*ef270ab1SKenneth D. Merry 	uint32_t els_retries_remaining;	/*<< Retries remaining */
135*ef270ab1SKenneth D. Merry 	void (*els_callback)(ocs_node_t *node, ocs_node_cb_t *cbdata, void *cbarg);
136*ef270ab1SKenneth D. Merry 	void *els_callback_arg;
137*ef270ab1SKenneth D. Merry 	uint32_t els_timeout_sec;	/**< timeout */
138*ef270ab1SKenneth D. Merry 
139*ef270ab1SKenneth D. Merry 	ocs_timer_t delay_timer;	/**< delay timer */
140*ef270ab1SKenneth D. Merry 
141*ef270ab1SKenneth D. Merry 	/* for abort handling */
142*ef270ab1SKenneth D. Merry 	ocs_io_t *io_to_abort;		/**< pointer to IO to abort */
143*ef270ab1SKenneth D. Merry 
144*ef270ab1SKenneth D. Merry 	ocs_list_link_t link;		/**< linked list link */
145*ef270ab1SKenneth D. Merry 	ocs_dma_t cmdbuf;		/**< SCSI Command buffer, used for CDB (initiator) */
146*ef270ab1SKenneth D. Merry 	ocs_dma_t rspbuf;		/**< SCSI Response buffer (i+t) */
147*ef270ab1SKenneth D. Merry 	uint32_t  timeout;		/**< Timeout value in seconds for this IO */
148*ef270ab1SKenneth D. Merry 	uint8_t   cs_ctl;		/**< CS_CTL priority for this IO */
149*ef270ab1SKenneth D. Merry 	uint8_t	  io_free;		/**< Is io object in freelist > */
150*ef270ab1SKenneth D. Merry 	uint32_t  app_id;
151*ef270ab1SKenneth D. Merry };
152*ef270ab1SKenneth D. Merry 
153*ef270ab1SKenneth D. Merry /**
154*ef270ab1SKenneth D. Merry  * @brief common IO callback argument
155*ef270ab1SKenneth D. Merry  *
156*ef270ab1SKenneth D. Merry  * Callback argument used as common I/O callback argument
157*ef270ab1SKenneth D. Merry  */
158*ef270ab1SKenneth D. Merry 
159*ef270ab1SKenneth D. Merry typedef struct {
160*ef270ab1SKenneth D. Merry 	int32_t status;				/**< completion status */
161*ef270ab1SKenneth D. Merry 	int32_t ext_status;			/**< extended completion status */
162*ef270ab1SKenneth D. Merry 	void *app;				/**< application argument */
163*ef270ab1SKenneth D. Merry } ocs_io_cb_arg_t;
164*ef270ab1SKenneth D. Merry 
165*ef270ab1SKenneth D. Merry /**
166*ef270ab1SKenneth D. Merry  * @brief Test if IO object is busy
167*ef270ab1SKenneth D. Merry  *
168*ef270ab1SKenneth D. Merry  * Return True if IO object is busy.   Busy is defined as the IO object not being on
169*ef270ab1SKenneth D. Merry  * the free list
170*ef270ab1SKenneth D. Merry  *
171*ef270ab1SKenneth D. Merry  * @param io Pointer to IO object
172*ef270ab1SKenneth D. Merry  *
173*ef270ab1SKenneth D. Merry  * @return returns True if IO is busy
174*ef270ab1SKenneth D. Merry  */
175*ef270ab1SKenneth D. Merry 
176*ef270ab1SKenneth D. Merry static inline int32_t
ocs_io_busy(ocs_io_t * io)177*ef270ab1SKenneth D. Merry ocs_io_busy(ocs_io_t *io)
178*ef270ab1SKenneth D. Merry {
179*ef270ab1SKenneth D. Merry 	return !(io->io_free);
180*ef270ab1SKenneth D. Merry }
181*ef270ab1SKenneth D. Merry 
182*ef270ab1SKenneth D. Merry typedef struct ocs_io_pool_s ocs_io_pool_t;
183*ef270ab1SKenneth D. Merry 
184*ef270ab1SKenneth D. Merry extern ocs_io_pool_t *ocs_io_pool_create(ocs_t *ocs, uint32_t num_io, uint32_t num_sgl);
185*ef270ab1SKenneth D. Merry extern int32_t ocs_io_pool_free(ocs_io_pool_t *io_pool);
186*ef270ab1SKenneth D. Merry extern uint32_t ocs_io_pool_allocated(ocs_io_pool_t *io_pool);
187*ef270ab1SKenneth D. Merry 
188*ef270ab1SKenneth D. Merry extern ocs_io_t *ocs_io_pool_io_alloc(ocs_io_pool_t *io_pool);
189*ef270ab1SKenneth D. Merry extern void ocs_io_pool_io_free(ocs_io_pool_t *io_pool, ocs_io_t *io);
190*ef270ab1SKenneth D. Merry extern ocs_io_t *ocs_io_find_tgt_io(ocs_t *ocs, ocs_node_t *node, uint16_t ox_id, uint16_t rx_id);
191*ef270ab1SKenneth D. Merry extern void ocs_ddump_io(ocs_textbuf_t *textbuf, ocs_io_t *io);
192*ef270ab1SKenneth D. Merry 
193*ef270ab1SKenneth D. Merry #endif
194