提交 c27ff608 编写于 作者: J Jan Kiszka 提交者: Mark McLoughlin

net: Fix and improved ordered packet delivery

Fix a race in qemu_send_packet when delivering deferred packets and
add proper deferring also to qemu_sendv_packet.
Signed-off-by: NJan Kiszka <jan.kiszka@siemens.com>
Signed-off-by: NMark McLoughlin <markmc@redhat.com>
上级 c8aa237c
...@@ -438,8 +438,8 @@ void qemu_send_packet(VLANClientState *vc, const uint8_t *buf, int size) ...@@ -438,8 +438,8 @@ void qemu_send_packet(VLANClientState *vc, const uint8_t *buf, int size)
vlan->delivering = 1; vlan->delivering = 1;
qemu_deliver_packet(vc, buf, size); qemu_deliver_packet(vc, buf, size);
while ((packet = vlan->send_queue) != NULL) { while ((packet = vlan->send_queue) != NULL) {
qemu_deliver_packet(packet->sender, packet->data, packet->size);
vlan->send_queue = packet->next; vlan->send_queue = packet->next;
qemu_deliver_packet(packet->sender, packet->data, packet->size);
qemu_free(packet); qemu_free(packet);
} }
vlan->delivering = 0; vlan->delivering = 0;
...@@ -476,30 +476,57 @@ static ssize_t calc_iov_length(const struct iovec *iov, int iovcnt) ...@@ -476,30 +476,57 @@ static ssize_t calc_iov_length(const struct iovec *iov, int iovcnt)
return offset; return offset;
} }
ssize_t qemu_sendv_packet(VLANClientState *vc1, const struct iovec *iov, ssize_t qemu_sendv_packet(VLANClientState *sender, const struct iovec *iov,
int iovcnt) int iovcnt)
{ {
VLANState *vlan = vc1->vlan; VLANState *vlan = sender->vlan;
VLANClientState *vc; VLANClientState *vc;
VLANPacket *packet;
ssize_t max_len = 0; ssize_t max_len = 0;
int i;
if (vc1->link_down) if (sender->link_down)
return calc_iov_length(iov, iovcnt); return calc_iov_length(iov, iovcnt);
for (vc = vlan->first_client; vc != NULL; vc = vc->next) { if (vlan->delivering) {
ssize_t len = 0; max_len = calc_iov_length(iov, iovcnt);
if (vc == vc1) packet = qemu_malloc(sizeof(VLANPacket) + max_len);
continue; packet->next = vlan->send_queue;
packet->sender = sender;
packet->size = 0;
for (i = 0; i < iovcnt; i++) {
size_t len = iov[i].iov_len;
memcpy(packet->data + packet->size, iov[i].iov_base, len);
packet->size += len;
}
vlan->send_queue = packet;
} else {
vlan->delivering = 1;
for (vc = vlan->first_client; vc != NULL; vc = vc->next) {
ssize_t len = 0;
if (vc->link_down) if (vc == sender) {
len = calc_iov_length(iov, iovcnt); continue;
else if (vc->fd_readv) }
len = vc->fd_readv(vc->opaque, iov, iovcnt); if (vc->link_down) {
else if (vc->fd_read) len = calc_iov_length(iov, iovcnt);
len = vc_sendv_compat(vc, iov, iovcnt); } else if (vc->fd_readv) {
len = vc->fd_readv(vc->opaque, iov, iovcnt);
} else if (vc->fd_read) {
len = vc_sendv_compat(vc, iov, iovcnt);
}
max_len = MAX(max_len, len);
}
max_len = MAX(max_len, len); while ((packet = vlan->send_queue) != NULL) {
vlan->send_queue = packet->next;
qemu_deliver_packet(packet->sender, packet->data, packet->size);
qemu_free(packet);
}
vlan->delivering = 0;
} }
return max_len; return max_len;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册