| /* |
| * Virtio 9p backend |
| * |
| * Copyright IBM, Corp. 2010 |
| * |
| * Authors: |
| * Anthony Liguori <aliguori@us.ibm.com> |
| * |
| * This work is licensed under the terms of the GNU GPL, version 2. See |
| * the COPYING file in the top-level directory. |
| * |
| */ |
| |
| /* |
| * Not so fast! You might want to read the 9p developer docs first: |
| * https://wiki.qemu.org/Documentation/9p |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "hw/virtio/virtio.h" |
| #include "qemu/sockets.h" |
| #include "virtio-9p.h" |
| #include "fsdev/qemu-fsdev.h" |
| #include "coth.h" |
| #include "hw/qdev-properties.h" |
| #include "hw/virtio/virtio-access.h" |
| #include "qemu/iov.h" |
| #include "qemu/module.h" |
| #include "sysemu/qtest.h" |
| |
| static void virtio_9p_push_and_notify(V9fsPDU *pdu) |
| { |
| V9fsState *s = pdu->s; |
| V9fsVirtioState *v = container_of(s, V9fsVirtioState, state); |
| VirtQueueElement *elem = v->elems[pdu->idx]; |
| |
| /* push onto queue and notify */ |
| virtqueue_push(v->vq, elem, pdu->size); |
| g_free(elem); |
| v->elems[pdu->idx] = NULL; |
| |
| /* FIXME: we should batch these completions */ |
| virtio_notify(VIRTIO_DEVICE(v), v->vq); |
| } |
| |
| static void handle_9p_output(VirtIODevice *vdev, VirtQueue *vq) |
| { |
| V9fsVirtioState *v = (V9fsVirtioState *)vdev; |
| V9fsState *s = &v->state; |
| V9fsPDU *pdu; |
| ssize_t len; |
| VirtQueueElement *elem; |
| |
| while ((pdu = pdu_alloc(s))) { |
| P9MsgHeader out; |
| |
| elem = virtqueue_pop(vq, sizeof(VirtQueueElement)); |
| if (!elem) { |
| goto out_free_pdu; |
| } |
| |
| if (iov_size(elem->in_sg, elem->in_num) < 7) { |
| virtio_error(vdev, |
| "The guest sent a VirtFS request without space for " |
| "the reply"); |
| goto out_free_req; |
| } |
| |
| len = iov_to_buf(elem->out_sg, elem->out_num, 0, &out, 7); |
| if (len != 7) { |
| virtio_error(vdev, "The guest sent a malformed VirtFS request: " |
| "header size is %zd, should be 7", len); |
| goto out_free_req; |
| } |
| |
| v->elems[pdu->idx] = elem; |
| |
| pdu_submit(pdu, &out); |
| } |
| |
| return; |
| |
| out_free_req: |
| virtqueue_detach_element(vq, elem, 0); |
| g_free(elem); |
| out_free_pdu: |
| pdu_free(pdu); |
| } |
| |
| static uint64_t virtio_9p_get_features(VirtIODevice *vdev, uint64_t features, |
| Error **errp) |
| { |
| virtio_add_feature(&features, VIRTIO_9P_MOUNT_TAG); |
| return features; |
| } |
| |
| static void virtio_9p_get_config(VirtIODevice *vdev, uint8_t *config) |
| { |
| int len; |
| struct virtio_9p_config *cfg; |
| V9fsVirtioState *v = VIRTIO_9P(vdev); |
| V9fsState *s = &v->state; |
| |
| len = strlen(s->tag); |
| cfg = g_malloc0(sizeof(struct virtio_9p_config) + len); |
| virtio_stw_p(vdev, &cfg->tag_len, len); |
| /* We don't copy the terminating null to config space */ |
| memcpy(cfg->tag, s->tag, len); |
| memcpy(config, cfg, v->config_size); |
| g_free(cfg); |
| } |
| |
| static void virtio_9p_reset(VirtIODevice *vdev) |
| { |
| V9fsVirtioState *v = (V9fsVirtioState *)vdev; |
| |
| v9fs_reset(&v->state); |
| } |
| |
| static ssize_t virtio_pdu_vmarshal(V9fsPDU *pdu, size_t offset, |
| const char *fmt, va_list ap) |
| { |
| V9fsState *s = pdu->s; |
| V9fsVirtioState *v = container_of(s, V9fsVirtioState, state); |
| VirtQueueElement *elem = v->elems[pdu->idx]; |
| ssize_t ret; |
| |
| ret = v9fs_iov_vmarshal(elem->in_sg, elem->in_num, offset, 1, fmt, ap); |
| if (ret < 0) { |
| VirtIODevice *vdev = VIRTIO_DEVICE(v); |
| |
| virtio_error(vdev, "Failed to encode VirtFS reply type %d", |
| pdu->id + 1); |
| } |
| return ret; |
| } |
| |
| static ssize_t virtio_pdu_vunmarshal(V9fsPDU *pdu, size_t offset, |
| const char *fmt, va_list ap) |
| { |
| V9fsState *s = pdu->s; |
| V9fsVirtioState *v = container_of(s, V9fsVirtioState, state); |
| VirtQueueElement *elem = v->elems[pdu->idx]; |
| ssize_t ret; |
| |
| ret = v9fs_iov_vunmarshal(elem->out_sg, elem->out_num, offset, 1, fmt, ap); |
| if (ret < 0) { |
| VirtIODevice *vdev = VIRTIO_DEVICE(v); |
| |
| virtio_error(vdev, "Failed to decode VirtFS request type %d", pdu->id); |
| } |
| return ret; |
| } |
| |
| static void virtio_init_in_iov_from_pdu(V9fsPDU *pdu, struct iovec **piov, |
| unsigned int *pniov, size_t size) |
| { |
| V9fsState *s = pdu->s; |
| V9fsVirtioState *v = container_of(s, V9fsVirtioState, state); |
| VirtQueueElement *elem = v->elems[pdu->idx]; |
| size_t buf_size = iov_size(elem->in_sg, elem->in_num); |
| |
| if (buf_size < size) { |
| VirtIODevice *vdev = VIRTIO_DEVICE(v); |
| |
| virtio_error(vdev, |
| "VirtFS reply type %d needs %zu bytes, buffer has %zu", |
| pdu->id + 1, size, buf_size); |
| } |
| |
| *piov = elem->in_sg; |
| *pniov = elem->in_num; |
| } |
| |
| static void virtio_init_out_iov_from_pdu(V9fsPDU *pdu, struct iovec **piov, |
| unsigned int *pniov, size_t size) |
| { |
| V9fsState *s = pdu->s; |
| V9fsVirtioState *v = container_of(s, V9fsVirtioState, state); |
| VirtQueueElement *elem = v->elems[pdu->idx]; |
| size_t buf_size = iov_size(elem->out_sg, elem->out_num); |
| |
| if (buf_size < size) { |
| VirtIODevice *vdev = VIRTIO_DEVICE(v); |
| |
| virtio_error(vdev, |
| "VirtFS request type %d needs %zu bytes, buffer has %zu", |
| pdu->id, size, buf_size); |
| } |
| |
| *piov = elem->out_sg; |
| *pniov = elem->out_num; |
| } |
| |
| static const V9fsTransport virtio_9p_transport = { |
| .pdu_vmarshal = virtio_pdu_vmarshal, |
| .pdu_vunmarshal = virtio_pdu_vunmarshal, |
| .init_in_iov_from_pdu = virtio_init_in_iov_from_pdu, |
| .init_out_iov_from_pdu = virtio_init_out_iov_from_pdu, |
| .push_and_notify = virtio_9p_push_and_notify, |
| }; |
| |
| static void virtio_9p_device_realize(DeviceState *dev, Error **errp) |
| { |
| VirtIODevice *vdev = VIRTIO_DEVICE(dev); |
| V9fsVirtioState *v = VIRTIO_9P(dev); |
| V9fsState *s = &v->state; |
| FsDriverEntry *fse = get_fsdev_fsentry(s->fsconf.fsdev_id); |
| |
| if (qtest_enabled() && fse) { |
| fse->export_flags |= V9FS_NO_PERF_WARN; |
| } |
| |
| if (v9fs_device_realize_common(s, &virtio_9p_transport, errp)) { |
| return; |
| } |
| |
| v->config_size = sizeof(struct virtio_9p_config) + strlen(s->fsconf.tag); |
| virtio_init(vdev, VIRTIO_ID_9P, v->config_size); |
| v->vq = virtio_add_queue(vdev, MAX_REQ, handle_9p_output); |
| } |
| |
| static void virtio_9p_device_unrealize(DeviceState *dev) |
| { |
| VirtIODevice *vdev = VIRTIO_DEVICE(dev); |
| V9fsVirtioState *v = VIRTIO_9P(dev); |
| V9fsState *s = &v->state; |
| |
| virtio_delete_queue(v->vq); |
| virtio_cleanup(vdev); |
| v9fs_device_unrealize_common(s); |
| } |
| |
| /* virtio-9p device */ |
| |
| static const VMStateDescription vmstate_virtio_9p = { |
| .name = "virtio-9p", |
| .minimum_version_id = 1, |
| .version_id = 1, |
| .fields = (const VMStateField[]) { |
| VMSTATE_VIRTIO_DEVICE, |
| VMSTATE_END_OF_LIST() |
| }, |
| }; |
| |
| static Property virtio_9p_properties[] = { |
| DEFINE_PROP_STRING("mount_tag", V9fsVirtioState, state.fsconf.tag), |
| DEFINE_PROP_STRING("fsdev", V9fsVirtioState, state.fsconf.fsdev_id), |
| DEFINE_PROP_END_OF_LIST(), |
| }; |
| |
| static void virtio_9p_class_init(ObjectClass *klass, void *data) |
| { |
| DeviceClass *dc = DEVICE_CLASS(klass); |
| VirtioDeviceClass *vdc = VIRTIO_DEVICE_CLASS(klass); |
| |
| device_class_set_props(dc, virtio_9p_properties); |
| dc->vmsd = &vmstate_virtio_9p; |
| set_bit(DEVICE_CATEGORY_STORAGE, dc->categories); |
| vdc->realize = virtio_9p_device_realize; |
| vdc->unrealize = virtio_9p_device_unrealize; |
| vdc->get_features = virtio_9p_get_features; |
| vdc->get_config = virtio_9p_get_config; |
| vdc->reset = virtio_9p_reset; |
| } |
| |
| static const TypeInfo virtio_device_info = { |
| .name = TYPE_VIRTIO_9P, |
| .parent = TYPE_VIRTIO_DEVICE, |
| .instance_size = sizeof(V9fsVirtioState), |
| .class_init = virtio_9p_class_init, |
| }; |
| |
| static void virtio_9p_register_types(void) |
| { |
| type_register_static(&virtio_device_info); |
| } |
| |
| type_init(virtio_9p_register_types) |