huge_gem_object.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  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 int huge_get_pages(struct drm_i915_gem_object *obj)
  36. {
  37. #define GFP (GFP_KERNEL | __GFP_NOWARN | __GFP_NORETRY)
  38. const unsigned long nreal = obj->scratch / PAGE_SIZE;
  39. const unsigned long npages = obj->base.size / PAGE_SIZE;
  40. struct scatterlist *sg, *src, *end;
  41. struct sg_table *pages;
  42. unsigned long n;
  43. pages = kmalloc(sizeof(*pages), GFP);
  44. if (!pages)
  45. return -ENOMEM;
  46. if (sg_alloc_table(pages, npages, GFP)) {
  47. kfree(pages);
  48. return -ENOMEM;
  49. }
  50. sg = pages->sgl;
  51. for (n = 0; n < nreal; n++) {
  52. struct page *page;
  53. page = alloc_page(GFP | __GFP_HIGHMEM);
  54. if (!page) {
  55. sg_mark_end(sg);
  56. goto err;
  57. }
  58. sg_set_page(sg, page, PAGE_SIZE, 0);
  59. sg = __sg_next(sg);
  60. }
  61. if (nreal < npages) {
  62. for (end = sg, src = pages->sgl; sg; sg = __sg_next(sg)) {
  63. sg_set_page(sg, sg_page(src), PAGE_SIZE, 0);
  64. src = __sg_next(src);
  65. if (src == end)
  66. src = pages->sgl;
  67. }
  68. }
  69. if (i915_gem_gtt_prepare_pages(obj, pages))
  70. goto err;
  71. __i915_gem_object_set_pages(obj, pages, PAGE_SIZE);
  72. return 0;
  73. err:
  74. huge_free_pages(obj, pages);
  75. return -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. unsigned int cache_level;
  98. GEM_BUG_ON(!phys_size || phys_size > dma_size);
  99. GEM_BUG_ON(!IS_ALIGNED(phys_size, PAGE_SIZE));
  100. GEM_BUG_ON(!IS_ALIGNED(dma_size, I915_GTT_PAGE_SIZE));
  101. if (overflows_type(dma_size, obj->base.size))
  102. return ERR_PTR(-E2BIG);
  103. obj = i915_gem_object_alloc(i915);
  104. if (!obj)
  105. return ERR_PTR(-ENOMEM);
  106. drm_gem_private_object_init(&i915->drm, &obj->base, dma_size);
  107. i915_gem_object_init(obj, &huge_ops);
  108. obj->read_domains = I915_GEM_DOMAIN_CPU;
  109. obj->write_domain = I915_GEM_DOMAIN_CPU;
  110. cache_level = HAS_LLC(i915) ? I915_CACHE_LLC : I915_CACHE_NONE;
  111. i915_gem_object_set_cache_coherency(obj, cache_level);
  112. obj->scratch = phys_size;
  113. return obj;
  114. }