summaryrefslogtreecommitdiff
path: root/sys/dev/ic/mpt.c
diff options
context:
space:
mode:
authorKenneth R Westerback <krw@cvs.openbsd.org>2004-03-20 03:54:17 +0000
committerKenneth R Westerback <krw@cvs.openbsd.org>2004-03-20 03:54:17 +0000
commit1b1d62eac8378ffe9933e2937e55e990078cd4d6 (patch)
tree4d798a3422c6d9b83f18d6fd46f388d1c91e4012 /sys/dev/ic/mpt.c
parentf56163e66caaae31b77570a65b994446117ad454 (diff)
Implement firmware download.
From Milos Urbanek via Marco Peereboom, with suggestions from LSI. ok deraadt@.
Diffstat (limited to 'sys/dev/ic/mpt.c')
-rw-r--r--sys/dev/ic/mpt.c185
1 files changed, 172 insertions, 13 deletions
diff --git a/sys/dev/ic/mpt.c b/sys/dev/ic/mpt.c
index f10a2b91c24..d5efe1cb2d8 100644
--- a/sys/dev/ic/mpt.c
+++ b/sys/dev/ic/mpt.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mpt.c,v 1.3 2004/03/19 02:47:36 krw Exp $ */
+/* $OpenBSD: mpt.c,v 1.4 2004/03/20 03:54:16 krw Exp $ */
/* $NetBSD: mpt.c,v 1.4 2003/11/02 11:07:45 wiz Exp $ */
/*
@@ -181,6 +181,9 @@ mpt_soft_reset(mpt_softc_t *mpt)
void
mpt_hard_reset(mpt_softc_t *mpt)
{
+ u_int32_t diag0;
+ int count;
+
/* This extra read comes for the Linux source
* released by LSI. It's function is undocumented!
*/
@@ -214,9 +217,25 @@ mpt_hard_reset(mpt_softc_t *mpt)
/* Note that if there is no valid firmware to run, the doorbell will
remain in the reset state (0x00000000) */
- if (mpt->download_fw) {
- /* FIXME do download boot we panic for now */
- panic("FWDownloadBoot not implemented yet.");
+
+ /* try to download firmware, if available */
+ if (mpt->fw) {
+ /* wait for the adapter to finish the reset */
+ for (count = 0; count < 30; count++) {
+ diag0 = mpt_read(mpt, MPI_DIAGNOSTIC_OFFSET);
+ mpt_prt(mpt, "diag0=%08x\n", diag0);
+ if (!(diag0 & MPI_DIAG_RESET_ADAPTER)) {
+ break;
+ }
+
+ /* wait 1 second */
+ DELAY(1000);
+ }
+ count = mpt_downloadboot(mpt);
+ if (count < 0) {
+ panic("firmware downloadboot failure (%d)!\n",
+ count);
+ }
}
}
@@ -1149,14 +1168,6 @@ mpt_init(mpt_softc_t *mpt, u_int32_t who)
mpt->fw_download_boot = facts.Flags
& MPI_IOCFACTS_FLAGS_FW_DOWNLOAD_BOOT;
- /* if download boot set download flag */
- if (mpt->fw_download_boot) {
- mpt->download_fw = 1;
- }
- else {
- mpt->download_fw = 0;
- }
-
mpt->fw_image_size = facts.FWImageSize;
if (mpt_get_portfacts(mpt, &pfp) != MPT_OK) {
@@ -1327,7 +1338,7 @@ mpt_do_upload(mpt_softc_t *mpt)
prequest->ImageType = MPI_FW_UPLOAD_ITYPE_FW_IOC_MEM;
prequest->Function = MPI_FUNCTION_FW_UPLOAD;
- prequest->MsgContext = 0; /* XXX MU ok? */
+ prequest->MsgContext = 0;
ptcsge = (FWUploadTCSGE_t *) &prequest->SGL;
ptcsge->Reserved = 0;
@@ -1407,3 +1418,151 @@ mpt_do_upload(mpt_softc_t *mpt)
return error;
}
+/*
+ * mpt_downloadboot - DownloadBoot code
+ * Returns 0 for success, <0 for failure
+ */
+int
+mpt_downloadboot(mpt_softc_t *mpt)
+{
+ MpiFwHeader_t *fwhdr = NULL;
+ MpiExtImageHeader_t *exthdr = NULL;
+ int fw_size;
+ u_int32_t diag0;
+#if MPT_DEBUG
+ u_int32_t diag1;
+#endif
+ int count = 0;
+ u_int32_t *ptr = NULL;
+ u_int32_t nextimg;
+ u_int32_t load_addr;
+ u_int32_t diagrw_data;
+
+#ifdef MPT_DEBUG
+ diag0 = mpt_read(mpt, MPT_OFFSET_DIAGNOSTIC);
+ if (mpt->mpt2)
+ diag1 = mpt_read(mpt->mpt2, MPT_OFFSET_DIAGNOSTIC);
+ mpt_prt(mpt, "diag0=%08x, diag1=%08x\n", diag0, diag1);
+#endif
+ mpt_prt(mpt, "fw size 0x%x, ioc FW ptr %p\n",
+ mpt->fw_image_size, mpt->fw);
+ if (mpt->mpt2)
+ mpt_prt(mpt->mpt2, "ioc FW ptr %p\n", mpt->mpt2->fw);
+
+ fw_size = mpt->fw_image_size;
+
+ if (fw_size == 0)
+ return -1;
+
+ mpt_prt(mpt, "FW Image @ %p\n", mpt->fw);
+
+ if (!mpt->fw)
+ return -2;
+
+ /*
+ * Write magic sequence to WriteSequence register
+ * until enter diagnostic mode
+ */
+ diag0 = mpt_read(mpt, MPT_OFFSET_DIAGNOSTIC);
+ while ((diag0 & MPI_DIAG_DRWE) == 0) {
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, 0xFF);
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, MPT_DIAG_SEQUENCE_1);
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, MPT_DIAG_SEQUENCE_2);
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, MPT_DIAG_SEQUENCE_3);
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, MPT_DIAG_SEQUENCE_4);
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, MPT_DIAG_SEQUENCE_5);
+
+ /* wait 100msec */
+ DELAY(100);
+
+ count++;
+ if (count > 20) {
+ mpt_prt(mpt, "enable diagnostic mode FAILED! (%02xh)\n",
+ diag0);
+ return -EFAULT;
+ }
+
+ diag0 = mpt_read(mpt, MPT_OFFSET_DIAGNOSTIC);
+#ifdef MPT_DEBUG
+ if (mpt->mpt2)
+ diag1 = mpt_read(mpt->mpt2, MPT_OFFSET_DIAGNOSTIC);
+ mpt_prt(mpt, "diag0=%08x, diag1=%08x\n", diag0, diag1);
+#endif
+ mpt_prt(mpt, "wrote magic DiagWriteEn sequence (%x)\n", diag0);
+ }
+
+ /* Set the DiagRwEn and Disable ARM bits */
+ diag0 |= (MPI_DIAG_RW_ENABLE | MPI_DIAG_DISABLE_ARM);
+ mpt_write(mpt, MPT_OFFSET_DIAGNOSTIC, diag0);
+
+#ifdef MPT_DEBUG
+ if (mpt->mpt2)
+ diag1 = mpt_read(mpt->mpt2, MPT_OFFSET_DIAGNOSTIC);
+ mpt_prt(mpt, "diag0=%08x, diag1=%08x\n", diag0, diag1);
+#endif
+
+ fwhdr = (MpiFwHeader_t *) mpt->fw;
+ ptr = (u_int32_t *) fwhdr;
+ count = (fwhdr->ImageSize + 3)/4;
+ nextimg = fwhdr->NextImageHeaderOffset;
+
+ /*
+ * write the LoadStartAddress to the DiagRw Address Register
+ * XXX linux is using programmed IO for the RWADDR and RWDATA
+ */
+ mpt_write(mpt, MPT_OFFSET_RWADDR, fwhdr->LoadStartAddress);
+
+ mpt_prt(mpt, "LoadStart addr written 0x%x \n", fwhdr->LoadStartAddress);
+ mpt_prt(mpt, "writing file image: 0x%x u32's @ %p\n", count, ptr);
+
+ while (count--) {
+ mpt_write(mpt, MPT_OFFSET_RWDATA, *ptr);
+ ptr++;
+ }
+
+ while (nextimg) {
+ ptr = (u_int32_t *) (mpt->fw + nextimg);
+ exthdr = (MpiExtImageHeader_t *) ptr;
+ count = (exthdr->ImageSize +3)/4;
+ nextimg = exthdr->NextImageHeaderOffset;
+ load_addr = exthdr->LoadStartAddress;
+
+ mpt_prt(mpt, "write ext image: 0x%x u32's @ %p\n", count, ptr);
+
+ mpt_write(mpt, MPT_OFFSET_RWADDR, load_addr);
+
+ while (count--) {
+ mpt_write(mpt, MPT_OFFSET_RWDATA, *ptr);
+ ptr++;
+ }
+ }
+
+ /* write the IopResetVectorRegAddr */
+ mpt_prt(mpt, "write IopResetVector addr!\n");
+ mpt_write(mpt, MPT_OFFSET_RWADDR, fwhdr->IopResetRegAddr);
+
+ /* write the IopResetVectorValue */
+ mpt_prt(mpt, "write IopResetVector value!\n");
+ mpt_write(mpt, MPT_OFFSET_RWDATA, fwhdr->IopResetVectorValue);
+
+ /*
+ * clear the internal flash bad bit - autoincrementing register,
+ * so must do two writes.
+ */
+ mpt_write(mpt, MPT_OFFSET_RWADDR, 0x3F000000);
+ diagrw_data = mpt_read(mpt, MPT_OFFSET_RWDATA);
+ diagrw_data |= 0x4000000;
+ mpt_write(mpt, MPT_OFFSET_RWADDR, 0x3F000000);
+ mpt_write(mpt, MPT_OFFSET_RWDATA, diagrw_data);
+
+ /* clear the RW enable and DISARM bits */
+ diag0 = mpt_read(mpt, MPT_OFFSET_DIAGNOSTIC);
+ diag0 &= ~(MPI_DIAG_DISABLE_ARM | MPI_DIAG_RW_ENABLE
+ | MPI_DIAG_FLASH_BAD_SIG);
+ mpt_write(mpt, MPT_OFFSET_DIAGNOSTIC, diag0);
+
+ /* write 0xFF to reset the sequencer */
+ mpt_write(mpt, MPT_OFFSET_SEQUENCE, 0xFF);
+
+ return 0;
+}