9pfs: introduce V9fsVirtioState
V9fsState now only contains generic fields. Introduce V9fsVirtioState for virtio transport. Change virtio-pci and virtio-ccw to use V9fsVirtioState. Signed-off-by: Wei Liu <wei.liu2@citrix.com> Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
This commit is contained in:
parent
2a0c56aa4c
commit
00588a0aa2
11
hw/9pfs/9p.c
11
hw/9pfs/9p.c
@ -1585,6 +1585,8 @@ static int v9fs_xattr_read(V9fsState *s, V9fsPDU *pdu, V9fsFidState *fidp,
|
|||||||
size_t offset = 7;
|
size_t offset = 7;
|
||||||
int read_count;
|
int read_count;
|
||||||
int64_t xattr_len;
|
int64_t xattr_len;
|
||||||
|
V9fsVirtioState *v = container_of(s, V9fsVirtioState, state);
|
||||||
|
VirtQueueElement *elem = &v->elems[pdu->idx];
|
||||||
|
|
||||||
xattr_len = fidp->fs.xattr.len;
|
xattr_len = fidp->fs.xattr.len;
|
||||||
read_count = xattr_len - off;
|
read_count = xattr_len - off;
|
||||||
@ -1601,7 +1603,8 @@ static int v9fs_xattr_read(V9fsState *s, V9fsPDU *pdu, V9fsFidState *fidp,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
offset += err;
|
offset += err;
|
||||||
err = v9fs_pack(pdu->elem.in_sg, pdu->elem.in_num, offset,
|
|
||||||
|
err = v9fs_pack(elem->in_sg, elem->in_num, offset,
|
||||||
((char *)fidp->fs.xattr.value) + off,
|
((char *)fidp->fs.xattr.value) + off,
|
||||||
read_count);
|
read_count);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
@ -3269,6 +3272,7 @@ void pdu_submit(V9fsPDU *pdu)
|
|||||||
/* Returns 0 on success, 1 on failure. */
|
/* Returns 0 on success, 1 on failure. */
|
||||||
int v9fs_device_realize_common(V9fsState *s, Error **errp)
|
int v9fs_device_realize_common(V9fsState *s, Error **errp)
|
||||||
{
|
{
|
||||||
|
V9fsVirtioState *v = container_of(s, V9fsVirtioState, state);
|
||||||
int i, len;
|
int i, len;
|
||||||
struct stat stat;
|
struct stat stat;
|
||||||
FsDriverEntry *fse;
|
FsDriverEntry *fse;
|
||||||
@ -3279,8 +3283,9 @@ int v9fs_device_realize_common(V9fsState *s, Error **errp)
|
|||||||
QLIST_INIT(&s->free_list);
|
QLIST_INIT(&s->free_list);
|
||||||
QLIST_INIT(&s->active_list);
|
QLIST_INIT(&s->active_list);
|
||||||
for (i = 0; i < (MAX_REQ - 1); i++) {
|
for (i = 0; i < (MAX_REQ - 1); i++) {
|
||||||
QLIST_INSERT_HEAD(&s->free_list, &s->pdus[i], next);
|
QLIST_INSERT_HEAD(&s->free_list, &v->pdus[i], next);
|
||||||
s->pdus[i].s = s;
|
v->pdus[i].s = s;
|
||||||
|
v->pdus[i].idx = i;
|
||||||
}
|
}
|
||||||
|
|
||||||
v9fs_path_init(&path);
|
v9fs_path_init(&path);
|
||||||
|
@ -131,9 +131,9 @@ struct V9fsPDU
|
|||||||
uint8_t id;
|
uint8_t id;
|
||||||
uint8_t cancelled;
|
uint8_t cancelled;
|
||||||
CoQueue complete;
|
CoQueue complete;
|
||||||
VirtQueueElement elem;
|
|
||||||
struct V9fsState *s;
|
struct V9fsState *s;
|
||||||
QLIST_ENTRY(V9fsPDU) next;
|
QLIST_ENTRY(V9fsPDU) next;
|
||||||
|
uint32_t idx;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -205,16 +205,12 @@ struct V9fsFidState
|
|||||||
|
|
||||||
typedef struct V9fsState
|
typedef struct V9fsState
|
||||||
{
|
{
|
||||||
VirtIODevice parent_obj;
|
|
||||||
VirtQueue *vq;
|
|
||||||
V9fsPDU pdus[MAX_REQ];
|
|
||||||
QLIST_HEAD(, V9fsPDU) free_list;
|
QLIST_HEAD(, V9fsPDU) free_list;
|
||||||
QLIST_HEAD(, V9fsPDU) active_list;
|
QLIST_HEAD(, V9fsPDU) active_list;
|
||||||
V9fsFidState *fid_list;
|
V9fsFidState *fid_list;
|
||||||
FileOperations *ops;
|
FileOperations *ops;
|
||||||
FsContext ctx;
|
FsContext ctx;
|
||||||
char *tag;
|
char *tag;
|
||||||
size_t config_size;
|
|
||||||
enum p9_proto_version proto_version;
|
enum p9_proto_version proto_version;
|
||||||
int32_t msize;
|
int32_t msize;
|
||||||
/*
|
/*
|
||||||
|
@ -24,33 +24,41 @@
|
|||||||
void virtio_9p_push_and_notify(V9fsPDU *pdu)
|
void virtio_9p_push_and_notify(V9fsPDU *pdu)
|
||||||
{
|
{
|
||||||
V9fsState *s = pdu->s;
|
V9fsState *s = pdu->s;
|
||||||
|
V9fsVirtioState *v = container_of(s, V9fsVirtioState, state);
|
||||||
|
VirtQueueElement *elem = &v->elems[pdu->idx];
|
||||||
|
|
||||||
/* push onto queue and notify */
|
/* push onto queue and notify */
|
||||||
virtqueue_push(s->vq, &pdu->elem, pdu->size);
|
virtqueue_push(v->vq, elem, pdu->size);
|
||||||
|
|
||||||
/* FIXME: we should batch these completions */
|
/* FIXME: we should batch these completions */
|
||||||
virtio_notify(VIRTIO_DEVICE(s), s->vq);
|
virtio_notify(VIRTIO_DEVICE(v), v->vq);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_9p_output(VirtIODevice *vdev, VirtQueue *vq)
|
static void handle_9p_output(VirtIODevice *vdev, VirtQueue *vq)
|
||||||
{
|
{
|
||||||
V9fsState *s = (V9fsState *)vdev;
|
V9fsVirtioState *v = (V9fsVirtioState *)vdev;
|
||||||
|
V9fsState *s = &v->state;
|
||||||
V9fsPDU *pdu;
|
V9fsPDU *pdu;
|
||||||
ssize_t len;
|
ssize_t len;
|
||||||
|
|
||||||
while ((pdu = pdu_alloc(s)) &&
|
while ((pdu = pdu_alloc(s))) {
|
||||||
(len = virtqueue_pop(vq, &pdu->elem)) != 0) {
|
|
||||||
struct {
|
struct {
|
||||||
uint32_t size_le;
|
uint32_t size_le;
|
||||||
uint8_t id;
|
uint8_t id;
|
||||||
uint16_t tag_le;
|
uint16_t tag_le;
|
||||||
} QEMU_PACKED out;
|
} QEMU_PACKED out;
|
||||||
int len;
|
VirtQueueElement *elem = &v->elems[pdu->idx];
|
||||||
|
|
||||||
BUG_ON(pdu->elem.out_num == 0 || pdu->elem.in_num == 0);
|
len = virtqueue_pop(vq, elem);
|
||||||
|
if (!len) {
|
||||||
|
pdu_free(pdu);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
BUG_ON(elem->out_num == 0 || elem->in_num == 0);
|
||||||
QEMU_BUILD_BUG_ON(sizeof out != 7);
|
QEMU_BUILD_BUG_ON(sizeof out != 7);
|
||||||
|
|
||||||
len = iov_to_buf(pdu->elem.out_sg, pdu->elem.out_num, 0,
|
len = iov_to_buf(elem->out_sg, elem->out_num, 0,
|
||||||
&out, sizeof out);
|
&out, sizeof out);
|
||||||
BUG_ON(len != sizeof out);
|
BUG_ON(len != sizeof out);
|
||||||
|
|
||||||
@ -62,7 +70,6 @@ static void handle_9p_output(VirtIODevice *vdev, VirtQueue *vq)
|
|||||||
qemu_co_queue_init(&pdu->complete);
|
qemu_co_queue_init(&pdu->complete);
|
||||||
pdu_submit(pdu);
|
pdu_submit(pdu);
|
||||||
}
|
}
|
||||||
pdu_free(pdu);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint64_t virtio_9p_get_features(VirtIODevice *vdev, uint64_t features,
|
static uint64_t virtio_9p_get_features(VirtIODevice *vdev, uint64_t features,
|
||||||
@ -76,14 +83,15 @@ static void virtio_9p_get_config(VirtIODevice *vdev, uint8_t *config)
|
|||||||
{
|
{
|
||||||
int len;
|
int len;
|
||||||
struct virtio_9p_config *cfg;
|
struct virtio_9p_config *cfg;
|
||||||
V9fsState *s = VIRTIO_9P(vdev);
|
V9fsVirtioState *v = VIRTIO_9P(vdev);
|
||||||
|
V9fsState *s = &v->state;
|
||||||
|
|
||||||
len = strlen(s->tag);
|
len = strlen(s->tag);
|
||||||
cfg = g_malloc0(sizeof(struct virtio_9p_config) + len);
|
cfg = g_malloc0(sizeof(struct virtio_9p_config) + len);
|
||||||
virtio_stw_p(vdev, &cfg->tag_len, len);
|
virtio_stw_p(vdev, &cfg->tag_len, len);
|
||||||
/* We don't copy the terminating null to config space */
|
/* We don't copy the terminating null to config space */
|
||||||
memcpy(cfg->tag, s->tag, len);
|
memcpy(cfg->tag, s->tag, len);
|
||||||
memcpy(config, cfg, s->config_size);
|
memcpy(config, cfg, v->config_size);
|
||||||
g_free(cfg);
|
g_free(cfg);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -100,16 +108,17 @@ static int virtio_9p_load(QEMUFile *f, void *opaque, int version_id)
|
|||||||
static void virtio_9p_device_realize(DeviceState *dev, Error **errp)
|
static void virtio_9p_device_realize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
VirtIODevice *vdev = VIRTIO_DEVICE(dev);
|
VirtIODevice *vdev = VIRTIO_DEVICE(dev);
|
||||||
V9fsState *s = VIRTIO_9P(dev);
|
V9fsVirtioState *v = VIRTIO_9P(dev);
|
||||||
|
V9fsState *s = &v->state;
|
||||||
|
|
||||||
if (v9fs_device_realize_common(s, errp)) {
|
if (v9fs_device_realize_common(s, errp)) {
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
s->config_size = sizeof(struct virtio_9p_config) + strlen(s->fsconf.tag);
|
v->config_size = sizeof(struct virtio_9p_config) + strlen(s->fsconf.tag);
|
||||||
virtio_init(vdev, "virtio-9p", VIRTIO_ID_9P, s->config_size);
|
virtio_init(vdev, "virtio-9p", VIRTIO_ID_9P, v->config_size);
|
||||||
s->vq = virtio_add_queue(vdev, MAX_REQ, handle_9p_output);
|
v->vq = virtio_add_queue(vdev, MAX_REQ, handle_9p_output);
|
||||||
register_savevm(dev, "virtio-9p", -1, 1, virtio_9p_save, virtio_9p_load, s);
|
register_savevm(dev, "virtio-9p", -1, 1, virtio_9p_save, virtio_9p_load, v);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return;
|
return;
|
||||||
@ -118,44 +127,55 @@ out:
|
|||||||
static void virtio_9p_device_unrealize(DeviceState *dev, Error **errp)
|
static void virtio_9p_device_unrealize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
VirtIODevice *vdev = VIRTIO_DEVICE(dev);
|
VirtIODevice *vdev = VIRTIO_DEVICE(dev);
|
||||||
V9fsState *s = VIRTIO_9P(dev);
|
V9fsVirtioState *v = VIRTIO_9P(dev);
|
||||||
|
V9fsState *s = &v->state;
|
||||||
|
|
||||||
virtio_cleanup(vdev);
|
virtio_cleanup(vdev);
|
||||||
unregister_savevm(dev, "virtio-9p", s);
|
unregister_savevm(dev, "virtio-9p", v);
|
||||||
v9fs_device_unrealize_common(s, errp);
|
v9fs_device_unrealize_common(s, errp);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t virtio_pdu_vmarshal(V9fsPDU *pdu, size_t offset,
|
ssize_t virtio_pdu_vmarshal(V9fsPDU *pdu, size_t offset,
|
||||||
const char *fmt, va_list ap)
|
const char *fmt, va_list ap)
|
||||||
{
|
{
|
||||||
return v9fs_iov_vmarshal(pdu->elem.in_sg, pdu->elem.in_num,
|
V9fsState *s = pdu->s;
|
||||||
offset, 1, fmt, ap);
|
V9fsVirtioState *v = container_of(s, V9fsVirtioState, state);
|
||||||
|
VirtQueueElement *elem = &v->elems[pdu->idx];
|
||||||
|
|
||||||
|
return v9fs_iov_vmarshal(elem->in_sg, elem->in_num, offset, 1, fmt, ap);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t virtio_pdu_vunmarshal(V9fsPDU *pdu, size_t offset,
|
ssize_t virtio_pdu_vunmarshal(V9fsPDU *pdu, size_t offset,
|
||||||
const char *fmt, va_list ap)
|
const char *fmt, va_list ap)
|
||||||
{
|
{
|
||||||
return v9fs_iov_vunmarshal(pdu->elem.out_sg, pdu->elem.out_num,
|
V9fsState *s = pdu->s;
|
||||||
offset, 1, fmt, ap);
|
V9fsVirtioState *v = container_of(s, V9fsVirtioState, state);
|
||||||
|
VirtQueueElement *elem = &v->elems[pdu->idx];
|
||||||
|
|
||||||
|
return v9fs_iov_vunmarshal(elem->out_sg, elem->out_num, offset, 1, fmt, ap);
|
||||||
}
|
}
|
||||||
|
|
||||||
void virtio_init_iov_from_pdu(V9fsPDU *pdu, struct iovec **piov,
|
void virtio_init_iov_from_pdu(V9fsPDU *pdu, struct iovec **piov,
|
||||||
unsigned int *pniov, bool is_write)
|
unsigned int *pniov, bool is_write)
|
||||||
{
|
{
|
||||||
|
V9fsState *s = pdu->s;
|
||||||
|
V9fsVirtioState *v = container_of(s, V9fsVirtioState, state);
|
||||||
|
VirtQueueElement *elem = &v->elems[pdu->idx];
|
||||||
|
|
||||||
if (is_write) {
|
if (is_write) {
|
||||||
*piov = pdu->elem.out_sg;
|
*piov = elem->out_sg;
|
||||||
*pniov = pdu->elem.out_num;
|
*pniov = elem->out_num;
|
||||||
} else {
|
} else {
|
||||||
*piov = pdu->elem.in_sg;
|
*piov = elem->in_sg;
|
||||||
*pniov = pdu->elem.in_num;
|
*pniov = elem->in_num;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* virtio-9p device */
|
/* virtio-9p device */
|
||||||
|
|
||||||
static Property virtio_9p_properties[] = {
|
static Property virtio_9p_properties[] = {
|
||||||
DEFINE_PROP_STRING("mount_tag", V9fsState, fsconf.tag),
|
DEFINE_PROP_STRING("mount_tag", V9fsVirtioState, state.fsconf.tag),
|
||||||
DEFINE_PROP_STRING("fsdev", V9fsState, fsconf.fsdev_id),
|
DEFINE_PROP_STRING("fsdev", V9fsVirtioState, state.fsconf.fsdev_id),
|
||||||
DEFINE_PROP_END_OF_LIST(),
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -175,7 +195,7 @@ static void virtio_9p_class_init(ObjectClass *klass, void *data)
|
|||||||
static const TypeInfo virtio_device_info = {
|
static const TypeInfo virtio_device_info = {
|
||||||
.name = TYPE_VIRTIO_9P,
|
.name = TYPE_VIRTIO_9P,
|
||||||
.parent = TYPE_VIRTIO_DEVICE,
|
.parent = TYPE_VIRTIO_DEVICE,
|
||||||
.instance_size = sizeof(V9fsState),
|
.instance_size = sizeof(V9fsVirtioState),
|
||||||
.class_init = virtio_9p_class_init,
|
.class_init = virtio_9p_class_init,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -5,6 +5,16 @@
|
|||||||
#include "hw/virtio/virtio.h"
|
#include "hw/virtio/virtio.h"
|
||||||
#include "9p.h"
|
#include "9p.h"
|
||||||
|
|
||||||
|
typedef struct V9fsVirtioState
|
||||||
|
{
|
||||||
|
VirtIODevice parent_obj;
|
||||||
|
VirtQueue *vq;
|
||||||
|
size_t config_size;
|
||||||
|
V9fsPDU pdus[MAX_REQ];
|
||||||
|
VirtQueueElement elems[MAX_REQ];
|
||||||
|
V9fsState state;
|
||||||
|
} V9fsVirtioState;
|
||||||
|
|
||||||
extern void virtio_9p_push_and_notify(V9fsPDU *pdu);
|
extern void virtio_9p_push_and_notify(V9fsPDU *pdu);
|
||||||
|
|
||||||
ssize_t virtio_pdu_vmarshal(V9fsPDU *pdu, size_t offset,
|
ssize_t virtio_pdu_vmarshal(V9fsPDU *pdu, size_t offset,
|
||||||
@ -16,6 +26,6 @@ void virtio_init_iov_from_pdu(V9fsPDU *pdu, struct iovec **piov,
|
|||||||
|
|
||||||
#define TYPE_VIRTIO_9P "virtio-9p-device"
|
#define TYPE_VIRTIO_9P "virtio-9p-device"
|
||||||
#define VIRTIO_9P(obj) \
|
#define VIRTIO_9P(obj) \
|
||||||
OBJECT_CHECK(V9fsState, (obj), TYPE_VIRTIO_9P)
|
OBJECT_CHECK(V9fsVirtioState, (obj), TYPE_VIRTIO_9P)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -210,7 +210,7 @@ VirtIODevice *virtio_ccw_get_vdev(SubchDev *sch);
|
|||||||
|
|
||||||
typedef struct V9fsCCWState {
|
typedef struct V9fsCCWState {
|
||||||
VirtioCcwDevice parent_obj;
|
VirtioCcwDevice parent_obj;
|
||||||
V9fsState vdev;
|
V9fsVirtioState vdev;
|
||||||
} V9fsCCWState;
|
} V9fsCCWState;
|
||||||
|
|
||||||
#endif /* CONFIG_VIRTFS */
|
#endif /* CONFIG_VIRTFS */
|
||||||
|
@ -242,7 +242,7 @@ struct VirtIONetPCI {
|
|||||||
|
|
||||||
typedef struct V9fsPCIState {
|
typedef struct V9fsPCIState {
|
||||||
VirtIOPCIProxy parent_obj;
|
VirtIOPCIProxy parent_obj;
|
||||||
V9fsState vdev;
|
V9fsVirtioState vdev;
|
||||||
} V9fsPCIState;
|
} V9fsPCIState;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
x
Reference in New Issue
Block a user