etnaviv_gem.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) 2015-2018 Etnaviv Project
  4. */
  5. #include <drm/drm_prime.h>
  6. #include <drm/drm_print.h>
  7. #include <linux/dma-mapping.h>
  8. #include <linux/shmem_fs.h>
  9. #include <linux/spinlock.h>
  10. #include <linux/vmalloc.h>
  11. #include "etnaviv_drv.h"
  12. #include "etnaviv_gem.h"
  13. #include "etnaviv_gpu.h"
  14. #include "etnaviv_mmu.h"
  15. static struct lock_class_key etnaviv_shm_lock_class;
  16. static struct lock_class_key etnaviv_userptr_lock_class;
  17. static void etnaviv_gem_scatter_map(struct etnaviv_gem_object *etnaviv_obj)
  18. {
  19. struct drm_device *dev = etnaviv_obj->base.dev;
  20. struct sg_table *sgt = etnaviv_obj->sgt;
  21. /*
  22. * For non-cached buffers, ensure the new pages are clean
  23. * because display controller, GPU, etc. are not coherent.
  24. */
  25. if (etnaviv_obj->flags & ETNA_BO_CACHE_MASK)
  26. dma_map_sgtable(dev->dev, sgt, DMA_BIDIRECTIONAL, 0);
  27. }
  28. static void etnaviv_gem_scatterlist_unmap(struct etnaviv_gem_object *etnaviv_obj)
  29. {
  30. struct drm_device *dev = etnaviv_obj->base.dev;
  31. struct sg_table *sgt = etnaviv_obj->sgt;
  32. /*
  33. * For non-cached buffers, ensure the new pages are clean
  34. * because display controller, GPU, etc. are not coherent:
  35. *
  36. * WARNING: The DMA API does not support concurrent CPU
  37. * and device access to the memory area. With BIDIRECTIONAL,
  38. * we will clean the cache lines which overlap the region,
  39. * and invalidate all cache lines (partially) contained in
  40. * the region.
  41. *
  42. * If you have dirty data in the overlapping cache lines,
  43. * that will corrupt the GPU-written data. If you have
  44. * written into the remainder of the region, this can
  45. * discard those writes.
  46. */
  47. if (etnaviv_obj->flags & ETNA_BO_CACHE_MASK)
  48. dma_unmap_sgtable(dev->dev, sgt, DMA_BIDIRECTIONAL, 0);
  49. }
  50. /* called with etnaviv_obj->lock held */
  51. static int etnaviv_gem_shmem_get_pages(struct etnaviv_gem_object *etnaviv_obj)
  52. {
  53. struct drm_device *dev = etnaviv_obj->base.dev;
  54. struct page **p = drm_gem_get_pages(&etnaviv_obj->base);
  55. if (IS_ERR(p)) {
  56. dev_dbg(dev->dev, "could not get pages: %ld\n", PTR_ERR(p));
  57. return PTR_ERR(p);
  58. }
  59. etnaviv_obj->pages = p;
  60. return 0;
  61. }
  62. static void put_pages(struct etnaviv_gem_object *etnaviv_obj)
  63. {
  64. if (etnaviv_obj->sgt) {
  65. etnaviv_gem_scatterlist_unmap(etnaviv_obj);
  66. sg_free_table(etnaviv_obj->sgt);
  67. kfree(etnaviv_obj->sgt);
  68. etnaviv_obj->sgt = NULL;
  69. }
  70. if (etnaviv_obj->pages) {
  71. drm_gem_put_pages(&etnaviv_obj->base, etnaviv_obj->pages,
  72. true, false);
  73. etnaviv_obj->pages = NULL;
  74. }
  75. }
  76. struct page **etnaviv_gem_get_pages(struct etnaviv_gem_object *etnaviv_obj)
  77. {
  78. int ret;
  79. lockdep_assert_held(&etnaviv_obj->lock);
  80. if (!etnaviv_obj->pages) {
  81. ret = etnaviv_obj->ops->get_pages(etnaviv_obj);
  82. if (ret < 0)
  83. return ERR_PTR(ret);
  84. }
  85. if (!etnaviv_obj->sgt) {
  86. struct drm_device *dev = etnaviv_obj->base.dev;
  87. unsigned int npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  88. struct sg_table *sgt;
  89. sgt = drm_prime_pages_to_sg(dev, etnaviv_obj->pages, npages);
  90. if (IS_ERR(sgt)) {
  91. dev_err(dev->dev, "failed to allocate sgt: %ld\n",
  92. PTR_ERR(sgt));
  93. return ERR_CAST(sgt);
  94. }
  95. etnaviv_obj->sgt = sgt;
  96. etnaviv_gem_scatter_map(etnaviv_obj);
  97. }
  98. return etnaviv_obj->pages;
  99. }
  100. void etnaviv_gem_put_pages(struct etnaviv_gem_object *etnaviv_obj)
  101. {
  102. lockdep_assert_held(&etnaviv_obj->lock);
  103. /* when we start tracking the pin count, then do something here */
  104. }
  105. static int etnaviv_gem_mmap_obj(struct etnaviv_gem_object *etnaviv_obj,
  106. struct vm_area_struct *vma)
  107. {
  108. pgprot_t vm_page_prot;
  109. vm_flags_set(vma, VM_PFNMAP | VM_DONTEXPAND | VM_DONTDUMP);
  110. vm_page_prot = vm_get_page_prot(vma->vm_flags);
  111. if (etnaviv_obj->flags & ETNA_BO_WC) {
  112. vma->vm_page_prot = pgprot_writecombine(vm_page_prot);
  113. } else if (etnaviv_obj->flags & ETNA_BO_UNCACHED) {
  114. vma->vm_page_prot = pgprot_noncached(vm_page_prot);
  115. } else {
  116. /*
  117. * Shunt off cached objs to shmem file so they have their own
  118. * address_space (so unmap_mapping_range does what we want,
  119. * in particular in the case of mmap'd dmabufs)
  120. */
  121. vma->vm_pgoff = 0;
  122. vma_set_file(vma, etnaviv_obj->base.filp);
  123. vma->vm_page_prot = vm_page_prot;
  124. }
  125. return 0;
  126. }
  127. static int etnaviv_gem_mmap(struct drm_gem_object *obj, struct vm_area_struct *vma)
  128. {
  129. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  130. return etnaviv_obj->ops->mmap(etnaviv_obj, vma);
  131. }
  132. static vm_fault_t etnaviv_gem_fault(struct vm_fault *vmf)
  133. {
  134. struct vm_area_struct *vma = vmf->vma;
  135. struct drm_gem_object *obj = vma->vm_private_data;
  136. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  137. struct page **pages;
  138. unsigned long pfn;
  139. pgoff_t pgoff;
  140. int err;
  141. /*
  142. * Make sure we don't parallel update on a fault, nor move or remove
  143. * something from beneath our feet. Note that vmf_insert_page() is
  144. * specifically coded to take care of this, so we don't have to.
  145. */
  146. err = mutex_lock_interruptible(&etnaviv_obj->lock);
  147. if (err)
  148. return VM_FAULT_NOPAGE;
  149. /* make sure we have pages attached now */
  150. pages = etnaviv_gem_get_pages(etnaviv_obj);
  151. mutex_unlock(&etnaviv_obj->lock);
  152. if (IS_ERR(pages)) {
  153. err = PTR_ERR(pages);
  154. return vmf_error(err);
  155. }
  156. /* We don't use vmf->pgoff since that has the fake offset: */
  157. pgoff = (vmf->address - vma->vm_start) >> PAGE_SHIFT;
  158. pfn = page_to_pfn(pages[pgoff]);
  159. VERB("Inserting %p pfn %lx, pa %lx", (void *)vmf->address,
  160. pfn, pfn << PAGE_SHIFT);
  161. return vmf_insert_pfn(vma, vmf->address, pfn);
  162. }
  163. int etnaviv_gem_mmap_offset(struct drm_gem_object *obj, u64 *offset)
  164. {
  165. int ret;
  166. /* Make it mmapable */
  167. ret = drm_gem_create_mmap_offset(obj);
  168. if (ret)
  169. dev_err(obj->dev->dev, "could not allocate mmap offset\n");
  170. else
  171. *offset = drm_vma_node_offset_addr(&obj->vma_node);
  172. return ret;
  173. }
  174. static struct etnaviv_vram_mapping *
  175. etnaviv_gem_get_vram_mapping(struct etnaviv_gem_object *obj,
  176. struct etnaviv_iommu_context *context)
  177. {
  178. struct etnaviv_vram_mapping *mapping;
  179. list_for_each_entry(mapping, &obj->vram_list, obj_node) {
  180. if (mapping->context == context)
  181. return mapping;
  182. }
  183. return NULL;
  184. }
  185. void etnaviv_gem_mapping_unreference(struct etnaviv_vram_mapping *mapping)
  186. {
  187. struct etnaviv_gem_object *etnaviv_obj = mapping->object;
  188. mutex_lock(&etnaviv_obj->lock);
  189. WARN_ON(mapping->use == 0);
  190. mapping->use -= 1;
  191. mutex_unlock(&etnaviv_obj->lock);
  192. drm_gem_object_put(&etnaviv_obj->base);
  193. }
  194. struct etnaviv_vram_mapping *etnaviv_gem_mapping_get(
  195. struct drm_gem_object *obj, struct etnaviv_iommu_context *mmu_context,
  196. u64 va)
  197. {
  198. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  199. struct etnaviv_vram_mapping *mapping;
  200. struct page **pages;
  201. int ret = 0;
  202. mutex_lock(&etnaviv_obj->lock);
  203. mapping = etnaviv_gem_get_vram_mapping(etnaviv_obj, mmu_context);
  204. if (mapping) {
  205. /*
  206. * Holding the object lock prevents the use count changing
  207. * beneath us. If the use count is zero, the MMU might be
  208. * reaping this object, so take the lock and re-check that
  209. * the MMU owns this mapping to close this race.
  210. */
  211. if (mapping->use == 0) {
  212. mutex_lock(&mmu_context->lock);
  213. if (mapping->context == mmu_context)
  214. if (va && mapping->iova != va) {
  215. etnaviv_iommu_reap_mapping(mapping);
  216. mapping = NULL;
  217. } else {
  218. mapping->use += 1;
  219. }
  220. else
  221. mapping = NULL;
  222. mutex_unlock(&mmu_context->lock);
  223. if (mapping)
  224. goto out;
  225. } else {
  226. mapping->use += 1;
  227. goto out;
  228. }
  229. }
  230. pages = etnaviv_gem_get_pages(etnaviv_obj);
  231. if (IS_ERR(pages)) {
  232. ret = PTR_ERR(pages);
  233. goto out;
  234. }
  235. /*
  236. * See if we have a reaped vram mapping we can re-use before
  237. * allocating a fresh mapping.
  238. */
  239. mapping = etnaviv_gem_get_vram_mapping(etnaviv_obj, NULL);
  240. if (!mapping) {
  241. mapping = kzalloc_obj(*mapping);
  242. if (!mapping) {
  243. ret = -ENOMEM;
  244. goto out;
  245. }
  246. INIT_LIST_HEAD(&mapping->scan_node);
  247. mapping->object = etnaviv_obj;
  248. } else {
  249. list_del(&mapping->obj_node);
  250. }
  251. mapping->use = 1;
  252. ret = etnaviv_iommu_map_gem(mmu_context, etnaviv_obj,
  253. mmu_context->global->memory_base,
  254. mapping, va);
  255. if (ret < 0)
  256. kfree(mapping);
  257. else
  258. list_add_tail(&mapping->obj_node, &etnaviv_obj->vram_list);
  259. out:
  260. mutex_unlock(&etnaviv_obj->lock);
  261. if (ret)
  262. return ERR_PTR(ret);
  263. /* Take a reference on the object */
  264. drm_gem_object_get(obj);
  265. return mapping;
  266. }
  267. void *etnaviv_gem_vmap(struct drm_gem_object *obj)
  268. {
  269. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  270. if (etnaviv_obj->vaddr)
  271. return etnaviv_obj->vaddr;
  272. mutex_lock(&etnaviv_obj->lock);
  273. /*
  274. * Need to check again, as we might have raced with another thread
  275. * while waiting for the mutex.
  276. */
  277. if (!etnaviv_obj->vaddr)
  278. etnaviv_obj->vaddr = etnaviv_obj->ops->vmap(etnaviv_obj);
  279. mutex_unlock(&etnaviv_obj->lock);
  280. return etnaviv_obj->vaddr;
  281. }
  282. static void *etnaviv_gem_vmap_impl(struct etnaviv_gem_object *obj)
  283. {
  284. struct page **pages;
  285. pgprot_t prot;
  286. lockdep_assert_held(&obj->lock);
  287. pages = etnaviv_gem_get_pages(obj);
  288. if (IS_ERR(pages))
  289. return NULL;
  290. switch (obj->flags & ETNA_BO_CACHE_MASK) {
  291. case ETNA_BO_CACHED:
  292. prot = PAGE_KERNEL;
  293. break;
  294. case ETNA_BO_UNCACHED:
  295. prot = pgprot_noncached(PAGE_KERNEL);
  296. break;
  297. case ETNA_BO_WC:
  298. default:
  299. prot = pgprot_writecombine(PAGE_KERNEL);
  300. }
  301. return vmap(pages, obj->base.size >> PAGE_SHIFT, VM_MAP, prot);
  302. }
  303. static inline enum dma_data_direction etnaviv_op_to_dma_dir(u32 op)
  304. {
  305. op &= ETNA_PREP_READ | ETNA_PREP_WRITE;
  306. if (op == ETNA_PREP_READ)
  307. return DMA_FROM_DEVICE;
  308. else if (op == ETNA_PREP_WRITE)
  309. return DMA_TO_DEVICE;
  310. else
  311. return DMA_BIDIRECTIONAL;
  312. }
  313. int etnaviv_gem_cpu_prep(struct drm_gem_object *obj, u32 op,
  314. struct drm_etnaviv_timespec *timeout)
  315. {
  316. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  317. struct drm_device *dev = obj->dev;
  318. bool write = !!(op & ETNA_PREP_WRITE);
  319. int ret;
  320. if (!etnaviv_obj->sgt) {
  321. void *ret;
  322. mutex_lock(&etnaviv_obj->lock);
  323. ret = etnaviv_gem_get_pages(etnaviv_obj);
  324. mutex_unlock(&etnaviv_obj->lock);
  325. if (IS_ERR(ret))
  326. return PTR_ERR(ret);
  327. }
  328. if (op & ETNA_PREP_NOSYNC) {
  329. if (!dma_resv_test_signaled(obj->resv,
  330. dma_resv_usage_rw(write)))
  331. return -EBUSY;
  332. } else {
  333. unsigned long remain = etnaviv_timeout_to_jiffies(timeout);
  334. ret = dma_resv_wait_timeout(obj->resv, dma_resv_usage_rw(write),
  335. true, remain);
  336. if (ret <= 0)
  337. return ret == 0 ? -ETIMEDOUT : ret;
  338. }
  339. if (etnaviv_obj->flags & ETNA_BO_CACHED) {
  340. dma_sync_sgtable_for_cpu(dev->dev, etnaviv_obj->sgt,
  341. etnaviv_op_to_dma_dir(op));
  342. etnaviv_obj->last_cpu_prep_op = op;
  343. }
  344. return 0;
  345. }
  346. int etnaviv_gem_cpu_fini(struct drm_gem_object *obj)
  347. {
  348. struct drm_device *dev = obj->dev;
  349. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  350. if (etnaviv_obj->flags & ETNA_BO_CACHED) {
  351. /* fini without a prep is almost certainly a userspace error */
  352. WARN_ON(etnaviv_obj->last_cpu_prep_op == 0);
  353. dma_sync_sgtable_for_device(dev->dev, etnaviv_obj->sgt,
  354. etnaviv_op_to_dma_dir(etnaviv_obj->last_cpu_prep_op));
  355. etnaviv_obj->last_cpu_prep_op = 0;
  356. }
  357. return 0;
  358. }
  359. int etnaviv_gem_wait_bo(struct etnaviv_gpu *gpu, struct drm_gem_object *obj,
  360. struct drm_etnaviv_timespec *timeout)
  361. {
  362. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  363. return etnaviv_gpu_wait_obj_inactive(gpu, etnaviv_obj, timeout);
  364. }
  365. #ifdef CONFIG_DEBUG_FS
  366. static void etnaviv_gem_describe(struct drm_gem_object *obj, struct seq_file *m)
  367. {
  368. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  369. struct dma_resv *robj = obj->resv;
  370. unsigned long off = drm_vma_node_start(&obj->vma_node);
  371. int r;
  372. seq_printf(m, "%08x: %c %2d (%2d) %08lx %p %zd\n",
  373. etnaviv_obj->flags, is_active(etnaviv_obj) ? 'A' : 'I',
  374. obj->name, kref_read(&obj->refcount),
  375. off, etnaviv_obj->vaddr, obj->size);
  376. r = dma_resv_lock(robj, NULL);
  377. if (r)
  378. return;
  379. dma_resv_describe(robj, m);
  380. dma_resv_unlock(robj);
  381. }
  382. void etnaviv_gem_describe_objects(struct etnaviv_drm_private *priv,
  383. struct seq_file *m)
  384. {
  385. struct etnaviv_gem_object *etnaviv_obj;
  386. int count = 0;
  387. size_t size = 0;
  388. mutex_lock(&priv->gem_lock);
  389. list_for_each_entry(etnaviv_obj, &priv->gem_list, gem_node) {
  390. struct drm_gem_object *obj = &etnaviv_obj->base;
  391. seq_puts(m, " ");
  392. etnaviv_gem_describe(obj, m);
  393. count++;
  394. size += obj->size;
  395. }
  396. mutex_unlock(&priv->gem_lock);
  397. seq_printf(m, "Total %d objects, %zu bytes\n", count, size);
  398. }
  399. #endif
  400. static void etnaviv_gem_shmem_release(struct etnaviv_gem_object *etnaviv_obj)
  401. {
  402. vunmap(etnaviv_obj->vaddr);
  403. put_pages(etnaviv_obj);
  404. }
  405. static const struct etnaviv_gem_ops etnaviv_gem_shmem_ops = {
  406. .get_pages = etnaviv_gem_shmem_get_pages,
  407. .release = etnaviv_gem_shmem_release,
  408. .vmap = etnaviv_gem_vmap_impl,
  409. .mmap = etnaviv_gem_mmap_obj,
  410. };
  411. void etnaviv_gem_free_object(struct drm_gem_object *obj)
  412. {
  413. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  414. struct etnaviv_drm_private *priv = obj->dev->dev_private;
  415. struct etnaviv_vram_mapping *mapping, *tmp;
  416. /* object should not be active */
  417. WARN_ON(is_active(etnaviv_obj));
  418. mutex_lock(&priv->gem_lock);
  419. list_del(&etnaviv_obj->gem_node);
  420. mutex_unlock(&priv->gem_lock);
  421. list_for_each_entry_safe(mapping, tmp, &etnaviv_obj->vram_list,
  422. obj_node) {
  423. struct etnaviv_iommu_context *context = mapping->context;
  424. WARN_ON(mapping->use);
  425. if (context)
  426. etnaviv_iommu_unmap_gem(context, mapping);
  427. list_del(&mapping->obj_node);
  428. kfree(mapping);
  429. }
  430. etnaviv_obj->ops->release(etnaviv_obj);
  431. drm_gem_object_release(obj);
  432. mutex_destroy(&etnaviv_obj->lock);
  433. kfree(etnaviv_obj);
  434. }
  435. void etnaviv_gem_obj_add(struct drm_device *dev, struct drm_gem_object *obj)
  436. {
  437. struct etnaviv_drm_private *priv = dev->dev_private;
  438. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  439. mutex_lock(&priv->gem_lock);
  440. list_add_tail(&etnaviv_obj->gem_node, &priv->gem_list);
  441. mutex_unlock(&priv->gem_lock);
  442. }
  443. static enum drm_gem_object_status etnaviv_gem_status(struct drm_gem_object *obj)
  444. {
  445. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  446. enum drm_gem_object_status status = 0;
  447. if (etnaviv_obj->pages)
  448. status |= DRM_GEM_OBJECT_RESIDENT;
  449. return status;
  450. }
  451. static const struct vm_operations_struct vm_ops = {
  452. .fault = etnaviv_gem_fault,
  453. .open = drm_gem_vm_open,
  454. .close = drm_gem_vm_close,
  455. };
  456. static const struct drm_gem_object_funcs etnaviv_gem_object_funcs = {
  457. .free = etnaviv_gem_free_object,
  458. .pin = etnaviv_gem_prime_pin,
  459. .unpin = etnaviv_gem_prime_unpin,
  460. .get_sg_table = etnaviv_gem_prime_get_sg_table,
  461. .vmap = etnaviv_gem_prime_vmap,
  462. .mmap = etnaviv_gem_mmap,
  463. .status = etnaviv_gem_status,
  464. .vm_ops = &vm_ops,
  465. };
  466. static int etnaviv_gem_new_impl(struct drm_device *dev, u32 size, u32 flags,
  467. const struct etnaviv_gem_ops *ops, struct drm_gem_object **obj)
  468. {
  469. struct etnaviv_gem_object *etnaviv_obj;
  470. unsigned sz = sizeof(*etnaviv_obj);
  471. bool valid = true;
  472. /* validate flags */
  473. switch (flags & ETNA_BO_CACHE_MASK) {
  474. case ETNA_BO_UNCACHED:
  475. case ETNA_BO_CACHED:
  476. case ETNA_BO_WC:
  477. break;
  478. default:
  479. valid = false;
  480. }
  481. if (!valid) {
  482. dev_err(dev->dev, "invalid cache flag: %x\n",
  483. (flags & ETNA_BO_CACHE_MASK));
  484. return -EINVAL;
  485. }
  486. etnaviv_obj = kzalloc(sz, GFP_KERNEL);
  487. if (!etnaviv_obj)
  488. return -ENOMEM;
  489. etnaviv_obj->size = ALIGN(size, SZ_4K);
  490. etnaviv_obj->flags = flags;
  491. etnaviv_obj->ops = ops;
  492. mutex_init(&etnaviv_obj->lock);
  493. INIT_LIST_HEAD(&etnaviv_obj->vram_list);
  494. *obj = &etnaviv_obj->base;
  495. (*obj)->funcs = &etnaviv_gem_object_funcs;
  496. return 0;
  497. }
  498. /* convenience method to construct a GEM buffer object, and userspace handle */
  499. int etnaviv_gem_new_handle(struct drm_device *dev, struct drm_file *file,
  500. u32 size, u32 flags, u32 *handle)
  501. {
  502. struct etnaviv_drm_private *priv = dev->dev_private;
  503. struct drm_gem_object *obj = NULL;
  504. int ret;
  505. ret = etnaviv_gem_new_impl(dev, size, flags, &etnaviv_gem_shmem_ops, &obj);
  506. if (ret)
  507. goto fail;
  508. lockdep_set_class(&to_etnaviv_bo(obj)->lock, &etnaviv_shm_lock_class);
  509. ret = drm_gem_object_init(dev, obj, PAGE_ALIGN(size));
  510. if (ret)
  511. goto fail;
  512. /*
  513. * Our buffers are kept pinned, so allocating them from the MOVABLE
  514. * zone is a really bad idea, and conflicts with CMA. See comments
  515. * above new_inode() why this is required _and_ expected if you're
  516. * going to pin these pages.
  517. */
  518. mapping_set_gfp_mask(obj->filp->f_mapping, priv->shm_gfp_mask);
  519. etnaviv_gem_obj_add(dev, obj);
  520. ret = drm_gem_handle_create(file, obj, handle);
  521. /* drop reference from allocate - handle holds it now */
  522. fail:
  523. drm_gem_object_put(obj);
  524. return ret;
  525. }
  526. int etnaviv_gem_new_private(struct drm_device *dev, size_t size, u32 flags,
  527. const struct etnaviv_gem_ops *ops, struct etnaviv_gem_object **res)
  528. {
  529. struct drm_gem_object *obj;
  530. int ret;
  531. ret = etnaviv_gem_new_impl(dev, size, flags, ops, &obj);
  532. if (ret)
  533. return ret;
  534. drm_gem_private_object_init(dev, obj, size);
  535. *res = to_etnaviv_bo(obj);
  536. return 0;
  537. }
  538. static int etnaviv_gem_userptr_get_pages(struct etnaviv_gem_object *etnaviv_obj)
  539. {
  540. struct page **pvec = NULL;
  541. struct etnaviv_gem_userptr *userptr = &etnaviv_obj->userptr;
  542. int ret, pinned = 0, npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  543. unsigned int gup_flags = FOLL_LONGTERM;
  544. might_lock_read(&current->mm->mmap_lock);
  545. if (userptr->mm != current->mm)
  546. return -EPERM;
  547. pvec = kvmalloc_objs(struct page *, npages);
  548. if (!pvec)
  549. return -ENOMEM;
  550. if (!userptr->ro)
  551. gup_flags |= FOLL_WRITE;
  552. do {
  553. unsigned num_pages = npages - pinned;
  554. uint64_t ptr = userptr->ptr + pinned * PAGE_SIZE;
  555. struct page **pages = pvec + pinned;
  556. ret = pin_user_pages_fast(ptr, num_pages, gup_flags, pages);
  557. if (ret < 0) {
  558. unpin_user_pages(pvec, pinned);
  559. kvfree(pvec);
  560. return ret;
  561. }
  562. pinned += ret;
  563. } while (pinned < npages);
  564. etnaviv_obj->pages = pvec;
  565. return 0;
  566. }
  567. static void etnaviv_gem_userptr_release(struct etnaviv_gem_object *etnaviv_obj)
  568. {
  569. if (etnaviv_obj->sgt) {
  570. etnaviv_gem_scatterlist_unmap(etnaviv_obj);
  571. sg_free_table(etnaviv_obj->sgt);
  572. kfree(etnaviv_obj->sgt);
  573. }
  574. if (etnaviv_obj->pages) {
  575. unsigned int npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  576. unpin_user_pages(etnaviv_obj->pages, npages);
  577. kvfree(etnaviv_obj->pages);
  578. }
  579. }
  580. static int etnaviv_gem_userptr_mmap_obj(struct etnaviv_gem_object *etnaviv_obj,
  581. struct vm_area_struct *vma)
  582. {
  583. return -EINVAL;
  584. }
  585. static const struct etnaviv_gem_ops etnaviv_gem_userptr_ops = {
  586. .get_pages = etnaviv_gem_userptr_get_pages,
  587. .release = etnaviv_gem_userptr_release,
  588. .vmap = etnaviv_gem_vmap_impl,
  589. .mmap = etnaviv_gem_userptr_mmap_obj,
  590. };
  591. int etnaviv_gem_new_userptr(struct drm_device *dev, struct drm_file *file,
  592. uintptr_t ptr, u32 size, u32 flags, u32 *handle)
  593. {
  594. struct etnaviv_gem_object *etnaviv_obj;
  595. int ret;
  596. ret = etnaviv_gem_new_private(dev, size, ETNA_BO_CACHED,
  597. &etnaviv_gem_userptr_ops, &etnaviv_obj);
  598. if (ret)
  599. return ret;
  600. lockdep_set_class(&etnaviv_obj->lock, &etnaviv_userptr_lock_class);
  601. etnaviv_obj->userptr.ptr = ptr;
  602. etnaviv_obj->userptr.mm = current->mm;
  603. etnaviv_obj->userptr.ro = !(flags & ETNA_USERPTR_WRITE);
  604. etnaviv_gem_obj_add(dev, &etnaviv_obj->base);
  605. ret = drm_gem_handle_create(file, &etnaviv_obj->base, handle);
  606. /* drop reference from allocate - handle holds it now */
  607. drm_gem_object_put(&etnaviv_obj->base);
  608. return ret;
  609. }