[PATCH 1/2] drm/hyperv: Add DRM driver for hyperv synthetic video device
Daniel Vetter
daniel at ffwll.ch
Tue Apr 4 13:20:55 UTC 2023
Yeah way late reply, but I just stumbled over this and got a bit puzzled ...
On Sat, 2 Jan 2021 at 07:03, Deepak Rawat <drawat.floss at gmail.com> wrote:
> +/*
> + * PCI/vmbus interface
> + */
> +
> +static int hyperv_pci_probe(struct pci_dev *pdev,
> + const struct pci_device_id *ent)
> +{
> + return 0;
> +}
Why do you have this dummy driver when it does nothing? Can it just be
deleted? If it's just to have a driver, then we really don't need that
on linux, there's no requirement to have a device driver for every
device in a system.
If you actually need to make sure that this pci device isn't passed to
a guest vm or something like that, then the main driver must ensure
that the pci driver is bound (ideally with component.c because
otherwise you'll get the unbind/rebind dance wrong in one of the
bazillion of subtle ways). Just having a driver doesn't stop anyone
from unbinding it and then wreaking havoc.
-Daniel
> +
> +static void hyperv_pci_remove(struct pci_dev *pdev)
> +{
> +}
> +
> +static const struct pci_device_id hyperv_pci_tbl[] = {
> + {
> + .vendor = PCI_VENDOR_ID_MICROSOFT,
> + .device = PCI_DEVICE_ID_HYPERV_VIDEO,
> + },
> + { /* end of list */ }
> +};
> +
> +static struct pci_driver hyperv_pci_driver = {
> + .name = KBUILD_MODNAME,
> + .id_table = hyperv_pci_tbl,
> + .probe = hyperv_pci_probe,
> + .remove = hyperv_pci_remove,
> +};
> +
> +static int hyperv_get_vram_gen1(struct hyperv_device *hv)
> +{
> + struct drm_device *dev = &hv->dev;
> + struct pci_dev *pdev;
> + int ret;
> +
> + pdev = pci_get_device(PCI_VENDOR_ID_MICROSOFT,
> + PCI_DEVICE_ID_HYPERV_VIDEO, NULL);
> + if (!pdev) {
> + drm_err(dev, "Unable to find PCI Hyper-V video\n");
> + return -ENODEV;
> + }
> +
> + ret = drm_fb_helper_remove_conflicting_pci_framebuffers(pdev, "hypervdrmfb");
> + if (ret) {
> + drm_err(dev, "Not able to remove boot fb\n");
> + return ret;
> + }
> +
> + if (pci_request_region(pdev, 0, DRIVER_NAME) != 0)
> + drm_warn(dev, "Cannot request framebuffer, boot fb still active?\n");
> +
> + if ((pdev->resource[0].flags & IORESOURCE_MEM) == 0) {
> + drm_err(dev, "Resource at bar 0 is not IORESOURCE_MEM\n");
> + ret = -ENODEV;
> + goto error;
> + }
> +
> + hv->fb_base = pci_resource_start(pdev, 0);
> + hv->fb_size = pci_resource_len(pdev, 0);
> + if (hv->fb_base == 0) {
> + drm_err(dev, "Resource not available\n");
> + ret = -ENODEV;
> + goto error;
> + }
> +
> + hv->fb_size = min(hv->fb_size,
> + (unsigned long)(hv->mmio_megabytes * 1024 * 1024));
> + hv->vram = devm_ioremap(&pdev->dev, hv->fb_base, hv->fb_size);
> + if (!hv->vram) {
> + drm_err(dev, "Failed to map vram\n");
> + ret = -ENOMEM;
> + }
> +
> +error:
> + pci_dev_put(pdev);
> + return ret;
> +}
> +
> +static int hyperv_get_vram_gen2(struct hyperv_device *hv,
> + struct hv_device *hdev)
> +{
> + struct drm_device *dev = &hv->dev;
> + struct apertures_struct *ap;
> + int ret;
> +
> + hv->fb_size = (unsigned long)(hv->mmio_megabytes * 1024 * 1024);
> +
> + ret = vmbus_allocate_mmio(&hv->mem, hdev, 0, -1, hv->fb_size, 0x100000,
> + true);
> + if (ret) {
> + drm_err(dev, "Failed to allocate mmio\n");
> + return -ENOMEM;
> + }
> +
> + hv->vram = ioremap(hv->mem->start, hv->fb_size);
> + if (!hv->vram) {
> + drm_err(dev, "Failed to map vram\n");
> + ret = -ENOMEM;
> + goto error;
> + }
> +
> + hv->fb_base = hv->mem->start;
> +
> + ap = alloc_apertures(1);
> + if (!ap) {
> + drm_err(dev, "Failed to get apertures\n");
> + ret = -ENOMEM;
> + goto error;
> + }
> +
> + ap->ranges[0].base = screen_info.lfb_base;
> + ap->ranges[0].size = screen_info.lfb_size;
> + remove_conflicting_framebuffers(ap, KBUILD_MODNAME, false);
> + kfree(ap);
> +
> + return 0;
> +
> +error:
> + vmbus_free_mmio(hv->mem->start, hv->fb_size);
> + return ret;
> +}
> +
> +static int hyperv_vmbus_probe(struct hv_device *hdev,
> + const struct hv_vmbus_device_id *dev_id)
> +{
> + struct hyperv_device *hv;
> + struct drm_device *dev;
> + int ret;
> +
> + hv = devm_drm_dev_alloc(&hdev->device, &hyperv_driver,
> + struct hyperv_device, dev);
> + if (IS_ERR(hv))
> + return PTR_ERR(hv);
> +
> + dev = &hv->dev;
> + init_completion(&hv->wait);
> + hv_set_drvdata(hdev, hv);
> + hv->hdev = hdev;
> +
> + /* Get the actual VRAM size from the device */
> + ret = synthvid_connect_vsp(hdev);
> + if (ret) {
> + drm_err(dev, "Failed to connect to vmbus.\n");
> + goto err_hv_set_drv_data;
> + }
> +
> + if (efi_enabled(EFI_BOOT))
> + ret = hyperv_get_vram_gen2(hv, hdev);
> + else
> + ret = hyperv_get_vram_gen1(hv);
> +
> + if (ret)
> + goto err_vmbus_close;
> +
> + /*
> + * Should be done only once during init and resume. Failing to update
> + * vram location is not fatal. Device will update dirty area till
> + * preferred resolution only.
> + */
> + ret = synthvid_update_vram_location(hdev, hv->fb_base);
> + if (ret)
> + drm_warn(dev, "Failed to update vram location.\n");
> +
> + ret = hyperv_mode_config_init(hv);
> + if (ret)
> + goto err_vmbus_close;
> +
> + ret = drm_dev_register(dev, 0);
> + if (ret) {
> + drm_err(dev, "Failed to register drm driver.\n");
> + goto err_vmbus_close;
> + }
> +
> + drm_fbdev_generic_setup(dev, 0);
> +
> + return 0;
> +
> +err_vmbus_close:
> + vmbus_close(hdev->channel);
> +err_hv_set_drv_data:
> + hv_set_drvdata(hdev, NULL);
> + return ret;
> +}
> +
> +static int hyperv_vmbus_remove(struct hv_device *hdev)
> +{
> + struct drm_device *dev = hv_get_drvdata(hdev);
> + struct hyperv_device *hv = to_hv(dev);
> +
> + drm_dev_unplug(dev);
> + drm_atomic_helper_shutdown(dev);
> + vmbus_close(hdev->channel);
> + hv_set_drvdata(hdev, NULL);
> + vmbus_free_mmio(hv->mem->start, hv->fb_size);
> +
> + return 0;
> +}
> +
> +static int hyperv_vmbus_suspend(struct hv_device *hdev)
> +{
> + struct drm_device *dev = hv_get_drvdata(hdev);
> + int ret;
> +
> + ret = drm_mode_config_helper_suspend(dev);
> +
> + vmbus_close(hdev->channel);
> +
> + return ret;
> +}
> +
> +static int hyperv_vmbus_resume(struct hv_device *hdev)
> +{
> + struct drm_device *dev = hv_get_drvdata(hdev);
> + int ret;
> +
> + ret = synthvid_connect_vsp(hdev);
> + if (ret)
> + return ret;
> +
> + return drm_mode_config_helper_resume(dev);
> +}
> +
> +static const struct hv_vmbus_device_id hyperv_vmbus_tbl[] = {
> + /* Synthetic Video Device GUID */
> + {HV_SYNTHVID_GUID},
> + {}
> +};
> +
> +static struct hv_driver hyperv_hv_driver = {
> + .name = KBUILD_MODNAME,
> + .id_table = hyperv_vmbus_tbl,
> + .probe = hyperv_vmbus_probe,
> + .remove = hyperv_vmbus_remove,
> + .suspend = hyperv_vmbus_suspend,
> + .resume = hyperv_vmbus_resume,
> + .driver = {
> + .probe_type = PROBE_PREFER_ASYNCHRONOUS,
> + },
> +};
> +
> +/* ---------------------------------------------------------------------- */
> +/* module init/exit */
> +
> +static int __init hyperv_init(void)
> +{
> + int ret;
> +
> + ret = pci_register_driver(&hyperv_pci_driver);
> + if (ret != 0)
> + return ret;
> +
> + return vmbus_driver_register(&hyperv_hv_driver);
> +}
> +
> +static void __exit hyperv_exit(void)
> +{
> + vmbus_driver_unregister(&hyperv_hv_driver);
> + pci_unregister_driver(&hyperv_pci_driver);
> +}
> +
> +module_init(hyperv_init);
> +module_exit(hyperv_exit);
> +
> +MODULE_DEVICE_TABLE(pci, hyperv_pci_tbl);
> +MODULE_DEVICE_TABLE(vmbus, hyperv_vmbus_tbl);
> +MODULE_LICENSE("GPL");
> +MODULE_AUTHOR("Deepak Rawat <drawat.floss at gmail.com>");
> +MODULE_DESCRIPTION("DRM driver for hyperv synthetic video device");
> --
> 2.29.2
>
--
Daniel Vetter
Software Engineer, Intel Corporation
http://blog.ffwll.ch
More information about the dri-devel
mailing list