huge_gem_object.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. /*
  2. * Copyright © 2016 Intel Corporation
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a
  5. * copy of this software and associated documentation files (the "Software"),
  6. * to deal in the Software without restriction, including without limitation
  7. * the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. * and/or sell copies of the Software, and to permit persons to whom the
  9. * Software is furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice (including the next
  12. * paragraph) shall be included in all copies or substantial portions of the
  13. * Software.
  14. *
  15. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
  18. * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  20. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  21. * IN THE SOFTWARE.
  22. *
  23. */
  24. #include "huge_gem_object.h"
  25. static void huge_free_pages(struct drm_i915_gem_object *obj,
  26. struct sg_table *pages)
  27. {
  28. unsigned long nreal = obj->scratch / PAGE_SIZE;
  29. struct scatterlist *sg;
  30. for (sg = pages->sgl; sg && nreal--; sg = __sg_next(sg))
  31. __free_page(sg_page(sg));
  32. sg_free_table(pages);
  33. kfree(pages);
  34. }
  35. static struct sg_table *
  36. huge_get_pages(struct drm_i915_gem_object *obj)
  37. {
  38. #define GFP (GFP_KERNEL | __GFP_NOWARN | __GFP_NORETRY)
  39. const unsigned long nreal = obj->scratch / PAGE_SIZE;
  40. const unsigned long npages = obj->base.size / PAGE_SIZE;
  41. struct scatterlist *sg, *src, *end;
  42. struct sg_table *pages;
  43. unsigned long n;
  44. pages = kmalloc(sizeof(*pages), GFP);
  45. if (!pages)
  46. return ERR_PTR(-ENOMEM);
  47. if (sg_alloc_table(pages, npages, GFP)) {
  48. kfree(pages);
  49. return ERR_PTR(-ENOMEM);
  50. }
  51. sg = pages->sgl;
  52. for (n = 0; n < nreal; n++) {
  53. struct page *page;
  54. page = alloc_page(GFP | __GFP_HIGHMEM);
  55. if (!page) {
  56. sg_mark_end(sg);
  57. goto err;
  58. }
  59. sg_set_page(sg, page, PAGE_SIZE, 0);
  60. sg = __sg_next(sg);
  61. }
  62. if (nreal < npages) {
  63. for (end = sg, src = pages->sgl; sg; sg = __sg_next(sg)) {
  64. sg_set_page(sg, sg_page(src), PAGE_SIZE, 0);
  65. src = __sg_next(src);
  66. if (src == end)
  67. src = pages->sgl;
  68. }
  69. }
  70. if (i915_gem_gtt_prepare_pages(obj, pages))
  71. goto err;
  72. return pages;
  73. err:
  74. huge_free_pages(obj, pages);
  75. return ERR_PTR(-ENOMEM);
  76. #undef GFP
  77. }
  78. static void huge_put_pages(struct drm_i915_gem_object *obj,
  79. struct sg_table *pages)
  80. {
  81. i915_gem_gtt_finish_pages(obj, pages);
  82. huge_free_pages(obj, pages);
  83. obj->mm.dirty = false;
  84. }
  85. static const struct drm_i915_gem_object_ops huge_ops = {
  86. .flags = I915_GEM_OBJECT_HAS_STRUCT_PAGE |
  87. I915_GEM_OBJECT_IS_SHRINKABLE,
  88. .get_pages = huge_get_pages,
  89. .put_pages = huge_put_pages,
  90. };
  91. struct drm_i915_gem_object *
  92. huge_gem_object(struct drm_i915_private *i915,
  93. phys_addr_t phys_size,
  94. dma_addr_t dma_size)
  95. {
  96. struct drm_i915_gem_object *obj;
  97. GEM_BUG_ON(!phys_size || phys_size > dma_size);
  98. GEM_BUG_ON(!IS_ALIGNED(phys_size, PAGE_SIZE));
  99. GEM_BUG_ON(!IS_ALIGNED(dma_size, I915_GTT_PAGE_SIZE));
  100. if (overflows_type(dma_size, obj->base.size))
  101. return ERR_PTR(-E2BIG);
  102. obj = i915_gem_object_alloc(i915);
  103. if (!obj)
  104. return ERR_PTR(-ENOMEM);
  105. drm_gem_private_object_init(&i915->drm, &obj->base, dma_size);
  106. i915_gem_object_init(obj, &huge_ops);
  107. obj->base.write_domain = I915_GEM_DOMAIN_CPU;
  108. obj->base.read_domains = I915_GEM_DOMAIN_CPU;
  109. obj->cache_level = HAS_LLC(i915) ? I915_CACHE_LLC : I915_CACHE_NONE;
  110. obj->scratch = phys_size;
  111. return obj;
  112. }