xref: /illumos-gate/usr/src/uts/common/sys/stmf.h (revision b1d7ec75953cd517f5b7c3d9cb427ff8ec5d7d07)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
23  */
24 #ifndef	_STMF_H
25 #define	_STMF_H
26 
27 #include <sys/stmf_defines.h>
28 
29 #ifdef	__cplusplus
30 extern "C" {
31 #endif
32 
33 typedef enum stmf_struct_id {
34 	STMF_STRUCT_LU_PROVIDER = 1,
35 	STMF_STRUCT_PORT_PROVIDER,
36 	STMF_STRUCT_STMF_LOCAL_PORT,
37 	STMF_STRUCT_STMF_LU,
38 	STMF_STRUCT_SCSI_SESSION,
39 	STMF_STRUCT_SCSI_TASK,
40 	STMF_STRUCT_DATA_BUF,
41 	STMF_STRUCT_DBUF_STORE,
42 	STMF_MAX_STRUCT_IDS
43 } stmf_struct_id_t;
44 
45 /*
46  * Provider callback commands
47  */
48 #define	STMF_PROVIDER_DATA_UPDATED	0x01
49 
50 /*
51  * Provider callback flags
52  */
53 #define	STMF_PCB_STMF_ONLINING		0x0001
54 #define	STMF_PCB_PREG_COMPLETE		0x0002
55 
56 typedef void *data_seg_handle_t;
57 #define	STMF_MAX_LU_CACHE_NTASKS 16
58 
59 #define	STMF_NO_HANDLE	0xffffffff
60 
61 #define	COMPANY_ID_NONE			0xFFFFFFFF
62 #define	COMPANY_ID_SUN			0x00144F
63 
64 /*
65  * The scatter/gather list buffer format is used in 2 different
66  * contexts within stmf:
67  * 1) supplied by the port provider that the LU provider uses to exchange
68  *    data with the backing store.
69  * 2) supplied by the LU provider that the port provider uses exchange
70  *    data with the host initiator.
71  * The second format is optionally supported by the port provided as
72  * indicated by the command task flags.
73  */
74 
75 typedef struct stmf_sglist_ent {
76 	uint32_t	seg_length;
77 	uint8_t		*seg_addr;
78 } stmf_sglist_ent_t;
79 
80 typedef struct stmf_data_buf {
81 	void		*db_stmf_private;
82 	void		*db_port_private;
83 	void		*db_lu_private;
84 	uint32_t	db_buf_size;	/* Total size of this buffer */
85 	uint32_t	db_data_size;	/* Intended xfer size of this buffer */
86 	uint32_t	db_relative_offset;
87 	uint16_t	db_sglist_length;
88 	uint16_t	db_flags;	/* Direction, auto status etc */
89 	stmf_status_t	db_xfer_status;
90 	uint8_t		db_handle;	/* To track parallel buffers */
91 	hrtime_t	db_xfer_start_timestamp;
92 	stmf_sglist_ent_t db_sglist[1];	/* PP scatter/gather list */
93 } stmf_data_buf_t;
94 
95 /*
96  * db_flags
97  */
98 #define	DB_DIRECTION_TO_RPORT		0x0001
99 #define	DB_DIRECTION_FROM_RPORT		0x0002
100 #define	DB_SEND_STATUS_GOOD		0x0004
101 #define	DB_STATUS_GOOD_SENT		0x0008
102 #define	DB_DONT_CACHE			0x0010
103 #define	DB_DONT_REUSE			0x0020
104 #define	DB_LU_DATA_BUF			0x0040
105 
106 typedef struct scsi_task {
107 	void		*task_stmf_private;
108 	void		*task_port_private;
109 
110 	void		*task_lu_private;
111 	struct stmf_scsi_session *task_session;
112 	struct stmf_local_port *task_lport;
113 	struct stmf_lu	*task_lu;
114 	void		*task_lu_itl_handle;	/* Assigned by LU */
115 
116 	/* CMD information from initiator */
117 	uint8_t		task_lun_no[8];
118 	uint8_t		task_flags;		/* See def. for task flags */
119 	uint8_t		task_priority;		/* As per SAM-3 */
120 	uint8_t		task_mgmt_function;	/* If this is a TM request */
121 	uint8_t		task_max_nbufs;
122 	uint8_t		task_cur_nbufs;
123 	uint8_t		task_csn_size;		/* cmd seq no size in bits */
124 	uint16_t	task_additional_flags;
125 	uint32_t	task_cmd_seq_no;
126 	uint32_t	task_expected_xfer_length;
127 	uint32_t	task_timeout;		/* In seconds */
128 	uint16_t	task_ext_id;
129 	uint16_t	task_cdb_length;
130 	uint8_t		*task_cdb;
131 
132 	/* Fields to manage data phase */
133 	uint32_t	task_cmd_xfer_length;	/* xfer len based on CDB */
134 	uint32_t	task_nbytes_transferred;
135 	uint32_t	task_max_xfer_len;	/* largest xfer allowed */
136 	uint32_t	task_1st_xfer_len;	/* 1st xfer hint */
137 	uint32_t	task_copy_threshold;	/* copy reduction threshold */
138 
139 
140 	/* Status Phase */
141 	stmf_status_t	task_completion_status;
142 	uint32_t	task_resid;
143 	uint8_t		task_status_ctrl;	/* See def. for status ctrl */
144 	uint8_t		task_scsi_status;
145 	uint16_t	task_sense_length;
146 	uint8_t		*task_sense_data;
147 
148 	/* Misc. task data */
149 	void		*task_extended_cmd;
150 
151 } scsi_task_t;
152 
153 /*
154  * Maximum expected transfer length.   Can also be used when the transfer
155  * length is unknown when the task is allocated (e.g. SAS)
156  */
157 
158 #define	TASK_MAX_XFER_LENGTH	0xFFFFFFFF
159 
160 /*
161  * task_flags definitions.
162  */
163 /*
164  * If TF_INITIAL_BURST is set, the dbuf passed with new_task() contains
165  * data from initial burst. Otherwise its just a buffer which the port
166  * passed to the LU.
167  */
168 #define	TF_INITIAL_BURST	0x80
169 /* Both READ_DATA and WRITE_DATA can be set for bidirectional xfers */
170 #define	TF_READ_DATA		0x40
171 #define	TF_WRITE_DATA		0x20
172 #define	TF_ATTR_MASK		0x07
173 #define	TF_ATTR_UNTAGGED	0x0
174 #define	TF_ATTR_SIMPLE_QUEUE	0x1
175 #define	TF_ATTR_ORDERED_QUEUE	0x2
176 #define	TF_ATTR_HEAD_OF_QUEUE	0x3
177 #define	TF_ATTR_ACA		0x4
178 
179 /*
180  * Task Management flags.
181  */
182 #define	TM_NONE			0x00
183 #define	TM_ABORT_TASK		0x01
184 #define	TM_ABORT_TASK_SET	0x02
185 #define	TM_CLEAR_ACA		0x03
186 #define	TM_CLEAR_TASK_SET	0x04
187 #define	TM_LUN_RESET		0x05
188 #define	TM_TARGET_WARM_RESET	0x06
189 #define	TM_TARGET_COLD_RESET	0x07
190 #define	TM_TASK_REASSIGN	0x08
191 #define	TM_TARGET_RESET		0x09
192 #define	TM_QUERY_TASK		0x0A
193 
194 /*
195  * additional flags
196  */
197 #define	TASK_AF_ENABLE_COMP_CONF	0x01
198 #define	TASK_AF_PORT_LOAD_HIGH		0x02
199 #define	TASK_AF_NO_EXPECTED_XFER_LENGTH	0x04
200 /*
201  * PP sets this flag if it can process dbufs created by the LU.
202  */
203 #define	TASK_AF_ACCEPT_LU_DBUF		0x08
204 
205 /*
206  * scsi_task_t extension identifiers
207  */
208 #define	STMF_TASK_EXT_NONE		0
209 
210 /*
211  * max_nbufs
212  */
213 #define	STMF_BUFS_MAX		255
214 
215 /*
216  * Task status ctrl
217  */
218 #define	TASK_SCTRL_OVER		1
219 #define	TASK_SCTRL_UNDER	2
220 
221 /*
222  * The flags used by I/O flow.
223  */
224 #define	STMF_IOF_LU_DONE		0x0001
225 #define	STMF_IOF_LPORT_DONE		0x0002
226 #define	STMF_IOF_STATS_ONLY		0x0004
227 
228 /*
229  * struct allocation flags
230  */
231 #define	AF_FORCE_NOSLEEP	0x0001
232 #define	AF_DONTZERO		0x0002
233 
234 typedef struct stmf_state_change_info {
235 	uint64_t	st_rflags;	/* Reason behind this change */
236 	char		*st_additional_info;
237 } stmf_state_change_info_t;
238 
239 typedef struct stmf_change_status {
240 	stmf_status_t	st_completion_status;
241 	char		*st_additional_info;
242 } stmf_change_status_t;
243 
244 /*
245  * conditions causing or affecting the change.
246  */
247 #define	STMF_RFLAG_USER_REQUEST		0x0001
248 #define	STMF_RFLAG_FATAL_ERROR		0x0002
249 #define	STMF_RFLAG_STAY_OFFLINED	0x0004
250 #define	STMF_RFLAG_RESET		0x0008
251 #define	STMF_RFLAG_COLLECT_DEBUG_DUMP	0x0010
252 #define	STMF_RFLAG_LU_ABORT		0x0020
253 #define	STMF_RFLAG_LPORT_ABORT		0x0040
254 
255 #define	STMF_CHANGE_INFO_LEN		160
256 
257 /*
258  * cmds to stmf_abort entry point
259  */
260 #define	STMF_QUEUE_TASK_ABORT		1
261 #define	STMF_REQUEUE_TASK_ABORT_LPORT	2
262 #define	STMF_REQUEUE_TASK_ABORT_LU	3
263 #define	STMF_QUEUE_ABORT_LU		4
264 
265 /*
266  * cmds to be used by stmf ctl
267  */
268 #define	STMF_CMD_LU_OP			0x0100
269 #define	STMF_CMD_LPORT_OP		0x0200
270 #define	STMF_CMD_MASK			0x00ff
271 #define	STMF_CMD_ONLINE			0x0001
272 #define	STMF_CMD_OFFLINE		0x0002
273 #define	STMF_CMD_GET_STATUS		0x0003
274 #define	STMF_CMD_ONLINE_COMPLETE	0x0004
275 #define	STMF_CMD_OFFLINE_COMPLETE	0x0005
276 #define	STMF_ACK_ONLINE_COMPLETE	0x0006
277 #define	STMF_ACK_OFFLINE_COMPLETE	0x0007
278 
279 #define	STMF_CMD_LU_ONLINE		(STMF_CMD_LU_OP | STMF_CMD_ONLINE)
280 #define	STMF_CMD_LU_OFFLINE		(STMF_CMD_LU_OP | STMF_CMD_OFFLINE)
281 #define	STMF_CMD_LPORT_ONLINE		(STMF_CMD_LPORT_OP | STMF_CMD_ONLINE)
282 #define	STMF_CMD_LPORT_OFFLINE		(STMF_CMD_LPORT_OP | STMF_CMD_OFFLINE)
283 #define	STMF_CMD_GET_LU_STATUS		(STMF_CMD_LU_OP | STMF_CMD_GET_STATUS)
284 #define	STMF_CMD_GET_LPORT_STATUS	\
285 			(STMF_CMD_LPORT_OP | STMF_CMD_GET_STATUS)
286 #define	STMF_CMD_LU_ONLINE_COMPLETE	\
287 			(STMF_CMD_LU_OP | STMF_CMD_ONLINE_COMPLETE)
288 #define	STMF_CMD_LPORT_ONLINE_COMPLETE	\
289 			(STMF_CMD_LPORT_OP | STMF_CMD_ONLINE_COMPLETE)
290 #define	STMF_ACK_LU_ONLINE_COMPLETE	\
291 			(STMF_CMD_LU_OP | STMF_ACK_ONLINE_COMPLETE)
292 #define	STMF_ACK_LPORT_ONLINE_COMPLETE	\
293 			(STMF_CMD_LPORT_OP | STMF_ACK_ONLINE_COMPLETE)
294 #define	STMF_CMD_LU_OFFLINE_COMPLETE	\
295 			(STMF_CMD_LU_OP | STMF_CMD_OFFLINE_COMPLETE)
296 #define	STMF_CMD_LPORT_OFFLINE_COMPLETE	\
297 			(STMF_CMD_LPORT_OP | STMF_CMD_OFFLINE_COMPLETE)
298 #define	STMF_ACK_LU_OFFLINE_COMPLETE	\
299 			(STMF_CMD_LU_OP | STMF_ACK_OFFLINE_COMPLETE)
300 #define	STMF_ACK_LPORT_OFFLINE_COMPLETE	\
301 			(STMF_CMD_LPORT_OP | STMF_ACK_OFFLINE_COMPLETE)
302 /*
303  * For LPORTs and LUs to create their own ctl cmds which dont
304  * conflict with stmf ctl cmds.
305  */
306 #define	STMF_LPORT_CTL_CMDS		0x1000
307 #define	STMF_LU_CTL_CMDS		0x2000
308 
309 /*
310  * Commands for various info routines.
311  */
312 /* Command classifiers */
313 #define	SI_LPORT		0x1000000
314 #define	SI_STMF			0x2000000
315 #define	SI_LU			0x4000000
316 #define	SI_LPORT_FC		0x0000000
317 #define	SI_LPORT_ISCSI		0x0010000
318 #define	SI_LPORT_SAS		0x0020000
319 #define	SI_STMF_LU		0x0010000
320 #define	SI_STMF_LPORT		0x0020000
321 
322 #define	SI_GET_CLASS(v)		((v) & 0xFF000000)
323 #define	SI_GET_SUBCLASS(v)	((v) & 0x00FF0000)
324 
325 /* Commands for LPORT info routines */
326 /* XXX - Implement these. */
327 #if 0
328 #define	SI_LPORT_FC_PORTINFO		(SI_LPORT | SI_LPORT_FC | 1)
329 #define	SI_RPORT_FC_PORTINFO		(SI_LPORT | SI_LPORT_FC | 2)
330 #endif
331 
332 /*
333  * Events
334  */
335 #define	STMF_EVENT_ALL			((int)-1)
336 #define	LPORT_EVENT_INITIAL_LUN_MAPPED	0
337 
338 /*
339  * This needs to go into common/ddi/sunddi.h
340  */
341 #define	DDI_NT_STMF		"ddi_scsi_target:framework"
342 #define	DDI_NT_STMF_LP		"ddi_scsi_target:lu_provider"
343 #define	DDI_NT_STMF_PP		"ddi_scsi_target:port_provider"
344 
345 /*
346  * VPD page bits.
347  */
348 #define	STMF_VPD_LU_ID		0x01
349 #define	STMF_VPD_TARGET_ID	0x02
350 #define	STMF_VPD_TP_GROUP	0x04
351 #define	STMF_VPD_RELATIVE_TP_ID	0x08
352 
353 /*
354  * Common macros to simplify coding
355  */
356 #define	STMF_SEC2TICK(x_sec)	(drv_usectohz((x_sec) * 1000000))
357 
358 void stmf_trace(caddr_t ident, const char *fmt, ...);
359 void *stmf_alloc(stmf_struct_id_t sid, int additional_size, int alloc_flags);
360 void stmf_free(void *struct_ptr);
361 struct scsi_task *stmf_task_alloc(struct stmf_local_port *lport,
362     struct stmf_scsi_session *ss, uint8_t *lun, uint16_t cdb_length,
363     uint16_t ext_id);
364 void stmf_post_task(scsi_task_t *task, stmf_data_buf_t *dbuf);
365 stmf_data_buf_t *stmf_alloc_dbuf(scsi_task_t *task, uint32_t size,
366     uint32_t *pminsize, uint32_t flags);
367 void stmf_free_dbuf(scsi_task_t *task, stmf_data_buf_t *dbuf);
368 stmf_status_t stmf_setup_dbuf(scsi_task_t *task, stmf_data_buf_t *dbuf,
369     uint32_t flags);
370 void stmf_teardown_dbuf(scsi_task_t *task, stmf_data_buf_t *dbuf);
371 stmf_status_t stmf_xfer_data(scsi_task_t *task, stmf_data_buf_t *dbuf,
372     uint32_t ioflags);
373 stmf_status_t stmf_send_scsi_status(scsi_task_t *task, uint32_t ioflags);
374 void stmf_data_xfer_done(scsi_task_t *task, stmf_data_buf_t *dbuf,
375     uint32_t iof);
376 void stmf_send_status_done(scsi_task_t *task, stmf_status_t s, uint32_t iof);
377 void stmf_task_lu_done(scsi_task_t *task);
378 void stmf_abort(int abort_cmd, scsi_task_t *task, stmf_status_t s, void *arg);
379 void stmf_task_lu_aborted(scsi_task_t *task, stmf_status_t s, uint32_t iof);
380 void stmf_task_lport_aborted(scsi_task_t *task, stmf_status_t s, uint32_t iof);
381 stmf_status_t stmf_task_poll_lu(scsi_task_t *task, uint32_t timeout);
382 stmf_status_t stmf_task_poll_lport(scsi_task_t *task, uint32_t timeout);
383 stmf_status_t stmf_ctl(int cmd, void *obj, void *arg);
384 stmf_status_t stmf_register_itl_handle(struct stmf_lu *lu, uint8_t *lun,
385     struct stmf_scsi_session *ss, uint64_t session_id, void *itl_handle);
386 stmf_status_t stmf_deregister_itl_handle(struct stmf_lu *lu, uint8_t *lun,
387     struct stmf_scsi_session *ss, uint64_t session_id, void *itl_handle);
388 stmf_status_t stmf_deregister_all_lu_itl_handles(struct stmf_lu *lu);
389 stmf_status_t stmf_get_itl_handle(struct stmf_lu *lu, uint8_t *lun,
390     struct stmf_scsi_session *ss, uint64_t session_id, void **itl_handle_retp);
391 stmf_data_buf_t *stmf_handle_to_buf(scsi_task_t *task, uint8_t h);
392 stmf_status_t stmf_lu_add_event(struct stmf_lu *lu, int eventid);
393 stmf_status_t stmf_lu_remove_event(struct stmf_lu *lu, int eventid);
394 stmf_status_t stmf_lport_add_event(struct stmf_local_port *lport, int eventid);
395 stmf_status_t stmf_lport_remove_event(struct stmf_local_port *lport,
396     int eventid);
397 void stmf_wwn_to_devid_desc(struct scsi_devid_desc *sdid, uint8_t *wwn,
398     uint8_t protocol_id);
399 stmf_status_t stmf_scsilib_uniq_lu_id(uint32_t company_id,
400     struct scsi_devid_desc *lu_id);
401 stmf_status_t stmf_scsilib_uniq_lu_id2(uint32_t company_id, uint32_t host_id,
402     struct scsi_devid_desc *lu_id);
403 void stmf_scsilib_send_status(scsi_task_t *task, uint8_t st, uint32_t saa);
404 uint32_t stmf_scsilib_prepare_vpd_page83(scsi_task_t *task, uint8_t *page,
405 		uint32_t page_len, uint8_t byte0, uint32_t vpd_mask);
406 uint16_t stmf_scsilib_get_lport_rtid(struct scsi_devid_desc *devid);
407 struct scsi_devid_desc *stmf_scsilib_get_devid_desc(uint16_t rtpid);
408 void stmf_scsilib_handle_report_tpgs(scsi_task_t *task, stmf_data_buf_t *dbuf);
409 void stmf_scsilib_handle_task_mgmt(scsi_task_t *task);
410 
411 #ifdef	__cplusplus
412 }
413 #endif
414 
415 #endif	/* _STMF_H */
416