提交 e5bd7b54 编写于 作者: M Mike Christie 提交者: James Bottomley

[SCSI] libiscsi: Support drivers initiating session removal

If the driver knows when hardware is removed like with cxgb3i,
bnx2i, qla4xxx and iser then we will want to remove the sessions/devices
that are bound to that device before removing the host.

cxgb3i and in the future bnx2i will remove the host and that will
remove all the sessions on the hba. iser can call iscsi_kill_session
when it gets an event that indicates that a hca is removed.
And when qla4xxx is hooked in to the lib (it is only hooked into
the class right now) it can call iscsi remove host like the
partial offload card drivers.
Signed-off-by: NMike Christie <michaelc@cs.wisc.edu>
Signed-off-by: NJames Bottomley <James.Bottomley@HansenPartnership.com>
上级 1d9edf02
...@@ -378,6 +378,7 @@ static void iscsi_iser_session_destroy(struct iscsi_cls_session *cls_session) ...@@ -378,6 +378,7 @@ static void iscsi_iser_session_destroy(struct iscsi_cls_session *cls_session)
{ {
struct Scsi_Host *shost = iscsi_session_to_shost(cls_session); struct Scsi_Host *shost = iscsi_session_to_shost(cls_session);
iscsi_session_teardown(cls_session);
iscsi_host_remove(shost); iscsi_host_remove(shost);
iscsi_host_free(shost); iscsi_host_free(shost);
} }
......
...@@ -1885,6 +1885,7 @@ static void iscsi_tcp_session_destroy(struct iscsi_cls_session *cls_session) ...@@ -1885,6 +1885,7 @@ static void iscsi_tcp_session_destroy(struct iscsi_cls_session *cls_session)
struct Scsi_Host *shost = iscsi_session_to_shost(cls_session); struct Scsi_Host *shost = iscsi_session_to_shost(cls_session);
iscsi_r2tpool_free(cls_session->dd_data); iscsi_r2tpool_free(cls_session->dd_data);
iscsi_session_teardown(cls_session);
iscsi_host_remove(shost); iscsi_host_remove(shost);
iscsi_host_free(shost); iscsi_host_free(shost);
......
...@@ -983,6 +983,38 @@ struct iscsi_task *iscsi_itt_to_ctask(struct iscsi_conn *conn, itt_t itt) ...@@ -983,6 +983,38 @@ struct iscsi_task *iscsi_itt_to_ctask(struct iscsi_conn *conn, itt_t itt)
} }
EXPORT_SYMBOL_GPL(iscsi_itt_to_ctask); EXPORT_SYMBOL_GPL(iscsi_itt_to_ctask);
void iscsi_session_failure(struct iscsi_cls_session *cls_session,
enum iscsi_err err)
{
struct iscsi_session *session = cls_session->dd_data;
struct iscsi_conn *conn;
struct device *dev;
unsigned long flags;
spin_lock_irqsave(&session->lock, flags);
conn = session->leadconn;
if (session->state == ISCSI_STATE_TERMINATE || !conn) {
spin_unlock_irqrestore(&session->lock, flags);
return;
}
dev = get_device(&conn->cls_conn->dev);
spin_unlock_irqrestore(&session->lock, flags);
if (!dev)
return;
/*
* if the host is being removed bypass the connection
* recovery initialization because we are going to kill
* the session.
*/
if (err == ISCSI_ERR_INVALID_HOST)
iscsi_conn_error_event(conn->cls_conn, err);
else
iscsi_conn_failure(conn, err);
put_device(dev);
}
EXPORT_SYMBOL_GPL(iscsi_session_failure);
void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err) void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err)
{ {
struct iscsi_session *session = conn->session; struct iscsi_session *session = conn->session;
...@@ -997,9 +1029,10 @@ void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err) ...@@ -997,9 +1029,10 @@ void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err)
if (conn->stop_stage == 0) if (conn->stop_stage == 0)
session->state = ISCSI_STATE_FAILED; session->state = ISCSI_STATE_FAILED;
spin_unlock_irqrestore(&session->lock, flags); spin_unlock_irqrestore(&session->lock, flags);
set_bit(ISCSI_SUSPEND_BIT, &conn->suspend_tx); set_bit(ISCSI_SUSPEND_BIT, &conn->suspend_tx);
set_bit(ISCSI_SUSPEND_BIT, &conn->suspend_rx); set_bit(ISCSI_SUSPEND_BIT, &conn->suspend_rx);
iscsi_conn_error(conn->cls_conn, err); iscsi_conn_error_event(conn->cls_conn, err);
} }
EXPORT_SYMBOL_GPL(iscsi_conn_failure); EXPORT_SYMBOL_GPL(iscsi_conn_failure);
...@@ -1905,6 +1938,7 @@ struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht, ...@@ -1905,6 +1938,7 @@ struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht,
int dd_data_size, uint16_t qdepth) int dd_data_size, uint16_t qdepth)
{ {
struct Scsi_Host *shost; struct Scsi_Host *shost;
struct iscsi_host *ihost;
shost = scsi_host_alloc(sht, sizeof(struct iscsi_host) + dd_data_size); shost = scsi_host_alloc(sht, sizeof(struct iscsi_host) + dd_data_size);
if (!shost) if (!shost)
...@@ -1919,22 +1953,43 @@ struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht, ...@@ -1919,22 +1953,43 @@ struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht,
qdepth = ISCSI_DEF_CMD_PER_LUN; qdepth = ISCSI_DEF_CMD_PER_LUN;
} }
shost->cmd_per_lun = qdepth; shost->cmd_per_lun = qdepth;
ihost = shost_priv(shost);
spin_lock_init(&ihost->lock);
ihost->state = ISCSI_HOST_SETUP;
ihost->num_sessions = 0;
init_waitqueue_head(&ihost->session_removal_wq);
return shost; return shost;
} }
EXPORT_SYMBOL_GPL(iscsi_host_alloc); EXPORT_SYMBOL_GPL(iscsi_host_alloc);
static void iscsi_notify_host_removed(struct iscsi_cls_session *cls_session)
{
iscsi_session_failure(cls_session, ISCSI_ERR_INVALID_HOST);
}
/** /**
* iscsi_host_remove - remove host and sessions * iscsi_host_remove - remove host and sessions
* @shost: scsi host * @shost: scsi host
* *
* This will also remove any sessions attached to the host, but if userspace * If there are any sessions left, this will initiate the removal and wait
* is managing the session at the same time this will break. TODO: add * for the completion.
* refcounting to the netlink iscsi interface so a rmmod or host hot unplug
* does not remove the memory from under us.
*/ */
void iscsi_host_remove(struct Scsi_Host *shost) void iscsi_host_remove(struct Scsi_Host *shost)
{ {
iscsi_host_for_each_session(shost, iscsi_session_teardown); struct iscsi_host *ihost = shost_priv(shost);
unsigned long flags;
spin_lock_irqsave(&ihost->lock, flags);
ihost->state = ISCSI_HOST_REMOVED;
spin_unlock_irqrestore(&ihost->lock, flags);
iscsi_host_for_each_session(shost, iscsi_notify_host_removed);
wait_event_interruptible(ihost->session_removal_wq,
ihost->num_sessions == 0);
if (signal_pending(current))
flush_signals(current);
scsi_remove_host(shost); scsi_remove_host(shost);
} }
EXPORT_SYMBOL_GPL(iscsi_host_remove); EXPORT_SYMBOL_GPL(iscsi_host_remove);
...@@ -1950,6 +2005,27 @@ void iscsi_host_free(struct Scsi_Host *shost) ...@@ -1950,6 +2005,27 @@ void iscsi_host_free(struct Scsi_Host *shost)
} }
EXPORT_SYMBOL_GPL(iscsi_host_free); EXPORT_SYMBOL_GPL(iscsi_host_free);
static void iscsi_host_dec_session_cnt(struct Scsi_Host *shost)
{
struct iscsi_host *ihost = shost_priv(shost);
unsigned long flags;
shost = scsi_host_get(shost);
if (!shost) {
printk(KERN_ERR "Invalid state. Cannot notify host removal "
"of session teardown event because host already "
"removed.\n");
return;
}
spin_lock_irqsave(&ihost->lock, flags);
ihost->num_sessions--;
if (ihost->num_sessions == 0)
wake_up(&ihost->session_removal_wq);
spin_unlock_irqrestore(&ihost->lock, flags);
scsi_host_put(shost);
}
/** /**
* iscsi_session_setup - create iscsi cls session and host and session * iscsi_session_setup - create iscsi cls session and host and session
* @iscsit: iscsi transport template * @iscsit: iscsi transport template
...@@ -1970,9 +2046,19 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost, ...@@ -1970,9 +2046,19 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost,
uint16_t cmds_max, int cmd_task_size, uint16_t cmds_max, int cmd_task_size,
uint32_t initial_cmdsn, unsigned int id) uint32_t initial_cmdsn, unsigned int id)
{ {
struct iscsi_host *ihost = shost_priv(shost);
struct iscsi_session *session; struct iscsi_session *session;
struct iscsi_cls_session *cls_session; struct iscsi_cls_session *cls_session;
int cmd_i, scsi_cmds, total_cmds = cmds_max; int cmd_i, scsi_cmds, total_cmds = cmds_max;
unsigned long flags;
spin_lock_irqsave(&ihost->lock, flags);
if (ihost->state == ISCSI_HOST_REMOVED) {
spin_unlock_irqrestore(&ihost->lock, flags);
return NULL;
}
ihost->num_sessions++;
spin_unlock_irqrestore(&ihost->lock, flags);
if (!total_cmds) if (!total_cmds)
total_cmds = ISCSI_DEF_XMIT_CMDS_MAX; total_cmds = ISCSI_DEF_XMIT_CMDS_MAX;
...@@ -1985,7 +2071,7 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost, ...@@ -1985,7 +2071,7 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost,
printk(KERN_ERR "iscsi: invalid can_queue of %d. can_queue " printk(KERN_ERR "iscsi: invalid can_queue of %d. can_queue "
"must be a power of two that is at least %d.\n", "must be a power of two that is at least %d.\n",
total_cmds, ISCSI_TOTAL_CMDS_MIN); total_cmds, ISCSI_TOTAL_CMDS_MIN);
return NULL; goto dec_session_count;
} }
if (total_cmds > ISCSI_TOTAL_CMDS_MAX) { if (total_cmds > ISCSI_TOTAL_CMDS_MAX) {
...@@ -2009,7 +2095,7 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost, ...@@ -2009,7 +2095,7 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost,
cls_session = iscsi_alloc_session(shost, iscsit, cls_session = iscsi_alloc_session(shost, iscsit,
sizeof(struct iscsi_session)); sizeof(struct iscsi_session));
if (!cls_session) if (!cls_session)
return NULL; goto dec_session_count;
session = cls_session->dd_data; session = cls_session->dd_data;
session->cls_session = cls_session; session->cls_session = cls_session;
session->host = shost; session->host = shost;
...@@ -2048,6 +2134,7 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost, ...@@ -2048,6 +2134,7 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost,
if (iscsi_add_session(cls_session, id)) if (iscsi_add_session(cls_session, id))
goto cls_session_fail; goto cls_session_fail;
return cls_session; return cls_session;
cls_session_fail: cls_session_fail:
...@@ -2056,6 +2143,8 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost, ...@@ -2056,6 +2143,8 @@ iscsi_session_setup(struct iscsi_transport *iscsit, struct Scsi_Host *shost,
iscsi_pool_free(&session->cmdpool); iscsi_pool_free(&session->cmdpool);
cmdpool_alloc_fail: cmdpool_alloc_fail:
iscsi_free_session(cls_session); iscsi_free_session(cls_session);
dec_session_count:
iscsi_host_dec_session_cnt(shost);
return NULL; return NULL;
} }
EXPORT_SYMBOL_GPL(iscsi_session_setup); EXPORT_SYMBOL_GPL(iscsi_session_setup);
...@@ -2071,6 +2160,7 @@ void iscsi_session_teardown(struct iscsi_cls_session *cls_session) ...@@ -2071,6 +2160,7 @@ void iscsi_session_teardown(struct iscsi_cls_session *cls_session)
{ {
struct iscsi_session *session = cls_session->dd_data; struct iscsi_session *session = cls_session->dd_data;
struct module *owner = cls_session->transport->owner; struct module *owner = cls_session->transport->owner;
struct Scsi_Host *shost = session->host;
iscsi_pool_free(&session->cmdpool); iscsi_pool_free(&session->cmdpool);
...@@ -2083,6 +2173,7 @@ void iscsi_session_teardown(struct iscsi_cls_session *cls_session) ...@@ -2083,6 +2173,7 @@ void iscsi_session_teardown(struct iscsi_cls_session *cls_session)
kfree(session->ifacename); kfree(session->ifacename);
iscsi_destroy_session(cls_session); iscsi_destroy_session(cls_session);
iscsi_host_dec_session_cnt(shost);
module_put(owner); module_put(owner);
} }
EXPORT_SYMBOL_GPL(iscsi_session_teardown); EXPORT_SYMBOL_GPL(iscsi_session_teardown);
......
...@@ -353,7 +353,7 @@ void qla4xxx_mark_device_missing(struct scsi_qla_host *ha, ...@@ -353,7 +353,7 @@ void qla4xxx_mark_device_missing(struct scsi_qla_host *ha,
ha->host_no, ddb_entry->bus, ddb_entry->target, ha->host_no, ddb_entry->bus, ddb_entry->target,
ddb_entry->fw_ddb_index)); ddb_entry->fw_ddb_index));
iscsi_block_session(ddb_entry->sess); iscsi_block_session(ddb_entry->sess);
iscsi_conn_error(ddb_entry->conn, ISCSI_ERR_CONN_FAILED); iscsi_conn_error_event(ddb_entry->conn, ISCSI_ERR_CONN_FAILED);
} }
static struct srb* qla4xxx_get_new_srb(struct scsi_qla_host *ha, static struct srb* qla4xxx_get_new_srb(struct scsi_qla_host *ha,
......
...@@ -1010,7 +1010,7 @@ int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr, ...@@ -1010,7 +1010,7 @@ int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr,
skb = alloc_skb(len, GFP_ATOMIC); skb = alloc_skb(len, GFP_ATOMIC);
if (!skb) { if (!skb) {
iscsi_conn_error(conn, ISCSI_ERR_CONN_FAILED); iscsi_conn_error_event(conn, ISCSI_ERR_CONN_FAILED);
iscsi_cls_conn_printk(KERN_ERR, conn, "can not deliver " iscsi_cls_conn_printk(KERN_ERR, conn, "can not deliver "
"control PDU: OOM\n"); "control PDU: OOM\n");
return -ENOMEM; return -ENOMEM;
...@@ -1031,7 +1031,7 @@ int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr, ...@@ -1031,7 +1031,7 @@ int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr,
} }
EXPORT_SYMBOL_GPL(iscsi_recv_pdu); EXPORT_SYMBOL_GPL(iscsi_recv_pdu);
void iscsi_conn_error(struct iscsi_cls_conn *conn, enum iscsi_err error) void iscsi_conn_error_event(struct iscsi_cls_conn *conn, enum iscsi_err error)
{ {
struct nlmsghdr *nlh; struct nlmsghdr *nlh;
struct sk_buff *skb; struct sk_buff *skb;
...@@ -1063,7 +1063,7 @@ void iscsi_conn_error(struct iscsi_cls_conn *conn, enum iscsi_err error) ...@@ -1063,7 +1063,7 @@ void iscsi_conn_error(struct iscsi_cls_conn *conn, enum iscsi_err error)
iscsi_cls_conn_printk(KERN_INFO, conn, "detected conn error (%d)\n", iscsi_cls_conn_printk(KERN_INFO, conn, "detected conn error (%d)\n",
error); error);
} }
EXPORT_SYMBOL_GPL(iscsi_conn_error); EXPORT_SYMBOL_GPL(iscsi_conn_error_event);
static int static int
iscsi_if_send_reply(int pid, int seq, int type, int done, int multi, iscsi_if_send_reply(int pid, int seq, int type, int done, int multi,
......
...@@ -213,6 +213,7 @@ enum iscsi_err { ...@@ -213,6 +213,7 @@ enum iscsi_err {
ISCSI_ERR_DATA_DGST = ISCSI_ERR_BASE + 15, ISCSI_ERR_DATA_DGST = ISCSI_ERR_BASE + 15,
ISCSI_ERR_PARAM_NOT_FOUND = ISCSI_ERR_BASE + 16, ISCSI_ERR_PARAM_NOT_FOUND = ISCSI_ERR_BASE + 16,
ISCSI_ERR_NO_SCSI_CMD = ISCSI_ERR_BASE + 17, ISCSI_ERR_NO_SCSI_CMD = ISCSI_ERR_BASE + 17,
ISCSI_ERR_INVALID_HOST = ISCSI_ERR_BASE + 18,
}; };
/* /*
......
...@@ -287,6 +287,11 @@ struct iscsi_session { ...@@ -287,6 +287,11 @@ struct iscsi_session {
struct iscsi_pool cmdpool; /* PDU's pool */ struct iscsi_pool cmdpool; /* PDU's pool */
}; };
enum {
ISCSI_HOST_SETUP,
ISCSI_HOST_REMOVED,
};
struct iscsi_host { struct iscsi_host {
char *initiatorname; char *initiatorname;
/* hw address or netdev iscsi connection is bound to */ /* hw address or netdev iscsi connection is bound to */
...@@ -295,6 +300,12 @@ struct iscsi_host { ...@@ -295,6 +300,12 @@ struct iscsi_host {
/* local address */ /* local address */
int local_port; int local_port;
char local_address[ISCSI_ADDRESS_BUF_LEN]; char local_address[ISCSI_ADDRESS_BUF_LEN];
wait_queue_head_t session_removal_wq;
/* protects sessions and state */
spinlock_t lock;
int num_sessions;
int state;
}; };
/* /*
...@@ -351,6 +362,8 @@ extern void iscsi_conn_stop(struct iscsi_cls_conn *, int); ...@@ -351,6 +362,8 @@ extern void iscsi_conn_stop(struct iscsi_cls_conn *, int);
extern int iscsi_conn_bind(struct iscsi_cls_session *, struct iscsi_cls_conn *, extern int iscsi_conn_bind(struct iscsi_cls_session *, struct iscsi_cls_conn *,
int); int);
extern void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err); extern void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err);
extern void iscsi_session_failure(struct iscsi_cls_session *cls_session,
enum iscsi_err err);
extern int iscsi_conn_get_param(struct iscsi_cls_conn *cls_conn, extern int iscsi_conn_get_param(struct iscsi_cls_conn *cls_conn,
enum iscsi_param param, char *buf); enum iscsi_param param, char *buf);
extern void iscsi_suspend_tx(struct iscsi_conn *conn); extern void iscsi_suspend_tx(struct iscsi_conn *conn);
......
...@@ -135,7 +135,8 @@ extern int iscsi_unregister_transport(struct iscsi_transport *tt); ...@@ -135,7 +135,8 @@ extern int iscsi_unregister_transport(struct iscsi_transport *tt);
/* /*
* control plane upcalls * control plane upcalls
*/ */
extern void iscsi_conn_error(struct iscsi_cls_conn *conn, enum iscsi_err error); extern void iscsi_conn_error_event(struct iscsi_cls_conn *conn,
enum iscsi_err error);
extern int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr, extern int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr,
char *data, uint32_t data_size); char *data, uint32_t data_size);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册