forked from Minki/linux
6542e9adc0
4.19 is out, Lyude asked for a backmerge, and it's been a while. All very good reasons on their own :-) Signed-off-by: Sean Paul <seanpaul@chromium.org>
1218 lines
30 KiB
C
1218 lines
30 KiB
C
/*
|
|
* Copyright 2017 Red Hat
|
|
* Parts ported from amdgpu (fence wait code).
|
|
* Copyright 2016 Advanced Micro Devices, Inc.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice (including the next
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
* Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
* IN THE SOFTWARE.
|
|
*
|
|
* Authors:
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* DOC: Overview
|
|
*
|
|
* DRM synchronisation objects (syncobj, see struct &drm_syncobj) are
|
|
* persistent objects that contain an optional fence. The fence can be updated
|
|
* with a new fence, or be NULL.
|
|
*
|
|
* syncobj's can be waited upon, where it will wait for the underlying
|
|
* fence.
|
|
*
|
|
* syncobj's can be export to fd's and back, these fd's are opaque and
|
|
* have no other use case, except passing the syncobj between processes.
|
|
*
|
|
* Their primary use-case is to implement Vulkan fences and semaphores.
|
|
*
|
|
* syncobj have a kref reference count, but also have an optional file.
|
|
* The file is only created once the syncobj is exported.
|
|
* The file takes a reference on the kref.
|
|
*/
|
|
|
|
#include <drm/drmP.h>
|
|
#include <linux/file.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/anon_inodes.h>
|
|
#include <linux/sync_file.h>
|
|
#include <linux/sched/signal.h>
|
|
|
|
#include "drm_internal.h"
|
|
#include <drm/drm_syncobj.h>
|
|
|
|
/* merge normal syncobj to timeline syncobj, the point interval is 1 */
|
|
#define DRM_SYNCOBJ_BINARY_POINT 1
|
|
|
|
struct drm_syncobj_stub_fence {
|
|
struct dma_fence base;
|
|
spinlock_t lock;
|
|
};
|
|
|
|
static const char *drm_syncobj_stub_fence_get_name(struct dma_fence *fence)
|
|
{
|
|
return "syncobjstub";
|
|
}
|
|
|
|
static const struct dma_fence_ops drm_syncobj_stub_fence_ops = {
|
|
.get_driver_name = drm_syncobj_stub_fence_get_name,
|
|
.get_timeline_name = drm_syncobj_stub_fence_get_name,
|
|
};
|
|
|
|
struct drm_syncobj_signal_pt {
|
|
struct dma_fence_array *fence_array;
|
|
u64 value;
|
|
struct list_head list;
|
|
};
|
|
|
|
/**
|
|
* drm_syncobj_find - lookup and reference a sync object.
|
|
* @file_private: drm file private pointer
|
|
* @handle: sync object handle to lookup.
|
|
*
|
|
* Returns a reference to the syncobj pointed to by handle or NULL. The
|
|
* reference must be released by calling drm_syncobj_put().
|
|
*/
|
|
struct drm_syncobj *drm_syncobj_find(struct drm_file *file_private,
|
|
u32 handle)
|
|
{
|
|
struct drm_syncobj *syncobj;
|
|
|
|
spin_lock(&file_private->syncobj_table_lock);
|
|
|
|
/* Check if we currently have a reference on the object */
|
|
syncobj = idr_find(&file_private->syncobj_idr, handle);
|
|
if (syncobj)
|
|
drm_syncobj_get(syncobj);
|
|
|
|
spin_unlock(&file_private->syncobj_table_lock);
|
|
|
|
return syncobj;
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_find);
|
|
|
|
static struct dma_fence
|
|
*drm_syncobj_find_signal_pt_for_point(struct drm_syncobj *syncobj,
|
|
uint64_t point)
|
|
{
|
|
struct drm_syncobj_signal_pt *signal_pt;
|
|
|
|
if ((syncobj->type == DRM_SYNCOBJ_TYPE_TIMELINE) &&
|
|
(point <= syncobj->timeline)) {
|
|
struct drm_syncobj_stub_fence *fence =
|
|
kzalloc(sizeof(struct drm_syncobj_stub_fence),
|
|
GFP_KERNEL);
|
|
|
|
if (!fence)
|
|
return NULL;
|
|
spin_lock_init(&fence->lock);
|
|
dma_fence_init(&fence->base,
|
|
&drm_syncobj_stub_fence_ops,
|
|
&fence->lock,
|
|
syncobj->timeline_context,
|
|
point);
|
|
|
|
dma_fence_signal(&fence->base);
|
|
return &fence->base;
|
|
}
|
|
|
|
list_for_each_entry(signal_pt, &syncobj->signal_pt_list, list) {
|
|
if (point > signal_pt->value)
|
|
continue;
|
|
if ((syncobj->type == DRM_SYNCOBJ_TYPE_BINARY) &&
|
|
(point != signal_pt->value))
|
|
continue;
|
|
return dma_fence_get(&signal_pt->fence_array->base);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void drm_syncobj_add_callback_locked(struct drm_syncobj *syncobj,
|
|
struct drm_syncobj_cb *cb,
|
|
drm_syncobj_func_t func)
|
|
{
|
|
cb->func = func;
|
|
list_add_tail(&cb->node, &syncobj->cb_list);
|
|
}
|
|
|
|
static void drm_syncobj_fence_get_or_add_callback(struct drm_syncobj *syncobj,
|
|
struct dma_fence **fence,
|
|
struct drm_syncobj_cb *cb,
|
|
drm_syncobj_func_t func)
|
|
{
|
|
u64 pt_value = 0;
|
|
|
|
WARN_ON(*fence);
|
|
|
|
if (syncobj->type == DRM_SYNCOBJ_TYPE_BINARY) {
|
|
/*BINARY syncobj always wait on last pt */
|
|
pt_value = syncobj->signal_point;
|
|
|
|
if (pt_value == 0)
|
|
pt_value += DRM_SYNCOBJ_BINARY_POINT;
|
|
}
|
|
|
|
mutex_lock(&syncobj->cb_mutex);
|
|
spin_lock(&syncobj->pt_lock);
|
|
*fence = drm_syncobj_find_signal_pt_for_point(syncobj, pt_value);
|
|
spin_unlock(&syncobj->pt_lock);
|
|
if (!*fence)
|
|
drm_syncobj_add_callback_locked(syncobj, cb, func);
|
|
mutex_unlock(&syncobj->cb_mutex);
|
|
}
|
|
|
|
void drm_syncobj_add_callback(struct drm_syncobj *syncobj,
|
|
struct drm_syncobj_cb *cb,
|
|
drm_syncobj_func_t func)
|
|
{
|
|
mutex_lock(&syncobj->cb_mutex);
|
|
drm_syncobj_add_callback_locked(syncobj, cb, func);
|
|
mutex_unlock(&syncobj->cb_mutex);
|
|
}
|
|
|
|
void drm_syncobj_remove_callback(struct drm_syncobj *syncobj,
|
|
struct drm_syncobj_cb *cb)
|
|
{
|
|
mutex_lock(&syncobj->cb_mutex);
|
|
list_del_init(&cb->node);
|
|
mutex_unlock(&syncobj->cb_mutex);
|
|
}
|
|
|
|
static void drm_syncobj_init(struct drm_syncobj *syncobj)
|
|
{
|
|
spin_lock(&syncobj->pt_lock);
|
|
syncobj->timeline_context = dma_fence_context_alloc(1);
|
|
syncobj->timeline = 0;
|
|
syncobj->signal_point = 0;
|
|
init_waitqueue_head(&syncobj->wq);
|
|
|
|
INIT_LIST_HEAD(&syncobj->signal_pt_list);
|
|
spin_unlock(&syncobj->pt_lock);
|
|
}
|
|
|
|
static void drm_syncobj_fini(struct drm_syncobj *syncobj)
|
|
{
|
|
struct drm_syncobj_signal_pt *signal_pt = NULL, *tmp;
|
|
|
|
spin_lock(&syncobj->pt_lock);
|
|
list_for_each_entry_safe(signal_pt, tmp,
|
|
&syncobj->signal_pt_list, list) {
|
|
list_del(&signal_pt->list);
|
|
dma_fence_put(&signal_pt->fence_array->base);
|
|
kfree(signal_pt);
|
|
}
|
|
spin_unlock(&syncobj->pt_lock);
|
|
}
|
|
|
|
static int drm_syncobj_create_signal_pt(struct drm_syncobj *syncobj,
|
|
struct dma_fence *fence,
|
|
u64 point)
|
|
{
|
|
struct drm_syncobj_signal_pt *signal_pt =
|
|
kzalloc(sizeof(struct drm_syncobj_signal_pt), GFP_KERNEL);
|
|
struct drm_syncobj_signal_pt *tail_pt;
|
|
struct dma_fence **fences;
|
|
int num_fences = 0;
|
|
int ret = 0, i;
|
|
|
|
if (!signal_pt)
|
|
return -ENOMEM;
|
|
if (!fence)
|
|
goto out;
|
|
|
|
fences = kmalloc_array(sizeof(void *), 2, GFP_KERNEL);
|
|
if (!fences) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
fences[num_fences++] = dma_fence_get(fence);
|
|
/* timeline syncobj must take this dependency */
|
|
if (syncobj->type == DRM_SYNCOBJ_TYPE_TIMELINE) {
|
|
spin_lock(&syncobj->pt_lock);
|
|
if (!list_empty(&syncobj->signal_pt_list)) {
|
|
tail_pt = list_last_entry(&syncobj->signal_pt_list,
|
|
struct drm_syncobj_signal_pt, list);
|
|
fences[num_fences++] =
|
|
dma_fence_get(&tail_pt->fence_array->base);
|
|
}
|
|
spin_unlock(&syncobj->pt_lock);
|
|
}
|
|
signal_pt->fence_array = dma_fence_array_create(num_fences, fences,
|
|
syncobj->timeline_context,
|
|
point, false);
|
|
if (!signal_pt->fence_array) {
|
|
ret = -ENOMEM;
|
|
goto fail;
|
|
}
|
|
|
|
spin_lock(&syncobj->pt_lock);
|
|
if (syncobj->signal_point >= point) {
|
|
DRM_WARN("A later signal is ready!");
|
|
spin_unlock(&syncobj->pt_lock);
|
|
goto exist;
|
|
}
|
|
signal_pt->value = point;
|
|
list_add_tail(&signal_pt->list, &syncobj->signal_pt_list);
|
|
syncobj->signal_point = point;
|
|
spin_unlock(&syncobj->pt_lock);
|
|
wake_up_all(&syncobj->wq);
|
|
|
|
return 0;
|
|
exist:
|
|
dma_fence_put(&signal_pt->fence_array->base);
|
|
fail:
|
|
for (i = 0; i < num_fences; i++)
|
|
dma_fence_put(fences[i]);
|
|
kfree(fences);
|
|
out:
|
|
kfree(signal_pt);
|
|
return ret;
|
|
}
|
|
|
|
static void drm_syncobj_garbage_collection(struct drm_syncobj *syncobj)
|
|
{
|
|
struct drm_syncobj_signal_pt *signal_pt, *tmp, *tail_pt;
|
|
|
|
spin_lock(&syncobj->pt_lock);
|
|
tail_pt = list_last_entry(&syncobj->signal_pt_list,
|
|
struct drm_syncobj_signal_pt,
|
|
list);
|
|
list_for_each_entry_safe(signal_pt, tmp,
|
|
&syncobj->signal_pt_list, list) {
|
|
if (syncobj->type == DRM_SYNCOBJ_TYPE_BINARY &&
|
|
signal_pt == tail_pt)
|
|
continue;
|
|
if (dma_fence_is_signaled(&signal_pt->fence_array->base)) {
|
|
syncobj->timeline = signal_pt->value;
|
|
list_del(&signal_pt->list);
|
|
dma_fence_put(&signal_pt->fence_array->base);
|
|
kfree(signal_pt);
|
|
} else {
|
|
/*signal_pt is in order in list, from small to big, so
|
|
* the later must not be signal either */
|
|
break;
|
|
}
|
|
}
|
|
|
|
spin_unlock(&syncobj->pt_lock);
|
|
}
|
|
/**
|
|
* drm_syncobj_replace_fence - replace fence in a sync object.
|
|
* @syncobj: Sync object to replace fence in
|
|
* @point: timeline point
|
|
* @fence: fence to install in sync file.
|
|
*
|
|
* This replaces the fence on a sync object, or a timeline point fence.
|
|
*/
|
|
void drm_syncobj_replace_fence(struct drm_syncobj *syncobj,
|
|
u64 point,
|
|
struct dma_fence *fence)
|
|
{
|
|
u64 pt_value = point;
|
|
|
|
drm_syncobj_garbage_collection(syncobj);
|
|
if (syncobj->type == DRM_SYNCOBJ_TYPE_BINARY) {
|
|
if (!fence) {
|
|
drm_syncobj_fini(syncobj);
|
|
drm_syncobj_init(syncobj);
|
|
return;
|
|
}
|
|
pt_value = syncobj->signal_point +
|
|
DRM_SYNCOBJ_BINARY_POINT;
|
|
}
|
|
drm_syncobj_create_signal_pt(syncobj, fence, pt_value);
|
|
if (fence) {
|
|
struct drm_syncobj_cb *cur, *tmp;
|
|
LIST_HEAD(cb_list);
|
|
|
|
mutex_lock(&syncobj->cb_mutex);
|
|
list_for_each_entry_safe(cur, tmp, &syncobj->cb_list, node) {
|
|
list_del_init(&cur->node);
|
|
cur->func(syncobj, cur);
|
|
}
|
|
mutex_unlock(&syncobj->cb_mutex);
|
|
}
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_replace_fence);
|
|
|
|
static int drm_syncobj_assign_null_handle(struct drm_syncobj *syncobj)
|
|
{
|
|
struct drm_syncobj_stub_fence *fence;
|
|
fence = kzalloc(sizeof(*fence), GFP_KERNEL);
|
|
if (fence == NULL)
|
|
return -ENOMEM;
|
|
|
|
spin_lock_init(&fence->lock);
|
|
dma_fence_init(&fence->base, &drm_syncobj_stub_fence_ops,
|
|
&fence->lock, 0, 0);
|
|
dma_fence_signal(&fence->base);
|
|
|
|
drm_syncobj_replace_fence(syncobj, 0, &fence->base);
|
|
|
|
dma_fence_put(&fence->base);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
drm_syncobj_point_get(struct drm_syncobj *syncobj, u64 point, u64 flags,
|
|
struct dma_fence **fence)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (flags & DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT) {
|
|
ret = wait_event_interruptible(syncobj->wq,
|
|
point <= syncobj->signal_point);
|
|
if (ret < 0)
|
|
return ret;
|
|
}
|
|
spin_lock(&syncobj->pt_lock);
|
|
*fence = drm_syncobj_find_signal_pt_for_point(syncobj, point);
|
|
if (!*fence)
|
|
ret = -EINVAL;
|
|
spin_unlock(&syncobj->pt_lock);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* drm_syncobj_search_fence - lookup and reference the fence in a sync object or
|
|
* in a timeline point
|
|
* @syncobj: sync object pointer
|
|
* @point: timeline point
|
|
* @flags: DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT or not
|
|
* @fence: out parameter for the fence
|
|
*
|
|
* if flags is DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT, the function will block
|
|
* here until specific timeline points is reached.
|
|
* if not, you need a submit thread and block in userspace until all future
|
|
* timeline points have materialized, only then you can submit to the kernel,
|
|
* otherwise, function will fail to return fence.
|
|
*
|
|
* Returns 0 on success or a negative error value on failure. On success @fence
|
|
* contains a reference to the fence, which must be released by calling
|
|
* dma_fence_put().
|
|
*/
|
|
int drm_syncobj_search_fence(struct drm_syncobj *syncobj, u64 point,
|
|
u64 flags, struct dma_fence **fence)
|
|
{
|
|
u64 pt_value = point;
|
|
|
|
if (!syncobj)
|
|
return -ENOENT;
|
|
|
|
drm_syncobj_garbage_collection(syncobj);
|
|
if (syncobj->type == DRM_SYNCOBJ_TYPE_BINARY) {
|
|
/*BINARY syncobj always wait on last pt */
|
|
pt_value = syncobj->signal_point;
|
|
|
|
if (pt_value == 0)
|
|
pt_value += DRM_SYNCOBJ_BINARY_POINT;
|
|
}
|
|
return drm_syncobj_point_get(syncobj, pt_value, flags, fence);
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_search_fence);
|
|
|
|
/**
|
|
* drm_syncobj_find_fence - lookup and reference the fence in a sync object
|
|
* @file_private: drm file private pointer
|
|
* @handle: sync object handle to lookup.
|
|
* @point: timeline point
|
|
* @flags: DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT or not
|
|
* @fence: out parameter for the fence
|
|
*
|
|
* This is just a convenience function that combines drm_syncobj_find() and
|
|
* drm_syncobj_lookup_fence().
|
|
*
|
|
* Returns 0 on success or a negative error value on failure. On success @fence
|
|
* contains a reference to the fence, which must be released by calling
|
|
* dma_fence_put().
|
|
*/
|
|
int drm_syncobj_find_fence(struct drm_file *file_private,
|
|
u32 handle, u64 point, u64 flags,
|
|
struct dma_fence **fence)
|
|
{
|
|
struct drm_syncobj *syncobj = drm_syncobj_find(file_private, handle);
|
|
int ret;
|
|
|
|
ret = drm_syncobj_search_fence(syncobj, point, flags, fence);
|
|
drm_syncobj_put(syncobj);
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_find_fence);
|
|
|
|
/**
|
|
* drm_syncobj_free - free a sync object.
|
|
* @kref: kref to free.
|
|
*
|
|
* Only to be called from kref_put in drm_syncobj_put.
|
|
*/
|
|
void drm_syncobj_free(struct kref *kref)
|
|
{
|
|
struct drm_syncobj *syncobj = container_of(kref,
|
|
struct drm_syncobj,
|
|
refcount);
|
|
drm_syncobj_fini(syncobj);
|
|
kfree(syncobj);
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_free);
|
|
|
|
/**
|
|
* drm_syncobj_create - create a new syncobj
|
|
* @out_syncobj: returned syncobj
|
|
* @flags: DRM_SYNCOBJ_* flags
|
|
* @fence: if non-NULL, the syncobj will represent this fence
|
|
*
|
|
* This is the first function to create a sync object. After creating, drivers
|
|
* probably want to make it available to userspace, either through
|
|
* drm_syncobj_get_handle() or drm_syncobj_get_fd().
|
|
*
|
|
* Returns 0 on success or a negative error value on failure.
|
|
*/
|
|
int drm_syncobj_create(struct drm_syncobj **out_syncobj, uint32_t flags,
|
|
struct dma_fence *fence)
|
|
{
|
|
int ret;
|
|
struct drm_syncobj *syncobj;
|
|
|
|
syncobj = kzalloc(sizeof(struct drm_syncobj), GFP_KERNEL);
|
|
if (!syncobj)
|
|
return -ENOMEM;
|
|
|
|
kref_init(&syncobj->refcount);
|
|
INIT_LIST_HEAD(&syncobj->cb_list);
|
|
spin_lock_init(&syncobj->pt_lock);
|
|
mutex_init(&syncobj->cb_mutex);
|
|
if (flags & DRM_SYNCOBJ_CREATE_TYPE_TIMELINE)
|
|
syncobj->type = DRM_SYNCOBJ_TYPE_TIMELINE;
|
|
else
|
|
syncobj->type = DRM_SYNCOBJ_TYPE_BINARY;
|
|
drm_syncobj_init(syncobj);
|
|
|
|
if (flags & DRM_SYNCOBJ_CREATE_SIGNALED) {
|
|
ret = drm_syncobj_assign_null_handle(syncobj);
|
|
if (ret < 0) {
|
|
drm_syncobj_put(syncobj);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
if (fence)
|
|
drm_syncobj_replace_fence(syncobj, 0, fence);
|
|
|
|
*out_syncobj = syncobj;
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_create);
|
|
|
|
/**
|
|
* drm_syncobj_get_handle - get a handle from a syncobj
|
|
* @file_private: drm file private pointer
|
|
* @syncobj: Sync object to export
|
|
* @handle: out parameter with the new handle
|
|
*
|
|
* Exports a sync object created with drm_syncobj_create() as a handle on
|
|
* @file_private to userspace.
|
|
*
|
|
* Returns 0 on success or a negative error value on failure.
|
|
*/
|
|
int drm_syncobj_get_handle(struct drm_file *file_private,
|
|
struct drm_syncobj *syncobj, u32 *handle)
|
|
{
|
|
int ret;
|
|
|
|
/* take a reference to put in the idr */
|
|
drm_syncobj_get(syncobj);
|
|
|
|
idr_preload(GFP_KERNEL);
|
|
spin_lock(&file_private->syncobj_table_lock);
|
|
ret = idr_alloc(&file_private->syncobj_idr, syncobj, 1, 0, GFP_NOWAIT);
|
|
spin_unlock(&file_private->syncobj_table_lock);
|
|
|
|
idr_preload_end();
|
|
|
|
if (ret < 0) {
|
|
drm_syncobj_put(syncobj);
|
|
return ret;
|
|
}
|
|
|
|
*handle = ret;
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_get_handle);
|
|
|
|
static int drm_syncobj_create_as_handle(struct drm_file *file_private,
|
|
u32 *handle, uint32_t flags)
|
|
{
|
|
int ret;
|
|
struct drm_syncobj *syncobj;
|
|
|
|
ret = drm_syncobj_create(&syncobj, flags, NULL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = drm_syncobj_get_handle(file_private, syncobj, handle);
|
|
drm_syncobj_put(syncobj);
|
|
return ret;
|
|
}
|
|
|
|
static int drm_syncobj_destroy(struct drm_file *file_private,
|
|
u32 handle)
|
|
{
|
|
struct drm_syncobj *syncobj;
|
|
|
|
spin_lock(&file_private->syncobj_table_lock);
|
|
syncobj = idr_remove(&file_private->syncobj_idr, handle);
|
|
spin_unlock(&file_private->syncobj_table_lock);
|
|
|
|
if (!syncobj)
|
|
return -EINVAL;
|
|
|
|
drm_syncobj_put(syncobj);
|
|
return 0;
|
|
}
|
|
|
|
static int drm_syncobj_file_release(struct inode *inode, struct file *file)
|
|
{
|
|
struct drm_syncobj *syncobj = file->private_data;
|
|
|
|
drm_syncobj_put(syncobj);
|
|
return 0;
|
|
}
|
|
|
|
static const struct file_operations drm_syncobj_file_fops = {
|
|
.release = drm_syncobj_file_release,
|
|
};
|
|
|
|
/**
|
|
* drm_syncobj_get_fd - get a file descriptor from a syncobj
|
|
* @syncobj: Sync object to export
|
|
* @p_fd: out parameter with the new file descriptor
|
|
*
|
|
* Exports a sync object created with drm_syncobj_create() as a file descriptor.
|
|
*
|
|
* Returns 0 on success or a negative error value on failure.
|
|
*/
|
|
int drm_syncobj_get_fd(struct drm_syncobj *syncobj, int *p_fd)
|
|
{
|
|
struct file *file;
|
|
int fd;
|
|
|
|
fd = get_unused_fd_flags(O_CLOEXEC);
|
|
if (fd < 0)
|
|
return fd;
|
|
|
|
file = anon_inode_getfile("syncobj_file",
|
|
&drm_syncobj_file_fops,
|
|
syncobj, 0);
|
|
if (IS_ERR(file)) {
|
|
put_unused_fd(fd);
|
|
return PTR_ERR(file);
|
|
}
|
|
|
|
drm_syncobj_get(syncobj);
|
|
fd_install(fd, file);
|
|
|
|
*p_fd = fd;
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL(drm_syncobj_get_fd);
|
|
|
|
static int drm_syncobj_handle_to_fd(struct drm_file *file_private,
|
|
u32 handle, int *p_fd)
|
|
{
|
|
struct drm_syncobj *syncobj = drm_syncobj_find(file_private, handle);
|
|
int ret;
|
|
|
|
if (!syncobj)
|
|
return -EINVAL;
|
|
|
|
ret = drm_syncobj_get_fd(syncobj, p_fd);
|
|
drm_syncobj_put(syncobj);
|
|
return ret;
|
|
}
|
|
|
|
static int drm_syncobj_fd_to_handle(struct drm_file *file_private,
|
|
int fd, u32 *handle)
|
|
{
|
|
struct drm_syncobj *syncobj;
|
|
struct file *file;
|
|
int ret;
|
|
|
|
file = fget(fd);
|
|
if (!file)
|
|
return -EINVAL;
|
|
|
|
if (file->f_op != &drm_syncobj_file_fops) {
|
|
fput(file);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* take a reference to put in the idr */
|
|
syncobj = file->private_data;
|
|
drm_syncobj_get(syncobj);
|
|
|
|
idr_preload(GFP_KERNEL);
|
|
spin_lock(&file_private->syncobj_table_lock);
|
|
ret = idr_alloc(&file_private->syncobj_idr, syncobj, 1, 0, GFP_NOWAIT);
|
|
spin_unlock(&file_private->syncobj_table_lock);
|
|
idr_preload_end();
|
|
|
|
if (ret > 0) {
|
|
*handle = ret;
|
|
ret = 0;
|
|
} else
|
|
drm_syncobj_put(syncobj);
|
|
|
|
fput(file);
|
|
return ret;
|
|
}
|
|
|
|
static int drm_syncobj_import_sync_file_fence(struct drm_file *file_private,
|
|
int fd, int handle)
|
|
{
|
|
struct dma_fence *fence = sync_file_get_fence(fd);
|
|
struct drm_syncobj *syncobj;
|
|
|
|
if (!fence)
|
|
return -EINVAL;
|
|
|
|
syncobj = drm_syncobj_find(file_private, handle);
|
|
if (!syncobj) {
|
|
dma_fence_put(fence);
|
|
return -ENOENT;
|
|
}
|
|
|
|
drm_syncobj_replace_fence(syncobj, 0, fence);
|
|
dma_fence_put(fence);
|
|
drm_syncobj_put(syncobj);
|
|
return 0;
|
|
}
|
|
|
|
static int drm_syncobj_export_sync_file(struct drm_file *file_private,
|
|
int handle, int *p_fd)
|
|
{
|
|
int ret;
|
|
struct dma_fence *fence;
|
|
struct sync_file *sync_file;
|
|
int fd = get_unused_fd_flags(O_CLOEXEC);
|
|
|
|
if (fd < 0)
|
|
return fd;
|
|
|
|
ret = drm_syncobj_find_fence(file_private, handle, 0, 0, &fence);
|
|
if (ret)
|
|
goto err_put_fd;
|
|
|
|
sync_file = sync_file_create(fence);
|
|
|
|
dma_fence_put(fence);
|
|
|
|
if (!sync_file) {
|
|
ret = -EINVAL;
|
|
goto err_put_fd;
|
|
}
|
|
|
|
fd_install(fd, sync_file->file);
|
|
|
|
*p_fd = fd;
|
|
return 0;
|
|
err_put_fd:
|
|
put_unused_fd(fd);
|
|
return ret;
|
|
}
|
|
/**
|
|
* drm_syncobj_open - initalizes syncobj file-private structures at devnode open time
|
|
* @file_private: drm file-private structure to set up
|
|
*
|
|
* Called at device open time, sets up the structure for handling refcounting
|
|
* of sync objects.
|
|
*/
|
|
void
|
|
drm_syncobj_open(struct drm_file *file_private)
|
|
{
|
|
idr_init_base(&file_private->syncobj_idr, 1);
|
|
spin_lock_init(&file_private->syncobj_table_lock);
|
|
}
|
|
|
|
static int
|
|
drm_syncobj_release_handle(int id, void *ptr, void *data)
|
|
{
|
|
struct drm_syncobj *syncobj = ptr;
|
|
|
|
drm_syncobj_put(syncobj);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* drm_syncobj_release - release file-private sync object resources
|
|
* @file_private: drm file-private structure to clean up
|
|
*
|
|
* Called at close time when the filp is going away.
|
|
*
|
|
* Releases any remaining references on objects by this filp.
|
|
*/
|
|
void
|
|
drm_syncobj_release(struct drm_file *file_private)
|
|
{
|
|
idr_for_each(&file_private->syncobj_idr,
|
|
&drm_syncobj_release_handle, file_private);
|
|
idr_destroy(&file_private->syncobj_idr);
|
|
}
|
|
|
|
int
|
|
drm_syncobj_create_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_create *args = data;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
/* no valid flags yet */
|
|
if (args->flags & ~(DRM_SYNCOBJ_CREATE_SIGNALED |
|
|
DRM_SYNCOBJ_CREATE_TYPE_TIMELINE))
|
|
return -EINVAL;
|
|
|
|
return drm_syncobj_create_as_handle(file_private,
|
|
&args->handle, args->flags);
|
|
}
|
|
|
|
int
|
|
drm_syncobj_destroy_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_destroy *args = data;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
/* make sure padding is empty */
|
|
if (args->pad)
|
|
return -EINVAL;
|
|
return drm_syncobj_destroy(file_private, args->handle);
|
|
}
|
|
|
|
int
|
|
drm_syncobj_handle_to_fd_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_handle *args = data;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
if (args->pad)
|
|
return -EINVAL;
|
|
|
|
if (args->flags != 0 &&
|
|
args->flags != DRM_SYNCOBJ_HANDLE_TO_FD_FLAGS_EXPORT_SYNC_FILE)
|
|
return -EINVAL;
|
|
|
|
if (args->flags & DRM_SYNCOBJ_HANDLE_TO_FD_FLAGS_EXPORT_SYNC_FILE)
|
|
return drm_syncobj_export_sync_file(file_private, args->handle,
|
|
&args->fd);
|
|
|
|
return drm_syncobj_handle_to_fd(file_private, args->handle,
|
|
&args->fd);
|
|
}
|
|
|
|
int
|
|
drm_syncobj_fd_to_handle_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_handle *args = data;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
if (args->pad)
|
|
return -EINVAL;
|
|
|
|
if (args->flags != 0 &&
|
|
args->flags != DRM_SYNCOBJ_FD_TO_HANDLE_FLAGS_IMPORT_SYNC_FILE)
|
|
return -EINVAL;
|
|
|
|
if (args->flags & DRM_SYNCOBJ_FD_TO_HANDLE_FLAGS_IMPORT_SYNC_FILE)
|
|
return drm_syncobj_import_sync_file_fence(file_private,
|
|
args->fd,
|
|
args->handle);
|
|
|
|
return drm_syncobj_fd_to_handle(file_private, args->fd,
|
|
&args->handle);
|
|
}
|
|
|
|
struct syncobj_wait_entry {
|
|
struct task_struct *task;
|
|
struct dma_fence *fence;
|
|
struct dma_fence_cb fence_cb;
|
|
struct drm_syncobj_cb syncobj_cb;
|
|
};
|
|
|
|
static void syncobj_wait_fence_func(struct dma_fence *fence,
|
|
struct dma_fence_cb *cb)
|
|
{
|
|
struct syncobj_wait_entry *wait =
|
|
container_of(cb, struct syncobj_wait_entry, fence_cb);
|
|
|
|
wake_up_process(wait->task);
|
|
}
|
|
|
|
static void syncobj_wait_syncobj_func(struct drm_syncobj *syncobj,
|
|
struct drm_syncobj_cb *cb)
|
|
{
|
|
struct syncobj_wait_entry *wait =
|
|
container_of(cb, struct syncobj_wait_entry, syncobj_cb);
|
|
|
|
drm_syncobj_search_fence(syncobj, 0, 0, &wait->fence);
|
|
|
|
wake_up_process(wait->task);
|
|
}
|
|
|
|
static signed long drm_syncobj_array_wait_timeout(struct drm_syncobj **syncobjs,
|
|
uint32_t count,
|
|
uint32_t flags,
|
|
signed long timeout,
|
|
uint32_t *idx)
|
|
{
|
|
struct syncobj_wait_entry *entries;
|
|
struct dma_fence *fence;
|
|
uint32_t signaled_count, i;
|
|
|
|
entries = kcalloc(count, sizeof(*entries), GFP_KERNEL);
|
|
if (!entries)
|
|
return -ENOMEM;
|
|
|
|
/* Walk the list of sync objects and initialize entries. We do
|
|
* this up-front so that we can properly return -EINVAL if there is
|
|
* a syncobj with a missing fence and then never have the chance of
|
|
* returning -EINVAL again.
|
|
*/
|
|
signaled_count = 0;
|
|
for (i = 0; i < count; ++i) {
|
|
entries[i].task = current;
|
|
drm_syncobj_search_fence(syncobjs[i], 0, 0,
|
|
&entries[i].fence);
|
|
if (!entries[i].fence) {
|
|
if (flags & DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT) {
|
|
continue;
|
|
} else {
|
|
timeout = -EINVAL;
|
|
goto cleanup_entries;
|
|
}
|
|
}
|
|
|
|
if (dma_fence_is_signaled(entries[i].fence)) {
|
|
if (signaled_count == 0 && idx)
|
|
*idx = i;
|
|
signaled_count++;
|
|
}
|
|
}
|
|
|
|
if (signaled_count == count ||
|
|
(signaled_count > 0 &&
|
|
!(flags & DRM_SYNCOBJ_WAIT_FLAGS_WAIT_ALL)))
|
|
goto cleanup_entries;
|
|
|
|
/* There's a very annoying laxness in the dma_fence API here, in
|
|
* that backends are not required to automatically report when a
|
|
* fence is signaled prior to fence->ops->enable_signaling() being
|
|
* called. So here if we fail to match signaled_count, we need to
|
|
* fallthough and try a 0 timeout wait!
|
|
*/
|
|
|
|
if (flags & DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT) {
|
|
for (i = 0; i < count; ++i) {
|
|
if (entries[i].fence)
|
|
continue;
|
|
|
|
drm_syncobj_fence_get_or_add_callback(syncobjs[i],
|
|
&entries[i].fence,
|
|
&entries[i].syncobj_cb,
|
|
syncobj_wait_syncobj_func);
|
|
}
|
|
}
|
|
|
|
do {
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
|
|
signaled_count = 0;
|
|
for (i = 0; i < count; ++i) {
|
|
fence = entries[i].fence;
|
|
if (!fence)
|
|
continue;
|
|
|
|
if (dma_fence_is_signaled(fence) ||
|
|
(!entries[i].fence_cb.func &&
|
|
dma_fence_add_callback(fence,
|
|
&entries[i].fence_cb,
|
|
syncobj_wait_fence_func))) {
|
|
/* The fence has been signaled */
|
|
if (flags & DRM_SYNCOBJ_WAIT_FLAGS_WAIT_ALL) {
|
|
signaled_count++;
|
|
} else {
|
|
if (idx)
|
|
*idx = i;
|
|
goto done_waiting;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (signaled_count == count)
|
|
goto done_waiting;
|
|
|
|
if (timeout == 0) {
|
|
timeout = -ETIME;
|
|
goto done_waiting;
|
|
}
|
|
|
|
if (signal_pending(current)) {
|
|
timeout = -ERESTARTSYS;
|
|
goto done_waiting;
|
|
}
|
|
|
|
timeout = schedule_timeout(timeout);
|
|
} while (1);
|
|
|
|
done_waiting:
|
|
__set_current_state(TASK_RUNNING);
|
|
|
|
cleanup_entries:
|
|
for (i = 0; i < count; ++i) {
|
|
if (entries[i].syncobj_cb.func)
|
|
drm_syncobj_remove_callback(syncobjs[i],
|
|
&entries[i].syncobj_cb);
|
|
if (entries[i].fence_cb.func)
|
|
dma_fence_remove_callback(entries[i].fence,
|
|
&entries[i].fence_cb);
|
|
dma_fence_put(entries[i].fence);
|
|
}
|
|
kfree(entries);
|
|
|
|
return timeout;
|
|
}
|
|
|
|
/**
|
|
* drm_timeout_abs_to_jiffies - calculate jiffies timeout from absolute value
|
|
*
|
|
* @timeout_nsec: timeout nsec component in ns, 0 for poll
|
|
*
|
|
* Calculate the timeout in jiffies from an absolute time in sec/nsec.
|
|
*/
|
|
static signed long drm_timeout_abs_to_jiffies(int64_t timeout_nsec)
|
|
{
|
|
ktime_t abs_timeout, now;
|
|
u64 timeout_ns, timeout_jiffies64;
|
|
|
|
/* make 0 timeout means poll - absolute 0 doesn't seem valid */
|
|
if (timeout_nsec == 0)
|
|
return 0;
|
|
|
|
abs_timeout = ns_to_ktime(timeout_nsec);
|
|
now = ktime_get();
|
|
|
|
if (!ktime_after(abs_timeout, now))
|
|
return 0;
|
|
|
|
timeout_ns = ktime_to_ns(ktime_sub(abs_timeout, now));
|
|
|
|
timeout_jiffies64 = nsecs_to_jiffies64(timeout_ns);
|
|
/* clamp timeout to avoid infinite timeout */
|
|
if (timeout_jiffies64 >= MAX_SCHEDULE_TIMEOUT - 1)
|
|
return MAX_SCHEDULE_TIMEOUT - 1;
|
|
|
|
return timeout_jiffies64 + 1;
|
|
}
|
|
|
|
static int drm_syncobj_array_wait(struct drm_device *dev,
|
|
struct drm_file *file_private,
|
|
struct drm_syncobj_wait *wait,
|
|
struct drm_syncobj **syncobjs)
|
|
{
|
|
signed long timeout = drm_timeout_abs_to_jiffies(wait->timeout_nsec);
|
|
uint32_t first = ~0;
|
|
|
|
timeout = drm_syncobj_array_wait_timeout(syncobjs,
|
|
wait->count_handles,
|
|
wait->flags,
|
|
timeout, &first);
|
|
if (timeout < 0)
|
|
return timeout;
|
|
|
|
wait->first_signaled = first;
|
|
return 0;
|
|
}
|
|
|
|
static int drm_syncobj_array_find(struct drm_file *file_private,
|
|
void __user *user_handles,
|
|
uint32_t count_handles,
|
|
struct drm_syncobj ***syncobjs_out)
|
|
{
|
|
uint32_t i, *handles;
|
|
struct drm_syncobj **syncobjs;
|
|
int ret;
|
|
|
|
handles = kmalloc_array(count_handles, sizeof(*handles), GFP_KERNEL);
|
|
if (handles == NULL)
|
|
return -ENOMEM;
|
|
|
|
if (copy_from_user(handles, user_handles,
|
|
sizeof(uint32_t) * count_handles)) {
|
|
ret = -EFAULT;
|
|
goto err_free_handles;
|
|
}
|
|
|
|
syncobjs = kmalloc_array(count_handles, sizeof(*syncobjs), GFP_KERNEL);
|
|
if (syncobjs == NULL) {
|
|
ret = -ENOMEM;
|
|
goto err_free_handles;
|
|
}
|
|
|
|
for (i = 0; i < count_handles; i++) {
|
|
syncobjs[i] = drm_syncobj_find(file_private, handles[i]);
|
|
if (!syncobjs[i]) {
|
|
ret = -ENOENT;
|
|
goto err_put_syncobjs;
|
|
}
|
|
}
|
|
|
|
kfree(handles);
|
|
*syncobjs_out = syncobjs;
|
|
return 0;
|
|
|
|
err_put_syncobjs:
|
|
while (i-- > 0)
|
|
drm_syncobj_put(syncobjs[i]);
|
|
kfree(syncobjs);
|
|
err_free_handles:
|
|
kfree(handles);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void drm_syncobj_array_free(struct drm_syncobj **syncobjs,
|
|
uint32_t count)
|
|
{
|
|
uint32_t i;
|
|
for (i = 0; i < count; i++)
|
|
drm_syncobj_put(syncobjs[i]);
|
|
kfree(syncobjs);
|
|
}
|
|
|
|
int
|
|
drm_syncobj_wait_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_wait *args = data;
|
|
struct drm_syncobj **syncobjs;
|
|
int ret = 0;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
if (args->flags & ~(DRM_SYNCOBJ_WAIT_FLAGS_WAIT_ALL |
|
|
DRM_SYNCOBJ_WAIT_FLAGS_WAIT_FOR_SUBMIT))
|
|
return -EINVAL;
|
|
|
|
if (args->count_handles == 0)
|
|
return -EINVAL;
|
|
|
|
ret = drm_syncobj_array_find(file_private,
|
|
u64_to_user_ptr(args->handles),
|
|
args->count_handles,
|
|
&syncobjs);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
ret = drm_syncobj_array_wait(dev, file_private,
|
|
args, syncobjs);
|
|
|
|
drm_syncobj_array_free(syncobjs, args->count_handles);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
drm_syncobj_reset_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_array *args = data;
|
|
struct drm_syncobj **syncobjs;
|
|
uint32_t i;
|
|
int ret;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
if (args->pad != 0)
|
|
return -EINVAL;
|
|
|
|
if (args->count_handles == 0)
|
|
return -EINVAL;
|
|
|
|
ret = drm_syncobj_array_find(file_private,
|
|
u64_to_user_ptr(args->handles),
|
|
args->count_handles,
|
|
&syncobjs);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
for (i = 0; i < args->count_handles; i++) {
|
|
drm_syncobj_fini(syncobjs[i]);
|
|
drm_syncobj_init(syncobjs[i]);
|
|
}
|
|
drm_syncobj_array_free(syncobjs, args->count_handles);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
drm_syncobj_signal_ioctl(struct drm_device *dev, void *data,
|
|
struct drm_file *file_private)
|
|
{
|
|
struct drm_syncobj_array *args = data;
|
|
struct drm_syncobj **syncobjs;
|
|
uint32_t i;
|
|
int ret;
|
|
|
|
if (!drm_core_check_feature(dev, DRIVER_SYNCOBJ))
|
|
return -EOPNOTSUPP;
|
|
|
|
if (args->pad != 0)
|
|
return -EINVAL;
|
|
|
|
if (args->count_handles == 0)
|
|
return -EINVAL;
|
|
|
|
ret = drm_syncobj_array_find(file_private,
|
|
u64_to_user_ptr(args->handles),
|
|
args->count_handles,
|
|
&syncobjs);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
for (i = 0; i < args->count_handles; i++) {
|
|
ret = drm_syncobj_assign_null_handle(syncobjs[i]);
|
|
if (ret < 0)
|
|
break;
|
|
}
|
|
|
|
drm_syncobj_array_free(syncobjs, args->count_handles);
|
|
|
|
return ret;
|
|
}
|