Home | History | Annotate | Line # | Download | only in drm
drm_prime.c revision 1.1.1.1.6.1
      1          1.1  riastrad /*
      2          1.1  riastrad  * Copyright  2012 Red Hat
      3          1.1  riastrad  *
      4          1.1  riastrad  * Permission is hereby granted, free of charge, to any person obtaining a
      5          1.1  riastrad  * copy of this software and associated documentation files (the "Software"),
      6          1.1  riastrad  * to deal in the Software without restriction, including without limitation
      7          1.1  riastrad  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
      8          1.1  riastrad  * and/or sell copies of the Software, and to permit persons to whom the
      9          1.1  riastrad  * Software is furnished to do so, subject to the following conditions:
     10          1.1  riastrad  *
     11          1.1  riastrad  * The above copyright notice and this permission notice (including the next
     12          1.1  riastrad  * paragraph) shall be included in all copies or substantial portions of the
     13          1.1  riastrad  * Software.
     14          1.1  riastrad  *
     15          1.1  riastrad  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     16          1.1  riastrad  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
     17          1.1  riastrad  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
     18          1.1  riastrad  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
     19          1.1  riastrad  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
     20          1.1  riastrad  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
     21          1.1  riastrad  * IN THE SOFTWARE.
     22          1.1  riastrad  *
     23          1.1  riastrad  * Authors:
     24          1.1  riastrad  *      Dave Airlie <airlied (at) redhat.com>
     25          1.1  riastrad  *      Rob Clark <rob.clark (at) linaro.org>
     26          1.1  riastrad  *
     27          1.1  riastrad  */
     28          1.1  riastrad 
     29          1.1  riastrad #include <linux/export.h>
     30          1.1  riastrad #include <linux/dma-buf.h>
     31          1.1  riastrad #include <drm/drmP.h>
     32          1.1  riastrad 
     33          1.1  riastrad /*
     34          1.1  riastrad  * DMA-BUF/GEM Object references and lifetime overview:
     35          1.1  riastrad  *
     36          1.1  riastrad  * On the export the dma_buf holds a reference to the exporting GEM
     37          1.1  riastrad  * object. It takes this reference in handle_to_fd_ioctl, when it
     38          1.1  riastrad  * first calls .prime_export and stores the exporting GEM object in
     39          1.1  riastrad  * the dma_buf priv. This reference is released when the dma_buf
     40          1.1  riastrad  * object goes away in the driver .release function.
     41          1.1  riastrad  *
     42          1.1  riastrad  * On the import the importing GEM object holds a reference to the
     43          1.1  riastrad  * dma_buf (which in turn holds a ref to the exporting GEM object).
     44          1.1  riastrad  * It takes that reference in the fd_to_handle ioctl.
     45          1.1  riastrad  * It calls dma_buf_get, creates an attachment to it and stores the
     46          1.1  riastrad  * attachment in the GEM object. When this attachment is destroyed
     47          1.1  riastrad  * when the imported object is destroyed, we remove the attachment
     48          1.1  riastrad  * and drop the reference to the dma_buf.
     49          1.1  riastrad  *
     50          1.1  riastrad  * Thus the chain of references always flows in one direction
     51          1.1  riastrad  * (avoiding loops): importing_gem -> dmabuf -> exporting_gem
     52          1.1  riastrad  *
     53          1.1  riastrad  * Self-importing: if userspace is using PRIME as a replacement for flink
     54          1.1  riastrad  * then it will get a fd->handle request for a GEM object that it created.
     55          1.1  riastrad  * Drivers should detect this situation and return back the gem object
     56  1.1.1.1.6.1       tls  * from the dma-buf private.  Prime will do this automatically for drivers that
     57  1.1.1.1.6.1       tls  * use the drm_gem_prime_{import,export} helpers.
     58          1.1  riastrad  */
     59          1.1  riastrad 
     60          1.1  riastrad struct drm_prime_member {
     61          1.1  riastrad 	struct list_head entry;
     62          1.1  riastrad 	struct dma_buf *dma_buf;
     63          1.1  riastrad 	uint32_t handle;
     64          1.1  riastrad };
     65          1.1  riastrad 
     66  1.1.1.1.6.1       tls struct drm_prime_attachment {
     67  1.1.1.1.6.1       tls 	struct sg_table *sgt;
     68  1.1.1.1.6.1       tls 	enum dma_data_direction dir;
     69  1.1.1.1.6.1       tls };
     70  1.1.1.1.6.1       tls 
     71  1.1.1.1.6.1       tls static int drm_prime_add_buf_handle(struct drm_prime_file_private *prime_fpriv,
     72  1.1.1.1.6.1       tls 				    struct dma_buf *dma_buf, uint32_t handle)
     73  1.1.1.1.6.1       tls {
     74  1.1.1.1.6.1       tls 	struct drm_prime_member *member;
     75  1.1.1.1.6.1       tls 
     76  1.1.1.1.6.1       tls 	member = kmalloc(sizeof(*member), GFP_KERNEL);
     77  1.1.1.1.6.1       tls 	if (!member)
     78  1.1.1.1.6.1       tls 		return -ENOMEM;
     79  1.1.1.1.6.1       tls 
     80  1.1.1.1.6.1       tls 	get_dma_buf(dma_buf);
     81  1.1.1.1.6.1       tls 	member->dma_buf = dma_buf;
     82  1.1.1.1.6.1       tls 	member->handle = handle;
     83  1.1.1.1.6.1       tls 	list_add(&member->entry, &prime_fpriv->head);
     84  1.1.1.1.6.1       tls 	return 0;
     85  1.1.1.1.6.1       tls }
     86  1.1.1.1.6.1       tls 
     87  1.1.1.1.6.1       tls static struct dma_buf *drm_prime_lookup_buf_by_handle(struct drm_prime_file_private *prime_fpriv,
     88  1.1.1.1.6.1       tls 						      uint32_t handle)
     89  1.1.1.1.6.1       tls {
     90  1.1.1.1.6.1       tls 	struct drm_prime_member *member;
     91  1.1.1.1.6.1       tls 
     92  1.1.1.1.6.1       tls 	list_for_each_entry(member, &prime_fpriv->head, entry) {
     93  1.1.1.1.6.1       tls 		if (member->handle == handle)
     94  1.1.1.1.6.1       tls 			return member->dma_buf;
     95  1.1.1.1.6.1       tls 	}
     96  1.1.1.1.6.1       tls 
     97  1.1.1.1.6.1       tls 	return NULL;
     98  1.1.1.1.6.1       tls }
     99  1.1.1.1.6.1       tls 
    100  1.1.1.1.6.1       tls static int drm_prime_lookup_buf_handle(struct drm_prime_file_private *prime_fpriv,
    101  1.1.1.1.6.1       tls 				       struct dma_buf *dma_buf,
    102  1.1.1.1.6.1       tls 				       uint32_t *handle)
    103  1.1.1.1.6.1       tls {
    104  1.1.1.1.6.1       tls 	struct drm_prime_member *member;
    105  1.1.1.1.6.1       tls 
    106  1.1.1.1.6.1       tls 	list_for_each_entry(member, &prime_fpriv->head, entry) {
    107  1.1.1.1.6.1       tls 		if (member->dma_buf == dma_buf) {
    108  1.1.1.1.6.1       tls 			*handle = member->handle;
    109  1.1.1.1.6.1       tls 			return 0;
    110  1.1.1.1.6.1       tls 		}
    111  1.1.1.1.6.1       tls 	}
    112  1.1.1.1.6.1       tls 	return -ENOENT;
    113  1.1.1.1.6.1       tls }
    114  1.1.1.1.6.1       tls 
    115  1.1.1.1.6.1       tls static int drm_gem_map_attach(struct dma_buf *dma_buf,
    116  1.1.1.1.6.1       tls 			      struct device *target_dev,
    117  1.1.1.1.6.1       tls 			      struct dma_buf_attachment *attach)
    118  1.1.1.1.6.1       tls {
    119  1.1.1.1.6.1       tls 	struct drm_prime_attachment *prime_attach;
    120  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = dma_buf->priv;
    121  1.1.1.1.6.1       tls 	struct drm_device *dev = obj->dev;
    122  1.1.1.1.6.1       tls 
    123  1.1.1.1.6.1       tls 	prime_attach = kzalloc(sizeof(*prime_attach), GFP_KERNEL);
    124  1.1.1.1.6.1       tls 	if (!prime_attach)
    125  1.1.1.1.6.1       tls 		return -ENOMEM;
    126  1.1.1.1.6.1       tls 
    127  1.1.1.1.6.1       tls 	prime_attach->dir = DMA_NONE;
    128  1.1.1.1.6.1       tls 	attach->priv = prime_attach;
    129  1.1.1.1.6.1       tls 
    130  1.1.1.1.6.1       tls 	if (!dev->driver->gem_prime_pin)
    131  1.1.1.1.6.1       tls 		return 0;
    132  1.1.1.1.6.1       tls 
    133  1.1.1.1.6.1       tls 	return dev->driver->gem_prime_pin(obj);
    134  1.1.1.1.6.1       tls }
    135  1.1.1.1.6.1       tls 
    136  1.1.1.1.6.1       tls static void drm_gem_map_detach(struct dma_buf *dma_buf,
    137  1.1.1.1.6.1       tls 			       struct dma_buf_attachment *attach)
    138  1.1.1.1.6.1       tls {
    139  1.1.1.1.6.1       tls 	struct drm_prime_attachment *prime_attach = attach->priv;
    140  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = dma_buf->priv;
    141  1.1.1.1.6.1       tls 	struct drm_device *dev = obj->dev;
    142  1.1.1.1.6.1       tls 	struct sg_table *sgt;
    143  1.1.1.1.6.1       tls 
    144  1.1.1.1.6.1       tls 	if (dev->driver->gem_prime_unpin)
    145  1.1.1.1.6.1       tls 		dev->driver->gem_prime_unpin(obj);
    146  1.1.1.1.6.1       tls 
    147  1.1.1.1.6.1       tls 	if (!prime_attach)
    148  1.1.1.1.6.1       tls 		return;
    149  1.1.1.1.6.1       tls 
    150  1.1.1.1.6.1       tls 	sgt = prime_attach->sgt;
    151  1.1.1.1.6.1       tls 	if (sgt) {
    152  1.1.1.1.6.1       tls 		if (prime_attach->dir != DMA_NONE)
    153  1.1.1.1.6.1       tls 			dma_unmap_sg(attach->dev, sgt->sgl, sgt->nents,
    154  1.1.1.1.6.1       tls 					prime_attach->dir);
    155  1.1.1.1.6.1       tls 		sg_free_table(sgt);
    156  1.1.1.1.6.1       tls 	}
    157  1.1.1.1.6.1       tls 
    158  1.1.1.1.6.1       tls 	kfree(sgt);
    159  1.1.1.1.6.1       tls 	kfree(prime_attach);
    160  1.1.1.1.6.1       tls 	attach->priv = NULL;
    161  1.1.1.1.6.1       tls }
    162  1.1.1.1.6.1       tls 
    163  1.1.1.1.6.1       tls void drm_prime_remove_buf_handle_locked(struct drm_prime_file_private *prime_fpriv,
    164  1.1.1.1.6.1       tls 					struct dma_buf *dma_buf)
    165  1.1.1.1.6.1       tls {
    166  1.1.1.1.6.1       tls 	struct drm_prime_member *member, *safe;
    167  1.1.1.1.6.1       tls 
    168  1.1.1.1.6.1       tls 	list_for_each_entry_safe(member, safe, &prime_fpriv->head, entry) {
    169  1.1.1.1.6.1       tls 		if (member->dma_buf == dma_buf) {
    170  1.1.1.1.6.1       tls 			dma_buf_put(dma_buf);
    171  1.1.1.1.6.1       tls 			list_del(&member->entry);
    172  1.1.1.1.6.1       tls 			kfree(member);
    173  1.1.1.1.6.1       tls 		}
    174  1.1.1.1.6.1       tls 	}
    175  1.1.1.1.6.1       tls }
    176  1.1.1.1.6.1       tls 
    177  1.1.1.1.6.1       tls static struct sg_table *drm_gem_map_dma_buf(struct dma_buf_attachment *attach,
    178  1.1.1.1.6.1       tls 					    enum dma_data_direction dir)
    179  1.1.1.1.6.1       tls {
    180  1.1.1.1.6.1       tls 	struct drm_prime_attachment *prime_attach = attach->priv;
    181  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = attach->dmabuf->priv;
    182  1.1.1.1.6.1       tls 	struct sg_table *sgt;
    183  1.1.1.1.6.1       tls 
    184  1.1.1.1.6.1       tls 	if (WARN_ON(dir == DMA_NONE || !prime_attach))
    185  1.1.1.1.6.1       tls 		return ERR_PTR(-EINVAL);
    186  1.1.1.1.6.1       tls 
    187  1.1.1.1.6.1       tls 	/* return the cached mapping when possible */
    188  1.1.1.1.6.1       tls 	if (prime_attach->dir == dir)
    189  1.1.1.1.6.1       tls 		return prime_attach->sgt;
    190  1.1.1.1.6.1       tls 
    191  1.1.1.1.6.1       tls 	/*
    192  1.1.1.1.6.1       tls 	 * two mappings with different directions for the same attachment are
    193  1.1.1.1.6.1       tls 	 * not allowed
    194  1.1.1.1.6.1       tls 	 */
    195  1.1.1.1.6.1       tls 	if (WARN_ON(prime_attach->dir != DMA_NONE))
    196  1.1.1.1.6.1       tls 		return ERR_PTR(-EBUSY);
    197  1.1.1.1.6.1       tls 
    198  1.1.1.1.6.1       tls 	sgt = obj->dev->driver->gem_prime_get_sg_table(obj);
    199  1.1.1.1.6.1       tls 
    200  1.1.1.1.6.1       tls 	if (!IS_ERR(sgt)) {
    201  1.1.1.1.6.1       tls 		if (!dma_map_sg(attach->dev, sgt->sgl, sgt->nents, dir)) {
    202  1.1.1.1.6.1       tls 			sg_free_table(sgt);
    203  1.1.1.1.6.1       tls 			kfree(sgt);
    204  1.1.1.1.6.1       tls 			sgt = ERR_PTR(-ENOMEM);
    205  1.1.1.1.6.1       tls 		} else {
    206  1.1.1.1.6.1       tls 			prime_attach->sgt = sgt;
    207  1.1.1.1.6.1       tls 			prime_attach->dir = dir;
    208  1.1.1.1.6.1       tls 		}
    209  1.1.1.1.6.1       tls 	}
    210  1.1.1.1.6.1       tls 
    211  1.1.1.1.6.1       tls 	return sgt;
    212  1.1.1.1.6.1       tls }
    213  1.1.1.1.6.1       tls 
    214  1.1.1.1.6.1       tls static void drm_gem_unmap_dma_buf(struct dma_buf_attachment *attach,
    215  1.1.1.1.6.1       tls 				  struct sg_table *sgt,
    216  1.1.1.1.6.1       tls 				  enum dma_data_direction dir)
    217  1.1.1.1.6.1       tls {
    218  1.1.1.1.6.1       tls 	/* nothing to be done here */
    219  1.1.1.1.6.1       tls }
    220  1.1.1.1.6.1       tls 
    221  1.1.1.1.6.1       tls /**
    222  1.1.1.1.6.1       tls  * drm_gem_dmabuf_release - dma_buf release implementation for GEM
    223  1.1.1.1.6.1       tls  * @dma_buf: buffer to be released
    224  1.1.1.1.6.1       tls  *
    225  1.1.1.1.6.1       tls  * Generic release function for dma_bufs exported as PRIME buffers. GEM drivers
    226  1.1.1.1.6.1       tls  * must use this in their dma_buf ops structure as the release callback.
    227  1.1.1.1.6.1       tls  */
    228  1.1.1.1.6.1       tls void drm_gem_dmabuf_release(struct dma_buf *dma_buf)
    229  1.1.1.1.6.1       tls {
    230  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = dma_buf->priv;
    231  1.1.1.1.6.1       tls 
    232  1.1.1.1.6.1       tls 	/* drop the reference on the export fd holds */
    233  1.1.1.1.6.1       tls 	drm_gem_object_unreference_unlocked(obj);
    234  1.1.1.1.6.1       tls }
    235  1.1.1.1.6.1       tls EXPORT_SYMBOL(drm_gem_dmabuf_release);
    236  1.1.1.1.6.1       tls 
    237  1.1.1.1.6.1       tls static void *drm_gem_dmabuf_vmap(struct dma_buf *dma_buf)
    238  1.1.1.1.6.1       tls {
    239  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = dma_buf->priv;
    240  1.1.1.1.6.1       tls 	struct drm_device *dev = obj->dev;
    241  1.1.1.1.6.1       tls 
    242  1.1.1.1.6.1       tls 	return dev->driver->gem_prime_vmap(obj);
    243  1.1.1.1.6.1       tls }
    244  1.1.1.1.6.1       tls 
    245  1.1.1.1.6.1       tls static void drm_gem_dmabuf_vunmap(struct dma_buf *dma_buf, void *vaddr)
    246  1.1.1.1.6.1       tls {
    247  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = dma_buf->priv;
    248  1.1.1.1.6.1       tls 	struct drm_device *dev = obj->dev;
    249  1.1.1.1.6.1       tls 
    250  1.1.1.1.6.1       tls 	dev->driver->gem_prime_vunmap(obj, vaddr);
    251  1.1.1.1.6.1       tls }
    252  1.1.1.1.6.1       tls 
    253  1.1.1.1.6.1       tls static void *drm_gem_dmabuf_kmap_atomic(struct dma_buf *dma_buf,
    254  1.1.1.1.6.1       tls 					unsigned long page_num)
    255  1.1.1.1.6.1       tls {
    256  1.1.1.1.6.1       tls 	return NULL;
    257  1.1.1.1.6.1       tls }
    258  1.1.1.1.6.1       tls 
    259  1.1.1.1.6.1       tls static void drm_gem_dmabuf_kunmap_atomic(struct dma_buf *dma_buf,
    260  1.1.1.1.6.1       tls 					 unsigned long page_num, void *addr)
    261  1.1.1.1.6.1       tls {
    262  1.1.1.1.6.1       tls 
    263  1.1.1.1.6.1       tls }
    264  1.1.1.1.6.1       tls static void *drm_gem_dmabuf_kmap(struct dma_buf *dma_buf,
    265  1.1.1.1.6.1       tls 				 unsigned long page_num)
    266  1.1.1.1.6.1       tls {
    267  1.1.1.1.6.1       tls 	return NULL;
    268  1.1.1.1.6.1       tls }
    269  1.1.1.1.6.1       tls 
    270  1.1.1.1.6.1       tls static void drm_gem_dmabuf_kunmap(struct dma_buf *dma_buf,
    271  1.1.1.1.6.1       tls 				  unsigned long page_num, void *addr)
    272  1.1.1.1.6.1       tls {
    273  1.1.1.1.6.1       tls 
    274  1.1.1.1.6.1       tls }
    275  1.1.1.1.6.1       tls 
    276  1.1.1.1.6.1       tls static int drm_gem_dmabuf_mmap(struct dma_buf *dma_buf,
    277  1.1.1.1.6.1       tls 			       struct vm_area_struct *vma)
    278  1.1.1.1.6.1       tls {
    279  1.1.1.1.6.1       tls 	struct drm_gem_object *obj = dma_buf->priv;
    280  1.1.1.1.6.1       tls 	struct drm_device *dev = obj->dev;
    281  1.1.1.1.6.1       tls 
    282  1.1.1.1.6.1       tls 	if (!dev->driver->gem_prime_mmap)
    283  1.1.1.1.6.1       tls 		return -ENOSYS;
    284  1.1.1.1.6.1       tls 
    285  1.1.1.1.6.1       tls 	return dev->driver->gem_prime_mmap(obj, vma);
    286  1.1.1.1.6.1       tls }
    287  1.1.1.1.6.1       tls 
    288  1.1.1.1.6.1       tls static const struct dma_buf_ops drm_gem_prime_dmabuf_ops =  {
    289  1.1.1.1.6.1       tls 	.attach = drm_gem_map_attach,
    290  1.1.1.1.6.1       tls 	.detach = drm_gem_map_detach,
    291  1.1.1.1.6.1       tls 	.map_dma_buf = drm_gem_map_dma_buf,
    292  1.1.1.1.6.1       tls 	.unmap_dma_buf = drm_gem_unmap_dma_buf,
    293  1.1.1.1.6.1       tls 	.release = drm_gem_dmabuf_release,
    294  1.1.1.1.6.1       tls 	.kmap = drm_gem_dmabuf_kmap,
    295  1.1.1.1.6.1       tls 	.kmap_atomic = drm_gem_dmabuf_kmap_atomic,
    296  1.1.1.1.6.1       tls 	.kunmap = drm_gem_dmabuf_kunmap,
    297  1.1.1.1.6.1       tls 	.kunmap_atomic = drm_gem_dmabuf_kunmap_atomic,
    298  1.1.1.1.6.1       tls 	.mmap = drm_gem_dmabuf_mmap,
    299  1.1.1.1.6.1       tls 	.vmap = drm_gem_dmabuf_vmap,
    300  1.1.1.1.6.1       tls 	.vunmap = drm_gem_dmabuf_vunmap,
    301  1.1.1.1.6.1       tls };
    302  1.1.1.1.6.1       tls 
    303  1.1.1.1.6.1       tls /**
    304  1.1.1.1.6.1       tls  * DOC: PRIME Helpers
    305  1.1.1.1.6.1       tls  *
    306  1.1.1.1.6.1       tls  * Drivers can implement @gem_prime_export and @gem_prime_import in terms of
    307  1.1.1.1.6.1       tls  * simpler APIs by using the helper functions @drm_gem_prime_export and
    308  1.1.1.1.6.1       tls  * @drm_gem_prime_import.  These functions implement dma-buf support in terms of
    309  1.1.1.1.6.1       tls  * five lower-level driver callbacks:
    310  1.1.1.1.6.1       tls  *
    311  1.1.1.1.6.1       tls  * Export callbacks:
    312  1.1.1.1.6.1       tls  *
    313  1.1.1.1.6.1       tls  *  - @gem_prime_pin (optional): prepare a GEM object for exporting
    314  1.1.1.1.6.1       tls  *
    315  1.1.1.1.6.1       tls  *  - @gem_prime_get_sg_table: provide a scatter/gather table of pinned pages
    316  1.1.1.1.6.1       tls  *
    317  1.1.1.1.6.1       tls  *  - @gem_prime_vmap: vmap a buffer exported by your driver
    318  1.1.1.1.6.1       tls  *
    319  1.1.1.1.6.1       tls  *  - @gem_prime_vunmap: vunmap a buffer exported by your driver
    320  1.1.1.1.6.1       tls  *
    321  1.1.1.1.6.1       tls  * Import callback:
    322  1.1.1.1.6.1       tls  *
    323  1.1.1.1.6.1       tls  *  - @gem_prime_import_sg_table (import): produce a GEM object from another
    324  1.1.1.1.6.1       tls  *    driver's scatter/gather table
    325  1.1.1.1.6.1       tls  */
    326  1.1.1.1.6.1       tls 
    327  1.1.1.1.6.1       tls /**
    328  1.1.1.1.6.1       tls  * drm_gem_prime_export - helper library implemention of the export callback
    329  1.1.1.1.6.1       tls  * @dev: drm_device to export from
    330  1.1.1.1.6.1       tls  * @obj: GEM object to export
    331  1.1.1.1.6.1       tls  * @flags: flags like DRM_CLOEXEC
    332  1.1.1.1.6.1       tls  *
    333  1.1.1.1.6.1       tls  * This is the implementation of the gem_prime_export functions for GEM drivers
    334  1.1.1.1.6.1       tls  * using the PRIME helpers.
    335  1.1.1.1.6.1       tls  */
    336  1.1.1.1.6.1       tls struct dma_buf *drm_gem_prime_export(struct drm_device *dev,
    337  1.1.1.1.6.1       tls 				     struct drm_gem_object *obj, int flags)
    338  1.1.1.1.6.1       tls {
    339  1.1.1.1.6.1       tls 	return dma_buf_export(obj, &drm_gem_prime_dmabuf_ops, obj->size, flags);
    340  1.1.1.1.6.1       tls }
    341  1.1.1.1.6.1       tls EXPORT_SYMBOL(drm_gem_prime_export);
    342  1.1.1.1.6.1       tls 
    343  1.1.1.1.6.1       tls static struct dma_buf *export_and_register_object(struct drm_device *dev,
    344  1.1.1.1.6.1       tls 						  struct drm_gem_object *obj,
    345  1.1.1.1.6.1       tls 						  uint32_t flags)
    346  1.1.1.1.6.1       tls {
    347  1.1.1.1.6.1       tls 	struct dma_buf *dmabuf;
    348  1.1.1.1.6.1       tls 
    349  1.1.1.1.6.1       tls 	/* prevent races with concurrent gem_close. */
    350  1.1.1.1.6.1       tls 	if (obj->handle_count == 0) {
    351  1.1.1.1.6.1       tls 		dmabuf = ERR_PTR(-ENOENT);
    352  1.1.1.1.6.1       tls 		return dmabuf;
    353  1.1.1.1.6.1       tls 	}
    354  1.1.1.1.6.1       tls 
    355  1.1.1.1.6.1       tls 	dmabuf = dev->driver->gem_prime_export(dev, obj, flags);
    356  1.1.1.1.6.1       tls 	if (IS_ERR(dmabuf)) {
    357  1.1.1.1.6.1       tls 		/* normally the created dma-buf takes ownership of the ref,
    358  1.1.1.1.6.1       tls 		 * but if that fails then drop the ref
    359  1.1.1.1.6.1       tls 		 */
    360  1.1.1.1.6.1       tls 		return dmabuf;
    361  1.1.1.1.6.1       tls 	}
    362  1.1.1.1.6.1       tls 
    363  1.1.1.1.6.1       tls 	/*
    364  1.1.1.1.6.1       tls 	 * Note that callers do not need to clean up the export cache
    365  1.1.1.1.6.1       tls 	 * since the check for obj->handle_count guarantees that someone
    366  1.1.1.1.6.1       tls 	 * will clean it up.
    367  1.1.1.1.6.1       tls 	 */
    368  1.1.1.1.6.1       tls 	obj->dma_buf = dmabuf;
    369  1.1.1.1.6.1       tls 	get_dma_buf(obj->dma_buf);
    370  1.1.1.1.6.1       tls 	/* Grab a new ref since the callers is now used by the dma-buf */
    371  1.1.1.1.6.1       tls 	drm_gem_object_reference(obj);
    372  1.1.1.1.6.1       tls 
    373  1.1.1.1.6.1       tls 	return dmabuf;
    374  1.1.1.1.6.1       tls }
    375  1.1.1.1.6.1       tls 
    376  1.1.1.1.6.1       tls /**
    377  1.1.1.1.6.1       tls  * drm_gem_prime_handle_to_fd - PRIME export function for GEM drivers
    378  1.1.1.1.6.1       tls  * @dev: dev to export the buffer from
    379  1.1.1.1.6.1       tls  * @file_priv: drm file-private structure
    380  1.1.1.1.6.1       tls  * @handle: buffer handle to export
    381  1.1.1.1.6.1       tls  * @flags: flags like DRM_CLOEXEC
    382  1.1.1.1.6.1       tls  * @prime_fd: pointer to storage for the fd id of the create dma-buf
    383  1.1.1.1.6.1       tls  *
    384  1.1.1.1.6.1       tls  * This is the PRIME export function which must be used mandatorily by GEM
    385  1.1.1.1.6.1       tls  * drivers to ensure correct lifetime management of the underlying GEM object.
    386  1.1.1.1.6.1       tls  * The actual exporting from GEM object to a dma-buf is done through the
    387  1.1.1.1.6.1       tls  * gem_prime_export driver callback.
    388  1.1.1.1.6.1       tls  */
    389          1.1  riastrad int drm_gem_prime_handle_to_fd(struct drm_device *dev,
    390  1.1.1.1.6.1       tls 			       struct drm_file *file_priv, uint32_t handle,
    391  1.1.1.1.6.1       tls 			       uint32_t flags,
    392  1.1.1.1.6.1       tls 			       int *prime_fd)
    393          1.1  riastrad {
    394          1.1  riastrad 	struct drm_gem_object *obj;
    395  1.1.1.1.6.1       tls 	int ret = 0;
    396  1.1.1.1.6.1       tls 	struct dma_buf *dmabuf;
    397          1.1  riastrad 
    398  1.1.1.1.6.1       tls 	mutex_lock(&file_priv->prime.lock);
    399          1.1  riastrad 	obj = drm_gem_object_lookup(dev, file_priv, handle);
    400  1.1.1.1.6.1       tls 	if (!obj)  {
    401  1.1.1.1.6.1       tls 		ret = -ENOENT;
    402  1.1.1.1.6.1       tls 		goto out_unlock;
    403  1.1.1.1.6.1       tls 	}
    404          1.1  riastrad 
    405  1.1.1.1.6.1       tls 	dmabuf = drm_prime_lookup_buf_by_handle(&file_priv->prime, handle);
    406  1.1.1.1.6.1       tls 	if (dmabuf) {
    407  1.1.1.1.6.1       tls 		get_dma_buf(dmabuf);
    408  1.1.1.1.6.1       tls 		goto out_have_handle;
    409  1.1.1.1.6.1       tls 	}
    410  1.1.1.1.6.1       tls 
    411  1.1.1.1.6.1       tls 	mutex_lock(&dev->object_name_lock);
    412          1.1  riastrad 	/* re-export the original imported object */
    413          1.1  riastrad 	if (obj->import_attach) {
    414  1.1.1.1.6.1       tls 		dmabuf = obj->import_attach->dmabuf;
    415  1.1.1.1.6.1       tls 		get_dma_buf(dmabuf);
    416  1.1.1.1.6.1       tls 		goto out_have_obj;
    417          1.1  riastrad 	}
    418          1.1  riastrad 
    419  1.1.1.1.6.1       tls 	if (obj->dma_buf) {
    420  1.1.1.1.6.1       tls 		get_dma_buf(obj->dma_buf);
    421  1.1.1.1.6.1       tls 		dmabuf = obj->dma_buf;
    422  1.1.1.1.6.1       tls 		goto out_have_obj;
    423          1.1  riastrad 	}
    424  1.1.1.1.6.1       tls 
    425  1.1.1.1.6.1       tls 	dmabuf = export_and_register_object(dev, obj, flags);
    426  1.1.1.1.6.1       tls 	if (IS_ERR(dmabuf)) {
    427  1.1.1.1.6.1       tls 		/* normally the created dma-buf takes ownership of the ref,
    428  1.1.1.1.6.1       tls 		 * but if that fails then drop the ref
    429  1.1.1.1.6.1       tls 		 */
    430  1.1.1.1.6.1       tls 		ret = PTR_ERR(dmabuf);
    431  1.1.1.1.6.1       tls 		mutex_unlock(&dev->object_name_lock);
    432  1.1.1.1.6.1       tls 		goto out;
    433  1.1.1.1.6.1       tls 	}
    434  1.1.1.1.6.1       tls 
    435  1.1.1.1.6.1       tls out_have_obj:
    436  1.1.1.1.6.1       tls 	/*
    437  1.1.1.1.6.1       tls 	 * If we've exported this buffer then cheat and add it to the import list
    438  1.1.1.1.6.1       tls 	 * so we get the correct handle back. We must do this under the
    439  1.1.1.1.6.1       tls 	 * protection of dev->object_name_lock to ensure that a racing gem close
    440  1.1.1.1.6.1       tls 	 * ioctl doesn't miss to remove this buffer handle from the cache.
    441  1.1.1.1.6.1       tls 	 */
    442  1.1.1.1.6.1       tls 	ret = drm_prime_add_buf_handle(&file_priv->prime,
    443  1.1.1.1.6.1       tls 				       dmabuf, handle);
    444  1.1.1.1.6.1       tls 	mutex_unlock(&dev->object_name_lock);
    445  1.1.1.1.6.1       tls 	if (ret)
    446  1.1.1.1.6.1       tls 		goto fail_put_dmabuf;
    447  1.1.1.1.6.1       tls 
    448  1.1.1.1.6.1       tls out_have_handle:
    449  1.1.1.1.6.1       tls 	ret = dma_buf_fd(dmabuf, flags);
    450  1.1.1.1.6.1       tls 	/*
    451  1.1.1.1.6.1       tls 	 * We must _not_ remove the buffer from the handle cache since the newly
    452  1.1.1.1.6.1       tls 	 * created dma buf is already linked in the global obj->dma_buf pointer,
    453  1.1.1.1.6.1       tls 	 * and that is invariant as long as a userspace gem handle exists.
    454  1.1.1.1.6.1       tls 	 * Closing the handle will clean out the cache anyway, so we don't leak.
    455          1.1  riastrad 	 */
    456  1.1.1.1.6.1       tls 	if (ret < 0) {
    457  1.1.1.1.6.1       tls 		goto fail_put_dmabuf;
    458  1.1.1.1.6.1       tls 	} else {
    459  1.1.1.1.6.1       tls 		*prime_fd = ret;
    460  1.1.1.1.6.1       tls 		ret = 0;
    461          1.1  riastrad 	}
    462          1.1  riastrad 
    463  1.1.1.1.6.1       tls 	goto out;
    464  1.1.1.1.6.1       tls 
    465  1.1.1.1.6.1       tls fail_put_dmabuf:
    466  1.1.1.1.6.1       tls 	dma_buf_put(dmabuf);
    467  1.1.1.1.6.1       tls out:
    468  1.1.1.1.6.1       tls 	drm_gem_object_unreference_unlocked(obj);
    469  1.1.1.1.6.1       tls out_unlock:
    470          1.1  riastrad 	mutex_unlock(&file_priv->prime.lock);
    471  1.1.1.1.6.1       tls 
    472  1.1.1.1.6.1       tls 	return ret;
    473          1.1  riastrad }
    474          1.1  riastrad EXPORT_SYMBOL(drm_gem_prime_handle_to_fd);
    475          1.1  riastrad 
    476  1.1.1.1.6.1       tls /**
    477  1.1.1.1.6.1       tls  * drm_gem_prime_import - helper library implemention of the import callback
    478  1.1.1.1.6.1       tls  * @dev: drm_device to import into
    479  1.1.1.1.6.1       tls  * @dma_buf: dma-buf object to import
    480  1.1.1.1.6.1       tls  *
    481  1.1.1.1.6.1       tls  * This is the implementation of the gem_prime_import functions for GEM drivers
    482  1.1.1.1.6.1       tls  * using the PRIME helpers.
    483  1.1.1.1.6.1       tls  */
    484  1.1.1.1.6.1       tls struct drm_gem_object *drm_gem_prime_import(struct drm_device *dev,
    485  1.1.1.1.6.1       tls 					    struct dma_buf *dma_buf)
    486  1.1.1.1.6.1       tls {
    487  1.1.1.1.6.1       tls 	struct dma_buf_attachment *attach;
    488  1.1.1.1.6.1       tls 	struct sg_table *sgt;
    489  1.1.1.1.6.1       tls 	struct drm_gem_object *obj;
    490  1.1.1.1.6.1       tls 	int ret;
    491  1.1.1.1.6.1       tls 
    492  1.1.1.1.6.1       tls 	if (!dev->driver->gem_prime_import_sg_table)
    493  1.1.1.1.6.1       tls 		return ERR_PTR(-EINVAL);
    494  1.1.1.1.6.1       tls 
    495  1.1.1.1.6.1       tls 	if (dma_buf->ops == &drm_gem_prime_dmabuf_ops) {
    496  1.1.1.1.6.1       tls 		obj = dma_buf->priv;
    497  1.1.1.1.6.1       tls 		if (obj->dev == dev) {
    498  1.1.1.1.6.1       tls 			/*
    499  1.1.1.1.6.1       tls 			 * Importing dmabuf exported from out own gem increases
    500  1.1.1.1.6.1       tls 			 * refcount on gem itself instead of f_count of dmabuf.
    501  1.1.1.1.6.1       tls 			 */
    502  1.1.1.1.6.1       tls 			drm_gem_object_reference(obj);
    503  1.1.1.1.6.1       tls 			return obj;
    504  1.1.1.1.6.1       tls 		}
    505  1.1.1.1.6.1       tls 	}
    506  1.1.1.1.6.1       tls 
    507  1.1.1.1.6.1       tls 	attach = dma_buf_attach(dma_buf, dev->dev);
    508  1.1.1.1.6.1       tls 	if (IS_ERR(attach))
    509  1.1.1.1.6.1       tls 		return ERR_CAST(attach);
    510  1.1.1.1.6.1       tls 
    511  1.1.1.1.6.1       tls 	get_dma_buf(dma_buf);
    512  1.1.1.1.6.1       tls 
    513  1.1.1.1.6.1       tls 	sgt = dma_buf_map_attachment(attach, DMA_BIDIRECTIONAL);
    514  1.1.1.1.6.1       tls 	if (IS_ERR(sgt)) {
    515  1.1.1.1.6.1       tls 		ret = PTR_ERR(sgt);
    516  1.1.1.1.6.1       tls 		goto fail_detach;
    517  1.1.1.1.6.1       tls 	}
    518  1.1.1.1.6.1       tls 
    519  1.1.1.1.6.1       tls 	obj = dev->driver->gem_prime_import_sg_table(dev, dma_buf->size, sgt);
    520  1.1.1.1.6.1       tls 	if (IS_ERR(obj)) {
    521  1.1.1.1.6.1       tls 		ret = PTR_ERR(obj);
    522  1.1.1.1.6.1       tls 		goto fail_unmap;
    523  1.1.1.1.6.1       tls 	}
    524  1.1.1.1.6.1       tls 
    525  1.1.1.1.6.1       tls 	obj->import_attach = attach;
    526  1.1.1.1.6.1       tls 
    527  1.1.1.1.6.1       tls 	return obj;
    528  1.1.1.1.6.1       tls 
    529  1.1.1.1.6.1       tls fail_unmap:
    530  1.1.1.1.6.1       tls 	dma_buf_unmap_attachment(attach, sgt, DMA_BIDIRECTIONAL);
    531  1.1.1.1.6.1       tls fail_detach:
    532  1.1.1.1.6.1       tls 	dma_buf_detach(dma_buf, attach);
    533  1.1.1.1.6.1       tls 	dma_buf_put(dma_buf);
    534  1.1.1.1.6.1       tls 
    535  1.1.1.1.6.1       tls 	return ERR_PTR(ret);
    536  1.1.1.1.6.1       tls }
    537  1.1.1.1.6.1       tls EXPORT_SYMBOL(drm_gem_prime_import);
    538  1.1.1.1.6.1       tls 
    539  1.1.1.1.6.1       tls /**
    540  1.1.1.1.6.1       tls  * drm_gem_prime_fd_to_handle - PRIME import function for GEM drivers
    541  1.1.1.1.6.1       tls  * @dev: dev to export the buffer from
    542  1.1.1.1.6.1       tls  * @file_priv: drm file-private structure
    543  1.1.1.1.6.1       tls  * @prime_fd: fd id of the dma-buf which should be imported
    544  1.1.1.1.6.1       tls  * @handle: pointer to storage for the handle of the imported buffer object
    545  1.1.1.1.6.1       tls  *
    546  1.1.1.1.6.1       tls  * This is the PRIME import function which must be used mandatorily by GEM
    547  1.1.1.1.6.1       tls  * drivers to ensure correct lifetime management of the underlying GEM object.
    548  1.1.1.1.6.1       tls  * The actual importing of GEM object from the dma-buf is done through the
    549  1.1.1.1.6.1       tls  * gem_import_export driver callback.
    550  1.1.1.1.6.1       tls  */
    551          1.1  riastrad int drm_gem_prime_fd_to_handle(struct drm_device *dev,
    552  1.1.1.1.6.1       tls 			       struct drm_file *file_priv, int prime_fd,
    553  1.1.1.1.6.1       tls 			       uint32_t *handle)
    554          1.1  riastrad {
    555          1.1  riastrad 	struct dma_buf *dma_buf;
    556          1.1  riastrad 	struct drm_gem_object *obj;
    557          1.1  riastrad 	int ret;
    558          1.1  riastrad 
    559          1.1  riastrad 	dma_buf = dma_buf_get(prime_fd);
    560          1.1  riastrad 	if (IS_ERR(dma_buf))
    561          1.1  riastrad 		return PTR_ERR(dma_buf);
    562          1.1  riastrad 
    563          1.1  riastrad 	mutex_lock(&file_priv->prime.lock);
    564          1.1  riastrad 
    565  1.1.1.1.6.1       tls 	ret = drm_prime_lookup_buf_handle(&file_priv->prime,
    566          1.1  riastrad 			dma_buf, handle);
    567  1.1.1.1.6.1       tls 	if (ret == 0)
    568          1.1  riastrad 		goto out_put;
    569          1.1  riastrad 
    570          1.1  riastrad 	/* never seen this one, need to import */
    571  1.1.1.1.6.1       tls 	mutex_lock(&dev->object_name_lock);
    572          1.1  riastrad 	obj = dev->driver->gem_prime_import(dev, dma_buf);
    573          1.1  riastrad 	if (IS_ERR(obj)) {
    574          1.1  riastrad 		ret = PTR_ERR(obj);
    575  1.1.1.1.6.1       tls 		goto out_unlock;
    576          1.1  riastrad 	}
    577          1.1  riastrad 
    578  1.1.1.1.6.1       tls 	if (obj->dma_buf) {
    579  1.1.1.1.6.1       tls 		WARN_ON(obj->dma_buf != dma_buf);
    580  1.1.1.1.6.1       tls 	} else {
    581  1.1.1.1.6.1       tls 		obj->dma_buf = dma_buf;
    582  1.1.1.1.6.1       tls 		get_dma_buf(dma_buf);
    583  1.1.1.1.6.1       tls 	}
    584  1.1.1.1.6.1       tls 
    585  1.1.1.1.6.1       tls 	/* drm_gem_handle_create_tail unlocks dev->object_name_lock. */
    586  1.1.1.1.6.1       tls 	ret = drm_gem_handle_create_tail(file_priv, obj, handle);
    587          1.1  riastrad 	drm_gem_object_unreference_unlocked(obj);
    588          1.1  riastrad 	if (ret)
    589          1.1  riastrad 		goto out_put;
    590          1.1  riastrad 
    591  1.1.1.1.6.1       tls 	ret = drm_prime_add_buf_handle(&file_priv->prime,
    592          1.1  riastrad 			dma_buf, *handle);
    593          1.1  riastrad 	if (ret)
    594          1.1  riastrad 		goto fail;
    595          1.1  riastrad 
    596          1.1  riastrad 	mutex_unlock(&file_priv->prime.lock);
    597  1.1.1.1.6.1       tls 
    598  1.1.1.1.6.1       tls 	dma_buf_put(dma_buf);
    599  1.1.1.1.6.1       tls 
    600          1.1  riastrad 	return 0;
    601          1.1  riastrad 
    602          1.1  riastrad fail:
    603          1.1  riastrad 	/* hmm, if driver attached, we are relying on the free-object path
    604          1.1  riastrad 	 * to detach.. which seems ok..
    605          1.1  riastrad 	 */
    606  1.1.1.1.6.1       tls 	drm_gem_handle_delete(file_priv, *handle);
    607  1.1.1.1.6.1       tls out_unlock:
    608  1.1.1.1.6.1       tls 	mutex_unlock(&dev->object_name_lock);
    609          1.1  riastrad out_put:
    610          1.1  riastrad 	dma_buf_put(dma_buf);
    611          1.1  riastrad 	mutex_unlock(&file_priv->prime.lock);
    612          1.1  riastrad 	return ret;
    613          1.1  riastrad }
    614          1.1  riastrad EXPORT_SYMBOL(drm_gem_prime_fd_to_handle);
    615          1.1  riastrad 
    616          1.1  riastrad int drm_prime_handle_to_fd_ioctl(struct drm_device *dev, void *data,
    617          1.1  riastrad 				 struct drm_file *file_priv)
    618          1.1  riastrad {
    619          1.1  riastrad 	struct drm_prime_handle *args = data;
    620          1.1  riastrad 	uint32_t flags;
    621          1.1  riastrad 
    622          1.1  riastrad 	if (!drm_core_check_feature(dev, DRIVER_PRIME))
    623          1.1  riastrad 		return -EINVAL;
    624          1.1  riastrad 
    625          1.1  riastrad 	if (!dev->driver->prime_handle_to_fd)
    626          1.1  riastrad 		return -ENOSYS;
    627          1.1  riastrad 
    628          1.1  riastrad 	/* check flags are valid */
    629          1.1  riastrad 	if (args->flags & ~DRM_CLOEXEC)
    630          1.1  riastrad 		return -EINVAL;
    631          1.1  riastrad 
    632          1.1  riastrad 	/* we only want to pass DRM_CLOEXEC which is == O_CLOEXEC */
    633          1.1  riastrad 	flags = args->flags & DRM_CLOEXEC;
    634          1.1  riastrad 
    635          1.1  riastrad 	return dev->driver->prime_handle_to_fd(dev, file_priv,
    636          1.1  riastrad 			args->handle, flags, &args->fd);
    637          1.1  riastrad }
    638          1.1  riastrad 
    639          1.1  riastrad int drm_prime_fd_to_handle_ioctl(struct drm_device *dev, void *data,
    640          1.1  riastrad 				 struct drm_file *file_priv)
    641          1.1  riastrad {
    642          1.1  riastrad 	struct drm_prime_handle *args = data;
    643          1.1  riastrad 
    644          1.1  riastrad 	if (!drm_core_check_feature(dev, DRIVER_PRIME))
    645          1.1  riastrad 		return -EINVAL;
    646          1.1  riastrad 
    647          1.1  riastrad 	if (!dev->driver->prime_fd_to_handle)
    648          1.1  riastrad 		return -ENOSYS;
    649          1.1  riastrad 
    650          1.1  riastrad 	return dev->driver->prime_fd_to_handle(dev, file_priv,
    651          1.1  riastrad 			args->fd, &args->handle);
    652          1.1  riastrad }
    653          1.1  riastrad 
    654  1.1.1.1.6.1       tls /**
    655  1.1.1.1.6.1       tls  * drm_prime_pages_to_sg - converts a page array into an sg list
    656  1.1.1.1.6.1       tls  * @pages: pointer to the array of page pointers to convert
    657  1.1.1.1.6.1       tls  * @nr_pages: length of the page vector
    658          1.1  riastrad  *
    659  1.1.1.1.6.1       tls  * This helper creates an sg table object from a set of pages
    660          1.1  riastrad  * the driver is responsible for mapping the pages into the
    661  1.1.1.1.6.1       tls  * importers address space for use with dma_buf itself.
    662          1.1  riastrad  */
    663          1.1  riastrad struct sg_table *drm_prime_pages_to_sg(struct page **pages, int nr_pages)
    664          1.1  riastrad {
    665          1.1  riastrad 	struct sg_table *sg = NULL;
    666          1.1  riastrad 	int ret;
    667          1.1  riastrad 
    668          1.1  riastrad 	sg = kmalloc(sizeof(struct sg_table), GFP_KERNEL);
    669  1.1.1.1.6.1       tls 	if (!sg) {
    670  1.1.1.1.6.1       tls 		ret = -ENOMEM;
    671          1.1  riastrad 		goto out;
    672  1.1.1.1.6.1       tls 	}
    673          1.1  riastrad 
    674  1.1.1.1.6.1       tls 	ret = sg_alloc_table_from_pages(sg, pages, nr_pages, 0,
    675  1.1.1.1.6.1       tls 				nr_pages << PAGE_SHIFT, GFP_KERNEL);
    676          1.1  riastrad 	if (ret)
    677          1.1  riastrad 		goto out;
    678          1.1  riastrad 
    679          1.1  riastrad 	return sg;
    680          1.1  riastrad out:
    681          1.1  riastrad 	kfree(sg);
    682  1.1.1.1.6.1       tls 	return ERR_PTR(ret);
    683          1.1  riastrad }
    684          1.1  riastrad EXPORT_SYMBOL(drm_prime_pages_to_sg);
    685          1.1  riastrad 
    686  1.1.1.1.6.1       tls /**
    687  1.1.1.1.6.1       tls  * drm_prime_sg_to_page_addr_arrays - convert an sg table into a page array
    688  1.1.1.1.6.1       tls  * @sgt: scatter-gather table to convert
    689  1.1.1.1.6.1       tls  * @pages: array of page pointers to store the page array in
    690  1.1.1.1.6.1       tls  * @addrs: optional array to store the dma bus address of each page
    691  1.1.1.1.6.1       tls  * @max_pages: size of both the passed-in arrays
    692  1.1.1.1.6.1       tls  *
    693  1.1.1.1.6.1       tls  * Exports an sg table into an array of pages and addresses. This is currently
    694  1.1.1.1.6.1       tls  * required by the TTM driver in order to do correct fault handling.
    695  1.1.1.1.6.1       tls  */
    696          1.1  riastrad int drm_prime_sg_to_page_addr_arrays(struct sg_table *sgt, struct page **pages,
    697          1.1  riastrad 				     dma_addr_t *addrs, int max_pages)
    698          1.1  riastrad {
    699          1.1  riastrad 	unsigned count;
    700          1.1  riastrad 	struct scatterlist *sg;
    701          1.1  riastrad 	struct page *page;
    702  1.1.1.1.6.1       tls 	u32 len;
    703          1.1  riastrad 	int pg_index;
    704          1.1  riastrad 	dma_addr_t addr;
    705          1.1  riastrad 
    706          1.1  riastrad 	pg_index = 0;
    707          1.1  riastrad 	for_each_sg(sgt->sgl, sg, sgt->nents, count) {
    708          1.1  riastrad 		len = sg->length;
    709          1.1  riastrad 		page = sg_page(sg);
    710          1.1  riastrad 		addr = sg_dma_address(sg);
    711          1.1  riastrad 
    712          1.1  riastrad 		while (len > 0) {
    713          1.1  riastrad 			if (WARN_ON(pg_index >= max_pages))
    714          1.1  riastrad 				return -1;
    715          1.1  riastrad 			pages[pg_index] = page;
    716          1.1  riastrad 			if (addrs)
    717          1.1  riastrad 				addrs[pg_index] = addr;
    718          1.1  riastrad 
    719          1.1  riastrad 			page++;
    720          1.1  riastrad 			addr += PAGE_SIZE;
    721          1.1  riastrad 			len -= PAGE_SIZE;
    722          1.1  riastrad 			pg_index++;
    723          1.1  riastrad 		}
    724          1.1  riastrad 	}
    725          1.1  riastrad 	return 0;
    726          1.1  riastrad }
    727          1.1  riastrad EXPORT_SYMBOL(drm_prime_sg_to_page_addr_arrays);
    728  1.1.1.1.6.1       tls 
    729  1.1.1.1.6.1       tls /**
    730  1.1.1.1.6.1       tls  * drm_prime_gem_destroy - helper to clean up a PRIME-imported GEM object
    731  1.1.1.1.6.1       tls  * @obj: GEM object which was created from a dma-buf
    732  1.1.1.1.6.1       tls  * @sg: the sg-table which was pinned at import time
    733  1.1.1.1.6.1       tls  *
    734  1.1.1.1.6.1       tls  * This is the cleanup functions which GEM drivers need to call when they use
    735  1.1.1.1.6.1       tls  * @drm_gem_prime_import to import dma-bufs.
    736  1.1.1.1.6.1       tls  */
    737          1.1  riastrad void drm_prime_gem_destroy(struct drm_gem_object *obj, struct sg_table *sg)
    738          1.1  riastrad {
    739          1.1  riastrad 	struct dma_buf_attachment *attach;
    740          1.1  riastrad 	struct dma_buf *dma_buf;
    741          1.1  riastrad 	attach = obj->import_attach;
    742          1.1  riastrad 	if (sg)
    743          1.1  riastrad 		dma_buf_unmap_attachment(attach, sg, DMA_BIDIRECTIONAL);
    744          1.1  riastrad 	dma_buf = attach->dmabuf;
    745          1.1  riastrad 	dma_buf_detach(attach->dmabuf, attach);
    746          1.1  riastrad 	/* remove the reference */
    747          1.1  riastrad 	dma_buf_put(dma_buf);
    748          1.1  riastrad }
    749          1.1  riastrad EXPORT_SYMBOL(drm_prime_gem_destroy);
    750          1.1  riastrad 
    751          1.1  riastrad void drm_prime_init_file_private(struct drm_prime_file_private *prime_fpriv)
    752          1.1  riastrad {
    753          1.1  riastrad 	INIT_LIST_HEAD(&prime_fpriv->head);
    754          1.1  riastrad 	mutex_init(&prime_fpriv->lock);
    755          1.1  riastrad }
    756          1.1  riastrad 
    757          1.1  riastrad void drm_prime_destroy_file_private(struct drm_prime_file_private *prime_fpriv)
    758          1.1  riastrad {
    759  1.1.1.1.6.1       tls 	/* by now drm_gem_release should've made sure the list is empty */
    760  1.1.1.1.6.1       tls 	WARN_ON(!list_empty(&prime_fpriv->head));
    761          1.1  riastrad }
    762