2029 lines
52 KiB
C
2029 lines
52 KiB
C
/* $NetBSD: iscsi_ioctl.c,v 1.30 2018/12/07 14:59:19 mlelstv Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 2004,2005,2006,2011 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Wasabi Systems, Inc.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "iscsi_globals.h"
|
|
|
|
#include <sys/file.h>
|
|
#include <sys/filedesc.h>
|
|
#include <sys/proc.h>
|
|
|
|
#ifndef ISCSI_MINIMAL
|
|
#include <uvm/uvm.h>
|
|
#include <uvm/uvm_pmap.h>
|
|
#endif
|
|
|
|
static kmutex_t iscsi_cleanup_mtx;
|
|
static kcondvar_t iscsi_cleanup_cv;
|
|
static kcondvar_t iscsi_event_cv;
|
|
static struct lwp *iscsi_cleanproc = NULL;
|
|
|
|
static uint16_t current_id = 0; /* Global session ID counter */
|
|
|
|
/* list of event handlers */
|
|
static event_handler_list_t event_handlers =
|
|
TAILQ_HEAD_INITIALIZER(event_handlers);
|
|
|
|
static connection_list_t iscsi_timeout_conn_list =
|
|
TAILQ_HEAD_INITIALIZER(iscsi_timeout_conn_list);
|
|
|
|
static ccb_list_t iscsi_timeout_ccb_list =
|
|
TAILQ_HEAD_INITIALIZER(iscsi_timeout_ccb_list);
|
|
|
|
static session_list_t iscsi_cleanups_list =
|
|
TAILQ_HEAD_INITIALIZER(iscsi_cleanups_list);
|
|
|
|
static connection_list_t iscsi_cleanupc_list =
|
|
TAILQ_HEAD_INITIALIZER(iscsi_cleanupc_list);
|
|
|
|
static uint32_t handler_id = 0; /* Handler ID counter */
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
/* Event management functions */
|
|
|
|
/*
|
|
* find_handler:
|
|
* Search the event handler list for the given ID.
|
|
*
|
|
* Parameter:
|
|
* id The handler ID.
|
|
*
|
|
* Returns:
|
|
* Pointer to handler if found, else NULL.
|
|
*/
|
|
|
|
|
|
static event_handler_t *
|
|
find_handler(uint32_t id)
|
|
{
|
|
event_handler_t *curr;
|
|
|
|
KASSERT(mutex_owned(&iscsi_cleanup_mtx));
|
|
|
|
TAILQ_FOREACH(curr, &event_handlers, evh_link)
|
|
if (curr->evh_id == id)
|
|
break;
|
|
|
|
return curr;
|
|
}
|
|
|
|
|
|
/*
|
|
* register_event:
|
|
* Create event handler entry, return ID.
|
|
*
|
|
* Parameter:
|
|
* par The parameter.
|
|
*/
|
|
|
|
static void
|
|
register_event(iscsi_register_event_parameters_t *par)
|
|
{
|
|
event_handler_t *handler;
|
|
int was_empty;
|
|
|
|
handler = malloc(sizeof(event_handler_t), M_DEVBUF, M_WAITOK | M_ZERO);
|
|
if (handler == NULL) {
|
|
DEBOUT(("No mem for event handler\n"));
|
|
par->status = ISCSI_STATUS_NO_RESOURCES;
|
|
return;
|
|
}
|
|
|
|
TAILQ_INIT(&handler->evh_events);
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
/* create a unique ID */
|
|
do {
|
|
++handler_id;
|
|
} while (!handler_id || find_handler(handler_id) != NULL);
|
|
par->event_id = handler->evh_id = handler_id;
|
|
|
|
was_empty = TAILQ_FIRST(&event_handlers) == NULL;
|
|
TAILQ_INSERT_TAIL(&event_handlers, handler, evh_link);
|
|
if (was_empty)
|
|
iscsi_notify_cleanup();
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
DEB(5, ("Register Event OK, ID %d\n", par->event_id));
|
|
}
|
|
|
|
|
|
/*
|
|
* deregister_event:
|
|
* Destroy handler entry and any waiting events, wake up waiter.
|
|
*
|
|
* Parameter:
|
|
* par The parameter.
|
|
*/
|
|
|
|
static void
|
|
deregister_event(iscsi_register_event_parameters_t *par)
|
|
{
|
|
event_handler_t *handler;
|
|
event_t *evt;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
handler = find_handler(par->event_id);
|
|
if (handler == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEB(1, ("Deregister Event ID %d not found\n", par->event_id));
|
|
par->status = ISCSI_STATUS_INVALID_EVENT_ID;
|
|
return;
|
|
}
|
|
|
|
TAILQ_REMOVE(&event_handlers, handler, evh_link);
|
|
if (handler->evh_waiter != NULL) {
|
|
handler->evh_waiter->status = ISCSI_STATUS_EVENT_DEREGISTERED;
|
|
cv_broadcast(&iscsi_event_cv);
|
|
}
|
|
|
|
while ((evt = TAILQ_FIRST(&handler->evh_events)) != NULL) {
|
|
TAILQ_REMOVE(&handler->evh_events, evt, ev_link);
|
|
free(evt, M_TEMP);
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
free(handler, M_DEVBUF);
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
DEB(5, ("Deregister Event ID %d complete\n", par->event_id));
|
|
}
|
|
|
|
|
|
/*
|
|
* check_event:
|
|
* Return first queued event. Optionally wait for arrival of event.
|
|
*
|
|
* Parameter:
|
|
* par The parameter.
|
|
* wait Wait for event if true
|
|
*/
|
|
|
|
static void
|
|
check_event(iscsi_wait_event_parameters_t *par, bool wait)
|
|
{
|
|
event_handler_t *handler;
|
|
event_t *evt;
|
|
int rc;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
handler = find_handler(par->event_id);
|
|
if (handler == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Wait Event ID %d not found\n", par->event_id));
|
|
par->status = ISCSI_STATUS_INVALID_EVENT_ID;
|
|
return;
|
|
}
|
|
if (handler->evh_waiter != NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Wait Event ID %d already waiting\n", par->event_id));
|
|
par->status = ISCSI_STATUS_EVENT_WAITING;
|
|
return;
|
|
}
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
DEB(99, ("Wait Event ID %d\n", par->event_id));
|
|
|
|
do {
|
|
evt = TAILQ_FIRST(&handler->evh_events);
|
|
if (evt != NULL) {
|
|
TAILQ_REMOVE(&handler->evh_events, evt, ev_link);
|
|
} else {
|
|
if (!wait) {
|
|
par->status = ISCSI_STATUS_LIST_EMPTY;
|
|
return;
|
|
}
|
|
if (par->status != ISCSI_STATUS_SUCCESS) {
|
|
return;
|
|
}
|
|
handler->evh_waiter = par;
|
|
rc = cv_wait_sig(&iscsi_event_cv, &iscsi_cleanup_mtx);
|
|
if (rc) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
par->status = ISCSI_STATUS_LIST_EMPTY;
|
|
return;
|
|
}
|
|
}
|
|
} while (evt == NULL);
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
par->connection_id = evt->ev_connection_id;
|
|
par->session_id = evt->ev_session_id;
|
|
par->event_kind = evt->ev_event_kind;
|
|
par->reason = evt->ev_reason;
|
|
|
|
free(evt, M_TEMP);
|
|
}
|
|
|
|
/*
|
|
* add_event
|
|
* Adds an event entry to each registered handler queue.
|
|
* Note that events are simply duplicated because we expect the number of
|
|
* handlers to be very small, usually 1 (the daemon).
|
|
*
|
|
* Parameters:
|
|
* kind The event kind
|
|
* sid The ID of the affected session
|
|
* cid The ID of the affected connection
|
|
* reason The reason code
|
|
*/
|
|
|
|
void
|
|
add_event(iscsi_event_t kind, uint32_t sid, uint32_t cid, uint32_t reason)
|
|
{
|
|
event_handler_t *curr;
|
|
event_t *evt;
|
|
|
|
DEB(9, ("Add_event kind %d, sid %d, cid %d, reason %d\n",
|
|
kind, sid, cid, reason));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
TAILQ_FOREACH(curr, &event_handlers, evh_link) {
|
|
evt = malloc(sizeof(*evt), M_TEMP, M_NOWAIT);
|
|
if (evt == NULL) {
|
|
DEBOUT(("Cannot allocate event\n"));
|
|
break;
|
|
}
|
|
evt->ev_event_kind = kind;
|
|
evt->ev_session_id = sid;
|
|
evt->ev_connection_id = cid;
|
|
evt->ev_reason = reason;
|
|
|
|
TAILQ_INSERT_TAIL(&curr->evh_events, evt, ev_link);
|
|
if (curr->evh_waiter != NULL) {
|
|
curr->evh_waiter = NULL;
|
|
cv_broadcast(&iscsi_event_cv);
|
|
}
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
|
|
/*
|
|
* check_event_handlers
|
|
* Checks for dead event handlers. A dead event handler would deplete
|
|
* memory over time, so we have to make sure someone at the other
|
|
* end is actively monitoring events.
|
|
* This function is called every 30 seconds or so (less frequent if there
|
|
* is other activity for the cleanup thread to deal with) to go through
|
|
* the list of handlers and check whether the first element in the event
|
|
* list has changed at all. If not, the event is deregistered.
|
|
* Note that this will not detect dead handlers if no events are pending,
|
|
* but we don't care as long as events don't accumulate in the list.
|
|
*
|
|
*/
|
|
|
|
static void
|
|
check_event_handlers(void)
|
|
{
|
|
event_handler_t *curr, *next;
|
|
event_t *evt;
|
|
|
|
KASSERT(mutex_owned(&iscsi_cleanup_mtx));
|
|
|
|
for (curr = TAILQ_FIRST(&event_handlers); curr != NULL; curr = next) {
|
|
next = TAILQ_NEXT(curr, evh_link);
|
|
evt = TAILQ_FIRST(&curr->evh_events);
|
|
|
|
if (evt != NULL && evt == curr->evh_first_in_list) {
|
|
DEBOUT(("Found Dead Event Handler %d, removing\n", curr->evh_id));
|
|
|
|
TAILQ_REMOVE(&event_handlers, curr, evh_link);
|
|
while ((evt = TAILQ_FIRST(&curr->evh_events)) != NULL) {
|
|
TAILQ_REMOVE(&curr->evh_events, evt, ev_link);
|
|
free(evt, M_TEMP);
|
|
}
|
|
free(curr, M_DEVBUF);
|
|
} else
|
|
curr->evh_first_in_list = evt;
|
|
}
|
|
}
|
|
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
/*
|
|
* get_socket:
|
|
* Get the file pointer from the socket handle passed into login.
|
|
*
|
|
* Parameter:
|
|
* fdes IN: The socket handle
|
|
* fpp OUT: The pointer to the resulting file pointer
|
|
*
|
|
* Returns: 0 on success, else an error code.
|
|
*
|
|
*/
|
|
|
|
static int
|
|
get_socket(int fdes, struct file **fpp)
|
|
{
|
|
struct file *fp;
|
|
|
|
if ((fp = fd_getfile(fdes)) == NULL) {
|
|
return EBADF;
|
|
}
|
|
if (fp->f_type != DTYPE_SOCKET) {
|
|
return ENOTSOCK;
|
|
}
|
|
|
|
/* Add the reference */
|
|
mutex_enter(&fp->f_lock);
|
|
fp->f_count++;
|
|
mutex_exit(&fp->f_lock);
|
|
|
|
*fpp = fp;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* release_socket:
|
|
* Release the file pointer from the socket handle passed into login.
|
|
*
|
|
* Parameter:
|
|
* fp IN: The pointer to the resulting file pointer
|
|
*
|
|
*/
|
|
|
|
static void
|
|
release_socket(struct file *fp)
|
|
{
|
|
/* Add the reference */
|
|
mutex_enter(&fp->f_lock);
|
|
fp->f_count--;
|
|
mutex_exit(&fp->f_lock);
|
|
}
|
|
|
|
|
|
/*
|
|
* find_session:
|
|
* Find a session by ID.
|
|
*
|
|
* Parameter: the session ID
|
|
*
|
|
* Returns: The pointer to the session (or NULL if not found)
|
|
*/
|
|
|
|
session_t *
|
|
find_session(uint32_t id)
|
|
{
|
|
session_t *sess;
|
|
|
|
KASSERT(mutex_owned(&iscsi_cleanup_mtx));
|
|
|
|
TAILQ_FOREACH(sess, &iscsi_sessions, s_sessions)
|
|
if (sess->s_id == id) {
|
|
break;
|
|
}
|
|
return sess;
|
|
}
|
|
|
|
|
|
/*
|
|
* find_connection:
|
|
* Find a connection by ID.
|
|
*
|
|
* Parameter: the session pointer and the connection ID
|
|
*
|
|
* Returns: The pointer to the connection (or NULL if not found)
|
|
*/
|
|
|
|
connection_t *
|
|
find_connection(session_t *sess, uint32_t id)
|
|
{
|
|
connection_t *conn;
|
|
|
|
KASSERT(mutex_owned(&iscsi_cleanup_mtx));
|
|
|
|
TAILQ_FOREACH(conn, &sess->s_conn_list, c_connections)
|
|
if (conn->c_id == id) {
|
|
break;
|
|
}
|
|
return conn;
|
|
}
|
|
|
|
/*
|
|
* ref_session:
|
|
* Reference a session
|
|
*
|
|
* Session cannot be release until reference count reaches zero
|
|
*
|
|
* Returns: 1 if reference counter would overflow
|
|
*/
|
|
|
|
int
|
|
ref_session(session_t *sess)
|
|
{
|
|
int rc = 1;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
KASSERT(sess != NULL);
|
|
if (sess->s_refcount <= CCBS_PER_SESSION) {
|
|
sess->s_refcount++;
|
|
rc = 0;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* unref_session:
|
|
* Unreference a session
|
|
*
|
|
* Release session reference, trigger cleanup
|
|
*/
|
|
|
|
void
|
|
unref_session(session_t *session)
|
|
{
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
KASSERT(session != NULL);
|
|
KASSERT(session->s_refcount > 0);
|
|
if (--session->s_refcount == 0)
|
|
cv_broadcast(&session->s_sess_cv);
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
|
|
/*
|
|
* kill_connection:
|
|
* Terminate the connection as gracefully as possible.
|
|
*
|
|
* Parameter:
|
|
* conn The connection to terminate
|
|
* status The status code for the connection's "terminating" field
|
|
* logout The logout reason code
|
|
* recover Attempt to recover connection
|
|
*/
|
|
|
|
void
|
|
kill_connection(connection_t *conn, uint32_t status, int logout, bool recover)
|
|
{
|
|
session_t *sess = conn->c_session;
|
|
int terminating;
|
|
|
|
DEBC(conn, 1, ("Kill_connection: terminating=%d, status=%d, logout=%d, "
|
|
"state=%d\n",
|
|
conn->c_terminating, status, logout, conn->c_state));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (recover &&
|
|
!conn->c_destroy &&
|
|
conn->c_recover > MAX_RECOVERY_ATTEMPTS) {
|
|
DEBC(conn, 1,
|
|
("Kill_connection: Too many recovery attempts, destroying\n"));
|
|
conn->c_destroy = TRUE;
|
|
}
|
|
|
|
if (!recover || conn->c_destroy) {
|
|
|
|
if (conn->c_in_session) {
|
|
conn->c_in_session = FALSE;
|
|
TAILQ_REMOVE(&sess->s_conn_list, conn, c_connections);
|
|
sess->s_mru_connection = TAILQ_FIRST(&sess->s_conn_list);
|
|
}
|
|
|
|
if (!conn->c_destroy) {
|
|
DEBC(conn, 1, ("Kill_connection setting destroy flag\n"));
|
|
conn->c_destroy = TRUE;
|
|
}
|
|
}
|
|
|
|
terminating = conn->c_terminating;
|
|
if (!terminating)
|
|
conn->c_terminating = status;
|
|
|
|
/* Don't recurse */
|
|
if (terminating) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
KASSERT(conn->c_state != ST_FULL_FEATURE);
|
|
DEBC(conn, 1, ("Kill_connection exiting (already terminating)\n"));
|
|
goto done;
|
|
}
|
|
|
|
if (conn->c_state == ST_FULL_FEATURE) {
|
|
sess->s_active_connections--;
|
|
conn->c_state = ST_WINDING_DOWN;
|
|
|
|
/* If this is the last connection and ERL < 2, reset TSIH */
|
|
if (!sess->s_active_connections && sess->s_ErrorRecoveryLevel < 2)
|
|
sess->s_TSIH = 0;
|
|
|
|
/* Don't try to log out if the socket is broken or we're in the middle */
|
|
/* of logging in */
|
|
if (logout >= 0) {
|
|
if (sess->s_ErrorRecoveryLevel < 2 &&
|
|
logout == RECOVER_CONNECTION) {
|
|
logout = LOGOUT_CONNECTION;
|
|
}
|
|
if (!sess->s_active_connections &&
|
|
logout == LOGOUT_CONNECTION) {
|
|
logout = LOGOUT_SESSION;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
connection_timeout_start(conn, CONNECTION_TIMEOUT);
|
|
|
|
if (!send_logout(conn, conn, logout, FALSE)) {
|
|
conn->c_terminating = ISCSI_STATUS_SUCCESS;
|
|
return;
|
|
}
|
|
/*
|
|
* if the logout request was successfully sent,
|
|
* the logout response handler will do the rest
|
|
* of the termination processing. If the logout
|
|
* doesn't get a response, we'll get back in here
|
|
* once the timeout hits.
|
|
*/
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
}
|
|
|
|
conn->c_state = ST_SETTLING;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
done:
|
|
/* let send thread take over next step of cleanup */
|
|
mutex_enter(&conn->c_lock);
|
|
cv_broadcast(&conn->c_conn_cv);
|
|
mutex_exit(&conn->c_lock);
|
|
|
|
DEBC(conn, 5, ("kill_connection returns\n"));
|
|
}
|
|
|
|
|
|
/*
|
|
* kill_session:
|
|
* Terminate the session as gracefully as possible.
|
|
*
|
|
* Parameter:
|
|
* session Session to terminate
|
|
* status The status code for the termination
|
|
* logout The logout reason code
|
|
|
|
*/
|
|
|
|
void
|
|
kill_session(session_t *sess, uint32_t status, int logout, bool recover)
|
|
{
|
|
connection_t *conn;
|
|
|
|
DEB(1, ("ISCSI: kill_session %d, status %d, logout %d, recover %d\n",
|
|
sess->s_id, status, logout, recover));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (sess->s_terminating) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
DEB(5, ("Session is being killed with status %d\n",sess->s_terminating));
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* don't do anything if session isn't established yet, termination will be
|
|
* handled elsewhere
|
|
*/
|
|
if (sess->s_sessions.tqe_next == NULL && sess->s_sessions.tqe_prev == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
DEB(5, ("Session is being killed which is not yet established\n"));
|
|
return;
|
|
}
|
|
|
|
if (recover) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
/*
|
|
* Only recover when there's just one active connection left.
|
|
* Otherwise we get in all sorts of timing problems, and it doesn't
|
|
* make much sense anyway to recover when the other side has
|
|
* requested that we kill a multipathed session.
|
|
*/
|
|
if (sess->s_active_connections == 1) {
|
|
conn = assign_connection(sess, FALSE);
|
|
if (conn != NULL)
|
|
kill_connection(conn, status, logout, TRUE);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (sess->s_refcount > 0) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
DEB(5, ("Session is being killed while in use (refcnt = %d)\n",
|
|
sess->s_refcount));
|
|
return;
|
|
}
|
|
|
|
/* Remove session from global list */
|
|
sess->s_terminating = status;
|
|
TAILQ_REMOVE(&iscsi_sessions, sess, s_sessions);
|
|
sess->s_sessions.tqe_next = NULL;
|
|
sess->s_sessions.tqe_prev = NULL;
|
|
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
/* kill all connections */
|
|
while ((conn = TAILQ_FIRST(&sess->s_conn_list)) != NULL) {
|
|
kill_connection(conn, status, logout, FALSE);
|
|
logout = NO_LOGOUT;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* create_connection:
|
|
* Create and init the necessary framework for a connection:
|
|
* Alloc the connection structure itself
|
|
* Copy connection parameters
|
|
* Create the send and receive threads
|
|
* And finally, log in.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The login parameters
|
|
* session IN: The owning session
|
|
* l IN: The lwp pointer of the caller
|
|
*
|
|
* Returns: 0 on success
|
|
* >0 on failure, connection structure deleted
|
|
* <0 on failure, connection is still terminating
|
|
*/
|
|
|
|
static int
|
|
create_connection(iscsi_login_parameters_t *par, session_t *sess,
|
|
struct lwp *l)
|
|
{
|
|
connection_t *conn;
|
|
int rc;
|
|
|
|
DEB(1, ("Create Connection for Session %d\n", sess->s_id));
|
|
|
|
if (sess->s_MaxConnections &&
|
|
sess->s_active_connections >= sess->s_MaxConnections) {
|
|
DEBOUT(("Too many connections (max = %d, curr = %d)\n",
|
|
sess->s_MaxConnections, sess->s_active_connections));
|
|
par->status = ISCSI_STATUS_MAXED_CONNECTIONS;
|
|
return EIO;
|
|
}
|
|
|
|
conn = malloc(sizeof(*conn), M_DEVBUF, M_WAITOK | M_ZERO);
|
|
if (conn == NULL) {
|
|
DEBOUT(("No mem for connection\n"));
|
|
par->status = ISCSI_STATUS_NO_RESOURCES;
|
|
return EIO;
|
|
}
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
/* create a unique ID */
|
|
do {
|
|
++sess->s_conn_id;
|
|
} while (!sess->s_conn_id ||
|
|
find_connection(sess, sess->s_conn_id) != NULL);
|
|
par->connection_id = conn->c_id = sess->s_conn_id;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEB(99, ("Connection ID = %d\n", conn->c_id));
|
|
|
|
conn->c_session = sess;
|
|
|
|
TAILQ_INIT(&conn->c_ccbs_waiting);
|
|
TAILQ_INIT(&conn->c_pdus_to_send);
|
|
TAILQ_INIT(&conn->c_pdu_pool);
|
|
|
|
mutex_init(&conn->c_lock, MUTEX_DEFAULT, IPL_BIO);
|
|
cv_init(&conn->c_conn_cv, "conn");
|
|
cv_init(&conn->c_pdu_cv, "pdupool");
|
|
cv_init(&conn->c_ccb_cv, "ccbwait");
|
|
cv_init(&conn->c_idle_cv, "idle");
|
|
|
|
callout_init(&conn->c_timeout, CALLOUT_MPSAFE);
|
|
callout_setfunc(&conn->c_timeout, connection_timeout_co, conn);
|
|
conn->c_idle_timeout_val = CONNECTION_IDLE_TIMEOUT;
|
|
|
|
init_sernum(&conn->c_StatSN_buf);
|
|
create_pdus(conn);
|
|
|
|
if ((rc = get_socket(par->socket, &conn->c_sock)) != 0) {
|
|
DEBOUT(("Invalid socket %d\n", par->socket));
|
|
|
|
callout_destroy(&conn->c_timeout);
|
|
cv_destroy(&conn->c_idle_cv);
|
|
cv_destroy(&conn->c_ccb_cv);
|
|
cv_destroy(&conn->c_pdu_cv);
|
|
cv_destroy(&conn->c_conn_cv);
|
|
mutex_destroy(&conn->c_lock);
|
|
free(conn, M_DEVBUF);
|
|
par->status = ISCSI_STATUS_INVALID_SOCKET;
|
|
return rc;
|
|
}
|
|
DEBC(conn, 1, ("get_socket: par_sock=%d, fdesc=%p\n",
|
|
par->socket, conn->c_sock));
|
|
|
|
/* close the file descriptor */
|
|
fd_close(par->socket);
|
|
|
|
conn->c_threadobj = l;
|
|
conn->c_login_par = par;
|
|
|
|
DEB(5, ("Creating receive thread\n"));
|
|
if ((rc = kthread_create(PRI_BIO, KTHREAD_MPSAFE, NULL, iscsi_rcv_thread,
|
|
conn, &conn->c_rcvproc,
|
|
"ConnRcv")) != 0) {
|
|
DEBOUT(("Can't create rcv thread (rc %d)\n", rc));
|
|
|
|
release_socket(conn->c_sock);
|
|
callout_destroy(&conn->c_timeout);
|
|
cv_destroy(&conn->c_idle_cv);
|
|
cv_destroy(&conn->c_ccb_cv);
|
|
cv_destroy(&conn->c_pdu_cv);
|
|
cv_destroy(&conn->c_conn_cv);
|
|
mutex_destroy(&conn->c_lock);
|
|
free(conn, M_DEVBUF);
|
|
par->status = ISCSI_STATUS_NO_RESOURCES;
|
|
return rc;
|
|
}
|
|
DEB(5, ("Creating send thread\n"));
|
|
if ((rc = kthread_create(PRI_BIO, KTHREAD_MPSAFE, NULL, iscsi_send_thread,
|
|
conn, &conn->c_sendproc,
|
|
"ConnSend")) != 0) {
|
|
DEBOUT(("Can't create send thread (rc %d)\n", rc));
|
|
|
|
conn->c_terminating = ISCSI_STATUS_NO_RESOURCES;
|
|
|
|
/*
|
|
* We must close the socket here to force the receive
|
|
* thread to wake up
|
|
*/
|
|
DEBC(conn, 1, ("Closing Socket %p\n", conn->c_sock));
|
|
mutex_enter(&conn->c_sock->f_lock);
|
|
conn->c_sock->f_count += 1;
|
|
mutex_exit(&conn->c_sock->f_lock);
|
|
closef(conn->c_sock);
|
|
|
|
/* give receive thread time to exit */
|
|
kpause("settle", false, 2 * hz, NULL);
|
|
|
|
release_socket(conn->c_sock);
|
|
callout_destroy(&conn->c_timeout);
|
|
cv_destroy(&conn->c_idle_cv);
|
|
cv_destroy(&conn->c_ccb_cv);
|
|
cv_destroy(&conn->c_pdu_cv);
|
|
cv_destroy(&conn->c_conn_cv);
|
|
mutex_destroy(&conn->c_lock);
|
|
free(conn, M_DEVBUF);
|
|
par->status = ISCSI_STATUS_NO_RESOURCES;
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* At this point, each thread will tie 'sock' into its own file descriptor
|
|
* tables w/o increasing the use count - they will inherit the use
|
|
* increments performed in get_socket().
|
|
*/
|
|
|
|
if ((rc = send_login(conn)) != 0) {
|
|
DEBC(conn, 0, ("Login failed (rc %d)\n", rc));
|
|
/* Don't attempt to recover, there seems to be something amiss */
|
|
kill_connection(conn, rc, NO_LOGOUT, FALSE);
|
|
par->status = rc;
|
|
return -1;
|
|
}
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (sess->s_terminating) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBC(conn, 0, ("Session terminating\n"));
|
|
kill_connection(conn, rc, NO_LOGOUT, FALSE);
|
|
par->status = sess->s_terminating;
|
|
return -1;
|
|
}
|
|
conn->c_state = ST_FULL_FEATURE;
|
|
TAILQ_INSERT_TAIL(&sess->s_conn_list, conn, c_connections);
|
|
conn->c_in_session = TRUE;
|
|
sess->s_total_connections++;
|
|
sess->s_active_connections++;
|
|
sess->s_mru_connection = conn;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
DEBC(conn, 5, ("Connection created successfully!\n"));
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* recreate_connection:
|
|
* Revive dead connection
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The login parameters
|
|
* conn IN: The connection
|
|
* l IN: The lwp pointer of the caller
|
|
*
|
|
* Returns: 0 on success
|
|
* >0 on failure, connection structure deleted
|
|
* <0 on failure, connection is still terminating
|
|
*/
|
|
|
|
static int
|
|
recreate_connection(iscsi_login_parameters_t *par, session_t *sess,
|
|
connection_t *conn, struct lwp *l)
|
|
{
|
|
int rc;
|
|
ccb_t *ccb;
|
|
ccb_list_t old_waiting;
|
|
pdu_t *pdu;
|
|
uint32_t sn;
|
|
|
|
DEB(1, ("ReCreate Connection %d for Session %d, ERL=%d\n",
|
|
conn->c_id, conn->c_session->s_id,
|
|
conn->c_session->s_ErrorRecoveryLevel));
|
|
|
|
if (sess->s_MaxConnections &&
|
|
sess->s_active_connections >= sess->s_MaxConnections) {
|
|
DEBOUT(("Too many connections (max = %d, curr = %d)\n",
|
|
sess->s_MaxConnections, sess->s_active_connections));
|
|
par->status = ISCSI_STATUS_MAXED_CONNECTIONS;
|
|
return EIO;
|
|
}
|
|
|
|
/* close old socket */
|
|
if (conn->c_sock != NULL) {
|
|
closef(conn->c_sock);
|
|
conn->c_sock = NULL;
|
|
}
|
|
|
|
if ((rc = get_socket(par->socket, &conn->c_sock)) != 0) {
|
|
DEBOUT(("Invalid socket %d\n", par->socket));
|
|
par->status = ISCSI_STATUS_INVALID_SOCKET;
|
|
return rc;
|
|
}
|
|
DEBC(conn, 1, ("get_socket: par_sock=%d, fdesc=%p\n",
|
|
par->socket, conn->c_sock));
|
|
|
|
/* close the file descriptor */
|
|
fd_close(par->socket);
|
|
|
|
conn->c_threadobj = l;
|
|
conn->c_login_par = par;
|
|
conn->c_terminating = ISCSI_STATUS_SUCCESS;
|
|
conn->c_recover++;
|
|
conn->c_num_timeouts = 0;
|
|
conn->c_state = ST_SEC_NEG;
|
|
conn->c_HeaderDigest = 0;
|
|
conn->c_DataDigest = 0;
|
|
|
|
sess->s_active_connections++;
|
|
|
|
TAILQ_INIT(&old_waiting);
|
|
|
|
mutex_enter(&conn->c_lock);
|
|
while ((ccb = TAILQ_FIRST(&conn->c_ccbs_waiting)) != NULL) {
|
|
suspend_ccb(ccb, FALSE);
|
|
TAILQ_INSERT_TAIL(&old_waiting, ccb, ccb_chain);
|
|
}
|
|
init_sernum(&conn->c_StatSN_buf);
|
|
cv_broadcast(&conn->c_idle_cv);
|
|
mutex_exit(&conn->c_lock);
|
|
|
|
if ((rc = send_login(conn)) != 0) {
|
|
DEBOUT(("Login failed (rc %d)\n", rc));
|
|
while ((ccb = TAILQ_FIRST(&old_waiting)) != NULL) {
|
|
TAILQ_REMOVE(&old_waiting, ccb, ccb_chain);
|
|
wake_ccb(ccb, rc);
|
|
}
|
|
/* Don't attempt to recover, there seems to be something amiss */
|
|
kill_connection(conn, rc, NO_LOGOUT, FALSE);
|
|
par->status = rc;
|
|
return -1;
|
|
}
|
|
|
|
DEBC(conn, 9, ("Re-Login successful\n"));
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
|
|
conn->c_state = ST_FULL_FEATURE;
|
|
sess->s_mru_connection = conn;
|
|
|
|
while ((ccb = TAILQ_FIRST(&old_waiting)) != NULL) {
|
|
TAILQ_REMOVE(&old_waiting, ccb, ccb_chain);
|
|
mutex_enter(&conn->c_lock);
|
|
suspend_ccb(ccb, TRUE);
|
|
mutex_exit(&conn->c_lock);
|
|
|
|
rc = send_task_management(conn, ccb, NULL, TASK_REASSIGN);
|
|
/* if we get an error on reassign, restart the original request */
|
|
if (rc && ccb->ccb_pdu_waiting != NULL) {
|
|
mutex_enter(&sess->s_lock);
|
|
if (sn_a_lt_b(ccb->ccb_CmdSN, sess->s_ExpCmdSN)) {
|
|
pdu = ccb->ccb_pdu_waiting;
|
|
sn = get_sernum(sess, pdu);
|
|
|
|
/* update CmdSN */
|
|
DEBC(conn, 0, ("Resend ccb %p (%d) - updating CmdSN old %u, new %u\n",
|
|
ccb, rc, ccb->ccb_CmdSN, sn));
|
|
ccb->ccb_CmdSN = sn;
|
|
pdu->pdu_hdr.pduh_p.command.CmdSN = htonl(ccb->ccb_CmdSN);
|
|
} else {
|
|
DEBC(conn, 0, ("Resend ccb %p (%d) - CmdSN %u\n",
|
|
ccb, rc, ccb->ccb_CmdSN));
|
|
}
|
|
mutex_exit(&sess->s_lock);
|
|
resend_pdu(ccb);
|
|
} else {
|
|
DEBC(conn, 0, ("Resend ccb %p (%d) CmdSN %u - reassigned\n",
|
|
ccb, rc, ccb->ccb_CmdSN));
|
|
ccb_timeout_start(ccb, COMMAND_TIMEOUT);
|
|
}
|
|
}
|
|
|
|
mutex_enter(&sess->s_lock);
|
|
cv_broadcast(&sess->s_sess_cv);
|
|
mutex_exit(&sess->s_lock);
|
|
|
|
DEBC(conn, 0, ("Connection ReCreated successfully - status %d\n",
|
|
par->status));
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
/*
|
|
* check_login_pars:
|
|
* Check the parameters passed into login/add_connection
|
|
* for validity and consistency.
|
|
*
|
|
* Parameter:
|
|
* par The login parameters
|
|
*
|
|
* Returns: 0 on success, else an error code.
|
|
*/
|
|
|
|
static int
|
|
check_login_pars(iscsi_login_parameters_t *par)
|
|
{
|
|
int i, n;
|
|
|
|
if (par->is_present.auth_info) {
|
|
/* check consistency of authentication parameters */
|
|
|
|
if (par->auth_info.auth_number > ISCSI_AUTH_OPTIONS) {
|
|
DEBOUT(("Auth number invalid: %d\n", par->auth_info.auth_number));
|
|
return ISCSI_STATUS_PARAMETER_INVALID;
|
|
}
|
|
|
|
if (par->auth_info.auth_number > 2) {
|
|
DEBOUT(("Auth number invalid: %d\n", par->auth_info.auth_number));
|
|
return ISCSI_STATUS_NOTIMPL;
|
|
}
|
|
|
|
for (i = 0, n = 0; i < par->auth_info.auth_number; i++) {
|
|
#if 0
|
|
if (par->auth_info.auth_type[i] < ISCSI_AUTH_None) {
|
|
DEBOUT(("Auth type invalid: %d\n",
|
|
par->auth_info.auth_type[i]));
|
|
return ISCSI_STATUS_PARAMETER_INVALID;
|
|
}
|
|
#endif
|
|
if (par->auth_info.auth_type[i] > ISCSI_AUTH_CHAP) {
|
|
DEBOUT(("Auth type invalid: %d\n",
|
|
par->auth_info.auth_type[i]));
|
|
return ISCSI_STATUS_NOTIMPL;
|
|
}
|
|
n = max(n, par->auth_info.auth_type[i]);
|
|
}
|
|
if (n) {
|
|
if (!par->is_present.password ||
|
|
(par->auth_info.mutual_auth &&
|
|
!par->is_present.target_password)) {
|
|
DEBOUT(("Password missing\n"));
|
|
return ISCSI_STATUS_PARAMETER_MISSING;
|
|
}
|
|
/* Note: Default for user-name is initiator name */
|
|
}
|
|
}
|
|
if (par->login_type != ISCSI_LOGINTYPE_DISCOVERY &&
|
|
!par->is_present.TargetName) {
|
|
DEBOUT(("Target name missing, login type %d\n", par->login_type));
|
|
return ISCSI_STATUS_PARAMETER_MISSING;
|
|
}
|
|
if (par->is_present.MaxRecvDataSegmentLength) {
|
|
if (par->MaxRecvDataSegmentLength < 512 ||
|
|
par->MaxRecvDataSegmentLength > 0xffffff) {
|
|
DEBOUT(("MaxRecvDataSegmentLength invalid: %d\n",
|
|
par->MaxRecvDataSegmentLength));
|
|
return ISCSI_STATUS_PARAMETER_INVALID;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* login:
|
|
* Handle the login ioctl - Create a session:
|
|
* Alloc the session structure
|
|
* Copy session parameters
|
|
* And call create_connection to establish the connection.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The login parameters
|
|
* l IN: The lwp pointer of the caller
|
|
*/
|
|
|
|
static void
|
|
login(iscsi_login_parameters_t *par, struct lwp *l, device_t dev)
|
|
{
|
|
session_t *sess;
|
|
int rc;
|
|
|
|
DEB(99, ("ISCSI: login\n"));
|
|
|
|
if (!iscsi_InitiatorName[0]) {
|
|
DEB(1, ("No Initiator Name\n"));
|
|
par->status = ISCSI_STATUS_NO_INITIATOR_NAME;
|
|
return;
|
|
}
|
|
|
|
if ((par->status = check_login_pars(par)) != 0)
|
|
return;
|
|
|
|
/* alloc the session */
|
|
sess = malloc(sizeof(*sess), M_DEVBUF, M_WAITOK | M_ZERO);
|
|
if (sess == NULL) {
|
|
DEBOUT(("No mem for session\n"));
|
|
par->status = ISCSI_STATUS_NO_RESOURCES;
|
|
return;
|
|
}
|
|
TAILQ_INIT(&sess->s_conn_list);
|
|
TAILQ_INIT(&sess->s_ccb_pool);
|
|
|
|
mutex_init(&sess->s_lock, MUTEX_DEFAULT, IPL_BIO);
|
|
cv_init(&sess->s_sess_cv, "session");
|
|
cv_init(&sess->s_ccb_cv, "ccb");
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
/* create a unique ID */
|
|
do {
|
|
++current_id;
|
|
} while (!current_id || find_session(current_id) != NULL);
|
|
par->session_id = sess->s_id = current_id;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
create_ccbs(sess);
|
|
sess->s_login_type = par->login_type;
|
|
sess->s_CmdSN = 1;
|
|
|
|
if ((rc = create_connection(par, sess, l)) != 0) {
|
|
if (rc > 0) {
|
|
destroy_ccbs(sess);
|
|
cv_destroy(&sess->s_ccb_cv);
|
|
cv_destroy(&sess->s_sess_cv);
|
|
mutex_destroy(&sess->s_lock);
|
|
free(sess, M_DEVBUF);
|
|
}
|
|
return;
|
|
}
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
TAILQ_INSERT_HEAD(&iscsi_sessions, sess, s_sessions);
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
/* Session established, map LUNs? */
|
|
if (par->login_type == ISCSI_LOGINTYPE_MAP) {
|
|
copyinstr(par->TargetName, sess->s_tgtname,
|
|
sizeof(sess->s_tgtname), NULL);
|
|
DEB(1, ("Login: map session %d\n", sess->s_id));
|
|
if (!map_session(sess, dev)) {
|
|
DEB(1, ("Login: map session %d failed\n", sess->s_id));
|
|
kill_session(sess, ISCSI_STATUS_MAP_FAILED,
|
|
LOGOUT_SESSION, FALSE);
|
|
par->status = ISCSI_STATUS_MAP_FAILED;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* logout:
|
|
* Handle the logout ioctl - Kill a session.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The login parameters
|
|
*/
|
|
|
|
static void
|
|
logout(iscsi_logout_parameters_t *par)
|
|
{
|
|
session_t *session;
|
|
|
|
DEB(5, ("ISCSI: logout session %d\n", par->session_id));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((session = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Session %d not found\n", par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
/* If the session exists, this always succeeds */
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
|
|
kill_session(session, ISCSI_STATUS_LOGOUT, LOGOUT_SESSION, FALSE);
|
|
}
|
|
|
|
|
|
/*
|
|
* add_connection:
|
|
* Handle the add_connection ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The login parameters
|
|
* l IN: The lwp pointer of the caller
|
|
*/
|
|
|
|
static void
|
|
add_connection(iscsi_login_parameters_t *par, struct lwp *l)
|
|
{
|
|
session_t *session;
|
|
|
|
DEB(5, ("ISCSI: add_connection to session %d\n", par->session_id));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((session = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Session %d not found\n", par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
if ((par->status = check_login_pars(par)) == 0) {
|
|
create_connection(par, session, l);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* remove_connection:
|
|
* Handle the remove_connection ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The remove parameters
|
|
*/
|
|
|
|
static void
|
|
remove_connection(iscsi_remove_parameters_t *par)
|
|
{
|
|
connection_t *conn;
|
|
session_t *session;
|
|
|
|
DEB(5, ("ISCSI: remove_connection %d from session %d\n",
|
|
par->connection_id, par->session_id));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((session = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Session %d not found\n", par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
|
|
if ((conn = find_connection(session, par->connection_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Connection %d not found in session %d\n",
|
|
par->connection_id, par->session_id));
|
|
|
|
par->status = ISCSI_STATUS_INVALID_CONNECTION_ID;
|
|
} else {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
kill_connection(conn, ISCSI_STATUS_LOGOUT, LOGOUT_CONNECTION,
|
|
FALSE);
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* restore_connection:
|
|
* Handle the restore_connection ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The login parameters
|
|
* l IN: The lwp pointer of the caller
|
|
*/
|
|
|
|
static void
|
|
restore_connection(iscsi_login_parameters_t *par, struct lwp *l)
|
|
{
|
|
session_t *sess;
|
|
connection_t *conn;
|
|
|
|
DEB(1, ("ISCSI: restore_connection %d of session %d\n",
|
|
par->connection_id, par->session_id));
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((sess = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Session %d not found\n", par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
|
|
if ((conn = find_connection(sess, par->connection_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Connection %d not found in session %d\n",
|
|
par->connection_id, par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_CONNECTION_ID;
|
|
return;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
if ((par->status = check_login_pars(par)) == 0) {
|
|
recreate_connection(par, sess, conn, l);
|
|
}
|
|
}
|
|
|
|
|
|
#ifndef ISCSI_MINIMAL
|
|
|
|
/*
|
|
* map_databuf:
|
|
* Map user-supplied data buffer into kernel space.
|
|
*
|
|
* Parameter:
|
|
* p IN: The proc pointer of the caller
|
|
* buf IN/OUT: The virtual address of the buffer, modified
|
|
* on exit to reflect kernel VA.
|
|
* datalen IN: The size of the data buffer
|
|
*
|
|
* Returns:
|
|
* An ISCSI status code on error, else 0.
|
|
*/
|
|
|
|
uint32_t
|
|
map_databuf(struct proc *p, void **buf, uint32_t datalen)
|
|
{
|
|
vaddr_t kva, databuf, offs;
|
|
int error;
|
|
|
|
/* page align address */
|
|
databuf = (vaddr_t) * buf & ~PAGE_MASK;
|
|
/* offset of VA into page */
|
|
offs = (vaddr_t) * buf & PAGE_MASK;
|
|
/* round to full page including offset */
|
|
datalen = (datalen + offs + PAGE_MASK) & ~PAGE_MASK;
|
|
|
|
/* Do some magic to the vm space reference count (copied from "copyin_proc") */
|
|
if ((p->p_sflag & PS_WEXIT) || (p->p_vmspace->vm_refcnt < 1)) {
|
|
return ISCSI_STATUS_NO_RESOURCES;
|
|
}
|
|
p->p_vmspace->vm_refcnt++;
|
|
|
|
/* this is lifted from uvm_io */
|
|
error = uvm_map_extract(&p->p_vmspace->vm_map, databuf, datalen,
|
|
kernel_map, &kva,
|
|
UVM_EXTRACT_QREF | UVM_EXTRACT_CONTIG |
|
|
UVM_EXTRACT_FIXPROT);
|
|
if (error) {
|
|
DEBOUT(("uvm_map_extract failed, error = %d\n", error));
|
|
return ISCSI_STATUS_NO_RESOURCES;
|
|
}
|
|
/* add offset back into kernel VA */
|
|
*buf = (void *) (kva + offs);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* unmap_databuf:
|
|
* Remove kernel space mapping of data buffer.
|
|
*
|
|
* Parameter:
|
|
* p IN: The proc pointer of the caller
|
|
* buf IN: The kernel virtual address of the buffer
|
|
* datalen IN: The size of the data buffer
|
|
*
|
|
* Returns:
|
|
* An ISCSI status code on error, else 0.
|
|
*/
|
|
|
|
void
|
|
unmap_databuf(struct proc *p, void *buf, uint32_t datalen)
|
|
{
|
|
struct vm_map_entry *dead_entries;
|
|
vaddr_t databuf;
|
|
|
|
/* round to full page */
|
|
datalen = (datalen + ((uintptr_t) buf & PAGE_MASK) + PAGE_MASK) & ~PAGE_MASK;
|
|
/* page align address */
|
|
databuf = (vaddr_t) buf & ~PAGE_MASK;
|
|
|
|
/* following code lifted almost verbatim from uvm_io.c */
|
|
vm_map_lock(kernel_map);
|
|
uvm_unmap_remove(kernel_map, databuf, databuf + datalen, &dead_entries,
|
|
0);
|
|
vm_map_unlock(kernel_map);
|
|
if (dead_entries != NULL) {
|
|
uvm_unmap_detach(dead_entries, AMAP_REFALL);
|
|
}
|
|
/* this apparently reverses the magic to the vm ref count, from copyin_proc */
|
|
uvmspace_free(p->p_vmspace);
|
|
}
|
|
|
|
|
|
/*
|
|
* io_command:
|
|
* Handle the io_command ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The iocommand parameters
|
|
* l IN: The lwp pointer of the caller
|
|
*/
|
|
|
|
static void
|
|
io_command(iscsi_iocommand_parameters_t *par, struct lwp *l)
|
|
{
|
|
uint32_t datalen = par->req.datalen;
|
|
void *databuf = par->req.databuf;
|
|
session_t *session;
|
|
|
|
DEB(9, ("ISCSI: io_command, SID=%d, lun=%" PRIu64 "\n", par->session_id, par->lun));
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((session = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Session %d not found\n", par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
par->req.senselen_used = 0;
|
|
par->req.datalen_used = 0;
|
|
par->req.error = 0;
|
|
par->req.status = 0;
|
|
par->req.retsts = SCCMD_UNKNOWN; /* init to failure code */
|
|
|
|
if (par->req.cmdlen > 16 || par->req.senselen > sizeof(par->req.sense)) {
|
|
par->status = ISCSI_STATUS_PARAMETER_INVALID;
|
|
return;
|
|
}
|
|
|
|
if (datalen && (par->status = map_databuf(l->l_proc,
|
|
&par->req.databuf, datalen)) != 0) {
|
|
return;
|
|
}
|
|
par->status = send_io_command(session, par->lun, &par->req,
|
|
par->options.immediate, par->connection_id);
|
|
|
|
if (datalen) {
|
|
unmap_databuf(l->l_proc, par->req.databuf, datalen);
|
|
par->req.databuf = databuf; /* restore original addr */
|
|
}
|
|
|
|
switch (par->status) {
|
|
case ISCSI_STATUS_SUCCESS:
|
|
par->req.retsts = SCCMD_OK;
|
|
break;
|
|
|
|
case ISCSI_STATUS_TARGET_BUSY:
|
|
par->req.retsts = SCCMD_BUSY;
|
|
break;
|
|
|
|
case ISCSI_STATUS_TIMEOUT:
|
|
case ISCSI_STATUS_SOCKET_ERROR:
|
|
par->req.retsts = SCCMD_TIMEOUT;
|
|
break;
|
|
|
|
default:
|
|
par->req.retsts = (par->req.senselen_used) ? SCCMD_SENSE
|
|
: SCCMD_UNKNOWN;
|
|
break;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* send_targets:
|
|
* Handle the send_targets ioctl.
|
|
* Note: If the passed buffer is too small to hold the complete response,
|
|
* the response is kept in the session structure so it can be
|
|
* retrieved with the next call to this function without having to go to
|
|
* the target again. Once the complete response has been retrieved, it
|
|
* is discarded.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The send_targets parameters
|
|
*/
|
|
|
|
static void
|
|
send_targets(iscsi_send_targets_parameters_t *par)
|
|
{
|
|
int rc;
|
|
uint32_t rlen, cplen;
|
|
session_t *sess;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((sess = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEBOUT(("Session %d not found\n", par->session_id));
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
DEB(9, ("ISCSI: send_targets, rsp_size=%d; Saved list: %p\n",
|
|
par->response_size, sess->s_target_list));
|
|
|
|
if (sess->s_target_list == NULL) {
|
|
rc = send_send_targets(sess, par->key);
|
|
if (rc) {
|
|
par->status = rc;
|
|
return;
|
|
}
|
|
}
|
|
rlen = sess->s_target_list_len;
|
|
par->response_total = rlen;
|
|
cplen = min(par->response_size, rlen);
|
|
if (cplen) {
|
|
copyout(sess->s_target_list, par->response_buffer, cplen);
|
|
}
|
|
par->response_used = cplen;
|
|
|
|
/* If all of the response was copied, don't keep it around */
|
|
if (rlen && par->response_used == rlen) {
|
|
free(sess->s_target_list, M_TEMP);
|
|
sess->s_target_list = NULL;
|
|
}
|
|
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*
|
|
* set_node_name:
|
|
* Handle the set_node_name ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The set_node_name parameters
|
|
*/
|
|
|
|
static void
|
|
set_node_name(iscsi_set_node_name_parameters_t *par)
|
|
{
|
|
|
|
if (strlen(par->InitiatorName) >= ISCSI_STRING_LENGTH ||
|
|
strlen(par->InitiatorAlias) >= ISCSI_STRING_LENGTH) {
|
|
DEBOUT(("*** set_node_name string too long!\n"));
|
|
par->status = ISCSI_STATUS_PARAMETER_INVALID;
|
|
return;
|
|
}
|
|
strlcpy(iscsi_InitiatorName, par->InitiatorName, sizeof(iscsi_InitiatorName));
|
|
strlcpy(iscsi_InitiatorAlias, par->InitiatorAlias, sizeof(iscsi_InitiatorAlias));
|
|
memcpy(&iscsi_InitiatorISID, par->ISID, 6);
|
|
DEB(5, ("ISCSI: set_node_name, ISID A=%x, B=%x, C=%x, D=%x\n",
|
|
iscsi_InitiatorISID.ISID_A, iscsi_InitiatorISID.ISID_B,
|
|
iscsi_InitiatorISID.ISID_C, iscsi_InitiatorISID.ISID_D));
|
|
|
|
if (!iscsi_InitiatorISID.ISID_A && !iscsi_InitiatorISID.ISID_B &&
|
|
!iscsi_InitiatorISID.ISID_C && !iscsi_InitiatorISID.ISID_D) {
|
|
iscsi_InitiatorISID.ISID_A = T_FORMAT_EN;
|
|
iscsi_InitiatorISID.ISID_B = htons(0x1);
|
|
iscsi_InitiatorISID.ISID_C = 0x37;
|
|
iscsi_InitiatorISID.ISID_D = 0;
|
|
}
|
|
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
/*
|
|
* connection_status:
|
|
* Handle the connection_status ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The status parameters
|
|
*/
|
|
|
|
static void
|
|
connection_status(iscsi_conn_status_parameters_t *par)
|
|
{
|
|
connection_t *conn;
|
|
session_t *sess;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if ((sess = find_session(par->session_id)) == NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
par->status = ISCSI_STATUS_INVALID_SESSION_ID;
|
|
return;
|
|
}
|
|
|
|
if (par->connection_id) {
|
|
conn = find_connection(sess, par->connection_id);
|
|
} else {
|
|
conn = TAILQ_FIRST(&sess->s_conn_list);
|
|
}
|
|
par->status = (conn == NULL) ? ISCSI_STATUS_INVALID_CONNECTION_ID :
|
|
ISCSI_STATUS_SUCCESS;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
DEB(9, ("ISCSI: connection_status, session %d connection %d --> %d\n",
|
|
par->session_id, par->connection_id, par->status));
|
|
}
|
|
|
|
|
|
/*
|
|
* get_version:
|
|
* Handle the get_version ioctl.
|
|
*
|
|
* Parameter:
|
|
* par IN/OUT: The version parameters
|
|
*/
|
|
|
|
static void
|
|
get_version(iscsi_get_version_parameters_t *par)
|
|
{
|
|
par->status = ISCSI_STATUS_SUCCESS;
|
|
par->interface_version = INTERFACE_VERSION;
|
|
par->major = VERSION_MAJOR;
|
|
par->minor = VERSION_MINOR;
|
|
strlcpy(par->version_string, VERSION_STRING,
|
|
sizeof(par->version_string));
|
|
}
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/*
|
|
* kill_all_sessions:
|
|
* Terminate all sessions (called when the driver unloads).
|
|
*/
|
|
|
|
int
|
|
kill_all_sessions(void)
|
|
{
|
|
session_t *sess;
|
|
int rc = 0;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
while ((sess = TAILQ_FIRST(&iscsi_sessions)) != NULL) {
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
kill_session(sess, ISCSI_STATUS_DRIVER_UNLOAD, LOGOUT_SESSION,
|
|
FALSE);
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
}
|
|
if (TAILQ_FIRST(&iscsi_sessions) != NULL) {
|
|
DEBOUT(("Failed to kill all sessions\n"));
|
|
rc = EBUSY;
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* handle_connection_error:
|
|
* Deal with a problem during send or receive.
|
|
*
|
|
* Parameter:
|
|
* conn The connection the problem is associated with
|
|
* status The status code to insert into any unfinished CCBs
|
|
* dologout Whether Logout should be attempted
|
|
*/
|
|
|
|
void
|
|
handle_connection_error(connection_t *conn, uint32_t status, int dologout)
|
|
{
|
|
|
|
DEBC(conn, 0, ("*** Connection Error, status=%d, logout=%d, state=%d\n",
|
|
status, dologout, conn->c_state));
|
|
|
|
if (!conn->c_terminating && conn->c_state <= ST_LOGOUT_SENT) {
|
|
/* if we get an error while winding down, escalate it */
|
|
if (dologout >= 0 && conn->c_state >= ST_WINDING_DOWN) {
|
|
dologout = NO_LOGOUT;
|
|
}
|
|
kill_connection(conn, status, dologout, TRUE);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* remove a connection from session and add to the cleanup list
|
|
*/
|
|
void
|
|
add_connection_cleanup(connection_t *conn)
|
|
{
|
|
session_t *sess;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (conn->c_in_session) {
|
|
sess = conn->c_session;
|
|
conn->c_in_session = FALSE;
|
|
conn->c_session = NULL;
|
|
TAILQ_REMOVE(&sess->s_conn_list, conn, c_connections);
|
|
sess->s_mru_connection = TAILQ_FIRST(&sess->s_conn_list);
|
|
}
|
|
TAILQ_INSERT_TAIL(&iscsi_cleanupc_list, conn, c_connections);
|
|
iscsi_notify_cleanup();
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
/*
|
|
* callout wrappers for timeouts, the work is done by the cleanup thread
|
|
*/
|
|
void
|
|
connection_timeout_co(void *par)
|
|
{
|
|
connection_t *conn = par;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
conn->c_timedout = TOUT_QUEUED;
|
|
TAILQ_INSERT_TAIL(&iscsi_timeout_conn_list, conn, c_tchain);
|
|
iscsi_notify_cleanup();
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
void
|
|
connection_timeout_start(connection_t *conn, int ticks)
|
|
{
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (conn->c_timedout != TOUT_QUEUED) {
|
|
conn->c_timedout = TOUT_ARMED;
|
|
callout_schedule(&conn->c_timeout, ticks);
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
void
|
|
connection_timeout_stop(connection_t *conn)
|
|
{
|
|
callout_stop(&conn->c_timeout);
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (conn->c_timedout == TOUT_QUEUED) {
|
|
TAILQ_REMOVE(&iscsi_timeout_conn_list, conn, c_tchain);
|
|
conn->c_timedout = TOUT_NONE;
|
|
}
|
|
if (curlwp != iscsi_cleanproc) {
|
|
while (conn->c_timedout == TOUT_BUSY)
|
|
kpause("connbusy", false, 1, &iscsi_cleanup_mtx);
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
void
|
|
ccb_timeout_co(void *par)
|
|
{
|
|
ccb_t *ccb = par;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
ccb->ccb_timedout = TOUT_QUEUED;
|
|
TAILQ_INSERT_TAIL(&iscsi_timeout_ccb_list, ccb, ccb_tchain);
|
|
iscsi_notify_cleanup();
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
void
|
|
ccb_timeout_start(ccb_t *ccb, int ticks)
|
|
{
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (ccb->ccb_timedout != TOUT_QUEUED) {
|
|
ccb->ccb_timedout = TOUT_ARMED;
|
|
callout_schedule(&ccb->ccb_timeout, ticks);
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
void
|
|
ccb_timeout_stop(ccb_t *ccb)
|
|
{
|
|
callout_stop(&ccb->ccb_timeout);
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (ccb->ccb_timedout == TOUT_QUEUED) {
|
|
TAILQ_REMOVE(&iscsi_timeout_ccb_list, ccb, ccb_tchain);
|
|
ccb->ccb_timedout = TOUT_NONE;
|
|
}
|
|
if (curlwp != iscsi_cleanproc) {
|
|
while (ccb->ccb_timedout == TOUT_BUSY)
|
|
kpause("ccbbusy", false, 1, &iscsi_cleanup_mtx);
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
/*
|
|
* iscsi_cleanup_thread
|
|
* Global thread to handle connection and session cleanup after termination.
|
|
*/
|
|
|
|
static void
|
|
iscsi_cleanup_thread(void *par)
|
|
{
|
|
int s, rc;
|
|
session_t *sess, *nxts;
|
|
connection_t *conn, *nxtc;
|
|
ccb_t *ccb;
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
while (iscsi_num_send_threads || !iscsi_detaching ||
|
|
!TAILQ_EMPTY(&iscsi_cleanupc_list) || !TAILQ_EMPTY(&iscsi_cleanups_list)) {
|
|
TAILQ_FOREACH_SAFE(conn, &iscsi_cleanupc_list, c_connections, nxtc) {
|
|
|
|
TAILQ_REMOVE(&iscsi_cleanupc_list, conn, c_connections);
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
sess = conn->c_session;
|
|
|
|
/*
|
|
* This implies that connection cleanup only runs when
|
|
* the send/recv threads have been killed
|
|
*/
|
|
DEBC(conn, 5, ("Cleanup: Waiting for threads to exit\n"));
|
|
while (conn->c_sendproc || conn->c_rcvproc)
|
|
kpause("threads", false, hz, NULL);
|
|
|
|
for (s=1; conn->c_usecount > 0 && s < 3; ++s)
|
|
kpause("usecount", false, hz, NULL);
|
|
|
|
if (conn->c_usecount > 0) {
|
|
DEBC(conn, 5, ("Cleanup: %d CCBs busy\n", conn->c_usecount));
|
|
/* retry later */
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
TAILQ_INSERT_HEAD(&iscsi_cleanupc_list, conn, c_connections);
|
|
continue;
|
|
}
|
|
|
|
KASSERT(!conn->c_in_session);
|
|
|
|
callout_halt(&conn->c_timeout, NULL);
|
|
closef(conn->c_sock);
|
|
callout_destroy(&conn->c_timeout);
|
|
cv_destroy(&conn->c_idle_cv);
|
|
cv_destroy(&conn->c_ccb_cv);
|
|
cv_destroy(&conn->c_pdu_cv);
|
|
cv_destroy(&conn->c_conn_cv);
|
|
mutex_destroy(&conn->c_lock);
|
|
free(conn, M_DEVBUF);
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
|
|
if (--sess->s_total_connections == 0) {
|
|
DEB(1, ("Cleanup: session %d\n", sess->s_id));
|
|
if (!sess->s_terminating) {
|
|
sess->s_terminating = ISCSI_CONNECTION_TERMINATED;
|
|
KASSERT(sess->s_sessions.tqe_prev != NULL);
|
|
TAILQ_REMOVE(&iscsi_sessions, sess, s_sessions);
|
|
sess->s_sessions.tqe_next = NULL;
|
|
sess->s_sessions.tqe_prev = NULL;
|
|
}
|
|
KASSERT(sess->s_sessions.tqe_prev == NULL);
|
|
TAILQ_INSERT_HEAD(&iscsi_cleanups_list, sess, s_sessions);
|
|
}
|
|
}
|
|
|
|
TAILQ_FOREACH_SAFE(sess, &iscsi_cleanups_list, s_sessions, nxts) {
|
|
if (sess->s_refcount > 0)
|
|
continue;
|
|
TAILQ_REMOVE(&iscsi_cleanups_list, sess, s_sessions);
|
|
sess->s_sessions.tqe_next = NULL;
|
|
sess->s_sessions.tqe_prev = NULL;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
DEB(1, ("Cleanup: Unmap session %d\n", sess->s_id));
|
|
if (unmap_session(sess) == 0) {
|
|
DEB(1, ("Cleanup: Unmap session %d failed\n", sess->s_id));
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
TAILQ_INSERT_HEAD(&iscsi_cleanups_list, sess, s_sessions);
|
|
continue;
|
|
}
|
|
|
|
if (sess->s_target_list != NULL)
|
|
free(sess->s_target_list, M_TEMP);
|
|
|
|
/* notify event handlers of session shutdown */
|
|
add_event(ISCSI_SESSION_TERMINATED, sess->s_id, 0, sess->s_terminating);
|
|
DEB(1, ("Cleanup: session ended %d\n", sess->s_id));
|
|
|
|
destroy_ccbs(sess);
|
|
cv_destroy(&sess->s_ccb_cv);
|
|
cv_destroy(&sess->s_sess_cv);
|
|
mutex_destroy(&sess->s_lock);
|
|
free(sess, M_DEVBUF);
|
|
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
}
|
|
|
|
/* handle ccb timeouts */
|
|
while ((ccb = TAILQ_FIRST(&iscsi_timeout_ccb_list)) != NULL) {
|
|
TAILQ_REMOVE(&iscsi_timeout_ccb_list, ccb, ccb_tchain);
|
|
KASSERT(ccb->ccb_timedout == TOUT_QUEUED);
|
|
ccb->ccb_timedout = TOUT_BUSY;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
ccb_timeout(ccb);
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (ccb->ccb_timedout == TOUT_BUSY)
|
|
ccb->ccb_timedout = TOUT_NONE;
|
|
}
|
|
|
|
/* handle connection timeouts */
|
|
while ((conn = TAILQ_FIRST(&iscsi_timeout_conn_list)) != NULL) {
|
|
TAILQ_REMOVE(&iscsi_timeout_conn_list, conn, c_tchain);
|
|
KASSERT(conn->c_timedout == TOUT_QUEUED);
|
|
conn->c_timedout = TOUT_BUSY;
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
connection_timeout(conn);
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
if (conn->c_timedout == TOUT_BUSY)
|
|
conn->c_timedout = TOUT_NONE;
|
|
}
|
|
|
|
/* Go to sleep, but wake up every 30 seconds to
|
|
* check for dead event handlers */
|
|
rc = cv_timedwait(&iscsi_cleanup_cv, &iscsi_cleanup_mtx,
|
|
(TAILQ_FIRST(&event_handlers)) ? 120 * hz : 0);
|
|
|
|
/* if timed out, not woken up */
|
|
if (rc == EWOULDBLOCK)
|
|
check_event_handlers();
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
add_event(ISCSI_DRIVER_TERMINATING, 0, 0, ISCSI_STATUS_DRIVER_UNLOAD);
|
|
|
|
/*
|
|
* Wait for all event handlers to deregister, but don't wait more
|
|
* than 1 minute (assume registering app has died if it takes longer).
|
|
*/
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
for (s = 0; TAILQ_FIRST(&event_handlers) != NULL && s < 60; s++)
|
|
kpause("waiteventclr", true, hz, &iscsi_cleanup_mtx);
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
iscsi_cleanproc = NULL;
|
|
DEB(5, ("Cleanup thread exits\n"));
|
|
kthread_exit(0);
|
|
}
|
|
|
|
void
|
|
iscsi_init_cleanup(void)
|
|
{
|
|
|
|
mutex_init(&iscsi_cleanup_mtx, MUTEX_DEFAULT, IPL_BIO);
|
|
cv_init(&iscsi_cleanup_cv, "cleanup");
|
|
cv_init(&iscsi_event_cv, "iscsievtwait");
|
|
|
|
if (kthread_create(PRI_NONE, KTHREAD_MPSAFE, NULL, iscsi_cleanup_thread,
|
|
NULL, &iscsi_cleanproc, "iscsi_cleanup") != 0) {
|
|
panic("Can't create cleanup thread!");
|
|
}
|
|
}
|
|
|
|
int
|
|
iscsi_destroy_cleanup(void)
|
|
{
|
|
|
|
iscsi_detaching = true;
|
|
mutex_enter(&iscsi_cleanup_mtx);
|
|
while (iscsi_cleanproc != NULL) {
|
|
iscsi_notify_cleanup();
|
|
kpause("detach_wait", false, hz, &iscsi_cleanup_mtx);
|
|
}
|
|
mutex_exit(&iscsi_cleanup_mtx);
|
|
|
|
cv_destroy(&iscsi_event_cv);
|
|
cv_destroy(&iscsi_cleanup_cv);
|
|
mutex_destroy(&iscsi_cleanup_mtx);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
iscsi_notify_cleanup(void)
|
|
{
|
|
KASSERT(mutex_owned(&iscsi_cleanup_mtx));
|
|
|
|
cv_signal(&iscsi_cleanup_cv);
|
|
}
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/*
|
|
* iscsi_ioctl:
|
|
* Driver ioctl entry.
|
|
*
|
|
* Parameter:
|
|
* file File structure
|
|
* cmd The ioctl Command
|
|
* addr IN/OUT: The command parameter
|
|
* flag Flags (ignored)
|
|
* l IN: The lwp object of the caller
|
|
*/
|
|
|
|
int
|
|
iscsiioctl(struct file *fp, u_long cmd, void *addr)
|
|
{
|
|
struct lwp *l = curlwp;
|
|
struct iscsifd *d = fp->f_iscsi;
|
|
|
|
DEB(1, ("ISCSI Ioctl cmd = %x\n", (int) cmd));
|
|
|
|
switch (cmd) {
|
|
case ISCSI_GET_VERSION:
|
|
get_version((iscsi_get_version_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_LOGIN:
|
|
login((iscsi_login_parameters_t *) addr, l, d->fd_dev);
|
|
break;
|
|
|
|
case ISCSI_ADD_CONNECTION:
|
|
add_connection((iscsi_login_parameters_t *) addr, l);
|
|
break;
|
|
|
|
case ISCSI_RESTORE_CONNECTION:
|
|
restore_connection((iscsi_login_parameters_t *) addr, l);
|
|
break;
|
|
|
|
case ISCSI_LOGOUT:
|
|
logout((iscsi_logout_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_REMOVE_CONNECTION:
|
|
remove_connection((iscsi_remove_parameters_t *) addr);
|
|
break;
|
|
|
|
#ifndef ISCSI_MINIMAL
|
|
case ISCSI_IO_COMMAND:
|
|
io_command((iscsi_iocommand_parameters_t *) addr, l);
|
|
break;
|
|
#endif
|
|
|
|
case ISCSI_SEND_TARGETS:
|
|
send_targets((iscsi_send_targets_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_SET_NODE_NAME:
|
|
set_node_name((iscsi_set_node_name_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_CONNECTION_STATUS:
|
|
connection_status((iscsi_conn_status_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_REGISTER_EVENT:
|
|
register_event((iscsi_register_event_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_DEREGISTER_EVENT:
|
|
deregister_event((iscsi_register_event_parameters_t *) addr);
|
|
break;
|
|
|
|
case ISCSI_WAIT_EVENT:
|
|
check_event((iscsi_wait_event_parameters_t *) addr, TRUE);
|
|
break;
|
|
|
|
case ISCSI_POLL_EVENT:
|
|
check_event((iscsi_wait_event_parameters_t *) addr, FALSE);
|
|
break;
|
|
|
|
default:
|
|
DEBOUT(("Invalid IO-Control Code\n"));
|
|
return ENOTTY;
|
|
}
|
|
|
|
/*
|
|
* NOTE: We return 0 even if the function fails as long as the ioctl code
|
|
* is good, so the status code is copied back to the caller.
|
|
*/
|
|
return 0;
|
|
}
|