diff options
author | Mark Kettenis <kettenis@cvs.openbsd.org> | 2009-01-12 20:11:14 +0000 |
---|---|---|
committer | Mark Kettenis <kettenis@cvs.openbsd.org> | 2009-01-12 20:11:14 +0000 |
commit | bfa63b300f95ba373af190274515f1e2fe749876 (patch) | |
tree | 145b9844ab0ece70457f2473477309961498c82b /sys/arch/sparc64 | |
parent | acbb8396dea49cce77c5e4ab0f56536242c207c8 (diff) |
Initial stab at a driver for virtual disks found on sun4v logical domains.
Still needs some work, but reading from and writing to a disk image works.
Diffstat (limited to 'sys/arch/sparc64')
-rw-r--r-- | sys/arch/sparc64/conf/GENERIC | 4 | ||||
-rw-r--r-- | sys/arch/sparc64/conf/files.sparc64 | 7 | ||||
-rw-r--r-- | sys/arch/sparc64/dev/vdsk.c | 1077 |
3 files changed, 1086 insertions, 2 deletions
diff --git a/sys/arch/sparc64/conf/GENERIC b/sys/arch/sparc64/conf/GENERIC index de8b22911ab..123105adf95 100644 --- a/sys/arch/sparc64/conf/GENERIC +++ b/sys/arch/sparc64/conf/GENERIC @@ -1,4 +1,4 @@ -# $OpenBSD: GENERIC,v 1.228 2009/01/10 19:04:57 kettenis Exp $ +# $OpenBSD: GENERIC,v 1.229 2009/01/12 20:11:13 kettenis Exp $ # # For further information on compiling OpenBSD kernels, see the config(8) # man page. @@ -334,6 +334,8 @@ timer* at mainbus0 # Virtual devices for sun4v systems. vcons0 at vbus? vrtc0 at vbus? +vdsk* at cbus? +scsibus* at vdsk? vnet* at cbus? ## Lance Ethernet - AMD7990 diff --git a/sys/arch/sparc64/conf/files.sparc64 b/sys/arch/sparc64/conf/files.sparc64 index 77218c72937..ccf34ead7dc 100644 --- a/sys/arch/sparc64/conf/files.sparc64 +++ b/sys/arch/sparc64/conf/files.sparc64 @@ -1,4 +1,4 @@ -# $OpenBSD: files.sparc64,v 1.116 2009/01/10 17:28:48 kettenis Exp $ +# $OpenBSD: files.sparc64,v 1.117 2009/01/12 20:11:13 kettenis Exp $ # $NetBSD: files.sparc64,v 1.50 2001/08/10 20:53:50 eeh Exp $ # maxpartitions must be first item in files.${ARCH} @@ -389,6 +389,11 @@ file arch/sparc64/dev/vrtc.c vrtc define ldc file arch/sparc64/dev/ldc.c ldc +# Virtual disk +device vdsk: ldc, scsi +attach vdsk at cbus +file arch/sparc64/dev/vdsk.c vdsk + # Virtual network device vnet: ldc attach vnet at cbus diff --git a/sys/arch/sparc64/dev/vdsk.c b/sys/arch/sparc64/dev/vdsk.c new file mode 100644 index 00000000000..8e4860b6264 --- /dev/null +++ b/sys/arch/sparc64/dev/vdsk.c @@ -0,0 +1,1077 @@ +/* $OpenBSD: vdsk.c,v 1.1 2009/01/12 20:11:13 kettenis Exp $ */ +/* + * Copyright (c) 2009 Mark Kettenis + * + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +#include <sys/param.h> +#include <sys/buf.h> +#include <sys/device.h> +#include <sys/malloc.h> +#include <sys/systm.h> + +#include <machine/autoconf.h> +#include <machine/hypervisor.h> + +#include <uvm/uvm.h> + +#include <scsi/scsi_all.h> +#include <scsi/scsi_disk.h> +#include <scsi/scsiconf.h> + +#include <sparc64/dev/cbusvar.h> +#include <sparc64/dev/ldcvar.h> +#include <sparc64/dev/viovar.h> + +#ifdef VDSK_DEBUG +#define DPRINTF(x) printf x +#else +#define DPRINTF(x) +#endif + +#define VDSK_TX_ENTRIES 32 +#define VDSK_RX_ENTRIES 32 + +struct vd_attr_info { + struct vio_msg_tag tag; + uint8_t xfer_mode; + uint8_t vd_type; + uint8_t vd_mtype; + uint8_t _reserved1; + uint32_t vdisk_block_size; + uint64_t operations; + uint64_t vdisk_size; + uint64_t max_xfer_sz; + uint64_t reserved2[2]; +}; + +#define VD_DISK_TYPE_SLICE 0x01 +#define VD_DISK_TYPE_DISK 0x02 + +#define VD_MEDIA_TYPE_FIXED 0x01 +#define VD_MEDIA_TYPE_CD 0x02 +#define VD_MEDIA_TYPE_DVD 0x03 + +/* vDisk version 1.0. */ +#define VD_OP_BREAD 0x01 +#define VD_OP_BWRITE 0x02 +#define VD_OP_FLUSH 0x03 +#define VD_OP_GET_WCE 0x04 +#define VD_OP_SET_WCE 0x05 +#define VD_OP_GET_VTOC 0x06 +#define VD_OP_SET_VTOC 0x07 +#define VD_OP_GET_DISKGEOM 0x08 +#define VD_OP_SET_DISKGEOM 0x09 +#define VD_OP_GET_DEVID 0x0b +#define VD_OP_GET_EFI 0x0c +#define VD_OP_SET_EFI 0x0d + +/* vDisk version 1.1 */ +#define VD_OP_SCSICMD 0x0a +#define VD_OP_RESET 0x0e +#define VD_OP_GET_ACCESS 0x0f +#define VD_OP_SET_ACCESS 0x10 +#define VD_OP_GET_CAPACITY 0x11 + +struct vd_desc { + struct vio_dring_hdr hdr; + uint64_t req_id; + uint8_t operation; + uint8_t slice; + uint16_t _reserved1; + uint32_t status; + uint64_t offset; + uint64_t size; + uint32_t ncookies; + uint32_t _reserved2; + struct ldc_cookie cookie[MAXPHYS / PAGE_SIZE]; +}; + +#define VD_SLICE_NONE 0xff + +struct vdsk_dring { + bus_dmamap_t vd_map; + bus_dma_segment_t vd_seg; + struct vd_desc *vd_desc; + int vd_nentries; +}; + +struct vdsk_dring *vdsk_dring_alloc(bus_dma_tag_t, int); +void vdsk_dring_free(bus_dma_tag_t, struct vdsk_dring *); + +/* + * For now, we only support vDisk 1.0. + */ +#define VDSK_MAJOR 1 +#define VDSK_MINOR 0 + +struct vdsk_soft_desc { + int vsd_map_idx[MAXPHYS / PAGE_SIZE]; + struct scsi_xfer *vsd_xs; +}; + +struct vdsk_softc { + struct device sc_dv; + bus_space_tag_t sc_bustag; + bus_dma_tag_t sc_dmatag; + + void *sc_tx_ih; + void *sc_rx_ih; + + struct ldc_conn sc_lc; + + uint8_t sc_vio_state; +#define VIO_ESTABLISHED 8 + + uint32_t sc_local_sid; + uint64_t sc_dring_ident; + uint64_t sc_seq_no; + + int sc_tx_cnt; + int sc_tx_prod; + int sc_tx_cons; + + struct ldc_map *sc_lm; + struct vdsk_dring *sc_vd; + struct vdsk_soft_desc *sc_vsd; + + struct scsi_adapter sc_switch; + struct scsi_link sc_link; + + uint32_t sc_vdisk_block_size; + uint64_t sc_vdisk_size; +}; + +int vdsk_match(struct device *, void *, void *); +void vdsk_attach(struct device *, struct device *, void *); + +struct cfattach vdsk_ca = { + sizeof(struct vdsk_softc), vdsk_match, vdsk_attach +}; + +struct cfdriver vdsk_cd = { + NULL, "vdsk", DV_IFNET +}; + +struct scsi_device vdsk_device = { + NULL, NULL, NULL, NULL +}; + +int vdsk_tx_intr(void *); +int vdsk_rx_intr(void *); + +void vdsk_rx_data(struct ldc_conn *, struct ldc_pkt *); +void vdsk_rx_vio_ctrl(struct vdsk_softc *, struct vio_msg *); +void vdsk_rx_vio_ver_info(struct vdsk_softc *, struct vio_msg_tag *); +void vdsk_rx_vio_attr_info(struct vdsk_softc *, struct vio_msg_tag *); +void vdsk_rx_vio_dring_reg(struct vdsk_softc *, struct vio_msg_tag *); +void vdsk_rx_vio_rdx(struct vdsk_softc *sc, struct vio_msg_tag *); +void vdsk_rx_vio_data(struct vdsk_softc *sc, struct vio_msg *); +void vdsk_rx_vio_dring_data(struct vdsk_softc *sc, struct vio_msg_tag *); + +void vdsk_reset(struct ldc_conn *); +void vdsk_start(struct ldc_conn *); + +void vdsk_sendmsg(struct vdsk_softc *, void *, size_t); +void vdsk_send_ver_info(struct vdsk_softc *, uint16_t, uint16_t); +void vdsk_send_attr_info(struct vdsk_softc *); +void vdsk_send_dring_reg(struct vdsk_softc *); +void vdsk_send_rdx(struct vdsk_softc *); + +int vdsk_scsi_cmd(struct scsi_xfer *); +int vdsk_dev_probe(struct scsi_link *); +void vdsk_dev_free(struct scsi_link *); +int vdsk_ioctl(struct scsi_link *, u_long, caddr_t, int, struct proc *); + +int vdsk_scsi_inq(struct scsi_xfer *); +int vdsk_scsi_inquiry(struct scsi_xfer *); +int vdsk_scsi_capacity(struct scsi_xfer *); +int vdsk_scsi_done(struct scsi_xfer *, int); + +int +vdsk_match(struct device *parent, void *match, void *aux) +{ + struct cbus_attach_args *ca = aux; + + if (strcmp(ca->ca_name, "disk") == 0) + return (1); + + return (0); +} + +void +vdsk_attach(struct device *parent, struct device *self, void *aux) +{ + struct vdsk_softc *sc = (struct vdsk_softc *)self; + struct cbus_attach_args *ca = aux; + struct scsibus_attach_args saa; + struct ldc_conn *lc; + uint64_t sysino[2]; + int err, s; + int timeout; + + sc->sc_bustag = ca->ca_bustag; + sc->sc_dmatag = ca->ca_dmatag; + + if (cbus_intr_map(ca->ca_node, ca->ca_tx_ino, &sysino[0]) || + cbus_intr_map(ca->ca_node, ca->ca_rx_ino, &sysino[1])) { + printf(": can't map interrupt\n"); + return; + } + printf(": ivec 0x%lx, 0x%lx", sysino[0], sysino[1]); + + /* + * Un-configure queues before registering interrupt handlers, + * such that we dont get any stale LDC packets or events. + */ + hv_ldc_tx_qconf(ca->ca_id, 0, 0); + hv_ldc_rx_qconf(ca->ca_id, 0, 0); + + sc->sc_tx_ih = bus_intr_establish(ca->ca_bustag, sysino[0], IPL_BIO, + 0, vdsk_tx_intr, sc, sc->sc_dv.dv_xname); + sc->sc_rx_ih = bus_intr_establish(ca->ca_bustag, sysino[1], IPL_BIO, + 0, vdsk_rx_intr, sc, sc->sc_dv.dv_xname); + if (sc->sc_tx_ih == NULL || sc->sc_rx_ih == NULL) { + printf(", can't establish interrupt\n"); + return; + } + + lc = &sc->sc_lc; + lc->lc_id = ca->ca_id; + lc->lc_sc = sc; + lc->lc_reset = vdsk_reset; + lc->lc_start = vdsk_start; + lc->lc_rx_data = vdsk_rx_data; + + lc->lc_txq = ldc_queue_alloc(sc->sc_dmatag, VDSK_TX_ENTRIES); + if (lc->lc_txq == NULL) { + printf(", can't allocate tx queue\n"); + return; + } + + lc->lc_rxq = ldc_queue_alloc(sc->sc_dmatag, VDSK_RX_ENTRIES); + if (lc->lc_rxq == NULL) { + printf(", can't allocate rx queue\n"); + goto free_txqueue; + } + + sc->sc_lm = ldc_map_alloc(sc->sc_dmatag, 2048); + if (sc->sc_lm == NULL) { + printf(", can't allocate LDC mapping table\n"); + goto free_rxqueue; + } + + err = hv_ldc_set_map_table(lc->lc_id, + sc->sc_lm->lm_map->dm_segs[0].ds_addr, sc->sc_lm->lm_nentries); + if (err != H_EOK) { + printf("hv_ldc_set_map_table %d\n", err); + goto free_map; + } + + sc->sc_vd = vdsk_dring_alloc(sc->sc_dmatag, 32); + if (sc->sc_vd == NULL) { + printf(", can't allocate dring\n"); + goto free_map; + } + sc->sc_vsd = malloc(32 * sizeof(*sc->sc_vsd), M_DEVBUF, M_NOWAIT); + if (sc->sc_vsd == NULL) { + printf(", can't allocate software ring\n"); + goto free_dring; + } + + sc->sc_lm->lm_slot[0].entry = sc->sc_vd->vd_map->dm_segs[0].ds_addr; + sc->sc_lm->lm_slot[0].entry &= LDC_MTE_RA_MASK; + sc->sc_lm->lm_slot[0].entry |= LDC_MTE_CPR | LDC_MTE_CPW; + sc->sc_lm->lm_slot[0].entry |= LDC_MTE_R | LDC_MTE_W; + sc->sc_lm->lm_next = 1; + sc->sc_lm->lm_count = 1; + + err = hv_ldc_tx_qconf(lc->lc_id, + lc->lc_txq->lq_map->dm_segs[0].ds_addr, lc->lc_txq->lq_nentries); + if (err != H_EOK) + printf("hv_ldc_tx_qconf %d\n", err); + + err = hv_ldc_rx_qconf(lc->lc_id, + lc->lc_rxq->lq_map->dm_segs[0].ds_addr, lc->lc_rxq->lq_nentries); + if (err != H_EOK) + printf("hv_ldc_rx_qconf %d\n", err); + + ldc_send_vers(lc); + + printf("\n"); + + /* + * Interrupts aren't enabled during autoconf, so poll for VIO + * peer-to-peer hanshake completion. + */ + s = splbio(); + timeout = 1000; + do { + if (vdsk_rx_intr(sc) && sc->sc_vio_state == VIO_ESTABLISHED) + break; + + delay(1000); + } while(--timeout > 0); + splx(s); + + if (sc->sc_vio_state != VIO_ESTABLISHED) + return; + + sc->sc_switch.scsi_cmd = vdsk_scsi_cmd; + sc->sc_switch.scsi_minphys = minphys; + sc->sc_switch.dev_probe = vdsk_dev_probe; + sc->sc_switch.dev_free = vdsk_dev_free; + sc->sc_switch.ioctl = vdsk_ioctl; + + sc->sc_link.device = &vdsk_device; + sc->sc_link.adapter = &sc->sc_switch; + sc->sc_link.adapter_softc = self; + sc->sc_link.adapter_buswidth = 2; + sc->sc_link.luns = 1; /* XXX slices should be presented as luns? */ + sc->sc_link.adapter_target = 2; + sc->sc_link.openings = 32; + + bzero(&saa, sizeof(saa)); + saa.saa_sc_link = &sc->sc_link; + config_found(self, &saa, scsiprint); + + return; + +free_dring: + vdsk_dring_free(sc->sc_dmatag, sc->sc_vd); +free_map: + hv_ldc_set_map_table(lc->lc_id, 0, 0); + ldc_map_free(sc->sc_dmatag, sc->sc_lm); +free_rxqueue: + ldc_queue_free(sc->sc_dmatag, lc->lc_rxq); +free_txqueue: + ldc_queue_free(sc->sc_dmatag, lc->lc_txq); + return; +} + +int +vdsk_tx_intr(void *arg) +{ + struct vdsk_softc *sc = arg; + struct ldc_conn *lc = &sc->sc_lc; + uint64_t tx_head, tx_tail, tx_state; + + hv_ldc_tx_get_state(lc->lc_id, &tx_head, &tx_tail, &tx_state); + if (tx_state != lc->lc_tx_state) { + switch (tx_state) { + case LDC_CHANNEL_DOWN: + DPRINTF(("Tx link down\n")); + break; + case LDC_CHANNEL_UP: + DPRINTF(("Tx link up\n")); + break; + case LDC_CHANNEL_RESET: + DPRINTF(("Tx link reset\n")); + break; + } + lc->lc_tx_state = tx_state; + } + + return (1); +} + +int +vdsk_rx_intr(void *arg) +{ + struct vdsk_softc *sc = arg; + struct ldc_conn *lc = &sc->sc_lc; + uint64_t rx_head, rx_tail, rx_state; + struct ldc_pkt *lp; + uint64_t *msg; + int err; + + err = hv_ldc_rx_get_state(lc->lc_id, &rx_head, &rx_tail, &rx_state); + if (err == H_EINVAL) + return (0); + if (err != H_EOK) { + printf("hv_ldc_rx_get_state %d\n", err); + return (0); + } + + if (rx_state != lc->lc_rx_state) { + sc->sc_tx_cnt = sc->sc_tx_prod = sc->sc_tx_cons = 0; + sc->sc_vio_state = 0; + lc->lc_tx_seqid = 0; + lc->lc_state = 0; + switch (rx_state) { + case LDC_CHANNEL_DOWN: + DPRINTF(("Rx link down\n")); + break; + case LDC_CHANNEL_UP: + DPRINTF(("Rx link up\n")); + ldc_send_vers(lc); + break; + case LDC_CHANNEL_RESET: + DPRINTF(("Rx link reset\n")); + break; + } + lc->lc_rx_state = rx_state; + hv_ldc_rx_set_qhead(lc->lc_id, rx_tail); + return (1); + } + + if (rx_head == rx_tail) + return (0); + + msg = (uint64_t *)(lc->lc_rxq->lq_va + rx_head); +#if 0 +{ + int i; + + printf("%s: rx intr, head %lx, tail %lx\n", sc->sc_dv.dv_xname, + rx_head, rx_tail); + for (i = 0; i < 8; i++) + printf("word %d: 0x%016lx\n", i, msg[i]); +} +#endif + lp = (struct ldc_pkt *)(lc->lc_rxq->lq_va + rx_head); + switch (lp->type) { + case LDC_CTRL: + ldc_rx_ctrl(lc, lp); + break; + + case LDC_DATA: + ldc_rx_data(lc, lp); + break; + + default: + DPRINTF(("%0x02/%0x02/%0x02\n", lp->type, lp->stype, + lp->ctrl)); + ldc_reset(lc); + break; + } + + if (lc->lc_state == 0) + return (1); + + rx_head += sizeof(*lp); + rx_head &= ((lc->lc_rxq->lq_nentries * sizeof(*lp)) - 1); + err = hv_ldc_rx_set_qhead(lc->lc_id, rx_head); + if (err != H_EOK) + printf("%s: hv_ldc_rx_set_qhead %d\n", __func__, err); + + return (1); +} + +void +vdsk_rx_data(struct ldc_conn *lc, struct ldc_pkt *lp) +{ + struct vio_msg *vm = (struct vio_msg *)lp; + + switch (vm->type) { + case VIO_TYPE_CTRL: + if ((lp->env & LDC_FRAG_START) == 0 && + (lp->env & LDC_FRAG_STOP) == 0) + return; + vdsk_rx_vio_ctrl(lc->lc_sc, vm); + break; + + case VIO_TYPE_DATA: + if((lp->env & LDC_FRAG_START) == 0) + return; + vdsk_rx_vio_data(lc->lc_sc, vm); + break; + + default: + DPRINTF(("Unhandled packet type 0x%02x\n", vm->type)); + ldc_reset(lc); + break; + } +} + +void +vdsk_rx_vio_ctrl(struct vdsk_softc *sc, struct vio_msg *vm) +{ + struct vio_msg_tag *tag = (struct vio_msg_tag *)&vm->type; + + switch (tag->stype_env) { + case VIO_VER_INFO: + vdsk_rx_vio_ver_info(sc, tag); + break; + case VIO_ATTR_INFO: + vdsk_rx_vio_attr_info(sc, tag); + break; + case VIO_DRING_REG: + vdsk_rx_vio_dring_reg(sc, tag); + break; + case VIO_RDX: + vdsk_rx_vio_rdx(sc, tag); + break; + default: + DPRINTF(("CTRL/0x%02x/0x%04x\n", tag->stype, tag->stype_env)); + break; + } +} + +void +vdsk_rx_vio_ver_info(struct vdsk_softc *sc, struct vio_msg_tag *tag) +{ + struct vio_ver_info *vi = (struct vio_ver_info *)tag; + + switch (vi->tag.stype) { + case VIO_SUBTYPE_INFO: + DPRINTF(("CTRL/INFO/VER_INFO\n")); + break; + + case VIO_SUBTYPE_ACK: + DPRINTF(("CTRL/ACK/VER_INFO\n")); + vdsk_send_attr_info(sc); + break; + + default: + DPRINTF(("CTRL/0x%02x/VER_INFO\n", vi->tag.stype)); + break; + } +} + +void +vdsk_rx_vio_attr_info(struct vdsk_softc *sc, struct vio_msg_tag *tag) +{ + struct vd_attr_info *ai = (struct vd_attr_info *)tag; + + switch (ai->tag.stype) { + case VIO_SUBTYPE_INFO: + DPRINTF(("CTRL/INFO/ATTR_INFO\n")); + break; + + case VIO_SUBTYPE_ACK: + DPRINTF(("CTRL/ACK/ATTR_INFO\n")); + sc->sc_vdisk_block_size = ai->vdisk_block_size; + sc->sc_vdisk_size = ai->vdisk_size; + vdsk_send_dring_reg(sc); + break; + + default: + DPRINTF(("CTRL/0x%02x/ATTR_INFO\n", ai->tag.stype)); + break; + } +} + +void +vdsk_rx_vio_dring_reg(struct vdsk_softc *sc, struct vio_msg_tag *tag) +{ + struct vio_dring_reg *dr = (struct vio_dring_reg *)tag; + + switch (dr->tag.stype) { + case VIO_SUBTYPE_INFO: + DPRINTF(("CTRL/INFO/DRING_REG\n")); + break; + + case VIO_SUBTYPE_ACK: + DPRINTF(("CTRL/ACK/DRING_REG\n")); + + sc->sc_dring_ident = dr->dring_ident; + sc->sc_seq_no = 1; + + vdsk_send_rdx(sc); + break; + + default: + DPRINTF(("CTRL/0x%02x/DRING_REG\n", dr->tag.stype)); + break; + } +} + +void +vdsk_rx_vio_rdx(struct vdsk_softc *sc, struct vio_msg_tag *tag) +{ + switch(tag->stype) { + case VIO_SUBTYPE_INFO: + DPRINTF(("CTRL/INFO/RDX\n")); + break; + + case VIO_SUBTYPE_ACK: + DPRINTF(("CTRL/ACK/RDX\n")); + + /* Link is up! */ + sc->sc_vio_state = VIO_ESTABLISHED; + break; + + default: + DPRINTF(("CTRL/0x%02x/RDX (VIO)\n", tag->stype)); + break; + } +} + +void +vdsk_rx_vio_data(struct vdsk_softc *sc, struct vio_msg *vm) +{ + struct vio_msg_tag *tag = (struct vio_msg_tag *)&vm->type; + + if (sc->sc_vio_state != VIO_ESTABLISHED) { + DPRINTF(("Spurious DATA/0x%02x/0x%04x\n", tag->stype, + tag->stype_env)); + return; + } + + switch(tag->stype_env) { + case VIO_DRING_DATA: + vdsk_rx_vio_dring_data(sc, tag); + break; + + default: + DPRINTF(("DATA/0x%02x/0x%04x\n", tag->stype, tag->stype_env)); + break; + } +} + +void +vdsk_rx_vio_dring_data(struct vdsk_softc *sc, struct vio_msg_tag *tag) +{ + switch(tag->stype) { + case VIO_SUBTYPE_INFO: + DPRINTF(("DATA/INFO/DRING_DATA\n")); + break; + + case VIO_SUBTYPE_ACK: + { + struct scsi_xfer *xs; + struct ldc_map *map = sc->sc_lm; + int cons, error; + int cookie, idx; + int len; + + cons = sc->sc_tx_cons; + while (sc->sc_vd->vd_desc[cons].hdr.dstate == VIO_DESC_DONE) { + xs = sc->sc_vsd[cons].vsd_xs; + + cookie = 0; + len = xs->datalen; + while (len > 0) { + idx = sc->sc_vsd[cons].vsd_map_idx[cookie++]; + map->lm_slot[idx].entry = 0; + map->lm_count--; + len -= PAGE_SIZE; + } + + error = XS_NOERROR; + if (sc->sc_vd->vd_desc[cons].status != 0) + error = XS_DRIVER_STUFFUP; + xs->resid = xs->datalen - + sc->sc_vd->vd_desc[cons].size; + vdsk_scsi_done(xs, error); + + sc->sc_vd->vd_desc[cons++].hdr.dstate = VIO_DESC_FREE; + cons &= (sc->sc_vd->vd_nentries - 1); + sc->sc_tx_cnt--; + } + sc->sc_tx_cons = cons; + break; + } + + case VIO_SUBTYPE_NACK: + DPRINTF(("DATA/NACK/DRING_DATA\n")); + printf("state 0x%02x\n", sc->sc_vd->vd_desc[0].hdr.dstate); + printf("status 0x%04x\n", sc->sc_vd->vd_desc[0].status); + printf("size 0x%04x\n", sc->sc_vd->vd_desc[0].size); + break; + + default: + DPRINTF(("DATA/0x%02x/DRING_DATA\n", tag->stype)); + break; + } +} + +void +vdsk_reset(struct ldc_conn *lc) +{ + struct vdsk_softc *sc = lc->lc_sc; + + sc->sc_tx_cnt = sc->sc_tx_prod = sc->sc_tx_cons = 0; + sc->sc_vio_state = 0; +} + +void +vdsk_start(struct ldc_conn *lc) +{ + struct vdsk_softc *sc = lc->lc_sc; + + vdsk_send_ver_info(sc, VDSK_MAJOR, VDSK_MINOR); +} + +void +vdsk_sendmsg(struct vdsk_softc *sc, void *msg, size_t len) +{ + struct ldc_conn *lc = &sc->sc_lc; + struct ldc_pkt *lp; + uint64_t tx_head, tx_tail, tx_state; + int err; + + err = hv_ldc_tx_get_state(lc->lc_id, &tx_head, &tx_tail, &tx_state); + if (err != H_EOK) + return; + + lp = (struct ldc_pkt *)(lc->lc_txq->lq_va + tx_tail); + bzero(lp, sizeof(struct ldc_pkt)); + lp->type = LDC_DATA; + lp->stype = LDC_INFO; + KASSERT((len & ~LDC_LEN_MASK) == 0); + lp->env = len | LDC_FRAG_STOP | LDC_FRAG_START; + lp->seqid = lc->lc_tx_seqid++; + bcopy(msg, &lp->major, len); + + tx_tail += sizeof(*lp); + tx_tail &= ((lc->lc_txq->lq_nentries * sizeof(*lp)) - 1); + err = hv_ldc_tx_set_qtail(lc->lc_id, tx_tail); + if (err != H_EOK) + printf("hv_ldc_tx_set_qtail: %d\n", err); +} + +void +vdsk_send_ver_info(struct vdsk_softc *sc, uint16_t major, uint16_t minor) +{ + struct vio_ver_info vi; + + /* Allocate new session ID. */ + sc->sc_local_sid = tick(); + + bzero(&vi, sizeof(vi)); + vi.tag.type = VIO_TYPE_CTRL; + vi.tag.stype = VIO_SUBTYPE_INFO; + vi.tag.stype_env = VIO_VER_INFO; + vi.tag.sid = sc->sc_local_sid; + vi.major = major; + vi.minor = minor; + vi.dev_class = VDEV_DISK; + vdsk_sendmsg(sc, &vi, sizeof(vi)); +} + +void +vdsk_send_attr_info(struct vdsk_softc *sc) +{ + struct vd_attr_info ai; + + bzero(&ai, sizeof(ai)); + ai.tag.type = VIO_TYPE_CTRL; + ai.tag.stype = VIO_SUBTYPE_INFO; + ai.tag.stype_env = VIO_ATTR_INFO; + ai.tag.sid = sc->sc_local_sid; + ai.xfer_mode = VIO_DRING_MODE; + ai.vdisk_block_size = DEV_BSIZE; + ai.max_xfer_sz = MAXPHYS / DEV_BSIZE; + vdsk_sendmsg(sc, &ai, sizeof(ai)); +} + +void +vdsk_send_dring_reg(struct vdsk_softc *sc) +{ + struct vio_dring_reg dr; + + bzero(&dr, sizeof(dr)); + dr.tag.type = VIO_TYPE_CTRL; + dr.tag.stype = VIO_SUBTYPE_INFO; + dr.tag.stype_env = VIO_DRING_REG; + dr.tag.sid = sc->sc_local_sid; + dr.dring_ident = 0; + dr.num_descriptors = sc->sc_vd->vd_nentries; + dr.descriptor_size = sizeof(struct vd_desc); + dr.options = VIO_TX_RING | VIO_RX_RING; + dr.ncookies = 1; + dr.cookie[0].addr = 0; + dr.cookie[0].size = PAGE_SIZE; + vdsk_sendmsg(sc, &dr, sizeof(dr)); +}; + +void +vdsk_send_rdx(struct vdsk_softc *sc) +{ + struct vio_rdx rdx; + + bzero(&rdx, sizeof(rdx)); + rdx.tag.type = VIO_TYPE_CTRL; + rdx.tag.stype = VIO_SUBTYPE_INFO; + rdx.tag.stype_env = VIO_RDX; + rdx.tag.sid = sc->sc_local_sid; + vdsk_sendmsg(sc, &rdx, sizeof(rdx)); +} + +struct vdsk_dring * +vdsk_dring_alloc(bus_dma_tag_t t, int nentries) +{ + struct vdsk_dring *vd; + bus_size_t size; + caddr_t va; + int nsegs; + int i; + + vd = malloc(sizeof(struct vdsk_dring), M_DEVBUF, M_NOWAIT); + if (vd == NULL) + return NULL; + + size = roundup(nentries * sizeof(struct vd_desc), PAGE_SIZE); + + if (bus_dmamap_create(t, size, 1, size, 0, + BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW, &vd->vd_map) != 0) + return (NULL); + + if (bus_dmamem_alloc(t, size, PAGE_SIZE, 0, &vd->vd_seg, 1, + &nsegs, BUS_DMA_NOWAIT) != 0) + goto destroy; + + if (bus_dmamem_map(t, &vd->vd_seg, 1, size, &va, + BUS_DMA_NOWAIT) != 0) + goto free; + + if (bus_dmamap_load(t, vd->vd_map, va, size, NULL, + BUS_DMA_NOWAIT) != 0) + goto unmap; + + vd->vd_desc = (struct vd_desc *)va; + vd->vd_nentries = nentries; + bzero(vd->vd_desc, nentries * sizeof(struct vd_desc)); + for (i = 0; i < vd->vd_nentries; i++) + vd->vd_desc[i].hdr.dstate = VIO_DESC_FREE; + return (vd); + +unmap: + bus_dmamem_unmap(t, va, size); +free: + bus_dmamem_free(t, &vd->vd_seg, 1); +destroy: + bus_dmamap_destroy(t, vd->vd_map); + + return (NULL); +} + +void +vdsk_dring_free(bus_dma_tag_t t, struct vdsk_dring *vd) +{ + bus_size_t size; + + size = vd->vd_nentries * sizeof(struct vd_desc); + size = roundup(size, PAGE_SIZE); + + bus_dmamap_unload(t, vd->vd_map); + bus_dmamem_unmap(t, (caddr_t)vd->vd_desc, size); + bus_dmamem_free(t, &vd->vd_seg, 1); + bus_dmamap_destroy(t, vd->vd_map); + free(vd, M_DEVBUF); +} + +int +vdsk_scsi_cmd(struct scsi_xfer *xs) +{ + struct scsi_rw *rw; + struct scsi_rw_big *rwb; + u_int64_t lba; + u_int32_t sector_count; + uint8_t operation; + + switch (xs->cmd->opcode) { + case READ_BIG: + case READ_COMMAND: + operation = VD_OP_BREAD; + break; + case WRITE_BIG: + case WRITE_COMMAND: + operation = VD_OP_BWRITE; + break; + + case INQUIRY: + return (vdsk_scsi_inq(xs)); + case READ_CAPACITY: + return (vdsk_scsi_capacity(xs)); + + case TEST_UNIT_READY: + case START_STOP: + case PREVENT_ALLOW: + return (vdsk_scsi_done(xs, XS_NOERROR)); + + default: + printf("%s cmd 0x%02x\n", __func__, xs->cmd->opcode); + case MODE_SENSE: + case MODE_SENSE_BIG: + case REPORT_LUNS: + return (vdsk_scsi_done(xs, XS_DRIVER_STUFFUP)); + } + + if (xs->cmdlen == 6) { + rw = (struct scsi_rw *)xs->cmd; + lba = _3btol(rw->addr) & (SRW_TOPADDR << 16 | 0xffff); + sector_count = rw->length ? rw->length : 0x100; + } else { + rwb = (struct scsi_rw_big *)xs->cmd; + lba = _4btol(rwb->addr); + sector_count = _2btol(rwb->length); + } + +{ + struct vdsk_softc *sc = xs->sc_link->adapter_softc; + struct ldc_map *map = sc->sc_lm; + struct vio_dring_msg dm; + vaddr_t va; + paddr_t pa; + int len, ncookies; + int desc; + + desc = sc->sc_tx_prod; + + ncookies = 0; + len = xs->datalen; + va = (vaddr_t)xs->data; + while (len > 0) { + pmap_extract(pmap_kernel(), va, &pa); + while (map->lm_slot[map->lm_next].entry != 0) { + map->lm_next++; + map->lm_next &= (map->lm_nentries - 1); + } + /* Don't take slot 0; it's used by our descriptor ring. */ + if (map->lm_next == 0) + map->lm_next++; + map->lm_slot[map->lm_next].entry = (pa & LDC_MTE_RA_MASK); + map->lm_slot[map->lm_next].entry |= LDC_MTE_CPR | LDC_MTE_CPW; + map->lm_slot[map->lm_next].entry |= LDC_MTE_IOR | LDC_MTE_IOW; + map->lm_slot[map->lm_next].entry |= LDC_MTE_R | LDC_MTE_W; + map->lm_count++; + + sc->sc_vd->vd_desc[desc].cookie[ncookies].addr = + map->lm_next << PAGE_SHIFT | (pa & PAGE_MASK); + sc->sc_vd->vd_desc[desc].cookie[ncookies].size = + min(len, PAGE_SIZE); + + sc->sc_vsd[desc].vsd_map_idx[ncookies] = map->lm_next; + va += PAGE_SIZE; + len -= PAGE_SIZE; + } + + sc->sc_vd->vd_desc[desc].hdr.ack = 1; + sc->sc_vd->vd_desc[desc].operation = operation; + sc->sc_vd->vd_desc[desc].slice = VD_SLICE_NONE; + sc->sc_vd->vd_desc[desc].status = 0xffffffff; + sc->sc_vd->vd_desc[desc].offset = lba; + sc->sc_vd->vd_desc[desc].size = xs->datalen; + sc->sc_vd->vd_desc[desc].ncookies = 1; + membar(Sync); + sc->sc_vd->vd_desc[desc].hdr.dstate = VIO_DESC_READY; + + sc->sc_vsd[desc].vsd_xs = xs; + + desc++; + desc &= (sc->sc_vd->vd_nentries - 1); + sc->sc_tx_cnt++; + + bzero(&dm, sizeof(dm)); + dm.tag.type = VIO_TYPE_DATA; + dm.tag.stype = VIO_SUBTYPE_INFO; + dm.tag.stype_env = VIO_DRING_DATA; + dm.tag.sid = sc->sc_local_sid; + dm.seq_no = sc->sc_seq_no++; + dm.dring_ident = sc->sc_dring_ident; + dm.start_idx = sc->sc_tx_prod; + dm.end_idx = sc->sc_tx_prod; + vdsk_sendmsg(sc, &dm, sizeof(dm)); + + sc->sc_tx_prod = desc; +} + + return (SUCCESSFULLY_QUEUED); +} + +int +vdsk_scsi_inq(struct scsi_xfer *xs) +{ + struct scsi_inquiry *inq = (struct scsi_inquiry *)xs->cmd; + + if (ISSET(inq->flags, SI_EVPD)) + return (vdsk_scsi_done(xs, XS_DRIVER_STUFFUP)); + + return (vdsk_scsi_inquiry(xs)); +} + +int +vdsk_scsi_inquiry(struct scsi_xfer *xs) +{ + struct scsi_inquiry_data inq; + + bzero(&inq, sizeof(inq)); + + inq.device = T_DIRECT; + inq.version = 0x05; /* SPC-3 */ + inq.response_format = 2; + inq.additional_length = 32; + bcopy("SUN ", inq.vendor, sizeof(inq.vendor)); + bcopy("Virtual Disk ", inq.product, sizeof(inq.product)); + bcopy("1.0 ", inq.revision, sizeof(inq.revision)); + + bcopy(&inq, xs->data, MIN(sizeof(inq), xs->datalen)); + + return (vdsk_scsi_done(xs, XS_NOERROR)); +} + +int +vdsk_scsi_capacity(struct scsi_xfer *xs) +{ + struct vdsk_softc *sc = xs->sc_link->adapter_softc; + struct scsi_read_cap_data rcd; + uint64_t capacity; + + bzero(&rcd, sizeof(rcd)); + + capacity = sc->sc_vdisk_size; + if (capacity > 0xffffffff) + capacity = 0xffffffff; + + _lto4b(capacity - 1, rcd.addr); + _lto4b(sc->sc_vdisk_block_size, rcd.length); + + bcopy(&rcd, xs->data, MIN(sizeof(rcd), xs->datalen)); + + return (vdsk_scsi_done(xs, XS_NOERROR)); +} + +int +vdsk_scsi_done(struct scsi_xfer *xs, int error) +{ + int s; + + xs->error = error; + xs->flags |= ITSDONE; + + s = splbio(); + scsi_done(xs); + splx(s); + return (COMPLETE); +} + +int +vdsk_dev_probe(struct scsi_link *link) +{ + KASSERT(link->scsibus == 0); + KASSERT(link->lun == 0); + + if (link->target == 0) + return (0); + + return (ENODEV); +} + +void +vdsk_dev_free(struct scsi_link *link) +{ + printf("%s\n", __func__); +} + +int +vdsk_ioctl(struct scsi_link *link, u_long cmd, caddr_t addr, int flags, + struct proc *p) +{ + printf("%s\n", __func__); + return (ENOTTY); +} + |