Signed-off-by: David Rosca <david.ro...@amd.com>
---
.../gpu/drm/amd/amdgpu/amdgpu_userq_fence.c | 49 +++++++++++++++----
include/uapi/drm/amdgpu_drm.h | 5 ++
2 files changed, 45 insertions(+), 9 deletions(-)
diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_userq_fence.c
b/drivers/gpu/drm/amd/amdgpu/amdgpu_userq_fence.c
index 95e91d1dc58a..5b2cdc49a28c 100644
--- a/drivers/gpu/drm/amd/amdgpu/amdgpu_userq_fence.c
+++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_userq_fence.c
@@ -460,10 +460,11 @@ int amdgpu_userq_signal_ioctl(struct drm_device *dev,
void *data,
struct drm_gem_object **gobj_read = NULL;
struct amdgpu_usermode_queue *queue;
struct amdgpu_userq_fence *userq_fence;
- struct drm_syncobj **syncobj = NULL;
+ struct amdgpu_cs_post_dep *syncobj = NULL;
u32 *bo_handles_write, num_write_bo_handles;
u32 *syncobj_handles, num_syncobj_handles;
u32 *bo_handles_read, num_read_bo_handles;
+ u64 *syncobj_points = NULL;
int r, i, entry, rentry, wentry;
struct dma_fence *fence;
struct drm_exec exec;
@@ -475,19 +476,37 @@ int amdgpu_userq_signal_ioctl(struct drm_device *dev,
void *data,
if (IS_ERR(syncobj_handles))
return PTR_ERR(syncobj_handles);
+ if (args->syncobj_points) {
+ syncobj_points =
memdup_user(u64_to_user_ptr(args->syncobj_points),
+ sizeof(u64) * num_syncobj_handles);
+ if (IS_ERR(syncobj_points)) {
+ r = PTR_ERR(syncobj_points);
+ goto free_syncobj_handles;
+ }
+ }
+
/* Array of pointers to the looked up syncobjs */
syncobj = kmalloc_array(num_syncobj_handles, sizeof(*syncobj),
GFP_KERNEL);
if (!syncobj) {
r = -ENOMEM;
- goto free_syncobj_handles;
+ goto free_syncobj_points;
}
for (entry = 0; entry < num_syncobj_handles; entry++) {
- syncobj[entry] = drm_syncobj_find(filp, syncobj_handles[entry]);
- if (!syncobj[entry]) {
+ syncobj[entry].chain = NULL;
+ syncobj[entry].syncobj = drm_syncobj_find(filp,
syncobj_handles[entry]);
+ if (!syncobj[entry].syncobj) {
r = -ENOENT;
goto free_syncobj;
}
+ if (syncobj_points && syncobj_points[entry]) {
+ syncobj[entry].point = syncobj_points[entry];
+ syncobj[entry].chain = dma_fence_chain_alloc();
+ if (!syncobj[entry].chain) {
+ r = -ENOMEM;
+ goto free_syncobj;
+ }
+ }
}
num_read_bo_handles = args->num_bo_read_handles;
@@ -603,8 +622,15 @@ int amdgpu_userq_signal_ioctl(struct drm_device *dev, void
*data,
}
/* Add the created fence to syncobj/BO's */
- for (i = 0; i < num_syncobj_handles; i++)
- drm_syncobj_replace_fence(syncobj[i], fence);
+ for (i = 0; i < num_syncobj_handles; i++) {
+ if (syncobj[i].chain) {
+ drm_syncobj_add_point(syncobj[i].syncobj,
syncobj[i].chain,
+ fence,
syncobj[i].point);
+ syncobj[i].chain = NULL;
+ } else {
+ drm_syncobj_replace_fence(syncobj[i].syncobj, fence);
+ }
+ }
/* drop the reference acquired in fence creation function */
dma_fence_put(fence);
@@ -624,10 +650,15 @@ int amdgpu_userq_signal_ioctl(struct drm_device *dev,
void *data,
free_bo_handles_read:
kfree(bo_handles_read);
free_syncobj:
- while (entry-- > 0)
- if (syncobj[entry])
- drm_syncobj_put(syncobj[entry]);
+ while (entry-- > 0) {
+ if (syncobj[entry].syncobj)
+ drm_syncobj_put(syncobj[entry].syncobj);
+ if (syncobj[entry].chain)
+ dma_fence_chain_free(syncobj[entry].chain);
+ }
kfree(syncobj);
+free_syncobj_points:
+ kfree(syncobj_points);
free_syncobj_handles:
kfree(syncobj_handles);
diff --git a/include/uapi/drm/amdgpu_drm.h b/include/uapi/drm/amdgpu_drm.h
index 85b3ca14f81e..197ea3bfb559 100644
--- a/include/uapi/drm/amdgpu_drm.h
+++ b/include/uapi/drm/amdgpu_drm.h
@@ -504,6 +504,11 @@ struct drm_amdgpu_userq_signal {
* @bo_write_handles.
*/
__u32 num_bo_write_handles;
+ /**
+ * @syncobj_points: The list of syncobj points submitted by the user
queue job
+ * for the corresponding @syncobj_handles.
+ */
+ __u64 syncobj_points;
};
struct drm_amdgpu_userq_fence_info {