diff options
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/scsi/qla2xxx/qla_attr.c | 57 | ||||
-rw-r--r-- | drivers/scsi/qla2xxx/qla_def.h | 4 | ||||
-rw-r--r-- | drivers/scsi/qla2xxx/qla_fw.h | 1 | ||||
-rw-r--r-- | drivers/scsi/qla2xxx/qla_gbl.h | 3 | ||||
-rw-r--r-- | drivers/scsi/qla2xxx/qla_mbx.c | 38 | ||||
-rw-r--r-- | drivers/scsi/qla2xxx/qla_os.c | 4 |
6 files changed, 107 insertions, 0 deletions
diff --git a/drivers/scsi/qla2xxx/qla_attr.c b/drivers/scsi/qla2xxx/qla_attr.c index 524ceb9b9288..e8c1c9e01a7b 100644 --- a/drivers/scsi/qla2xxx/qla_attr.c +++ b/drivers/scsi/qla2xxx/qla_attr.c | |||
@@ -692,6 +692,58 @@ static struct bin_attribute sysfs_edc_status_attr = { | |||
692 | .read = qla2x00_sysfs_read_edc_status, | 692 | .read = qla2x00_sysfs_read_edc_status, |
693 | }; | 693 | }; |
694 | 694 | ||
695 | static ssize_t | ||
696 | qla2x00_sysfs_read_xgmac_stats(struct kobject *kobj, | ||
697 | struct bin_attribute *bin_attr, | ||
698 | char *buf, loff_t off, size_t count) | ||
699 | { | ||
700 | struct scsi_qla_host *vha = shost_priv(dev_to_shost(container_of(kobj, | ||
701 | struct device, kobj))); | ||
702 | struct qla_hw_data *ha = vha->hw; | ||
703 | int rval; | ||
704 | uint16_t actual_size; | ||
705 | |||
706 | if (!capable(CAP_SYS_ADMIN) || off != 0 || count > XGMAC_DATA_SIZE) | ||
707 | return 0; | ||
708 | |||
709 | if (ha->xgmac_data) | ||
710 | goto do_read; | ||
711 | |||
712 | ha->xgmac_data = dma_alloc_coherent(&ha->pdev->dev, XGMAC_DATA_SIZE, | ||
713 | &ha->xgmac_data_dma, GFP_KERNEL); | ||
714 | if (!ha->xgmac_data) { | ||
715 | qla_printk(KERN_WARNING, ha, | ||
716 | "Unable to allocate memory for XGMAC read-data.\n"); | ||
717 | return 0; | ||
718 | } | ||
719 | |||
720 | do_read: | ||
721 | actual_size = 0; | ||
722 | memset(ha->xgmac_data, 0, XGMAC_DATA_SIZE); | ||
723 | |||
724 | rval = qla2x00_get_xgmac_stats(vha, ha->xgmac_data_dma, | ||
725 | XGMAC_DATA_SIZE, &actual_size); | ||
726 | if (rval != QLA_SUCCESS) { | ||
727 | qla_printk(KERN_WARNING, ha, | ||
728 | "Unable to read XGMAC data (%x).\n", rval); | ||
729 | count = 0; | ||
730 | } | ||
731 | |||
732 | count = actual_size > count ? count: actual_size; | ||
733 | memcpy(buf, ha->xgmac_data, count); | ||
734 | |||
735 | return count; | ||
736 | } | ||
737 | |||
738 | static struct bin_attribute sysfs_xgmac_stats_attr = { | ||
739 | .attr = { | ||
740 | .name = "xgmac_stats", | ||
741 | .mode = S_IRUSR, | ||
742 | }, | ||
743 | .size = 0, | ||
744 | .read = qla2x00_sysfs_read_xgmac_stats, | ||
745 | }; | ||
746 | |||
695 | static struct sysfs_entry { | 747 | static struct sysfs_entry { |
696 | char *name; | 748 | char *name; |
697 | struct bin_attribute *attr; | 749 | struct bin_attribute *attr; |
@@ -706,6 +758,7 @@ static struct sysfs_entry { | |||
706 | { "reset", &sysfs_reset_attr, }, | 758 | { "reset", &sysfs_reset_attr, }, |
707 | { "edc", &sysfs_edc_attr, 2 }, | 759 | { "edc", &sysfs_edc_attr, 2 }, |
708 | { "edc_status", &sysfs_edc_status_attr, 2 }, | 760 | { "edc_status", &sysfs_edc_status_attr, 2 }, |
761 | { "xgmac_stats", &sysfs_xgmac_stats_attr, 3 }, | ||
709 | { NULL }, | 762 | { NULL }, |
710 | }; | 763 | }; |
711 | 764 | ||
@@ -721,6 +774,8 @@ qla2x00_alloc_sysfs_attr(scsi_qla_host_t *vha) | |||
721 | continue; | 774 | continue; |
722 | if (iter->is4GBp_only == 2 && !IS_QLA25XX(vha->hw)) | 775 | if (iter->is4GBp_only == 2 && !IS_QLA25XX(vha->hw)) |
723 | continue; | 776 | continue; |
777 | if (iter->is4GBp_only == 3 && !IS_QLA81XX(vha->hw)) | ||
778 | continue; | ||
724 | 779 | ||
725 | ret = sysfs_create_bin_file(&host->shost_gendev.kobj, | 780 | ret = sysfs_create_bin_file(&host->shost_gendev.kobj, |
726 | iter->attr); | 781 | iter->attr); |
@@ -743,6 +798,8 @@ qla2x00_free_sysfs_attr(scsi_qla_host_t *vha) | |||
743 | continue; | 798 | continue; |
744 | if (iter->is4GBp_only == 2 && !IS_QLA25XX(ha)) | 799 | if (iter->is4GBp_only == 2 && !IS_QLA25XX(ha)) |
745 | continue; | 800 | continue; |
801 | if (iter->is4GBp_only == 3 && !IS_QLA81XX(ha)) | ||
802 | continue; | ||
746 | 803 | ||
747 | sysfs_remove_bin_file(&host->shost_gendev.kobj, | 804 | sysfs_remove_bin_file(&host->shost_gendev.kobj, |
748 | iter->attr); | 805 | iter->attr); |
diff --git a/drivers/scsi/qla2xxx/qla_def.h b/drivers/scsi/qla2xxx/qla_def.h index 721bae94e437..da941be9b182 100644 --- a/drivers/scsi/qla2xxx/qla_def.h +++ b/drivers/scsi/qla2xxx/qla_def.h | |||
@@ -2397,6 +2397,10 @@ struct qla_hw_data { | |||
2397 | dma_addr_t edc_data_dma; | 2397 | dma_addr_t edc_data_dma; |
2398 | uint16_t edc_data_len; | 2398 | uint16_t edc_data_len; |
2399 | 2399 | ||
2400 | #define XGMAC_DATA_SIZE PAGE_SIZE | ||
2401 | void *xgmac_data; | ||
2402 | dma_addr_t xgmac_data_dma; | ||
2403 | |||
2400 | struct task_struct *dpc_thread; | 2404 | struct task_struct *dpc_thread; |
2401 | uint8_t dpc_active; /* DPC routine is active */ | 2405 | uint8_t dpc_active; /* DPC routine is active */ |
2402 | 2406 | ||
diff --git a/drivers/scsi/qla2xxx/qla_fw.h b/drivers/scsi/qla2xxx/qla_fw.h index f389f3da0bab..80ab46cfaca6 100644 --- a/drivers/scsi/qla2xxx/qla_fw.h +++ b/drivers/scsi/qla2xxx/qla_fw.h | |||
@@ -1405,6 +1405,7 @@ struct access_chip_rsp_84xx { | |||
1405 | #define MBC_IDC_ACK 0x101 | 1405 | #define MBC_IDC_ACK 0x101 |
1406 | #define MBC_RESTART_MPI_FW 0x3d | 1406 | #define MBC_RESTART_MPI_FW 0x3d |
1407 | #define MBC_FLASH_ACCESS_CTRL 0x3e /* Control flash access. */ | 1407 | #define MBC_FLASH_ACCESS_CTRL 0x3e /* Control flash access. */ |
1408 | #define MBC_GET_XGMAC_STATS 0x7a | ||
1408 | 1409 | ||
1409 | /* Flash access control option field bit definitions */ | 1410 | /* Flash access control option field bit definitions */ |
1410 | #define FAC_OPT_FORCE_SEMAPHORE BIT_15 | 1411 | #define FAC_OPT_FORCE_SEMAPHORE BIT_15 |
diff --git a/drivers/scsi/qla2xxx/qla_gbl.h b/drivers/scsi/qla2xxx/qla_gbl.h index f17d525897a0..66ba3997c91c 100644 --- a/drivers/scsi/qla2xxx/qla_gbl.h +++ b/drivers/scsi/qla2xxx/qla_gbl.h | |||
@@ -293,6 +293,9 @@ qla81xx_fac_do_write_enable(scsi_qla_host_t *, int); | |||
293 | extern int | 293 | extern int |
294 | qla81xx_fac_erase_sector(scsi_qla_host_t *, uint32_t, uint32_t); | 294 | qla81xx_fac_erase_sector(scsi_qla_host_t *, uint32_t, uint32_t); |
295 | 295 | ||
296 | extern int | ||
297 | qla2x00_get_xgmac_stats(scsi_qla_host_t *, dma_addr_t, uint16_t, uint16_t *); | ||
298 | |||
296 | /* | 299 | /* |
297 | * Global Function Prototypes in qla_isr.c source file. | 300 | * Global Function Prototypes in qla_isr.c source file. |
298 | */ | 301 | */ |
diff --git a/drivers/scsi/qla2xxx/qla_mbx.c b/drivers/scsi/qla2xxx/qla_mbx.c index 7d0eeec9ba57..2497fe4ce5aa 100644 --- a/drivers/scsi/qla2xxx/qla_mbx.c +++ b/drivers/scsi/qla2xxx/qla_mbx.c | |||
@@ -3462,3 +3462,41 @@ qla2x00_write_edc(scsi_qla_host_t *vha, uint16_t dev, uint16_t adr, | |||
3462 | 3462 | ||
3463 | return rval; | 3463 | return rval; |
3464 | } | 3464 | } |
3465 | |||
3466 | int | ||
3467 | qla2x00_get_xgmac_stats(scsi_qla_host_t *vha, dma_addr_t stats_dma, | ||
3468 | uint16_t size_in_bytes, uint16_t *actual_size) | ||
3469 | { | ||
3470 | int rval; | ||
3471 | mbx_cmd_t mc; | ||
3472 | mbx_cmd_t *mcp = &mc; | ||
3473 | |||
3474 | if (!IS_QLA81XX(vha->hw)) | ||
3475 | return QLA_FUNCTION_FAILED; | ||
3476 | |||
3477 | DEBUG11(printk("%s(%ld): entered.\n", __func__, vha->host_no)); | ||
3478 | |||
3479 | mcp->mb[0] = MBC_GET_XGMAC_STATS; | ||
3480 | mcp->mb[2] = MSW(stats_dma); | ||
3481 | mcp->mb[3] = LSW(stats_dma); | ||
3482 | mcp->mb[6] = MSW(MSD(stats_dma)); | ||
3483 | mcp->mb[7] = LSW(MSD(stats_dma)); | ||
3484 | mcp->mb[8] = size_in_bytes >> 2; | ||
3485 | mcp->out_mb = MBX_8|MBX_7|MBX_6|MBX_3|MBX_2|MBX_0; | ||
3486 | mcp->in_mb = MBX_2|MBX_1|MBX_0; | ||
3487 | mcp->tov = MBX_TOV_SECONDS; | ||
3488 | mcp->flags = 0; | ||
3489 | rval = qla2x00_mailbox_command(vha, mcp); | ||
3490 | |||
3491 | if (rval != QLA_SUCCESS) { | ||
3492 | DEBUG2_3_11(printk("%s(%ld): failed=%x mb[0]=0x%x " | ||
3493 | "mb[1]=0x%x mb[2]=0x%x.\n", __func__, vha->host_no, rval, | ||
3494 | mcp->mb[0], mcp->mb[1], mcp->mb[2])); | ||
3495 | } else { | ||
3496 | DEBUG11(printk("%s(%ld): done.\n", __func__, vha->host_no)); | ||
3497 | |||
3498 | *actual_size = mcp->mb[2] << 2; | ||
3499 | } | ||
3500 | |||
3501 | return rval; | ||
3502 | } | ||
diff --git a/drivers/scsi/qla2xxx/qla_os.c b/drivers/scsi/qla2xxx/qla_os.c index f4f535536952..642e976083e8 100644 --- a/drivers/scsi/qla2xxx/qla_os.c +++ b/drivers/scsi/qla2xxx/qla_os.c | |||
@@ -2434,6 +2434,10 @@ qla2x00_mem_free(struct qla_hw_data *ha) | |||
2434 | vfree(ha->fw_dump); | 2434 | vfree(ha->fw_dump); |
2435 | } | 2435 | } |
2436 | 2436 | ||
2437 | if (ha->xgmac_data) | ||
2438 | dma_free_coherent(&ha->pdev->dev, XGMAC_DATA_SIZE, | ||
2439 | ha->xgmac_data, ha->xgmac_data_dma); | ||
2440 | |||
2437 | if (ha->sns_cmd) | 2441 | if (ha->sns_cmd) |
2438 | dma_free_coherent(&ha->pdev->dev, sizeof(struct sns_cmd_pkt), | 2442 | dma_free_coherent(&ha->pdev->dev, sizeof(struct sns_cmd_pkt), |
2439 | ha->sns_cmd, ha->sns_cmd_dma); | 2443 | ha->sns_cmd, ha->sns_cmd_dma); |