diff options
author | Mike Christie <michaelc@cs.wisc.edu> | 2006-05-02 20:46:47 -0400 |
---|---|---|
committer | James Bottomley <jejb@mulgrave.il.steeleye.com> | 2006-05-10 11:13:21 -0400 |
commit | 8d2860b3c3e933304f49171770658c00ed26fd79 (patch) | |
tree | 979395449d53d41bd22732a4be1387f9af73a994 /drivers/scsi/libiscsi.c | |
parent | be2df72e7ec5fa5e6e1ccccab6cef97ecbb9c191 (diff) |
[SCSI] iscsi: increment expstatsn during login
debugged by Ming and Rohan:
The problem Ming and Rohan debugged was that during a normal session
login, open-iscsi is not incrementing the exp_statsn counter. It was
stuck at zero. From the RFC, it looks like if the login response PDU has
a successful status then we should be incrementing that value. Also from
the RFC, it looks like if when we drop a connection then reconnect, we
should be using the exp_statsn from the old connection in the next
relogin attempt.
Signed-off-by: Mike Christie <michaelc@cs.wisc.edu>
Signed-off-by: James Bottomley <James.Bottomley@SteelEye.com>
Diffstat (limited to 'drivers/scsi/libiscsi.c')
-rw-r--r-- | drivers/scsi/libiscsi.c | 38 |
1 files changed, 19 insertions, 19 deletions
diff --git a/drivers/scsi/libiscsi.c b/drivers/scsi/libiscsi.c index a99f2ef44e87..4750d4888100 100644 --- a/drivers/scsi/libiscsi.c +++ b/drivers/scsi/libiscsi.c | |||
@@ -333,15 +333,21 @@ int __iscsi_complete_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
333 | debug_scsi("immrsp [op 0x%x cid %d itt 0x%x len %d]\n", | 333 | debug_scsi("immrsp [op 0x%x cid %d itt 0x%x len %d]\n", |
334 | opcode, conn->id, mtask->itt, datalen); | 334 | opcode, conn->id, mtask->itt, datalen); |
335 | 335 | ||
336 | rc = iscsi_check_assign_cmdsn(session, | ||
337 | (struct iscsi_nopin*)hdr); | ||
338 | if (rc) | ||
339 | goto done; | ||
340 | |||
336 | switch(opcode) { | 341 | switch(opcode) { |
342 | case ISCSI_OP_LOGOUT_RSP: | ||
343 | conn->exp_statsn = be32_to_cpu(hdr->statsn) + 1; | ||
344 | /* fall through */ | ||
337 | case ISCSI_OP_LOGIN_RSP: | 345 | case ISCSI_OP_LOGIN_RSP: |
338 | case ISCSI_OP_TEXT_RSP: | 346 | case ISCSI_OP_TEXT_RSP: |
339 | case ISCSI_OP_LOGOUT_RSP: | 347 | /* |
340 | rc = iscsi_check_assign_cmdsn(session, | 348 | * login related PDU's exp_statsn is handled in |
341 | (struct iscsi_nopin*)hdr); | 349 | * userspace |
342 | if (rc) | 350 | */ |
343 | break; | ||
344 | |||
345 | rc = iscsi_recv_pdu(conn->cls_conn, hdr, data, datalen); | 351 | rc = iscsi_recv_pdu(conn->cls_conn, hdr, data, datalen); |
346 | list_del(&mtask->running); | 352 | list_del(&mtask->running); |
347 | if (conn->login_mtask != mtask) | 353 | if (conn->login_mtask != mtask) |
@@ -349,15 +355,12 @@ int __iscsi_complete_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
349 | (void*)&mtask, sizeof(void*)); | 355 | (void*)&mtask, sizeof(void*)); |
350 | break; | 356 | break; |
351 | case ISCSI_OP_SCSI_TMFUNC_RSP: | 357 | case ISCSI_OP_SCSI_TMFUNC_RSP: |
352 | rc = iscsi_check_assign_cmdsn(session, | ||
353 | (struct iscsi_nopin*)hdr); | ||
354 | if (rc) | ||
355 | break; | ||
356 | |||
357 | if (datalen) { | 358 | if (datalen) { |
358 | rc = ISCSI_ERR_PROTO; | 359 | rc = ISCSI_ERR_PROTO; |
359 | break; | 360 | break; |
360 | } | 361 | } |
362 | |||
363 | conn->exp_statsn = be32_to_cpu(hdr->statsn) + 1; | ||
361 | conn->tmfrsp_pdus_cnt++; | 364 | conn->tmfrsp_pdus_cnt++; |
362 | if (conn->tmabort_state == TMABORT_INITIAL) { | 365 | if (conn->tmabort_state == TMABORT_INITIAL) { |
363 | conn->tmabort_state = | 366 | conn->tmabort_state = |
@@ -373,10 +376,6 @@ int __iscsi_complete_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
373 | rc = ISCSI_ERR_PROTO; | 376 | rc = ISCSI_ERR_PROTO; |
374 | break; | 377 | break; |
375 | } | 378 | } |
376 | rc = iscsi_check_assign_cmdsn(session, | ||
377 | (struct iscsi_nopin*)hdr); | ||
378 | if (rc) | ||
379 | break; | ||
380 | conn->exp_statsn = be32_to_cpu(hdr->statsn) + 1; | 379 | conn->exp_statsn = be32_to_cpu(hdr->statsn) + 1; |
381 | 380 | ||
382 | rc = iscsi_recv_pdu(conn->cls_conn, hdr, data, datalen); | 381 | rc = iscsi_recv_pdu(conn->cls_conn, hdr, data, datalen); |
@@ -404,6 +403,7 @@ int __iscsi_complete_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
404 | case ISCSI_OP_REJECT: | 403 | case ISCSI_OP_REJECT: |
405 | /* we need sth like iscsi_reject_rsp()*/ | 404 | /* we need sth like iscsi_reject_rsp()*/ |
406 | case ISCSI_OP_ASYNC_EVENT: | 405 | case ISCSI_OP_ASYNC_EVENT: |
406 | conn->exp_statsn = be32_to_cpu(hdr->statsn) + 1; | ||
407 | /* we need sth like iscsi_async_event_rsp() */ | 407 | /* we need sth like iscsi_async_event_rsp() */ |
408 | rc = ISCSI_ERR_BAD_OPCODE; | 408 | rc = ISCSI_ERR_BAD_OPCODE; |
409 | break; | 409 | break; |
@@ -414,6 +414,7 @@ int __iscsi_complete_pdu(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
414 | } else | 414 | } else |
415 | rc = ISCSI_ERR_BAD_ITT; | 415 | rc = ISCSI_ERR_BAD_ITT; |
416 | 416 | ||
417 | done: | ||
417 | return rc; | 418 | return rc; |
418 | } | 419 | } |
419 | EXPORT_SYMBOL_GPL(__iscsi_complete_pdu); | 420 | EXPORT_SYMBOL_GPL(__iscsi_complete_pdu); |
@@ -730,6 +731,7 @@ iscsi_conn_send_generic(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
730 | BUG_ON(conn->c_stage == ISCSI_CONN_INITIAL_STAGE); | 731 | BUG_ON(conn->c_stage == ISCSI_CONN_INITIAL_STAGE); |
731 | BUG_ON(conn->c_stage == ISCSI_CONN_STOPPED); | 732 | BUG_ON(conn->c_stage == ISCSI_CONN_STOPPED); |
732 | 733 | ||
734 | nop->exp_statsn = cpu_to_be32(conn->exp_statsn); | ||
733 | if (!__kfifo_get(session->mgmtpool.queue, | 735 | if (!__kfifo_get(session->mgmtpool.queue, |
734 | (void*)&mtask, sizeof(void*))) { | 736 | (void*)&mtask, sizeof(void*))) { |
735 | spin_unlock_bh(&session->lock); | 737 | spin_unlock_bh(&session->lock); |
@@ -738,7 +740,7 @@ iscsi_conn_send_generic(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
738 | } | 740 | } |
739 | 741 | ||
740 | /* | 742 | /* |
741 | * pre-format CmdSN and ExpStatSN for outgoing PDU. | 743 | * pre-format CmdSN for outgoing PDU. |
742 | */ | 744 | */ |
743 | if (hdr->itt != cpu_to_be32(ISCSI_RESERVED_TAG)) { | 745 | if (hdr->itt != cpu_to_be32(ISCSI_RESERVED_TAG)) { |
744 | hdr->itt = mtask->itt | (conn->id << ISCSI_CID_SHIFT) | | 746 | hdr->itt = mtask->itt | (conn->id << ISCSI_CID_SHIFT) | |
@@ -751,8 +753,6 @@ iscsi_conn_send_generic(struct iscsi_conn *conn, struct iscsi_hdr *hdr, | |||
751 | /* do not advance CmdSN */ | 753 | /* do not advance CmdSN */ |
752 | nop->cmdsn = cpu_to_be32(session->cmdsn); | 754 | nop->cmdsn = cpu_to_be32(session->cmdsn); |
753 | 755 | ||
754 | nop->exp_statsn = cpu_to_be32(conn->exp_statsn); | ||
755 | |||
756 | if (data_size) { | 756 | if (data_size) { |
757 | memcpy(mtask->data, data, data_size); | 757 | memcpy(mtask->data, data, data_size); |
758 | mtask->data_count = data_size; | 758 | mtask->data_count = data_size; |
@@ -1647,7 +1647,7 @@ void iscsi_conn_stop(struct iscsi_cls_conn *cls_conn, int flag) | |||
1647 | case STOP_CONN_RECOVER: | 1647 | case STOP_CONN_RECOVER: |
1648 | case STOP_CONN_TERM: | 1648 | case STOP_CONN_TERM: |
1649 | iscsi_start_session_recovery(session, conn, flag); | 1649 | iscsi_start_session_recovery(session, conn, flag); |
1650 | return; | 1650 | break; |
1651 | case STOP_CONN_SUSPEND: | 1651 | case STOP_CONN_SUSPEND: |
1652 | if (session->tt->suspend_conn_recv) | 1652 | if (session->tt->suspend_conn_recv) |
1653 | session->tt->suspend_conn_recv(conn); | 1653 | session->tt->suspend_conn_recv(conn); |