diff options
Diffstat (limited to 'drivers/scsi/qla2xxx/qla_sup.c')
-rw-r--r-- | drivers/scsi/qla2xxx/qla_sup.c | 669 |
1 files changed, 669 insertions, 0 deletions
diff --git a/drivers/scsi/qla2xxx/qla_sup.c b/drivers/scsi/qla2xxx/qla_sup.c index 3105bb93cc19..3866a5760f15 100644 --- a/drivers/scsi/qla2xxx/qla_sup.c +++ b/drivers/scsi/qla2xxx/qla_sup.c | |||
@@ -989,3 +989,672 @@ qla24xx_beacon_off(struct scsi_qla_host *ha) | |||
989 | 989 | ||
990 | return QLA_SUCCESS; | 990 | return QLA_SUCCESS; |
991 | } | 991 | } |
992 | |||
993 | |||
994 | /* | ||
995 | * Flash support routines | ||
996 | */ | ||
997 | |||
998 | /** | ||
999 | * qla2x00_flash_enable() - Setup flash for reading and writing. | ||
1000 | * @ha: HA context | ||
1001 | */ | ||
1002 | static void | ||
1003 | qla2x00_flash_enable(scsi_qla_host_t *ha) | ||
1004 | { | ||
1005 | uint16_t data; | ||
1006 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1007 | |||
1008 | data = RD_REG_WORD(®->ctrl_status); | ||
1009 | data |= CSR_FLASH_ENABLE; | ||
1010 | WRT_REG_WORD(®->ctrl_status, data); | ||
1011 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1012 | } | ||
1013 | |||
1014 | /** | ||
1015 | * qla2x00_flash_disable() - Disable flash and allow RISC to run. | ||
1016 | * @ha: HA context | ||
1017 | */ | ||
1018 | static void | ||
1019 | qla2x00_flash_disable(scsi_qla_host_t *ha) | ||
1020 | { | ||
1021 | uint16_t data; | ||
1022 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1023 | |||
1024 | data = RD_REG_WORD(®->ctrl_status); | ||
1025 | data &= ~(CSR_FLASH_ENABLE); | ||
1026 | WRT_REG_WORD(®->ctrl_status, data); | ||
1027 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1028 | } | ||
1029 | |||
1030 | /** | ||
1031 | * qla2x00_read_flash_byte() - Reads a byte from flash | ||
1032 | * @ha: HA context | ||
1033 | * @addr: Address in flash to read | ||
1034 | * | ||
1035 | * A word is read from the chip, but, only the lower byte is valid. | ||
1036 | * | ||
1037 | * Returns the byte read from flash @addr. | ||
1038 | */ | ||
1039 | static uint8_t | ||
1040 | qla2x00_read_flash_byte(scsi_qla_host_t *ha, uint32_t addr) | ||
1041 | { | ||
1042 | uint16_t data; | ||
1043 | uint16_t bank_select; | ||
1044 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1045 | |||
1046 | bank_select = RD_REG_WORD(®->ctrl_status); | ||
1047 | |||
1048 | if (IS_QLA2322(ha) || IS_QLA6322(ha)) { | ||
1049 | /* Specify 64K address range: */ | ||
1050 | /* clear out Module Select and Flash Address bits [19:16]. */ | ||
1051 | bank_select &= ~0xf8; | ||
1052 | bank_select |= addr >> 12 & 0xf0; | ||
1053 | bank_select |= CSR_FLASH_64K_BANK; | ||
1054 | WRT_REG_WORD(®->ctrl_status, bank_select); | ||
1055 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1056 | |||
1057 | WRT_REG_WORD(®->flash_address, (uint16_t)addr); | ||
1058 | data = RD_REG_WORD(®->flash_data); | ||
1059 | |||
1060 | return (uint8_t)data; | ||
1061 | } | ||
1062 | |||
1063 | /* Setup bit 16 of flash address. */ | ||
1064 | if ((addr & BIT_16) && ((bank_select & CSR_FLASH_64K_BANK) == 0)) { | ||
1065 | bank_select |= CSR_FLASH_64K_BANK; | ||
1066 | WRT_REG_WORD(®->ctrl_status, bank_select); | ||
1067 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1068 | } else if (((addr & BIT_16) == 0) && | ||
1069 | (bank_select & CSR_FLASH_64K_BANK)) { | ||
1070 | bank_select &= ~(CSR_FLASH_64K_BANK); | ||
1071 | WRT_REG_WORD(®->ctrl_status, bank_select); | ||
1072 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1073 | } | ||
1074 | |||
1075 | /* Always perform IO mapped accesses to the FLASH registers. */ | ||
1076 | if (ha->pio_address) { | ||
1077 | uint16_t data2; | ||
1078 | |||
1079 | reg = (struct device_reg_2xxx __iomem *)ha->pio_address; | ||
1080 | WRT_REG_WORD_PIO(®->flash_address, (uint16_t)addr); | ||
1081 | do { | ||
1082 | data = RD_REG_WORD_PIO(®->flash_data); | ||
1083 | barrier(); | ||
1084 | cpu_relax(); | ||
1085 | data2 = RD_REG_WORD_PIO(®->flash_data); | ||
1086 | } while (data != data2); | ||
1087 | } else { | ||
1088 | WRT_REG_WORD(®->flash_address, (uint16_t)addr); | ||
1089 | data = qla2x00_debounce_register(®->flash_data); | ||
1090 | } | ||
1091 | |||
1092 | return (uint8_t)data; | ||
1093 | } | ||
1094 | |||
1095 | /** | ||
1096 | * qla2x00_write_flash_byte() - Write a byte to flash | ||
1097 | * @ha: HA context | ||
1098 | * @addr: Address in flash to write | ||
1099 | * @data: Data to write | ||
1100 | */ | ||
1101 | static void | ||
1102 | qla2x00_write_flash_byte(scsi_qla_host_t *ha, uint32_t addr, uint8_t data) | ||
1103 | { | ||
1104 | uint16_t bank_select; | ||
1105 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1106 | |||
1107 | bank_select = RD_REG_WORD(®->ctrl_status); | ||
1108 | if (IS_QLA2322(ha) || IS_QLA6322(ha)) { | ||
1109 | /* Specify 64K address range: */ | ||
1110 | /* clear out Module Select and Flash Address bits [19:16]. */ | ||
1111 | bank_select &= ~0xf8; | ||
1112 | bank_select |= addr >> 12 & 0xf0; | ||
1113 | bank_select |= CSR_FLASH_64K_BANK; | ||
1114 | WRT_REG_WORD(®->ctrl_status, bank_select); | ||
1115 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1116 | |||
1117 | WRT_REG_WORD(®->flash_address, (uint16_t)addr); | ||
1118 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1119 | WRT_REG_WORD(®->flash_data, (uint16_t)data); | ||
1120 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1121 | |||
1122 | return; | ||
1123 | } | ||
1124 | |||
1125 | /* Setup bit 16 of flash address. */ | ||
1126 | if ((addr & BIT_16) && ((bank_select & CSR_FLASH_64K_BANK) == 0)) { | ||
1127 | bank_select |= CSR_FLASH_64K_BANK; | ||
1128 | WRT_REG_WORD(®->ctrl_status, bank_select); | ||
1129 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1130 | } else if (((addr & BIT_16) == 0) && | ||
1131 | (bank_select & CSR_FLASH_64K_BANK)) { | ||
1132 | bank_select &= ~(CSR_FLASH_64K_BANK); | ||
1133 | WRT_REG_WORD(®->ctrl_status, bank_select); | ||
1134 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1135 | } | ||
1136 | |||
1137 | /* Always perform IO mapped accesses to the FLASH registers. */ | ||
1138 | if (ha->pio_address) { | ||
1139 | reg = (struct device_reg_2xxx __iomem *)ha->pio_address; | ||
1140 | WRT_REG_WORD_PIO(®->flash_address, (uint16_t)addr); | ||
1141 | WRT_REG_WORD_PIO(®->flash_data, (uint16_t)data); | ||
1142 | } else { | ||
1143 | WRT_REG_WORD(®->flash_address, (uint16_t)addr); | ||
1144 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1145 | WRT_REG_WORD(®->flash_data, (uint16_t)data); | ||
1146 | RD_REG_WORD(®->ctrl_status); /* PCI Posting. */ | ||
1147 | } | ||
1148 | } | ||
1149 | |||
1150 | /** | ||
1151 | * qla2x00_poll_flash() - Polls flash for completion. | ||
1152 | * @ha: HA context | ||
1153 | * @addr: Address in flash to poll | ||
1154 | * @poll_data: Data to be polled | ||
1155 | * @man_id: Flash manufacturer ID | ||
1156 | * @flash_id: Flash ID | ||
1157 | * | ||
1158 | * This function polls the device until bit 7 of what is read matches data | ||
1159 | * bit 7 or until data bit 5 becomes a 1. If that hapens, the flash ROM timed | ||
1160 | * out (a fatal error). The flash book recommeds reading bit 7 again after | ||
1161 | * reading bit 5 as a 1. | ||
1162 | * | ||
1163 | * Returns 0 on success, else non-zero. | ||
1164 | */ | ||
1165 | static int | ||
1166 | qla2x00_poll_flash(scsi_qla_host_t *ha, uint32_t addr, uint8_t poll_data, | ||
1167 | uint8_t man_id, uint8_t flash_id) | ||
1168 | { | ||
1169 | int status; | ||
1170 | uint8_t flash_data; | ||
1171 | uint32_t cnt; | ||
1172 | |||
1173 | status = 1; | ||
1174 | |||
1175 | /* Wait for 30 seconds for command to finish. */ | ||
1176 | poll_data &= BIT_7; | ||
1177 | for (cnt = 3000000; cnt; cnt--) { | ||
1178 | flash_data = qla2x00_read_flash_byte(ha, addr); | ||
1179 | if ((flash_data & BIT_7) == poll_data) { | ||
1180 | status = 0; | ||
1181 | break; | ||
1182 | } | ||
1183 | |||
1184 | if (man_id != 0x40 && man_id != 0xda) { | ||
1185 | if ((flash_data & BIT_5) && cnt > 2) | ||
1186 | cnt = 2; | ||
1187 | } | ||
1188 | udelay(10); | ||
1189 | barrier(); | ||
1190 | } | ||
1191 | return status; | ||
1192 | } | ||
1193 | |||
1194 | #define IS_OEM_001(ha) \ | ||
1195 | ((ha)->pdev->device == PCI_DEVICE_ID_QLOGIC_ISP2322 && \ | ||
1196 | (ha)->pdev->subsystem_vendor == 0x1028 && \ | ||
1197 | (ha)->pdev->subsystem_device == 0x0170) | ||
1198 | |||
1199 | /** | ||
1200 | * qla2x00_program_flash_address() - Programs a flash address | ||
1201 | * @ha: HA context | ||
1202 | * @addr: Address in flash to program | ||
1203 | * @data: Data to be written in flash | ||
1204 | * @man_id: Flash manufacturer ID | ||
1205 | * @flash_id: Flash ID | ||
1206 | * | ||
1207 | * Returns 0 on success, else non-zero. | ||
1208 | */ | ||
1209 | static int | ||
1210 | qla2x00_program_flash_address(scsi_qla_host_t *ha, uint32_t addr, uint8_t data, | ||
1211 | uint8_t man_id, uint8_t flash_id) | ||
1212 | { | ||
1213 | /* Write Program Command Sequence. */ | ||
1214 | if (IS_OEM_001(ha)) { | ||
1215 | qla2x00_write_flash_byte(ha, 0xaaa, 0xaa); | ||
1216 | qla2x00_write_flash_byte(ha, 0x555, 0x55); | ||
1217 | qla2x00_write_flash_byte(ha, 0xaaa, 0xa0); | ||
1218 | qla2x00_write_flash_byte(ha, addr, data); | ||
1219 | } else { | ||
1220 | if (man_id == 0xda && flash_id == 0xc1) { | ||
1221 | qla2x00_write_flash_byte(ha, addr, data); | ||
1222 | if (addr & 0x7e) | ||
1223 | return 0; | ||
1224 | } else { | ||
1225 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1226 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1227 | qla2x00_write_flash_byte(ha, 0x5555, 0xa0); | ||
1228 | qla2x00_write_flash_byte(ha, addr, data); | ||
1229 | } | ||
1230 | } | ||
1231 | |||
1232 | udelay(150); | ||
1233 | |||
1234 | /* Wait for write to complete. */ | ||
1235 | return qla2x00_poll_flash(ha, addr, data, man_id, flash_id); | ||
1236 | } | ||
1237 | |||
1238 | /** | ||
1239 | * qla2x00_erase_flash() - Erase the flash. | ||
1240 | * @ha: HA context | ||
1241 | * @man_id: Flash manufacturer ID | ||
1242 | * @flash_id: Flash ID | ||
1243 | * | ||
1244 | * Returns 0 on success, else non-zero. | ||
1245 | */ | ||
1246 | static int | ||
1247 | qla2x00_erase_flash(scsi_qla_host_t *ha, uint8_t man_id, uint8_t flash_id) | ||
1248 | { | ||
1249 | /* Individual Sector Erase Command Sequence */ | ||
1250 | if (IS_OEM_001(ha)) { | ||
1251 | qla2x00_write_flash_byte(ha, 0xaaa, 0xaa); | ||
1252 | qla2x00_write_flash_byte(ha, 0x555, 0x55); | ||
1253 | qla2x00_write_flash_byte(ha, 0xaaa, 0x80); | ||
1254 | qla2x00_write_flash_byte(ha, 0xaaa, 0xaa); | ||
1255 | qla2x00_write_flash_byte(ha, 0x555, 0x55); | ||
1256 | qla2x00_write_flash_byte(ha, 0xaaa, 0x10); | ||
1257 | } else { | ||
1258 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1259 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1260 | qla2x00_write_flash_byte(ha, 0x5555, 0x80); | ||
1261 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1262 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1263 | qla2x00_write_flash_byte(ha, 0x5555, 0x10); | ||
1264 | } | ||
1265 | |||
1266 | udelay(150); | ||
1267 | |||
1268 | /* Wait for erase to complete. */ | ||
1269 | return qla2x00_poll_flash(ha, 0x00, 0x80, man_id, flash_id); | ||
1270 | } | ||
1271 | |||
1272 | /** | ||
1273 | * qla2x00_erase_flash_sector() - Erase a flash sector. | ||
1274 | * @ha: HA context | ||
1275 | * @addr: Flash sector to erase | ||
1276 | * @sec_mask: Sector address mask | ||
1277 | * @man_id: Flash manufacturer ID | ||
1278 | * @flash_id: Flash ID | ||
1279 | * | ||
1280 | * Returns 0 on success, else non-zero. | ||
1281 | */ | ||
1282 | static int | ||
1283 | qla2x00_erase_flash_sector(scsi_qla_host_t *ha, uint32_t addr, | ||
1284 | uint32_t sec_mask, uint8_t man_id, uint8_t flash_id) | ||
1285 | { | ||
1286 | /* Individual Sector Erase Command Sequence */ | ||
1287 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1288 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1289 | qla2x00_write_flash_byte(ha, 0x5555, 0x80); | ||
1290 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1291 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1292 | if (man_id == 0x1f && flash_id == 0x13) | ||
1293 | qla2x00_write_flash_byte(ha, addr & sec_mask, 0x10); | ||
1294 | else | ||
1295 | qla2x00_write_flash_byte(ha, addr & sec_mask, 0x30); | ||
1296 | |||
1297 | udelay(150); | ||
1298 | |||
1299 | /* Wait for erase to complete. */ | ||
1300 | return qla2x00_poll_flash(ha, addr, 0x80, man_id, flash_id); | ||
1301 | } | ||
1302 | |||
1303 | /** | ||
1304 | * qla2x00_get_flash_manufacturer() - Read manufacturer ID from flash chip. | ||
1305 | * @man_id: Flash manufacturer ID | ||
1306 | * @flash_id: Flash ID | ||
1307 | */ | ||
1308 | static void | ||
1309 | qla2x00_get_flash_manufacturer(scsi_qla_host_t *ha, uint8_t *man_id, | ||
1310 | uint8_t *flash_id) | ||
1311 | { | ||
1312 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1313 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1314 | qla2x00_write_flash_byte(ha, 0x5555, 0x90); | ||
1315 | *man_id = qla2x00_read_flash_byte(ha, 0x0000); | ||
1316 | *flash_id = qla2x00_read_flash_byte(ha, 0x0001); | ||
1317 | qla2x00_write_flash_byte(ha, 0x5555, 0xaa); | ||
1318 | qla2x00_write_flash_byte(ha, 0x2aaa, 0x55); | ||
1319 | qla2x00_write_flash_byte(ha, 0x5555, 0xf0); | ||
1320 | } | ||
1321 | |||
1322 | |||
1323 | static inline void | ||
1324 | qla2x00_suspend_hba(struct scsi_qla_host *ha) | ||
1325 | { | ||
1326 | int cnt; | ||
1327 | unsigned long flags; | ||
1328 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1329 | |||
1330 | /* Suspend HBA. */ | ||
1331 | scsi_block_requests(ha->host); | ||
1332 | ha->isp_ops.disable_intrs(ha); | ||
1333 | set_bit(MBX_UPDATE_FLASH_ACTIVE, &ha->mbx_cmd_flags); | ||
1334 | |||
1335 | /* Pause RISC. */ | ||
1336 | spin_lock_irqsave(&ha->hardware_lock, flags); | ||
1337 | WRT_REG_WORD(®->hccr, HCCR_PAUSE_RISC); | ||
1338 | RD_REG_WORD(®->hccr); | ||
1339 | if (IS_QLA2100(ha) || IS_QLA2200(ha) || IS_QLA2300(ha)) { | ||
1340 | for (cnt = 0; cnt < 30000; cnt++) { | ||
1341 | if ((RD_REG_WORD(®->hccr) & HCCR_RISC_PAUSE) != 0) | ||
1342 | break; | ||
1343 | udelay(100); | ||
1344 | } | ||
1345 | } else { | ||
1346 | udelay(10); | ||
1347 | } | ||
1348 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | ||
1349 | } | ||
1350 | |||
1351 | static inline void | ||
1352 | qla2x00_resume_hba(struct scsi_qla_host *ha) | ||
1353 | { | ||
1354 | /* Resume HBA. */ | ||
1355 | clear_bit(MBX_UPDATE_FLASH_ACTIVE, &ha->mbx_cmd_flags); | ||
1356 | set_bit(ISP_ABORT_NEEDED, &ha->dpc_flags); | ||
1357 | up(ha->dpc_wait); | ||
1358 | qla2x00_wait_for_hba_online(ha); | ||
1359 | scsi_unblock_requests(ha->host); | ||
1360 | } | ||
1361 | |||
1362 | uint8_t * | ||
1363 | qla2x00_read_optrom_data(struct scsi_qla_host *ha, uint8_t *buf, | ||
1364 | uint32_t offset, uint32_t length) | ||
1365 | { | ||
1366 | unsigned long flags; | ||
1367 | uint32_t addr, midpoint; | ||
1368 | uint8_t *data; | ||
1369 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1370 | |||
1371 | /* Suspend HBA. */ | ||
1372 | qla2x00_suspend_hba(ha); | ||
1373 | |||
1374 | /* Go with read. */ | ||
1375 | spin_lock_irqsave(&ha->hardware_lock, flags); | ||
1376 | midpoint = ha->optrom_size / 2; | ||
1377 | |||
1378 | qla2x00_flash_enable(ha); | ||
1379 | WRT_REG_WORD(®->nvram, 0); | ||
1380 | RD_REG_WORD(®->nvram); /* PCI Posting. */ | ||
1381 | for (addr = offset, data = buf; addr < length; addr++, data++) { | ||
1382 | if (addr == midpoint) { | ||
1383 | WRT_REG_WORD(®->nvram, NVR_SELECT); | ||
1384 | RD_REG_WORD(®->nvram); /* PCI Posting. */ | ||
1385 | } | ||
1386 | |||
1387 | *data = qla2x00_read_flash_byte(ha, addr); | ||
1388 | } | ||
1389 | qla2x00_flash_disable(ha); | ||
1390 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | ||
1391 | |||
1392 | /* Resume HBA. */ | ||
1393 | qla2x00_resume_hba(ha); | ||
1394 | |||
1395 | return buf; | ||
1396 | } | ||
1397 | |||
1398 | int | ||
1399 | qla2x00_write_optrom_data(struct scsi_qla_host *ha, uint8_t *buf, | ||
1400 | uint32_t offset, uint32_t length) | ||
1401 | { | ||
1402 | |||
1403 | int rval; | ||
1404 | unsigned long flags; | ||
1405 | uint8_t man_id, flash_id, sec_number, data; | ||
1406 | uint16_t wd; | ||
1407 | uint32_t addr, liter, sec_mask, rest_addr; | ||
1408 | struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; | ||
1409 | |||
1410 | /* Suspend HBA. */ | ||
1411 | qla2x00_suspend_hba(ha); | ||
1412 | |||
1413 | rval = QLA_SUCCESS; | ||
1414 | sec_number = 0; | ||
1415 | |||
1416 | /* Reset ISP chip. */ | ||
1417 | spin_lock_irqsave(&ha->hardware_lock, flags); | ||
1418 | WRT_REG_WORD(®->ctrl_status, CSR_ISP_SOFT_RESET); | ||
1419 | pci_read_config_word(ha->pdev, PCI_COMMAND, &wd); | ||
1420 | |||
1421 | /* Go with write. */ | ||
1422 | qla2x00_flash_enable(ha); | ||
1423 | do { /* Loop once to provide quick error exit */ | ||
1424 | /* Structure of flash memory based on manufacturer */ | ||
1425 | if (IS_OEM_001(ha)) { | ||
1426 | /* OEM variant with special flash part. */ | ||
1427 | man_id = flash_id = 0; | ||
1428 | rest_addr = 0xffff; | ||
1429 | sec_mask = 0x10000; | ||
1430 | goto update_flash; | ||
1431 | } | ||
1432 | qla2x00_get_flash_manufacturer(ha, &man_id, &flash_id); | ||
1433 | switch (man_id) { | ||
1434 | case 0x20: /* ST flash. */ | ||
1435 | if (flash_id == 0xd2 || flash_id == 0xe3) { | ||
1436 | /* | ||
1437 | * ST m29w008at part - 64kb sector size with | ||
1438 | * 32kb,8kb,8kb,16kb sectors at memory address | ||
1439 | * 0xf0000. | ||
1440 | */ | ||
1441 | rest_addr = 0xffff; | ||
1442 | sec_mask = 0x10000; | ||
1443 | break; | ||
1444 | } | ||
1445 | /* | ||
1446 | * ST m29w010b part - 16kb sector size | ||
1447 | * Default to 16kb sectors | ||
1448 | */ | ||
1449 | rest_addr = 0x3fff; | ||
1450 | sec_mask = 0x1c000; | ||
1451 | break; | ||
1452 | case 0x40: /* Mostel flash. */ | ||
1453 | /* Mostel v29c51001 part - 512 byte sector size. */ | ||
1454 | rest_addr = 0x1ff; | ||
1455 | sec_mask = 0x1fe00; | ||
1456 | break; | ||
1457 | case 0xbf: /* SST flash. */ | ||
1458 | /* SST39sf10 part - 4kb sector size. */ | ||
1459 | rest_addr = 0xfff; | ||
1460 | sec_mask = 0x1f000; | ||
1461 | break; | ||
1462 | case 0xda: /* Winbond flash. */ | ||
1463 | /* Winbond W29EE011 part - 256 byte sector size. */ | ||
1464 | rest_addr = 0x7f; | ||
1465 | sec_mask = 0x1ff80; | ||
1466 | break; | ||
1467 | case 0xc2: /* Macronix flash. */ | ||
1468 | /* 64k sector size. */ | ||
1469 | if (flash_id == 0x38 || flash_id == 0x4f) { | ||
1470 | rest_addr = 0xffff; | ||
1471 | sec_mask = 0x10000; | ||
1472 | break; | ||
1473 | } | ||
1474 | /* Fall through... */ | ||
1475 | |||
1476 | case 0x1f: /* Atmel flash. */ | ||
1477 | /* 512k sector size. */ | ||
1478 | if (flash_id == 0x13) { | ||
1479 | rest_addr = 0x7fffffff; | ||
1480 | sec_mask = 0x80000000; | ||
1481 | break; | ||
1482 | } | ||
1483 | /* Fall through... */ | ||
1484 | |||
1485 | case 0x01: /* AMD flash. */ | ||
1486 | if (flash_id == 0x38 || flash_id == 0x40 || | ||
1487 | flash_id == 0x4f) { | ||
1488 | /* Am29LV081 part - 64kb sector size. */ | ||
1489 | /* Am29LV002BT part - 64kb sector size. */ | ||
1490 | rest_addr = 0xffff; | ||
1491 | sec_mask = 0x10000; | ||
1492 | break; | ||
1493 | } else if (flash_id == 0x3e) { | ||
1494 | /* | ||
1495 | * Am29LV008b part - 64kb sector size with | ||
1496 | * 32kb,8kb,8kb,16kb sector at memory address | ||
1497 | * h0xf0000. | ||
1498 | */ | ||
1499 | rest_addr = 0xffff; | ||
1500 | sec_mask = 0x10000; | ||
1501 | break; | ||
1502 | } else if (flash_id == 0x20 || flash_id == 0x6e) { | ||
1503 | /* | ||
1504 | * Am29LV010 part or AM29f010 - 16kb sector | ||
1505 | * size. | ||
1506 | */ | ||
1507 | rest_addr = 0x3fff; | ||
1508 | sec_mask = 0x1c000; | ||
1509 | break; | ||
1510 | } else if (flash_id == 0x6d) { | ||
1511 | /* Am29LV001 part - 8kb sector size. */ | ||
1512 | rest_addr = 0x1fff; | ||
1513 | sec_mask = 0x1e000; | ||
1514 | break; | ||
1515 | } | ||
1516 | default: | ||
1517 | /* Default to 16 kb sector size. */ | ||
1518 | rest_addr = 0x3fff; | ||
1519 | sec_mask = 0x1c000; | ||
1520 | break; | ||
1521 | } | ||
1522 | |||
1523 | update_flash: | ||
1524 | if (IS_QLA2322(ha) || IS_QLA6322(ha)) { | ||
1525 | if (qla2x00_erase_flash(ha, man_id, flash_id)) { | ||
1526 | rval = QLA_FUNCTION_FAILED; | ||
1527 | break; | ||
1528 | } | ||
1529 | } | ||
1530 | |||
1531 | for (addr = offset, liter = 0; liter < length; liter++, | ||
1532 | addr++) { | ||
1533 | data = buf[liter]; | ||
1534 | /* Are we at the beginning of a sector? */ | ||
1535 | if ((addr & rest_addr) == 0) { | ||
1536 | if (IS_QLA2322(ha) || IS_QLA6322(ha)) { | ||
1537 | if (addr >= 0x10000UL) { | ||
1538 | if (((addr >> 12) & 0xf0) && | ||
1539 | ((man_id == 0x01 && | ||
1540 | flash_id == 0x3e) || | ||
1541 | (man_id == 0x20 && | ||
1542 | flash_id == 0xd2))) { | ||
1543 | sec_number++; | ||
1544 | if (sec_number == 1) { | ||
1545 | rest_addr = | ||
1546 | 0x7fff; | ||
1547 | sec_mask = | ||
1548 | 0x18000; | ||
1549 | } else if ( | ||
1550 | sec_number == 2 || | ||
1551 | sec_number == 3) { | ||
1552 | rest_addr = | ||
1553 | 0x1fff; | ||
1554 | sec_mask = | ||
1555 | 0x1e000; | ||
1556 | } else if ( | ||
1557 | sec_number == 4) { | ||
1558 | rest_addr = | ||
1559 | 0x3fff; | ||
1560 | sec_mask = | ||
1561 | 0x1c000; | ||
1562 | } | ||
1563 | } | ||
1564 | } | ||
1565 | } else if (addr == ha->optrom_size / 2) { | ||
1566 | WRT_REG_WORD(®->nvram, NVR_SELECT); | ||
1567 | RD_REG_WORD(®->nvram); | ||
1568 | } | ||
1569 | |||
1570 | if (flash_id == 0xda && man_id == 0xc1) { | ||
1571 | qla2x00_write_flash_byte(ha, 0x5555, | ||
1572 | 0xaa); | ||
1573 | qla2x00_write_flash_byte(ha, 0x2aaa, | ||
1574 | 0x55); | ||
1575 | qla2x00_write_flash_byte(ha, 0x5555, | ||
1576 | 0xa0); | ||
1577 | } else if (!IS_QLA2322(ha) && !IS_QLA6322(ha)) { | ||
1578 | /* Then erase it */ | ||
1579 | if (qla2x00_erase_flash_sector(ha, | ||
1580 | addr, sec_mask, man_id, | ||
1581 | flash_id)) { | ||
1582 | rval = QLA_FUNCTION_FAILED; | ||
1583 | break; | ||
1584 | } | ||
1585 | if (man_id == 0x01 && flash_id == 0x6d) | ||
1586 | sec_number++; | ||
1587 | } | ||
1588 | } | ||
1589 | |||
1590 | if (man_id == 0x01 && flash_id == 0x6d) { | ||
1591 | if (sec_number == 1 && | ||
1592 | addr == (rest_addr - 1)) { | ||
1593 | rest_addr = 0x0fff; | ||
1594 | sec_mask = 0x1f000; | ||
1595 | } else if (sec_number == 3 && (addr & 0x7ffe)) { | ||
1596 | rest_addr = 0x3fff; | ||
1597 | sec_mask = 0x1c000; | ||
1598 | } | ||
1599 | } | ||
1600 | |||
1601 | if (qla2x00_program_flash_address(ha, addr, data, | ||
1602 | man_id, flash_id)) { | ||
1603 | rval = QLA_FUNCTION_FAILED; | ||
1604 | break; | ||
1605 | } | ||
1606 | } | ||
1607 | } while (0); | ||
1608 | qla2x00_flash_disable(ha); | ||
1609 | spin_unlock_irqrestore(&ha->hardware_lock, flags); | ||
1610 | |||
1611 | /* Resume HBA. */ | ||
1612 | qla2x00_resume_hba(ha); | ||
1613 | |||
1614 | return rval; | ||
1615 | } | ||
1616 | |||
1617 | uint8_t * | ||
1618 | qla24xx_read_optrom_data(struct scsi_qla_host *ha, uint8_t *buf, | ||
1619 | uint32_t offset, uint32_t length) | ||
1620 | { | ||
1621 | /* Suspend HBA. */ | ||
1622 | scsi_block_requests(ha->host); | ||
1623 | ha->isp_ops.disable_intrs(ha); | ||
1624 | set_bit(MBX_UPDATE_FLASH_ACTIVE, &ha->mbx_cmd_flags); | ||
1625 | |||
1626 | /* Go with read. */ | ||
1627 | qla24xx_read_flash_data(ha, (uint32_t *)buf, offset >> 2, length >> 2); | ||
1628 | |||
1629 | /* Resume HBA. */ | ||
1630 | clear_bit(MBX_UPDATE_FLASH_ACTIVE, &ha->mbx_cmd_flags); | ||
1631 | ha->isp_ops.enable_intrs(ha); | ||
1632 | scsi_unblock_requests(ha->host); | ||
1633 | |||
1634 | return buf; | ||
1635 | } | ||
1636 | |||
1637 | int | ||
1638 | qla24xx_write_optrom_data(struct scsi_qla_host *ha, uint8_t *buf, | ||
1639 | uint32_t offset, uint32_t length) | ||
1640 | { | ||
1641 | int rval; | ||
1642 | |||
1643 | /* Suspend HBA. */ | ||
1644 | scsi_block_requests(ha->host); | ||
1645 | ha->isp_ops.disable_intrs(ha); | ||
1646 | set_bit(MBX_UPDATE_FLASH_ACTIVE, &ha->mbx_cmd_flags); | ||
1647 | |||
1648 | /* Go with write. */ | ||
1649 | rval = qla24xx_write_flash_data(ha, (uint32_t *)buf, offset >> 2, | ||
1650 | length >> 2); | ||
1651 | |||
1652 | /* Resume HBA -- RISC reset needed. */ | ||
1653 | clear_bit(MBX_UPDATE_FLASH_ACTIVE, &ha->mbx_cmd_flags); | ||
1654 | set_bit(ISP_ABORT_NEEDED, &ha->dpc_flags); | ||
1655 | up(ha->dpc_wait); | ||
1656 | qla2x00_wait_for_hba_online(ha); | ||
1657 | scsi_unblock_requests(ha->host); | ||
1658 | |||
1659 | return rval; | ||
1660 | } | ||