Justin M. Forbes 696046
diff --git a/Makefile b/Makefile
Justin M. Forbes 696046
index 7d511e8..fb9bfaf 100644
Justin M. Forbes 696046
--- a/Makefile
Justin M. Forbes 696046
+++ b/Makefile
Justin M. Forbes 696046
@@ -12,10 +12,10 @@ OUT=out/
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 # Source files
Justin M. Forbes 696046
 SRCBOTH=misc.c stacks.c pmm.c output.c util.c block.c floppy.c ata.c mouse.c \
Justin M. Forbes 696046
-        kbd.c pci.c serial.c clock.c pic.c cdrom.c ps2port.c smp.c resume.c \
Justin M. Forbes 696046
-        pnpbios.c pirtable.c vgahooks.c ramdisk.c pcibios.c blockcmd.c \
Justin M. Forbes 696046
-        usb.c usb-uhci.c usb-ohci.c usb-ehci.c usb-hid.c usb-msc.c \
Justin M. Forbes 696046
-        virtio-ring.c virtio-pci.c virtio-blk.c apm.c ahci.c
Justin M. Forbes 696046
+    kbd.c pci.c serial.c clock.c pic.c cdrom.c ps2port.c smp.c resume.c \
Justin M. Forbes 696046
+    pnpbios.c pirtable.c vgahooks.c ramdisk.c pcibios.c blockcmd.c \
Justin M. Forbes 696046
+    usb.c usb-uhci.c usb-ohci.c usb-ehci.c usb-hid.c usb-msc.c \
Justin M. Forbes 696046
+    virtio-ring.c virtio-pci.c virtio-blk.c virtio-scsi.c apm.c ahci.c
Justin M. Forbes 696046
 SRC16=$(SRCBOTH) system.c disk.c font.c
Justin M. Forbes 696046
 SRC32FLAT=$(SRCBOTH) post.c shadow.c memmap.c coreboot.c boot.c \
Justin M. Forbes 696046
       acpi.c smm.c mptable.c smbios.c pciinit.c optionroms.c mtrr.c \
Justin M. Forbes 696046
diff --git a/src/Kconfig b/src/Kconfig
Justin M. Forbes 696046
index 338f51a..8de3503 100644
Justin M. Forbes 696046
--- a/src/Kconfig
Justin M. Forbes 696046
+++ b/src/Kconfig
Justin M. Forbes 696046
@@ -109,10 +109,16 @@ menu "Hardware support"
Justin M. Forbes 696046
             Support for AHCI disk code.
Justin M. Forbes 696046
     config VIRTIO_BLK
Justin M. Forbes 696046
         depends on DRIVES && !COREBOOT
Justin M. Forbes 696046
-        bool "VirtIO controllers"
Justin M. Forbes 696046
+        bool "virtio-blk controllers"
Justin M. Forbes 696046
         default y
Justin M. Forbes 696046
         help
Justin M. Forbes 696046
-            Support boot from virtio storage.
Justin M. Forbes 696046
+            Support boot from virtio-blk storage.
Justin M. Forbes 696046
+    config VIRTIO_SCSI
Justin M. Forbes 696046
+        depends on DRIVES && !COREBOOT
Justin M. Forbes 696046
+        bool "virtio-scsi controllers"
Justin M. Forbes 696046
+        default y
Justin M. Forbes 696046
+        help
Justin M. Forbes 696046
+            Support boot from virtio-scsi storage.
Justin M. Forbes 696046
     config FLOPPY
Justin M. Forbes 696046
         depends on DRIVES
Justin M. Forbes 696046
         bool "Floppy controller"
Justin M. Forbes 696046
diff --git a/src/block.c b/src/block.c
Justin M. Forbes 696046
index f7e7851..e607d67 100644
Justin M. Forbes 696046
--- a/src/block.c
Justin M. Forbes 696046
+++ b/src/block.c
Justin M. Forbes 696046
@@ -11,8 +11,8 @@
Justin M. Forbes 696046
 #include "util.h" // dprintf
Justin M. Forbes 696046
 #include "ata.h" // process_ata_op
Justin M. Forbes 696046
 #include "ahci.h" // process_ahci_op
Justin M. Forbes 696046
-#include "usb-msc.h" // process_usb_op
Justin M. Forbes 696046
-#include "virtio-blk.h" // process_virtio_op
Justin M. Forbes 696046
+#include "virtio-blk.h" // process_virtio_blk_op
Justin M. Forbes 696046
+#include "blockcmd.h" // cdb_*
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 u8 FloppyCount VAR16VISIBLE;
Justin M. Forbes 696046
 u8 CDCount;
Justin M. Forbes 696046
@@ -276,6 +276,28 @@ map_floppy_drive(struct drive_s *drive_g)
Justin M. Forbes 696046
  * 16bit calling interface
Justin M. Forbes 696046
  ****************************************************************/
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
+process_scsi_op(struct disk_op_s *op)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    if (!CONFIG_VIRTIO_SCSI && !CONFIG_USB_MSC)
Justin M. Forbes 696046
+        return 0;
Justin M. Forbes 696046
+    switch (op->command) {
Justin M. Forbes 696046
+    case CMD_READ:
Justin M. Forbes 696046
+        return cdb_read(op);
Justin M. Forbes 696046
+    case CMD_WRITE:
Justin M. Forbes 696046
+        return cdb_write(op);
Justin M. Forbes 696046
+    case CMD_FORMAT:
Justin M. Forbes 696046
+    case CMD_RESET:
Justin M. Forbes 696046
+    case CMD_ISREADY:
Justin M. Forbes 696046
+    case CMD_VERIFY:
Justin M. Forbes 696046
+    case CMD_SEEK:
Justin M. Forbes 696046
+        return DISK_RET_SUCCESS;
Justin M. Forbes 696046
+    default:
Justin M. Forbes 696046
+        op->count = 0;
Justin M. Forbes 696046
+        return DISK_RET_EPARAM;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 // Execute a disk_op request.
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
 process_op(struct disk_op_s *op)
Justin M. Forbes 696046
@@ -293,12 +315,13 @@ process_op(struct disk_op_s *op)
Justin M. Forbes 696046
         return process_ramdisk_op(op);
Justin M. Forbes 696046
     case DTYPE_CDEMU:
Justin M. Forbes 696046
         return process_cdemu_op(op);
Justin M. Forbes 696046
-    case DTYPE_USB:
Justin M. Forbes 696046
-        return process_usb_op(op);
Justin M. Forbes 696046
-    case DTYPE_VIRTIO:
Justin M. Forbes 696046
-	return process_virtio_op(op);
Justin M. Forbes 696046
+    case DTYPE_VIRTIO_BLK:
Justin M. Forbes 696046
+        return process_virtio_blk_op(op);
Justin M. Forbes 696046
     case DTYPE_AHCI:
Justin M. Forbes 696046
 	return process_ahci_op(op);
Justin M. Forbes 696046
+    case DTYPE_USB:
Justin M. Forbes 696046
+    case DTYPE_VIRTIO_SCSI:
Justin M. Forbes 696046
+        return process_scsi_op(op);
Justin M. Forbes 696046
     default:
Justin M. Forbes 696046
         op->count = 0;
Justin M. Forbes 696046
         return DISK_RET_EPARAM;
Justin M. Forbes 696046
diff --git a/src/blockcmd.c b/src/blockcmd.c
Justin M. Forbes 696046
index c9c6845..b2a8d71 100644
Justin M. Forbes 696046
--- a/src/blockcmd.c
Justin M. Forbes 696046
+++ b/src/blockcmd.c
Justin M. Forbes 696046
@@ -12,6 +12,7 @@
Justin M. Forbes 696046
 #include "ata.h" // atapi_cmd_data
Justin M. Forbes 696046
 #include "ahci.h" // atapi_cmd_data
Justin M. Forbes 696046
 #include "usb-msc.h" // usb_cmd_data
Justin M. Forbes 696046
+#include "virtio-scsi.h" // virtio_scsi_cmd_data
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 // Route command to low-level handler.
Justin M. Forbes 696046
 static int
Justin M. Forbes 696046
@@ -25,6 +26,8 @@ cdb_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize)
Justin M. Forbes 696046
         return usb_cmd_data(op, cdbcmd, blocksize);
Justin M. Forbes 696046
     case DTYPE_AHCI:
Justin M. Forbes 696046
         return ahci_cmd_data(op, cdbcmd, blocksize);
Justin M. Forbes 696046
+    case DTYPE_VIRTIO_SCSI:
Justin M. Forbes 696046
+        return virtio_scsi_cmd_data(op, cdbcmd, blocksize);
Justin M. Forbes 696046
     default:
Justin M. Forbes 696046
         op->count = 0;
Justin M. Forbes 696046
         return DISK_RET_EPARAM;
Justin M. Forbes 696046
@@ -32,6 +35,128 @@ cdb_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize)
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
+scsi_is_ready(struct disk_op_s *op)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    dprintf(6, "scsi_is_ready (drive=%p)\n", op->drive_g);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    /* Retry TEST UNIT READY for 5 seconds unless MEDIUM NOT PRESENT is
Justin M. Forbes 696046
+     * reported by the device.  If the device reports "IN PROGRESS",
Justin M. Forbes 696046
+     * 30 seconds is added. */
Justin M. Forbes 696046
+    int in_progress = 0;
Justin M. Forbes 696046
+    u64 end = calc_future_tsc(5000);
Justin M. Forbes 696046
+    for (;;) {
Justin M. Forbes 696046
+        if (check_tsc(end)) {
Justin M. Forbes 696046
+            dprintf(1, "test unit ready failed\n");
Justin M. Forbes 696046
+            return -1;
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+        int ret = cdb_test_unit_ready(op);
Justin M. Forbes 696046
+        if (!ret)
Justin M. Forbes 696046
+            // Success
Justin M. Forbes 696046
+            break;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+        struct cdbres_request_sense sense;
Justin M. Forbes 696046
+        ret = cdb_get_sense(op, &sense);
Justin M. Forbes 696046
+        if (ret)
Justin M. Forbes 696046
+            // Error - retry.
Justin M. Forbes 696046
+            continue;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+        // Sense succeeded.
Justin M. Forbes 696046
+        if (sense.asc == 0x3a) { /* MEDIUM NOT PRESENT */
Justin M. Forbes 696046
+            dprintf(1, "Device reports MEDIUM NOT PRESENT\n");
Justin M. Forbes 696046
+            return -1;
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+        if (sense.asc == 0x04 && sense.ascq == 0x01 && !in_progress) {
Justin M. Forbes 696046
+            /* IN PROGRESS OF BECOMING READY */
Justin M. Forbes 696046
+            printf("Waiting for device to detect medium... ");
Justin M. Forbes 696046
+            /* Allow 30 seconds more */
Justin M. Forbes 696046
+            end = calc_future_tsc(30000);
Justin M. Forbes 696046
+            in_progress = 1;
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+    return 0;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+// Validate drive and find block size and sector count.
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
+scsi_init_drive(struct drive_s *drive, const char *s, int *pdt, char **desc)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    if (!CONFIG_USB_MSC && !CONFIG_VIRTIO_SCSI)
Justin M. Forbes 696046
+        return 0;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    struct disk_op_s dop;
Justin M. Forbes 696046
+    memset(&dop, 0, sizeof(dop));
Justin M. Forbes 696046
+    dop.drive_g = drive;
Justin M. Forbes 696046
+    struct cdbres_inquiry data;
Justin M. Forbes 696046
+    int ret = cdb_get_inquiry(&dop, &data);
Justin M. Forbes 696046
+    if (ret)
Justin M. Forbes 696046
+        return ret;
Justin M. Forbes 696046
+    char vendor[sizeof(data.vendor)+1], product[sizeof(data.product)+1];
Justin M. Forbes 696046
+    char rev[sizeof(data.rev)+1];
Justin M. Forbes 696046
+    strtcpy(vendor, data.vendor, sizeof(vendor));
Justin M. Forbes 696046
+    nullTrailingSpace(vendor);
Justin M. Forbes 696046
+    strtcpy(product, data.product, sizeof(product));
Justin M. Forbes 696046
+    nullTrailingSpace(product);
Justin M. Forbes 696046
+    strtcpy(rev, data.rev, sizeof(rev));
Justin M. Forbes 696046
+    nullTrailingSpace(rev);
Justin M. Forbes 696046
+    *pdt = data.pdt & 0x1f;
Justin M. Forbes 696046
+    int removable = !!(data.removable & 0x80);
Justin M. Forbes 696046
+    dprintf(1, "%s vendor='%s' product='%s' rev='%s' type=%d removable=%d\n"
Justin M. Forbes 696046
+            , s, vendor, product, rev, *pdt, removable);
Justin M. Forbes 696046
+    drive->removable = removable;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    if (*pdt == SCSI_TYPE_CDROM) {
Justin M. Forbes 696046
+        drive->blksize = CDROM_SECTOR_SIZE;
Justin M. Forbes 696046
+        drive->sectors = (u64)-1;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+        *desc = znprintf(MAXDESCSIZE, "DVD/CD [%s Drive %s %s %s]"
Justin M. Forbes 696046
+                              , s, vendor, product, rev);
Justin M. Forbes 696046
+        return 0;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    ret = scsi_is_ready(&dop);
Justin M. Forbes 696046
+    if (ret) {
Justin M. Forbes 696046
+        dprintf(1, "scsi_is_ready returned %d\n", ret);
Justin M. Forbes 696046
+        return ret;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    struct cdbres_read_capacity capdata;
Justin M. Forbes 696046
+    ret = cdb_read_capacity(&dop, &capdata);
Justin M. Forbes 696046
+    if (ret)
Justin M. Forbes 696046
+        return ret;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    // READ CAPACITY returns the address of the last block.
Justin M. Forbes 696046
+    // We do not bother with READ CAPACITY(16) because BIOS does not support
Justin M. Forbes 696046
+    // 64-bit LBA anyway.
Justin M. Forbes 696046
+    drive->blksize = ntohl(capdata.blksize);
Justin M. Forbes 696046
+    drive->sectors = (u64)ntohl(capdata.sectors) + 1;
Justin M. Forbes 696046
+    dprintf(1, "%s blksize=%d sectors=%d\n"
Justin M. Forbes 696046
+            , s, drive->blksize, (unsigned)drive->sectors);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    struct cdbres_mode_sense_geom geomdata;
Justin M. Forbes 696046
+    ret = cdb_mode_sense_geom(&dop, &geomdata);
Justin M. Forbes 696046
+    if (ret == 0) {
Justin M. Forbes 696046
+        u32 cylinders;
Justin M. Forbes 696046
+        cylinders = geomdata.cyl[0] << 16;
Justin M. Forbes 696046
+        cylinders |= geomdata.cyl[1] << 8;
Justin M. Forbes 696046
+        cylinders |= geomdata.cyl[2];
Justin M. Forbes 696046
+        if (cylinders && geomdata.heads &&
Justin M. Forbes 696046
+            drive->sectors <= 0xFFFFFFFFULL &&
Justin M. Forbes 696046
+            ((u32)drive->sectors % (geomdata.heads * cylinders) == 0)) {
Justin M. Forbes 696046
+            drive->pchs.cylinders = cylinders;
Justin M. Forbes 696046
+            drive->pchs.heads = geomdata.heads;
Justin M. Forbes 696046
+            drive->pchs.spt = (u32)drive->sectors
Justin M. Forbes 696046
+     / (geomdata.heads * cylinders);
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    *desc = znprintf(MAXDESCSIZE, "%s Drive %s %s %s"
Justin M. Forbes 696046
+                      , s, vendor, product, rev);
Justin M. Forbes 696046
+    return 0;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
 cdb_get_inquiry(struct disk_op_s *op, struct cdbres_inquiry *data)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
     struct cdb_request_sense cmd;
Justin M. Forbes 696046
@@ -56,6 +181,18 @@ cdb_get_sense(struct disk_op_s *op, struct cdbres_request_sense *data)
Justin M. Forbes 696046
     return cdb_cmd_data(op, &cmd, sizeof(*data));
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+// Test unit ready
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
+cdb_test_unit_ready(struct disk_op_s *op)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct cdb_request_sense cmd;
Justin M. Forbes 696046
+    memset(&cmd, 0, sizeof(cmd));
Justin M. Forbes 696046
+    cmd.command = CDB_CMD_TEST_UNIT_READY;
Justin M. Forbes 696046
+    op->count = 0;
Justin M. Forbes 696046
+    op->buf_fl = NULL;
Justin M. Forbes 696046
+    return cdb_cmd_data(op, &cmd, 0);
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 // Request capacity
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
 cdb_read_capacity(struct disk_op_s *op, struct cdbres_read_capacity *data)
Justin M. Forbes 696046
@@ -68,6 +205,21 @@ cdb_read_capacity(struct disk_op_s *op, struct cdbres_read_capacity *data)
Justin M. Forbes 696046
     return cdb_cmd_data(op, &cmd, sizeof(*data));
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+// Mode sense, geometry page.
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
+cdb_mode_sense_geom(struct disk_op_s *op, struct cdbres_mode_sense_geom *data)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct cdb_mode_sense cmd;
Justin M. Forbes 696046
+    memset(&cmd, 0, sizeof(cmd));
Justin M. Forbes 696046
+    cmd.command = CDB_CMD_MODE_SENSE;
Justin M. Forbes 696046
+    cmd.flags = 8; /* DBD */
Justin M. Forbes 696046
+    cmd.page = MODE_PAGE_HD_GEOMETRY;
Justin M. Forbes 696046
+    cmd.count = htons(sizeof(*data));
Justin M. Forbes 696046
+    op->count = 1;
Justin M. Forbes 696046
+    op->buf_fl = data;
Justin M. Forbes 696046
+    return cdb_cmd_data(op, &cmd, sizeof(*data));
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 // Read sectors.
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
 cdb_read(struct disk_op_s *op)
Justin M. Forbes 696046
@@ -79,3 +231,15 @@ cdb_read(struct disk_op_s *op)
Justin M. Forbes 696046
     cmd.count = htons(op->count);
Justin M. Forbes 696046
     return cdb_cmd_data(op, &cmd, GET_GLOBAL(op->drive_g->blksize));
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+// Write sectors.
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
+cdb_write(struct disk_op_s *op)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct cdb_rwdata_10 cmd;
Justin M. Forbes 696046
+    memset(&cmd, 0, sizeof(cmd));
Justin M. Forbes 696046
+    cmd.command = CDB_CMD_WRITE_10;
Justin M. Forbes 696046
+    cmd.lba = htonl(op->lba);
Justin M. Forbes 696046
+    cmd.count = htons(op->count);
Justin M. Forbes 696046
+    return cdb_cmd_data(op, &cmd, GET_GLOBAL(op->drive_g->blksize));
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
diff --git a/src/blockcmd.h b/src/blockcmd.h
Justin M. Forbes 696046
index 903c435..bace649 100644
Justin M. Forbes 696046
--- a/src/blockcmd.h
Justin M. Forbes 696046
+++ b/src/blockcmd.h
Justin M. Forbes 696046
@@ -32,8 +32,9 @@ struct cdbres_read_capacity {
Justin M. Forbes 696046
     u32 blksize;
Justin M. Forbes 696046
 } PACKED;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-#define CDB_CMD_INQUIRY 0x12
Justin M. Forbes 696046
-#define CDB_CMD_REQUEST_SENSE 0x03
Justin M. Forbes 696046
+#define CDB_CMD_TEST_UNIT_READY  0x00
Justin M. Forbes 696046
+#define CDB_CMD_INQUIRY          0x12
Justin M. Forbes 696046
+#define CDB_CMD_REQUEST_SENSE    0x03
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 struct cdb_request_sense {
Justin M. Forbes 696046
     u8 command;
Justin M. Forbes 696046
@@ -56,6 +57,9 @@ struct cdbres_request_sense {
Justin M. Forbes 696046
     u32 reserved_0e;
Justin M. Forbes 696046
 } PACKED;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+#define SCSI_TYPE_DISK  0x00
Justin M. Forbes 696046
+#define SCSI_TYPE_CDROM 0x05
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 struct cdbres_inquiry {
Justin M. Forbes 696046
     u8 pdt;
Justin M. Forbes 696046
     u8 removable;
Justin M. Forbes 696046
@@ -67,11 +71,45 @@ struct cdbres_inquiry {
Justin M. Forbes 696046
     char rev[4];
Justin M. Forbes 696046
 } PACKED;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+#define CDB_CMD_MODE_SENSE    0x5A
Justin M. Forbes 696046
+#define MODE_PAGE_HD_GEOMETRY 0x04
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+struct cdb_mode_sense {
Justin M. Forbes 696046
+    u8 command;
Justin M. Forbes 696046
+    u8 flags;
Justin M. Forbes 696046
+    u8 page;
Justin M. Forbes 696046
+    u32 reserved_03;
Justin M. Forbes 696046
+    u16 count;
Justin M. Forbes 696046
+    u8 reserved_09;
Justin M. Forbes 696046
+    u8 pad[6];
Justin M. Forbes 696046
+} PACKED;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+struct cdbres_mode_sense_geom {
Justin M. Forbes 696046
+    u8 unused_00[3];
Justin M. Forbes 696046
+    u8 read_only;
Justin M. Forbes 696046
+    u32 unused_04;
Justin M. Forbes 696046
+    u8 page;
Justin M. Forbes 696046
+    u8 length;
Justin M. Forbes 696046
+    u8 cyl[3];
Justin M. Forbes 696046
+    u8 heads;
Justin M. Forbes 696046
+    u8 precomp[3];
Justin M. Forbes 696046
+    u8 reduced[3];
Justin M. Forbes 696046
+    u16 step_rate;
Justin M. Forbes 696046
+    u8 landing[3];
Justin M. Forbes 696046
+    u16 rpm;
Justin M. Forbes 696046
+} PACKED;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 // blockcmd.c
Justin M. Forbes 696046
 int cdb_get_inquiry(struct disk_op_s *op, struct cdbres_inquiry *data);
Justin M. Forbes 696046
 int cdb_get_sense(struct disk_op_s *op, struct cdbres_request_sense *data);
Justin M. Forbes 696046
+int cdb_test_unit_ready(struct disk_op_s *op);
Justin M. Forbes 696046
 int cdb_read_capacity(struct disk_op_s *op, struct cdbres_read_capacity *data);
Justin M. Forbes 696046
+int cdb_mode_sense_geom(struct disk_op_s *op, struct cdbres_mode_sense_geom *data);
Justin M. Forbes 696046
 int cdb_inquiry(struct disk_op_s *op, struct cdbres_inquiry *data);
Justin M. Forbes 696046
 int cdb_read(struct disk_op_s *op);
Justin M. Forbes 696046
+int cdb_write(struct disk_op_s *op);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+int scsi_is_ready(struct disk_op_s *op);
Justin M. Forbes 696046
+int scsi_init_drive(struct drive_s *drive, const char *s, int *pdt, char **desc);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 #endif // blockcmd.h
Justin M. Forbes 696046
diff --git a/src/boot.c b/src/boot.c
Justin M. Forbes 696046
index 93928d3..c0991cd 100644
Justin M. Forbes 696046
--- a/src/boot.c
Justin M. Forbes 696046
+++ b/src/boot.c
Justin M. Forbes 696046
@@ -128,6 +128,20 @@ int bootprio_find_pci_device(struct pci_device *pci)
Justin M. Forbes 696046
     return find_prio(desc);
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+int bootprio_find_scsi_device(struct pci_device *pci, int target, int lun)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    if (!CONFIG_BOOTORDER)
Justin M. Forbes 696046
+        return -1;
Justin M. Forbes 696046
+    if (!pci)
Justin M. Forbes 696046
+        // support only pci machine for now
Justin M. Forbes 696046
+        return -1;
Justin M. Forbes 696046
+    // Find scsi drive - for example: /pci@i0cf8/scsi@5/channel@0/disk@1,0
Justin M. Forbes 696046
+    char desc[256], *p;
Justin M. Forbes 696046
+    p = build_pci_path(desc, sizeof(desc), "*", pci);
Justin M. Forbes 696046
+    snprintf(p, desc+sizeof(desc)-p, "/*@0/*@%d,%d", target, lun);
Justin M. Forbes 696046
+    return find_prio(desc);
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 int bootprio_find_ata_device(struct pci_device *pci, int chanid, int slave)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
     if (!CONFIG_BOOTORDER)
Justin M. Forbes 696046
diff --git a/src/boot.h b/src/boot.h
Justin M. Forbes 696046
index d776aa1..686f04d 100644
Justin M. Forbes 696046
--- a/src/boot.h
Justin M. Forbes 696046
+++ b/src/boot.h
Justin M. Forbes 696046
@@ -14,6 +14,7 @@ void boot_add_cbfs(void *data, const char *desc, int prio);
Justin M. Forbes 696046
 void boot_prep(void);
Justin M. Forbes 696046
 struct pci_device;
Justin M. Forbes 696046
 int bootprio_find_pci_device(struct pci_device *pci);
Justin M. Forbes 696046
+int bootprio_find_scsi_device(struct pci_device *pci, int target, int lun);
Justin M. Forbes 696046
 int bootprio_find_ata_device(struct pci_device *pci, int chanid, int slave);
Justin M. Forbes 696046
 int bootprio_find_fdc_device(struct pci_device *pci, int port, int fdid);
Justin M. Forbes 696046
 int bootprio_find_pci_rom(struct pci_device *pci, int instance);
Justin M. Forbes 696046
diff --git a/src/cdrom.c b/src/cdrom.c
Justin M. Forbes 696046
index 6351fec..170ffc4 100644
Justin M. Forbes 696046
--- a/src/cdrom.c
Justin M. Forbes 696046
+++ b/src/cdrom.c
Justin M. Forbes 696046
@@ -184,60 +184,6 @@ cdemu_134b(struct bregs *regs)
Justin M. Forbes 696046
  * CD booting
Justin M. Forbes 696046
  ****************************************************************/
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-static int
Justin M. Forbes 696046
-atapi_is_ready(struct disk_op_s *op)
Justin M. Forbes 696046
-{
Justin M. Forbes 696046
-    dprintf(6, "atapi_is_ready (drive=%p)\n", op->drive_g);
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    /* Retry READ CAPACITY for 5 seconds unless MEDIUM NOT PRESENT is
Justin M. Forbes 696046
-     * reported by the device.  If the device reports "IN PROGRESS",
Justin M. Forbes 696046
-     * 30 seconds is added. */
Justin M. Forbes 696046
-    struct cdbres_read_capacity info;
Justin M. Forbes 696046
-    int in_progress = 0;
Justin M. Forbes 696046
-    u64 end = calc_future_tsc(5000);
Justin M. Forbes 696046
-    for (;;) {
Justin M. Forbes 696046
-        if (check_tsc(end)) {
Justin M. Forbes 696046
-            dprintf(1, "read capacity failed\n");
Justin M. Forbes 696046
-            return -1;
Justin M. Forbes 696046
-        }
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-        int ret = cdb_read_capacity(op, &info;;
Justin M. Forbes 696046
-        if (!ret)
Justin M. Forbes 696046
-            // Success
Justin M. Forbes 696046
-            break;
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-        struct cdbres_request_sense sense;
Justin M. Forbes 696046
-        ret = cdb_get_sense(op, &sense);
Justin M. Forbes 696046
-        if (ret)
Justin M. Forbes 696046
-            // Error - retry.
Justin M. Forbes 696046
-            continue;
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-        // Sense succeeded.
Justin M. Forbes 696046
-        if (sense.asc == 0x3a) { /* MEDIUM NOT PRESENT */
Justin M. Forbes 696046
-            dprintf(1, "Device reports MEDIUM NOT PRESENT\n");
Justin M. Forbes 696046
-            return -1;
Justin M. Forbes 696046
-        }
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-        if (sense.asc == 0x04 && sense.ascq == 0x01 && !in_progress) {
Justin M. Forbes 696046
-            /* IN PROGRESS OF BECOMING READY */
Justin M. Forbes 696046
-            printf("Waiting for device to detect medium... ");
Justin M. Forbes 696046
-            /* Allow 30 seconds more */
Justin M. Forbes 696046
-            end = calc_future_tsc(30000);
Justin M. Forbes 696046
-            in_progress = 1;
Justin M. Forbes 696046
-        }
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    u32 blksize = ntohl(info.blksize), sectors = ntohl(info.sectors);
Justin M. Forbes 696046
-    if (blksize != GET_GLOBAL(op->drive_g->blksize)) {
Justin M. Forbes 696046
-        printf("Unsupported sector size %u\n", blksize);
Justin M. Forbes 696046
-        return -1;
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    dprintf(6, "sectors=%u\n", sectors);
Justin M. Forbes 696046
-    printf("%dMB medium detected\n", sectors>>(20-11));
Justin M. Forbes 696046
-    return 0;
Justin M. Forbes 696046
-}
Justin M. Forbes 696046
-
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
 cdrom_boot(struct drive_s *drive_g)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
@@ -248,9 +194,9 @@ cdrom_boot(struct drive_s *drive_g)
Justin M. Forbes 696046
     if (!dop.drive_g || cdid < 0)
Justin M. Forbes 696046
         return 1;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-    int ret = atapi_is_ready(&dop);
Justin M. Forbes 696046
+    int ret = scsi_is_ready(&dop);
Justin M. Forbes 696046
     if (ret)
Justin M. Forbes 696046
-        dprintf(1, "atapi_is_ready returned %d\n", ret);
Justin M. Forbes 696046
+        dprintf(1, "scsi_is_ready returned %d\n", ret);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Read the Boot Record Volume Descriptor
Justin M. Forbes 696046
     u8 buffer[2048];
Justin M. Forbes 696046
diff --git a/src/disk.c b/src/disk.c
Justin M. Forbes 696046
index 8f7c61f..6a170fd 100644
Justin M. Forbes 696046
--- a/src/disk.c
Justin M. Forbes 696046
+++ b/src/disk.c
Justin M. Forbes 696046
@@ -546,7 +546,8 @@ disk_1348(struct bregs *regs, struct drive_s *drive_g)
Justin M. Forbes 696046
     SET_INT13DPT(regs, blksize, blksize);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     if (size < 30 ||
Justin M. Forbes 696046
-        (type != DTYPE_ATA && type != DTYPE_ATAPI && type != DTYPE_VIRTIO)) {
Justin M. Forbes 696046
+        (type != DTYPE_ATA && type != DTYPE_ATAPI &&
Justin M. Forbes 696046
+         type != DTYPE_VIRTIO_BLK && type != DTYPE_VIRTIO_SCSI)) {
Justin M. Forbes 696046
         disk_ret(regs, DISK_RET_SUCCESS);
Justin M. Forbes 696046
         return;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
@@ -651,7 +652,7 @@ disk_1348(struct bregs *regs, struct drive_s *drive_g)
Justin M. Forbes 696046
         SET_INT13DPT(regs, iface_path, iobase1);
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-    if (type != DTYPE_VIRTIO) {
Justin M. Forbes 696046
+    if (type != DTYPE_VIRTIO_BLK) {
Justin M. Forbes 696046
         SET_INT13DPT(regs, iface_type[0], 'A');
Justin M. Forbes 696046
         SET_INT13DPT(regs, iface_type[1], 'T');
Justin M. Forbes 696046
         SET_INT13DPT(regs, iface_type[2], 'A');
Justin M. Forbes 696046
diff --git a/src/disk.h b/src/disk.h
Justin M. Forbes 696046
index ac33518..d344399 100644
Justin M. Forbes 696046
--- a/src/disk.h
Justin M. Forbes 696046
+++ b/src/disk.h
Justin M. Forbes 696046
@@ -198,15 +198,16 @@ struct drive_s {
Justin M. Forbes 696046
 #define DISK_SECTOR_SIZE  512
Justin M. Forbes 696046
 #define CDROM_SECTOR_SIZE 2048
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-#define DTYPE_NONE     0x00
Justin M. Forbes 696046
-#define DTYPE_FLOPPY   0x01
Justin M. Forbes 696046
-#define DTYPE_ATA      0x02
Justin M. Forbes 696046
-#define DTYPE_ATAPI    0x03
Justin M. Forbes 696046
-#define DTYPE_RAMDISK  0x04
Justin M. Forbes 696046
-#define DTYPE_CDEMU    0x05
Justin M. Forbes 696046
-#define DTYPE_USB      0x06
Justin M. Forbes 696046
-#define DTYPE_VIRTIO   0x07
Justin M. Forbes 696046
-#define DTYPE_AHCI     0x08
Justin M. Forbes 696046
+#define DTYPE_NONE         0x00
Justin M. Forbes 696046
+#define DTYPE_FLOPPY       0x01
Justin M. Forbes 696046
+#define DTYPE_ATA          0x02
Justin M. Forbes 696046
+#define DTYPE_ATAPI        0x03
Justin M. Forbes 696046
+#define DTYPE_RAMDISK      0x04
Justin M. Forbes 696046
+#define DTYPE_CDEMU        0x05
Justin M. Forbes 696046
+#define DTYPE_USB          0x06
Justin M. Forbes 696046
+#define DTYPE_VIRTIO_BLK   0x07
Justin M. Forbes 696046
+#define DTYPE_AHCI         0x08
Justin M. Forbes 696046
+#define DTYPE_VIRTIO_SCSI  0x09
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 #define MAXDESCSIZE 80
Justin M. Forbes 696046
 
Justin M. Forbes 696046
diff --git a/src/pci_ids.h b/src/pci_ids.h
Justin M. Forbes 696046
index e1cded2..4b59585 100644
Justin M. Forbes 696046
--- a/src/pci_ids.h
Justin M. Forbes 696046
+++ b/src/pci_ids.h
Justin M. Forbes 696046
@@ -2608,3 +2608,4 @@
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 #define PCI_VENDOR_ID_REDHAT_QUMRANET	0x1af4
Justin M. Forbes 696046
 #define PCI_DEVICE_ID_VIRTIO_BLK	0x1001
Justin M. Forbes 696046
+#define PCI_DEVICE_ID_VIRTIO_SCSI	0x1004
Justin M. Forbes 696046
diff --git a/src/post.c b/src/post.c
Justin M. Forbes 696046
index b4ad1fa..d7bbcdd 100644
Justin M. Forbes 696046
--- a/src/post.c
Justin M. Forbes 696046
+++ b/src/post.c
Justin M. Forbes 696046
@@ -26,6 +26,7 @@
Justin M. Forbes 696046
 #include "xen.h" // xen_probe_hvm_info
Justin M. Forbes 696046
 #include "ps2port.h" // ps2port_setup
Justin M. Forbes 696046
 #include "virtio-blk.h" // virtio_blk_setup
Justin M. Forbes 696046
+#include "virtio-scsi.h" // virtio_scsi_setup
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 /****************************************************************
Justin M. Forbes 696046
@@ -190,6 +191,7 @@ init_hw(void)
Justin M. Forbes 696046
     cbfs_payload_setup();
Justin M. Forbes 696046
     ramdisk_setup();
Justin M. Forbes 696046
     virtio_blk_setup();
Justin M. Forbes 696046
+    virtio_scsi_setup();
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 // Begin the boot process by invoking an int0x19 in 16bit mode.
Justin M. Forbes 696046
diff --git a/src/usb-ehci.c b/src/usb-ehci.c
Justin M. Forbes 696046
index a60c607..9bdd638 100644
Justin M. Forbes 696046
--- a/src/usb-ehci.c
Justin M. Forbes 696046
+++ b/src/usb-ehci.c
Justin M. Forbes 696046
@@ -303,22 +303,12 @@ ehci_init(struct pci_device *pci, int busid, struct pci_device *comppci)
Justin M. Forbes 696046
  * End point communication
Justin M. Forbes 696046
  ****************************************************************/
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-static int
Justin M. Forbes 696046
-ehci_wait_qh(struct usb_ehci_s *cntl, struct ehci_qh *qh)
Justin M. Forbes 696046
-{
Justin M. Forbes 696046
-    // XXX - 500ms just a guess
Justin M. Forbes 696046
-    u64 end = calc_future_tsc(500);
Justin M. Forbes 696046
-    for (;;) {
Justin M. Forbes 696046
-        if (qh->qtd_next & EHCI_PTR_TERM)
Justin M. Forbes 696046
-            // XXX - confirm
Justin M. Forbes 696046
-            return 0;
Justin M. Forbes 696046
-        if (check_tsc(end)) {
Justin M. Forbes 696046
-            warn_timeout();
Justin M. Forbes 696046
-            return -1;
Justin M. Forbes 696046
-        }
Justin M. Forbes 696046
-        yield();
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-}
Justin M. Forbes 696046
+struct ehci_pipe {
Justin M. Forbes 696046
+    struct ehci_qh qh;
Justin M. Forbes 696046
+    struct ehci_qtd *next_td, *tds;
Justin M. Forbes 696046
+    void *data;
Justin M. Forbes 696046
+    struct usb_pipe pipe;
Justin M. Forbes 696046
+};
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 // Wait for next USB async frame to start - for ensuring safe memory release.
Justin M. Forbes 696046
 static void
Justin M. Forbes 696046
@@ -362,12 +352,46 @@ ehci_waittick(struct usb_ehci_s *cntl)
Justin M. Forbes 696046
     writel(&cntl->regs->usbsts, STS_IAA);
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-struct ehci_pipe {
Justin M. Forbes 696046
-    struct ehci_qh qh;
Justin M. Forbes 696046
-    struct ehci_qtd *next_td, *tds;
Justin M. Forbes 696046
-    void *data;
Justin M. Forbes 696046
-    struct usb_pipe pipe;
Justin M. Forbes 696046
-};
Justin M. Forbes 696046
+static void
Justin M. Forbes 696046
+ehci_reset_pipe(struct ehci_pipe *pipe)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    SET_FLATPTR(pipe->qh.qtd_next, EHCI_PTR_TERM);
Justin M. Forbes 696046
+    SET_FLATPTR(pipe->qh.alt_next, EHCI_PTR_TERM);
Justin M. Forbes 696046
+    barrier();
Justin M. Forbes 696046
+    SET_FLATPTR(pipe->qh.token, GET_FLATPTR(pipe->qh.token) & QTD_TOGGLE);
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+ehci_wait_td(struct ehci_pipe *pipe, struct ehci_qtd *td, int timeout)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    u64 end = calc_future_tsc(timeout);
Justin M. Forbes 696046
+    u32 status;
Justin M. Forbes 696046
+    for (;;) {
Justin M. Forbes 696046
+        status = td->token;
Justin M. Forbes 696046
+        if (!(status & QTD_STS_ACTIVE))
Justin M. Forbes 696046
+            break;
Justin M. Forbes 696046
+        if (check_tsc(end)) {
Justin M. Forbes 696046
+            u32 cur = GET_FLATPTR(pipe->qh.current);
Justin M. Forbes 696046
+            u32 tok = GET_FLATPTR(pipe->qh.token);
Justin M. Forbes 696046
+            u32 next = GET_FLATPTR(pipe->qh.qtd_next);
Justin M. Forbes 696046
+            warn_timeout();
Justin M. Forbes 696046
+            dprintf(1, "ehci pipe=%p cur=%08x tok=%08x next=%x td=%p status=%x\n"
Justin M. Forbes 696046
+                    , pipe, cur, tok, next, td, status);
Justin M. Forbes 696046
+            ehci_reset_pipe(pipe);
Justin M. Forbes 696046
+            struct usb_ehci_s *cntl = container_of(
Justin M. Forbes 696046
+                GET_FLATPTR(pipe->pipe.cntl), struct usb_ehci_s, usb);
Justin M. Forbes 696046
+            ehci_waittick(cntl);
Justin M. Forbes 696046
+            return -1;
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+        yield();
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+    if (status & QTD_STS_HALT) {
Justin M. Forbes 696046
+        dprintf(1, "ehci_wait_td error - status=%x\n", status);
Justin M. Forbes 696046
+        ehci_reset_pipe(pipe);
Justin M. Forbes 696046
+        return -2;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+    return 0;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 void
Justin M. Forbes 696046
 ehci_free_pipe(struct usb_pipe *p)
Justin M. Forbes 696046
@@ -416,7 +440,6 @@ ehci_alloc_control_pipe(struct usb_pipe *dummy)
Justin M. Forbes 696046
     memset(pipe, 0, sizeof(*pipe));
Justin M. Forbes 696046
     memcpy(&pipe->pipe, dummy, sizeof(pipe->pipe));
Justin M. Forbes 696046
     pipe->qh.qtd_next = pipe->qh.alt_next = EHCI_PTR_TERM;
Justin M. Forbes 696046
-    pipe->qh.token = QTD_STS_HALT;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Add queue head to controller list.
Justin M. Forbes 696046
     struct ehci_qh *async_qh = cntl->async_qh;
Justin M. Forbes 696046
@@ -455,15 +478,14 @@ ehci_control(struct usb_pipe *p, int dir, const void *cmd, int cmdsize
Justin M. Forbes 696046
     ASSERT32FLAT();
Justin M. Forbes 696046
     if (! CONFIG_USB_EHCI)
Justin M. Forbes 696046
         return -1;
Justin M. Forbes 696046
-    dprintf(5, "ehci_control %p\n", p);
Justin M. Forbes 696046
+    dprintf(5, "ehci_control %p (dir=%d cmd=%d data=%d)\n"
Justin M. Forbes 696046
+            , p, dir, cmdsize, datasize);
Justin M. Forbes 696046
     if (datasize > 4*4096 || cmdsize > 4*4096) {
Justin M. Forbes 696046
         // XXX - should support larger sizes.
Justin M. Forbes 696046
         warn_noalloc();
Justin M. Forbes 696046
         return -1;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
     struct ehci_pipe *pipe = container_of(p, struct ehci_pipe, pipe);
Justin M. Forbes 696046
-    struct usb_ehci_s *cntl = container_of(
Justin M. Forbes 696046
-        pipe->pipe.cntl, struct usb_ehci_s, usb);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     u16 maxpacket = pipe->pipe.maxpacket;
Justin M. Forbes 696046
     int speed = pipe->pipe.speed;
Justin M. Forbes 696046
@@ -513,14 +535,12 @@ ehci_control(struct usb_pipe *p, int dir, const void *cmd, int cmdsize
Justin M. Forbes 696046
     // Transfer data
Justin M. Forbes 696046
     barrier();
Justin M. Forbes 696046
     pipe->qh.qtd_next = (u32)tds;
Justin M. Forbes 696046
-    barrier();
Justin M. Forbes 696046
-    pipe->qh.token = 0;
Justin M. Forbes 696046
-    int ret = ehci_wait_qh(cntl, &pipe->qh);
Justin M. Forbes 696046
-    pipe->qh.token = QTD_STS_HALT;
Justin M. Forbes 696046
-    if (ret) {
Justin M. Forbes 696046
-        pipe->qh.qtd_next = pipe->qh.alt_next = EHCI_PTR_TERM;
Justin M. Forbes 696046
-        // XXX - halt qh?
Justin M. Forbes 696046
-        ehci_waittick(cntl);
Justin M. Forbes 696046
+    int i, ret=0;
Justin M. Forbes 696046
+    for (i=0; i<3; i++) {
Justin M. Forbes 696046
+        struct ehci_qtd *td = &tds[i];
Justin M. Forbes 696046
+        ret = ehci_wait_td(pipe, td, 500);
Justin M. Forbes 696046
+        if (ret)
Justin M. Forbes 696046
+            break;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
     free(tds);
Justin M. Forbes 696046
     return ret;
Justin M. Forbes 696046
@@ -545,7 +565,6 @@ ehci_alloc_bulk_pipe(struct usb_pipe *dummy)
Justin M. Forbes 696046
     memset(pipe, 0, sizeof(*pipe));
Justin M. Forbes 696046
     memcpy(&pipe->pipe, dummy, sizeof(pipe->pipe));
Justin M. Forbes 696046
     pipe->qh.qtd_next = pipe->qh.alt_next = EHCI_PTR_TERM;
Justin M. Forbes 696046
-    pipe->qh.token = QTD_STS_HALT;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Add queue head to controller list.
Justin M. Forbes 696046
     struct ehci_qh *async_qh = cntl->async_qh;
Justin M. Forbes 696046
@@ -555,28 +574,6 @@ ehci_alloc_bulk_pipe(struct usb_pipe *dummy)
Justin M. Forbes 696046
     return &pipe->pipe;
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-static int
Justin M. Forbes 696046
-ehci_wait_td(struct ehci_qtd *td)
Justin M. Forbes 696046
-{
Justin M. Forbes 696046
-    u64 end = calc_future_tsc(5000); // XXX - lookup real time.
Justin M. Forbes 696046
-    u32 status;
Justin M. Forbes 696046
-    for (;;) {
Justin M. Forbes 696046
-        status = td->token;
Justin M. Forbes 696046
-        if (!(status & QTD_STS_ACTIVE))
Justin M. Forbes 696046
-            break;
Justin M. Forbes 696046
-        if (check_tsc(end)) {
Justin M. Forbes 696046
-            warn_timeout();
Justin M. Forbes 696046
-            return -1;
Justin M. Forbes 696046
-        }
Justin M. Forbes 696046
-        yield();
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-    if (status & QTD_STS_HALT) {
Justin M. Forbes 696046
-        dprintf(1, "ehci_wait_td error - status=%x\n", status);
Justin M. Forbes 696046
-        return -2;
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-    return 0;
Justin M. Forbes 696046
-}
Justin M. Forbes 696046
-
Justin M. Forbes 696046
 #define STACKQTDS 4
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
@@ -607,15 +604,13 @@ ehci_send_bulk(struct usb_pipe *p, int dir, void *data, int datasize)
Justin M. Forbes 696046
                    | (GET_FLATPTR(pipe->pipe.tt_devaddr) << QH_HUBADDR_SHIFT)));
Justin M. Forbes 696046
     barrier();
Justin M. Forbes 696046
     SET_FLATPTR(pipe->qh.qtd_next, (u32)MAKE_FLATPTR(GET_SEG(SS), tds));
Justin M. Forbes 696046
-    barrier();
Justin M. Forbes 696046
-    SET_FLATPTR(pipe->qh.token, GET_FLATPTR(pipe->qh.token) & QTD_TOGGLE);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     int tdpos = 0;
Justin M. Forbes 696046
     while (datasize) {
Justin M. Forbes 696046
         struct ehci_qtd *td = &tds[tdpos++ % STACKQTDS];
Justin M. Forbes 696046
-        int ret = ehci_wait_td(td);
Justin M. Forbes 696046
+        int ret = ehci_wait_td(pipe, td, 5000);
Justin M. Forbes 696046
         if (ret)
Justin M. Forbes 696046
-            goto fail;
Justin M. Forbes 696046
+            return -1;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
         struct ehci_qtd *nexttd_fl = MAKE_FLATPTR(GET_SEG(SS)
Justin M. Forbes 696046
                                                  , &tds[tdpos % STACKQTDS]);
Justin M. Forbes 696046
@@ -633,21 +628,12 @@ ehci_send_bulk(struct usb_pipe *p, int dir, void *data, int datasize)
Justin M. Forbes 696046
     int i;
Justin M. Forbes 696046
     for (i=0; i
Justin M. Forbes 696046
         struct ehci_qtd *td = &tds[tdpos++ % STACKQTDS];
Justin M. Forbes 696046
-        int ret = ehci_wait_td(td);
Justin M. Forbes 696046
+        int ret = ehci_wait_td(pipe, td, 5000);
Justin M. Forbes 696046
         if (ret)
Justin M. Forbes 696046
-            goto fail;
Justin M. Forbes 696046
+            return -1;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     return 0;
Justin M. Forbes 696046
-fail:
Justin M. Forbes 696046
-    dprintf(1, "ehci_send_bulk failed\n");
Justin M. Forbes 696046
-    SET_FLATPTR(pipe->qh.qtd_next, EHCI_PTR_TERM);
Justin M. Forbes 696046
-    SET_FLATPTR(pipe->qh.alt_next, EHCI_PTR_TERM);
Justin M. Forbes 696046
-    // XXX - halt qh?
Justin M. Forbes 696046
-    struct usb_ehci_s *cntl = container_of(
Justin M. Forbes 696046
-        GET_FLATPTR(pipe->pipe.cntl), struct usb_ehci_s, usb);
Justin M. Forbes 696046
-    ehci_waittick(cntl);
Justin M. Forbes 696046
-    return -1;
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 struct usb_pipe *
Justin M. Forbes 696046
diff --git a/src/usb-msc.c b/src/usb-msc.c
Justin M. Forbes 696046
index 13ef93e..4a09972 100644
Justin M. Forbes 696046
--- a/src/usb-msc.c
Justin M. Forbes 696046
+++ b/src/usb-msc.c
Justin M. Forbes 696046
@@ -46,6 +46,17 @@ struct csw_s {
Justin M. Forbes 696046
     u8 bCSWStatus;
Justin M. Forbes 696046
 } PACKED;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+usb_msc_send(struct usbdrive_s *udrive_g, int dir, void *buf, u32 bytes)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct usb_pipe *pipe;
Justin M. Forbes 696046
+    if (dir == USB_DIR_OUT)
Justin M. Forbes 696046
+        pipe = GET_GLOBAL(udrive_g->bulkout);
Justin M. Forbes 696046
+    else
Justin M. Forbes 696046
+        pipe = GET_GLOBAL(udrive_g->bulkin);
Justin M. Forbes 696046
+    return usb_send_bulk(pipe, dir, buf, bytes);
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
 // Low-level usb command transmit function.
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
 usb_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize)
Justin M. Forbes 696046
@@ -57,35 +68,35 @@ usb_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize)
Justin M. Forbes 696046
             , op->drive_g, 0, op->count, blocksize, op->buf_fl);
Justin M. Forbes 696046
     struct usbdrive_s *udrive_g = container_of(
Justin M. Forbes 696046
         op->drive_g, struct usbdrive_s, drive);
Justin M. Forbes 696046
-    struct usb_pipe *bulkin = GET_GLOBAL(udrive_g->bulkin);
Justin M. Forbes 696046
-    struct usb_pipe *bulkout = GET_GLOBAL(udrive_g->bulkout);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Setup command block wrapper.
Justin M. Forbes 696046
     u32 bytes = blocksize * op->count;
Justin M. Forbes 696046
     struct cbw_s cbw;
Justin M. Forbes 696046
     memset(&cbw, 0, sizeof(cbw));
Justin M. Forbes 696046
+    memcpy(cbw.CBWCB, cdbcmd, USB_CDB_SIZE);
Justin M. Forbes 696046
     cbw.dCBWSignature = CBW_SIGNATURE;
Justin M. Forbes 696046
     cbw.dCBWTag = 999; // XXX
Justin M. Forbes 696046
     cbw.dCBWDataTransferLength = bytes;
Justin M. Forbes 696046
-    cbw.bmCBWFlags = USB_DIR_IN; // XXX
Justin M. Forbes 696046
+    cbw.bmCBWFlags = (cbw.CBWCB[0] == CDB_CMD_WRITE_10) ? USB_DIR_OUT : USB_DIR_IN;
Justin M. Forbes 696046
     cbw.bCBWLUN = 0; // XXX
Justin M. Forbes 696046
     cbw.bCBWCBLength = USB_CDB_SIZE;
Justin M. Forbes 696046
-    memcpy(cbw.CBWCB, cdbcmd, USB_CDB_SIZE);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Transfer cbw to device.
Justin M. Forbes 696046
-    int ret = usb_send_bulk(bulkout, USB_DIR_OUT
Justin M. Forbes 696046
+    int ret = usb_msc_send(udrive_g, USB_DIR_OUT
Justin M. Forbes 696046
                             , MAKE_FLATPTR(GET_SEG(SS), &cbw), sizeof(cbw));
Justin M. Forbes 696046
     if (ret)
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-    // Transfer data from device.
Justin M. Forbes 696046
-    ret = usb_send_bulk(bulkin, USB_DIR_IN, op->buf_fl, bytes);
Justin M. Forbes 696046
-    if (ret)
Justin M. Forbes 696046
-        goto fail;
Justin M. Forbes 696046
+    // Transfer data to/from device.
Justin M. Forbes 696046
+    if (bytes) {
Justin M. Forbes 696046
+        ret = usb_msc_send(udrive_g, cbw.bmCBWFlags, op->buf_fl, bytes);
Justin M. Forbes 696046
+        if (ret)
Justin M. Forbes 696046
+            goto fail;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Transfer csw info.
Justin M. Forbes 696046
     struct csw_s csw;
Justin M. Forbes 696046
-    ret = usb_send_bulk(bulkin, USB_DIR_IN
Justin M. Forbes 696046
+    ret = usb_msc_send(udrive_g, USB_DIR_IN
Justin M. Forbes 696046
                         , MAKE_FLATPTR(GET_SEG(SS), &csw), sizeof(csw));
Justin M. Forbes 696046
     if (ret)
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
@@ -95,7 +106,8 @@ usb_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize)
Justin M. Forbes 696046
     if (csw.bCSWStatus == 2)
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-    op->count -= csw.dCSWDataResidue / blocksize;
Justin M. Forbes 696046
+    if (blocksize)
Justin M. Forbes 696046
+        op->count -= csw.dCSWDataResidue / blocksize;
Justin M. Forbes 696046
     return DISK_RET_EBADTRACK;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 fail:
Justin M. Forbes 696046
@@ -107,75 +119,33 @@ fail:
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 /****************************************************************
Justin M. Forbes 696046
- * Drive ops
Justin M. Forbes 696046
- ****************************************************************/
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-// 16bit command demuxer for ATAPI cdroms.
Justin M. Forbes 696046
-int
Justin M. Forbes 696046
-process_usb_op(struct disk_op_s *op)
Justin M. Forbes 696046
-{
Justin M. Forbes 696046
-    if (!CONFIG_USB_MSC)
Justin M. Forbes 696046
-        return 0;
Justin M. Forbes 696046
-    switch (op->command) {
Justin M. Forbes 696046
-    case CMD_READ:
Justin M. Forbes 696046
-        return cdb_read(op);
Justin M. Forbes 696046
-    case CMD_FORMAT:
Justin M. Forbes 696046
-    case CMD_WRITE:
Justin M. Forbes 696046
-        return DISK_RET_EWRITEPROTECT;
Justin M. Forbes 696046
-    case CMD_RESET:
Justin M. Forbes 696046
-    case CMD_ISREADY:
Justin M. Forbes 696046
-    case CMD_VERIFY:
Justin M. Forbes 696046
-    case CMD_SEEK:
Justin M. Forbes 696046
-        return DISK_RET_SUCCESS;
Justin M. Forbes 696046
-    default:
Justin M. Forbes 696046
-        op->count = 0;
Justin M. Forbes 696046
-        return DISK_RET_EPARAM;
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-}
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-/****************************************************************
Justin M. Forbes 696046
  * Setup
Justin M. Forbes 696046
  ****************************************************************/
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 static int
Justin M. Forbes 696046
-setup_drive_cdrom(struct disk_op_s *op, char *desc)
Justin M. Forbes 696046
+setup_drive_cdrom(struct drive_s *drive, char *desc)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
-    op->drive_g->blksize = CDROM_SECTOR_SIZE;
Justin M. Forbes 696046
-    op->drive_g->sectors = (u64)-1;
Justin M. Forbes 696046
+    drive->sectors = (u64)-1;
Justin M. Forbes 696046
     struct usb_pipe *pipe = container_of(
Justin M. Forbes 696046
-        op->drive_g, struct usbdrive_s, drive)->bulkout;
Justin M. Forbes 696046
+        drive, struct usbdrive_s, drive)->bulkout;
Justin M. Forbes 696046
     int prio = bootprio_find_usb(pipe->cntl->pci, pipe->path);
Justin M. Forbes 696046
-    boot_add_cd(op->drive_g, desc, prio);
Justin M. Forbes 696046
+    boot_add_cd(drive, desc, prio);
Justin M. Forbes 696046
     return 0;
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 static int
Justin M. Forbes 696046
-setup_drive_hd(struct disk_op_s *op, char *desc)
Justin M. Forbes 696046
+setup_drive_hd(struct drive_s *drive, char *desc)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
-    struct cdbres_read_capacity info;
Justin M. Forbes 696046
-    int ret = cdb_read_capacity(op, &info;;
Justin M. Forbes 696046
-    if (ret)
Justin M. Forbes 696046
-        return ret;
Justin M. Forbes 696046
-    // XXX - retry for some timeout?
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    u32 blksize = ntohl(info.blksize), sectors = ntohl(info.sectors);
Justin M. Forbes 696046
-    if (blksize != DISK_SECTOR_SIZE) {
Justin M. Forbes 696046
-        if (blksize == CDROM_SECTOR_SIZE)
Justin M. Forbes 696046
-            return setup_drive_cdrom(op, desc);
Justin M. Forbes 696046
-        dprintf(1, "Unsupported USB MSC block size %d\n", blksize);
Justin M. Forbes 696046
+    if (drive->blksize != DISK_SECTOR_SIZE) {
Justin M. Forbes 696046
+        dprintf(1, "Unsupported USB MSC block size %d\n", drive->blksize);
Justin M. Forbes 696046
         return -1;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
-    op->drive_g->blksize = blksize;
Justin M. Forbes 696046
-    op->drive_g->sectors = sectors;
Justin M. Forbes 696046
-    dprintf(1, "USB MSC blksize=%d sectors=%d\n", blksize, sectors);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     // Register with bcv system.
Justin M. Forbes 696046
     struct usb_pipe *pipe = container_of(
Justin M. Forbes 696046
-        op->drive_g, struct usbdrive_s, drive)->bulkout;
Justin M. Forbes 696046
+        drive, struct usbdrive_s, drive)->bulkout;
Justin M. Forbes 696046
     int prio = bootprio_find_usb(pipe->cntl->pci, pipe->path);
Justin M. Forbes 696046
-    boot_add_hd(op->drive_g, desc, prio);
Justin M. Forbes 696046
-
Justin M. Forbes 696046
+    boot_add_hd(drive, desc, prio);
Justin M. Forbes 696046
     return 0;
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
@@ -218,37 +188,17 @@ usb_msc_init(struct usb_pipe *pipe
Justin M. Forbes 696046
     if (!udrive_g->bulkin || !udrive_g->bulkout)
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-    // Validate drive and find block size and sector count.
Justin M. Forbes 696046
-    struct disk_op_s dop;
Justin M. Forbes 696046
-    memset(&dop, 0, sizeof(dop));
Justin M. Forbes 696046
-    dop.drive_g = &udrive_g->drive;
Justin M. Forbes 696046
-    struct cdbres_inquiry data;
Justin M. Forbes 696046
-    int ret = cdb_get_inquiry(&dop, &data);
Justin M. Forbes 696046
+    int ret, pdt;
Justin M. Forbes 696046
+    char *desc = NULL;
Justin M. Forbes 696046
+    ret = scsi_init_drive(&udrive_g->drive, "USB MSC", &pdt, &desc);
Justin M. Forbes 696046
     if (ret)
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
-    char vendor[sizeof(data.vendor)+1], product[sizeof(data.product)+1];
Justin M. Forbes 696046
-    char rev[sizeof(data.rev)+1];
Justin M. Forbes 696046
-    strtcpy(vendor, data.vendor, sizeof(vendor));
Justin M. Forbes 696046
-    nullTrailingSpace(vendor);
Justin M. Forbes 696046
-    strtcpy(product, data.product, sizeof(product));
Justin M. Forbes 696046
-    nullTrailingSpace(product);
Justin M. Forbes 696046
-    strtcpy(rev, data.rev, sizeof(rev));
Justin M. Forbes 696046
-    nullTrailingSpace(rev);
Justin M. Forbes 696046
-    int pdt = data.pdt & 0x1f;
Justin M. Forbes 696046
-    int removable = !!(data.removable & 0x80);
Justin M. Forbes 696046
-    dprintf(1, "USB MSC vendor='%s' product='%s' rev='%s' type=%d removable=%d\n"
Justin M. Forbes 696046
-            , vendor, product, rev, pdt, removable);
Justin M. Forbes 696046
-    udrive_g->drive.removable = removable;
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    if (pdt == USB_MSC_TYPE_CDROM) {
Justin M. Forbes 696046
-        char *desc = znprintf(MAXDESCSIZE, "DVD/CD [USB Drive %s %s %s]"
Justin M. Forbes 696046
-                              , vendor, product, rev);
Justin M. Forbes 696046
-        ret = setup_drive_cdrom(&dop, desc);
Justin M. Forbes 696046
-    } else {
Justin M. Forbes 696046
-        char *desc = znprintf(MAXDESCSIZE, "USB Drive %s %s %s"
Justin M. Forbes 696046
-                              , vendor, product, rev);
Justin M. Forbes 696046
-        ret = setup_drive_hd(&dop, desc);
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    if (pdt == SCSI_TYPE_CDROM)
Justin M. Forbes 696046
+        ret = setup_drive_cdrom(&udrive_g->drive, desc);
Justin M. Forbes 696046
+    else
Justin M. Forbes 696046
+        ret = setup_drive_hd(&udrive_g->drive, desc);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
     if (ret)
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
diff --git a/src/usb-msc.h b/src/usb-msc.h
Justin M. Forbes 696046
index 71adb20..a8686a3 100644
Justin M. Forbes 696046
--- a/src/usb-msc.h
Justin M. Forbes 696046
+++ b/src/usb-msc.h
Justin M. Forbes 696046
@@ -21,7 +21,4 @@ int process_usb_op(struct disk_op_s *op);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 #define US_PR_BULK         0x50
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-#define USB_MSC_TYPE_DISK  0x00
Justin M. Forbes 696046
-#define USB_MSC_TYPE_CDROM 0x05
Justin M. Forbes 696046
-
Justin M. Forbes 696046
 #endif // ush-msc.h
Justin M. Forbes 696046
diff --git a/src/usb-uhci.c b/src/usb-uhci.c
Justin M. Forbes 696046
index f3680d3..a78dbca 100644
Justin M. Forbes 696046
--- a/src/usb-uhci.c
Justin M. Forbes 696046
+++ b/src/usb-uhci.c
Justin M. Forbes 696046
@@ -212,26 +212,13 @@ uhci_init(struct pci_device *pci, int busid)
Justin M. Forbes 696046
  * End point communication
Justin M. Forbes 696046
  ****************************************************************/
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-static int
Justin M. Forbes 696046
-wait_qh(struct usb_uhci_s *cntl, struct uhci_qh *qh)
Justin M. Forbes 696046
-{
Justin M. Forbes 696046
-    // XXX - 500ms just a guess
Justin M. Forbes 696046
-    u64 end = calc_future_tsc(500);
Justin M. Forbes 696046
-    for (;;) {
Justin M. Forbes 696046
-        if (qh->element & UHCI_PTR_TERM)
Justin M. Forbes 696046
-            return 0;
Justin M. Forbes 696046
-        if (check_tsc(end)) {
Justin M. Forbes 696046
-            warn_timeout();
Justin M. Forbes 696046
-            struct uhci_td *td = (void*)(qh->element & ~UHCI_PTR_BITS);
Justin M. Forbes 696046
-            dprintf(1, "Timeout on wait_qh %p (td=%p s=%x c=%x/%x)\n"
Justin M. Forbes 696046
-                    , qh, td, td->status
Justin M. Forbes 696046
-                    , inw(cntl->iobase + USBCMD)
Justin M. Forbes 696046
-                    , inw(cntl->iobase + USBSTS));
Justin M. Forbes 696046
-            return -1;
Justin M. Forbes 696046
-        }
Justin M. Forbes 696046
-        yield();
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-}
Justin M. Forbes 696046
+struct uhci_pipe {
Justin M. Forbes 696046
+    struct uhci_qh qh;
Justin M. Forbes 696046
+    struct uhci_td *next_td;
Justin M. Forbes 696046
+    struct usb_pipe pipe;
Justin M. Forbes 696046
+    u16 iobase;
Justin M. Forbes 696046
+    u8 toggle;
Justin M. Forbes 696046
+};
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 // Wait for next USB frame to start - for ensuring safe memory release.
Justin M. Forbes 696046
 static void
Justin M. Forbes 696046
@@ -251,13 +238,29 @@ uhci_waittick(u16 iobase)
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
-struct uhci_pipe {
Justin M. Forbes 696046
-    struct uhci_qh qh;
Justin M. Forbes 696046
-    struct uhci_td *next_td;
Justin M. Forbes 696046
-    struct usb_pipe pipe;
Justin M. Forbes 696046
-    u16 iobase;
Justin M. Forbes 696046
-    u8 toggle;
Justin M. Forbes 696046
-};
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+wait_pipe(struct uhci_pipe *pipe, int timeout)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    u64 end = calc_future_tsc(timeout);
Justin M. Forbes 696046
+    for (;;) {
Justin M. Forbes 696046
+        u32 el_link = GET_FLATPTR(pipe->qh.element);
Justin M. Forbes 696046
+        if (el_link & UHCI_PTR_TERM)
Justin M. Forbes 696046
+            return 0;
Justin M. Forbes 696046
+        if (check_tsc(end)) {
Justin M. Forbes 696046
+            warn_timeout();
Justin M. Forbes 696046
+            u16 iobase = GET_FLATPTR(pipe->iobase);
Justin M. Forbes 696046
+            struct uhci_td *td = (void*)(el_link & ~UHCI_PTR_BITS);
Justin M. Forbes 696046
+            dprintf(1, "Timeout on wait_pipe %p (td=%p s=%x c=%x/%x)\n"
Justin M. Forbes 696046
+                    , pipe, (void*)el_link, GET_FLATPTR(td->status)
Justin M. Forbes 696046
+                    , inw(iobase + USBCMD)
Justin M. Forbes 696046
+                    , inw(iobase + USBSTS));
Justin M. Forbes 696046
+            SET_FLATPTR(pipe->qh.element, UHCI_PTR_TERM);
Justin M. Forbes 696046
+            uhci_waittick(iobase);
Justin M. Forbes 696046
+            return -1;
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+        yield();
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 void
Justin M. Forbes 696046
 uhci_free_pipe(struct usb_pipe *p)
Justin M. Forbes 696046
@@ -331,8 +334,6 @@ uhci_control(struct usb_pipe *p, int dir, const void *cmd, int cmdsize
Justin M. Forbes 696046
         return -1;
Justin M. Forbes 696046
     dprintf(5, "uhci_control %p\n", p);
Justin M. Forbes 696046
     struct uhci_pipe *pipe = container_of(p, struct uhci_pipe, pipe);
Justin M. Forbes 696046
-    struct usb_uhci_s *cntl = container_of(
Justin M. Forbes 696046
-        pipe->pipe.cntl, struct usb_uhci_s, usb);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
     int maxpacket = pipe->pipe.maxpacket;
Justin M. Forbes 696046
     int lowspeed = pipe->pipe.speed;
Justin M. Forbes 696046
@@ -376,11 +377,7 @@ uhci_control(struct usb_pipe *p, int dir, const void *cmd, int cmdsize
Justin M. Forbes 696046
     // Transfer data
Justin M. Forbes 696046
     barrier();
Justin M. Forbes 696046
     pipe->qh.element = (u32)&tds[0];
Justin M. Forbes 696046
-    int ret = wait_qh(cntl, &pipe->qh);
Justin M. Forbes 696046
-    if (ret) {
Justin M. Forbes 696046
-        pipe->qh.element = UHCI_PTR_TERM;
Justin M. Forbes 696046
-        uhci_waittick(pipe->iobase);
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
+    int ret = wait_pipe(pipe, 500);
Justin M. Forbes 696046
     free(tds);
Justin M. Forbes 696046
     return ret;
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
@@ -487,16 +484,8 @@ uhci_send_bulk(struct usb_pipe *p, int dir, void *data, int datasize)
Justin M. Forbes 696046
         data += transfer;
Justin M. Forbes 696046
         datasize -= transfer;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
-    int i;
Justin M. Forbes 696046
-    for (i=0; i
Justin M. Forbes 696046
-        struct uhci_td *td = &tds[tdpos++ % STACKTDS];
Justin M. Forbes 696046
-        int ret = wait_td(td);
Justin M. Forbes 696046
-        if (ret)
Justin M. Forbes 696046
-            goto fail;
Justin M. Forbes 696046
-    }
Justin M. Forbes 696046
-
Justin M. Forbes 696046
     SET_FLATPTR(pipe->toggle, !!toggle);
Justin M. Forbes 696046
-    return 0;
Justin M. Forbes 696046
+    return wait_pipe(pipe, 5000);
Justin M. Forbes 696046
 fail:
Justin M. Forbes 696046
     dprintf(1, "uhci_send_bulk failed\n");
Justin M. Forbes 696046
     SET_FLATPTR(pipe->qh.element, UHCI_PTR_TERM);
Justin M. Forbes 696046
diff --git a/src/virtio-blk.c b/src/virtio-blk.c
Justin M. Forbes 696046
index b1274fc..b869189 100644
Justin M. Forbes 696046
--- a/src/virtio-blk.c
Justin M. Forbes 696046
+++ b/src/virtio-blk.c
Justin M. Forbes 696046
@@ -75,7 +75,7 @@ virtio_blk_op(struct disk_op_s *op, int write)
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 int
Justin M. Forbes 696046
-process_virtio_op(struct disk_op_s *op)
Justin M. Forbes 696046
+process_virtio_blk_op(struct disk_op_s *op)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
     if (! CONFIG_VIRTIO_BLK || CONFIG_COREBOOT)
Justin M. Forbes 696046
         return 0;
Justin M. Forbes 696046
@@ -103,27 +103,17 @@ init_virtio_blk(struct pci_device *pci)
Justin M. Forbes 696046
     dprintf(1, "found virtio-blk at %x:%x\n", pci_bdf_to_bus(bdf),
Justin M. Forbes 696046
             pci_bdf_to_dev(bdf));
Justin M. Forbes 696046
     struct virtiodrive_s *vdrive_g = malloc_fseg(sizeof(*vdrive_g));
Justin M. Forbes 696046
-    struct vring_virtqueue *vq = memalign_low(PAGE_SIZE, sizeof(*vq));
Justin M. Forbes 696046
-    if (!vdrive_g || !vq) {
Justin M. Forbes 696046
+    if (!vdrive_g) {
Justin M. Forbes 696046
         warn_noalloc();
Justin M. Forbes 696046
-        goto fail;
Justin M. Forbes 696046
+        return;
Justin M. Forbes 696046
     }
Justin M. Forbes 696046
     memset(vdrive_g, 0, sizeof(*vdrive_g));
Justin M. Forbes 696046
-    memset(vq, 0, sizeof(*vq));
Justin M. Forbes 696046
-    vdrive_g->drive.type = DTYPE_VIRTIO;
Justin M. Forbes 696046
+    vdrive_g->drive.type = DTYPE_VIRTIO_BLK;
Justin M. Forbes 696046
     vdrive_g->drive.cntl_id = bdf;
Justin M. Forbes 696046
-    vdrive_g->vq = vq;
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    u16 ioaddr = pci_config_readl(bdf, PCI_BASE_ADDRESS_0) &
Justin M. Forbes 696046
-        PCI_BASE_ADDRESS_IO_MASK;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+    u16 ioaddr = vp_init_simple(bdf);
Justin M. Forbes 696046
     vdrive_g->ioaddr = ioaddr;
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    vp_reset(ioaddr);
Justin M. Forbes 696046
-    vp_set_status(ioaddr, VIRTIO_CONFIG_S_ACKNOWLEDGE |
Justin M. Forbes 696046
-                  VIRTIO_CONFIG_S_DRIVER );
Justin M. Forbes 696046
-
Justin M. Forbes 696046
-    if (vp_find_vq(ioaddr, 0, vdrive_g->vq) < 0 ) {
Justin M. Forbes 696046
+    if (vp_find_vq(ioaddr, 0, &vdrive_g->vq) < 0 ) {
Justin M. Forbes 696046
         dprintf(1, "fail to find vq for virtio-blk %x:%x\n",
Justin M. Forbes 696046
                 pci_bdf_to_bus(bdf), pci_bdf_to_dev(bdf));
Justin M. Forbes 696046
         goto fail;
Justin M. Forbes 696046
@@ -161,8 +151,8 @@ init_virtio_blk(struct pci_device *pci)
Justin M. Forbes 696046
     return;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 fail:
Justin M. Forbes 696046
+    free(vdrive_g->vq);
Justin M. Forbes 696046
     free(vdrive_g);
Justin M. Forbes 696046
-    free(vq);
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 void
Justin M. Forbes 696046
diff --git a/src/virtio-blk.h b/src/virtio-blk.h
Justin M. Forbes 696046
index 7243704..0825f09 100644
Justin M. Forbes 696046
--- a/src/virtio-blk.h
Justin M. Forbes 696046
+++ b/src/virtio-blk.h
Justin M. Forbes 696046
@@ -37,7 +37,7 @@ struct virtio_blk_outhdr {
Justin M. Forbes 696046
 #define VIRTIO_BLK_S_UNSUPP	2
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 struct disk_op_s;
Justin M. Forbes 696046
-int process_virtio_op(struct disk_op_s *op);
Justin M. Forbes 696046
+int process_virtio_blk_op(struct disk_op_s *op);
Justin M. Forbes 696046
 void virtio_blk_setup(void);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 #endif /* _VIRTIO_BLK_H */
Justin M. Forbes 696046
diff --git a/src/virtio-pci.c b/src/virtio-pci.c
Justin M. Forbes 696046
index db19e97..7e0c1a5 100644
Justin M. Forbes 696046
--- a/src/virtio-pci.c
Justin M. Forbes 696046
+++ b/src/virtio-pci.c
Justin M. Forbes 696046
@@ -21,12 +21,18 @@
Justin M. Forbes 696046
 #include "util.h" // dprintf
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 int vp_find_vq(unsigned int ioaddr, int queue_index,
Justin M. Forbes 696046
-               struct vring_virtqueue *vq)
Justin M. Forbes 696046
+               struct vring_virtqueue **p_vq)
Justin M. Forbes 696046
 {
Justin M. Forbes 696046
-   struct vring * vr = &vq->vring;
Justin M. Forbes 696046
    u16 num;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    ASSERT32FLAT();
Justin M. Forbes 696046
+   struct vring_virtqueue *vq = *p_vq = memalign_low(PAGE_SIZE, sizeof(*vq));
Justin M. Forbes 696046
+   if (!vq) {
Justin M. Forbes 696046
+       warn_noalloc();
Justin M. Forbes 696046
+       goto fail;
Justin M. Forbes 696046
+   }
Justin M. Forbes 696046
+   memset(vq, 0, sizeof(*vq));
Justin M. Forbes 696046
+
Justin M. Forbes 696046
    /* select the queue */
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    outw(queue_index, ioaddr + VIRTIO_PCI_QUEUE_SEL);
Justin M. Forbes 696046
@@ -36,25 +42,26 @@ int vp_find_vq(unsigned int ioaddr, int queue_index,
Justin M. Forbes 696046
    num = inw(ioaddr + VIRTIO_PCI_QUEUE_NUM);
Justin M. Forbes 696046
    if (!num) {
Justin M. Forbes 696046
        dprintf(1, "ERROR: queue size is 0\n");
Justin M. Forbes 696046
-       return -1;
Justin M. Forbes 696046
+       goto fail;
Justin M. Forbes 696046
    }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    if (num > MAX_QUEUE_NUM) {
Justin M. Forbes 696046
        dprintf(1, "ERROR: queue size %d > %d\n", num, MAX_QUEUE_NUM);
Justin M. Forbes 696046
-       return -1;
Justin M. Forbes 696046
+       goto fail;
Justin M. Forbes 696046
    }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    /* check if the queue is already active */
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    if (inl(ioaddr + VIRTIO_PCI_QUEUE_PFN)) {
Justin M. Forbes 696046
        dprintf(1, "ERROR: queue already active\n");
Justin M. Forbes 696046
-       return -1;
Justin M. Forbes 696046
+       goto fail;
Justin M. Forbes 696046
    }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    vq->queue_index = queue_index;
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    /* initialize the queue */
Justin M. Forbes 696046
 
Justin M. Forbes 696046
+   struct vring * vr = &vq->vring;
Justin M. Forbes 696046
    vring_init(vr, num, (unsigned char*)&vq->queue);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    /* activate the queue
Justin M. Forbes 696046
@@ -66,4 +73,20 @@ int vp_find_vq(unsigned int ioaddr, int queue_index,
Justin M. Forbes 696046
         ioaddr + VIRTIO_PCI_QUEUE_PFN);
Justin M. Forbes 696046
 
Justin M. Forbes 696046
    return num;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+fail:
Justin M. Forbes 696046
+   free(vq);
Justin M. Forbes 696046
+   *p_vq = NULL;
Justin M. Forbes 696046
+   return -1;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+u16 vp_init_simple(u16 bdf)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    u16 ioaddr = pci_config_readl(bdf, PCI_BASE_ADDRESS_0) &
Justin M. Forbes 696046
+        PCI_BASE_ADDRESS_IO_MASK;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    vp_reset(ioaddr);
Justin M. Forbes 696046
+    vp_set_status(ioaddr, VIRTIO_CONFIG_S_ACKNOWLEDGE |
Justin M. Forbes 696046
+                  VIRTIO_CONFIG_S_DRIVER );
Justin M. Forbes 696046
+    return ioaddr;
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
diff --git a/src/virtio-pci.h b/src/virtio-pci.h
Justin M. Forbes 696046
index d21d5a5..e1d972d 100644
Justin M. Forbes 696046
--- a/src/virtio-pci.h
Justin M. Forbes 696046
+++ b/src/virtio-pci.h
Justin M. Forbes 696046
@@ -99,6 +99,7 @@ static inline void vp_del_vq(unsigned int ioaddr, int queue_index)
Justin M. Forbes 696046
 }
Justin M. Forbes 696046
 
Justin M. Forbes 696046
 struct vring_virtqueue;
Justin M. Forbes 696046
+u16 vp_init_simple(u16 bdf);
Justin M. Forbes 696046
 int vp_find_vq(unsigned int ioaddr, int queue_index,
Justin M. Forbes 696046
-               struct vring_virtqueue *vq);
Justin M. Forbes 696046
+               struct vring_virtqueue **p_vq);
Justin M. Forbes 696046
 #endif /* _VIRTIO_PCI_H_ */
Justin M. Forbes 696046
diff --git a/src/virtio-scsi.c b/src/virtio-scsi.c
Justin M. Forbes 696046
new file mode 100644
Justin M. Forbes 696046
index 0000000..2f6c48a
Justin M. Forbes 696046
--- /dev/null
Justin M. Forbes 696046
+++ b/src/virtio-scsi.c
Justin M. Forbes 696046
@@ -0,0 +1,211 @@
Justin M. Forbes 696046
+// Virtio SCSI boot support.
Justin M. Forbes 696046
+//
Justin M. Forbes 696046
+// Copyright (C) 2011 Red Hat Inc.
Justin M. Forbes 696046
+//
Justin M. Forbes 696046
+// Authors:
Justin M. Forbes 696046
+//  Paolo Bonzini <pbonzini@redhat.com>
Justin M. Forbes 696046
+//
Justin M. Forbes 696046
+// This file may be distributed under the terms of the GNU LGPLv3 license.
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+#include "util.h" // dprintf
Justin M. Forbes 696046
+#include "pci.h" // foreachpci
Justin M. Forbes 696046
+#include "config.h" // CONFIG_*
Justin M. Forbes 696046
+#include "biosvar.h" // GET_GLOBAL
Justin M. Forbes 696046
+#include "pci_ids.h" // PCI_DEVICE_ID_VIRTIO_BLK
Justin M. Forbes 696046
+#include "pci_regs.h" // PCI_VENDOR_ID
Justin M. Forbes 696046
+#include "boot.h" // boot_add_hd
Justin M. Forbes 696046
+#include "blockcmd.h" // CDB_CMD_WRITE_10, SCSI_TYPE_CDROM
Justin M. Forbes 696046
+#include "virtio-pci.h"
Justin M. Forbes 696046
+#include "virtio-ring.h"
Justin M. Forbes 696046
+#include "virtio-scsi.h"
Justin M. Forbes 696046
+#include "disk.h"
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+struct virtio_lun_s {
Justin M. Forbes 696046
+    struct drive_s drive;
Justin M. Forbes 696046
+    struct pci_device *pci;
Justin M. Forbes 696046
+    struct vring_virtqueue *vq;
Justin M. Forbes 696046
+    u16 ioaddr;
Justin M. Forbes 696046
+    u16 target;
Justin M. Forbes 696046
+    u16 lun;
Justin M. Forbes 696046
+};
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+virtio_scsi_cmd(u16 ioaddr, struct vring_virtqueue *vq, struct disk_op_s *op,
Justin M. Forbes 696046
+                void *cdbcmd, u16 target, u16 lun, u32 len)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct virtio_scsi_req_cmd req;
Justin M. Forbes 696046
+    struct virtio_scsi_resp_cmd resp;
Justin M. Forbes 696046
+    struct vring_list sg[3];
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    memset(&req, 0, sizeof(req));
Justin M. Forbes 696046
+    req.lun[0] = 1;
Justin M. Forbes 696046
+    req.lun[1] = target;
Justin M. Forbes 696046
+    req.lun[2] = (lun >> 8) | 0x40;
Justin M. Forbes 696046
+    req.lun[3] = (lun & 0xff);
Justin M. Forbes 696046
+    memcpy(req.cdb, cdbcmd, 16);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    int datain = (req.cdb[0] != CDB_CMD_WRITE_10);
Justin M. Forbes 696046
+    int data_idx = (datain ? 2 : 1);
Justin M. Forbes 696046
+    int out_num = (datain ? 1 : 2);
Justin M. Forbes 696046
+    int in_num = (len ? 3 : 2) - out_num;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    sg[0].addr   = MAKE_FLATPTR(GET_SEG(SS), &req;;
Justin M. Forbes 696046
+    sg[0].length = sizeof(req);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    sg[out_num].addr   = MAKE_FLATPTR(GET_SEG(SS), &resp);
Justin M. Forbes 696046
+    sg[out_num].length = sizeof(resp);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    sg[data_idx].addr   = op->buf_fl;
Justin M. Forbes 696046
+    sg[data_idx].length = len;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    /* Add to virtqueue and kick host */
Justin M. Forbes 696046
+    vring_add_buf(vq, sg, out_num, in_num, 0, 0);
Justin M. Forbes 696046
+    vring_kick(ioaddr, vq, 1);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    /* Wait for reply */
Justin M. Forbes 696046
+    while (!vring_more_used(vq))
Justin M. Forbes 696046
+        usleep(5);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    /* Reclaim virtqueue element */
Justin M. Forbes 696046
+    vring_get_buf(vq, NULL);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    /* Clear interrupt status register.  Avoid leaving interrupts stuck if
Justin M. Forbes 696046
+     * VRING_AVAIL_F_NO_INTERRUPT was ignored and interrupts were raised.
Justin M. Forbes 696046
+     */
Justin M. Forbes 696046
+    vp_get_isr(ioaddr);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    if (resp.response == VIRTIO_SCSI_S_OK && resp.status == 0) {
Justin M. Forbes 696046
+        return DISK_RET_SUCCESS;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+    return DISK_RET_EBADTRACK;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+int
Justin M. Forbes 696046
+virtio_scsi_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct virtio_lun_s *vlun =
Justin M. Forbes 696046
+        container_of(op->drive_g, struct virtio_lun_s, drive);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    return virtio_scsi_cmd(GET_GLOBAL(vlun->ioaddr),
Justin M. Forbes 696046
+                           GET_GLOBAL(vlun->vq), op, cdbcmd,
Justin M. Forbes 696046
+                           GET_GLOBAL(vlun->target), GET_GLOBAL(vlun->lun),
Justin M. Forbes 696046
+                           blocksize * op->count);
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+setup_lun_cdrom(struct virtio_lun_s *vlun, char *desc)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    int prio = bootprio_find_scsi_device(vlun->pci, vlun->target, vlun->lun);
Justin M. Forbes 696046
+    boot_add_cd(&vlun->drive, desc, prio);
Justin M. Forbes 696046
+    return 0;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+setup_lun_hd(struct virtio_lun_s *vlun, char *desc)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    if (vlun->drive.blksize != DISK_SECTOR_SIZE) {
Justin M. Forbes 696046
+        dprintf(1, "Unsupported block size %d\n", vlun->drive.blksize);
Justin M. Forbes 696046
+        return -1;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    // Register with bcv system.
Justin M. Forbes 696046
+    int prio = bootprio_find_scsi_device(vlun->pci, vlun->target, vlun->lun);
Justin M. Forbes 696046
+    boot_add_hd(&vlun->drive, desc, prio);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    return 0;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+virtio_scsi_add_lun(struct pci_device *pci, u16 ioaddr,
Justin M. Forbes 696046
+                    struct vring_virtqueue *vq, u16 target, u16 lun)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    struct virtio_lun_s *vlun = malloc_fseg(sizeof(*vlun));
Justin M. Forbes 696046
+    if (!vlun) {
Justin M. Forbes 696046
+        warn_noalloc();
Justin M. Forbes 696046
+        return -1;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+    memset(vlun, 0, sizeof(*vlun));
Justin M. Forbes 696046
+    vlun->drive.type = DTYPE_VIRTIO_SCSI;
Justin M. Forbes 696046
+    vlun->drive.cntl_id = pci->bdf;
Justin M. Forbes 696046
+    vlun->pci = pci;
Justin M. Forbes 696046
+    vlun->ioaddr = ioaddr;
Justin M. Forbes 696046
+    vlun->vq = vq;
Justin M. Forbes 696046
+    vlun->target = target;
Justin M. Forbes 696046
+    vlun->lun = lun;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    int pdt, ret;
Justin M. Forbes 696046
+    char *desc = NULL;
Justin M. Forbes 696046
+    ret = scsi_init_drive(&vlun->drive, "virtio-scsi", &pdt, &desc);
Justin M. Forbes 696046
+    if (ret)
Justin M. Forbes 696046
+        goto fail;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    if (pdt == SCSI_TYPE_CDROM)
Justin M. Forbes 696046
+        ret = setup_lun_cdrom(vlun, desc);
Justin M. Forbes 696046
+    else
Justin M. Forbes 696046
+        ret = setup_lun_hd(vlun, desc);
Justin M. Forbes 696046
+    if (ret)
Justin M. Forbes 696046
+        goto fail;
Justin M. Forbes 696046
+    return ret;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+fail:
Justin M. Forbes 696046
+    free(vlun);
Justin M. Forbes 696046
+    return -1;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static int
Justin M. Forbes 696046
+virtio_scsi_scan_target(struct pci_device *pci, u16 ioaddr,
Justin M. Forbes 696046
+                        struct vring_virtqueue *vq, u16 target)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    /* TODO: send REPORT LUNS.  For now, only LUN 0 is recognized.  */
Justin M. Forbes 696046
+    int ret = virtio_scsi_add_lun(pci, ioaddr, vq, target, 0);
Justin M. Forbes 696046
+    return ret < 0 ? ret : 1;
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+static void
Justin M. Forbes 696046
+init_virtio_scsi(struct pci_device *pci)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    u16 bdf = pci->bdf;
Justin M. Forbes 696046
+    dprintf(1, "found virtio-scsi at %x:%x\n", pci_bdf_to_bus(bdf),
Justin M. Forbes 696046
+            pci_bdf_to_dev(bdf));
Justin M. Forbes 696046
+    struct vring_virtqueue *vq = NULL;
Justin M. Forbes 696046
+    u16 ioaddr = vp_init_simple(bdf);
Justin M. Forbes 696046
+    if (vp_find_vq(ioaddr, 2, &vq) < 0 ) {
Justin M. Forbes 696046
+        if (vq) {
Justin M. Forbes 696046
+            dprintf(1, "fail to find vq for virtio-scsi %x:%x\n",
Justin M. Forbes 696046
+                    pci_bdf_to_bus(bdf), pci_bdf_to_dev(bdf));
Justin M. Forbes 696046
+        }
Justin M. Forbes 696046
+        goto fail;
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    int i, tot;
Justin M. Forbes 696046
+    for (tot = 0, i = 0; i < 256; i++)
Justin M. Forbes 696046
+        tot += virtio_scsi_scan_target(pci, ioaddr, vq, i);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    if (!tot)
Justin M. Forbes 696046
+        goto fail;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    vp_set_status(ioaddr, VIRTIO_CONFIG_S_ACKNOWLEDGE |
Justin M. Forbes 696046
+                  VIRTIO_CONFIG_S_DRIVER | VIRTIO_CONFIG_S_DRIVER_OK);
Justin M. Forbes 696046
+    return;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+fail:
Justin M. Forbes 696046
+    free(vq);
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+void
Justin M. Forbes 696046
+virtio_scsi_setup(void)
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    ASSERT32FLAT();
Justin M. Forbes 696046
+    if (! CONFIG_VIRTIO_SCSI || CONFIG_COREBOOT)
Justin M. Forbes 696046
+        return;
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    dprintf(3, "init virtio-scsi\n");
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+    struct pci_device *pci;
Justin M. Forbes 696046
+    foreachpci(pci) {
Justin M. Forbes 696046
+        if (pci->vendor != PCI_VENDOR_ID_REDHAT_QUMRANET
Justin M. Forbes 696046
+            || pci->device != PCI_DEVICE_ID_VIRTIO_SCSI)
Justin M. Forbes 696046
+            continue;
Justin M. Forbes 696046
+        init_virtio_scsi(pci);
Justin M. Forbes 696046
+    }
Justin M. Forbes 696046
+}
Justin M. Forbes 696046
diff --git a/src/virtio-scsi.h b/src/virtio-scsi.h
Justin M. Forbes 696046
new file mode 100644
Justin M. Forbes 696046
index 0000000..28d48b0
Justin M. Forbes 696046
--- /dev/null
Justin M. Forbes 696046
+++ b/src/virtio-scsi.h
Justin M. Forbes 696046
@@ -0,0 +1,48 @@
Justin M. Forbes 696046
+#ifndef _VIRTIO_SCSI_H
Justin M. Forbes 696046
+#define _VIRTIO_SCSI_H
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+#define VIRTIO_SCSI_CDB_SIZE      32
Justin M. Forbes 696046
+#define VIRTIO_SCSI_SENSE_SIZE    96
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+struct virtio_scsi_config
Justin M. Forbes 696046
+{
Justin M. Forbes 696046
+    u32 num_queues;
Justin M. Forbes 696046
+    u32 seg_max;
Justin M. Forbes 696046
+    u32 max_sectors;
Justin M. Forbes 696046
+    u32 cmd_per_lun;
Justin M. Forbes 696046
+    u32 event_info_size;
Justin M. Forbes 696046
+    u32 sense_size;
Justin M. Forbes 696046
+    u32 cdb_size;
Justin M. Forbes 696046
+    u16 max_channel;
Justin M. Forbes 696046
+    u16 max_target;
Justin M. Forbes 696046
+    u32 max_lun;
Justin M. Forbes 696046
+} __attribute__((packed));
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+/* This is the first element of the "out" scatter-gather list. */
Justin M. Forbes 696046
+struct virtio_scsi_req_cmd {
Justin M. Forbes 696046
+    u8 lun[8];
Justin M. Forbes 696046
+    u64 id;
Justin M. Forbes 696046
+    u8 task_attr;
Justin M. Forbes 696046
+    u8 prio;
Justin M. Forbes 696046
+    u8 crn;
Justin M. Forbes 696046
+    char cdb[VIRTIO_SCSI_CDB_SIZE];
Justin M. Forbes 696046
+};
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+/* This is the first element of the "in" scatter-gather list. */
Justin M. Forbes 696046
+struct virtio_scsi_resp_cmd {
Justin M. Forbes 696046
+    u32 sense_len;
Justin M. Forbes 696046
+    u32 residual;
Justin M. Forbes 696046
+    u16 status_qualifier;
Justin M. Forbes 696046
+    u8 status;
Justin M. Forbes 696046
+    u8 response;
Justin M. Forbes 696046
+    u8 sense[VIRTIO_SCSI_SENSE_SIZE];
Justin M. Forbes 696046
+};
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+#define VIRTIO_SCSI_S_OK            0
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+struct disk_op_s;
Justin M. Forbes 696046
+int process_virtio_scsi_op(struct disk_op_s *op);
Justin M. Forbes 696046
+int virtio_scsi_cmd_data(struct disk_op_s *op, void *cdbcmd, u16 blocksize);
Justin M. Forbes 696046
+void virtio_scsi_setup(void);
Justin M. Forbes 696046
+
Justin M. Forbes 696046
+#endif /* _VIRTIO_SCSI_H */