diff options
author | Sarang Radke <sarang.radke@qlogic.com> | 2010-05-28 18:08:21 -0400 |
---|---|---|
committer | James Bottomley <James.Bottomley@suse.de> | 2010-07-27 13:01:22 -0400 |
commit | 23f2ebd17a13835c5b34994d2c2e5faacc127947 (patch) | |
tree | 2fce122a5ecceffa8c52e334d9dfeee0e33e3e7f /drivers/scsi/qla2xxx/qla_bsg.c | |
parent | 3a6478df74c271cb3be5895b39fddf75e9cef89c (diff) |
[SCSI] qla2xxx: Add internal loopback support for ISP81xx.
Signed-off-by: Giridhar Malavali <giridhar.malavali@qlogic.com>
Signed-off-by: James Bottomley <James.Bottomley@suse.de>
Diffstat (limited to 'drivers/scsi/qla2xxx/qla_bsg.c')
-rw-r--r-- | drivers/scsi/qla2xxx/qla_bsg.c | 196 |
1 files changed, 181 insertions, 15 deletions
diff --git a/drivers/scsi/qla2xxx/qla_bsg.c b/drivers/scsi/qla2xxx/qla_bsg.c index b905dfe5ea61..3a0248388505 100644 --- a/drivers/scsi/qla2xxx/qla_bsg.c +++ b/drivers/scsi/qla2xxx/qla_bsg.c | |||
@@ -483,6 +483,98 @@ done: | |||
483 | return rval; | 483 | return rval; |
484 | } | 484 | } |
485 | 485 | ||
486 | /* Set the port configuration to enable the | ||
487 | * internal loopback on ISP81XX | ||
488 | */ | ||
489 | static inline int | ||
490 | qla81xx_set_internal_loopback(scsi_qla_host_t *vha, uint16_t *config, | ||
491 | uint16_t *new_config) | ||
492 | { | ||
493 | int ret = 0; | ||
494 | int rval = 0; | ||
495 | struct qla_hw_data *ha = vha->hw; | ||
496 | |||
497 | if (!IS_QLA81XX(ha)) | ||
498 | goto done_set_internal; | ||
499 | |||
500 | new_config[0] = config[0] | (ENABLE_INTERNAL_LOOPBACK << 1); | ||
501 | memcpy(&new_config[1], &config[1], sizeof(uint16_t) * 3) ; | ||
502 | |||
503 | ha->notify_dcbx_comp = 1; | ||
504 | ret = qla81xx_set_port_config(vha, new_config); | ||
505 | if (ret != QLA_SUCCESS) { | ||
506 | DEBUG2(printk(KERN_ERR | ||
507 | "%s(%lu): Set port config failed\n", | ||
508 | __func__, vha->host_no)); | ||
509 | ha->notify_dcbx_comp = 0; | ||
510 | rval = -EINVAL; | ||
511 | goto done_set_internal; | ||
512 | } | ||
513 | |||
514 | /* Wait for DCBX complete event */ | ||
515 | if (!wait_for_completion_timeout(&ha->dcbx_comp, (20 * HZ))) { | ||
516 | DEBUG2(qla_printk(KERN_WARNING, ha, | ||
517 | "State change notificaition not received.\n")); | ||
518 | } else | ||
519 | DEBUG2(qla_printk(KERN_INFO, ha, | ||
520 | "State change RECEIVED\n")); | ||
521 | |||
522 | ha->notify_dcbx_comp = 0; | ||
523 | |||
524 | done_set_internal: | ||
525 | return rval; | ||
526 | } | ||
527 | |||
528 | /* Set the port configuration to disable the | ||
529 | * internal loopback on ISP81XX | ||
530 | */ | ||
531 | static inline int | ||
532 | qla81xx_reset_internal_loopback(scsi_qla_host_t *vha, uint16_t *config, | ||
533 | int wait) | ||
534 | { | ||
535 | int ret = 0; | ||
536 | int rval = 0; | ||
537 | uint16_t new_config[4]; | ||
538 | struct qla_hw_data *ha = vha->hw; | ||
539 | |||
540 | if (!IS_QLA81XX(ha)) | ||
541 | goto done_reset_internal; | ||
542 | |||
543 | memset(new_config, 0 , sizeof(new_config)); | ||
544 | if ((config[0] & INTERNAL_LOOPBACK_MASK) >> 1 == | ||
545 | ENABLE_INTERNAL_LOOPBACK) { | ||
546 | new_config[0] = config[0] & ~INTERNAL_LOOPBACK_MASK; | ||
547 | memcpy(&new_config[1], &config[1], sizeof(uint16_t) * 3) ; | ||
548 | |||
549 | ha->notify_dcbx_comp = wait; | ||
550 | ret = qla81xx_set_port_config(vha, new_config); | ||
551 | if (ret != QLA_SUCCESS) { | ||
552 | DEBUG2(printk(KERN_ERR | ||
553 | "%s(%lu): Set port config failed\n", | ||
554 | __func__, vha->host_no)); | ||
555 | ha->notify_dcbx_comp = 0; | ||
556 | rval = -EINVAL; | ||
557 | goto done_reset_internal; | ||
558 | } | ||
559 | |||
560 | /* Wait for DCBX complete event */ | ||
561 | if (wait && !wait_for_completion_timeout(&ha->dcbx_comp, | ||
562 | (20 * HZ))) { | ||
563 | DEBUG2(qla_printk(KERN_WARNING, ha, | ||
564 | "State change notificaition not received.\n")); | ||
565 | ha->notify_dcbx_comp = 0; | ||
566 | rval = -EINVAL; | ||
567 | goto done_reset_internal; | ||
568 | } else | ||
569 | DEBUG2(qla_printk(KERN_INFO, ha, | ||
570 | "State change RECEIVED\n")); | ||
571 | |||
572 | ha->notify_dcbx_comp = 0; | ||
573 | } | ||
574 | done_reset_internal: | ||
575 | return rval; | ||
576 | } | ||
577 | |||
486 | static int | 578 | static int |
487 | qla2x00_process_loopback(struct fc_bsg_job *bsg_job) | 579 | qla2x00_process_loopback(struct fc_bsg_job *bsg_job) |
488 | { | 580 | { |
@@ -494,6 +586,7 @@ qla2x00_process_loopback(struct fc_bsg_job *bsg_job) | |||
494 | char *type; | 586 | char *type; |
495 | struct msg_echo_lb elreq; | 587 | struct msg_echo_lb elreq; |
496 | uint16_t response[MAILBOX_REGISTER_COUNT]; | 588 | uint16_t response[MAILBOX_REGISTER_COUNT]; |
589 | uint16_t config[4], new_config[4]; | ||
497 | uint8_t *fw_sts_ptr; | 590 | uint8_t *fw_sts_ptr; |
498 | uint8_t *req_data = NULL; | 591 | uint8_t *req_data = NULL; |
499 | dma_addr_t req_data_dma; | 592 | dma_addr_t req_data_dma; |
@@ -568,29 +661,102 @@ qla2x00_process_loopback(struct fc_bsg_job *bsg_job) | |||
568 | 661 | ||
569 | elreq.options = bsg_job->request->rqst_data.h_vendor.vendor_cmd[1]; | 662 | elreq.options = bsg_job->request->rqst_data.h_vendor.vendor_cmd[1]; |
570 | 663 | ||
571 | if (ha->current_topology != ISP_CFG_F) { | 664 | if ((ha->current_topology == ISP_CFG_F || |
572 | type = "FC_BSG_HST_VENDOR_LOOPBACK"; | 665 | (IS_QLA81XX(ha) && |
666 | le32_to_cpu(*(uint32_t *)req_data) == ELS_OPCODE_BYTE | ||
667 | && req_data_len == MAX_ELS_FRAME_PAYLOAD)) && | ||
668 | elreq.options == EXTERNAL_LOOPBACK) { | ||
669 | type = "FC_BSG_HST_VENDOR_ECHO_DIAG"; | ||
573 | DEBUG2(qla_printk(KERN_INFO, ha, | 670 | DEBUG2(qla_printk(KERN_INFO, ha, |
574 | "scsi(%ld) bsg rqst type: %s\n", | 671 | "scsi(%ld) bsg rqst type: %s\n", vha->host_no, type)); |
575 | vha->host_no, type)); | 672 | command_sent = INT_DEF_LB_ECHO_CMD; |
576 | 673 | rval = qla2x00_echo_test(vha, &elreq, response); | |
577 | command_sent = INT_DEF_LB_LOOPBACK_CMD; | 674 | } else { |
578 | rval = qla2x00_loopback_test(vha, &elreq, response); | ||
579 | if (IS_QLA81XX(ha)) { | 675 | if (IS_QLA81XX(ha)) { |
676 | memset(config, 0, sizeof(config)); | ||
677 | memset(new_config, 0, sizeof(new_config)); | ||
678 | if (qla81xx_get_port_config(vha, config)) { | ||
679 | DEBUG2(printk(KERN_ERR | ||
680 | "%s(%lu): Get port config failed\n", | ||
681 | __func__, vha->host_no)); | ||
682 | bsg_job->reply->reply_payload_rcv_len = 0; | ||
683 | bsg_job->reply->result = (DID_ERROR << 16); | ||
684 | rval = -EPERM; | ||
685 | goto done_free_dma_req; | ||
686 | } | ||
687 | |||
688 | if (elreq.options != EXTERNAL_LOOPBACK) { | ||
689 | DEBUG2(qla_printk(KERN_INFO, ha, | ||
690 | "Internal: current port config = %x\n", | ||
691 | config[0])); | ||
692 | if (qla81xx_set_internal_loopback(vha, config, | ||
693 | new_config)) { | ||
694 | bsg_job->reply->reply_payload_rcv_len = | ||
695 | 0; | ||
696 | bsg_job->reply->result = | ||
697 | (DID_ERROR << 16); | ||
698 | rval = -EPERM; | ||
699 | goto done_free_dma_req; | ||
700 | } | ||
701 | } else { | ||
702 | /* For external loopback to work | ||
703 | * ensure internal loopback is disabled | ||
704 | */ | ||
705 | if (qla81xx_reset_internal_loopback(vha, | ||
706 | config, 1)) { | ||
707 | bsg_job->reply->reply_payload_rcv_len = | ||
708 | 0; | ||
709 | bsg_job->reply->result = | ||
710 | (DID_ERROR << 16); | ||
711 | rval = -EPERM; | ||
712 | goto done_free_dma_req; | ||
713 | } | ||
714 | } | ||
715 | |||
716 | type = "FC_BSG_HST_VENDOR_LOOPBACK"; | ||
717 | DEBUG2(qla_printk(KERN_INFO, ha, | ||
718 | "scsi(%ld) bsg rqst type: %s\n", | ||
719 | vha->host_no, type)); | ||
720 | |||
721 | command_sent = INT_DEF_LB_LOOPBACK_CMD; | ||
722 | rval = qla2x00_loopback_test(vha, &elreq, response); | ||
723 | |||
724 | if (new_config[1]) { | ||
725 | /* Revert back to original port config | ||
726 | * Also clear internal loopback | ||
727 | */ | ||
728 | qla81xx_reset_internal_loopback(vha, | ||
729 | new_config, 0); | ||
730 | } | ||
731 | |||
580 | if (response[0] == MBS_COMMAND_ERROR && | 732 | if (response[0] == MBS_COMMAND_ERROR && |
581 | response[1] == MBS_LB_RESET) { | 733 | response[1] == MBS_LB_RESET) { |
582 | DEBUG2(printk(KERN_ERR "%s(%ld): ABORTing " | 734 | DEBUG2(printk(KERN_ERR "%s(%ld): ABORTing " |
583 | "ISP\n", __func__, vha->host_no)); | 735 | "ISP\n", __func__, vha->host_no)); |
584 | set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags); | 736 | set_bit(ISP_ABORT_NEEDED, &vha->dpc_flags); |
585 | qla2xxx_wake_dpc(vha); | 737 | qla2xxx_wake_dpc(vha); |
738 | qla2x00_wait_for_chip_reset(vha); | ||
739 | /* Also reset the MPI */ | ||
740 | if (qla81xx_restart_mpi_firmware(vha) != | ||
741 | QLA_SUCCESS) { | ||
742 | qla_printk(KERN_INFO, ha, | ||
743 | "MPI reset failed for host%ld.\n", | ||
744 | vha->host_no); | ||
745 | } | ||
746 | |||
747 | bsg_job->reply->reply_payload_rcv_len = 0; | ||
748 | bsg_job->reply->result = (DID_ERROR << 16); | ||
749 | rval = -EIO; | ||
750 | goto done_free_dma_req; | ||
586 | } | 751 | } |
752 | } else { | ||
753 | type = "FC_BSG_HST_VENDOR_LOOPBACK"; | ||
754 | DEBUG2(qla_printk(KERN_INFO, ha, | ||
755 | "scsi(%ld) bsg rqst type: %s\n", | ||
756 | vha->host_no, type)); | ||
757 | command_sent = INT_DEF_LB_LOOPBACK_CMD; | ||
758 | rval = qla2x00_loopback_test(vha, &elreq, response); | ||
587 | } | 759 | } |
588 | } else { | ||
589 | type = "FC_BSG_HST_VENDOR_ECHO_DIAG"; | ||
590 | DEBUG2(qla_printk(KERN_INFO, ha, | ||
591 | "scsi(%ld) bsg rqst type: %s\n", vha->host_no, type)); | ||
592 | command_sent = INT_DEF_LB_ECHO_CMD; | ||
593 | rval = qla2x00_echo_test(vha, &elreq, response); | ||
594 | } | 760 | } |
595 | 761 | ||
596 | if (rval) { | 762 | if (rval) { |