]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
drm/hyperv: During panic do VMBus unload after frame buffer is flushed
authorMichael Kelley <mhklinux@outlook.com>
Tue, 17 Feb 2026 18:23:35 +0000 (10:23 -0800)
committerWei Liu <wei.liu@kernel.org>
Wed, 13 May 2026 22:31:01 +0000 (22:31 +0000)
In a VM, Linux panic information (reason for the panic, stack trace,
etc.) may be written to a serial console and/or a virtual frame buffer
for a graphics console. The latter may need to be flushed back to the
host hypervisor for display.

The current Hyper-V DRM driver for the frame buffer does the flushing
*after* the VMBus connection has been unloaded, such that panic messages
are not displayed on the graphics console. A user with a Hyper-V graphics
console is left with just a hung empty screen after a panic. The enhanced
control that DRM provides over the panic display in the graphics console
is similarly non-functional.

Commit 3671f3777758 ("drm/hyperv: Add support for drm_panic") added
the Hyper-V DRM driver support to flush the virtual frame buffer. It
provided necessary functionality but did not handle the sequencing
problem with VMBus unload.

Fix the full problem by using VMBus functions to suppress the VMBus
unload that is normally done by the VMBus driver in the panic path. Then
after the frame buffer has been flushed, do the VMBus unload so that a
kdump kernel can start cleanly. As expected, CONFIG_DRM_PANIC must be
selected for these changes to have effect. As a side benefit, the
enhanced features of the DRM panic path are also functional.

Fixes: 3671f3777758 ("drm/hyperv: Add support for drm_panic")
Signed-off-by: Michael Kelley <mhklinux@outlook.com>
Reviewed-by: Jocelyn Falempe <jfalempe@redhat.com>
Signed-off-by: Wei Liu <wei.liu@kernel.org>
drivers/gpu/drm/hyperv/hyperv_drm_drv.c
drivers/gpu/drm/hyperv/hyperv_drm_modeset.c

index 06b5d96e6eaf6baaa723f460f0b87c47f97085c0..b6bf6412ae34ac4387ba505690373195c1875cc2 100644 (file)
@@ -150,6 +150,10 @@ static int hyperv_vmbus_probe(struct hv_device *hdev,
                goto err_free_mmio;
        }
 
+       /* If DRM panic path is stubbed out VMBus code must do the unload */
+       if (IS_ENABLED(CONFIG_DRM_PANIC))
+               vmbus_set_skip_unload(true);
+
        drm_client_setup(dev, NULL);
 
        return 0;
@@ -169,6 +173,7 @@ static void hyperv_vmbus_remove(struct hv_device *hdev)
        struct drm_device *dev = hv_get_drvdata(hdev);
        struct hyperv_drm_device *hv = to_hv(dev);
 
+       vmbus_set_skip_unload(false);
        drm_dev_unplug(dev);
        drm_atomic_helper_shutdown(dev);
        vmbus_close(hdev->channel);
index 7978f8c8108c237290bc1d878c4ed0fef046a7ee..d48ca6c23b7c5e4af429eafd9b342116c544b875 100644 (file)
@@ -212,15 +212,16 @@ static void hyperv_plane_panic_flush(struct drm_plane *plane)
        struct hyperv_drm_device *hv = to_hv(plane->dev);
        struct drm_rect rect;
 
-       if (!plane->state || !plane->state->fb)
-               return;
+       if (plane->state && plane->state->fb) {
+               rect.x1 = 0;
+               rect.y1 = 0;
+               rect.x2 = plane->state->fb->width;
+               rect.y2 = plane->state->fb->height;
 
-       rect.x1 = 0;
-       rect.y1 = 0;
-       rect.x2 = plane->state->fb->width;
-       rect.y2 = plane->state->fb->height;
+               hyperv_update_dirt(hv->hdev, &rect);
+       }
 
-       hyperv_update_dirt(hv->hdev, &rect);
+       vmbus_initiate_unload(true);
 }
 
 static const struct drm_plane_helper_funcs hyperv_plane_helper_funcs = {