diff options
-rw-r--r-- | drivers/gpu/drm/i915/i915_gem_gtt.c | 22 |
1 files changed, 18 insertions, 4 deletions
diff --git a/drivers/gpu/drm/i915/i915_gem_gtt.c b/drivers/gpu/drm/i915/i915_gem_gtt.c index 00520f27bea6..f698006fe883 100644 --- a/drivers/gpu/drm/i915/i915_gem_gtt.c +++ b/drivers/gpu/drm/i915/i915_gem_gtt.c | |||
@@ -2361,10 +2361,24 @@ void i915_gem_suspend_gtt_mappings(struct drm_i915_private *dev_priv) | |||
2361 | int i915_gem_gtt_prepare_pages(struct drm_i915_gem_object *obj, | 2361 | int i915_gem_gtt_prepare_pages(struct drm_i915_gem_object *obj, |
2362 | struct sg_table *pages) | 2362 | struct sg_table *pages) |
2363 | { | 2363 | { |
2364 | if (dma_map_sg(&obj->base.dev->pdev->dev, | 2364 | do { |
2365 | pages->sgl, pages->nents, | 2365 | if (dma_map_sg(&obj->base.dev->pdev->dev, |
2366 | PCI_DMA_BIDIRECTIONAL)) | 2366 | pages->sgl, pages->nents, |
2367 | return 0; | 2367 | PCI_DMA_BIDIRECTIONAL)) |
2368 | return 0; | ||
2369 | |||
2370 | /* If the DMA remap fails, one cause can be that we have | ||
2371 | * too many objects pinned in a small remapping table, | ||
2372 | * such as swiotlb. Incrementally purge all other objects and | ||
2373 | * try again - if there are no more pages to remove from | ||
2374 | * the DMA remapper, i915_gem_shrink will return 0. | ||
2375 | */ | ||
2376 | GEM_BUG_ON(obj->mm.pages == pages); | ||
2377 | } while (i915_gem_shrink(to_i915(obj->base.dev), | ||
2378 | obj->base.size >> PAGE_SHIFT, | ||
2379 | I915_SHRINK_BOUND | | ||
2380 | I915_SHRINK_UNBOUND | | ||
2381 | I915_SHRINK_ACTIVE)); | ||
2368 | 2382 | ||
2369 | return -ENOSPC; | 2383 | return -ENOSPC; |
2370 | } | 2384 | } |