mirror of https://gitee.com/openkylin/qemu.git
xen/9pfs: receive requests from the frontend
Upon receiving an event channel notification from the frontend, schedule the bottom half. From the bottom half, read one request from the ring, create a pdu and call pdu_submit to handle it. For now, only handle one request per ring at a time. Signed-off-by: Stefano Stabellini <stefano@aporeto.com> CC: anthony.perard@citrix.com CC: jgross@suse.com CC: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> CC: Greg Kurz <groug@kaod.org>
This commit is contained in:
parent
f23ef34a5d
commit
47b70fb1e4
|
@ -100,12 +100,62 @@ static int xen_9pfs_init(struct XenDevice *xendev)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int xen_9pfs_receive(Xen9pfsRing *ring)
|
||||
{
|
||||
P9MsgHeader h;
|
||||
RING_IDX cons, prod, masked_prod, masked_cons;
|
||||
V9fsPDU *pdu;
|
||||
|
||||
if (ring->inprogress) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
cons = ring->intf->out_cons;
|
||||
prod = ring->intf->out_prod;
|
||||
xen_rmb();
|
||||
|
||||
if (xen_9pfs_queued(prod, cons, XEN_FLEX_RING_SIZE(ring->ring_order)) <
|
||||
sizeof(h)) {
|
||||
return 0;
|
||||
}
|
||||
ring->inprogress = true;
|
||||
|
||||
masked_prod = xen_9pfs_mask(prod, XEN_FLEX_RING_SIZE(ring->ring_order));
|
||||
masked_cons = xen_9pfs_mask(cons, XEN_FLEX_RING_SIZE(ring->ring_order));
|
||||
|
||||
xen_9pfs_read_packet((uint8_t *) &h, ring->ring.out, sizeof(h),
|
||||
masked_prod, &masked_cons,
|
||||
XEN_FLEX_RING_SIZE(ring->ring_order));
|
||||
|
||||
/* cannot fail, because we only handle one request per ring at a time */
|
||||
pdu = pdu_alloc(&ring->priv->state);
|
||||
pdu->size = le32_to_cpu(h.size_le);
|
||||
pdu->id = h.id;
|
||||
pdu->tag = le32_to_cpu(h.tag_le);
|
||||
ring->out_size = le32_to_cpu(h.size_le);
|
||||
ring->out_cons = cons + le32_to_cpu(h.size_le);
|
||||
|
||||
qemu_co_queue_init(&pdu->complete);
|
||||
pdu_submit(pdu);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void xen_9pfs_bh(void *opaque)
|
||||
{
|
||||
Xen9pfsRing *ring = opaque;
|
||||
xen_9pfs_receive(ring);
|
||||
}
|
||||
|
||||
static void xen_9pfs_evtchn_event(void *opaque)
|
||||
{
|
||||
Xen9pfsRing *ring = opaque;
|
||||
evtchn_port_t port;
|
||||
|
||||
port = xenevtchn_pending(ring->evtchndev);
|
||||
xenevtchn_unmask(ring->evtchndev, port);
|
||||
|
||||
qemu_bh_schedule(ring->bh);
|
||||
}
|
||||
|
||||
static int xen_9pfs_free(struct XenDevice *xendev)
|
||||
|
|
Loading…
Reference in New Issue