aboutsummaryrefslogtreecommitdiff
path: root/usr.sbin/bhyve
diff options
context:
space:
mode:
authorChuck Tuffli <chuck@FreeBSD.org>2020-06-29 00:31:51 +0000
committerChuck Tuffli <chuck@FreeBSD.org>2020-06-29 00:31:51 +0000
commit206edceb73ec8024b582f9b6b739cf95dd1a8560 (patch)
tree5e47a2c95cb9efcbf42bb82b1045628adf2f130e /usr.sbin/bhyve
parenta0900f46d1ecd8a81176e790fac5e19842fe1f99 (diff)
downloadsrc-206edceb73ec8024b582f9b6b739cf95dd1a8560.tar.gz
src-206edceb73ec8024b582f9b6b739cf95dd1a8560.zip
bhyve: refactor NVMe I/O read/write
Split the NVM I/O function (i.e. nvme_opc_write_read) into separate functions - one for RAM based backing-store and another for disk based backing-store for easier maintenance. No functional changes. Tested by: Jason Tubnor MFC after: 2 weeks Differential Revision: https://reviews.freebsd.org/D24890
Notes
Notes: svn path=/head/; revision=362756
Diffstat (limited to 'usr.sbin/bhyve')
-rw-r--r--usr.sbin/bhyve/pci_nvme.c184
1 files changed, 107 insertions, 77 deletions
diff --git a/usr.sbin/bhyve/pci_nvme.c b/usr.sbin/bhyve/pci_nvme.c
index 9e7c386d6deb..a74b933f7348 100644
--- a/usr.sbin/bhyve/pci_nvme.c
+++ b/usr.sbin/bhyve/pci_nvme.c
@@ -1683,6 +1683,108 @@ nvme_opc_flush(struct pci_nvme_softc *sc,
return (pending);
}
+static uint16_t
+nvme_write_read_ram(struct pci_nvme_softc *sc,
+ struct pci_nvme_blockstore *nvstore,
+ uint64_t prp1, uint64_t prp2,
+ size_t offset, uint64_t bytes,
+ bool is_write)
+{
+ uint8_t *buf = nvstore->ctx;
+ enum nvme_copy_dir dir;
+ uint16_t status;
+
+ if (is_write)
+ dir = NVME_COPY_TO_PRP;
+ else
+ dir = NVME_COPY_FROM_PRP;
+
+ if (nvme_prp_memcpy(sc->nsc_pi->pi_vmctx, prp1, prp2,
+ buf + offset, bytes, dir))
+ pci_nvme_status_genc(&status,
+ NVME_SC_DATA_TRANSFER_ERROR);
+ else
+ pci_nvme_status_genc(&status, NVME_SC_SUCCESS);
+
+ return (status);
+}
+
+static uint16_t
+nvme_write_read_blockif(struct pci_nvme_softc *sc,
+ struct pci_nvme_blockstore *nvstore,
+ struct pci_nvme_ioreq *req,
+ uint64_t prp1, uint64_t prp2,
+ size_t offset, uint64_t bytes,
+ bool is_write)
+{
+ uint64_t size;
+ int err;
+ uint16_t status = NVME_NO_STATUS;
+
+ size = MIN(PAGE_SIZE - (prp1 % PAGE_SIZE), bytes);
+ if (pci_nvme_append_iov_req(sc, req, prp1,
+ size, is_write, offset)) {
+ pci_nvme_status_genc(&status,
+ NVME_SC_DATA_TRANSFER_ERROR);
+ goto out;
+ }
+
+ offset += size;
+ bytes -= size;
+
+ if (bytes == 0) {
+ ;
+ } else if (bytes <= PAGE_SIZE) {
+ size = bytes;
+ if (pci_nvme_append_iov_req(sc, req, prp2,
+ size, is_write, offset)) {
+ pci_nvme_status_genc(&status,
+ NVME_SC_DATA_TRANSFER_ERROR);
+ goto out;
+ }
+ } else {
+ void *vmctx = sc->nsc_pi->pi_vmctx;
+ uint64_t *prp_list = &prp2;
+ uint64_t *last = prp_list;
+
+ /* PRP2 is pointer to a physical region page list */
+ while (bytes) {
+ /* Last entry in list points to the next list */
+ if (prp_list == last) {
+ uint64_t prp = *prp_list;
+
+ prp_list = paddr_guest2host(vmctx, prp,
+ PAGE_SIZE - (prp % PAGE_SIZE));
+ last = prp_list + (NVME_PRP2_ITEMS - 1);
+ }
+
+ size = MIN(bytes, PAGE_SIZE);
+
+ if (pci_nvme_append_iov_req(sc, req, *prp_list,
+ size, is_write, offset)) {
+ pci_nvme_status_genc(&status,
+ NVME_SC_DATA_TRANSFER_ERROR);
+ goto out;
+ }
+
+ offset += size;
+ bytes -= size;
+
+ prp_list++;
+ }
+ }
+ req->io_req.br_callback = pci_nvme_io_done;
+ if (is_write)
+ err = blockif_write(nvstore->ctx, &req->io_req);
+ else
+ err = blockif_read(nvstore->ctx, &req->io_req);
+
+ if (err)
+ pci_nvme_status_genc(&status, NVME_SC_DATA_TRANSFER_ERROR);
+out:
+ return (status);
+}
+
static bool
nvme_opc_write_read(struct pci_nvme_softc *sc,
struct nvme_command *cmd,
@@ -1714,85 +1816,13 @@ nvme_opc_write_read(struct pci_nvme_softc *sc,
cmd->prp2 &= ~0x3UL;
if (nvstore->type == NVME_STOR_RAM) {
- uint8_t *buf = nvstore->ctx;
- enum nvme_copy_dir dir;
-
- if (is_write)
- dir = NVME_COPY_TO_PRP;
- else
- dir = NVME_COPY_FROM_PRP;
-
- if (nvme_prp_memcpy(sc->nsc_pi->pi_vmctx, cmd->prp1, cmd->prp2,
- buf + offset, bytes, dir))
- pci_nvme_status_genc(status,
- NVME_SC_DATA_TRANSFER_ERROR);
- else
- pci_nvme_status_genc(status, NVME_SC_SUCCESS);
+ *status = nvme_write_read_ram(sc, nvstore, cmd->prp1,
+ cmd->prp2, offset, bytes, is_write);
} else {
- uint64_t size;
- int err;
-
- size = MIN(PAGE_SIZE - (cmd->prp1 % PAGE_SIZE), bytes);
- if (pci_nvme_append_iov_req(sc, req, cmd->prp1,
- size, is_write, offset)) {
- pci_nvme_status_genc(status,
- NVME_SC_DATA_TRANSFER_ERROR);
- goto out;
- }
-
- offset += size;
- bytes -= size;
+ *status = nvme_write_read_blockif(sc, nvstore, req,
+ cmd->prp1, cmd->prp2, offset, bytes, is_write);
- if (bytes == 0) {
- ;
- } else if (bytes <= PAGE_SIZE) {
- size = bytes;
- if (pci_nvme_append_iov_req(sc, req, cmd->prp2,
- size, is_write, offset)) {
- pci_nvme_status_genc(status,
- NVME_SC_DATA_TRANSFER_ERROR);
- goto out;
- }
- } else {
- void *vmctx = sc->nsc_pi->pi_vmctx;
- uint64_t *prp_list = &cmd->prp2;
- uint64_t *last = prp_list;
-
- /* PRP2 is pointer to a physical region page list */
- while (bytes) {
- /* Last entry in list points to the next list */
- if (prp_list == last) {
- uint64_t prp = *prp_list;
-
- prp_list = paddr_guest2host(vmctx, prp,
- PAGE_SIZE - (prp % PAGE_SIZE));
- last = prp_list + (NVME_PRP2_ITEMS - 1);
- }
-
- size = MIN(bytes, PAGE_SIZE);
-
- if (pci_nvme_append_iov_req(sc, req, *prp_list,
- size, is_write, offset)) {
- pci_nvme_status_genc(status,
- NVME_SC_DATA_TRANSFER_ERROR);
- goto out;
- }
-
- offset += size;
- bytes -= size;
-
- prp_list++;
- }
- }
- req->io_req.br_callback = pci_nvme_io_done;
- if (is_write)
- err = blockif_write(nvstore->ctx, &req->io_req);
- else
- err = blockif_read(nvstore->ctx, &req->io_req);
-
- if (err)
- pci_nvme_status_genc(status, NVME_SC_DATA_TRANSFER_ERROR);
- else
+ if (*status == NVME_NO_STATUS)
pending = true;
}
out: