etnaviv_gem.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922
  1. /*
  2. * Copyright (C) 2015 Etnaviv Project
  3. *
  4. * This program is free software; you can redistribute it and/or modify it
  5. * under the terms of the GNU General Public License version 2 as published by
  6. * the Free Software Foundation.
  7. *
  8. * This program is distributed in the hope that it will be useful, but WITHOUT
  9. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  11. * more details.
  12. *
  13. * You should have received a copy of the GNU General Public License along with
  14. * this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include <linux/spinlock.h>
  17. #include <linux/shmem_fs.h>
  18. #include "etnaviv_drv.h"
  19. #include "etnaviv_gem.h"
  20. #include "etnaviv_gpu.h"
  21. #include "etnaviv_mmu.h"
  22. static void etnaviv_gem_scatter_map(struct etnaviv_gem_object *etnaviv_obj)
  23. {
  24. struct drm_device *dev = etnaviv_obj->base.dev;
  25. struct sg_table *sgt = etnaviv_obj->sgt;
  26. /*
  27. * For non-cached buffers, ensure the new pages are clean
  28. * because display controller, GPU, etc. are not coherent.
  29. */
  30. if (etnaviv_obj->flags & ETNA_BO_CACHE_MASK)
  31. dma_map_sg(dev->dev, sgt->sgl, sgt->nents, DMA_BIDIRECTIONAL);
  32. }
  33. static void etnaviv_gem_scatterlist_unmap(struct etnaviv_gem_object *etnaviv_obj)
  34. {
  35. struct drm_device *dev = etnaviv_obj->base.dev;
  36. struct sg_table *sgt = etnaviv_obj->sgt;
  37. /*
  38. * For non-cached buffers, ensure the new pages are clean
  39. * because display controller, GPU, etc. are not coherent:
  40. *
  41. * WARNING: The DMA API does not support concurrent CPU
  42. * and device access to the memory area. With BIDIRECTIONAL,
  43. * we will clean the cache lines which overlap the region,
  44. * and invalidate all cache lines (partially) contained in
  45. * the region.
  46. *
  47. * If you have dirty data in the overlapping cache lines,
  48. * that will corrupt the GPU-written data. If you have
  49. * written into the remainder of the region, this can
  50. * discard those writes.
  51. */
  52. if (etnaviv_obj->flags & ETNA_BO_CACHE_MASK)
  53. dma_unmap_sg(dev->dev, sgt->sgl, sgt->nents, DMA_BIDIRECTIONAL);
  54. }
  55. /* called with etnaviv_obj->lock held */
  56. static int etnaviv_gem_shmem_get_pages(struct etnaviv_gem_object *etnaviv_obj)
  57. {
  58. struct drm_device *dev = etnaviv_obj->base.dev;
  59. struct page **p = drm_gem_get_pages(&etnaviv_obj->base);
  60. if (IS_ERR(p)) {
  61. dev_err(dev->dev, "could not get pages: %ld\n", PTR_ERR(p));
  62. return PTR_ERR(p);
  63. }
  64. etnaviv_obj->pages = p;
  65. return 0;
  66. }
  67. static void put_pages(struct etnaviv_gem_object *etnaviv_obj)
  68. {
  69. if (etnaviv_obj->sgt) {
  70. etnaviv_gem_scatterlist_unmap(etnaviv_obj);
  71. sg_free_table(etnaviv_obj->sgt);
  72. kfree(etnaviv_obj->sgt);
  73. etnaviv_obj->sgt = NULL;
  74. }
  75. if (etnaviv_obj->pages) {
  76. drm_gem_put_pages(&etnaviv_obj->base, etnaviv_obj->pages,
  77. true, false);
  78. etnaviv_obj->pages = NULL;
  79. }
  80. }
  81. struct page **etnaviv_gem_get_pages(struct etnaviv_gem_object *etnaviv_obj)
  82. {
  83. int ret;
  84. lockdep_assert_held(&etnaviv_obj->lock);
  85. if (!etnaviv_obj->pages) {
  86. ret = etnaviv_obj->ops->get_pages(etnaviv_obj);
  87. if (ret < 0)
  88. return ERR_PTR(ret);
  89. }
  90. if (!etnaviv_obj->sgt) {
  91. struct drm_device *dev = etnaviv_obj->base.dev;
  92. int npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  93. struct sg_table *sgt;
  94. sgt = drm_prime_pages_to_sg(etnaviv_obj->pages, npages);
  95. if (IS_ERR(sgt)) {
  96. dev_err(dev->dev, "failed to allocate sgt: %ld\n",
  97. PTR_ERR(sgt));
  98. return ERR_CAST(sgt);
  99. }
  100. etnaviv_obj->sgt = sgt;
  101. etnaviv_gem_scatter_map(etnaviv_obj);
  102. }
  103. return etnaviv_obj->pages;
  104. }
  105. void etnaviv_gem_put_pages(struct etnaviv_gem_object *etnaviv_obj)
  106. {
  107. lockdep_assert_held(&etnaviv_obj->lock);
  108. /* when we start tracking the pin count, then do something here */
  109. }
  110. static int etnaviv_gem_mmap_obj(struct etnaviv_gem_object *etnaviv_obj,
  111. struct vm_area_struct *vma)
  112. {
  113. pgprot_t vm_page_prot;
  114. vma->vm_flags &= ~VM_PFNMAP;
  115. vma->vm_flags |= VM_MIXEDMAP;
  116. vm_page_prot = vm_get_page_prot(vma->vm_flags);
  117. if (etnaviv_obj->flags & ETNA_BO_WC) {
  118. vma->vm_page_prot = pgprot_writecombine(vm_page_prot);
  119. } else if (etnaviv_obj->flags & ETNA_BO_UNCACHED) {
  120. vma->vm_page_prot = pgprot_noncached(vm_page_prot);
  121. } else {
  122. /*
  123. * Shunt off cached objs to shmem file so they have their own
  124. * address_space (so unmap_mapping_range does what we want,
  125. * in particular in the case of mmap'd dmabufs)
  126. */
  127. fput(vma->vm_file);
  128. get_file(etnaviv_obj->base.filp);
  129. vma->vm_pgoff = 0;
  130. vma->vm_file = etnaviv_obj->base.filp;
  131. vma->vm_page_prot = vm_page_prot;
  132. }
  133. return 0;
  134. }
  135. int etnaviv_gem_mmap(struct file *filp, struct vm_area_struct *vma)
  136. {
  137. struct etnaviv_gem_object *obj;
  138. int ret;
  139. ret = drm_gem_mmap(filp, vma);
  140. if (ret) {
  141. DBG("mmap failed: %d", ret);
  142. return ret;
  143. }
  144. obj = to_etnaviv_bo(vma->vm_private_data);
  145. return obj->ops->mmap(obj, vma);
  146. }
  147. int etnaviv_gem_fault(struct vm_area_struct *vma, struct vm_fault *vmf)
  148. {
  149. struct drm_gem_object *obj = vma->vm_private_data;
  150. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  151. struct page **pages, *page;
  152. pgoff_t pgoff;
  153. int ret;
  154. /*
  155. * Make sure we don't parallel update on a fault, nor move or remove
  156. * something from beneath our feet. Note that vm_insert_page() is
  157. * specifically coded to take care of this, so we don't have to.
  158. */
  159. ret = mutex_lock_interruptible(&etnaviv_obj->lock);
  160. if (ret)
  161. goto out;
  162. /* make sure we have pages attached now */
  163. pages = etnaviv_gem_get_pages(etnaviv_obj);
  164. mutex_unlock(&etnaviv_obj->lock);
  165. if (IS_ERR(pages)) {
  166. ret = PTR_ERR(pages);
  167. goto out;
  168. }
  169. /* We don't use vmf->pgoff since that has the fake offset: */
  170. pgoff = (vmf->address - vma->vm_start) >> PAGE_SHIFT;
  171. page = pages[pgoff];
  172. VERB("Inserting %p pfn %lx, pa %lx", (void *)vmf->address,
  173. page_to_pfn(page), page_to_pfn(page) << PAGE_SHIFT);
  174. ret = vm_insert_page(vma, vmf->address, page);
  175. out:
  176. switch (ret) {
  177. case -EAGAIN:
  178. case 0:
  179. case -ERESTARTSYS:
  180. case -EINTR:
  181. case -EBUSY:
  182. /*
  183. * EBUSY is ok: this just means that another thread
  184. * already did the job.
  185. */
  186. return VM_FAULT_NOPAGE;
  187. case -ENOMEM:
  188. return VM_FAULT_OOM;
  189. default:
  190. return VM_FAULT_SIGBUS;
  191. }
  192. }
  193. int etnaviv_gem_mmap_offset(struct drm_gem_object *obj, u64 *offset)
  194. {
  195. int ret;
  196. /* Make it mmapable */
  197. ret = drm_gem_create_mmap_offset(obj);
  198. if (ret)
  199. dev_err(obj->dev->dev, "could not allocate mmap offset\n");
  200. else
  201. *offset = drm_vma_node_offset_addr(&obj->vma_node);
  202. return ret;
  203. }
  204. static struct etnaviv_vram_mapping *
  205. etnaviv_gem_get_vram_mapping(struct etnaviv_gem_object *obj,
  206. struct etnaviv_iommu *mmu)
  207. {
  208. struct etnaviv_vram_mapping *mapping;
  209. list_for_each_entry(mapping, &obj->vram_list, obj_node) {
  210. if (mapping->mmu == mmu)
  211. return mapping;
  212. }
  213. return NULL;
  214. }
  215. void etnaviv_gem_mapping_reference(struct etnaviv_vram_mapping *mapping)
  216. {
  217. struct etnaviv_gem_object *etnaviv_obj = mapping->object;
  218. drm_gem_object_reference(&etnaviv_obj->base);
  219. mutex_lock(&etnaviv_obj->lock);
  220. WARN_ON(mapping->use == 0);
  221. mapping->use += 1;
  222. mutex_unlock(&etnaviv_obj->lock);
  223. }
  224. void etnaviv_gem_mapping_unreference(struct etnaviv_vram_mapping *mapping)
  225. {
  226. struct etnaviv_gem_object *etnaviv_obj = mapping->object;
  227. mutex_lock(&etnaviv_obj->lock);
  228. WARN_ON(mapping->use == 0);
  229. mapping->use -= 1;
  230. mutex_unlock(&etnaviv_obj->lock);
  231. drm_gem_object_unreference_unlocked(&etnaviv_obj->base);
  232. }
  233. struct etnaviv_vram_mapping *etnaviv_gem_mapping_get(
  234. struct drm_gem_object *obj, struct etnaviv_gpu *gpu)
  235. {
  236. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  237. struct etnaviv_vram_mapping *mapping;
  238. struct page **pages;
  239. int ret = 0;
  240. mutex_lock(&etnaviv_obj->lock);
  241. mapping = etnaviv_gem_get_vram_mapping(etnaviv_obj, gpu->mmu);
  242. if (mapping) {
  243. /*
  244. * Holding the object lock prevents the use count changing
  245. * beneath us. If the use count is zero, the MMU might be
  246. * reaping this object, so take the lock and re-check that
  247. * the MMU owns this mapping to close this race.
  248. */
  249. if (mapping->use == 0) {
  250. mutex_lock(&gpu->mmu->lock);
  251. if (mapping->mmu == gpu->mmu)
  252. mapping->use += 1;
  253. else
  254. mapping = NULL;
  255. mutex_unlock(&gpu->mmu->lock);
  256. if (mapping)
  257. goto out;
  258. } else {
  259. mapping->use += 1;
  260. goto out;
  261. }
  262. }
  263. pages = etnaviv_gem_get_pages(etnaviv_obj);
  264. if (IS_ERR(pages)) {
  265. ret = PTR_ERR(pages);
  266. goto out;
  267. }
  268. /*
  269. * See if we have a reaped vram mapping we can re-use before
  270. * allocating a fresh mapping.
  271. */
  272. mapping = etnaviv_gem_get_vram_mapping(etnaviv_obj, NULL);
  273. if (!mapping) {
  274. mapping = kzalloc(sizeof(*mapping), GFP_KERNEL);
  275. if (!mapping) {
  276. ret = -ENOMEM;
  277. goto out;
  278. }
  279. INIT_LIST_HEAD(&mapping->scan_node);
  280. mapping->object = etnaviv_obj;
  281. } else {
  282. list_del(&mapping->obj_node);
  283. }
  284. mapping->mmu = gpu->mmu;
  285. mapping->use = 1;
  286. ret = etnaviv_iommu_map_gem(gpu->mmu, etnaviv_obj, gpu->memory_base,
  287. mapping);
  288. if (ret < 0)
  289. kfree(mapping);
  290. else
  291. list_add_tail(&mapping->obj_node, &etnaviv_obj->vram_list);
  292. out:
  293. mutex_unlock(&etnaviv_obj->lock);
  294. if (ret)
  295. return ERR_PTR(ret);
  296. /* Take a reference on the object */
  297. drm_gem_object_reference(obj);
  298. return mapping;
  299. }
  300. void *etnaviv_gem_vmap(struct drm_gem_object *obj)
  301. {
  302. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  303. if (etnaviv_obj->vaddr)
  304. return etnaviv_obj->vaddr;
  305. mutex_lock(&etnaviv_obj->lock);
  306. /*
  307. * Need to check again, as we might have raced with another thread
  308. * while waiting for the mutex.
  309. */
  310. if (!etnaviv_obj->vaddr)
  311. etnaviv_obj->vaddr = etnaviv_obj->ops->vmap(etnaviv_obj);
  312. mutex_unlock(&etnaviv_obj->lock);
  313. return etnaviv_obj->vaddr;
  314. }
  315. static void *etnaviv_gem_vmap_impl(struct etnaviv_gem_object *obj)
  316. {
  317. struct page **pages;
  318. lockdep_assert_held(&obj->lock);
  319. pages = etnaviv_gem_get_pages(obj);
  320. if (IS_ERR(pages))
  321. return NULL;
  322. return vmap(pages, obj->base.size >> PAGE_SHIFT,
  323. VM_MAP, pgprot_writecombine(PAGE_KERNEL));
  324. }
  325. static inline enum dma_data_direction etnaviv_op_to_dma_dir(u32 op)
  326. {
  327. if (op & ETNA_PREP_READ)
  328. return DMA_FROM_DEVICE;
  329. else if (op & ETNA_PREP_WRITE)
  330. return DMA_TO_DEVICE;
  331. else
  332. return DMA_BIDIRECTIONAL;
  333. }
  334. int etnaviv_gem_cpu_prep(struct drm_gem_object *obj, u32 op,
  335. struct timespec *timeout)
  336. {
  337. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  338. struct drm_device *dev = obj->dev;
  339. bool write = !!(op & ETNA_PREP_WRITE);
  340. unsigned long remain =
  341. op & ETNA_PREP_NOSYNC ? 0 : etnaviv_timeout_to_jiffies(timeout);
  342. long lret;
  343. lret = reservation_object_wait_timeout_rcu(etnaviv_obj->resv,
  344. write, true, remain);
  345. if (lret < 0)
  346. return lret;
  347. else if (lret == 0)
  348. return remain == 0 ? -EBUSY : -ETIMEDOUT;
  349. if (etnaviv_obj->flags & ETNA_BO_CACHED) {
  350. if (!etnaviv_obj->sgt) {
  351. void *ret;
  352. mutex_lock(&etnaviv_obj->lock);
  353. ret = etnaviv_gem_get_pages(etnaviv_obj);
  354. mutex_unlock(&etnaviv_obj->lock);
  355. if (IS_ERR(ret))
  356. return PTR_ERR(ret);
  357. }
  358. dma_sync_sg_for_cpu(dev->dev, etnaviv_obj->sgt->sgl,
  359. etnaviv_obj->sgt->nents,
  360. etnaviv_op_to_dma_dir(op));
  361. etnaviv_obj->last_cpu_prep_op = op;
  362. }
  363. return 0;
  364. }
  365. int etnaviv_gem_cpu_fini(struct drm_gem_object *obj)
  366. {
  367. struct drm_device *dev = obj->dev;
  368. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  369. if (etnaviv_obj->flags & ETNA_BO_CACHED) {
  370. /* fini without a prep is almost certainly a userspace error */
  371. WARN_ON(etnaviv_obj->last_cpu_prep_op == 0);
  372. dma_sync_sg_for_device(dev->dev, etnaviv_obj->sgt->sgl,
  373. etnaviv_obj->sgt->nents,
  374. etnaviv_op_to_dma_dir(etnaviv_obj->last_cpu_prep_op));
  375. etnaviv_obj->last_cpu_prep_op = 0;
  376. }
  377. return 0;
  378. }
  379. int etnaviv_gem_wait_bo(struct etnaviv_gpu *gpu, struct drm_gem_object *obj,
  380. struct timespec *timeout)
  381. {
  382. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  383. return etnaviv_gpu_wait_obj_inactive(gpu, etnaviv_obj, timeout);
  384. }
  385. #ifdef CONFIG_DEBUG_FS
  386. static void etnaviv_gem_describe_fence(struct dma_fence *fence,
  387. const char *type, struct seq_file *m)
  388. {
  389. if (!test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags))
  390. seq_printf(m, "\t%9s: %s %s seq %u\n",
  391. type,
  392. fence->ops->get_driver_name(fence),
  393. fence->ops->get_timeline_name(fence),
  394. fence->seqno);
  395. }
  396. static void etnaviv_gem_describe(struct drm_gem_object *obj, struct seq_file *m)
  397. {
  398. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  399. struct reservation_object *robj = etnaviv_obj->resv;
  400. struct reservation_object_list *fobj;
  401. struct dma_fence *fence;
  402. unsigned long off = drm_vma_node_start(&obj->vma_node);
  403. seq_printf(m, "%08x: %c %2d (%2d) %08lx %p %zd\n",
  404. etnaviv_obj->flags, is_active(etnaviv_obj) ? 'A' : 'I',
  405. obj->name, obj->refcount.refcount.counter,
  406. off, etnaviv_obj->vaddr, obj->size);
  407. rcu_read_lock();
  408. fobj = rcu_dereference(robj->fence);
  409. if (fobj) {
  410. unsigned int i, shared_count = fobj->shared_count;
  411. for (i = 0; i < shared_count; i++) {
  412. fence = rcu_dereference(fobj->shared[i]);
  413. etnaviv_gem_describe_fence(fence, "Shared", m);
  414. }
  415. }
  416. fence = rcu_dereference(robj->fence_excl);
  417. if (fence)
  418. etnaviv_gem_describe_fence(fence, "Exclusive", m);
  419. rcu_read_unlock();
  420. }
  421. void etnaviv_gem_describe_objects(struct etnaviv_drm_private *priv,
  422. struct seq_file *m)
  423. {
  424. struct etnaviv_gem_object *etnaviv_obj;
  425. int count = 0;
  426. size_t size = 0;
  427. mutex_lock(&priv->gem_lock);
  428. list_for_each_entry(etnaviv_obj, &priv->gem_list, gem_node) {
  429. struct drm_gem_object *obj = &etnaviv_obj->base;
  430. seq_puts(m, " ");
  431. etnaviv_gem_describe(obj, m);
  432. count++;
  433. size += obj->size;
  434. }
  435. mutex_unlock(&priv->gem_lock);
  436. seq_printf(m, "Total %d objects, %zu bytes\n", count, size);
  437. }
  438. #endif
  439. static void etnaviv_gem_shmem_release(struct etnaviv_gem_object *etnaviv_obj)
  440. {
  441. vunmap(etnaviv_obj->vaddr);
  442. put_pages(etnaviv_obj);
  443. }
  444. static const struct etnaviv_gem_ops etnaviv_gem_shmem_ops = {
  445. .get_pages = etnaviv_gem_shmem_get_pages,
  446. .release = etnaviv_gem_shmem_release,
  447. .vmap = etnaviv_gem_vmap_impl,
  448. .mmap = etnaviv_gem_mmap_obj,
  449. };
  450. void etnaviv_gem_free_object(struct drm_gem_object *obj)
  451. {
  452. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  453. struct etnaviv_vram_mapping *mapping, *tmp;
  454. /* object should not be active */
  455. WARN_ON(is_active(etnaviv_obj));
  456. list_del(&etnaviv_obj->gem_node);
  457. list_for_each_entry_safe(mapping, tmp, &etnaviv_obj->vram_list,
  458. obj_node) {
  459. struct etnaviv_iommu *mmu = mapping->mmu;
  460. WARN_ON(mapping->use);
  461. if (mmu)
  462. etnaviv_iommu_unmap_gem(mmu, mapping);
  463. list_del(&mapping->obj_node);
  464. kfree(mapping);
  465. }
  466. drm_gem_free_mmap_offset(obj);
  467. etnaviv_obj->ops->release(etnaviv_obj);
  468. if (etnaviv_obj->resv == &etnaviv_obj->_resv)
  469. reservation_object_fini(&etnaviv_obj->_resv);
  470. drm_gem_object_release(obj);
  471. kfree(etnaviv_obj);
  472. }
  473. int etnaviv_gem_obj_add(struct drm_device *dev, struct drm_gem_object *obj)
  474. {
  475. struct etnaviv_drm_private *priv = dev->dev_private;
  476. struct etnaviv_gem_object *etnaviv_obj = to_etnaviv_bo(obj);
  477. mutex_lock(&priv->gem_lock);
  478. list_add_tail(&etnaviv_obj->gem_node, &priv->gem_list);
  479. mutex_unlock(&priv->gem_lock);
  480. return 0;
  481. }
  482. static int etnaviv_gem_new_impl(struct drm_device *dev, u32 size, u32 flags,
  483. struct reservation_object *robj, const struct etnaviv_gem_ops *ops,
  484. struct drm_gem_object **obj)
  485. {
  486. struct etnaviv_gem_object *etnaviv_obj;
  487. unsigned sz = sizeof(*etnaviv_obj);
  488. bool valid = true;
  489. /* validate flags */
  490. switch (flags & ETNA_BO_CACHE_MASK) {
  491. case ETNA_BO_UNCACHED:
  492. case ETNA_BO_CACHED:
  493. case ETNA_BO_WC:
  494. break;
  495. default:
  496. valid = false;
  497. }
  498. if (!valid) {
  499. dev_err(dev->dev, "invalid cache flag: %x\n",
  500. (flags & ETNA_BO_CACHE_MASK));
  501. return -EINVAL;
  502. }
  503. etnaviv_obj = kzalloc(sz, GFP_KERNEL);
  504. if (!etnaviv_obj)
  505. return -ENOMEM;
  506. etnaviv_obj->flags = flags;
  507. etnaviv_obj->ops = ops;
  508. if (robj) {
  509. etnaviv_obj->resv = robj;
  510. } else {
  511. etnaviv_obj->resv = &etnaviv_obj->_resv;
  512. reservation_object_init(&etnaviv_obj->_resv);
  513. }
  514. mutex_init(&etnaviv_obj->lock);
  515. INIT_LIST_HEAD(&etnaviv_obj->vram_list);
  516. *obj = &etnaviv_obj->base;
  517. return 0;
  518. }
  519. static struct drm_gem_object *__etnaviv_gem_new(struct drm_device *dev,
  520. u32 size, u32 flags)
  521. {
  522. struct drm_gem_object *obj = NULL;
  523. int ret;
  524. size = PAGE_ALIGN(size);
  525. ret = etnaviv_gem_new_impl(dev, size, flags, NULL,
  526. &etnaviv_gem_shmem_ops, &obj);
  527. if (ret)
  528. goto fail;
  529. ret = drm_gem_object_init(dev, obj, size);
  530. if (ret == 0) {
  531. struct address_space *mapping;
  532. /*
  533. * Our buffers are kept pinned, so allocating them
  534. * from the MOVABLE zone is a really bad idea, and
  535. * conflicts with CMA. See coments above new_inode()
  536. * why this is required _and_ expected if you're
  537. * going to pin these pages.
  538. */
  539. mapping = obj->filp->f_mapping;
  540. mapping_set_gfp_mask(mapping, GFP_HIGHUSER);
  541. }
  542. if (ret)
  543. goto fail;
  544. return obj;
  545. fail:
  546. drm_gem_object_unreference_unlocked(obj);
  547. return ERR_PTR(ret);
  548. }
  549. /* convenience method to construct a GEM buffer object, and userspace handle */
  550. int etnaviv_gem_new_handle(struct drm_device *dev, struct drm_file *file,
  551. u32 size, u32 flags, u32 *handle)
  552. {
  553. struct drm_gem_object *obj;
  554. int ret;
  555. obj = __etnaviv_gem_new(dev, size, flags);
  556. if (IS_ERR(obj))
  557. return PTR_ERR(obj);
  558. ret = etnaviv_gem_obj_add(dev, obj);
  559. if (ret < 0) {
  560. drm_gem_object_unreference_unlocked(obj);
  561. return ret;
  562. }
  563. ret = drm_gem_handle_create(file, obj, handle);
  564. /* drop reference from allocate - handle holds it now */
  565. drm_gem_object_unreference_unlocked(obj);
  566. return ret;
  567. }
  568. struct drm_gem_object *etnaviv_gem_new(struct drm_device *dev,
  569. u32 size, u32 flags)
  570. {
  571. struct drm_gem_object *obj;
  572. int ret;
  573. obj = __etnaviv_gem_new(dev, size, flags);
  574. if (IS_ERR(obj))
  575. return obj;
  576. ret = etnaviv_gem_obj_add(dev, obj);
  577. if (ret < 0) {
  578. drm_gem_object_unreference_unlocked(obj);
  579. return ERR_PTR(ret);
  580. }
  581. return obj;
  582. }
  583. int etnaviv_gem_new_private(struct drm_device *dev, size_t size, u32 flags,
  584. struct reservation_object *robj, const struct etnaviv_gem_ops *ops,
  585. struct etnaviv_gem_object **res)
  586. {
  587. struct drm_gem_object *obj;
  588. int ret;
  589. ret = etnaviv_gem_new_impl(dev, size, flags, robj, ops, &obj);
  590. if (ret)
  591. return ret;
  592. drm_gem_private_object_init(dev, obj, size);
  593. *res = to_etnaviv_bo(obj);
  594. return 0;
  595. }
  596. struct get_pages_work {
  597. struct work_struct work;
  598. struct mm_struct *mm;
  599. struct task_struct *task;
  600. struct etnaviv_gem_object *etnaviv_obj;
  601. };
  602. static struct page **etnaviv_gem_userptr_do_get_pages(
  603. struct etnaviv_gem_object *etnaviv_obj, struct mm_struct *mm, struct task_struct *task)
  604. {
  605. int ret = 0, pinned, npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  606. struct page **pvec;
  607. uintptr_t ptr;
  608. unsigned int flags = 0;
  609. pvec = drm_malloc_ab(npages, sizeof(struct page *));
  610. if (!pvec)
  611. return ERR_PTR(-ENOMEM);
  612. if (!etnaviv_obj->userptr.ro)
  613. flags |= FOLL_WRITE;
  614. pinned = 0;
  615. ptr = etnaviv_obj->userptr.ptr;
  616. down_read(&mm->mmap_sem);
  617. while (pinned < npages) {
  618. ret = get_user_pages_remote(task, mm, ptr, npages - pinned,
  619. flags, pvec + pinned, NULL, NULL);
  620. if (ret < 0)
  621. break;
  622. ptr += ret * PAGE_SIZE;
  623. pinned += ret;
  624. }
  625. up_read(&mm->mmap_sem);
  626. if (ret < 0) {
  627. release_pages(pvec, pinned, 0);
  628. drm_free_large(pvec);
  629. return ERR_PTR(ret);
  630. }
  631. return pvec;
  632. }
  633. static void __etnaviv_gem_userptr_get_pages(struct work_struct *_work)
  634. {
  635. struct get_pages_work *work = container_of(_work, typeof(*work), work);
  636. struct etnaviv_gem_object *etnaviv_obj = work->etnaviv_obj;
  637. struct page **pvec;
  638. pvec = etnaviv_gem_userptr_do_get_pages(etnaviv_obj, work->mm, work->task);
  639. mutex_lock(&etnaviv_obj->lock);
  640. if (IS_ERR(pvec)) {
  641. etnaviv_obj->userptr.work = ERR_CAST(pvec);
  642. } else {
  643. etnaviv_obj->userptr.work = NULL;
  644. etnaviv_obj->pages = pvec;
  645. }
  646. mutex_unlock(&etnaviv_obj->lock);
  647. drm_gem_object_unreference_unlocked(&etnaviv_obj->base);
  648. mmput(work->mm);
  649. put_task_struct(work->task);
  650. kfree(work);
  651. }
  652. static int etnaviv_gem_userptr_get_pages(struct etnaviv_gem_object *etnaviv_obj)
  653. {
  654. struct page **pvec = NULL;
  655. struct get_pages_work *work;
  656. struct mm_struct *mm;
  657. int ret, pinned, npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  658. if (etnaviv_obj->userptr.work) {
  659. if (IS_ERR(etnaviv_obj->userptr.work)) {
  660. ret = PTR_ERR(etnaviv_obj->userptr.work);
  661. etnaviv_obj->userptr.work = NULL;
  662. } else {
  663. ret = -EAGAIN;
  664. }
  665. return ret;
  666. }
  667. mm = get_task_mm(etnaviv_obj->userptr.task);
  668. pinned = 0;
  669. if (mm == current->mm) {
  670. pvec = drm_malloc_ab(npages, sizeof(struct page *));
  671. if (!pvec) {
  672. mmput(mm);
  673. return -ENOMEM;
  674. }
  675. pinned = __get_user_pages_fast(etnaviv_obj->userptr.ptr, npages,
  676. !etnaviv_obj->userptr.ro, pvec);
  677. if (pinned < 0) {
  678. drm_free_large(pvec);
  679. mmput(mm);
  680. return pinned;
  681. }
  682. if (pinned == npages) {
  683. etnaviv_obj->pages = pvec;
  684. mmput(mm);
  685. return 0;
  686. }
  687. }
  688. release_pages(pvec, pinned, 0);
  689. drm_free_large(pvec);
  690. work = kmalloc(sizeof(*work), GFP_KERNEL);
  691. if (!work) {
  692. mmput(mm);
  693. return -ENOMEM;
  694. }
  695. get_task_struct(current);
  696. drm_gem_object_reference(&etnaviv_obj->base);
  697. work->mm = mm;
  698. work->task = current;
  699. work->etnaviv_obj = etnaviv_obj;
  700. etnaviv_obj->userptr.work = &work->work;
  701. INIT_WORK(&work->work, __etnaviv_gem_userptr_get_pages);
  702. etnaviv_queue_work(etnaviv_obj->base.dev, &work->work);
  703. return -EAGAIN;
  704. }
  705. static void etnaviv_gem_userptr_release(struct etnaviv_gem_object *etnaviv_obj)
  706. {
  707. if (etnaviv_obj->sgt) {
  708. etnaviv_gem_scatterlist_unmap(etnaviv_obj);
  709. sg_free_table(etnaviv_obj->sgt);
  710. kfree(etnaviv_obj->sgt);
  711. }
  712. if (etnaviv_obj->pages) {
  713. int npages = etnaviv_obj->base.size >> PAGE_SHIFT;
  714. release_pages(etnaviv_obj->pages, npages, 0);
  715. drm_free_large(etnaviv_obj->pages);
  716. }
  717. put_task_struct(etnaviv_obj->userptr.task);
  718. }
  719. static int etnaviv_gem_userptr_mmap_obj(struct etnaviv_gem_object *etnaviv_obj,
  720. struct vm_area_struct *vma)
  721. {
  722. return -EINVAL;
  723. }
  724. static const struct etnaviv_gem_ops etnaviv_gem_userptr_ops = {
  725. .get_pages = etnaviv_gem_userptr_get_pages,
  726. .release = etnaviv_gem_userptr_release,
  727. .vmap = etnaviv_gem_vmap_impl,
  728. .mmap = etnaviv_gem_userptr_mmap_obj,
  729. };
  730. int etnaviv_gem_new_userptr(struct drm_device *dev, struct drm_file *file,
  731. uintptr_t ptr, u32 size, u32 flags, u32 *handle)
  732. {
  733. struct etnaviv_gem_object *etnaviv_obj;
  734. int ret;
  735. ret = etnaviv_gem_new_private(dev, size, ETNA_BO_CACHED, NULL,
  736. &etnaviv_gem_userptr_ops, &etnaviv_obj);
  737. if (ret)
  738. return ret;
  739. etnaviv_obj->userptr.ptr = ptr;
  740. etnaviv_obj->userptr.task = current;
  741. etnaviv_obj->userptr.ro = !(flags & ETNA_USERPTR_WRITE);
  742. get_task_struct(current);
  743. ret = etnaviv_gem_obj_add(dev, &etnaviv_obj->base);
  744. if (ret)
  745. goto unreference;
  746. ret = drm_gem_handle_create(file, &etnaviv_obj->base, handle);
  747. unreference:
  748. /* drop reference from allocate - handle holds it now */
  749. drm_gem_object_unreference_unlocked(&etnaviv_obj->base);
  750. return ret;
  751. }