提交 27e0e638 编写于 作者: J Jan Beulich 提交者: David Vrabel

xen/evtchn: fix ring resize when binding new events

The copying of ring data was wrong for two cases: For a full ring
nothing got copied at all (as in that case the canonicalized producer
and consumer indexes are identical). And in case one or both of the
canonicalized (after the resize) indexes would point into the second
half of the buffer, the copied data ended up in the wrong (free) part
of the new buffer. In both cases uninitialized data would get passed
back to the caller.

Fix this by simply copying the old ring contents twice: Once to the
low half of the new buffer, and a second time to the high half.

This addresses the inability to boot a HVM guest with 64 or more
vCPUs.  This regression was caused by 86200154 (xen/evtchn:
dynamically grow pending event channel ring).
Reported-by: NKonrad Rzeszutek Wilk <konrad.wilk@oracle.com>
Signed-off-by: NJan Beulich <jbeulich@suse.com>
Cc: <stable@vger.kernel.org> # 4.4+
Signed-off-by: NDavid Vrabel <david.vrabel@citrix.com>
上级 dfd74a1e
...@@ -316,7 +316,6 @@ static int evtchn_resize_ring(struct per_user_data *u) ...@@ -316,7 +316,6 @@ static int evtchn_resize_ring(struct per_user_data *u)
{ {
unsigned int new_size; unsigned int new_size;
evtchn_port_t *new_ring, *old_ring; evtchn_port_t *new_ring, *old_ring;
unsigned int p, c;
/* /*
* Ensure the ring is large enough to capture all possible * Ensure the ring is large enough to capture all possible
...@@ -346,20 +345,17 @@ static int evtchn_resize_ring(struct per_user_data *u) ...@@ -346,20 +345,17 @@ static int evtchn_resize_ring(struct per_user_data *u)
/* /*
* Copy the old ring contents to the new ring. * Copy the old ring contents to the new ring.
* *
* If the ring contents crosses the end of the current ring, * To take care of wrapping, a full ring, and the new index
* it needs to be copied in two chunks. * pointing into the second half, simply copy the old contents
* twice.
* *
* +---------+ +------------------+ * +---------+ +------------------+
* |34567 12| -> | 1234567 | * |34567 12| -> |34567 1234567 12|
* +-----p-c-+ +------------------+ * +-----p-c-+ +-------c------p---+
*/ */
p = evtchn_ring_offset(u, u->ring_prod); memcpy(new_ring, old_ring, u->ring_size * sizeof(*u->ring));
c = evtchn_ring_offset(u, u->ring_cons); memcpy(new_ring + u->ring_size, old_ring,
if (p < c) { u->ring_size * sizeof(*u->ring));
memcpy(new_ring + c, u->ring + c, (u->ring_size - c) * sizeof(*u->ring));
memcpy(new_ring + u->ring_size, u->ring, p * sizeof(*u->ring));
} else
memcpy(new_ring + c, u->ring + c, (p - c) * sizeof(*u->ring));
u->ring = new_ring; u->ring = new_ring;
u->ring_size = new_size; u->ring_size = new_size;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册