diff options
Diffstat (limited to 'drivers/scsi/qla2xxx/qla_sup.c')
-rw-r--r-- | drivers/scsi/qla2xxx/qla_sup.c | 153 |
1 files changed, 110 insertions, 43 deletions
diff --git a/drivers/scsi/qla2xxx/qla_sup.c b/drivers/scsi/qla2xxx/qla_sup.c index c10ccc75c398..eda11f6f5e52 100644 --- a/drivers/scsi/qla2xxx/qla_sup.c +++ b/drivers/scsi/qla2xxx/qla_sup.c | |||
@@ -543,6 +543,96 @@ qla24xx_get_flash_manufacturer(scsi_qla_host_t *ha, uint8_t *man_id, | |||
543 | } | 543 | } |
544 | } | 544 | } |
545 | 545 | ||
546 | void | ||
547 | qla2xxx_get_flash_info(scsi_qla_host_t *ha) | ||
548 | { | ||
549 | #define FLASH_BLK_SIZE_32K 0x8000 | ||
550 | #define FLASH_BLK_SIZE_64K 0x10000 | ||
551 | uint16_t cnt, chksum; | ||
552 | uint16_t *wptr; | ||
553 | struct qla_fdt_layout *fdt; | ||
554 | uint8_t man_id, flash_id; | ||
555 | |||
556 | if (!IS_QLA24XX(ha) && !IS_QLA25XX(ha)) | ||
557 | return; | ||
558 | |||
559 | wptr = (uint16_t *)ha->request_ring; | ||
560 | fdt = (struct qla_fdt_layout *)ha->request_ring; | ||
561 | ha->isp_ops->read_optrom(ha, (uint8_t *)ha->request_ring, | ||
562 | FA_FLASH_DESCR_ADDR << 2, OPTROM_BURST_SIZE); | ||
563 | if (*wptr == __constant_cpu_to_le16(0xffff)) | ||
564 | goto no_flash_data; | ||
565 | if (fdt->sig[0] != 'Q' || fdt->sig[1] != 'L' || fdt->sig[2] != 'I' || | ||
566 | fdt->sig[3] != 'D') | ||
567 | goto no_flash_data; | ||
568 | |||
569 | for (cnt = 0, chksum = 0; cnt < sizeof(struct qla_fdt_layout) >> 1; | ||
570 | cnt++) | ||
571 | chksum += le16_to_cpu(*wptr++); | ||
572 | if (chksum) { | ||
573 | DEBUG2(qla_printk(KERN_INFO, ha, "Inconsistent FDT detected: " | ||
574 | "checksum=0x%x id=%c version=0x%x.\n", chksum, fdt->sig[0], | ||
575 | le16_to_cpu(fdt->version))); | ||
576 | DEBUG9(qla2x00_dump_buffer((uint8_t *)fdt, sizeof(*fdt))); | ||
577 | goto no_flash_data; | ||
578 | } | ||
579 | |||
580 | ha->fdt_odd_index = le16_to_cpu(fdt->man_id) == 0x1f; | ||
581 | ha->fdt_wrt_disable = fdt->wrt_disable_bits; | ||
582 | ha->fdt_erase_cmd = flash_conf_to_access_addr(0x0300 | fdt->erase_cmd); | ||
583 | ha->fdt_block_size = le32_to_cpu(fdt->block_size); | ||
584 | if (fdt->unprotect_sec_cmd) { | ||
585 | ha->fdt_unprotect_sec_cmd = flash_conf_to_access_addr(0x0300 | | ||
586 | fdt->unprotect_sec_cmd); | ||
587 | ha->fdt_protect_sec_cmd = fdt->protect_sec_cmd ? | ||
588 | flash_conf_to_access_addr(0x0300 | fdt->protect_sec_cmd): | ||
589 | flash_conf_to_access_addr(0x0336); | ||
590 | } | ||
591 | |||
592 | DEBUG2(qla_printk(KERN_DEBUG, ha, "Flash[FDT]: (0x%x/0x%x) erase=0x%x " | ||
593 | "pro=%x upro=%x idx=%d wrtd=0x%x blk=0x%x.\n", | ||
594 | le16_to_cpu(fdt->man_id), le16_to_cpu(fdt->id), ha->fdt_erase_cmd, | ||
595 | ha->fdt_protect_sec_cmd, ha->fdt_unprotect_sec_cmd, | ||
596 | ha->fdt_odd_index, ha->fdt_wrt_disable, ha->fdt_block_size)); | ||
597 | return; | ||
598 | |||
599 | no_flash_data: | ||
600 | qla24xx_get_flash_manufacturer(ha, &man_id, &flash_id); | ||
601 | ha->fdt_wrt_disable = 0x9c; | ||
602 | ha->fdt_erase_cmd = flash_conf_to_access_addr(0x03d8); | ||
603 | switch (man_id) { | ||
604 | case 0xbf: /* STT flash. */ | ||
605 | if (flash_id == 0x8e) | ||
606 | ha->fdt_block_size = FLASH_BLK_SIZE_64K; | ||
607 | else | ||
608 | ha->fdt_block_size = FLASH_BLK_SIZE_32K; | ||
609 | |||
610 | if (flash_id == 0x80) | ||
611 | ha->fdt_erase_cmd = flash_conf_to_access_addr(0x0352); | ||
612 | break; | ||
613 | case 0x13: /* ST M25P80. */ | ||
614 | ha->fdt_block_size = FLASH_BLK_SIZE_64K; | ||
615 | break; | ||
616 | case 0x1f: /* Atmel 26DF081A. */ | ||
617 | ha->fdt_odd_index = 1; | ||
618 | ha->fdt_block_size = FLASH_BLK_SIZE_64K; | ||
619 | ha->fdt_erase_cmd = flash_conf_to_access_addr(0x0320); | ||
620 | ha->fdt_unprotect_sec_cmd = flash_conf_to_access_addr(0x0339); | ||
621 | ha->fdt_protect_sec_cmd = flash_conf_to_access_addr(0x0336); | ||
622 | break; | ||
623 | default: | ||
624 | /* Default to 64 kb sector size. */ | ||
625 | ha->fdt_block_size = FLASH_BLK_SIZE_64K; | ||
626 | break; | ||
627 | } | ||
628 | |||
629 | DEBUG2(qla_printk(KERN_DEBUG, ha, "Flash[MID]: (0x%x/0x%x) erase=0x%x " | ||
630 | "pro=%x upro=%x idx=%d wrtd=0x%x blk=0x%x.\n", man_id, flash_id, | ||
631 | ha->fdt_erase_cmd, ha->fdt_protect_sec_cmd, | ||
632 | ha->fdt_unprotect_sec_cmd, ha->fdt_odd_index, ha->fdt_wrt_disable, | ||
633 | ha->fdt_block_size)); | ||
634 | } | ||
635 | |||
546 | static void | 636 | static void |
547 | qla24xx_unprotect_flash(scsi_qla_host_t *ha) | 637 | qla24xx_unprotect_flash(scsi_qla_host_t *ha) |
548 | { | 638 | { |
@@ -553,6 +643,9 @@ qla24xx_unprotect_flash(scsi_qla_host_t *ha) | |||
553 | RD_REG_DWORD(®->ctrl_status) | CSRX_FLASH_ENABLE); | 643 | RD_REG_DWORD(®->ctrl_status) | CSRX_FLASH_ENABLE); |
554 | RD_REG_DWORD(®->ctrl_status); /* PCI Posting. */ | 644 | RD_REG_DWORD(®->ctrl_status); /* PCI Posting. */ |
555 | 645 | ||
646 | if (!ha->fdt_wrt_disable) | ||
647 | return; | ||
648 | |||
556 | /* Disable flash write-protection. */ | 649 | /* Disable flash write-protection. */ |
557 | qla24xx_write_flash_dword(ha, flash_conf_to_access_addr(0x101), 0); | 650 | qla24xx_write_flash_dword(ha, flash_conf_to_access_addr(0x101), 0); |
558 | /* Some flash parts need an additional zero-write to clear bits.*/ | 651 | /* Some flash parts need an additional zero-write to clear bits.*/ |
@@ -565,8 +658,12 @@ qla24xx_protect_flash(scsi_qla_host_t *ha) | |||
565 | uint32_t cnt; | 658 | uint32_t cnt; |
566 | struct device_reg_24xx __iomem *reg = &ha->iobase->isp24; | 659 | struct device_reg_24xx __iomem *reg = &ha->iobase->isp24; |
567 | 660 | ||
661 | if (!ha->fdt_wrt_disable) | ||
662 | goto skip_wrt_protect; | ||
663 | |||
568 | /* Enable flash write-protection and wait for completion. */ | 664 | /* Enable flash write-protection and wait for completion. */ |
569 | qla24xx_write_flash_dword(ha, flash_conf_to_access_addr(0x101), 0x9c); | 665 | qla24xx_write_flash_dword(ha, flash_conf_to_access_addr(0x101), |
666 | ha->fdt_wrt_disable); | ||
570 | for (cnt = 300; cnt && | 667 | for (cnt = 300; cnt && |
571 | qla24xx_read_flash_dword(ha, | 668 | qla24xx_read_flash_dword(ha, |
572 | flash_conf_to_access_addr(0x005)) & BIT_0; | 669 | flash_conf_to_access_addr(0x005)) & BIT_0; |
@@ -574,6 +671,7 @@ qla24xx_protect_flash(scsi_qla_host_t *ha) | |||
574 | udelay(10); | 671 | udelay(10); |
575 | } | 672 | } |
576 | 673 | ||
674 | skip_wrt_protect: | ||
577 | /* Disable flash write. */ | 675 | /* Disable flash write. */ |
578 | WRT_REG_DWORD(®->ctrl_status, | 676 | WRT_REG_DWORD(®->ctrl_status, |
579 | RD_REG_DWORD(®->ctrl_status) & ~CSRX_FLASH_ENABLE); | 677 | RD_REG_DWORD(®->ctrl_status) & ~CSRX_FLASH_ENABLE); |
@@ -586,9 +684,8 @@ qla24xx_write_flash_data(scsi_qla_host_t *ha, uint32_t *dwptr, uint32_t faddr, | |||
586 | { | 684 | { |
587 | int ret; | 685 | int ret; |
588 | uint32_t liter, miter; | 686 | uint32_t liter, miter; |
589 | uint32_t sec_mask, rest_addr, conf_addr; | 687 | uint32_t sec_mask, rest_addr; |
590 | uint32_t fdata, findex; | 688 | uint32_t fdata, findex; |
591 | uint8_t man_id, flash_id; | ||
592 | dma_addr_t optrom_dma; | 689 | dma_addr_t optrom_dma; |
593 | void *optrom = NULL; | 690 | void *optrom = NULL; |
594 | uint32_t *s, *d; | 691 | uint32_t *s, *d; |
@@ -607,43 +704,13 @@ qla24xx_write_flash_data(scsi_qla_host_t *ha, uint32_t *dwptr, uint32_t faddr, | |||
607 | } | 704 | } |
608 | } | 705 | } |
609 | 706 | ||
610 | qla24xx_get_flash_manufacturer(ha, &man_id, &flash_id); | 707 | rest_addr = (ha->fdt_block_size >> 2) - 1; |
611 | DEBUG9(printk("%s(%ld): Flash man_id=%d flash_id=%d\n", __func__, | 708 | sec_mask = 0x80000 - (ha->fdt_block_size >> 2); |
612 | ha->host_no, man_id, flash_id)); | ||
613 | |||
614 | conf_addr = flash_conf_to_access_addr(0x03d8); | ||
615 | switch (man_id) { | ||
616 | case 0xbf: /* STT flash. */ | ||
617 | if (flash_id == 0x8e) { | ||
618 | rest_addr = 0x3fff; | ||
619 | sec_mask = 0x7c000; | ||
620 | } else { | ||
621 | rest_addr = 0x1fff; | ||
622 | sec_mask = 0x7e000; | ||
623 | } | ||
624 | if (flash_id == 0x80) | ||
625 | conf_addr = flash_conf_to_access_addr(0x0352); | ||
626 | break; | ||
627 | case 0x13: /* ST M25P80. */ | ||
628 | rest_addr = 0x3fff; | ||
629 | sec_mask = 0x7c000; | ||
630 | break; | ||
631 | case 0x1f: // Atmel 26DF081A | ||
632 | rest_addr = 0x3fff; | ||
633 | sec_mask = 0x7c000; | ||
634 | conf_addr = flash_conf_to_access_addr(0x0320); | ||
635 | break; | ||
636 | default: | ||
637 | /* Default to 64 kb sector size. */ | ||
638 | rest_addr = 0x3fff; | ||
639 | sec_mask = 0x7c000; | ||
640 | break; | ||
641 | } | ||
642 | 709 | ||
643 | qla24xx_unprotect_flash(ha); | 710 | qla24xx_unprotect_flash(ha); |
644 | 711 | ||
645 | for (liter = 0; liter < dwords; liter++, faddr++, dwptr++) { | 712 | for (liter = 0; liter < dwords; liter++, faddr++, dwptr++) { |
646 | if (man_id == 0x1f) { | 713 | if (ha->fdt_odd_index) { |
647 | findex = faddr << 2; | 714 | findex = faddr << 2; |
648 | fdata = findex & sec_mask; | 715 | fdata = findex & sec_mask; |
649 | } else { | 716 | } else { |
@@ -653,13 +720,13 @@ qla24xx_write_flash_data(scsi_qla_host_t *ha, uint32_t *dwptr, uint32_t faddr, | |||
653 | 720 | ||
654 | /* Are we at the beginning of a sector? */ | 721 | /* Are we at the beginning of a sector? */ |
655 | if ((findex & rest_addr) == 0) { | 722 | if ((findex & rest_addr) == 0) { |
656 | /* Do sector unprotect at 4K boundry for Atmel part. */ | 723 | /* Do sector unprotect. */ |
657 | if (man_id == 0x1f) | 724 | if (ha->fdt_unprotect_sec_cmd) |
658 | qla24xx_write_flash_dword(ha, | 725 | qla24xx_write_flash_dword(ha, |
659 | flash_conf_to_access_addr(0x0339), | 726 | ha->fdt_unprotect_sec_cmd, |
660 | (fdata & 0xff00) | ((fdata << 16) & | 727 | (fdata & 0xff00) | ((fdata << 16) & |
661 | 0xff0000) | ((fdata >> 16) & 0xff)); | 728 | 0xff0000) | ((fdata >> 16) & 0xff)); |
662 | ret = qla24xx_write_flash_dword(ha, conf_addr, | 729 | ret = qla24xx_write_flash_dword(ha, ha->fdt_erase_cmd, |
663 | (fdata & 0xff00) |((fdata << 16) & | 730 | (fdata & 0xff00) |((fdata << 16) & |
664 | 0xff0000) | ((fdata >> 16) & 0xff)); | 731 | 0xff0000) | ((fdata >> 16) & 0xff)); |
665 | if (ret != QLA_SUCCESS) { | 732 | if (ret != QLA_SUCCESS) { |
@@ -709,11 +776,11 @@ qla24xx_write_flash_data(scsi_qla_host_t *ha, uint32_t *dwptr, uint32_t faddr, | |||
709 | break; | 776 | break; |
710 | } | 777 | } |
711 | 778 | ||
712 | /* Do sector protect at 4K boundry for Atmel part. */ | 779 | /* Do sector protect. */ |
713 | if (man_id == 0x1f && | 780 | if (ha->fdt_unprotect_sec_cmd && |
714 | ((faddr & rest_addr) == rest_addr)) | 781 | ((faddr & rest_addr) == rest_addr)) |
715 | qla24xx_write_flash_dword(ha, | 782 | qla24xx_write_flash_dword(ha, |
716 | flash_conf_to_access_addr(0x0336), | 783 | ha->fdt_protect_sec_cmd, |
717 | (fdata & 0xff00) | ((fdata << 16) & | 784 | (fdata & 0xff00) | ((fdata << 16) & |
718 | 0xff0000) | ((fdata >> 16) & 0xff)); | 785 | 0xff0000) | ((fdata >> 16) & 0xff)); |
719 | } | 786 | } |