]> git.baikalelectronics.ru Git - kernel.git/commitdiff
drm/vmwgfx: Be a lot more flexible with MOB limits
authorZack Rusin <zackr@vmware.com>
Fri, 23 Jul 2021 16:51:52 +0000 (12:51 -0400)
committerZack Rusin <zackr@vmware.com>
Wed, 28 Jul 2021 18:53:25 +0000 (14:53 -0400)
The code was trying to keep a strict limit on the amount of mob
memory that was used in the guest by making it match the host
settings. There's technically no reason to do that (guests can
certainly use more than the host can have resident in renderers
at the same time).

In particular this is problematic because our userspace is not
great at handling OOM conditions and running out of MOB space
results in GL apps crashing, e.g. gnome-shell likes to allocate
huge surfaces (~61MB for the desktop on 2560x1600 with two workspaces)
and running out of memory there means that the gnome-shell crashes
on startup taking us back to the login and resulting in a system
where one can not login in graphically anymore.

Instead of letting the userspace crash we can extend available
MOB space, we just don't want to use all of the RAM for graphics,
so we're going to limit it to half of RAM.

With the addition of some extra logging this should make the
"guest has been configured with not enough graphics memory"
errors a lot easier to diagnose in cases where the automatic
expansion of MOB space fails.

Signed-off-by: Zack Rusin <zackr@vmware.com>
Reviewed-by: Martin Krastev <krastevm@vmware.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20210723165153.113198-3-zackr@vmware.com
drivers/gpu/drm/vmwgfx/vmwgfx_drv.c
drivers/gpu/drm/vmwgfx/vmwgfx_gmrid_manager.c

index 049ea3b8229b74388a39f9934f979052a207a325..ab9a1750e1dffba51e491e40afbfc312b207b1e8 100644 (file)
@@ -942,6 +942,15 @@ static int vmw_driver_load(struct vmw_private *dev_priv, u32 pci_id)
                dev_priv->texture_max_height = 8192;
                dev_priv->max_primary_mem = dev_priv->vram_size;
        }
+       drm_info(&dev_priv->drm,
+                "Legacy memory limits: VRAM = %llu kB, FIFO = %llu kB, surface = %u kB\n",
+                (u64)dev_priv->vram_size / 1024,
+                (u64)dev_priv->fifo_mem_size / 1024,
+                dev_priv->memory_size / 1024);
+
+       drm_info(&dev_priv->drm,
+                "MOB limits: max mob size = %u kB, max mob pages = %u\n",
+                dev_priv->max_mob_size / 1024, dev_priv->max_mob_pages);
 
        vmw_print_bitmap(&dev_priv->drm, "Capabilities",
                         dev_priv->capabilities,
index 28ceb749a7339687fd46f5ba5c7ab07407b10b53..b2c4af331c9d518a2c77d886c60ee86185dce4e8 100644 (file)
@@ -71,8 +71,40 @@ static int vmw_gmrid_man_get_node(struct ttm_resource_manager *man,
 
        if (gman->max_gmr_pages > 0) {
                gman->used_gmr_pages += (*res)->num_pages;
-               if (unlikely(gman->used_gmr_pages > gman->max_gmr_pages))
-                       goto nospace;
+               /*
+                * Because the graphics memory is a soft limit we can try to
+                * expand it instead of letting the userspace apps crash.
+                * We're just going to have a sane limit (half of RAM)
+                * on the number of MOB's that we create and will try to keep
+                * the system running until we reach that.
+                */
+               if (unlikely(gman->used_gmr_pages > gman->max_gmr_pages)) {
+                       const unsigned long max_graphics_pages = totalram_pages() / 2;
+                       uint32_t new_max_pages = 0;
+
+                       DRM_WARN("vmwgfx: mob memory overflow. Consider increasing guest RAM and graphicsMemory.\n");
+                       vmw_host_printf("vmwgfx, warning: mob memory overflow. Consider increasing guest RAM and graphicsMemory.\n");
+
+                       if (gman->max_gmr_pages > (max_graphics_pages / 2)) {
+                               DRM_WARN("vmwgfx: guest requires more than half of RAM for graphics.\n");
+                               new_max_pages = max_graphics_pages;
+                       } else
+                               new_max_pages = gman->max_gmr_pages * 2;
+                       if (new_max_pages > gman->max_gmr_pages && new_max_pages >= gman->used_gmr_pages) {
+                               DRM_WARN("vmwgfx: increasing guest mob limits to %u kB.\n",
+                                        ((new_max_pages) << (PAGE_SHIFT - 10)));
+
+                               gman->max_gmr_pages = new_max_pages;
+                       } else {
+                               char buf[256];
+                               snprintf(buf, sizeof(buf),
+                                        "vmwgfx, error: guest graphics is out of memory (mob limit at: %ukB).\n",
+                                        ((gman->max_gmr_pages) << (PAGE_SHIFT - 10)));
+                               vmw_host_printf(buf);
+                               DRM_WARN("%s", buf);
+                               goto nospace;
+                       }
+               }
        }
 
        (*res)->start = id;