diff options
-rw-r--r-- | drivers/scsi/megaraid/megaraid_sas.c | 80 | ||||
-rw-r--r-- | drivers/scsi/megaraid/megaraid_sas.h | 9 |
2 files changed, 82 insertions, 7 deletions
diff --git a/drivers/scsi/megaraid/megaraid_sas.c b/drivers/scsi/megaraid/megaraid_sas.c index 5afd65121725..4c04a68bad6c 100644 --- a/drivers/scsi/megaraid/megaraid_sas.c +++ b/drivers/scsi/megaraid/megaraid_sas.c | |||
@@ -696,6 +696,35 @@ megasas_make_sgl64(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
696 | return sge_count; | 696 | return sge_count; |
697 | } | 697 | } |
698 | 698 | ||
699 | /** | ||
700 | * megasas_make_sgl_skinny - Prepares IEEE SGL | ||
701 | * @instance: Adapter soft state | ||
702 | * @scp: SCSI command from the mid-layer | ||
703 | * @mfi_sgl: SGL to be filled in | ||
704 | * | ||
705 | * If successful, this function returns the number of SG elements. Otherwise, | ||
706 | * it returnes -1. | ||
707 | */ | ||
708 | static int | ||
709 | megasas_make_sgl_skinny(struct megasas_instance *instance, | ||
710 | struct scsi_cmnd *scp, union megasas_sgl *mfi_sgl) | ||
711 | { | ||
712 | int i; | ||
713 | int sge_count; | ||
714 | struct scatterlist *os_sgl; | ||
715 | |||
716 | sge_count = scsi_dma_map(scp); | ||
717 | |||
718 | if (sge_count) { | ||
719 | scsi_for_each_sg(scp, os_sgl, sge_count, i) { | ||
720 | mfi_sgl->sge_skinny[i].length = sg_dma_len(os_sgl); | ||
721 | mfi_sgl->sge_skinny[i].phys_addr = | ||
722 | sg_dma_address(os_sgl); | ||
723 | } | ||
724 | } | ||
725 | return sge_count; | ||
726 | } | ||
727 | |||
699 | /** | 728 | /** |
700 | * megasas_get_frame_count - Computes the number of frames | 729 | * megasas_get_frame_count - Computes the number of frames |
701 | * @frame_type : type of frame- io or pthru frame | 730 | * @frame_type : type of frame- io or pthru frame |
@@ -704,7 +733,8 @@ megasas_make_sgl64(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
704 | * Returns the number of frames required for numnber of sge's (sge_count) | 733 | * Returns the number of frames required for numnber of sge's (sge_count) |
705 | */ | 734 | */ |
706 | 735 | ||
707 | static u32 megasas_get_frame_count(u8 sge_count, u8 frame_type) | 736 | static u32 megasas_get_frame_count(struct megasas_instance *instance, |
737 | u8 sge_count, u8 frame_type) | ||
708 | { | 738 | { |
709 | int num_cnt; | 739 | int num_cnt; |
710 | int sge_bytes; | 740 | int sge_bytes; |
@@ -714,6 +744,10 @@ static u32 megasas_get_frame_count(u8 sge_count, u8 frame_type) | |||
714 | sge_sz = (IS_DMA64) ? sizeof(struct megasas_sge64) : | 744 | sge_sz = (IS_DMA64) ? sizeof(struct megasas_sge64) : |
715 | sizeof(struct megasas_sge32); | 745 | sizeof(struct megasas_sge32); |
716 | 746 | ||
747 | if (instance->flag_ieee) { | ||
748 | sge_sz = sizeof(struct megasas_sge_skinny); | ||
749 | } | ||
750 | |||
717 | /* | 751 | /* |
718 | * Main frame can contain 2 SGEs for 64-bit SGLs and | 752 | * Main frame can contain 2 SGEs for 64-bit SGLs and |
719 | * 3 SGEs for 32-bit SGLs for ldio & | 753 | * 3 SGEs for 32-bit SGLs for ldio & |
@@ -721,12 +755,16 @@ static u32 megasas_get_frame_count(u8 sge_count, u8 frame_type) | |||
721 | * 2 SGEs for 32-bit SGLs for pthru frame | 755 | * 2 SGEs for 32-bit SGLs for pthru frame |
722 | */ | 756 | */ |
723 | if (unlikely(frame_type == PTHRU_FRAME)) { | 757 | if (unlikely(frame_type == PTHRU_FRAME)) { |
724 | if (IS_DMA64) | 758 | if (instance->flag_ieee == 1) { |
759 | num_cnt = sge_count - 1; | ||
760 | } else if (IS_DMA64) | ||
725 | num_cnt = sge_count - 1; | 761 | num_cnt = sge_count - 1; |
726 | else | 762 | else |
727 | num_cnt = sge_count - 2; | 763 | num_cnt = sge_count - 2; |
728 | } else { | 764 | } else { |
729 | if (IS_DMA64) | 765 | if (instance->flag_ieee == 1) { |
766 | num_cnt = sge_count - 1; | ||
767 | } else if (IS_DMA64) | ||
730 | num_cnt = sge_count - 2; | 768 | num_cnt = sge_count - 2; |
731 | else | 769 | else |
732 | num_cnt = sge_count - 3; | 770 | num_cnt = sge_count - 3; |
@@ -775,6 +813,10 @@ megasas_build_dcdb(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
775 | else if (scp->sc_data_direction == PCI_DMA_NONE) | 813 | else if (scp->sc_data_direction == PCI_DMA_NONE) |
776 | flags = MFI_FRAME_DIR_NONE; | 814 | flags = MFI_FRAME_DIR_NONE; |
777 | 815 | ||
816 | if (instance->flag_ieee == 1) { | ||
817 | flags |= MFI_FRAME_IEEE; | ||
818 | } | ||
819 | |||
778 | /* | 820 | /* |
779 | * Prepare the DCDB frame | 821 | * Prepare the DCDB frame |
780 | */ | 822 | */ |
@@ -804,7 +846,11 @@ megasas_build_dcdb(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
804 | /* | 846 | /* |
805 | * Construct SGL | 847 | * Construct SGL |
806 | */ | 848 | */ |
807 | if (IS_DMA64) { | 849 | if (instance->flag_ieee == 1) { |
850 | pthru->flags |= MFI_FRAME_SGL64; | ||
851 | pthru->sge_count = megasas_make_sgl_skinny(instance, scp, | ||
852 | &pthru->sgl); | ||
853 | } else if (IS_DMA64) { | ||
808 | pthru->flags |= MFI_FRAME_SGL64; | 854 | pthru->flags |= MFI_FRAME_SGL64; |
809 | pthru->sge_count = megasas_make_sgl64(instance, scp, | 855 | pthru->sge_count = megasas_make_sgl64(instance, scp, |
810 | &pthru->sgl); | 856 | &pthru->sgl); |
@@ -823,7 +869,7 @@ megasas_build_dcdb(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
823 | * Compute the total number of frames this command consumes. FW uses | 869 | * Compute the total number of frames this command consumes. FW uses |
824 | * this number to pull sufficient number of frames from host memory. | 870 | * this number to pull sufficient number of frames from host memory. |
825 | */ | 871 | */ |
826 | cmd->frame_count = megasas_get_frame_count(pthru->sge_count, | 872 | cmd->frame_count = megasas_get_frame_count(instance, pthru->sge_count, |
827 | PTHRU_FRAME); | 873 | PTHRU_FRAME); |
828 | 874 | ||
829 | return cmd->frame_count; | 875 | return cmd->frame_count; |
@@ -854,6 +900,10 @@ megasas_build_ldio(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
854 | else if (scp->sc_data_direction == PCI_DMA_FROMDEVICE) | 900 | else if (scp->sc_data_direction == PCI_DMA_FROMDEVICE) |
855 | flags = MFI_FRAME_DIR_READ; | 901 | flags = MFI_FRAME_DIR_READ; |
856 | 902 | ||
903 | if (instance->flag_ieee == 1) { | ||
904 | flags |= MFI_FRAME_IEEE; | ||
905 | } | ||
906 | |||
857 | /* | 907 | /* |
858 | * Prepare the Logical IO frame: 2nd bit is zero for all read cmds | 908 | * Prepare the Logical IO frame: 2nd bit is zero for all read cmds |
859 | */ | 909 | */ |
@@ -924,7 +974,11 @@ megasas_build_ldio(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
924 | /* | 974 | /* |
925 | * Construct SGL | 975 | * Construct SGL |
926 | */ | 976 | */ |
927 | if (IS_DMA64) { | 977 | if (instance->flag_ieee) { |
978 | ldio->flags |= MFI_FRAME_SGL64; | ||
979 | ldio->sge_count = megasas_make_sgl_skinny(instance, scp, | ||
980 | &ldio->sgl); | ||
981 | } else if (IS_DMA64) { | ||
928 | ldio->flags |= MFI_FRAME_SGL64; | 982 | ldio->flags |= MFI_FRAME_SGL64; |
929 | ldio->sge_count = megasas_make_sgl64(instance, scp, &ldio->sgl); | 983 | ldio->sge_count = megasas_make_sgl64(instance, scp, &ldio->sgl); |
930 | } else | 984 | } else |
@@ -941,7 +995,8 @@ megasas_build_ldio(struct megasas_instance *instance, struct scsi_cmnd *scp, | |||
941 | * Compute the total number of frames this command consumes. FW uses | 995 | * Compute the total number of frames this command consumes. FW uses |
942 | * this number to pull sufficient number of frames from host memory. | 996 | * this number to pull sufficient number of frames from host memory. |
943 | */ | 997 | */ |
944 | cmd->frame_count = megasas_get_frame_count(ldio->sge_count, IO_FRAME); | 998 | cmd->frame_count = megasas_get_frame_count(instance, |
999 | ldio->sge_count, IO_FRAME); | ||
945 | 1000 | ||
946 | return cmd->frame_count; | 1001 | return cmd->frame_count; |
947 | } | 1002 | } |
@@ -1929,6 +1984,10 @@ static int megasas_create_frame_pool(struct megasas_instance *instance) | |||
1929 | sge_sz = (IS_DMA64) ? sizeof(struct megasas_sge64) : | 1984 | sge_sz = (IS_DMA64) ? sizeof(struct megasas_sge64) : |
1930 | sizeof(struct megasas_sge32); | 1985 | sizeof(struct megasas_sge32); |
1931 | 1986 | ||
1987 | if (instance->flag_ieee) { | ||
1988 | sge_sz = sizeof(struct megasas_sge_skinny); | ||
1989 | } | ||
1990 | |||
1932 | /* | 1991 | /* |
1933 | * Calculated the number of 64byte frames required for SGL | 1992 | * Calculated the number of 64byte frames required for SGL |
1934 | */ | 1993 | */ |
@@ -2725,6 +2784,11 @@ megasas_register_aen(struct megasas_instance *instance, u32 seq_num, | |||
2725 | dcmd->sgl.sge32[0].phys_addr = (u32) instance->evt_detail_h; | 2784 | dcmd->sgl.sge32[0].phys_addr = (u32) instance->evt_detail_h; |
2726 | dcmd->sgl.sge32[0].length = sizeof(struct megasas_evt_detail); | 2785 | dcmd->sgl.sge32[0].length = sizeof(struct megasas_evt_detail); |
2727 | 2786 | ||
2787 | if (instance->aen_cmd != NULL) { | ||
2788 | megasas_return_cmd(instance, cmd); | ||
2789 | return 0; | ||
2790 | } | ||
2791 | |||
2728 | /* | 2792 | /* |
2729 | * Store reference to the cmd used to register for AEN. When an | 2793 | * Store reference to the cmd used to register for AEN. When an |
2730 | * application wants us to register for AEN, we have to abort this | 2794 | * application wants us to register for AEN, we have to abort this |
@@ -2895,6 +2959,7 @@ megasas_probe_one(struct pci_dev *pdev, const struct pci_device_id *id) | |||
2895 | *instance->producer = 0; | 2959 | *instance->producer = 0; |
2896 | *instance->consumer = 0; | 2960 | *instance->consumer = 0; |
2897 | megasas_poll_wait_aen = 0; | 2961 | megasas_poll_wait_aen = 0; |
2962 | instance->flag_ieee = 0; | ||
2898 | 2963 | ||
2899 | instance->evt_detail = pci_alloc_consistent(pdev, | 2964 | instance->evt_detail = pci_alloc_consistent(pdev, |
2900 | sizeof(struct | 2965 | sizeof(struct |
@@ -2933,6 +2998,7 @@ megasas_probe_one(struct pci_dev *pdev, const struct pci_device_id *id) | |||
2933 | 2998 | ||
2934 | if ((instance->pdev->device == PCI_DEVICE_ID_LSI_SAS0073SKINNY) || | 2999 | if ((instance->pdev->device == PCI_DEVICE_ID_LSI_SAS0073SKINNY) || |
2935 | (instance->pdev->device == PCI_DEVICE_ID_LSI_SAS0071SKINNY)) { | 3000 | (instance->pdev->device == PCI_DEVICE_ID_LSI_SAS0071SKINNY)) { |
3001 | instance->flag_ieee = 1; | ||
2936 | sema_init(&instance->ioctl_sem, MEGASAS_SKINNY_INT_CMDS); | 3002 | sema_init(&instance->ioctl_sem, MEGASAS_SKINNY_INT_CMDS); |
2937 | } else | 3003 | } else |
2938 | sema_init(&instance->ioctl_sem, MEGASAS_INT_CMDS); | 3004 | sema_init(&instance->ioctl_sem, MEGASAS_INT_CMDS); |
diff --git a/drivers/scsi/megaraid/megaraid_sas.h b/drivers/scsi/megaraid/megaraid_sas.h index 4c78cd32e757..a1fd44bc1817 100644 --- a/drivers/scsi/megaraid/megaraid_sas.h +++ b/drivers/scsi/megaraid/megaraid_sas.h | |||
@@ -96,6 +96,7 @@ | |||
96 | #define MFI_FRAME_DIR_WRITE 0x0008 | 96 | #define MFI_FRAME_DIR_WRITE 0x0008 |
97 | #define MFI_FRAME_DIR_READ 0x0010 | 97 | #define MFI_FRAME_DIR_READ 0x0010 |
98 | #define MFI_FRAME_DIR_BOTH 0x0018 | 98 | #define MFI_FRAME_DIR_BOTH 0x0018 |
99 | #define MFI_FRAME_IEEE 0x0020 | ||
99 | 100 | ||
100 | /* | 101 | /* |
101 | * Definition for cmd_status | 102 | * Definition for cmd_status |
@@ -732,10 +733,17 @@ struct megasas_sge64 { | |||
732 | 733 | ||
733 | } __attribute__ ((packed)); | 734 | } __attribute__ ((packed)); |
734 | 735 | ||
736 | struct megasas_sge_skinny { | ||
737 | u64 phys_addr; | ||
738 | u32 length; | ||
739 | u32 flag; | ||
740 | } __packed; | ||
741 | |||
735 | union megasas_sgl { | 742 | union megasas_sgl { |
736 | 743 | ||
737 | struct megasas_sge32 sge32[1]; | 744 | struct megasas_sge32 sge32[1]; |
738 | struct megasas_sge64 sge64[1]; | 745 | struct megasas_sge64 sge64[1]; |
746 | struct megasas_sge_skinny sge_skinny[1]; | ||
739 | 747 | ||
740 | } __attribute__ ((packed)); | 748 | } __attribute__ ((packed)); |
741 | 749 | ||
@@ -1210,6 +1218,7 @@ struct megasas_instance { | |||
1210 | 1218 | ||
1211 | u8 flag; | 1219 | u8 flag; |
1212 | u8 unload; | 1220 | u8 unload; |
1221 | u8 flag_ieee; | ||
1213 | unsigned long last_time; | 1222 | unsigned long last_time; |
1214 | 1223 | ||
1215 | struct timer_list io_completion_timer; | 1224 | struct timer_list io_completion_timer; |