#include <linux/dma-buf.h>
#include <linux/anon_inodes.h>
#include <linux/export.h>
+#ifdef CONFIG_DMA_SHARED_BUFFER_USES_KDS
#include <linux/poll.h>
#include <linux/sched.h>
+#endif
static inline int is_dma_buf_file(struct file *);
dmabuf = file->private_data;
dmabuf->ops->release(dmabuf);
+#ifdef CONFIG_DMA_SHARED_BUFFER_USES_KDS
kds_callback_term(&dmabuf->kds_cb);
kds_resource_term(&dmabuf->kds);
+#endif
kfree(dmabuf);
return 0;
}
return dmabuf->ops->mmap(dmabuf, vma);
}
-
+#ifdef CONFIG_DMA_SHARED_BUFFER_USES_KDS
static void dma_buf_kds_cb_fn(void *param1, void *param2)
{
struct kds_resource_set **rset_ptr = param1;
}
return ret;
}
+#endif
static const struct file_operations dma_buf_fops = {
.release = dma_buf_release,
.mmap = dma_buf_mmap_internal,
+#ifdef CONFIG_DMA_SHARED_BUFFER_USES_KDS
.poll = dma_buf_poll,
+#endif
};
/*
mutex_init(&dmabuf->lock);
INIT_LIST_HEAD(&dmabuf->attachments);
+#ifdef CONFIG_DMA_SHARED_BUFFER_USES_KDS
init_waitqueue_head(&dmabuf->wq_exclusive);
init_waitqueue_head(&dmabuf->wq_shared);
kds_resource_init(&dmabuf->kds);
kds_callback_init(&dmabuf->kds_cb, 1, dma_buf_kds_cb_fn);
+#endif
return dmabuf;
}
int dma_buf_mmap(struct dma_buf *dmabuf, struct vm_area_struct *vma,
unsigned long pgoff)
{
+ struct file *oldfile;
+ int ret;
+
if (WARN_ON(!dmabuf || !vma))
return -EINVAL;
return -EINVAL;
/* readjust the vma */
- if (vma->vm_file)
- fput(vma->vm_file);
-
+ get_file(dmabuf->file);
+ oldfile = vma->vm_file;
vma->vm_file = dmabuf->file;
- get_file(vma->vm_file);
-
vma->vm_pgoff = pgoff;
- return dmabuf->ops->mmap(dmabuf, vma);
+ ret = dmabuf->ops->mmap(dmabuf, vma);
+ if (ret) {
+ /* restore old parameters on failure */
+ vma->vm_file = oldfile;
+ fput(dmabuf->file);
+ } else {
+ if (oldfile)
+ fput(oldfile);
+ }
+ return ret;
}
EXPORT_SYMBOL_GPL(dma_buf_mmap);