blob: e3673422aac8dce628e36cb073eb03ae87e0c216 [file] [log] [blame]
Alex Deucherd38ceaf2015-04-20 16:55:21 -04001/*
2 * Copyright 2014 Advanced Micro Devices, Inc.
3 * All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the
7 * "Software"), to deal in the Software without restriction, including
8 * without limitation the rights to use, copy, modify, merge, publish,
9 * distribute, sub license, and/or sell copies of the Software, and to
10 * permit persons to whom the Software is furnished to do so, subject to
11 * the following conditions:
12 *
13 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
16 * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM,
17 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
18 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
19 * USE OR OTHER DEALINGS IN THE SOFTWARE.
20 *
21 * The above copyright notice and this permission notice (including the
22 * next paragraph) shall be included in all copies or substantial portions
23 * of the Software.
24 *
25 */
26/*
27 * Authors:
28 * Christian König <christian.koenig@amd.com>
29 */
30
31#include <drm/drmP.h>
32#include "amdgpu.h"
33#include "amdgpu_trace.h"
34
Christian Königf91b3a62015-08-20 14:47:40 +080035struct amdgpu_sync_entry {
36 struct hlist_node node;
37 struct fence *fence;
38};
39
Alex Deucherd38ceaf2015-04-20 16:55:21 -040040/**
41 * amdgpu_sync_create - zero init sync object
42 *
43 * @sync: sync object to initialize
44 *
45 * Just clear the sync object for now.
46 */
47void amdgpu_sync_create(struct amdgpu_sync *sync)
48{
Christian Königf91b3a62015-08-20 14:47:40 +080049 hash_init(sync->fences);
Alex Deucherd38ceaf2015-04-20 16:55:21 -040050 sync->last_vm_update = NULL;
51}
52
Christian Königbcc634f2016-02-16 16:23:02 +010053/**
54 * amdgpu_sync_same_dev - test if fence belong to us
55 *
56 * @adev: amdgpu device to use for the test
57 * @f: fence to test
58 *
59 * Test if the fence was issued by us.
60 */
Chunming Zhou3c623382015-08-20 18:33:59 +080061static bool amdgpu_sync_same_dev(struct amdgpu_device *adev, struct fence *f)
62{
Chunming Zhou3c623382015-08-20 18:33:59 +080063 struct amd_sched_fence *s_fence = to_amd_sched_fence(f);
64
Christian König4f839a22015-09-08 20:22:31 +020065 if (s_fence) {
66 struct amdgpu_ring *ring;
67
68 ring = container_of(s_fence->sched, struct amdgpu_ring, sched);
69 return ring->adev == adev;
70 }
71
Chunming Zhou3c623382015-08-20 18:33:59 +080072 return false;
73}
74
Christian Königbcc634f2016-02-16 16:23:02 +010075/**
76 * amdgpu_sync_get_owner - extract the owner of a fence
77 *
78 * @fence: fence get the owner from
79 *
80 * Extract who originally created the fence.
81 */
82static void *amdgpu_sync_get_owner(struct fence *f)
Chunming Zhou3c623382015-08-20 18:33:59 +080083{
Chunming Zhou3c623382015-08-20 18:33:59 +080084 struct amd_sched_fence *s_fence = to_amd_sched_fence(f);
Christian Königbcc634f2016-02-16 16:23:02 +010085
Chunming Zhou3c623382015-08-20 18:33:59 +080086 if (s_fence)
Christian Königbcc634f2016-02-16 16:23:02 +010087 return s_fence->owner;
Christian König336d1f52016-02-16 10:57:10 +010088
Christian Königbcc634f2016-02-16 16:23:02 +010089 return AMDGPU_FENCE_OWNER_UNDEFINED;
Chunming Zhou3c623382015-08-20 18:33:59 +080090}
91
Christian Königbcc634f2016-02-16 16:23:02 +010092/**
93 * amdgpu_sync_keep_later - Keep the later fence
94 *
95 * @keep: existing fence to test
96 * @fence: new fence
97 *
98 * Either keep the existing fence or the new one, depending which one is later.
99 */
Christian König24233862015-10-22 10:53:16 +0200100static void amdgpu_sync_keep_later(struct fence **keep, struct fence *fence)
101{
102 if (*keep && fence_is_later(*keep, fence))
103 return;
104
105 fence_put(*keep);
106 *keep = fence_get(fence);
107}
108
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400109/**
Christian König91e1a522015-07-06 22:06:40 +0200110 * amdgpu_sync_fence - remember to sync to this fence
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400111 *
112 * @sync: sync object to add fence to
113 * @fence: fence to sync to
114 *
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400115 */
Christian König91e1a522015-07-06 22:06:40 +0200116int amdgpu_sync_fence(struct amdgpu_device *adev, struct amdgpu_sync *sync,
117 struct fence *f)
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400118{
Christian Königf91b3a62015-08-20 14:47:40 +0800119 struct amdgpu_sync_entry *e;
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400120
Christian König91e1a522015-07-06 22:06:40 +0200121 if (!f)
122 return 0;
123
Chunming Zhou3c623382015-08-20 18:33:59 +0800124 if (amdgpu_sync_same_dev(adev, f) &&
Christian Königbcc634f2016-02-16 16:23:02 +0100125 amdgpu_sync_get_owner(f) == AMDGPU_FENCE_OWNER_VM)
Christian König24233862015-10-22 10:53:16 +0200126 amdgpu_sync_keep_later(&sync->last_vm_update, f);
Chunming Zhou3c623382015-08-20 18:33:59 +0800127
Christian König046c12c2016-01-18 14:49:45 +0100128 hash_for_each_possible(sync->fences, e, node, f->context) {
129 if (unlikely(e->fence->context != f->context))
130 continue;
Christian König24233862015-10-22 10:53:16 +0200131
Christian König046c12c2016-01-18 14:49:45 +0100132 amdgpu_sync_keep_later(&e->fence, f);
Christian Königf91b3a62015-08-20 14:47:40 +0800133 return 0;
134 }
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400135
Christian König046c12c2016-01-18 14:49:45 +0100136 e = kmalloc(sizeof(struct amdgpu_sync_entry), GFP_KERNEL);
137 if (!e)
138 return -ENOMEM;
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400139
Christian König046c12c2016-01-18 14:49:45 +0100140 hash_add(sync->fences, &e->node, f->context);
141 e->fence = fence_get(f);
Christian König91e1a522015-07-06 22:06:40 +0200142 return 0;
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400143}
144
145/**
Chunming Zhou2f4b9402016-01-15 11:05:21 +0800146 * amdgpu_sync_resv - sync to a reservation object
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400147 *
148 * @sync: sync object to add fences from reservation object to
149 * @resv: reservation object with embedded fence
150 * @shared: true if we should only sync to the exclusive fence
151 *
Chunming Zhou2f4b9402016-01-15 11:05:21 +0800152 * Sync to the fence
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400153 */
154int amdgpu_sync_resv(struct amdgpu_device *adev,
155 struct amdgpu_sync *sync,
156 struct reservation_object *resv,
157 void *owner)
158{
159 struct reservation_object_list *flist;
160 struct fence *f;
Chunming Zhou423a9482015-08-24 16:59:54 +0800161 void *fence_owner;
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400162 unsigned i;
163 int r = 0;
164
Jammy Zhou4b095302015-05-12 23:17:19 +0800165 if (resv == NULL)
166 return -EINVAL;
167
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400168 /* always sync to the exclusive fence */
169 f = reservation_object_get_excl(resv);
Christian König91e1a522015-07-06 22:06:40 +0200170 r = amdgpu_sync_fence(adev, sync, f);
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400171
172 flist = reservation_object_get_list(resv);
173 if (!flist || r)
174 return r;
175
176 for (i = 0; i < flist->shared_count; ++i) {
177 f = rcu_dereference_protected(flist->shared[i],
178 reservation_object_held(resv));
Chunming Zhou423a9482015-08-24 16:59:54 +0800179 if (amdgpu_sync_same_dev(adev, f)) {
Christian König1d3897e2015-07-27 15:40:35 +0200180 /* VM updates are only interesting
181 * for other VM updates and moves.
182 */
Chunming Zhou423a9482015-08-24 16:59:54 +0800183 fence_owner = amdgpu_sync_get_owner(f);
Christian König7a91d6c2015-10-27 17:28:24 +0100184 if ((owner != AMDGPU_FENCE_OWNER_UNDEFINED) &&
185 (fence_owner != AMDGPU_FENCE_OWNER_UNDEFINED) &&
Christian König1d3897e2015-07-27 15:40:35 +0200186 ((owner == AMDGPU_FENCE_OWNER_VM) !=
Chunming Zhou423a9482015-08-24 16:59:54 +0800187 (fence_owner == AMDGPU_FENCE_OWNER_VM)))
Christian König91e1a522015-07-06 22:06:40 +0200188 continue;
189
Christian König1d3897e2015-07-27 15:40:35 +0200190 /* Ignore fence from the same owner as
191 * long as it isn't undefined.
192 */
193 if (owner != AMDGPU_FENCE_OWNER_UNDEFINED &&
Chunming Zhou423a9482015-08-24 16:59:54 +0800194 fence_owner == owner)
Christian König1d3897e2015-07-27 15:40:35 +0200195 continue;
196 }
197
Christian König91e1a522015-07-06 22:06:40 +0200198 r = amdgpu_sync_fence(adev, sync, f);
199 if (r)
200 break;
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400201 }
202 return r;
203}
204
Christian Könige61235d2015-08-25 11:05:36 +0200205struct fence *amdgpu_sync_get_fence(struct amdgpu_sync *sync)
206{
207 struct amdgpu_sync_entry *e;
208 struct hlist_node *tmp;
209 struct fence *f;
210 int i;
211
212 hash_for_each_safe(sync->fences, i, tmp, e, node) {
213
214 f = e->fence;
215
216 hash_del(&e->node);
217 kfree(e);
218
219 if (!fence_is_signaled(f))
220 return f;
221
222 fence_put(f);
223 }
224 return NULL;
225}
226
Christian Königf91b3a62015-08-20 14:47:40 +0800227int amdgpu_sync_wait(struct amdgpu_sync *sync)
228{
229 struct amdgpu_sync_entry *e;
230 struct hlist_node *tmp;
231 int i, r;
232
233 hash_for_each_safe(sync->fences, i, tmp, e, node) {
234 r = fence_wait(e->fence, false);
235 if (r)
236 return r;
237
238 hash_del(&e->node);
239 fence_put(e->fence);
240 kfree(e);
241 }
Christian König3daea9e3d2015-09-05 11:12:27 +0200242
Christian Königf91b3a62015-08-20 14:47:40 +0800243 return 0;
244}
245
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400246/**
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400247 * amdgpu_sync_free - free the sync object
248 *
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400249 * @sync: sync object to use
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400250 *
Chunming Zhou2f4b9402016-01-15 11:05:21 +0800251 * Free the sync object.
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400252 */
Christian König8a8f0b42016-02-03 15:11:39 +0100253void amdgpu_sync_free(struct amdgpu_sync *sync)
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400254{
Christian Königf91b3a62015-08-20 14:47:40 +0800255 struct amdgpu_sync_entry *e;
256 struct hlist_node *tmp;
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400257 unsigned i;
258
Christian Königf91b3a62015-08-20 14:47:40 +0800259 hash_for_each_safe(sync->fences, i, tmp, e, node) {
260 hash_del(&e->node);
261 fence_put(e->fence);
262 kfree(e);
263 }
264
Chunming Zhou3c623382015-08-20 18:33:59 +0800265 fence_put(sync->last_vm_update);
Alex Deucherd38ceaf2015-04-20 16:55:21 -0400266}