Skip to content

Commit 397d227

Browse files
committed
fix: do not panic if virtio device activation return Err(...)
When the guest driver sets a virtio devices status to `DRIVER_OK`, we proceed with calling `VirtioDevice::activate`. However, our MMIO transport layer assumes that this activation cannot go wrong, and calls `.expect(...)` on the result. For most devices, this is fine, as the activate method doesn't do much besides writing to an event_fd (and I can't think of a scenario in which this could go wrong). However, our vhost-user-blk device has some non-trivial logic inside of its `activate` method, which includes communication with the vhost-user-backend via a unix socket. If this unix socket gets closed early, this causes `activate` to return an error, and thus consequently a panic in the MMIO code. The virtio spec, in Section 2.2, has the following to say [1]: > The device SHOULD set DEVICE_NEEDS_RESET when it enters an error state that a reset is needed. If DRIVER_OK is set, after it sets DEVICE_NEEDS_RESET, the device MUST send a device configuration change notification to the driver. So the spec-conform way of handling an activation error is setting the `DEVICE_NEEDS_RESET` flag in the device_status field (which is what this commit does). This will fix the panic, however it will most certainly still not result in correct device operations (e.g. a vhost-user-backend dying will still be unrecoverable). This is because Firecracker does not actually implement device reset, see also #3074. Thus, the device will simply be "dead" to the guest: All operations where we currently simply abort and do nothing if the device is in the FAILED state will do the same in the DEVICE_NEEDS_RESET state. [1]: https://docs.oasis-open.org/virtio/virtio/v1.1/csprd01/virtio-v1.1-csprd01.pdf Signed-off-by: Patrick Roy <[email protected]>
1 parent c2bb2f6 commit 397d227

File tree

2 files changed

+21
-7
lines changed

2 files changed

+21
-7
lines changed

src/vmm/src/devices/virtio/mmio.rs

+20-7
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ use std::sync::{Arc, Mutex, MutexGuard};
1111

1212
use utils::byte_order;
1313

14-
use crate::devices::virtio::device::VirtioDevice;
14+
use crate::devices::virtio::device::{IrqType, VirtioDevice};
1515
use crate::devices::virtio::device_status;
1616
use crate::devices::virtio::queue::Queue;
1717
use crate::logger::warn;
@@ -186,10 +186,18 @@ impl MmioTransport {
186186
DRIVER_OK if self.device_status == (ACKNOWLEDGE | DRIVER | FEATURES_OK) => {
187187
self.device_status = status;
188188
let device_activated = self.locked_device().is_activated();
189-
if !device_activated && self.are_queues_valid() {
190-
self.locked_device()
191-
.activate(self.mem.clone())
192-
.expect("Failed to activate device");
189+
if !device_activated
190+
&& self.are_queues_valid()
191+
&& self.locked_device().activate(self.mem.clone()).is_err()
192+
{
193+
self.device_status |= DEVICE_NEEDS_RESET;
194+
195+
// Section 2.1.2 of the specification states that we need to send a device
196+
// configuration change interrupt
197+
let _ = self
198+
.locked_device()
199+
.interrupt_trigger()
200+
.trigger_irq(IrqType::Config);
193201
}
194202
}
195203
_ if (status & FAILED) != 0 => {
@@ -306,7 +314,9 @@ impl MmioTransport {
306314
0x20 => {
307315
if self.check_device_status(
308316
device_status::DRIVER,
309-
device_status::FEATURES_OK | device_status::FAILED,
317+
device_status::FEATURES_OK
318+
| device_status::FAILED
319+
| device_status::DEVICE_NEEDS_RESET,
310320
) {
311321
self.locked_device()
312322
.ack_features_by_page(self.acked_features_select, v);
@@ -339,7 +349,10 @@ impl MmioTransport {
339349
}
340350
}
341351
0x100..=0xfff => {
342-
if self.check_device_status(device_status::DRIVER, device_status::FAILED) {
352+
if self.check_device_status(
353+
device_status::DRIVER,
354+
device_status::FAILED | device_status::DEVICE_NEEDS_RESET,
355+
) {
343356
self.locked_device().write_config(offset - 0x100, data)
344357
} else {
345358
warn!("can not write to device config data area before driver is ready");

src/vmm/src/devices/virtio/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ mod device_status {
4040
pub const FAILED: u32 = 128;
4141
pub const FEATURES_OK: u32 = 8;
4242
pub const DRIVER_OK: u32 = 4;
43+
pub const DEVICE_NEEDS_RESET: u32 = 64;
4344
}
4445

4546
/// Types taken from linux/virtio_ids.h.

0 commit comments

Comments
 (0)