1a6d42e7dSPeter Dunlap /*
2a6d42e7dSPeter Dunlap  * CDDL HEADER START
3a6d42e7dSPeter Dunlap  *
4a6d42e7dSPeter Dunlap  * The contents of this file are subject to the terms of the
5a6d42e7dSPeter Dunlap  * Common Development and Distribution License (the "License").
6a6d42e7dSPeter Dunlap  * You may not use this file except in compliance with the License.
7a6d42e7dSPeter Dunlap  *
8a6d42e7dSPeter Dunlap  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9a6d42e7dSPeter Dunlap  * or http://www.opensolaris.org/os/licensing.
10a6d42e7dSPeter Dunlap  * See the License for the specific language governing permissions
11a6d42e7dSPeter Dunlap  * and limitations under the License.
12a6d42e7dSPeter Dunlap  *
13a6d42e7dSPeter Dunlap  * When distributing Covered Code, include this CDDL HEADER in each
14a6d42e7dSPeter Dunlap  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15a6d42e7dSPeter Dunlap  * If applicable, add the following below this CDDL HEADER, with the
16a6d42e7dSPeter Dunlap  * fields enclosed by brackets "[]" replaced with your own identifying
17a6d42e7dSPeter Dunlap  * information: Portions Copyright [yyyy] [name of copyright owner]
18a6d42e7dSPeter Dunlap  *
19a6d42e7dSPeter Dunlap  * CDDL HEADER END
20a6d42e7dSPeter Dunlap  */
21*483b029bSYuri Pankov 
22a6d42e7dSPeter Dunlap /*
23d618d68dSPriya Krishnan  * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
24*483b029bSYuri Pankov  * Copyright 2014 Nexenta Systems, Inc.  All rights reserved.
25a6d42e7dSPeter Dunlap  */
26*483b029bSYuri Pankov 
27a6d42e7dSPeter Dunlap #ifndef _ISCSIT_H_
28a6d42e7dSPeter Dunlap #define	_ISCSIT_H_
29a6d42e7dSPeter Dunlap 
30a6d42e7dSPeter Dunlap #include <sys/iscsit/iscsi_if.h>
31a6d42e7dSPeter Dunlap #include <sys/iscsit/iscsit_common.h>
32a6d42e7dSPeter Dunlap 
334558d122SViswanathan Kannappan #include "iscsit_authclient.h"
344558d122SViswanathan Kannappan 
35a6d42e7dSPeter Dunlap /*
36a6d42e7dSPeter Dunlap  * For some reason iscsi_protocol.h lists the max version as "0x02" and the
37a6d42e7dSPeter Dunlap  * min version as "0x00".  RFC3720 clearly states that the current version
38a6d42e7dSPeter Dunlap  * number is 0x00 so that is what we will use.
39a6d42e7dSPeter Dunlap  */
40a6d42e7dSPeter Dunlap #define	ISCSIT_MIN_VERSION			0x00
41a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_VERSION			0x00
42d618d68dSPriya Krishnan #define	ISCSIT_MAX_CONNECTIONS			32 /* MC/S support  */
43a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_RECV_DATA_SEGMENT_LENGTH	(32*1024)
44c050a449SPeter Gill #define	ISCSIT_MAX_BURST_LENGTH			(1024*1024)
45a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_FIRST_BURST_LENGTH		ISCSI_DEFAULT_FIRST_BURST_LENGTH
46c050a449SPeter Gill #define	ISCSIT_MAX_TIME2WAIT			ISCSI_MAX_TIME2WAIT
47a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_TIME2RETAIN			ISCSI_DEFAULT_TIME_TO_RETAIN
48a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_OUTSTANDING_R2T		ISCSI_DEFAULT_MAX_OUT_R2T
49a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_ERROR_RECOVERY_LEVEL		0
5056261083SCharles Ting #define	ISCSIT_MAX_OUTSTANDING_UNEXPECTED_PDUS	0
51a6d42e7dSPeter Dunlap 
52a6d42e7dSPeter Dunlap #define	ISCSIT_DEFAULT_TPG	"iscsit-default-tpg"
53a6d42e7dSPeter Dunlap #define	ISCSIT_DEFAULT_TPGT	1
54a6d42e7dSPeter Dunlap 
55a6d42e7dSPeter Dunlap #define	ISCSI_MAX_TSIH		0xffff
56a6d42e7dSPeter Dunlap #define	ISCSI_UNSPEC_TSIH	0
57a6d42e7dSPeter Dunlap 
58a6d42e7dSPeter Dunlap /* Max targets per system */
59a6d42e7dSPeter Dunlap #define	ISCSIT_MAX_TARGETS	1024
60a6d42e7dSPeter Dunlap 
61d618d68dSPriya Krishnan #define	ISCSIT_MAX_WINDOW	1024
62d618d68dSPriya Krishnan #define	ISCSIT_RXPDU_QUEUE_LEN	2048
63d618d68dSPriya Krishnan 
64d618d68dSPriya Krishnan #define	ISCSIT_CMDSN_LT_EXPCMDSN	-1
65d618d68dSPriya Krishnan #define	ISCSIT_CMDSN_EQ_EXPCMDSN	1
66d618d68dSPriya Krishnan #define	ISCSIT_CMDSN_GT_EXPCMDSN	0
67d618d68dSPriya Krishnan /*
68d618d68dSPriya Krishnan  * MC/S: A timeout is maintained to recover from lost CmdSN (holes in the
69d618d68dSPriya Krishnan  * CmdSN ordering). When the timeout is reached, the ExpCmdSN is advanced
70d618d68dSPriya Krishnan  * past the hole to continue processing the queued commands. This value is
71d618d68dSPriya Krishnan  * system-tunable (volatile rxpdu_queue_threshold) and should be in the
72d618d68dSPriya Krishnan  * range from 5 to 30 seconds.
73d618d68dSPriya Krishnan  */
74d618d68dSPriya Krishnan #define	ISCSIT_RXPDU_QUEUE_THRESHOLD		5	/* 5 seconds */
75d618d68dSPriya Krishnan #define	ISCSIT_RXPDU_QUEUE_MONITOR_INTERVAL	5	/* 5 seconds */
76d618d68dSPriya Krishnan 
77a6d42e7dSPeter Dunlap /* Time in seconds to wait between calls to stmf_deregister_local_port */
78a6d42e7dSPeter Dunlap #define	TGT_DEREG_RETRY_SECONDS	1
79a6d42e7dSPeter Dunlap 
80a6d42e7dSPeter Dunlap #define	ISCSIT_GLOBAL_LOCK(rw) rw_enter(&iscsit_global.global_rwlock, (rw))
81a6d42e7dSPeter Dunlap #define	ISCSIT_GLOBAL_UNLOCK() rw_exit(&iscsit_global.global_rwlock)
82a6d42e7dSPeter Dunlap 
83d618d68dSPriya Krishnan /* Circular buffer to hold the out-of-order PDUs in MC/S */
84d618d68dSPriya Krishnan typedef struct {
85d618d68dSPriya Krishnan 	idm_pdu_t	*cb_buffer[ISCSIT_RXPDU_QUEUE_LEN];
86d618d68dSPriya Krishnan 	int		cb_num_elems;
87d618d68dSPriya Krishnan } iscsit_cbuf_t;
88d618d68dSPriya Krishnan 
89a6d42e7dSPeter Dunlap /*
90a6d42e7dSPeter Dunlap  * Used for serial number arithmetic (RFC 1982)
91a6d42e7dSPeter Dunlap  */
92a6d42e7dSPeter Dunlap #define	ISCSIT_SNA32_CHECK	0x80000000
93a6d42e7dSPeter Dunlap 
94a6d42e7dSPeter Dunlap typedef struct {
95a6d42e7dSPeter Dunlap 	char		tpg_name[MAX_TPG_NAMELEN];
96a6d42e7dSPeter Dunlap 	kmutex_t	tpg_mutex;
97a6d42e7dSPeter Dunlap 	idm_refcnt_t	tpg_refcnt;
98a6d42e7dSPeter Dunlap 	int		tpg_online;
99a6d42e7dSPeter Dunlap 	avl_tree_t	tpg_portal_list;
100a6d42e7dSPeter Dunlap 	avl_node_t	tpg_global_ln;
101a6d42e7dSPeter Dunlap 	list_node_t	tpg_delete_ln;
102a6d42e7dSPeter Dunlap } iscsit_tpg_t;
103a6d42e7dSPeter Dunlap 
104a6d42e7dSPeter Dunlap #define	IS_DEFAULT_TPGT(TPGT) \
105a6d42e7dSPeter Dunlap 	(((TPGT) != NULL) && \
106a6d42e7dSPeter Dunlap 	    ((TPGT)->tpgt_tpg == iscsit_global.global_default_tpg))
107a6d42e7dSPeter Dunlap 
108a6d42e7dSPeter Dunlap typedef struct {
109a6d42e7dSPeter Dunlap 	iscsit_tpg_t	*tpgt_tpg;
110a6d42e7dSPeter Dunlap 	idm_refcnt_t	tpgt_refcnt;
111a6d42e7dSPeter Dunlap 	avl_node_t	tpgt_tgt_ln;
112a6d42e7dSPeter Dunlap 	list_node_t	tpgt_delete_ln;
113a6d42e7dSPeter Dunlap 	uint16_t	tpgt_tag;
114a6d42e7dSPeter Dunlap 	boolean_t	tpgt_needs_tpg_offline;
115a6d42e7dSPeter Dunlap } iscsit_tpgt_t;
116a6d42e7dSPeter Dunlap 
117a6d42e7dSPeter Dunlap typedef struct {
118a6d42e7dSPeter Dunlap 	struct sockaddr_storage portal_addr;
119a6d42e7dSPeter Dunlap 	int			portal_online;
120a6d42e7dSPeter Dunlap 	idm_refcnt_t		portal_refcnt;
121a6d42e7dSPeter Dunlap 	avl_node_t		portal_tpg_ln;
122a6d42e7dSPeter Dunlap 	iscsit_tpg_t		*portal_tpg;
123a6d42e7dSPeter Dunlap 	idm_svc_t		*portal_svc;
124e42a0851Speter dunlap 	boolean_t		portal_default;
125e42a0851Speter dunlap 	void			*portal_isns;
126a6d42e7dSPeter Dunlap } iscsit_portal_t;
127a6d42e7dSPeter Dunlap 
128a6d42e7dSPeter Dunlap 
129a6d42e7dSPeter Dunlap /* Target states and events, update iscsit_ts_name table whenever modified */
130a6d42e7dSPeter Dunlap typedef enum {
131a6d42e7dSPeter Dunlap 	TS_UNDEFINED = 0,
132a6d42e7dSPeter Dunlap 	TS_CREATED,
133a6d42e7dSPeter Dunlap 	TS_ONLINING,
134a6d42e7dSPeter Dunlap 	TS_ONLINE,
135a6d42e7dSPeter Dunlap 	TS_STMF_ONLINE,
136a6d42e7dSPeter Dunlap 	TS_DELETING_NEED_OFFLINE,
137a6d42e7dSPeter Dunlap 	TS_OFFLINING,
138a6d42e7dSPeter Dunlap 	TS_OFFLINE,
139a6d42e7dSPeter Dunlap 	TS_STMF_OFFLINE,
140a6d42e7dSPeter Dunlap 	TS_DELETING_STMF_DEREG,
141a6d42e7dSPeter Dunlap 	TS_DELETING_STMF_DEREG_FAIL,
142a6d42e7dSPeter Dunlap 	TS_DELETING,
143a6d42e7dSPeter Dunlap 	TS_MAX_STATE
144a6d42e7dSPeter Dunlap } iscsit_tgt_state_t;
145a6d42e7dSPeter Dunlap 
146a6d42e7dSPeter Dunlap #ifdef ISCSIT_TGT_SM_STRINGS
147a6d42e7dSPeter Dunlap static const char *iscsit_ts_name[TS_MAX_STATE+1] = {
148a6d42e7dSPeter Dunlap 	"TS_UNDEFINED",
149a6d42e7dSPeter Dunlap 	"TS_CREATED",
150a6d42e7dSPeter Dunlap 	"TS_ONLINING",
151a6d42e7dSPeter Dunlap 	"TS_ONLINE",
152a6d42e7dSPeter Dunlap 	"TS_STMF_ONLINE",
153a6d42e7dSPeter Dunlap 	"TS_DELETING_NEED_OFFLINE",
154a6d42e7dSPeter Dunlap 	"TS_OFFLINING",
155a6d42e7dSPeter Dunlap 	"TS_OFFLINE",
156a6d42e7dSPeter Dunlap 	"TS_STMF_OFFLINE",
157a6d42e7dSPeter Dunlap 	"TS_DELETING_STMF_DEREG",
158a6d42e7dSPeter Dunlap 	"TS_DELETING_STMF_DEREG_FAIL",
159a6d42e7dSPeter Dunlap 	"TS_DELETING",
160a6d42e7dSPeter Dunlap 	"TS_MAX_STATE"
161a6d42e7dSPeter Dunlap };
162a6d42e7dSPeter Dunlap #endif
163a6d42e7dSPeter Dunlap 
164a6d42e7dSPeter Dunlap typedef enum {
165a6d42e7dSPeter Dunlap 	TE_UNDEFINED = 0,
166a6d42e7dSPeter Dunlap 	TE_STMF_ONLINE_REQ,
167a6d42e7dSPeter Dunlap 	TE_ONLINE_SUCCESS,
168a6d42e7dSPeter Dunlap 	TE_ONLINE_FAIL,
169a6d42e7dSPeter Dunlap 	TE_STMF_ONLINE_COMPLETE_ACK,
170a6d42e7dSPeter Dunlap 	TE_STMF_OFFLINE_REQ,
171a6d42e7dSPeter Dunlap 	TE_OFFLINE_COMPLETE,
172a6d42e7dSPeter Dunlap 	TE_STMF_OFFLINE_COMPLETE_ACK,
173a6d42e7dSPeter Dunlap 	TE_DELETE,
174a6d42e7dSPeter Dunlap 	TE_STMF_DEREG_SUCCESS,
175a6d42e7dSPeter Dunlap 	TE_STMF_DEREG_FAIL,
176a6d42e7dSPeter Dunlap 	TE_STMF_DEREG_RETRY,
177a6d42e7dSPeter Dunlap 	TE_WAIT_REF_COMPLETE,
178a6d42e7dSPeter Dunlap 	TE_MAX_EVENT
179a6d42e7dSPeter Dunlap } iscsit_tgt_event_t;
180a6d42e7dSPeter Dunlap 
181a6d42e7dSPeter Dunlap #ifdef ISCSIT_TGT_SM_STRINGS
182a6d42e7dSPeter Dunlap static const char *iscsit_te_name[TE_MAX_EVENT+1] = {
183a6d42e7dSPeter Dunlap 	"TE_UNDEFINED",
184a6d42e7dSPeter Dunlap 	"TE_STMF_ONLINE_REQ",
185a6d42e7dSPeter Dunlap 	"TE_ONLINE_SUCCESS",
186a6d42e7dSPeter Dunlap 	"TE_ONLINE_FAIL",
187a6d42e7dSPeter Dunlap 	"TE_STMF_ONLINE_COMPLETE_ACK",
188a6d42e7dSPeter Dunlap 	"TE_STMF_OFFLINE_REQ",
189a6d42e7dSPeter Dunlap 	"TE_OFFLINE_COMPLETE",
190a6d42e7dSPeter Dunlap 	"TE_STMF_OFFLINE_COMPLETE_ACK",
191a6d42e7dSPeter Dunlap 	"TE_DELETE",
192a6d42e7dSPeter Dunlap 	"TE_STMF_DEREG_SUCCESS",
193a6d42e7dSPeter Dunlap 	"TE_STMF_DEREG_FAIL",
194a6d42e7dSPeter Dunlap 	"TE_STMF_DEREG_RETRY",
195a6d42e7dSPeter Dunlap 	"TE_WAIT_REF_COMPLETE",
196a6d42e7dSPeter Dunlap 	"TE_MAX_EVENT"
197a6d42e7dSPeter Dunlap };
198a6d42e7dSPeter Dunlap #endif
199a6d42e7dSPeter Dunlap 
200a6d42e7dSPeter Dunlap typedef struct {
201a6d42e7dSPeter Dunlap 	char			*target_name;
202a6d42e7dSPeter Dunlap 	nvlist_t		*target_props;
203a6d42e7dSPeter Dunlap 	kmutex_t		target_mutex;
204a6d42e7dSPeter Dunlap 	idm_refcnt_t		target_refcnt;
205a6d42e7dSPeter Dunlap 	idm_refcnt_t		target_sess_refcnt;
206a6d42e7dSPeter Dunlap 	avl_tree_t		target_tpgt_list;
207a6d42e7dSPeter Dunlap 	avl_tree_t		target_sess_list;
208a6d42e7dSPeter Dunlap 	avl_node_t		target_global_ln;
209a6d42e7dSPeter Dunlap 	avl_node_t		target_global_deleted_ln;
210a6d42e7dSPeter Dunlap 	/* STMF lport == iSCSI target */
211a6d42e7dSPeter Dunlap 	scsi_devid_desc_t	*target_devid;
212a6d42e7dSPeter Dunlap 	stmf_local_port_t	*target_stmf_lport;
213a6d42e7dSPeter Dunlap 	uint8_t			target_stmf_lport_registered;
214a6d42e7dSPeter Dunlap 
215a6d42e7dSPeter Dunlap 	/* Target state */
216a6d42e7dSPeter Dunlap 	boolean_t		target_sm_busy;
217a6d42e7dSPeter Dunlap 	boolean_t		target_deleting;
218a6d42e7dSPeter Dunlap 	iscsit_tgt_state_t	target_state;
219a6d42e7dSPeter Dunlap 	iscsit_tgt_state_t	target_last_state;
220a6d42e7dSPeter Dunlap 	sm_audit_buf_t		target_state_audit;
221a6d42e7dSPeter Dunlap 	list_t			target_events;
222a6d42e7dSPeter Dunlap 	uint64_t		target_generation;
223a6d42e7dSPeter Dunlap } iscsit_tgt_t;
224a6d42e7dSPeter Dunlap 
225a6d42e7dSPeter Dunlap typedef struct {
226a6d42e7dSPeter Dunlap 	char			ini_name[MAX_ISCSI_NODENAMELEN];
227a6d42e7dSPeter Dunlap 	nvlist_t		*ini_props;
228a6d42e7dSPeter Dunlap 	avl_node_t		ini_global_ln;
229a6d42e7dSPeter Dunlap } iscsit_ini_t;
230a6d42e7dSPeter Dunlap 
231a6d42e7dSPeter Dunlap /*
232a6d42e7dSPeter Dunlap  * iSCSI Auth Information
233a6d42e7dSPeter Dunlap  */
234a6d42e7dSPeter Dunlap typedef struct conn_auth {
23530e7468fSPeter Dunlap 	char			ca_tgt_chapuser[iscsitAuthStringMaxLength];
23630e7468fSPeter Dunlap 	uint8_t			ca_tgt_chapsecret[iscsitAuthStringMaxLength];
237a6d42e7dSPeter Dunlap 	int			ca_tgt_chapsecretlen;
238a6d42e7dSPeter Dunlap 
23930e7468fSPeter Dunlap 	char			ca_ini_chapuser[iscsitAuthStringMaxLength];
24030e7468fSPeter Dunlap 	uint8_t			ca_ini_chapsecret[iscsitAuthStringMaxLength];
241a6d42e7dSPeter Dunlap 	int			ca_ini_chapsecretlen;
242a6d42e7dSPeter Dunlap 
243a6d42e7dSPeter Dunlap 	/* RADIUS authentication information   	*/
244a6d42e7dSPeter Dunlap 	boolean_t		ca_use_radius;
245a6d42e7dSPeter Dunlap 	struct sockaddr_storage	ca_radius_server;
24630e7468fSPeter Dunlap 	uint8_t			ca_radius_secret[iscsitAuthStringMaxLength];
247a6d42e7dSPeter Dunlap 	int			ca_radius_secretlen;
248a6d42e7dSPeter Dunlap 
249a6d42e7dSPeter Dunlap 	/* authentication method list */
25030e7468fSPeter Dunlap 	iscsit_auth_method_t	ca_method_valid_list[iscsitAuthMethodMaxCount];
251a6d42e7dSPeter Dunlap 
252a6d42e7dSPeter Dunlap 	/* Target alias */
253a6d42e7dSPeter Dunlap 	char			ca_tgt_alias[MAX_ISCSI_NODENAMELEN];
254a6d42e7dSPeter Dunlap } conn_auth_t;
255a6d42e7dSPeter Dunlap 
256a6d42e7dSPeter Dunlap /*
257a6d42e7dSPeter Dunlap  * We have three state machines (so far) between the IDM connection state
258a6d42e7dSPeter Dunlap  * machine, the session state machine, and the login state machine.  All
259a6d42e7dSPeter Dunlap  * of these states have some concept of "full feature mode".  It's going
260a6d42e7dSPeter Dunlap  * to be obnoxious if we use a mixture of these "ffp" representations
261a6d42e7dSPeter Dunlap  * since it will be difficult to ensure the three state machines
262a6d42e7dSPeter Dunlap  * transition at exactly the same time.  We should drive decisions that
263a6d42e7dSPeter Dunlap  * depend on FFP from the IDM state machine which is actually snooping
264a6d42e7dSPeter Dunlap  * the iSCSI PDU's and will always transition at the correct time.
265a6d42e7dSPeter Dunlap  *
266a6d42e7dSPeter Dunlap  * A consequence of this approach is that there is a window just after
267a6d42e7dSPeter Dunlap  * login completes where we may get a SCSI request but the session
268a6d42e7dSPeter Dunlap  * or login state machine has not quite transitioned to "FFP".  Whether
269a6d42e7dSPeter Dunlap  * this is a problem depends on how we use those state machines.  This
270a6d42e7dSPeter Dunlap  * is what we should use them for:
271a6d42e7dSPeter Dunlap  *
272a6d42e7dSPeter Dunlap  * IDM Connection state machine - Decisions related to command processing
273a6d42e7dSPeter Dunlap  * including whether a connection is in FFP
274a6d42e7dSPeter Dunlap  *
275a6d42e7dSPeter Dunlap  * Session state machine - Summarize the state of all available connections
276a6d42e7dSPeter Dunlap  * for the purposes of ERL1, ERL2 and MC/S.  A session in LOGGED_IN state
277a6d42e7dSPeter Dunlap  * should always have at least one FFP connection but there may be a brief
278a6d42e7dSPeter Dunlap  * window where a session in ACTIVE might have one or more FFP connections
279a6d42e7dSPeter Dunlap  * even though ACTIVE is not strictly an FFP state according to the RFC.
280a6d42e7dSPeter Dunlap  *
281a6d42e7dSPeter Dunlap  * Login state machine -- drive the login process, collect negotiated
282a6d42e7dSPeter Dunlap  * parameters.  Another side effect of this approach is that we may get
283a6d42e7dSPeter Dunlap  * the "notify ffp" callback from the IDM connection state machine before
284a6d42e7dSPeter Dunlap  * the login state machine has actually transitioned to FFP state.
285a6d42e7dSPeter Dunlap  */
286a6d42e7dSPeter Dunlap 
287a6d42e7dSPeter Dunlap struct iscsit_conn_s;
288a6d42e7dSPeter Dunlap 
289a6d42e7dSPeter Dunlap /* Update iscsit_ss_name table whenever session states are modified */
290a6d42e7dSPeter Dunlap typedef enum {
291a6d42e7dSPeter Dunlap 	SS_UNDEFINED = 0,
292a6d42e7dSPeter Dunlap 	SS_Q1_FREE,
293a6d42e7dSPeter Dunlap 	SS_Q2_ACTIVE,
294a6d42e7dSPeter Dunlap 	SS_Q3_LOGGED_IN,
295a6d42e7dSPeter Dunlap 	SS_Q4_FAILED,
296a6d42e7dSPeter Dunlap 	SS_Q5_CONTINUE,
297a6d42e7dSPeter Dunlap 	SS_Q6_DONE,
298a6d42e7dSPeter Dunlap 	SS_Q7_ERROR,
299a6d42e7dSPeter Dunlap 	/* Add new session states above SS_MAX_STATE */
300a6d42e7dSPeter Dunlap 	SS_MAX_STATE
301a6d42e7dSPeter Dunlap } iscsit_session_state_t;
302a6d42e7dSPeter Dunlap 
303a6d42e7dSPeter Dunlap #ifdef ISCSIT_SESS_SM_STRINGS
304a6d42e7dSPeter Dunlap /* An array of state text values, for use in logging state transitions */
305a6d42e7dSPeter Dunlap static const char *iscsit_ss_name[SS_MAX_STATE+1] = {
306a6d42e7dSPeter Dunlap 	"SS_UNDEFINED",
307a6d42e7dSPeter Dunlap 	"SS_Q1_FREE",
308a6d42e7dSPeter Dunlap 	"SS_Q2_ACTIVE",
309a6d42e7dSPeter Dunlap 	"SS_Q3_LOGGED_IN",
310a6d42e7dSPeter Dunlap 	"SS_Q4_FAILED",
311a6d42e7dSPeter Dunlap 	"SS_Q5_CONTINUE",
312a6d42e7dSPeter Dunlap 	"SS_Q6_DONE",
313a6d42e7dSPeter Dunlap 	"SS_Q7_ERROR",
314a6d42e7dSPeter Dunlap 	"SS_MAX_STATE"
315a6d42e7dSPeter Dunlap };
316a6d42e7dSPeter Dunlap #endif
317a6d42e7dSPeter Dunlap 
318a6d42e7dSPeter Dunlap /* Update iscsit_se_name table whenever session events are modified */
319a6d42e7dSPeter Dunlap typedef enum {
320a6d42e7dSPeter Dunlap 	SE_UNDEFINED = 0,
321a6d42e7dSPeter Dunlap 	SE_CONN_IN_LOGIN,	/* From login state machine */
322a6d42e7dSPeter Dunlap 	SE_CONN_LOGGED_IN,	/* FFP enabled client notification */
323a6d42e7dSPeter Dunlap 	SE_CONN_FFP_FAIL,	/* FFP disabled client notification */
324a6d42e7dSPeter Dunlap 	SE_CONN_FFP_DISABLE,	/* FFP disabled client notification */
325a6d42e7dSPeter Dunlap 	SE_CONN_FAIL,		/* Conn destroy client notification */
326a6d42e7dSPeter Dunlap 	SE_SESSION_CLOSE,	/* FFP disabled client notification */
327a6d42e7dSPeter Dunlap 	SE_SESSION_REINSTATE,	/* From login state machine */
328a6d42e7dSPeter Dunlap 	SE_SESSION_TIMEOUT,	/* Internal */
329a6d42e7dSPeter Dunlap 	SE_SESSION_CONTINUE,	/* From login state machine */
330a6d42e7dSPeter Dunlap 	SE_SESSION_CONTINUE_FAIL, /* From login state machine? */
331a6d42e7dSPeter Dunlap 	/* Add new events above SE_MAX_EVENT */
332a6d42e7dSPeter Dunlap 	SE_MAX_EVENT
333a6d42e7dSPeter Dunlap } iscsit_session_event_t;
334a6d42e7dSPeter Dunlap 
335a6d42e7dSPeter Dunlap #ifdef ISCSIT_SESS_SM_STRINGS
336a6d42e7dSPeter Dunlap /* An array of event text values, for use in logging events */
337a6d42e7dSPeter Dunlap static const char *iscsit_se_name[SE_MAX_EVENT+1] = {
338a6d42e7dSPeter Dunlap 	"SE_UNDEFINED",
339a6d42e7dSPeter Dunlap 	"SE_CONN_IN_LOGIN",
340a6d42e7dSPeter Dunlap 	"SE_CONN_LOGGED_IN",
341a6d42e7dSPeter Dunlap 	"SE_CONN_FFP_FAIL",
342a6d42e7dSPeter Dunlap 	"SE_CONN_FFP_DISABLE",
343a6d42e7dSPeter Dunlap 	"SE_CONN_FAIL",
344a6d42e7dSPeter Dunlap 	"SE_SESSION_CLOSE",
345a6d42e7dSPeter Dunlap 	"SE_SESSION_REINSTATE",
346a6d42e7dSPeter Dunlap 	"SE_SESSION_TIMEOUT",
347a6d42e7dSPeter Dunlap 	"SE_SESSION_CONTINUE",
348a6d42e7dSPeter Dunlap 	"SE_SESSION_CONTINUE_FAIL",
349a6d42e7dSPeter Dunlap 	"SE_MAX_EVENT"
350a6d42e7dSPeter Dunlap };
351a6d42e7dSPeter Dunlap #endif
352a6d42e7dSPeter Dunlap 
353a6d42e7dSPeter Dunlap /*
354a6d42e7dSPeter Dunlap  * Set in ist_tgt after iscsit_tgt_unbind_sess to differentiate an unbound
355a6d42e7dSPeter Dunlap  * session from a discovery session.
356a6d42e7dSPeter Dunlap  */
357a6d42e7dSPeter Dunlap #define	SESS_UNBOUND_FROM_TGT	-1
358a6d42e7dSPeter Dunlap 
359a6d42e7dSPeter Dunlap typedef struct {
360a6d42e7dSPeter Dunlap 	stmf_scsi_session_t	*ist_stmf_sess;
361a6d42e7dSPeter Dunlap 	stmf_local_port_t	*ist_lport;
362a6d42e7dSPeter Dunlap 	iscsit_tgt_t		*ist_tgt;
363a6d42e7dSPeter Dunlap 	idm_refcnt_t		ist_refcnt;
364a6d42e7dSPeter Dunlap 	kmem_cache_t		*ist_task_cache;
365d618d68dSPriya Krishnan 	kmutex_t		ist_sn_mutex;
366a6d42e7dSPeter Dunlap 	kmutex_t		ist_mutex;
367a6d42e7dSPeter Dunlap 	kcondvar_t		ist_cv;
368a6d42e7dSPeter Dunlap 	iscsit_session_state_t	ist_state;
369a6d42e7dSPeter Dunlap 	iscsit_session_state_t	ist_last_state;
370a6d42e7dSPeter Dunlap 	sm_audit_buf_t		ist_state_audit;
371a6d42e7dSPeter Dunlap 	boolean_t		ist_sm_busy;
372a6d42e7dSPeter Dunlap 	boolean_t		ist_sm_complete;
373a6d42e7dSPeter Dunlap 	boolean_t		ist_admin_close;
374a6d42e7dSPeter Dunlap 	list_t			ist_events;
375a6d42e7dSPeter Dunlap 	int			ist_conn_count;
376a6d42e7dSPeter Dunlap 	int			ist_ffp_conn_count;
377a6d42e7dSPeter Dunlap 	struct iscsit_conn_s	*ist_failed_conn;
378a6d42e7dSPeter Dunlap 	timeout_id_t		ist_state_timeout;
379a6d42e7dSPeter Dunlap 	list_t			ist_conn_list;
380a6d42e7dSPeter Dunlap 	avl_node_t		ist_tgt_ln;
381a6d42e7dSPeter Dunlap 	char			*ist_initiator_name;
382a6d42e7dSPeter Dunlap 	char			*ist_initiator_alias;
383a6d42e7dSPeter Dunlap 	char			*ist_target_name;
384a6d42e7dSPeter Dunlap 	char			*ist_target_alias;
385a6d42e7dSPeter Dunlap 	uint8_t			ist_isid[ISCSI_ISID_LEN];
386a6d42e7dSPeter Dunlap 	uint16_t		ist_tsih;
387a6d42e7dSPeter Dunlap 	uint16_t		ist_tpgt_tag;
388a6d42e7dSPeter Dunlap 	uint32_t		ist_expcmdsn;
389a6d42e7dSPeter Dunlap 	uint32_t		ist_maxcmdsn;
39072cf3143Speter dunlap 	avl_tree_t		ist_task_list;
391d618d68dSPriya Krishnan 	iscsit_cbuf_t		*ist_rxpdu_queue;
392a6d42e7dSPeter Dunlap } iscsit_sess_t;
393a6d42e7dSPeter Dunlap 
394a6d42e7dSPeter Dunlap /* Update iscsit_ils_name table whenever login states are modified */
395a6d42e7dSPeter Dunlap typedef enum {
396a6d42e7dSPeter Dunlap 	ILS_UNDEFINED = 0,
397a6d42e7dSPeter Dunlap 	ILS_LOGIN_INIT,
398a6d42e7dSPeter Dunlap 	ILS_LOGIN_WAITING,	/* Waiting for more login PDU's */
399a6d42e7dSPeter Dunlap 	ILS_LOGIN_PROCESSING,	/* Processing login request */
400a6d42e7dSPeter Dunlap 	ILS_LOGIN_RESPONDING,	/* Sending login response */
401a6d42e7dSPeter Dunlap 	ILS_LOGIN_RESPONDED,	/* Sent login response (no trans. to FFP) */
402a6d42e7dSPeter Dunlap 	ILS_LOGIN_FFP,		/* Sending last login PDU for final response */
403a6d42e7dSPeter Dunlap 	ILS_LOGIN_DONE,		/* Last login PDU sent (so we can free it) */
404a6d42e7dSPeter Dunlap 	ILS_LOGIN_ERROR,	/* Login error, login failed */
405a6d42e7dSPeter Dunlap 	/* Add new login states above ILS_MAX_STATE */
406a6d42e7dSPeter Dunlap 	ILS_MAX_STATE
407a6d42e7dSPeter Dunlap } iscsit_login_state_t;
408a6d42e7dSPeter Dunlap 
409a6d42e7dSPeter Dunlap #ifdef ISCSIT_LOGIN_SM_STRINGS
410a6d42e7dSPeter Dunlap /* An array of login state text values, for use in logging login progress */
411a6d42e7dSPeter Dunlap static const char *iscsit_ils_name[ILS_MAX_STATE+1] = {
412a6d42e7dSPeter Dunlap 	"ILS_UNDEFINED",
413a6d42e7dSPeter Dunlap 	"ILS_LOGIN_INIT",
414a6d42e7dSPeter Dunlap 	"ILS_LOGIN_WAITING",
415a6d42e7dSPeter Dunlap 	"ILS_LOGIN_PROCESSING",
416a6d42e7dSPeter Dunlap 	"ILS_LOGIN_RESPONDING",
417a6d42e7dSPeter Dunlap 	"ILS_LOGIN_RESPONDED",
418a6d42e7dSPeter Dunlap 	"ILS_LOGIN_FFP",
419a6d42e7dSPeter Dunlap 	"ILS_LOGIN_DONE",
420a6d42e7dSPeter Dunlap 	"ILS_LOGIN_ERROR",
421a6d42e7dSPeter Dunlap 	"ILS_MAX_STATE"
422a6d42e7dSPeter Dunlap };
423a6d42e7dSPeter Dunlap #endif
424a6d42e7dSPeter Dunlap 
425a6d42e7dSPeter Dunlap /* Update iscsit_ile_name table whenever login events are modified */
426a6d42e7dSPeter Dunlap typedef enum {
427a6d42e7dSPeter Dunlap 	ILE_UNDEFINED = 0,
428a6d42e7dSPeter Dunlap 	ILE_LOGIN_RCV,
429a6d42e7dSPeter Dunlap 	ILE_LOGIN_RESP_READY,
430a6d42e7dSPeter Dunlap 	ILE_LOGIN_FFP,
431a6d42e7dSPeter Dunlap 	ILE_LOGIN_RESP_COMPLETE,
432a6d42e7dSPeter Dunlap 	ILE_LOGIN_ERROR,
433a6d42e7dSPeter Dunlap 	ILE_LOGIN_CONN_ERROR,
434a6d42e7dSPeter Dunlap 	/* Add new login events above ILE_MAX_EVENT */
435a6d42e7dSPeter Dunlap 	ILE_MAX_EVENT
436a6d42e7dSPeter Dunlap } iscsit_login_event_t;
437a6d42e7dSPeter Dunlap 
438a6d42e7dSPeter Dunlap #ifdef ISCSIT_LOGIN_SM_STRINGS
439a6d42e7dSPeter Dunlap /* An array of login event text values, for use in logging login events */
440a6d42e7dSPeter Dunlap static const char *iscsit_ile_name[ILE_MAX_EVENT+1] = {
441a6d42e7dSPeter Dunlap 	"ILE_UNDEFINED",
442a6d42e7dSPeter Dunlap 	"ILE_LOGIN_RCV",
443a6d42e7dSPeter Dunlap 	"ILE_LOGIN_RESP_READY",
444a6d42e7dSPeter Dunlap 	"ILE_LOGIN_FFP",
445a6d42e7dSPeter Dunlap 	"ILE_LOGIN_RESP_COMPLETE",
446a6d42e7dSPeter Dunlap 	"ILE_LOGIN_ERROR",
447a6d42e7dSPeter Dunlap 	"ILE_LOGIN_CONN_ERROR",
448a6d42e7dSPeter Dunlap 	"ILE_MAX_EVENT"
449a6d42e7dSPeter Dunlap };
450a6d42e7dSPeter Dunlap #endif
451a6d42e7dSPeter Dunlap 
452a6d42e7dSPeter Dunlap typedef struct {
453a6d42e7dSPeter Dunlap 	uint32_t		op_initial_params_set:1,
454a6d42e7dSPeter Dunlap 				op_discovery_session:1,
455a6d42e7dSPeter Dunlap 				op_initial_r2t:1,
456a6d42e7dSPeter Dunlap 				op_immed_data:1,
457a6d42e7dSPeter Dunlap 				op_data_pdu_in_order:1,
45878264421SCharles Ting 				op_data_sequence_in_order:1,
45978264421SCharles Ting 				op_declarative_params_set:1;
460a6d42e7dSPeter Dunlap 	uint64_t		op_max_connections;
461a6d42e7dSPeter Dunlap 	uint64_t		op_max_recv_data_segment_length;
462a6d42e7dSPeter Dunlap 	uint64_t		op_max_burst_length;
463a6d42e7dSPeter Dunlap 	uint64_t		op_first_burst_length;
464a6d42e7dSPeter Dunlap 	uint64_t		op_default_time_2_wait;
465a6d42e7dSPeter Dunlap 	uint64_t		op_default_time_2_retain;
466a6d42e7dSPeter Dunlap 	uint64_t		op_max_outstanding_r2t;
467a6d42e7dSPeter Dunlap 	uint64_t		op_error_recovery_level;
468a6d42e7dSPeter Dunlap } iscsit_op_params_t;
469a6d42e7dSPeter Dunlap 
470a6d42e7dSPeter Dunlap typedef struct {
471a6d42e7dSPeter Dunlap 	iscsit_login_state_t 	icl_login_state;
472a6d42e7dSPeter Dunlap 	iscsit_login_state_t 	icl_login_last_state;
473a6d42e7dSPeter Dunlap 	sm_audit_buf_t		icl_state_audit;
474a6d42e7dSPeter Dunlap 	boolean_t		icl_busy;
475a6d42e7dSPeter Dunlap 	boolean_t		icl_login_complete;
476a6d42e7dSPeter Dunlap 	kmutex_t		icl_mutex;
477a6d42e7dSPeter Dunlap 	uint32_t		icl_login_itt;
478a6d42e7dSPeter Dunlap 	uint8_t			icl_login_csg;
479a6d42e7dSPeter Dunlap 	uint8_t			icl_login_nsg;
480a6d42e7dSPeter Dunlap 	boolean_t		icl_login_transit;
481a6d42e7dSPeter Dunlap 	conn_auth_t		icl_auth;
482a6d42e7dSPeter Dunlap 	iscsit_auth_client_t	icl_auth_client;
483a6d42e7dSPeter Dunlap 	int			icl_auth_pass;
484a6d42e7dSPeter Dunlap 	list_t			icl_login_events;
485a6d42e7dSPeter Dunlap 	list_t			icl_pdu_list;
486a6d42e7dSPeter Dunlap 	uint16_t		icl_tsih;
487a6d42e7dSPeter Dunlap 	uint8_t			icl_isid[ISCSI_ISID_LEN];
488a6d42e7dSPeter Dunlap 	uint32_t		icl_cmdsn;
489a6d42e7dSPeter Dunlap 	uint16_t		icl_tpgt_tag;
490a6d42e7dSPeter Dunlap 	char			*icl_target_name;
491a6d42e7dSPeter Dunlap 	char			*icl_target_alias;
492a6d42e7dSPeter Dunlap 	char			*icl_initiator_name;
493a6d42e7dSPeter Dunlap 	char			*icl_login_resp_buf;
494a6d42e7dSPeter Dunlap 	void			*icl_login_resp_itb; /* mult-pdu idm buf */
495a6d42e7dSPeter Dunlap 	int			icl_login_resp_len; /* For kmem_free */
496a6d42e7dSPeter Dunlap 	int			icl_login_resp_valid_len;
497a6d42e7dSPeter Dunlap 	uint8_t			icl_login_resp_err_class;
498a6d42e7dSPeter Dunlap 	uint8_t			icl_login_resp_err_detail;
499a6d42e7dSPeter Dunlap 	iscsi_login_rsp_hdr_t	*icl_login_resp_tmpl;
500a6d42e7dSPeter Dunlap 	nvlist_t		*icl_request_nvlist;
501a6d42e7dSPeter Dunlap 	nvlist_t		*icl_response_nvlist;
502a6d42e7dSPeter Dunlap 	nvlist_t		*icl_negotiated_values;
503a6d42e7dSPeter Dunlap } iscsit_conn_login_t;
504a6d42e7dSPeter Dunlap 
505a6d42e7dSPeter Dunlap #define	SET_LOGIN_ERROR(SLE_ICT, SLE_CLASS, SLE_DETAIL) \
506a6d42e7dSPeter Dunlap 	(SLE_ICT)->ict_login_sm.icl_login_resp_err_class = (SLE_CLASS); \
507a6d42e7dSPeter Dunlap 	(SLE_ICT)->ict_login_sm.icl_login_resp_err_detail = (SLE_DETAIL);
508a6d42e7dSPeter Dunlap 
509a6d42e7dSPeter Dunlap typedef struct iscsit_conn_s {
510a6d42e7dSPeter Dunlap 	idm_conn_t		*ict_ic;
511a6d42e7dSPeter Dunlap 	iscsit_sess_t		*ict_sess;
512a6d42e7dSPeter Dunlap 	kmutex_t		ict_mutex;
513a6d42e7dSPeter Dunlap 	idm_refcnt_t		ict_refcnt;
514a6d42e7dSPeter Dunlap 	idm_refcnt_t		ict_dispatch_refcnt;
515a6d42e7dSPeter Dunlap 	list_node_t		ict_sess_ln;
516a6d42e7dSPeter Dunlap 	iscsit_conn_login_t	ict_login_sm;
517a6d42e7dSPeter Dunlap 	iscsit_op_params_t	ict_op;
518a6d42e7dSPeter Dunlap 	uint16_t		ict_cid;
519a6d42e7dSPeter Dunlap 	uint32_t		ict_statsn;
52060220f10SPriya Krishnan 	kmutex_t		ict_statsn_mutex;
52172cf3143Speter dunlap 	uint32_t		ict_keepalive_ttt;
522a6d42e7dSPeter Dunlap 	struct iscsit_conn_s	*ict_reinstate_conn;
523a6d42e7dSPeter Dunlap 	uint32_t		ict_reinstating:1,
524a6d42e7dSPeter Dunlap 				ict_lost:1,
525a6d42e7dSPeter Dunlap 				ict_destroyed:1;
5264142b486SJames Moore 	/*
5274142b486SJames Moore 	 * Parameters for processing text commands
5284142b486SJames Moore 	 */
5294142b486SJames Moore 	char			*ict_text_rsp_buf;
5304142b486SJames Moore 	uint32_t		ict_text_rsp_len;
5314142b486SJames Moore 	uint32_t		ict_text_rsp_valid_len;
5324142b486SJames Moore 	uint32_t		ict_text_rsp_off;
5334142b486SJames Moore 	uint32_t		ict_text_req_itt;	/* from initiator */
5344142b486SJames Moore 	uint32_t		ict_text_rsp_ttt;
535a6d42e7dSPeter Dunlap } iscsit_conn_t;
536a6d42e7dSPeter Dunlap 
537a6d42e7dSPeter Dunlap #define	ICT_FLAGS_DISCOVERY	0x00000001
538a6d42e7dSPeter Dunlap 
539a6d42e7dSPeter Dunlap typedef struct {
540a6d42e7dSPeter Dunlap 	idm_buf_t		*ibuf_idm_buf;
541a6d42e7dSPeter Dunlap 	stmf_data_buf_t		*ibuf_stmf_buf;
542a6d42e7dSPeter Dunlap 	idm_pdu_t		*ibuf_immed_data_pdu;
543a6d42e7dSPeter Dunlap 	boolean_t		ibuf_is_immed;
544a6d42e7dSPeter Dunlap } iscsit_buf_t;
545a6d42e7dSPeter Dunlap 
546a6d42e7dSPeter Dunlap typedef struct {
547a6d42e7dSPeter Dunlap 	scsi_task_t		*it_stmf_task;
548a6d42e7dSPeter Dunlap 	idm_task_t		*it_idm_task;
549a6d42e7dSPeter Dunlap 	iscsit_buf_t		*it_immed_data;
550a6d42e7dSPeter Dunlap 	iscsit_conn_t		*it_ict;
551a6d42e7dSPeter Dunlap 	kmutex_t		it_mutex;
552a6d42e7dSPeter Dunlap 	idm_pdu_t		*it_tm_pdu;
553a6d42e7dSPeter Dunlap 	uint32_t		it_stmf_abort:1,
554a6d42e7dSPeter Dunlap 				it_aborted:1,
55572cf3143Speter dunlap 				it_active:1,
556a6d42e7dSPeter Dunlap 				it_tm_task:1,
557a6d42e7dSPeter Dunlap 				it_tm_responded:1;
558a6d42e7dSPeter Dunlap 	uint32_t		it_cmdsn;
559a6d42e7dSPeter Dunlap 	uint32_t		it_itt;
560a6d42e7dSPeter Dunlap 	uint32_t		it_ttt;
56172cf3143Speter dunlap 	avl_node_t		it_sess_ln;
562a6d42e7dSPeter Dunlap } iscsit_task_t;
563a6d42e7dSPeter Dunlap 
564a6d42e7dSPeter Dunlap typedef struct iscsit_isns_cfg {
565a6d42e7dSPeter Dunlap 	kmutex_t		isns_mutex;
566a6d42e7dSPeter Dunlap 	boolean_t		isns_state;
567a6d42e7dSPeter Dunlap 	list_t			isns_svrs;
568a6d42e7dSPeter Dunlap } iscsit_isns_cfg_t;
569a6d42e7dSPeter Dunlap 
570a6d42e7dSPeter Dunlap /*
571a6d42e7dSPeter Dunlap  * State values for the iscsit service
572a6d42e7dSPeter Dunlap  */
573a6d42e7dSPeter Dunlap typedef enum {
574a6d42e7dSPeter Dunlap 	ISE_UNDEFINED = 0,
575a6d42e7dSPeter Dunlap 	ISE_DETACHED,
576a6d42e7dSPeter Dunlap 	ISE_DISABLED,
577a6d42e7dSPeter Dunlap 	ISE_ENABLING,
578a6d42e7dSPeter Dunlap 	ISE_ENABLED,
579a6d42e7dSPeter Dunlap 	ISE_BUSY,
580a6d42e7dSPeter Dunlap 	ISE_DISABLING
581a6d42e7dSPeter Dunlap } iscsit_service_enabled_t;
582a6d42e7dSPeter Dunlap 
583a6d42e7dSPeter Dunlap 
584a6d42e7dSPeter Dunlap typedef struct {
585a6d42e7dSPeter Dunlap 	iscsit_service_enabled_t	global_svc_state;
586a6d42e7dSPeter Dunlap 	dev_info_t			*global_dip;
587a6d42e7dSPeter Dunlap 	ldi_ident_t			global_li;
588a6d42e7dSPeter Dunlap 	nvlist_t			*global_props;
589a6d42e7dSPeter Dunlap 	stmf_port_provider_t		*global_pp;
590a6d42e7dSPeter Dunlap 	stmf_dbuf_store_t		*global_dbuf_store;
591a6d42e7dSPeter Dunlap 	taskq_t				*global_dispatch_taskq;
592a6d42e7dSPeter Dunlap 	idm_refcnt_t			global_refcnt;
593a6d42e7dSPeter Dunlap 	avl_tree_t			global_discovery_sessions;
594a6d42e7dSPeter Dunlap 	avl_tree_t			global_target_list;
595a6d42e7dSPeter Dunlap 	list_t				global_deleted_target_list;
596a6d42e7dSPeter Dunlap 	avl_tree_t			global_tpg_list;
597a6d42e7dSPeter Dunlap 	avl_tree_t			global_ini_list;
598a6d42e7dSPeter Dunlap 	iscsit_tpg_t			*global_default_tpg;
599a6d42e7dSPeter Dunlap 	vmem_t				*global_tsih_pool;
600a6d42e7dSPeter Dunlap 	iscsit_isns_cfg_t		global_isns_cfg;
601a6d42e7dSPeter Dunlap 	iscsi_radius_props_t		global_radius_server;
602a6d42e7dSPeter Dunlap 	krwlock_t			global_rwlock;
6038c629652SPeter Cudhea - Sun Microsystems - Burlington, MA United States 	kmutex_t			global_state_mutex;
604a6d42e7dSPeter Dunlap } iscsit_global_t;
605a6d42e7dSPeter Dunlap 
606a6d42e7dSPeter Dunlap extern iscsit_global_t iscsit_global;
607a6d42e7dSPeter Dunlap 
608a6d42e7dSPeter Dunlap void
609a6d42e7dSPeter Dunlap iscsit_global_hold();
610a6d42e7dSPeter Dunlap 
611a6d42e7dSPeter Dunlap void
612a6d42e7dSPeter Dunlap iscsit_global_rele();
613a6d42e7dSPeter Dunlap 
614a6d42e7dSPeter Dunlap void
615a6d42e7dSPeter Dunlap iscsit_global_wait_ref();
616a6d42e7dSPeter Dunlap 
617a6d42e7dSPeter Dunlap idm_status_t
618a6d42e7dSPeter Dunlap iscsit_login_sm_init(iscsit_conn_t *ict);
619a6d42e7dSPeter Dunlap 
620a6d42e7dSPeter Dunlap void
621a6d42e7dSPeter Dunlap iscsit_login_sm_fini(iscsit_conn_t *ict);
622a6d42e7dSPeter Dunlap 
623a6d42e7dSPeter Dunlap void
624a6d42e7dSPeter Dunlap iscsit_login_sm_event(iscsit_conn_t *ic, iscsit_login_event_t event,
625a6d42e7dSPeter Dunlap     idm_pdu_t *pdu);
626a6d42e7dSPeter Dunlap 
627a6d42e7dSPeter Dunlap void
628a6d42e7dSPeter Dunlap iscsit_login_sm_event_locked(iscsit_conn_t *ic, iscsit_login_event_t event,
629a6d42e7dSPeter Dunlap     idm_pdu_t *pdu);
630a6d42e7dSPeter Dunlap 
631*483b029bSYuri Pankov int
632*483b029bSYuri Pankov iscsit_is_v4_mapped(struct sockaddr_storage *sa, struct sockaddr_storage *v4sa);
633*483b029bSYuri Pankov 
634a6d42e7dSPeter Dunlap void
635a6d42e7dSPeter Dunlap iscsit_send_async_event(iscsit_conn_t *ict, uint8_t async_event);
636a6d42e7dSPeter Dunlap 
637a6d42e7dSPeter Dunlap void
638a6d42e7dSPeter Dunlap iscsit_pdu_tx(idm_pdu_t *pdu);
639a6d42e7dSPeter Dunlap 
6404142b486SJames Moore void
6414142b486SJames Moore iscsit_send_reject(iscsit_conn_t *ict, idm_pdu_t *rejected_pdu, uint8_t reason);
6424142b486SJames Moore 
6434142b486SJames Moore void
6444142b486SJames Moore iscsit_text_cmd_fini(iscsit_conn_t *ict);
6454142b486SJames Moore 
646a6d42e7dSPeter Dunlap /*
647a6d42e7dSPeter Dunlap  * IDM conn ops
648a6d42e7dSPeter Dunlap  */
649a6d42e7dSPeter Dunlap 
650a6d42e7dSPeter Dunlap idm_rx_pdu_cb_t		iscsit_op_scsi_cmd;
651a6d42e7dSPeter Dunlap idm_rx_pdu_cb_t		iscsit_rx_pdu;
652a6d42e7dSPeter Dunlap idm_rx_pdu_error_cb_t	iscsit_rx_pdu_error;
653a6d42e7dSPeter Dunlap idm_task_cb_t		iscsit_task_aborted;
654a6d42e7dSPeter Dunlap idm_client_notify_cb_t	iscsit_client_notify;
655a6d42e7dSPeter Dunlap idm_build_hdr_cb_t	iscsit_build_hdr;
65660220f10SPriya Krishnan idm_update_statsn_cb_t	iscsit_update_statsn;
65772cf3143Speter dunlap idm_keepalive_cb_t	iscsit_keepalive;
658a6d42e7dSPeter Dunlap 
659a6d42e7dSPeter Dunlap /*
660a6d42e7dSPeter Dunlap  * lport entry points
661a6d42e7dSPeter Dunlap  */
662a6d42e7dSPeter Dunlap stmf_status_t
663a6d42e7dSPeter Dunlap iscsit_xfer_scsi_data(scsi_task_t *task, stmf_data_buf_t *dbuf,
664a6d42e7dSPeter Dunlap     uint32_t ioflags);
665a6d42e7dSPeter Dunlap 
666a6d42e7dSPeter Dunlap stmf_status_t
667a6d42e7dSPeter Dunlap iscsit_send_scsi_status(scsi_task_t *task, uint32_t ioflags);
668a6d42e7dSPeter Dunlap 
669a6d42e7dSPeter Dunlap void
670a6d42e7dSPeter Dunlap iscsit_lport_task_free(scsi_task_t *task);
671a6d42e7dSPeter Dunlap 
672a6d42e7dSPeter Dunlap stmf_status_t
673a6d42e7dSPeter Dunlap iscsit_abort(stmf_local_port_t *lport, int abort_cmd, void *arg,
674a6d42e7dSPeter Dunlap     uint32_t flags);
675a6d42e7dSPeter Dunlap 
676a6d42e7dSPeter Dunlap void
677a6d42e7dSPeter Dunlap iscsit_ctl(stmf_local_port_t *lport, int cmd, void *arg);
678a6d42e7dSPeter Dunlap 
679a6d42e7dSPeter Dunlap /*
680a6d42e7dSPeter Dunlap  * Connection functions
681a6d42e7dSPeter Dunlap  */
682a6d42e7dSPeter Dunlap idm_status_t
683a6d42e7dSPeter Dunlap iscsit_conn_reinstate(iscsit_conn_t *existing_ict, iscsit_conn_t *ict);
684a6d42e7dSPeter Dunlap 
685a6d42e7dSPeter Dunlap void
686a6d42e7dSPeter Dunlap iscsit_conn_destroy_done(iscsit_conn_t *ict);
687a6d42e7dSPeter Dunlap 
688a6d42e7dSPeter Dunlap void
689a6d42e7dSPeter Dunlap iscsit_conn_set_auth(iscsit_conn_t *ict);
690a6d42e7dSPeter Dunlap 
691a6d42e7dSPeter Dunlap void
692a6d42e7dSPeter Dunlap iscsit_conn_hold(iscsit_conn_t *ict);
693a6d42e7dSPeter Dunlap 
694a6d42e7dSPeter Dunlap void
695a6d42e7dSPeter Dunlap iscsit_conn_rele(iscsit_conn_t *ict);
696a6d42e7dSPeter Dunlap 
69747715e7fSPriya Krishnan void
69847715e7fSPriya Krishnan iscsit_conn_logout(iscsit_conn_t *ict);
69947715e7fSPriya Krishnan 
700a6d42e7dSPeter Dunlap /*
701a6d42e7dSPeter Dunlap  * Session functions
702a6d42e7dSPeter Dunlap  */
703a6d42e7dSPeter Dunlap int
704a6d42e7dSPeter Dunlap iscsit_sess_avl_compare(const void *void_sess1, const void *void_sess2);
705a6d42e7dSPeter Dunlap 
706a6d42e7dSPeter Dunlap iscsit_sess_t *
707a6d42e7dSPeter Dunlap iscsit_sess_create(iscsit_tgt_t *tgt, iscsit_conn_t *ict,
708a6d42e7dSPeter Dunlap     uint32_t cmdsn, uint8_t *isid, uint16_t tag,
709a6d42e7dSPeter Dunlap     char *initiator_name, char *target_name,
710a6d42e7dSPeter Dunlap     uint8_t *error_class, uint8_t *error_detail);
711a6d42e7dSPeter Dunlap 
712a6d42e7dSPeter Dunlap void
713a6d42e7dSPeter Dunlap iscsit_sess_destroy(iscsit_sess_t *ist);
714a6d42e7dSPeter Dunlap 
715a6d42e7dSPeter Dunlap void
716a6d42e7dSPeter Dunlap iscsit_sess_hold(iscsit_sess_t *ist);
717a6d42e7dSPeter Dunlap 
71847715e7fSPriya Krishnan idm_status_t
71947715e7fSPriya Krishnan iscsit_sess_check_hold(iscsit_sess_t *ist);
72047715e7fSPriya Krishnan 
721a6d42e7dSPeter Dunlap void
722a6d42e7dSPeter Dunlap iscsit_sess_rele(iscsit_sess_t *ist);
723a6d42e7dSPeter Dunlap 
724a6d42e7dSPeter Dunlap iscsit_conn_t *
725a6d42e7dSPeter Dunlap iscsit_sess_lookup_conn(iscsit_sess_t *ist, uint16_t cid);
726a6d42e7dSPeter Dunlap 
727a6d42e7dSPeter Dunlap void
728a6d42e7dSPeter Dunlap iscsit_sess_bind_conn(iscsit_sess_t *ist, iscsit_conn_t *ict);
729a6d42e7dSPeter Dunlap 
730a6d42e7dSPeter Dunlap void
731a6d42e7dSPeter Dunlap iscsit_sess_unbind_conn(iscsit_sess_t *ist, iscsit_conn_t *ict);
732a6d42e7dSPeter Dunlap 
733a6d42e7dSPeter Dunlap void
734a6d42e7dSPeter Dunlap iscsit_sess_close(iscsit_sess_t *ist);
735a6d42e7dSPeter Dunlap 
736a6d42e7dSPeter Dunlap iscsit_sess_t *
737a6d42e7dSPeter Dunlap iscsit_sess_reinstate(iscsit_tgt_t *tgt, iscsit_sess_t *ist, iscsit_conn_t *ict,
738a6d42e7dSPeter Dunlap     uint8_t *error_class, uint8_t *error_detail);
739a6d42e7dSPeter Dunlap 
740a6d42e7dSPeter Dunlap void
741a6d42e7dSPeter Dunlap iscsit_sess_sm_event(iscsit_sess_t *ist, iscsit_session_event_t event,
742a6d42e7dSPeter Dunlap     iscsit_conn_t *ict);
743a6d42e7dSPeter Dunlap 
744a6d42e7dSPeter Dunlap /*
745a6d42e7dSPeter Dunlap  * Target, TPGT, TPGT and portal functions
746a6d42e7dSPeter Dunlap  */
747a6d42e7dSPeter Dunlap 
748a6d42e7dSPeter Dunlap void
749a6d42e7dSPeter Dunlap iscsit_tgt_sm_event(iscsit_tgt_t *tgt, iscsit_tgt_event_t event);
750a6d42e7dSPeter Dunlap 
751a6d42e7dSPeter Dunlap void
752a6d42e7dSPeter Dunlap tgt_sm_event_locked(iscsit_tgt_t *tgt, iscsit_tgt_event_t event);
753a6d42e7dSPeter Dunlap 
754a6d42e7dSPeter Dunlap it_cfg_status_t
755a6d42e7dSPeter Dunlap iscsit_config_merge_tgt(it_config_t *cfg);
756a6d42e7dSPeter Dunlap 
757a6d42e7dSPeter Dunlap void
758a6d42e7dSPeter Dunlap iscsit_config_destroy_tgts(list_t *tgt_del_list);
759a6d42e7dSPeter Dunlap 
760a6d42e7dSPeter Dunlap void
761a6d42e7dSPeter Dunlap iscsit_config_destroy_tpgts(list_t *tpgt_del_list);
762a6d42e7dSPeter Dunlap 
763a6d42e7dSPeter Dunlap iscsit_tgt_t *
764a6d42e7dSPeter Dunlap iscsit_tgt_lookup(char *target_name);
765a6d42e7dSPeter Dunlap 
766a6d42e7dSPeter Dunlap iscsit_tgt_t *
767a6d42e7dSPeter Dunlap iscsit_tgt_lookup_locked(char *target_name);
768a6d42e7dSPeter Dunlap 
769a6d42e7dSPeter Dunlap int
770a6d42e7dSPeter Dunlap iscsit_tgt_avl_compare(const void *void_tgt1, const void *void_tgt2);
771a6d42e7dSPeter Dunlap 
772a6d42e7dSPeter Dunlap int
773a6d42e7dSPeter Dunlap iscsit_tpgt_avl_compare(const void *void_tpgt1, const void *void_tpgt2);
774a6d42e7dSPeter Dunlap 
775a6d42e7dSPeter Dunlap void
776a6d42e7dSPeter Dunlap iscsit_tgt_hold(iscsit_tgt_t *tgt);
777a6d42e7dSPeter Dunlap 
778a6d42e7dSPeter Dunlap void
779a6d42e7dSPeter Dunlap iscsit_tgt_rele(iscsit_tgt_t *tgt);
780a6d42e7dSPeter Dunlap 
781a6d42e7dSPeter Dunlap iscsit_tpgt_t *
782a6d42e7dSPeter Dunlap iscsit_tgt_lookup_tpgt(iscsit_tgt_t *tgt, uint16_t tag);
783a6d42e7dSPeter Dunlap 
784a6d42e7dSPeter Dunlap void
785a6d42e7dSPeter Dunlap iscsit_tpgt_hold(iscsit_tpgt_t *tpgt);
786a6d42e7dSPeter Dunlap 
787a6d42e7dSPeter Dunlap void
788a6d42e7dSPeter Dunlap iscsit_tpgt_rele(iscsit_tpgt_t *tpgt);
789a6d42e7dSPeter Dunlap 
790a6d42e7dSPeter Dunlap iscsit_portal_t *
791a6d42e7dSPeter Dunlap iscsit_tgt_lookup_portal(iscsit_tgt_t *tgt, struct sockaddr_storage *sa,
792a6d42e7dSPeter Dunlap     iscsit_tpgt_t **output_tpgt);
793a6d42e7dSPeter Dunlap 
794a6d42e7dSPeter Dunlap iscsit_sess_t *
795a6d42e7dSPeter Dunlap iscsit_tgt_lookup_sess(iscsit_tgt_t *tgt, char *initiator_name,
796a6d42e7dSPeter Dunlap     uint8_t *isid, uint16_t tsih, uint16_t tag);
797a6d42e7dSPeter Dunlap 
798a6d42e7dSPeter Dunlap void
799a6d42e7dSPeter Dunlap iscsit_tgt_bind_sess(iscsit_tgt_t *tgt, iscsit_sess_t *sess);
800a6d42e7dSPeter Dunlap 
801a6d42e7dSPeter Dunlap void
802a6d42e7dSPeter Dunlap iscsit_tgt_unbind_sess(iscsit_tgt_t *tgt, iscsit_sess_t *sess);
803a6d42e7dSPeter Dunlap 
804a6d42e7dSPeter Dunlap it_cfg_status_t
805a6d42e7dSPeter Dunlap iscsit_config_merge_tpg(it_config_t *cfg, list_t *tpg_del_list);
806a6d42e7dSPeter Dunlap 
807a6d42e7dSPeter Dunlap void
808a6d42e7dSPeter Dunlap iscsit_config_destroy_tpgs(list_t *tpg_del_list);
809a6d42e7dSPeter Dunlap 
810a6d42e7dSPeter Dunlap iscsit_tpg_t *
811a6d42e7dSPeter Dunlap iscsit_tpg_lookup(char *tpg_name);
812a6d42e7dSPeter Dunlap 
813a6d42e7dSPeter Dunlap int
814a6d42e7dSPeter Dunlap iscsit_tpg_avl_compare(const void *void_tpg1, const void *void_tpg2);
815a6d42e7dSPeter Dunlap 
816a6d42e7dSPeter Dunlap void
817a6d42e7dSPeter Dunlap iscsit_tpg_hold(iscsit_tpg_t *tpg);
818a6d42e7dSPeter Dunlap 
819a6d42e7dSPeter Dunlap void
820a6d42e7dSPeter Dunlap iscsit_tpg_rele(iscsit_tpg_t *tpg);
821a6d42e7dSPeter Dunlap 
822a6d42e7dSPeter Dunlap iscsit_tpg_t *
823a6d42e7dSPeter Dunlap iscsit_tpg_createdefault();
824a6d42e7dSPeter Dunlap 
825a6d42e7dSPeter Dunlap void
826a6d42e7dSPeter Dunlap iscsit_tpg_destroydefault(iscsit_tpg_t *tpg);
827a6d42e7dSPeter Dunlap 
828a6d42e7dSPeter Dunlap idm_status_t
829a6d42e7dSPeter Dunlap iscsit_tpg_online(iscsit_tpg_t *tpg);
830a6d42e7dSPeter Dunlap 
831a6d42e7dSPeter Dunlap void
832a6d42e7dSPeter Dunlap iscsit_tpg_offline(iscsit_tpg_t *tpg);
833a6d42e7dSPeter Dunlap 
834a6d42e7dSPeter Dunlap iscsit_portal_t *
835a6d42e7dSPeter Dunlap iscsit_tpg_portal_lookup(iscsit_tpg_t *tpg, struct sockaddr_storage *sa);
836a6d42e7dSPeter Dunlap 
837a6d42e7dSPeter Dunlap void
838a6d42e7dSPeter Dunlap iscsit_portal_hold(iscsit_portal_t *portal);
839a6d42e7dSPeter Dunlap 
840a6d42e7dSPeter Dunlap void
841a6d42e7dSPeter Dunlap iscsit_portal_rele(iscsit_portal_t *portal);
842a6d42e7dSPeter Dunlap 
843a6d42e7dSPeter Dunlap it_cfg_status_t
844a6d42e7dSPeter Dunlap iscsit_config_merge_ini(it_config_t *cfg);
845a6d42e7dSPeter Dunlap 
846a6d42e7dSPeter Dunlap int
847a6d42e7dSPeter Dunlap iscsit_ini_avl_compare(const void *void_ini1, const void *void_ini2);
848a6d42e7dSPeter Dunlap 
849a6d42e7dSPeter Dunlap iscsit_ini_t *
850a6d42e7dSPeter Dunlap iscsit_ini_lookup_locked(char *ini_name);
851a6d42e7dSPeter Dunlap 
852a6d42e7dSPeter Dunlap int
853a6d42e7dSPeter Dunlap iscsit_portal_avl_compare(const void *void_portal1, const void *void_portal2);
854a6d42e7dSPeter Dunlap 
855a6d42e7dSPeter Dunlap int
856a6d42e7dSPeter Dunlap iscsit_verify_chap_resp(iscsit_conn_login_t *lsm,
857a6d42e7dSPeter Dunlap     unsigned int chap_i, uchar_t *chap_c, unsigned int challenge_len,
858a6d42e7dSPeter Dunlap     uchar_t *chap_r, unsigned int resp_len);
859a6d42e7dSPeter Dunlap 
860d618d68dSPriya Krishnan void
861d618d68dSPriya Krishnan iscsit_rxpdu_queue_monitor_start(void);
862d618d68dSPriya Krishnan 
863d618d68dSPriya Krishnan void
864d618d68dSPriya Krishnan iscsit_rxpdu_queue_monitor_stop(void);
865d618d68dSPriya Krishnan 
866a6d42e7dSPeter Dunlap #endif /* _ISCSIT_H_ */
867